From 632587c3b4bf982117bbc80f22c66d4d96ff98d4 Mon Sep 17 00:00:00 2001 From: David Anderson Date: Wed, 1 Oct 2008 20:58:28 +0000 Subject: [PATCH] - scheduler: enforce max_wus_to_end, daily_result_quota, and max_wus_in_progress when using score-based scheduling svn path=/trunk/boinc/; revision=16106 --- checkin_notes | 7 ++++++ sched/sched_send.cpp | 53 ++++++++++++++++++++++++++++++++++++++------ 2 files changed, 53 insertions(+), 7 deletions(-) diff --git a/checkin_notes b/checkin_notes index c35a56453b..124409d929 100644 --- a/checkin_notes +++ b/checkin_notes @@ -7920,3 +7920,10 @@ David 1 Oct 2008 handle_request.cpp sched_config.cpp,h server_types.cpp,h + +David 1 Oct 2008 + - scheduler: enforce max_wus_to_end, daily_result_quota, + and max_wus_in_progress when using score-based scheduling + + sched/ + sched_send.cpp diff --git a/sched/sched_send.cpp b/sched/sched_send.cpp index 6f4a172094..dc453d9004 100644 --- a/sched/sched_send.cpp +++ b/sched/sched_send.cpp @@ -1451,13 +1451,32 @@ struct JOB_SET { double est_time; double disk_usage; double disk_limit; + int max_jobs; std::list jobs; // sorted high to low - JOB_SET(double wr, double dl) { - work_req = wr; + JOB_SET(SCHEDULER_REQUEST& sreq, SCHEDULER_REPLY& reply) { + work_req = sreq.work_req_seconds; est_time = 0; disk_usage = 0; - disk_limit = dl; + disk_limit = reply.wreq.disk_available; + max_jobs = config.max_wus_to_send; + int ncpus = effective_ncpus(reply.host), n; + + if (config.daily_result_quota) { + if (reply.host.max_results_day == 0 || reply.host.max_results_day>config.daily_result_quota) { + reply.host.max_results_day = config.daily_result_quota; + } + reply.wreq.daily_result_quota = ncpus*reply.host.max_results_day; + n = reply.wreq.daily_result_quota - reply.host.nresults_today; + if (n < 0) n = 0; + if (n < max_jobs) max_jobs = n; + } + + if (config.max_wus_in_progress) { + n = config.max_wus_in_progress*ncpus - reply.wreq.nresults_on_host; + if (n < 0) n = 0; + if (n < max_jobs) max_jobs = n; + } } void add_job(JOB&); double higher_score_disk_usage(double); @@ -1667,9 +1686,10 @@ double JOB_SET::lowest_score() { return jobs.back().score; } -// add the given job, and remove lowest-score jobs -// that are in excess of work request -// or that cause the disk limit to be exceeded +// add the given job, and remove lowest-score jobs that +// - are in excess of work request +// - are in excess of per-request or per-day limits +// - cause the disk limit to be exceeded // void JOB_SET::add_job(JOB& job) { while (!jobs.empty()) { @@ -1694,6 +1714,13 @@ void JOB_SET::add_job(JOB& job) { break; } } + + if (jobs.size() == max_jobs) { + JOB& worst_job = jobs.back(); + jobs.pop_back(); + ssp->wu_results[worst_job.index].state = WR_STATE_PRESENT; + } + list::iterator i = jobs.begin(); while (i != jobs.end()) { if (i->score < job.score) { @@ -1749,7 +1776,7 @@ void JOB_SET::send(SCHEDULER_REQUEST& sreq, SCHEDULER_REPLY& reply) { void send_work_matchmaker(SCHEDULER_REQUEST& sreq, SCHEDULER_REPLY& reply) { int i, slots_locked=0; - JOB_SET jobs (sreq.work_req_seconds, reply.wreq.disk_available); + JOB_SET jobs (sreq, reply); int min_slots = config.mm_min_slots; if (!min_slots) min_slots = ssp->max_wu_results/2; int max_slots = config.mm_max_slots; @@ -1758,6 +1785,10 @@ void send_work_matchmaker(SCHEDULER_REQUEST& sreq, SCHEDULER_REPLY& reply) { lock_sema(); i = rand() % ssp->max_wu_results; + + // scan through the job cache, maintaining a JOB_SET of jobs + // that we can send to this client, ordered by score. + // for (int slots_scanned=0; slots_scannedmax_wu_results; WU_RESULT& wu_result = ssp->wu_results[i]; @@ -1774,6 +1805,11 @@ void send_work_matchmaker(SCHEDULER_REQUEST& sreq, SCHEDULER_REPLY& reply) { JOB job; job.index = i; + + // get score for this job, and skip it if it fails quick check. + // NOTE: the EDF check done in get_score() + // includes only in-progress jobs. + // if (!job.get_score(sreq, reply)) { continue; } @@ -1782,6 +1818,7 @@ void send_work_matchmaker(SCHEDULER_REQUEST& sreq, SCHEDULER_REPLY& reply) { "score for %s: %f\n", wu_result.workunit.name, job.score ); } + if (job.score > jobs.lowest_score() || !jobs.request_satisfied()) { ssp->wu_results[i].state = g_pid; unlock_sema(); @@ -1799,6 +1836,8 @@ void send_work_matchmaker(SCHEDULER_REQUEST& sreq, SCHEDULER_REPLY& reply) { if (jobs.request_satisfied() && slots_scanned>=min_slots) break; } + // TODO: trim jobs from tail of list until we pass the EDF check + // jobs.send(sreq, reply); unlock_sema(); if (slots_locked > max_locked) {