2009-01-26 05:00:49 +00:00
|
|
|
// This file is part of BOINC.
|
|
|
|
// http://boinc.berkeley.edu
|
|
|
|
// Copyright (C) 2008 University of California
|
|
|
|
//
|
|
|
|
// BOINC is free software; you can redistribute it and/or modify it
|
|
|
|
// under the terms of the GNU Lesser General Public License
|
|
|
|
// as published by the Free Software Foundation,
|
|
|
|
// either version 3 of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// BOINC is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
// See the GNU Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
|
|
// along with BOINC. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
// Work fetch logic for CPU, GPU, and other processing resources.
|
|
|
|
// See http://boinc.berkeley.edu/trac/wiki/GpuWorkFetch
|
|
|
|
|
2008-12-31 23:14:57 +00:00
|
|
|
#ifndef _WORK_FETCH_
|
|
|
|
#define _WORK_FETCH_
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
|
2011-01-07 20:23:22 +00:00
|
|
|
extern bool use_hyst_fetch;
|
2010-11-23 19:39:47 +00:00
|
|
|
|
2011-07-11 05:34:09 +00:00
|
|
|
#define RSC_TYPE_ANY -1
|
|
|
|
#define RSC_TYPE_CPU 0
|
2008-12-31 23:14:57 +00:00
|
|
|
|
2009-12-21 19:20:28 +00:00
|
|
|
struct PROJECT;
|
2008-12-31 23:14:57 +00:00
|
|
|
struct RESULT;
|
2009-12-21 19:20:28 +00:00
|
|
|
struct ACTIVE_TASK;
|
2008-12-31 23:14:57 +00:00
|
|
|
struct RSC_WORK_FETCH;
|
2009-09-16 17:34:19 +00:00
|
|
|
struct SCHEDULER_REPLY;
|
2010-09-24 21:24:02 +00:00
|
|
|
struct APP_VERSION;
|
2008-12-31 23:14:57 +00:00
|
|
|
|
2009-01-03 06:01:17 +00:00
|
|
|
// state per (resource, project) pair
|
2008-12-31 23:14:57 +00:00
|
|
|
//
|
|
|
|
struct RSC_PROJECT_WORK_FETCH {
|
2009-01-03 06:01:17 +00:00
|
|
|
// the following are persistent (saved in state file)
|
2008-12-31 23:14:57 +00:00
|
|
|
double backoff_time;
|
|
|
|
double backoff_interval;
|
2010-11-23 19:39:47 +00:00
|
|
|
|
|
|
|
// the following used by debt accounting
|
2008-12-31 23:14:57 +00:00
|
|
|
double secs_this_debt_interval;
|
|
|
|
inline void reset_debt_accounting() {
|
|
|
|
secs_this_debt_interval = 0;
|
|
|
|
}
|
2010-01-21 00:14:56 +00:00
|
|
|
double queue_est;
|
|
|
|
// an estimate of instance-secs of queued work;
|
|
|
|
// a temp used in computing overall debts
|
2010-02-11 22:19:22 +00:00
|
|
|
bool anon_skip;
|
|
|
|
// set if this project is anonymous platform
|
|
|
|
// and it has no app version that uses this resource
|
2009-01-03 06:01:17 +00:00
|
|
|
double fetchable_share;
|
|
|
|
// this project's share relative to projects from which
|
|
|
|
// we could probably get work for this resource;
|
|
|
|
// determines how many instances this project deserves
|
2009-02-02 05:15:12 +00:00
|
|
|
bool has_runnable_jobs;
|
2009-02-02 18:47:34 +00:00
|
|
|
double sim_nused;
|
2009-12-24 20:40:27 +00:00
|
|
|
double nused_total; // sum of instances over all runnable jobs
|
2009-04-10 19:01:38 +00:00
|
|
|
int deadlines_missed;
|
|
|
|
int deadlines_missed_copy;
|
2009-08-14 16:54:45 +00:00
|
|
|
// copy of the above used during schedule_cpus()
|
2008-12-31 23:14:57 +00:00
|
|
|
|
|
|
|
RSC_PROJECT_WORK_FETCH() {
|
|
|
|
memset(this, 0, sizeof(*this));
|
|
|
|
}
|
|
|
|
|
2009-01-23 18:29:28 +00:00
|
|
|
inline void reset() {
|
2008-12-31 23:14:57 +00:00
|
|
|
backoff_time = 0;
|
|
|
|
backoff_interval = 0;
|
|
|
|
}
|
2009-01-08 00:41:15 +00:00
|
|
|
|
|
|
|
bool may_have_work;
|
2009-09-28 04:24:18 +00:00
|
|
|
bool compute_may_have_work(PROJECT*, int rsc_type);
|
2009-04-03 21:55:26 +00:00
|
|
|
void backoff(PROJECT*, const char*);
|
2009-09-28 04:24:18 +00:00
|
|
|
void rr_init(PROJECT*, int rsc_type);
|
2009-01-10 00:48:22 +00:00
|
|
|
void clear_backoff() {
|
|
|
|
backoff_time = 0;
|
|
|
|
backoff_interval = 0;
|
|
|
|
}
|
2008-12-31 23:14:57 +00:00
|
|
|
};
|
|
|
|
|
2009-09-03 20:31:04 +00:00
|
|
|
// estimate the time a resources will be saturated
|
|
|
|
// with high-priority jobs.
|
|
|
|
//
|
|
|
|
struct BUSY_TIME_ESTIMATOR {
|
|
|
|
std::vector<double> busy_time;
|
|
|
|
int ninstances;
|
|
|
|
inline void reset() {
|
|
|
|
for (int i=0; i<ninstances; i++) {
|
|
|
|
busy_time[i] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
inline void init(int n) {
|
|
|
|
ninstances = n;
|
|
|
|
busy_time.resize(n);
|
|
|
|
reset();
|
|
|
|
}
|
|
|
|
// called for each high-priority job.
|
|
|
|
// Find the least-busy instance, and put this job
|
|
|
|
// on that and following instances
|
|
|
|
//
|
|
|
|
inline void update(double dur, double nused) {
|
|
|
|
int i, j;
|
|
|
|
if (nused < 1) return;
|
|
|
|
double best = busy_time[0];
|
|
|
|
int ibest = 0;
|
|
|
|
for (i=1; i<ninstances; i++) {
|
|
|
|
if (busy_time[i] < best) {
|
|
|
|
best = busy_time[i];
|
|
|
|
ibest = i;
|
|
|
|
}
|
|
|
|
}
|
2009-09-16 16:54:42 +00:00
|
|
|
int inused = (int) nused; // ignore fractional usage
|
|
|
|
for (i=0; i<inused; i++) {
|
2009-09-03 20:31:04 +00:00
|
|
|
j = (ibest + i) % ninstances;
|
|
|
|
busy_time[j] += dur;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// the overall busy time is the busy time of
|
|
|
|
// the least busy instance
|
|
|
|
//
|
|
|
|
inline double get_busy_time() {
|
2009-09-29 02:12:35 +00:00
|
|
|
if (!ninstances) return 0;
|
2009-09-03 20:31:04 +00:00
|
|
|
double best = busy_time[0];
|
|
|
|
for (int i=1; i<ninstances; i++) {
|
|
|
|
if (busy_time[i] < best) {
|
|
|
|
best = busy_time[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return best;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2008-12-31 23:14:57 +00:00
|
|
|
// per-resource state
|
|
|
|
//
|
|
|
|
struct RSC_WORK_FETCH {
|
|
|
|
int rsc_type;
|
|
|
|
int ninstances;
|
2010-01-21 00:14:56 +00:00
|
|
|
double relative_speed; // total FLOPS relative to CPU total FLOPS
|
2008-12-31 23:14:57 +00:00
|
|
|
|
|
|
|
// the following used/set by rr_simulation():
|
|
|
|
//
|
|
|
|
double shortfall;
|
2009-03-18 15:53:02 +00:00
|
|
|
// seconds of idle instances between now and now+work_buf_total()
|
2008-12-31 23:14:57 +00:00
|
|
|
double nidle_now;
|
2009-02-02 18:47:34 +00:00
|
|
|
double sim_nused;
|
2009-01-03 06:01:17 +00:00
|
|
|
double total_fetchable_share;
|
|
|
|
// total RS of projects from which we could fetch jobs for this device
|
2009-07-17 18:29:10 +00:00
|
|
|
double saturated_time;
|
|
|
|
// estimated time until resource is not saturated
|
|
|
|
// used to calculate work request
|
2009-03-19 16:55:04 +00:00
|
|
|
double deadline_missed_instances;
|
|
|
|
// instance count for jobs that miss deadline
|
2011-09-12 17:01:54 +00:00
|
|
|
std::vector<RESULT*> pending;
|
2009-09-03 20:31:04 +00:00
|
|
|
BUSY_TIME_ESTIMATOR busy_time_estimator;
|
2010-11-01 16:53:41 +00:00
|
|
|
#ifdef SIM
|
|
|
|
double estimated_delay;
|
|
|
|
#endif
|
2009-01-03 06:01:17 +00:00
|
|
|
|
2009-09-03 20:31:04 +00:00
|
|
|
void init(int t, int n, double sp) {
|
|
|
|
rsc_type = t;
|
|
|
|
ninstances = n;
|
2010-01-21 00:14:56 +00:00
|
|
|
relative_speed = sp;
|
2009-09-03 20:31:04 +00:00
|
|
|
busy_time_estimator.init(n);
|
|
|
|
}
|
2009-01-03 06:01:17 +00:00
|
|
|
// the following specify the work request for this resource
|
|
|
|
//
|
|
|
|
double req_secs;
|
2009-12-24 20:40:27 +00:00
|
|
|
double req_instances;
|
2008-12-31 23:14:57 +00:00
|
|
|
|
|
|
|
// debt accounting
|
|
|
|
double secs_this_debt_interval;
|
|
|
|
inline void reset_debt_accounting() {
|
2009-02-24 00:06:45 +00:00
|
|
|
this->secs_this_debt_interval = 0;
|
2008-12-31 23:14:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void rr_init();
|
2009-02-02 18:45:00 +00:00
|
|
|
void accumulate_shortfall(double d_time);
|
2009-07-17 18:29:10 +00:00
|
|
|
void update_saturated_time(double dt);
|
|
|
|
void update_busy_time(double dur, double nused);
|
2010-12-30 22:41:50 +00:00
|
|
|
PROJECT* choose_project_hyst();
|
2009-02-23 20:35:52 +00:00
|
|
|
PROJECT* choose_project(int);
|
2010-12-30 22:41:50 +00:00
|
|
|
void supplement(PROJECT*);
|
2008-12-31 23:14:57 +00:00
|
|
|
RSC_PROJECT_WORK_FETCH& project_state(PROJECT*);
|
2009-04-03 21:55:26 +00:00
|
|
|
void print_state(const char*);
|
2009-01-03 06:01:17 +00:00
|
|
|
void clear_request();
|
2011-09-13 22:58:39 +00:00
|
|
|
void set_request(PROJECT*);
|
2009-01-08 00:41:15 +00:00
|
|
|
bool may_have_work(PROJECT*);
|
2008-12-31 23:14:57 +00:00
|
|
|
RSC_WORK_FETCH() {
|
2010-07-15 21:43:51 +00:00
|
|
|
rsc_type = 0;
|
|
|
|
ninstances = 0;
|
|
|
|
relative_speed = 0;
|
|
|
|
shortfall = 0;
|
|
|
|
nidle_now = 0;
|
|
|
|
sim_nused = 0;
|
|
|
|
total_fetchable_share = 0;
|
|
|
|
saturated_time = 0;
|
|
|
|
deadline_missed_instances = 0;
|
2008-12-31 23:14:57 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// per project state
|
|
|
|
//
|
|
|
|
struct PROJECT_WORK_FETCH {
|
2010-10-29 23:41:34 +00:00
|
|
|
double rec;
|
|
|
|
// recent estimated credit
|
|
|
|
double rec_time;
|
|
|
|
// when it was last updated
|
2010-11-23 19:39:47 +00:00
|
|
|
double rec_temp;
|
|
|
|
// temporary copy used during schedule_cpus()
|
|
|
|
bool can_fetch_work;
|
|
|
|
bool compute_can_fetch_work(PROJECT*);
|
|
|
|
bool has_runnable_jobs;
|
2008-12-31 23:14:57 +00:00
|
|
|
PROJECT_WORK_FETCH() {
|
|
|
|
memset(this, 0, sizeof(*this));
|
|
|
|
}
|
2009-01-23 18:29:28 +00:00
|
|
|
void reset(PROJECT*);
|
2008-12-31 23:14:57 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// global work fetch state
|
|
|
|
//
|
|
|
|
struct WORK_FETCH {
|
|
|
|
PROJECT* choose_project();
|
2009-01-03 06:01:17 +00:00
|
|
|
// find a project to ask for work
|
2009-01-21 20:51:33 +00:00
|
|
|
PROJECT* non_cpu_intensive_project_needing_work();
|
2009-01-03 06:01:17 +00:00
|
|
|
void compute_work_request(PROJECT*);
|
|
|
|
// we're going to contact this project anyway;
|
|
|
|
// decide how much work to task for
|
2008-12-31 23:14:57 +00:00
|
|
|
void accumulate_inst_sec(ACTIVE_TASK*, double dt);
|
2009-09-02 21:36:25 +00:00
|
|
|
void write_request(FILE*, PROJECT*);
|
2009-09-16 17:34:19 +00:00
|
|
|
void handle_reply(
|
|
|
|
PROJECT*, SCHEDULER_REPLY*, std::vector<RESULT*>new_results
|
|
|
|
);
|
2009-01-10 00:48:22 +00:00
|
|
|
void set_initial_work_request();
|
2009-12-24 20:40:27 +00:00
|
|
|
void set_all_requests(PROJECT*);
|
2010-12-30 22:41:50 +00:00
|
|
|
void set_all_requests_hyst(PROJECT*, int rsc_type);
|
2008-12-31 23:14:57 +00:00
|
|
|
void print_state();
|
|
|
|
void init();
|
|
|
|
void rr_init();
|
2009-01-03 06:01:17 +00:00
|
|
|
void clear_request();
|
|
|
|
void compute_shares();
|
2010-09-24 21:24:02 +00:00
|
|
|
void clear_backoffs(APP_VERSION&);
|
2010-09-27 20:34:47 +00:00
|
|
|
void request_string(char*);
|
2008-12-31 23:14:57 +00:00
|
|
|
};
|
|
|
|
|
2011-03-25 03:44:09 +00:00
|
|
|
extern RSC_WORK_FETCH rsc_work_fetch[MAX_RSC];
|
2008-12-31 23:14:57 +00:00
|
|
|
extern WORK_FETCH work_fetch;
|
|
|
|
|
2011-07-29 00:07:20 +00:00
|
|
|
extern void set_no_rsc_config();
|
|
|
|
|
2011-09-12 17:01:54 +00:00
|
|
|
void project_priority_init(bool init_rec_temp=true);
|
2010-11-23 19:39:47 +00:00
|
|
|
double project_priority(PROJECT*);
|
2011-07-13 19:46:03 +00:00
|
|
|
void adjust_rec_sched(RESULT*);
|
|
|
|
void adjust_rec_work_fetch(RESULT*);
|
2010-11-23 19:39:47 +00:00
|
|
|
|
2008-12-31 23:14:57 +00:00
|
|
|
#endif
|