2008-08-06 18:36:30 +00:00
|
|
|
// This file is part of BOINC.
|
2005-08-04 03:50:04 +00:00
|
|
|
// http://boinc.berkeley.edu
|
2008-08-06 18:36:30 +00:00
|
|
|
// Copyright (C) 2008 University of California
|
2005-08-04 03:50:04 +00:00
|
|
|
//
|
2008-08-06 18:36:30 +00:00
|
|
|
// 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.
|
2005-08-04 03:50:04 +00:00
|
|
|
//
|
2008-08-06 18:36:30 +00:00
|
|
|
// BOINC is distributed in the hope that it will be useful,
|
2005-08-04 03:50:04 +00:00
|
|
|
// 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.
|
|
|
|
//
|
2008-08-06 18:36:30 +00:00
|
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
|
|
// along with BOINC. If not, see <http://www.gnu.org/licenses/>.
|
2005-08-04 03:50:04 +00:00
|
|
|
|
|
|
|
// scheduler code related to sending work
|
|
|
|
|
2008-02-27 23:26:38 +00:00
|
|
|
#include <cstdlib>
|
|
|
|
#include <string>
|
|
|
|
#include <cstring>
|
2005-08-04 03:50:04 +00:00
|
|
|
|
2005-11-21 18:34:44 +00:00
|
|
|
#include "config.h"
|
2010-11-08 17:51:57 +00:00
|
|
|
|
2009-08-10 04:49:02 +00:00
|
|
|
#include "sched_main.h"
|
|
|
|
#include "sched_types.h"
|
2005-08-04 03:50:04 +00:00
|
|
|
#include "sched_shmem.h"
|
|
|
|
#include "sched_hr.h"
|
|
|
|
#include "sched_config.h"
|
|
|
|
#include "sched_util.h"
|
|
|
|
#include "sched_msgs.h"
|
|
|
|
#include "sched_send.h"
|
2009-03-19 16:35:35 +00:00
|
|
|
#include "sched_version.h"
|
2010-03-03 19:29:23 +00:00
|
|
|
#ifdef _USING_FCGI_
|
|
|
|
#include "boinc_fcgi.h"
|
|
|
|
#endif
|
2005-08-04 03:58:00 +00:00
|
|
|
|
|
|
|
#include "sched_array.h"
|
2005-08-04 03:50:04 +00:00
|
|
|
|
2010-03-03 19:29:23 +00:00
|
|
|
// do fast checks on this job, i.e. ones that don't require DB access
|
|
|
|
// if any check fails, return false
|
|
|
|
//
|
|
|
|
static bool quick_check(
|
|
|
|
WU_RESULT& wu_result, WORKUNIT& wu, BEST_APP_VERSION* &bavp,
|
|
|
|
APP* &app, int& last_retval
|
|
|
|
) {
|
|
|
|
int retval;
|
|
|
|
|
|
|
|
if (wu_result.state != WR_STATE_PRESENT && wu_result.state != g_pid) {
|
|
|
|
return false;
|
|
|
|
}
|
2011-03-04 19:40:59 +00:00
|
|
|
|
2010-03-03 19:29:23 +00:00
|
|
|
app = ssp->lookup_app(wu_result.workunit.appid);
|
|
|
|
if (app == NULL) {
|
|
|
|
return false; // this should never happen
|
|
|
|
}
|
2010-03-04 04:16:00 +00:00
|
|
|
|
2010-06-16 22:07:19 +00:00
|
|
|
g_wreq->no_jobs_available = false;
|
|
|
|
|
2010-03-04 04:16:00 +00:00
|
|
|
// If we're looking for beta jobs and this isn't one, skip it
|
|
|
|
//
|
2010-03-03 19:29:23 +00:00
|
|
|
if (g_wreq->beta_only) {
|
|
|
|
if (!app->beta) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (config.debug_send) {
|
|
|
|
log_messages.printf(MSG_NORMAL,
|
|
|
|
"[send] [HOST#%d] beta work found: [RESULT#%d]\n",
|
|
|
|
g_reply->host.id, wu_result.resultid
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (app->beta) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2011-03-04 19:40:59 +00:00
|
|
|
|
2010-03-03 19:29:23 +00:00
|
|
|
// If this is a reliable host and we are checking for results that
|
|
|
|
// need a reliable host, then continue if the result is a normal result
|
|
|
|
// skip if the app is beta (beta apps don't use the reliable mechanism)
|
|
|
|
//
|
|
|
|
if (!app->beta) {
|
|
|
|
if (g_wreq->reliable_only && (!wu_result.need_reliable)) {
|
|
|
|
return false;
|
|
|
|
} else if (!g_wreq->reliable_only && wu_result.need_reliable) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2011-03-04 19:40:59 +00:00
|
|
|
|
2010-03-03 19:29:23 +00:00
|
|
|
// don't send if we are looking for infeasible results
|
|
|
|
// and the result is not infeasible
|
|
|
|
//
|
|
|
|
if (g_wreq->infeasible_only && (wu_result.infeasible_count==0)) {
|
|
|
|
return false;
|
|
|
|
}
|
2011-03-04 19:40:59 +00:00
|
|
|
|
|
|
|
// Find the app and best app_version for this host.
|
|
|
|
//
|
|
|
|
bavp = get_app_version(wu, true, g_wreq->reliable_only);
|
|
|
|
if (!bavp) {
|
|
|
|
if (config.debug_array) {
|
|
|
|
log_messages.printf(MSG_NORMAL,
|
|
|
|
"[array] No app version\n"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check app filter if needed.
|
|
|
|
// Do this AFTER get_app_version(), otherwise we could send
|
|
|
|
// a misleading message to user
|
2010-03-03 19:29:23 +00:00
|
|
|
//
|
|
|
|
if (g_wreq->user_apps_only &&
|
|
|
|
(!g_wreq->beta_only || config.distinct_beta_apps)
|
|
|
|
) {
|
|
|
|
if (app_not_selected(wu)) {
|
|
|
|
g_wreq->no_allowed_apps_available = true;
|
|
|
|
#if 0
|
|
|
|
if (config.debug_send) {
|
|
|
|
log_messages.printf(MSG_NORMAL,
|
|
|
|
"[send] [USER#%d] [WU#%d] user doesn't want work for app %s\n",
|
|
|
|
g_reply->user.id, wu.id, app->name
|
|
|
|
);
|
|
|
|
}
|
2005-08-04 03:50:04 +00:00
|
|
|
#endif
|
2010-03-03 19:29:23 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// don't send job if host can't handle it
|
|
|
|
//
|
|
|
|
retval = wu_is_infeasible_fast(
|
|
|
|
wu,
|
|
|
|
wu_result.res_server_state, wu_result.res_priority,
|
|
|
|
wu_result.res_report_deadline,
|
|
|
|
*app, *bavp
|
|
|
|
);
|
|
|
|
if (retval) {
|
|
|
|
if (retval != last_retval && config.debug_send) {
|
|
|
|
log_messages.printf(MSG_NORMAL,
|
|
|
|
"[send] [HOST#%d] [WU#%d %s] WU is infeasible: %s\n",
|
|
|
|
g_reply->host.id, wu.id, wu.name, infeasible_string(retval)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
last_retval = retval;
|
|
|
|
if (config.debug_array) {
|
|
|
|
log_messages.printf(MSG_NORMAL, "[array] infeasible\n");
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// do slow checks (ones that require DB access)
|
2011-10-26 07:15:22 +00:00
|
|
|
// return true if OK to send
|
2010-03-03 19:29:23 +00:00
|
|
|
//
|
2011-10-26 07:15:22 +00:00
|
|
|
static bool slow_check(
|
|
|
|
WU_RESULT& wu_result, WORKUNIT& wu, APP* app, BEST_APP_VERSION* bavp
|
|
|
|
) {
|
2010-03-03 19:29:23 +00:00
|
|
|
int n, retval;
|
|
|
|
DB_RESULT result;
|
|
|
|
char buf[256];
|
|
|
|
|
|
|
|
// Don't send if we've already sent a result of this WU to this user.
|
|
|
|
//
|
|
|
|
if (config.one_result_per_user_per_wu) {
|
|
|
|
sprintf(buf,
|
|
|
|
"where workunitid=%d and userid=%d",
|
|
|
|
wu_result.workunit.id, g_reply->user.id
|
|
|
|
);
|
|
|
|
retval = result.count(n, buf);
|
|
|
|
if (retval) {
|
|
|
|
log_messages.printf(MSG_CRITICAL,
|
2010-11-08 17:51:57 +00:00
|
|
|
"send_work: can't get result count (%s)\n", boincerror(retval)
|
2010-03-03 19:29:23 +00:00
|
|
|
);
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
if (n>0) {
|
|
|
|
if (config.debug_send) {
|
|
|
|
log_messages.printf(MSG_NORMAL,
|
|
|
|
"[send] [USER#%d] already has %d result(s) for [WU#%d]\n",
|
|
|
|
g_reply->user.id, n, wu_result.workunit.id
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (config.one_result_per_host_per_wu) {
|
|
|
|
// Don't send if we've already sent a result
|
|
|
|
// of this WU to this host.
|
|
|
|
// We only have to check this
|
|
|
|
// if we don't send one result per user.
|
|
|
|
//
|
|
|
|
sprintf(buf,
|
|
|
|
"where workunitid=%d and hostid=%d",
|
|
|
|
wu_result.workunit.id, g_reply->host.id
|
|
|
|
);
|
|
|
|
retval = result.count(n, buf);
|
|
|
|
if (retval) {
|
|
|
|
log_messages.printf(MSG_CRITICAL,
|
2010-11-08 17:51:57 +00:00
|
|
|
"send_work: can't get result count (%s)\n", boincerror(retval)
|
2010-03-03 19:29:23 +00:00
|
|
|
);
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
if (n>0) {
|
|
|
|
if (config.debug_send) {
|
|
|
|
log_messages.printf(MSG_NORMAL,
|
|
|
|
"[send] [HOST#%d] already has %d result(s) for [WU#%d]\n",
|
|
|
|
g_reply->host.id, n, wu_result.workunit.id
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-26 07:15:22 +00:00
|
|
|
// Checks that require looking up the WU.
|
|
|
|
// Lump these together so we only do 1 lookup
|
|
|
|
//
|
|
|
|
if (app_hr_type(*app) || app->homogeneous_app_version) {
|
|
|
|
DB_WORKUNIT db_wu;
|
|
|
|
db_wu.id = wu_result.workunit.id;
|
|
|
|
int vals[2];
|
|
|
|
retval = db_wu.get_field_ints("hr_class, app_version_id", 2, vals);
|
|
|
|
if (retval) {
|
|
|
|
log_messages.printf(MSG_CRITICAL,
|
|
|
|
"can't get fields for [WU#%d]: %s\n", db_wu.id, boincerror(retval)
|
|
|
|
);
|
2010-03-03 19:29:23 +00:00
|
|
|
return false;
|
|
|
|
}
|
2011-10-26 07:15:22 +00:00
|
|
|
if (app_hr_type(*app)) {
|
|
|
|
wu_result.workunit.hr_class = vals[0];
|
|
|
|
if (already_sent_to_different_hr_class( wu_result.workunit, *app)) {
|
|
|
|
if (config.debug_send) {
|
|
|
|
log_messages.printf(MSG_NORMAL,
|
|
|
|
"[send] [HOST#%d] [WU#%d %s] is assigned to different HR class\n",
|
|
|
|
g_reply->host.id, wu.id, wu.name
|
|
|
|
);
|
|
|
|
}
|
|
|
|
// Mark the workunit as infeasible.
|
|
|
|
// This ensures that jobs already assigned to an HR class
|
|
|
|
// are processed first.
|
|
|
|
//
|
|
|
|
wu_result.infeasible_count++;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (app->homogeneous_app_version) {
|
|
|
|
int wu_avid = vals[1];
|
|
|
|
wu_result.workunit.app_version_id = wu_avid;
|
|
|
|
if (wu_avid && wu_avid != bavp->avp->id) {
|
|
|
|
if (config.debug_send) {
|
|
|
|
log_messages.printf(MSG_NORMAL,
|
|
|
|
"[send] [HOST#%d] [WU#%d %s] is assigned to different app version\n",
|
|
|
|
g_reply->host.id, wu.id, wu.name
|
|
|
|
);
|
|
|
|
}
|
|
|
|
wu_result.infeasible_count++;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2010-03-03 19:29:23 +00:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for pathological conditions that mean
|
|
|
|
// result is not sendable at all.
|
|
|
|
//
|
|
|
|
static bool result_still_sendable(DB_RESULT& result, WORKUNIT& wu) {
|
|
|
|
int retval = result.lookup_id(result.id);
|
|
|
|
if (retval) {
|
|
|
|
log_messages.printf(MSG_CRITICAL,
|
2010-11-08 17:51:57 +00:00
|
|
|
"[RESULT#%d] result.lookup_id() failed: %s\n",
|
|
|
|
result.id, boincerror(retval)
|
2010-03-03 19:29:23 +00:00
|
|
|
);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (result.server_state != RESULT_SERVER_STATE_UNSENT) {
|
|
|
|
log_messages.printf(MSG_NORMAL,
|
|
|
|
"[RESULT#%d] expected to be unsent; instead, state is %d\n",
|
|
|
|
result.id, result.server_state
|
|
|
|
);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (result.workunitid != wu.id) {
|
|
|
|
log_messages.printf(MSG_CRITICAL,
|
|
|
|
"[RESULT#%d] wrong WU ID: wanted %d, got %d\n",
|
|
|
|
result.id, wu.id, result.workunitid
|
|
|
|
);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2005-08-04 03:50:04 +00:00
|
|
|
|
|
|
|
// Make a pass through the wu/results array, sending work.
|
2008-12-22 00:10:02 +00:00
|
|
|
// The choice of jobs is limited by flags in g_wreq, as follows:
|
|
|
|
// infeasible_only:
|
|
|
|
// send only results that were previously infeasible for some host
|
2011-03-04 19:40:59 +00:00
|
|
|
// reliable_only:
|
2008-12-22 00:10:02 +00:00
|
|
|
// send only retries
|
|
|
|
// user_apps_only:
|
|
|
|
// Send only jobs for apps selected by user
|
|
|
|
// beta_only:
|
|
|
|
// Send only jobs for beta-test apps
|
|
|
|
//
|
2009-06-01 22:15:14 +00:00
|
|
|
// Return true if no more work is needed.
|
2005-08-04 03:50:04 +00:00
|
|
|
//
|
2009-06-01 22:15:14 +00:00
|
|
|
static bool scan_work_array() {
|
2010-03-03 19:29:23 +00:00
|
|
|
int i, j, retval, rnd_off, last_retval=0;;
|
2005-08-04 03:50:04 +00:00
|
|
|
APP* app;
|
2010-03-03 19:29:23 +00:00
|
|
|
BEST_APP_VERSION* bavp;
|
2009-06-01 22:15:14 +00:00
|
|
|
bool no_more_needed = false;
|
2011-06-06 03:40:42 +00:00
|
|
|
SCHED_DB_RESULT result;
|
2009-01-23 22:52:35 +00:00
|
|
|
|
2005-08-04 03:50:04 +00:00
|
|
|
lock_sema();
|
2011-03-04 19:40:59 +00:00
|
|
|
|
2008-02-22 22:21:00 +00:00
|
|
|
rnd_off = rand() % ssp->max_wu_results;
|
|
|
|
for (j=0; j<ssp->max_wu_results; j++) {
|
|
|
|
i = (j+rnd_off) % ssp->max_wu_results;
|
2005-08-04 03:50:04 +00:00
|
|
|
|
2008-02-22 22:21:00 +00:00
|
|
|
WU_RESULT& wu_result = ssp->wu_results[i];
|
2010-03-03 19:29:23 +00:00
|
|
|
WORKUNIT wu = wu_result.workunit;
|
2005-08-04 03:50:04 +00:00
|
|
|
|
2010-03-03 19:29:23 +00:00
|
|
|
// do fast (non-DB) checks
|
2005-08-04 03:50:04 +00:00
|
|
|
//
|
2010-03-03 19:29:23 +00:00
|
|
|
if (!quick_check(wu_result, wu, bavp, app, last_retval)) {
|
2005-08-04 03:50:04 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2007-06-14 18:02:00 +00:00
|
|
|
// mark wu_result as checked out and release semaphore.
|
2005-08-04 03:50:04 +00:00
|
|
|
// from here on in this loop, don't continue on failure;
|
|
|
|
// instead, goto dont_send (so that we reacquire semaphore)
|
2007-06-14 18:02:00 +00:00
|
|
|
//
|
|
|
|
// Note: without the semaphore we don't have mutual exclusion;
|
|
|
|
// ideally we should use a transaction from now until when
|
|
|
|
// we commit to sending the results.
|
2005-08-04 03:50:04 +00:00
|
|
|
|
2006-01-19 21:46:25 +00:00
|
|
|
wu_result.state = g_pid;
|
2005-08-04 03:50:04 +00:00
|
|
|
unlock_sema();
|
|
|
|
|
2011-10-26 07:15:22 +00:00
|
|
|
if (!slow_check(wu_result, wu, app, bavp)) {
|
2010-03-03 19:29:23 +00:00
|
|
|
// if we couldn't send the result to this host,
|
|
|
|
// set its state back to PRESENT
|
2007-03-11 03:15:17 +00:00
|
|
|
//
|
2010-03-03 19:29:23 +00:00
|
|
|
wu_result.state = WR_STATE_PRESENT;
|
|
|
|
} else {
|
|
|
|
result.id = wu_result.resultid;
|
2005-08-04 03:50:04 +00:00
|
|
|
|
2010-03-03 19:29:23 +00:00
|
|
|
// mark slot as empty AFTER we've copied out of it
|
|
|
|
// (since otherwise feeder might overwrite it)
|
|
|
|
//
|
|
|
|
wu_result.state = WR_STATE_EMPTY;
|
2005-08-04 03:50:04 +00:00
|
|
|
|
2010-03-03 19:29:23 +00:00
|
|
|
// reread result from DB, make sure it's still unsent
|
|
|
|
// TODO: from here to end of add_result_to_reply()
|
|
|
|
// (which updates the DB record) should be a transaction
|
|
|
|
//
|
|
|
|
if (result_still_sendable(result, wu)) {
|
|
|
|
retval = add_result_to_reply(result, wu, bavp, false);
|
|
|
|
|
|
|
|
// add_result_to_reply() fails only in pathological cases -
|
|
|
|
// e.g. we couldn't update the DB record or modify XML fields.
|
|
|
|
// If this happens, don't replace the record in the array
|
|
|
|
// (we can't anyway, since we marked the entry as "empty").
|
|
|
|
// The feeder will eventually pick it up again,
|
|
|
|
// and hopefully the problem won't happen twice.
|
|
|
|
}
|
2005-08-04 03:50:04 +00:00
|
|
|
}
|
|
|
|
lock_sema();
|
2009-06-01 22:15:14 +00:00
|
|
|
if (!work_needed(false)) {
|
|
|
|
no_more_needed = true;
|
|
|
|
break;
|
|
|
|
}
|
2005-08-04 03:50:04 +00:00
|
|
|
}
|
|
|
|
unlock_sema();
|
2009-06-01 22:15:14 +00:00
|
|
|
return no_more_needed;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Send work by scanning the job array multiple times,
|
|
|
|
// with different selection criteria on each scan.
|
|
|
|
// This has been superceded by send_work_matchmaker()
|
|
|
|
//
|
|
|
|
void send_work_old() {
|
|
|
|
g_wreq->beta_only = false;
|
|
|
|
g_wreq->user_apps_only = true;
|
|
|
|
g_wreq->infeasible_only = false;
|
|
|
|
|
|
|
|
// give top priority to results that require a 'reliable host'
|
|
|
|
//
|
- server: change the following from per-host to per-(host, app version):
- daily quota mechanism
- reliable mechanism (accelerated retries)
- "trusted" mechanism (adaptive replication)
- scheduler: enforce host scale probation only for apps with
host_scale_check set.
- validator: do scale probation on invalid results
(need this in addition to error and timeout cases)
- feeder: update app version scales every 10 min, not 10 sec
- back-end apps: support --foo as well as -foo for options
Notes:
- If you have, say, cuda, cuda23 and cuda_fermi plan classes,
a host will have separate quotas for each one.
That means it could error out on 100 jobs for cuda_fermi,
and when its quota goes to zero,
error out on 100 jobs for cuda23, etc.
This is intentional; there may be cases where one version
works but not the others.
- host.error_rate and host.max_results_day are deprecated
TODO:
- the values in the app table for limits on jobs in progress etc.
should override rather than config.xml.
Implementation notes:
scheduler:
process_request():
read all host_app_versions for host at start;
Compute "reliable" and "trusted" for each one.
write modified records at end
get_app_version():
add "reliable_only" arg; if set, use only reliable versions
skip over-quota versions
Multi-pass scheduling: if have at least one reliable version,
do a pass for jobs that need reliable,
and use only reliable versions.
Then clear best_app_versions cache.
Score-based scheduling: for need-reliable jobs,
it will pick the fastest version,
then give a score bonus if that version happens to be reliable.
When get back a successful result from client:
increase daily quota
When get back an error result from client:
impose scale probation
decrease daily quota if not aborted
Validator:
when handling a WU, create a vector of HOST_APP_VERSION
parallel to vector of RESULT.
Pass it to assign_credit_set().
Make copies of originals so we can update only modified ones
update HOST_APP_VERSION error rates
Transitioner:
decrease quota on timeout
svn path=/trunk/boinc/; revision=21181
2010-04-15 03:13:56 +00:00
|
|
|
if (g_wreq->has_reliable_version) {
|
2009-06-01 22:15:14 +00:00
|
|
|
g_wreq->reliable_only = true;
|
|
|
|
if (scan_work_array()) return;
|
- server: change the following from per-host to per-(host, app version):
- daily quota mechanism
- reliable mechanism (accelerated retries)
- "trusted" mechanism (adaptive replication)
- scheduler: enforce host scale probation only for apps with
host_scale_check set.
- validator: do scale probation on invalid results
(need this in addition to error and timeout cases)
- feeder: update app version scales every 10 min, not 10 sec
- back-end apps: support --foo as well as -foo for options
Notes:
- If you have, say, cuda, cuda23 and cuda_fermi plan classes,
a host will have separate quotas for each one.
That means it could error out on 100 jobs for cuda_fermi,
and when its quota goes to zero,
error out on 100 jobs for cuda23, etc.
This is intentional; there may be cases where one version
works but not the others.
- host.error_rate and host.max_results_day are deprecated
TODO:
- the values in the app table for limits on jobs in progress etc.
should override rather than config.xml.
Implementation notes:
scheduler:
process_request():
read all host_app_versions for host at start;
Compute "reliable" and "trusted" for each one.
write modified records at end
get_app_version():
add "reliable_only" arg; if set, use only reliable versions
skip over-quota versions
Multi-pass scheduling: if have at least one reliable version,
do a pass for jobs that need reliable,
and use only reliable versions.
Then clear best_app_versions cache.
Score-based scheduling: for need-reliable jobs,
it will pick the fastest version,
then give a score bonus if that version happens to be reliable.
When get back a successful result from client:
increase daily quota
When get back an error result from client:
impose scale probation
decrease daily quota if not aborted
Validator:
when handling a WU, create a vector of HOST_APP_VERSION
parallel to vector of RESULT.
Pass it to assign_credit_set().
Make copies of originals so we can update only modified ones
update HOST_APP_VERSION error rates
Transitioner:
decrease quota on timeout
svn path=/trunk/boinc/; revision=21181
2010-04-15 03:13:56 +00:00
|
|
|
g_wreq->reliable_only = false;
|
|
|
|
g_wreq->best_app_versions.clear();
|
2009-06-01 22:15:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// give 2nd priority to results for a beta app
|
|
|
|
// (projects should load beta work with care,
|
|
|
|
// otherwise your users won't get production work done!
|
|
|
|
//
|
|
|
|
if (g_wreq->allow_beta_work) {
|
|
|
|
g_wreq->beta_only = true;
|
|
|
|
if (config.debug_send) {
|
|
|
|
log_messages.printf(MSG_NORMAL,
|
|
|
|
"[send] [HOST#%d] will accept beta work. Scanning for beta work.\n",
|
|
|
|
g_reply->host.id
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (scan_work_array()) return;
|
|
|
|
}
|
|
|
|
g_wreq->beta_only = false;
|
|
|
|
|
|
|
|
// give next priority to results that were infeasible for some other host
|
|
|
|
//
|
|
|
|
g_wreq->infeasible_only = true;
|
|
|
|
if (scan_work_array()) return;;
|
|
|
|
|
2010-03-03 19:29:23 +00:00
|
|
|
// that takes care of high-priority cases. Now do general scan.
|
|
|
|
//
|
2009-06-01 22:15:14 +00:00
|
|
|
g_wreq->infeasible_only = false;
|
|
|
|
if (scan_work_array()) return;
|
|
|
|
|
|
|
|
// If user has selected apps but will accept any,
|
|
|
|
// and we haven't found any jobs for selected apps, try others
|
|
|
|
//
|
|
|
|
if (!g_wreq->njobs_sent && g_wreq->allow_non_preferred_apps ) {
|
|
|
|
g_wreq->user_apps_only = false;
|
|
|
|
preferred_app_message_index = g_wreq->no_work_messages.size();
|
|
|
|
if (config.debug_send) {
|
|
|
|
log_messages.printf(MSG_NORMAL,
|
|
|
|
"[send] [HOST#%d] is looking for work from a non-preferred application\n",
|
|
|
|
g_reply->host.id
|
|
|
|
);
|
|
|
|
}
|
|
|
|
scan_work_array();
|
|
|
|
}
|
2005-08-04 03:50:04 +00:00
|
|
|
}
|
|
|
|
|
2005-09-13 09:01:56 +00:00
|
|
|
const char *BOINC_RCSID_d9f764fd14="$Id$";
|