When should the core client ask a project for more work,
which project should it ask,
and how much work should it ask for?
The goals of the CPU scheduler and work-fetch policies are
(in descending priority):
- Results should be completed and reported by their deadline
(because results reported after their deadline
may not have any value to the project and may not be granted credit).
- NCPUS processors should be kept busy.
- At any given point, a computer should have enough work
so that NCPUS processors will be busy for at least
min_queue days (min_queue is a user preference).
- Project resource shares should be honored over the long term.
- Variety: if a computer is attached to multiple projects,
execution should rotate among projects on a frequent basis.
In previous versions of BOINC,
the core client attempted to maintain at least one result
for each attached project,
and would do weighted round-robin CPU scheduling among all projects.
In some scenarios (any combination of slow computer,
lots of projects, and tight deadlines) a computer could
miss the deadlines of all its results.
The new policies solve this problem as follows:
-
Work fetch is limited to ensure that deadlines can be met.
A computer attached to 10 projects might
have work for only a few (perhaps only one) at a given time.
-
If deadlines are threatened,
the CPU scheduling policy optimizes the likelihood of meeting deadlines,
at the expense of variety.
Concepts and terms
Wall CPU time
A result's wall CPU time is the amount of wall-clock time
its process has been runnable at the OS level.
The actual CPU time may be less than this,
e.g. if the process does a lot of paging,
or if other (non-BOINC) processing jobs run at the same time.
BOINC uses wall CPU time as the measure of how much resource
has been given to each project.
Why not use actual CPU time instead?
- Wall CPU time is more fair in the case of paging apps.
- The measurement of actual CPU time depends on apps to
report it correctly.
Sometimes apps have bugs that cause them to always report zero.
This screws up the scheduler.
Result states
R is runnable if
- Neither R nor R.project is suspended, and
- R's files have been downloaded, and
- R hasn't finished computing
Project states
P is runnable if
- P has at least one runnable result
(this implies that P is not suspended).
P is downloading if
- P is not suspended, and
- P has at least one result whose files are being downloaded
and none of the downloads is deferred.
P is fetchable
(i.e. the work-fetch policy allows work to be fetched from it) if
- P is not suspended, and
- P is not deferred (i.e. its minimum RPC time is in the past), and
- P's no-new-work flag is not set, and
- P is not overworked (see definition below), and
- a fetch of P's master file is not pending
P is latency-limited if
- The client's last scheduler RPC to P returned
a 'no work because of deadlines' flag, and
- the RPC reply's delay request has not yet elapsed.
This means that P has work available,
but didn't send any because the work's deadlines couldn't be met
given the existing work queue.
P is potentially runnable if
- P is either runnable, downloading, fetchable, overworked,
or latency-limited.
This means that, to the best of the client's knowledge,
it could do work for P if it wanted to.
Debt
Intuitively, a project's 'debt' is how much work is owed to it,
relative to other projects.
BOINC uses two types of debt;
each is defined for a set S of projects.
In each case, the debt is recalculated periodically as follows:
- A = the wall CPU time used by projects in S during this period
- R = sum of resource shares of projects in S
- For each project P in S:
- F = P.resource_share / R (i.e., P's fractional resource share)
- W = A*F (i.e., how much wall CPU time P should have gotten)
- P.debt += W - P.wall_cpu_time (i.e. what P should have gotten
minus what it got).
- P.debt is normalized so that the mean or minimum is zero.
Short-term debt is used by the CPU scheduler.
It is adjusted over the set of runnable projects.
It is normalized so that minimum short-term debt is zero,
and maximum short-term debt is no greater than 86,400 (i.e. one day).
Long-term debt is used by the work-fetch policy.
It is defined for all projects,
and adjusted over the set of potentially runnable projects.
It is normalized so that average long-term debt,
over all project, is zero.
CPU scheduling policy
The CPU scheduler uses an earliest-deadline-first (EDF) policy
for results that are in danger of missing their deadline,
and weighted round-robin among other projects if additional CPUs exist.
This allows the client to meet deadlines that would otherwise be missed,
while honoring resource shares over the long term.
The scheduler uses the following data,
which are obtained by doing a simulation of round-robin scheduling
applied to the current work queue:
- deadline_missed(R): whether result R would miss
its deadline with round-robin scheduling.
- deadlines_missed(P):
the number of results of P whose deadlines would
be missed with round-robin scheduling.
The scheduling policy is:
- Let P be the project with the earliest-deadline runnable result
among projects with deadlines_missed(P)>0.
Let R be P's earliest-deadline runnable result not scheduled yet.
Tiebreaker: least index in result array.
- If such an R exists, schedule R and decrement deadlines_missed(P).
- If there are more CPUs, and projects with deadlines_missed(P)>0, go to 1.
- If all CPUs are scheduled, stop.
- Set the 'anticipated debt' of each project to its short-term debt
- Find the project P with the greatest anticipated debt,
select one of P's runnable results
(picking one that is already running, if possible)
and schedule that result.
- Decrement P's anticipated debt by the 'expected payoff'
(the scheduling period divided by NCPUS).
- Repeat steps 6 and 7 for additional CPUs
The CPU scheduler runs when a result is completed,
when the end of the user-specified scheduling period is reached,
when new results become runnable,
or when the user performs a UI interaction
(e.g. suspending or resuming a project or result).
The CPU scheduler decides what result should run,
but it doesn't enforce this decision
(by preempting, resuming and starting applications).
This enforcement is done by a separate function,
which runs periodically, and is also called by
the CPU scheduler at its conclusion.
The following rules apply to application preemption:
- If the 'leave in memory' preference is not set,
an application scheduled for preemption is allowed to run for
up to sched_interval/2 additional seconds, or until it checkpoints.
-
The above does not apply for application being preempted
to run a result R for which deadline_missed(R).
- If an application has never checkpointed,
it is always left in memory on preemption.
Work-fetch policy
When a result runs in EDF mode,
its project may get more than its share of CPU time.
The work-fetch policy is responsible for
ensuring that this doesn't happen repeatedly.
It does this by suppressing work fetch for the project.
A project P is overworked if
- P.long_term_debt < -sched_period
This condition occurs if P's results run in EDF mode
(and in extreme cases, when a project with large negative LTD
is detached).
The work-fetch policy uses the functions
frs(project P)
P's fractional resource share among fetchable projects.
min_results(project P)
The minimum number of runnable results needed to
maintain P's resource share on this machine: namely,
ceil(ncpus*frs(P))
time_until_work_done(project P)
The estimated wall time until the number of
uncompleted results for this project will reach min_results(P)-1,
assuming round-robin scheduling among currently fetchable projects.
time_until_free_cpu()
The estimated wall time until there is a free CPU,
assuming round-robin scheduling among currently fetchable projects.
The work-fetch policy function is called every few minutes
(or as needed) by the scheduler RPC polling function.
It sets the variable work_request_size(P) for each project P,
which is the number of seconds of work to request
if we do a scheduler RPC to P.
This is computed as follows:
for each project P
if P is suspended, deferred, overworked, or no-new-work
P.work_request_size = 0
else
if time_until_work_done(P) > min_queue or CPU scheduler scheduled all CPUs by EDF
if time_until_cpu_free() < min_queue
P.work_request_size = 1
else
P.work_request_size = 0
else
y = estimated wall time of P's queued work
P.work_request_size = max(1, (min_queue*ncpus*frs(P)) - y)
if time_until_cpu_free() < min_queue and P.work_request_size==0 for all P
for each project P
if P is overworked
P.work_request_size = 1
The scheduler RPC mechanism may select a project to contact
because of a user request, an outstanding trickle-up message,
or a result that is overdue for reporting.
If it does so, it will also request work from that project.
Otherwise, the RPC mechanism chooses the project P for which
P.work_request_size>0 and
P.long_term_debt - time_until_work_done(P) is greatest
and gets work from that project.
Scheduler work-send policy
NOTE: the following has not been implemented,
and is independent of the above policies.
The scheduler should avoid sending results whose
deadlines are likely to be missed,
or which are likely to cause existing results to miss their deadlines.
This will be accomplished as follows:
-
Scheduler requests includes connection period,
list of queued result (with estimated time remaining and deadline)
and project resource fractions.
-
The scheduler won't send results whose deadlines are less than
now + min_queue.
-
The scheduler does an EDF simulation of the initial workload
to determine by how much each result misses its deadline.
For each result R being considered for sending,
the scheduler does an EDF simulation.
If R meets its deadline
(optional if the project does not need strict adherence),
and no result misses its deadline by more than it did previously, R is sent.
-
If the scheduler has work but doesn't send any because of deadline misses,
it returns a 'no work because of deadlines' flag.
If the last RPC to a project returned this flag,
it is marked as latency-limited and accumulates LTD.
Describing scenarios
We encourage the use of the following notation for
describing scheduling scenarios
(times are given in hours):
P(C, D, R)
This describes a project with
- C = CPU time per task
- D = delay bound
- R = fractional resource share
A scenario is described by a list of project,
plus the following optional parameters:
- NCPUS: number of CPUS (default 1)
- min_queue
Hence a typical scenario description is:
(P1(1000, 2000, .5), P2(1, 10, .5), NCPUS=4)
";
page_tail();
?>