2008-08-06 18:36:30 +00:00
|
|
|
// This file is part of BOINC.
|
2005-01-20 23:22:22 +00:00
|
|
|
// http://boinc.berkeley.edu
|
2008-08-06 18:36:30 +00:00
|
|
|
// Copyright (C) 2008 University of California
|
2003-07-02 02:02:18 +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.
|
2003-07-02 02:02:18 +00:00
|
|
|
//
|
2008-08-06 18:36:30 +00:00
|
|
|
// BOINC is distributed in the hope that it will be useful,
|
2005-01-20 23:22:22 +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.
|
2002-04-30 22:22:54 +00:00
|
|
|
//
|
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/>.
|
2002-04-30 22:22:54 +00:00
|
|
|
|
2003-02-24 21:25:16 +00:00
|
|
|
// High-level logic for communicating with scheduling servers,
|
2002-07-15 05:34:32 +00:00
|
|
|
// and for merging the result of a scheduler RPC into the client state
|
2002-06-21 06:52:47 +00:00
|
|
|
|
2004-01-30 22:19:19 +00:00
|
|
|
// The scheduler RPC mechanism is in scheduler_op.C
|
2002-06-21 06:52:47 +00:00
|
|
|
|
2003-10-16 19:03:49 +00:00
|
|
|
#include "cpp.h"
|
2004-03-04 11:41:43 +00:00
|
|
|
|
|
|
|
#ifdef _WIN32
|
2004-06-16 23:16:08 +00:00
|
|
|
#include "boinc_win.h"
|
2010-05-11 19:10:29 +00:00
|
|
|
#else
|
2005-11-21 18:34:44 +00:00
|
|
|
#include "config.h"
|
2009-02-26 00:23:23 +00:00
|
|
|
#include <cstdio>
|
|
|
|
#include <cmath>
|
|
|
|
#include <ctime>
|
|
|
|
#include <cstring>
|
2005-04-13 18:38:55 +00:00
|
|
|
#include <map>
|
|
|
|
#include <set>
|
2004-01-10 06:40:46 +00:00
|
|
|
#endif
|
2002-04-30 22:22:54 +00:00
|
|
|
|
2002-07-07 20:39:24 +00:00
|
|
|
#include "crypt.h"
|
2002-04-30 22:22:54 +00:00
|
|
|
#include "error_numbers.h"
|
|
|
|
#include "file_names.h"
|
2002-08-22 21:47:19 +00:00
|
|
|
#include "filesys.h"
|
2002-04-30 22:22:54 +00:00
|
|
|
#include "parse.h"
|
2007-02-21 16:26:51 +00:00
|
|
|
#include "str_util.h"
|
2009-06-16 20:54:44 +00:00
|
|
|
#include "str_replace.h"
|
2009-11-05 18:02:51 +00:00
|
|
|
#include "url.h"
|
2003-02-24 21:25:16 +00:00
|
|
|
#include "util.h"
|
2003-02-21 01:38:16 +00:00
|
|
|
|
2004-04-08 08:15:23 +00:00
|
|
|
#include "client_msgs.h"
|
2009-12-21 17:49:28 +00:00
|
|
|
#include "cs_notice.h"
|
2011-09-20 18:49:38 +00:00
|
|
|
#include "cs_trickle.h"
|
2012-04-30 21:00:28 +00:00
|
|
|
#include "project.h"
|
|
|
|
#include "result.h"
|
2002-06-21 06:52:47 +00:00
|
|
|
#include "scheduler_op.h"
|
2008-09-24 23:42:56 +00:00
|
|
|
#include "sandbox.h"
|
2002-04-30 22:22:54 +00:00
|
|
|
|
|
|
|
#include "client_state.h"
|
|
|
|
|
2004-07-02 03:20:17 +00:00
|
|
|
using std::max;
|
2004-06-30 18:17:21 +00:00
|
|
|
using std::vector;
|
|
|
|
using std::string;
|
|
|
|
|
2002-04-30 22:22:54 +00:00
|
|
|
// quantities like avg CPU time decay by a factor of e every week
|
2002-07-15 05:34:32 +00:00
|
|
|
//
|
2003-02-24 21:25:16 +00:00
|
|
|
#define EXP_DECAY_RATE (1./(SECONDS_PER_DAY*7))
|
2002-04-30 22:22:54 +00:00
|
|
|
|
2004-06-18 22:56:46 +00:00
|
|
|
// try to report results this much before their deadline
|
2004-05-05 21:15:34 +00:00
|
|
|
//
|
2006-08-23 21:14:47 +00:00
|
|
|
#define REPORT_DEADLINE_CUSHION ((double)SECONDS_PER_DAY)
|
2003-08-07 23:08:15 +00:00
|
|
|
|
2013-12-30 17:53:35 +00:00
|
|
|
// report results within this time after completion
|
|
|
|
//
|
|
|
|
#define MAX_REPORT_DELAY 3600
|
|
|
|
|
2008-12-30 19:01:25 +00:00
|
|
|
#ifndef SIM
|
2002-04-30 22:22:54 +00:00
|
|
|
|
2005-10-04 21:44:58 +00:00
|
|
|
// Write a scheduler request to a disk file,
|
|
|
|
// to be sent to a scheduling server
|
2002-07-15 23:21:20 +00:00
|
|
|
//
|
2005-05-31 23:43:34 +00:00
|
|
|
int CLIENT_STATE::make_scheduler_request(PROJECT* p) {
|
2004-09-22 22:09:54 +00:00
|
|
|
char buf[1024];
|
2004-06-12 04:45:36 +00:00
|
|
|
MIOFILE mf;
|
2002-04-30 22:22:54 +00:00
|
|
|
unsigned int i;
|
|
|
|
RESULT* rp;
|
2005-10-04 21:44:58 +00:00
|
|
|
|
2007-03-13 19:33:27 +00:00
|
|
|
get_sched_request_filename(*p, buf, sizeof(buf));
|
2005-10-04 21:44:58 +00:00
|
|
|
FILE* f = boinc_fopen(buf, "wb");
|
2007-06-29 15:45:15 +00:00
|
|
|
if (!f) return ERR_FOPEN;
|
2002-08-15 22:03:41 +00:00
|
|
|
|
2005-10-04 21:44:58 +00:00
|
|
|
double trs = total_resource_share();
|
2009-12-02 03:41:52 +00:00
|
|
|
double rrs = runnable_resource_share(RSC_TYPE_ANY);
|
2005-06-14 20:31:15 +00:00
|
|
|
double prrs = potentially_runnable_resource_share();
|
2005-10-04 21:44:58 +00:00
|
|
|
double resource_share_fraction, rrs_fraction, prrs_fraction;
|
|
|
|
if (trs) {
|
|
|
|
resource_share_fraction = p->resource_share / trs;
|
|
|
|
} else {
|
|
|
|
resource_share_fraction = 1;
|
|
|
|
}
|
|
|
|
if (rrs) {
|
|
|
|
rrs_fraction = p->resource_share / rrs;
|
|
|
|
} else {
|
|
|
|
rrs_fraction = 1;
|
|
|
|
}
|
2005-08-31 18:20:06 +00:00
|
|
|
if (prrs) {
|
2005-10-04 21:44:58 +00:00
|
|
|
prrs_fraction = p->resource_share / prrs;
|
2005-08-31 18:20:06 +00:00
|
|
|
} else {
|
2005-10-04 21:44:58 +00:00
|
|
|
prrs_fraction = 1;
|
2005-08-31 18:20:06 +00:00
|
|
|
}
|
2004-07-06 21:51:49 +00:00
|
|
|
|
2006-03-03 20:22:25 +00:00
|
|
|
// if hostid is zero, rpc_seqno better be also
|
|
|
|
//
|
|
|
|
if (!p->hostid) {
|
|
|
|
p->rpc_seqno = 0;
|
|
|
|
}
|
|
|
|
|
2004-06-12 04:45:36 +00:00
|
|
|
mf.init_file(f);
|
2002-04-30 22:22:54 +00:00
|
|
|
fprintf(f,
|
|
|
|
"<scheduler_request>\n"
|
|
|
|
" <authenticator>%s</authenticator>\n"
|
|
|
|
" <hostid>%d</hostid>\n"
|
|
|
|
" <rpc_seqno>%d</rpc_seqno>\n"
|
2002-12-01 06:14:28 +00:00
|
|
|
" <core_client_major_version>%d</core_client_major_version>\n"
|
|
|
|
" <core_client_minor_version>%d</core_client_minor_version>\n"
|
2005-08-31 00:18:36 +00:00
|
|
|
" <core_client_release>%d</core_client_release>\n"
|
2004-07-14 20:55:26 +00:00
|
|
|
" <resource_share_fraction>%f</resource_share_fraction>\n"
|
2005-10-04 21:44:58 +00:00
|
|
|
" <rrs_fraction>%f</rrs_fraction>\n"
|
|
|
|
" <prrs_fraction>%f</prrs_fraction>\n"
|
2008-09-24 23:42:56 +00:00
|
|
|
" <duration_correction_factor>%f</duration_correction_factor>\n"
|
- scheduler and client: fix the "allow multiple clients" feature.
This feature lets you run the BOINC client as a job on grid systems
that handle only 1-CPU jobs;
it disables various mechanisms that prevent multiple clients per host
(which is normally a bad thing).
Old:
- Run the client with a --allow_multiple_clients flag.
This tells it not to use a mutex that prevents
multiple clients per host.
- Run the project with the <multiple_clients_per_host> config flag.
This suppresses two mechanisms:
- (avoid duplicate host records)
on a scheduler request with no host ID,
looks for a host with same domain name, OS type,
and mem size, and assumes the request is from that host
- (job retry)
If we get a request that doesn't have a host ID
but does have a host CPID,
mark its in-progress results as over
NOTE: I CAN'T REMEMBER WHY WE SUPPRESS THIS;
MARK S, DO YOU REMEMBER?
Problem:
if the grid clients attach to a project that
doesn't use <multiple_clients_per_host>, bad things happen.
E.g., if there are several requests at about the same time,
most of them will fail with
"another RPC already in progress" errors.
If a project does include this flag,
it loses protection from duplicate host records.
New:
- If the client is run with --allow_multiple_clients flag,
it passes a <allow_multiple_clients> element
in scheduler requests.
- The scheduler skips the duplicate-host check on
requests that include this flag.
- There is no more <multiple_clients_per_host> scheduler option.
Note: if a project using the old mechanism upgrades to this change,
it will need to use new clients for its grid deployment.
svn path=/trunk/boinc/; revision=21839
2010-06-29 16:37:28 +00:00
|
|
|
" <allow_multiple_clients>%d</allow_multiple_clients>\n"
|
2014-07-03 07:33:24 +00:00
|
|
|
" <sandbox>%d</sandbox>\n"
|
|
|
|
" <dont_send_work>%d</dont_send_work>\n",
|
2002-04-30 22:22:54 +00:00
|
|
|
p->authenticator,
|
|
|
|
p->hostid,
|
|
|
|
p->rpc_seqno,
|
2006-12-14 00:42:43 +00:00
|
|
|
core_client_version.major,
|
|
|
|
core_client_version.minor,
|
|
|
|
core_client_version.release,
|
2005-08-31 18:20:06 +00:00
|
|
|
resource_share_fraction,
|
2005-10-04 21:44:58 +00:00
|
|
|
rrs_fraction,
|
|
|
|
prrs_fraction,
|
2008-09-24 23:42:56 +00:00
|
|
|
p->duration_correction_factor,
|
2014-05-08 07:51:18 +00:00
|
|
|
cc_config.allow_multiple_clients?1:0,
|
2014-07-03 07:33:24 +00:00
|
|
|
g_use_sandbox?1:0,
|
|
|
|
p->dont_request_more_work?1:0
|
2002-04-30 22:22:54 +00:00
|
|
|
);
|
2009-09-02 21:36:25 +00:00
|
|
|
work_fetch.write_request(f, p);
|
2007-05-02 17:53:35 +00:00
|
|
|
|
2008-04-02 19:05:08 +00:00
|
|
|
// write client capabilities
|
|
|
|
//
|
|
|
|
fprintf(f,
|
2008-04-07 16:04:02 +00:00
|
|
|
" <client_cap_plan_class>1</client_cap_plan_class>\n"
|
2008-04-02 19:05:08 +00:00
|
|
|
);
|
|
|
|
|
2007-05-06 20:22:39 +00:00
|
|
|
write_platforms(p, mf);
|
2007-05-02 17:53:35 +00:00
|
|
|
|
2003-02-11 19:41:08 +00:00
|
|
|
if (strlen(p->code_sign_key)) {
|
2011-08-31 16:36:17 +00:00
|
|
|
fprintf(f, " <code_sign_key>\n%s\n</code_sign_key>\n", p->code_sign_key);
|
2002-07-07 20:39:24 +00:00
|
|
|
}
|
2002-04-30 22:22:54 +00:00
|
|
|
|
2011-01-06 23:09:13 +00:00
|
|
|
// send working prefs
|
|
|
|
//
|
|
|
|
fprintf(f, "<working_global_preferences>\n");
|
|
|
|
global_prefs.write(mf);
|
|
|
|
fprintf(f, "</working_global_preferences>\n");
|
2006-10-06 19:16:59 +00:00
|
|
|
|
|
|
|
// send master global preferences if present and not host-specific
|
2002-09-29 00:32:11 +00:00
|
|
|
//
|
2006-10-06 19:16:59 +00:00
|
|
|
if (!global_prefs.host_specific && boinc_file_exists(GLOBAL_PREFS_FILE_NAME)) {
|
2004-03-01 19:11:12 +00:00
|
|
|
FILE* fprefs = fopen(GLOBAL_PREFS_FILE_NAME, "r");
|
|
|
|
if (fprefs) {
|
|
|
|
copy_stream(fprefs, f);
|
|
|
|
fclose(fprefs);
|
|
|
|
}
|
2004-10-06 19:09:37 +00:00
|
|
|
PROJECT* pp = lookup_project(global_prefs.source_project);
|
2004-05-03 02:18:35 +00:00
|
|
|
if (pp && strlen(pp->email_hash)) {
|
|
|
|
fprintf(f,
|
|
|
|
"<global_prefs_source_email_hash>%s</global_prefs_source_email_hash>\n",
|
|
|
|
pp->email_hash
|
|
|
|
);
|
|
|
|
}
|
2002-09-29 00:32:11 +00:00
|
|
|
}
|
2002-04-30 22:22:54 +00:00
|
|
|
|
2005-02-08 21:12:41 +00:00
|
|
|
// Of the projects with same email hash as this one,
|
|
|
|
// send the oldest cross-project ID.
|
|
|
|
// Use project URL as tie-breaker.
|
2004-04-18 18:40:13 +00:00
|
|
|
//
|
2005-02-08 21:12:41 +00:00
|
|
|
PROJECT* winner = p;
|
2004-04-18 18:40:13 +00:00
|
|
|
for (i=0; i<projects.size(); i++ ) {
|
|
|
|
PROJECT* project = projects[i];
|
|
|
|
if (project == p) continue;
|
|
|
|
if (strcmp(project->email_hash, p->email_hash)) continue;
|
2007-09-03 23:00:22 +00:00
|
|
|
if (project->cpid_time < winner->cpid_time) {
|
2005-02-08 21:12:41 +00:00
|
|
|
winner = project;
|
2007-09-03 23:00:22 +00:00
|
|
|
} else if (project->cpid_time == winner->cpid_time) {
|
2005-02-08 21:12:41 +00:00
|
|
|
if (strcmp(project->master_url, winner->master_url) < 0) {
|
|
|
|
winner = project;
|
|
|
|
}
|
2004-04-18 18:40:13 +00:00
|
|
|
}
|
|
|
|
}
|
2005-02-08 21:12:41 +00:00
|
|
|
fprintf(f,
|
|
|
|
"<cross_project_id>%s</cross_project_id>\n",
|
|
|
|
winner->cross_project_id
|
|
|
|
);
|
2004-04-18 18:40:13 +00:00
|
|
|
|
2010-04-24 03:18:24 +00:00
|
|
|
time_stats.write(mf, true);
|
|
|
|
net_stats.write(mf);
|
|
|
|
if (global_prefs.daily_xfer_period_days) {
|
2010-04-27 17:54:29 +00:00
|
|
|
daily_xfer_history.write_scheduler_request(
|
|
|
|
mf, global_prefs.daily_xfer_period_days
|
|
|
|
);
|
2010-04-24 03:18:24 +00:00
|
|
|
}
|
2005-10-04 21:44:58 +00:00
|
|
|
|
2006-01-16 20:08:50 +00:00
|
|
|
// update hardware info, and write host info
|
2005-10-04 21:44:58 +00:00
|
|
|
//
|
2014-11-25 22:46:58 +00:00
|
|
|
host_info.get_host_info(false);
|
2009-11-09 19:51:31 +00:00
|
|
|
set_ncpus();
|
2014-05-08 07:51:18 +00:00
|
|
|
host_info.write(mf, !cc_config.suppress_net_info, false);
|
2005-10-04 21:44:58 +00:00
|
|
|
|
|
|
|
// get and write disk usage
|
|
|
|
//
|
2011-12-28 14:52:03 +00:00
|
|
|
get_disk_usages();
|
2012-01-02 05:53:42 +00:00
|
|
|
get_disk_shares();
|
2005-10-04 21:44:58 +00:00
|
|
|
fprintf(f,
|
|
|
|
" <disk_usage>\n"
|
|
|
|
" <d_boinc_used_total>%f</d_boinc_used_total>\n"
|
|
|
|
" <d_boinc_used_project>%f</d_boinc_used_project>\n"
|
2012-01-02 05:53:42 +00:00
|
|
|
" <d_project_share>%f</d_project_share>\n"
|
2005-10-04 21:44:58 +00:00
|
|
|
" </disk_usage>\n",
|
2012-01-02 05:53:42 +00:00
|
|
|
total_disk_usage, p->disk_usage, p->disk_share
|
2005-10-04 21:44:58 +00:00
|
|
|
);
|
|
|
|
|
2011-03-25 03:44:09 +00:00
|
|
|
if (coprocs.n_rsc > 1) {
|
2014-08-10 04:44:39 +00:00
|
|
|
work_fetch.copy_requests();
|
2011-03-25 03:44:09 +00:00
|
|
|
coprocs.write_xml(mf, true);
|
2008-03-10 20:40:35 +00:00
|
|
|
}
|
|
|
|
|
2010-10-14 19:07:02 +00:00
|
|
|
// report completed jobs
|
2005-10-04 21:44:58 +00:00
|
|
|
//
|
2010-10-14 08:11:40 +00:00
|
|
|
unsigned int last_reported_index = 0;
|
2005-05-31 23:43:34 +00:00
|
|
|
p->nresults_returned = 0;
|
2002-04-30 22:22:54 +00:00
|
|
|
for (i=0; i<results.size(); i++) {
|
2003-02-06 19:01:49 +00:00
|
|
|
rp = results[i];
|
2003-09-23 23:19:41 +00:00
|
|
|
if (rp->project == p && rp->ready_to_report) {
|
2005-05-31 23:43:34 +00:00
|
|
|
p->nresults_returned++;
|
2004-06-12 04:45:36 +00:00
|
|
|
rp->write(mf, true);
|
2003-06-18 23:41:31 +00:00
|
|
|
}
|
2014-05-08 07:51:18 +00:00
|
|
|
if (cc_config.max_tasks_reported
|
|
|
|
&& (p->nresults_returned >= cc_config.max_tasks_reported)
|
2010-10-07 16:02:44 +00:00
|
|
|
) {
|
2010-10-13 23:21:19 +00:00
|
|
|
last_reported_index = i;
|
2010-10-07 16:02:44 +00:00
|
|
|
break;
|
|
|
|
}
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
2004-07-06 17:37:58 +00:00
|
|
|
|
2004-01-08 00:27:59 +00:00
|
|
|
read_trickle_files(p, f);
|
2004-09-10 00:41:48 +00:00
|
|
|
|
|
|
|
// report sticky files as needed
|
|
|
|
//
|
|
|
|
for (i=0; i<file_infos.size(); i++) {
|
|
|
|
FILE_INFO* fip = file_infos[i];
|
|
|
|
if (fip->project != p) continue;
|
2011-04-24 21:33:51 +00:00
|
|
|
if (!fip->sticky) continue;
|
2004-09-10 00:41:48 +00:00
|
|
|
fprintf(f,
|
|
|
|
" <file_info>\n"
|
|
|
|
" <name>%s</name>\n"
|
2005-01-28 20:03:57 +00:00
|
|
|
" <nbytes>%f</nbytes>\n"
|
|
|
|
" <status>%d</status>\n"
|
2004-09-10 00:41:48 +00:00
|
|
|
" </file_info>\n",
|
2005-01-28 20:03:57 +00:00
|
|
|
fip->name, fip->nbytes, fip->status
|
2004-09-10 00:41:48 +00:00
|
|
|
);
|
|
|
|
}
|
2005-05-31 21:59:29 +00:00
|
|
|
|
2007-04-13 22:55:18 +00:00
|
|
|
if (p->send_time_stats_log) {
|
|
|
|
fprintf(f, "<time_stats_log>\n");
|
2007-05-10 16:23:10 +00:00
|
|
|
time_stats.get_log_after(p->send_time_stats_log, mf);
|
2007-04-13 22:55:18 +00:00
|
|
|
fprintf(f, "</time_stats_log>\n");
|
|
|
|
}
|
|
|
|
if (p->send_job_log) {
|
|
|
|
fprintf(f, "<job_log>\n");
|
|
|
|
job_log_filename(*p, buf, sizeof(buf));
|
|
|
|
send_log_after(buf, p->send_job_log, mf);
|
|
|
|
fprintf(f, "</job_log>\n");
|
|
|
|
}
|
|
|
|
|
2010-05-13 20:18:27 +00:00
|
|
|
// send descriptions of app versions
|
|
|
|
//
|
|
|
|
fprintf(f, "<app_versions>\n");
|
2014-08-10 04:44:39 +00:00
|
|
|
int j=0;
|
2010-05-13 20:18:27 +00:00
|
|
|
for (i=0; i<app_versions.size(); i++) {
|
|
|
|
APP_VERSION* avp = app_versions[i];
|
2010-05-14 17:39:20 +00:00
|
|
|
if (avp->project != p) continue;
|
2010-05-14 03:08:23 +00:00
|
|
|
avp->write(mf, false);
|
2010-07-23 17:43:20 +00:00
|
|
|
avp->index = j++;
|
2010-05-13 20:18:27 +00:00
|
|
|
}
|
|
|
|
fprintf(f, "</app_versions>\n");
|
|
|
|
|
2010-10-14 19:07:02 +00:00
|
|
|
// send descriptions of jobs in progress for this project
|
2005-05-31 21:59:29 +00:00
|
|
|
//
|
|
|
|
fprintf(f, "<other_results>\n");
|
|
|
|
for (i=0; i<results.size(); i++) {
|
|
|
|
rp = results[i];
|
2010-10-13 23:21:19 +00:00
|
|
|
if (rp->project != p) continue;
|
|
|
|
if ((last_reported_index && (i > last_reported_index)) || !rp->ready_to_report) {
|
2005-05-31 21:59:29 +00:00
|
|
|
fprintf(f,
|
|
|
|
" <other_result>\n"
|
|
|
|
" <name>%s</name>\n"
|
2010-05-13 20:18:27 +00:00
|
|
|
" <app_version>%d</app_version>\n",
|
2009-05-31 16:38:37 +00:00
|
|
|
rp->name,
|
2010-05-13 20:18:27 +00:00
|
|
|
rp->avp->index
|
|
|
|
);
|
|
|
|
// the following is for backwards compatibility w/ old schedulers
|
|
|
|
//
|
|
|
|
if (strlen(rp->avp->plan_class)) {
|
|
|
|
fprintf(f,
|
2010-05-13 22:19:02 +00:00
|
|
|
" <plan_class>%s</plan_class>\n",
|
2010-05-13 20:18:27 +00:00
|
|
|
rp->avp->plan_class
|
|
|
|
);
|
|
|
|
}
|
|
|
|
fprintf(f,
|
|
|
|
" </other_result>\n"
|
2005-05-31 21:59:29 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fprintf(f, "</other_results>\n");
|
|
|
|
|
2010-05-13 20:18:27 +00:00
|
|
|
// if requested by project, send summary of all in-progress results
|
|
|
|
// (for EDF simulation by scheduler)
|
2005-05-31 21:59:29 +00:00
|
|
|
//
|
2010-05-13 20:18:27 +00:00
|
|
|
if (p->send_full_workload) {
|
|
|
|
fprintf(f, "<in_progress_results>\n");
|
|
|
|
for (i=0; i<results.size(); i++) {
|
|
|
|
rp = results[i];
|
2012-04-25 04:10:29 +00:00
|
|
|
double x = rp->estimated_runtime_remaining();
|
2010-05-13 20:18:27 +00:00
|
|
|
if (x == 0) continue;
|
2011-03-25 03:44:09 +00:00
|
|
|
strcpy(buf, "");
|
|
|
|
int rt = rp->avp->gpu_usage.rsc_type;
|
|
|
|
if (rt) {
|
2011-08-31 11:08:13 +00:00
|
|
|
if (rt == rsc_index(GPU_TYPE_NVIDIA)) {
|
2012-12-08 18:16:28 +00:00
|
|
|
sprintf(buf,
|
|
|
|
" <ncudas>%f</ncudas>\n",
|
|
|
|
rp->avp->gpu_usage.usage
|
|
|
|
);
|
2011-08-31 11:08:13 +00:00
|
|
|
} else if (rt == rsc_index(GPU_TYPE_ATI)) {
|
2012-12-08 18:16:28 +00:00
|
|
|
sprintf(buf,
|
|
|
|
" <natis>%f</natis>\n",
|
|
|
|
rp->avp->gpu_usage.usage
|
|
|
|
);
|
2012-12-05 13:11:20 +00:00
|
|
|
} else if (rt == rsc_index(GPU_TYPE_INTEL)) {
|
2012-12-08 18:16:28 +00:00
|
|
|
sprintf(buf,
|
|
|
|
" <nintel_gpus>%f</nintel_gpus>\n",
|
|
|
|
rp->avp->gpu_usage.usage
|
|
|
|
);
|
2011-03-25 03:44:09 +00:00
|
|
|
}
|
|
|
|
}
|
2010-05-13 20:18:27 +00:00
|
|
|
fprintf(f,
|
|
|
|
" <ip_result>\n"
|
|
|
|
" <name>%s</name>\n"
|
|
|
|
" <report_deadline>%.0f</report_deadline>\n"
|
|
|
|
" <time_remaining>%.2f</time_remaining>\n"
|
|
|
|
" <avg_ncpus>%f</avg_ncpus>\n"
|
2011-03-25 03:44:09 +00:00
|
|
|
"%s"
|
2010-05-13 20:18:27 +00:00
|
|
|
" </ip_result>\n",
|
|
|
|
rp->name,
|
|
|
|
rp->report_deadline,
|
|
|
|
x,
|
|
|
|
rp->avp->avg_ncpus,
|
2011-03-25 03:44:09 +00:00
|
|
|
buf
|
2010-05-13 20:18:27 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
fprintf(f, "</in_progress_results>\n");
|
2005-05-31 21:59:29 +00:00
|
|
|
}
|
2011-11-14 06:27:36 +00:00
|
|
|
FILE* cof = boinc_fopen(CLIENT_OPAQUE_FILENAME, "r");
|
|
|
|
if (cof) {
|
|
|
|
fprintf(f, "<client_opaque>\n<![CDATA[\n");
|
|
|
|
copy_stream(cof, f);
|
|
|
|
fprintf(f, "\n]]>\n</client_opaque>\n");
|
2012-11-01 17:54:48 +00:00
|
|
|
fclose(cof);
|
2011-11-14 06:27:36 +00:00
|
|
|
}
|
2010-05-13 20:18:27 +00:00
|
|
|
|
2014-07-17 03:18:06 +00:00
|
|
|
if (strlen(client_brand)) {
|
|
|
|
fprintf(f, " <client_brand>%s</client_brand>\n", client_brand);
|
|
|
|
}
|
|
|
|
|
2002-04-30 22:22:54 +00:00
|
|
|
fprintf(f, "</scheduler_request>\n");
|
2004-07-06 17:37:58 +00:00
|
|
|
|
2004-07-06 18:31:56 +00:00
|
|
|
fclose(f);
|
2002-04-30 22:22:54 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-04-08 09:02:41 +00:00
|
|
|
// the project is uploading, and it started recently
|
|
|
|
//
|
|
|
|
static inline bool actively_uploading(PROJECT* p) {
|
2013-06-15 05:40:43 +00:00
|
|
|
for (unsigned int i=0; i<gstate.file_xfers->file_xfers.size(); i++) {
|
|
|
|
FILE_XFER* fxp = gstate.file_xfers->file_xfers[i];
|
|
|
|
if (fxp->fip->project != p) continue;
|
|
|
|
if (!fxp->is_upload) continue;
|
2014-02-07 23:53:13 +00:00
|
|
|
if (gstate.now - fxp->start_time > WF_UPLOAD_DEFER_INTERVAL) continue;
|
2013-06-15 05:40:43 +00:00
|
|
|
//msg_printf(p, MSG_INFO, "actively uploading");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
//msg_printf(p, MSG_INFO, "not actively uploading");
|
|
|
|
return false;
|
2011-04-08 09:02:41 +00:00
|
|
|
}
|
|
|
|
|
2013-06-22 02:57:25 +00:00
|
|
|
// If there is a request for an idle instance, return true.
|
|
|
|
// Clear other requests
|
|
|
|
//
|
|
|
|
static inline bool idle_request() {
|
|
|
|
bool found = false;
|
|
|
|
for (int i=0; i<coprocs.n_rsc; i++) {
|
|
|
|
RSC_WORK_FETCH &rwf = rsc_work_fetch[i];
|
|
|
|
if (rwf.req_instances) {
|
|
|
|
found = true;
|
|
|
|
} else {
|
|
|
|
rwf.req_secs = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
2012-11-19 20:38:54 +00:00
|
|
|
// Called once/sec.
|
|
|
|
// Initiate scheduler RPC activity if needed and possible
|
2002-07-15 05:34:32 +00:00
|
|
|
//
|
2005-06-07 19:22:50 +00:00
|
|
|
bool CLIENT_STATE::scheduler_rpc_poll() {
|
2004-07-14 20:55:26 +00:00
|
|
|
PROJECT *p;
|
2004-10-14 22:01:05 +00:00
|
|
|
static double last_time=0;
|
2008-12-31 23:07:59 +00:00
|
|
|
static double last_work_fetch_time = 0;
|
2009-02-10 21:59:55 +00:00
|
|
|
double elapsed_time;
|
2004-10-14 22:01:05 +00:00
|
|
|
|
2012-11-19 20:38:54 +00:00
|
|
|
// are we currently doing a scheduler RPC?
|
|
|
|
// If so, see if it's finished
|
|
|
|
//
|
2010-04-26 19:31:28 +00:00
|
|
|
if (scheduler_op->state != SCHEDULER_OP_STATE_IDLE) {
|
|
|
|
last_time = now;
|
|
|
|
scheduler_op->poll();
|
|
|
|
return (scheduler_op->state == SCHEDULER_OP_STATE_IDLE);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (network_suspended) return false;
|
|
|
|
|
2011-01-06 23:09:13 +00:00
|
|
|
// check only every 5 sec
|
|
|
|
//
|
2013-03-15 03:43:29 +00:00
|
|
|
if (!clock_change && now - last_time < SCHEDULER_RPC_POLL_PERIOD) return false;
|
2007-05-10 16:23:10 +00:00
|
|
|
last_time = now;
|
2002-07-15 05:34:32 +00:00
|
|
|
|
2010-04-26 19:31:28 +00:00
|
|
|
if (scheduler_op->check_master_fetch_start()) {
|
|
|
|
return true;
|
|
|
|
}
|
2005-06-09 00:43:52 +00:00
|
|
|
|
2010-04-26 19:31:28 +00:00
|
|
|
// If we haven't run benchmarks yet, don't do a scheduler RPC.
|
|
|
|
// We need to know CPU speed to handle app versions
|
|
|
|
//
|
|
|
|
if (!host_info.p_calculated) return false;
|
2009-02-26 22:41:48 +00:00
|
|
|
|
2010-04-26 19:31:28 +00:00
|
|
|
// check for various reasons to contact particular projects.
|
|
|
|
// If we need to contact a project,
|
|
|
|
// see if we should ask it for work as well.
|
|
|
|
//
|
|
|
|
p = next_project_sched_rpc_pending();
|
|
|
|
if (p) {
|
2013-06-15 05:40:43 +00:00
|
|
|
if (log_flags.sched_op_debug) {
|
|
|
|
msg_printf(p, MSG_INFO, "sched RPC pending: %s",
|
|
|
|
rpc_reason_string(p->sched_rpc_pending)
|
|
|
|
);
|
|
|
|
}
|
2010-04-26 19:31:28 +00:00
|
|
|
// if the user requested the RPC,
|
|
|
|
// clear backoffs to allow work requests
|
2004-07-14 20:55:26 +00:00
|
|
|
//
|
2010-04-26 19:31:28 +00:00
|
|
|
if (p->sched_rpc_pending == RPC_REASON_USER_REQ) {
|
2011-03-25 03:44:09 +00:00
|
|
|
for (int i=0; i<coprocs.n_rsc; i++) {
|
|
|
|
p->rsc_pwf[i].clear_backoff();
|
|
|
|
}
|
2005-05-16 06:49:07 +00:00
|
|
|
}
|
2012-11-13 22:36:56 +00:00
|
|
|
work_fetch.piggyback_work_request(p);
|
2010-04-26 19:31:28 +00:00
|
|
|
scheduler_op->init_op_project(p, p->sched_rpc_pending);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
p = next_project_trickle_up_pending();
|
|
|
|
if (p) {
|
2012-11-13 22:36:56 +00:00
|
|
|
work_fetch.piggyback_work_request(p);
|
2010-04-26 19:31:28 +00:00
|
|
|
scheduler_op->init_op_project(p, RPC_REASON_TRICKLE_UP);
|
|
|
|
return true;
|
|
|
|
}
|
2011-01-06 23:09:13 +00:00
|
|
|
|
2012-11-19 20:38:54 +00:00
|
|
|
// stuff from here on is checked only once/minute,
|
|
|
|
// or if work fetch was requested.
|
2010-04-26 19:31:28 +00:00
|
|
|
//
|
2008-12-31 23:07:59 +00:00
|
|
|
|
2010-04-26 19:31:28 +00:00
|
|
|
if (must_check_work_fetch) {
|
|
|
|
last_work_fetch_time = 0;
|
|
|
|
}
|
|
|
|
elapsed_time = now - last_work_fetch_time;
|
2013-03-15 03:43:29 +00:00
|
|
|
if (!clock_change && elapsed_time < WORK_FETCH_PERIOD) return false;
|
2010-04-26 19:31:28 +00:00
|
|
|
must_check_work_fetch = false;
|
|
|
|
last_work_fetch_time = now;
|
|
|
|
|
2012-11-19 20:38:54 +00:00
|
|
|
// check if we should report finished results
|
|
|
|
//
|
|
|
|
bool suspend_soon = global_prefs.net_times.suspended(now + 1800);
|
|
|
|
suspend_soon |= global_prefs.cpu_times.suspended(now + 1800);
|
|
|
|
p = find_project_with_overdue_results(suspend_soon);
|
2010-04-26 19:31:28 +00:00
|
|
|
if (p) {
|
2012-11-19 20:38:54 +00:00
|
|
|
work_fetch.piggyback_work_request(p);
|
|
|
|
scheduler_op->init_op_project(p, RPC_REASON_RESULTS_DUE);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-06-14 03:50:21 +00:00
|
|
|
// check if we should fetch work (do this last)
|
2012-11-19 20:38:54 +00:00
|
|
|
//
|
|
|
|
|
2013-06-14 03:50:21 +00:00
|
|
|
switch (suspend_reason) {
|
|
|
|
case 0:
|
|
|
|
case SUSPEND_REASON_CPU_THROTTLE:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
2014-05-08 07:51:18 +00:00
|
|
|
if (cc_config.fetch_minimal_work && had_or_requested_work) {
|
2013-06-14 03:50:21 +00:00
|
|
|
return false;
|
|
|
|
}
|
2012-11-19 20:38:54 +00:00
|
|
|
|
2013-06-14 03:50:21 +00:00
|
|
|
p = work_fetch.choose_project();
|
|
|
|
if (p) {
|
|
|
|
if (actively_uploading(p)) {
|
2014-03-10 00:09:21 +00:00
|
|
|
bool dont_request = true;
|
|
|
|
if (p->pwf.request_if_idle_and_uploading) {
|
|
|
|
if (idle_request()) {
|
|
|
|
dont_request = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (dont_request) {
|
2013-06-22 02:57:25 +00:00
|
|
|
if (log_flags.work_fetch_debug) {
|
|
|
|
msg_printf(p, MSG_INFO,
|
|
|
|
"[work_fetch] deferring work fetch; upload active"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
p->sched_rpc_pending = 0;
|
|
|
|
return false;
|
2014-03-10 00:09:21 +00:00
|
|
|
}
|
2010-12-13 22:58:15 +00:00
|
|
|
}
|
2013-06-14 03:50:21 +00:00
|
|
|
scheduler_op->init_op_project(p, RPC_REASON_NEED_WORK);
|
|
|
|
return true;
|
2002-07-15 05:34:32 +00:00
|
|
|
}
|
2010-04-26 19:31:28 +00:00
|
|
|
return false;
|
2002-07-15 05:34:32 +00:00
|
|
|
}
|
|
|
|
|
2002-09-29 00:32:11 +00:00
|
|
|
// Handle the reply from a scheduler
|
2002-07-15 23:21:20 +00:00
|
|
|
//
|
2012-02-18 23:19:06 +00:00
|
|
|
int CLIENT_STATE::handle_scheduler_reply(
|
|
|
|
PROJECT* project, char* scheduler_url
|
|
|
|
) {
|
2002-04-30 22:22:54 +00:00
|
|
|
SCHEDULER_REPLY sr;
|
|
|
|
FILE* f;
|
|
|
|
int retval;
|
|
|
|
unsigned int i;
|
2004-10-01 08:15:58 +00:00
|
|
|
bool signature_valid, update_global_prefs=false, update_project_prefs=false;
|
2011-06-12 20:58:43 +00:00
|
|
|
char buf[1024], filename[256];
|
2005-07-25 22:02:26 +00:00
|
|
|
std::string old_gui_urls = project->gui_urls;
|
2006-03-07 23:21:30 +00:00
|
|
|
PROJECT* p2;
|
2008-12-31 23:07:59 +00:00
|
|
|
vector<RESULT*>new_results;
|
2002-04-30 22:22:54 +00:00
|
|
|
|
2006-10-12 19:43:55 +00:00
|
|
|
project->last_rpc_time = now;
|
2003-07-02 02:02:18 +00:00
|
|
|
|
2013-04-26 19:19:43 +00:00
|
|
|
if (work_fetch.requested_work()) {
|
2010-06-02 17:50:47 +00:00
|
|
|
had_or_requested_work = true;
|
|
|
|
}
|
|
|
|
|
2007-03-13 19:33:27 +00:00
|
|
|
get_sched_reply_filename(*project, filename, sizeof(filename));
|
2004-09-22 22:09:54 +00:00
|
|
|
|
|
|
|
f = fopen(filename, "r");
|
2003-01-02 23:12:05 +00:00
|
|
|
if (!f) return ERR_FOPEN;
|
2003-07-03 05:01:29 +00:00
|
|
|
retval = sr.parse(f, project);
|
2002-08-29 04:19:51 +00:00
|
|
|
fclose(f);
|
2003-02-08 00:22:38 +00:00
|
|
|
if (retval) return retval;
|
2002-04-30 22:22:54 +00:00
|
|
|
|
2006-06-23 20:05:12 +00:00
|
|
|
if (log_flags.sched_ops) {
|
2013-04-26 19:19:43 +00:00
|
|
|
if (work_fetch.requested_work()) {
|
2009-06-02 05:12:06 +00:00
|
|
|
sprintf(buf, ": got %d new tasks", (int)sr.results.size());
|
|
|
|
} else {
|
|
|
|
strcpy(buf, "");
|
|
|
|
}
|
|
|
|
msg_printf(project, MSG_INFO, "Scheduler request completed%s", buf);
|
2007-10-17 23:30:05 +00:00
|
|
|
}
|
|
|
|
if (log_flags.sched_op_debug) {
|
2006-07-28 16:09:10 +00:00
|
|
|
if (sr.scheduler_version) {
|
|
|
|
msg_printf(project, MSG_INFO,
|
2010-04-29 20:32:51 +00:00
|
|
|
"[sched_op] Server version %d",
|
2006-07-28 16:09:10 +00:00
|
|
|
sr.scheduler_version
|
|
|
|
);
|
|
|
|
}
|
2006-06-23 20:05:12 +00:00
|
|
|
}
|
|
|
|
|
2006-03-07 23:21:30 +00:00
|
|
|
// check that master URL is correct
|
|
|
|
//
|
|
|
|
if (strlen(sr.master_url)) {
|
2013-05-28 06:08:02 +00:00
|
|
|
canonicalize_master_url(sr.master_url, sizeof(sr.master_url));
|
2009-02-19 21:34:48 +00:00
|
|
|
string url1 = sr.master_url;
|
|
|
|
string url2 = project->master_url;
|
|
|
|
downcase_string(url1);
|
|
|
|
downcase_string(url2);
|
|
|
|
if (url1 != url2) {
|
2007-05-10 16:23:10 +00:00
|
|
|
p2 = lookup_project(sr.master_url);
|
2006-03-07 23:21:30 +00:00
|
|
|
if (p2) {
|
2010-09-17 05:24:37 +00:00
|
|
|
msg_printf(project, MSG_USER_ALERT,
|
|
|
|
"You are attached to this project twice. Please remove projects named %s, then add %s",
|
|
|
|
project->project_name,
|
|
|
|
sr.master_url
|
2006-03-07 23:21:30 +00:00
|
|
|
);
|
|
|
|
} else {
|
2015-11-09 06:42:58 +00:00
|
|
|
msg_printf(project, MSG_USER_ALERT,
|
|
|
|
_("This project is using an old URL. When convenient, remove the project, then add %s"),
|
2010-09-18 04:04:08 +00:00
|
|
|
sr.master_url
|
2006-03-07 23:21:30 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// make sure we don't already have a project of same name
|
|
|
|
//
|
2007-05-14 19:18:45 +00:00
|
|
|
bool dup_name = false;
|
|
|
|
for (i=0; i<projects.size(); i++) {
|
|
|
|
p2 = projects[i];
|
|
|
|
if (project == p2) continue;
|
|
|
|
if (!strcmp(p2->project_name, project->project_name)) {
|
|
|
|
dup_name = true;
|
|
|
|
break;
|
2006-03-07 23:21:30 +00:00
|
|
|
}
|
|
|
|
}
|
2007-05-14 19:18:45 +00:00
|
|
|
if (dup_name) {
|
2010-06-08 23:37:44 +00:00
|
|
|
msg_printf(project, MSG_INFO,
|
2007-05-14 19:18:45 +00:00
|
|
|
"Already attached to a project named %s (possibly with wrong URL)",
|
|
|
|
project->project_name
|
|
|
|
);
|
2010-06-08 23:37:44 +00:00
|
|
|
msg_printf(project, MSG_INFO,
|
2007-05-14 19:18:45 +00:00
|
|
|
"Consider detaching this project, then trying again"
|
|
|
|
);
|
|
|
|
}
|
2006-03-07 23:21:30 +00:00
|
|
|
|
2007-05-14 19:18:45 +00:00
|
|
|
// show messages from server
|
2005-09-28 05:47:14 +00:00
|
|
|
//
|
2013-08-13 21:56:37 +00:00
|
|
|
bool got_notice = false;
|
2005-01-31 22:19:03 +00:00
|
|
|
for (i=0; i<sr.messages.size(); i++) {
|
|
|
|
USER_MESSAGE& um = sr.messages[i];
|
2013-08-13 21:56:37 +00:00
|
|
|
int prio = MSG_INFO;
|
|
|
|
if (!strcmp(um.priority.c_str(), "notice")) {
|
|
|
|
prio = MSG_SCHEDULER_ALERT;
|
|
|
|
got_notice = true;
|
|
|
|
}
|
2013-02-08 19:04:19 +00:00
|
|
|
msg_printf(project, prio, "%s", um.message.c_str());
|
2002-05-17 22:33:57 +00:00
|
|
|
}
|
2014-02-10 19:22:35 +00:00
|
|
|
|
2013-08-13 21:56:37 +00:00
|
|
|
// if we requested work and didn't get notices,
|
|
|
|
// clear scheduler notices from this project
|
|
|
|
//
|
|
|
|
if (work_fetch.requested_work() && !got_notice) {
|
2014-02-11 01:52:53 +00:00
|
|
|
notices.remove_notices(project, REMOVE_SCHEDULER_MSG);
|
2013-08-13 21:56:37 +00:00
|
|
|
}
|
2002-05-17 22:33:57 +00:00
|
|
|
|
2006-11-06 23:11:07 +00:00
|
|
|
if (log_flags.sched_op_debug && sr.request_delay) {
|
|
|
|
msg_printf(project, MSG_INFO,
|
2009-02-10 19:33:46 +00:00
|
|
|
"Project requested delay of %.0f seconds", sr.request_delay
|
2006-11-06 23:11:07 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2004-06-22 22:56:50 +00:00
|
|
|
// if project is down, return error (so that we back off)
|
|
|
|
// and don't do anything else
|
|
|
|
//
|
|
|
|
if (sr.project_is_down) {
|
2005-08-24 02:04:46 +00:00
|
|
|
if (sr.request_delay) {
|
2007-05-10 16:23:10 +00:00
|
|
|
double x = now + sr.request_delay;
|
2014-07-08 19:35:31 +00:00
|
|
|
project->set_min_rpc_time(x, "project requested delay");
|
2005-08-24 02:04:46 +00:00
|
|
|
}
|
2004-06-22 22:56:50 +00:00
|
|
|
return ERR_PROJECT_DOWN;
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
2002-09-29 00:32:11 +00:00
|
|
|
|
|
|
|
// if the scheduler reply includes global preferences,
|
|
|
|
// insert extra elements, write to disk, and parse
|
2002-06-21 06:52:47 +00:00
|
|
|
//
|
2002-09-29 00:32:11 +00:00
|
|
|
if (sr.global_prefs_xml) {
|
2011-01-06 23:09:13 +00:00
|
|
|
// skip this if we have host-specific prefs
|
|
|
|
// and we're talking to an old scheduler
|
|
|
|
//
|
|
|
|
if (!global_prefs.host_specific || sr.scheduler_version >= 507) {
|
|
|
|
retval = save_global_prefs(
|
|
|
|
sr.global_prefs_xml, project->master_url, scheduler_url
|
|
|
|
);
|
|
|
|
if (retval) {
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
update_global_prefs = true;
|
|
|
|
} else {
|
|
|
|
if (log_flags.sched_op_debug) {
|
|
|
|
msg_printf(project, MSG_INFO,
|
|
|
|
"ignoring prefs from old server; we have host-specific prefs"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2005-04-13 23:28:47 +00:00
|
|
|
}
|
2003-08-13 20:00:20 +00:00
|
|
|
|
2006-10-26 21:22:38 +00:00
|
|
|
// see if we have a new venue from this project
|
|
|
|
// (this must go AFTER the above, since otherwise
|
|
|
|
// global_prefs_source_project() is meaningless)
|
|
|
|
//
|
|
|
|
if (strcmp(project->host_venue, sr.host_venue)) {
|
|
|
|
safe_strcpy(project->host_venue, sr.host_venue);
|
2007-10-04 17:30:28 +00:00
|
|
|
msg_printf(project, MSG_INFO, "New computer location: %s", sr.host_venue);
|
2006-10-26 21:22:38 +00:00
|
|
|
update_project_prefs = true;
|
2014-03-11 20:12:15 +00:00
|
|
|
#ifdef USE_NET_PREFS
|
2006-10-26 21:22:38 +00:00
|
|
|
if (project == global_prefs_source_project()) {
|
2013-06-04 03:24:48 +00:00
|
|
|
safe_strcpy(main_host_venue, sr.host_venue);
|
2006-10-26 21:22:38 +00:00
|
|
|
update_global_prefs = true;
|
|
|
|
}
|
2014-03-11 20:12:15 +00:00
|
|
|
#endif
|
2006-10-26 21:22:38 +00:00
|
|
|
}
|
|
|
|
|
2005-04-13 23:28:47 +00:00
|
|
|
if (update_global_prefs) {
|
2006-06-02 20:32:20 +00:00
|
|
|
read_global_prefs();
|
2002-09-29 00:32:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// deal with project preferences (should always be there)
|
2003-11-28 19:19:11 +00:00
|
|
|
// If they've changed, write to account file,
|
|
|
|
// then parse to get our venue, and pass to running apps
|
2002-09-29 00:32:11 +00:00
|
|
|
//
|
|
|
|
if (sr.project_prefs_xml) {
|
2003-11-28 19:19:11 +00:00
|
|
|
if (strcmp(project->project_prefs.c_str(), sr.project_prefs_xml)) {
|
|
|
|
project->project_prefs = string(sr.project_prefs_xml);
|
2004-10-01 08:15:58 +00:00
|
|
|
update_project_prefs = true;
|
2003-05-20 00:03:39 +00:00
|
|
|
}
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
2005-07-25 22:02:26 +00:00
|
|
|
|
2005-07-26 11:55:28 +00:00
|
|
|
// the account file has GUI URLs and project prefs.
|
|
|
|
// rewrite if either of these has changed
|
|
|
|
//
|
2005-07-25 22:02:26 +00:00
|
|
|
if (project->gui_urls != old_gui_urls || update_project_prefs) {
|
2004-10-01 18:48:34 +00:00
|
|
|
retval = project->write_account_file();
|
|
|
|
if (retval) {
|
2007-01-25 23:39:06 +00:00
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
2006-01-17 22:48:09 +00:00
|
|
|
"Can't write account file: %s", boincerror(retval)
|
|
|
|
);
|
2004-10-01 18:48:34 +00:00
|
|
|
return retval;
|
|
|
|
}
|
2005-07-25 22:02:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (update_project_prefs) {
|
2004-10-01 08:15:58 +00:00
|
|
|
project->parse_account_file();
|
|
|
|
project->parse_preferences_for_user_files();
|
|
|
|
active_tasks.request_reread_prefs(project);
|
|
|
|
}
|
2002-04-30 22:22:54 +00:00
|
|
|
|
2014-02-11 02:30:42 +00:00
|
|
|
// show notice if we can't possibly get work from this project.
|
2014-05-02 04:03:49 +00:00
|
|
|
// This must come after parsing project prefs
|
2014-02-11 02:30:42 +00:00
|
|
|
//
|
|
|
|
project->show_no_work_notice();
|
|
|
|
|
2002-07-07 20:39:24 +00:00
|
|
|
// if the scheduler reply includes a code-signing key,
|
|
|
|
// accept it if we don't already have one from the project.
|
|
|
|
// Otherwise verify its signature, using the key we already have.
|
|
|
|
//
|
|
|
|
|
|
|
|
if (sr.code_sign_key) {
|
2003-02-11 00:52:44 +00:00
|
|
|
if (!strlen(project->code_sign_key)) {
|
2003-04-03 18:35:40 +00:00
|
|
|
safe_strcpy(project->code_sign_key, sr.code_sign_key);
|
2002-07-07 20:39:24 +00:00
|
|
|
} else {
|
2002-08-25 07:54:33 +00:00
|
|
|
if (sr.code_sign_key_signature) {
|
2012-02-06 06:06:44 +00:00
|
|
|
retval = check_string_signature2(
|
2002-08-25 07:54:33 +00:00
|
|
|
sr.code_sign_key, sr.code_sign_key_signature,
|
|
|
|
project->code_sign_key, signature_valid
|
|
|
|
);
|
|
|
|
if (!retval && signature_valid) {
|
2003-04-03 18:35:40 +00:00
|
|
|
safe_strcpy(project->code_sign_key, sr.code_sign_key);
|
2002-08-25 07:54:33 +00:00
|
|
|
} else {
|
2007-01-25 23:39:06 +00:00
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
2006-01-17 22:48:09 +00:00
|
|
|
"New code signing key doesn't validate"
|
|
|
|
);
|
2002-08-25 07:54:33 +00:00
|
|
|
}
|
|
|
|
} else {
|
2007-01-25 23:39:06 +00:00
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
2006-01-17 22:48:09 +00:00
|
|
|
"Missing code sign key signature"
|
|
|
|
);
|
2002-08-25 07:54:33 +00:00
|
|
|
}
|
2002-07-07 20:39:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// copy new entities to client state
|
|
|
|
//
|
2002-04-30 22:22:54 +00:00
|
|
|
for (i=0; i<sr.apps.size(); i++) {
|
|
|
|
APP* app = lookup_app(project, sr.apps[i].name);
|
2007-01-23 17:24:43 +00:00
|
|
|
if (app) {
|
2014-05-04 07:02:32 +00:00
|
|
|
// update app attributes; they may have changed on server
|
|
|
|
//
|
2013-06-04 03:24:48 +00:00
|
|
|
safe_strcpy(app->user_friendly_name, sr.apps[i].user_friendly_name);
|
2014-05-04 07:02:32 +00:00
|
|
|
app->non_cpu_intensive = sr.apps[i].non_cpu_intensive;
|
|
|
|
app->fraction_done_exact = sr.apps[i].fraction_done_exact;
|
2007-01-23 17:24:43 +00:00
|
|
|
} else {
|
2002-04-30 22:22:54 +00:00
|
|
|
app = new APP;
|
|
|
|
*app = sr.apps[i];
|
|
|
|
retval = link_app(project, app);
|
2004-08-12 12:44:55 +00:00
|
|
|
if (retval) {
|
2007-01-25 23:39:06 +00:00
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
2006-01-17 22:48:09 +00:00
|
|
|
"Can't handle application %s in scheduler reply", app->name
|
2004-08-12 12:44:55 +00:00
|
|
|
);
|
|
|
|
delete app;
|
|
|
|
} else {
|
|
|
|
apps.push_back(app);
|
|
|
|
}
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
|
|
|
}
|
2004-07-06 17:37:58 +00:00
|
|
|
FILE_INFO* fip;
|
2002-04-30 22:22:54 +00:00
|
|
|
for (i=0; i<sr.file_infos.size(); i++) {
|
2004-07-06 17:37:58 +00:00
|
|
|
fip = lookup_file_info(project, sr.file_infos[i].name);
|
2004-08-12 10:13:01 +00:00
|
|
|
if (fip) {
|
2004-07-06 18:31:56 +00:00
|
|
|
fip->merge_info(sr.file_infos[i]);
|
|
|
|
} else {
|
2004-07-06 17:37:58 +00:00
|
|
|
fip = new FILE_INFO;
|
2002-04-30 22:22:54 +00:00
|
|
|
*fip = sr.file_infos[i];
|
2014-09-23 19:39:09 +00:00
|
|
|
if (fip->sticky_lifetime) {
|
|
|
|
fip->sticky_expire_time = now + fip->sticky_lifetime;
|
|
|
|
}
|
2004-09-22 21:08:26 +00:00
|
|
|
retval = link_file_info(project, fip);
|
2004-08-12 12:44:55 +00:00
|
|
|
if (retval) {
|
2007-01-25 23:39:06 +00:00
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
2006-01-17 22:48:09 +00:00
|
|
|
"Can't handle file %s in scheduler reply", fip->name
|
2004-08-12 12:44:55 +00:00
|
|
|
);
|
|
|
|
delete fip;
|
|
|
|
} else {
|
|
|
|
file_infos.push_back(fip);
|
|
|
|
}
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
2004-08-05 21:42:26 +00:00
|
|
|
}
|
|
|
|
for (i=0; i<sr.file_deletes.size(); i++) {
|
2005-03-09 23:28:37 +00:00
|
|
|
fip = lookup_file_info(project, sr.file_deletes[i].c_str());
|
2004-08-12 10:13:01 +00:00
|
|
|
if (fip) {
|
2011-06-06 03:40:42 +00:00
|
|
|
if (log_flags.file_xfer_debug) {
|
|
|
|
msg_printf(project, MSG_INFO,
|
|
|
|
"[file_xfer_debug] Got server request to delete file %s",
|
|
|
|
fip->name
|
|
|
|
);
|
|
|
|
}
|
2011-04-24 21:33:51 +00:00
|
|
|
fip->sticky = false;
|
2004-08-05 21:42:26 +00:00
|
|
|
}
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
|
|
|
for (i=0; i<sr.app_versions.size(); i++) {
|
2009-03-05 17:45:36 +00:00
|
|
|
if (project->anonymous_platform) {
|
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
|
|
|
"App version returned from anonymous platform project; ignoring"
|
|
|
|
);
|
|
|
|
continue;
|
|
|
|
}
|
2007-05-03 17:14:30 +00:00
|
|
|
APP_VERSION& avpp = sr.app_versions[i];
|
2007-05-11 03:27:25 +00:00
|
|
|
if (strlen(avpp.platform) == 0) {
|
2013-06-04 03:24:48 +00:00
|
|
|
safe_strcpy(avpp.platform, get_primary_platform());
|
2007-05-11 03:27:25 +00:00
|
|
|
} else {
|
|
|
|
if (!is_supported_platform(avpp.platform)) {
|
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
|
|
|
"App version has unsupported platform %s", avpp.platform
|
|
|
|
);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2011-03-25 03:44:09 +00:00
|
|
|
if (avpp.missing_coproc) {
|
2009-03-05 17:45:36 +00:00
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
2009-10-02 23:39:38 +00:00
|
|
|
"App version uses non-existent %s GPU",
|
2011-03-25 03:44:09 +00:00
|
|
|
avpp.missing_coproc_name
|
2009-03-05 17:45:36 +00:00
|
|
|
);
|
2009-02-06 00:22:21 +00:00
|
|
|
}
|
2007-05-03 17:14:30 +00:00
|
|
|
APP* app = lookup_app(project, avpp.app_name);
|
2009-10-02 23:39:38 +00:00
|
|
|
if (!app) {
|
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
|
|
|
"Missing app %s", avpp.app_name
|
|
|
|
);
|
|
|
|
continue;
|
|
|
|
}
|
2008-03-27 18:25:29 +00:00
|
|
|
APP_VERSION* avp = lookup_app_version(
|
|
|
|
app, avpp.platform, avpp.version_num, avpp.plan_class
|
|
|
|
);
|
2004-09-02 16:23:21 +00:00
|
|
|
if (avp) {
|
2014-05-04 07:02:32 +00:00
|
|
|
// update app version attributes in case they changed on server
|
2008-03-28 18:00:27 +00:00
|
|
|
//
|
|
|
|
avp->avg_ncpus = avpp.avg_ncpus;
|
|
|
|
avp->max_ncpus = avpp.max_ncpus;
|
|
|
|
avp->flops = avpp.flops;
|
2013-06-04 03:24:48 +00:00
|
|
|
safe_strcpy(avp->cmdline, avpp.cmdline);
|
2011-03-25 03:44:09 +00:00
|
|
|
avp->gpu_usage = avpp.gpu_usage;
|
2008-05-22 19:09:52 +00:00
|
|
|
strlcpy(avp->api_version, avpp.api_version, sizeof(avp->api_version));
|
2011-12-28 14:52:03 +00:00
|
|
|
avp->dont_throttle = avpp.dont_throttle;
|
|
|
|
avp->needs_network = avpp.needs_network;
|
2008-03-28 18:00:27 +00:00
|
|
|
|
2004-09-02 16:23:21 +00:00
|
|
|
// if we had download failures, clear them
|
|
|
|
//
|
|
|
|
avp->clear_errors();
|
2012-12-10 23:41:57 +00:00
|
|
|
|
2004-09-02 16:23:21 +00:00
|
|
|
continue;
|
|
|
|
}
|
2004-08-16 14:30:41 +00:00
|
|
|
avp = new APP_VERSION;
|
2007-05-11 03:27:25 +00:00
|
|
|
*avp = avpp;
|
2004-08-16 14:30:41 +00:00
|
|
|
retval = link_app_version(project, avp);
|
|
|
|
if (retval) {
|
|
|
|
delete avp;
|
|
|
|
continue;
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
2004-08-16 14:30:41 +00:00
|
|
|
app_versions.push_back(avp);
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
|
|
|
for (i=0; i<sr.workunits.size(); i++) {
|
2004-08-16 14:30:41 +00:00
|
|
|
if (lookup_workunit(project, sr.workunits[i].name)) continue;
|
|
|
|
WORKUNIT* wup = new WORKUNIT;
|
|
|
|
*wup = sr.workunits[i];
|
2006-08-10 17:36:28 +00:00
|
|
|
wup->project = project;
|
2004-08-16 14:30:41 +00:00
|
|
|
retval = link_workunit(project, wup);
|
|
|
|
if (retval) {
|
2007-01-25 23:39:06 +00:00
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
2006-01-17 22:48:09 +00:00
|
|
|
"Can't handle task %s in scheduler reply", wup->name
|
2004-08-16 14:30:41 +00:00
|
|
|
);
|
|
|
|
delete wup;
|
|
|
|
continue;
|
|
|
|
}
|
2005-01-13 18:57:27 +00:00
|
|
|
wup->clear_errors();
|
2004-08-16 14:30:41 +00:00
|
|
|
workunits.push_back(wup);
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
2012-04-25 04:10:29 +00:00
|
|
|
double est_rsc_runtime[MAX_RSC];
|
2011-03-25 03:44:09 +00:00
|
|
|
for (int j=0; j<coprocs.n_rsc; j++) {
|
2012-04-25 04:10:29 +00:00
|
|
|
est_rsc_runtime[j] = 0;
|
2011-03-25 03:44:09 +00:00
|
|
|
}
|
2002-04-30 22:22:54 +00:00
|
|
|
for (i=0; i<sr.results.size(); i++) {
|
2014-11-07 08:52:27 +00:00
|
|
|
RESULT* rp2 = lookup_result(project, sr.results[i].name);
|
|
|
|
if (rp2) {
|
|
|
|
// see if project wants to change the job's deadline
|
|
|
|
//
|
|
|
|
if (sr.results[i].report_deadline != rp2->report_deadline) {
|
|
|
|
rp2->report_deadline = sr.results[i].report_deadline;
|
|
|
|
} else {
|
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
|
|
|
"Already have task %s\n", sr.results[i].name
|
|
|
|
);
|
|
|
|
}
|
2004-08-16 14:30:41 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
RESULT* rp = new RESULT;
|
|
|
|
*rp = sr.results[i];
|
|
|
|
retval = link_result(project, rp);
|
|
|
|
if (retval) {
|
2007-01-25 23:39:06 +00:00
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
2006-01-17 22:48:09 +00:00
|
|
|
"Can't handle task %s in scheduler reply", rp->name
|
2004-08-16 14:30:41 +00:00
|
|
|
);
|
|
|
|
delete rp;
|
|
|
|
continue;
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
2007-05-03 17:14:30 +00:00
|
|
|
if (strlen(rp->platform) == 0) {
|
2013-06-04 03:24:48 +00:00
|
|
|
safe_strcpy(rp->platform, get_primary_platform());
|
2007-05-03 17:14:30 +00:00
|
|
|
rp->version_num = latest_version(rp->wup->app, rp->platform);
|
|
|
|
}
|
2008-03-27 18:25:29 +00:00
|
|
|
rp->avp = lookup_app_version(
|
|
|
|
rp->wup->app, rp->platform, rp->version_num, rp->plan_class
|
|
|
|
);
|
2007-05-03 17:14:30 +00:00
|
|
|
if (!rp->avp) {
|
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
2011-10-21 06:58:35 +00:00
|
|
|
"No app version found for app %s platform %s ver %d class %s; discarding %s",
|
2010-05-18 19:22:34 +00:00
|
|
|
rp->wup->app->name, rp->platform, rp->version_num, rp->plan_class, rp->name
|
2007-05-03 17:14:30 +00:00
|
|
|
);
|
|
|
|
delete rp;
|
|
|
|
continue;
|
|
|
|
}
|
2011-03-25 03:44:09 +00:00
|
|
|
if (rp->avp->missing_coproc) {
|
2009-10-02 23:39:38 +00:00
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
|
|
|
"Missing coprocessor for task %s; aborting", rp->name
|
|
|
|
);
|
2012-04-26 05:28:45 +00:00
|
|
|
rp->abort_inactive(EXIT_MISSING_COPROC);
|
2009-10-02 23:39:38 +00:00
|
|
|
} else {
|
|
|
|
rp->set_state(RESULT_NEW, "handle_scheduler_reply");
|
2011-03-25 03:44:09 +00:00
|
|
|
int rt = rp->avp->gpu_usage.rsc_type;
|
|
|
|
if (rt > 0) {
|
2012-04-25 04:10:29 +00:00
|
|
|
est_rsc_runtime[rt] += rp->estimated_runtime();
|
2009-12-07 00:42:03 +00:00
|
|
|
gpus_usable = true;
|
2009-10-14 19:11:11 +00:00
|
|
|
// trigger a check of whether GPU is actually usable
|
2009-10-02 23:39:38 +00:00
|
|
|
} else {
|
2012-04-25 04:10:29 +00:00
|
|
|
est_rsc_runtime[0] += rp->estimated_runtime();
|
2009-10-02 23:39:38 +00:00
|
|
|
}
|
|
|
|
}
|
2007-05-03 17:14:30 +00:00
|
|
|
rp->wup->version_num = rp->version_num;
|
2009-06-30 20:22:54 +00:00
|
|
|
rp->received_time = now;
|
2008-12-31 23:07:59 +00:00
|
|
|
new_results.push_back(rp);
|
2009-10-02 23:39:38 +00:00
|
|
|
results.push_back(rp);
|
2008-10-27 23:21:33 +00:00
|
|
|
}
|
2009-10-03 03:07:03 +00:00
|
|
|
sort_results();
|
|
|
|
|
2008-10-27 23:21:33 +00:00
|
|
|
if (log_flags.sched_op_debug) {
|
|
|
|
if (sr.results.size()) {
|
2011-06-12 20:58:43 +00:00
|
|
|
for (int j=0; j<coprocs.n_rsc; j++) {
|
2009-08-17 16:50:40 +00:00
|
|
|
msg_printf(project, MSG_INFO,
|
2011-03-25 03:44:09 +00:00
|
|
|
"[sched_op] estimated total %s task duration: %.0f seconds",
|
2014-02-25 04:54:42 +00:00
|
|
|
rsc_name_long(j),
|
2012-04-25 04:10:29 +00:00
|
|
|
est_rsc_runtime[j]/time_stats.availability_frac(j)
|
2009-08-17 16:50:40 +00:00
|
|
|
);
|
|
|
|
}
|
2008-10-27 23:21:33 +00:00
|
|
|
}
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// update records for ack'ed results
|
|
|
|
//
|
|
|
|
for (i=0; i<sr.result_acks.size(); i++) {
|
2006-06-22 19:40:30 +00:00
|
|
|
if (log_flags.sched_op_debug) {
|
2009-02-23 04:54:04 +00:00
|
|
|
msg_printf(project, MSG_INFO,
|
2010-04-29 20:32:51 +00:00
|
|
|
"[sched_op] handle_scheduler_reply(): got ack for task %s\n",
|
2006-06-22 19:40:30 +00:00
|
|
|
sr.result_acks[i].name
|
|
|
|
);
|
|
|
|
}
|
2006-12-07 23:39:23 +00:00
|
|
|
RESULT* rp = lookup_result(project, sr.result_acks[i].name);
|
2002-04-30 22:22:54 +00:00
|
|
|
if (rp) {
|
2003-09-23 23:19:41 +00:00
|
|
|
rp->got_server_ack = true;
|
2002-04-30 22:22:54 +00:00
|
|
|
} else {
|
2007-01-25 23:39:06 +00:00
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
2006-01-17 22:48:09 +00:00
|
|
|
"Got ack for task %s, but can't find it", sr.result_acks[i].name
|
2003-06-17 18:59:36 +00:00
|
|
|
);
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
|
|
|
}
|
2004-01-08 00:27:59 +00:00
|
|
|
|
2006-05-25 20:10:08 +00:00
|
|
|
// handle result abort requests
|
|
|
|
//
|
|
|
|
for (i=0; i<sr.result_abort.size(); i++) {
|
|
|
|
RESULT* rp = lookup_result(project, sr.result_abort[i].name);
|
|
|
|
if (rp) {
|
|
|
|
ACTIVE_TASK* atp = lookup_active_task_by_result(rp);
|
|
|
|
if (atp) {
|
2012-04-26 05:28:45 +00:00
|
|
|
atp->abort_task(EXIT_ABORTED_BY_PROJECT,
|
2007-06-20 16:27:27 +00:00
|
|
|
"aborted by project - no longer usable"
|
|
|
|
);
|
2006-05-25 20:10:08 +00:00
|
|
|
} else {
|
2012-04-26 05:28:45 +00:00
|
|
|
rp->abort_inactive(EXIT_ABORTED_BY_PROJECT);
|
2006-05-25 20:10:08 +00:00
|
|
|
}
|
2007-03-22 02:38:45 +00:00
|
|
|
} else {
|
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
|
|
|
"Server requested abort of unknown task %s",
|
|
|
|
sr.result_abort[i].name
|
|
|
|
);
|
2006-05-25 20:10:08 +00:00
|
|
|
}
|
|
|
|
}
|
2007-03-22 02:38:45 +00:00
|
|
|
for (i=0; i<sr.result_abort_if_not_started.size(); i++) {
|
|
|
|
RESULT* rp = lookup_result(project, sr.result_abort_if_not_started[i].name);
|
2009-02-18 19:47:02 +00:00
|
|
|
if (!rp) {
|
2007-03-22 02:38:45 +00:00
|
|
|
msg_printf(project, MSG_INTERNAL_ERROR,
|
|
|
|
"Server requested conditional abort of unknown task %s",
|
|
|
|
sr.result_abort_if_not_started[i].name
|
|
|
|
);
|
2009-02-18 19:47:02 +00:00
|
|
|
continue;
|
|
|
|
}
|
2011-11-04 09:27:01 +00:00
|
|
|
if (rp->not_started) {
|
2012-04-26 05:28:45 +00:00
|
|
|
rp->abort_inactive(EXIT_ABORTED_BY_PROJECT);
|
2006-05-25 20:10:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-01-08 00:27:59 +00:00
|
|
|
// remove acked trickle files
|
|
|
|
//
|
2004-07-06 04:01:15 +00:00
|
|
|
if (sr.message_ack) {
|
2004-01-08 00:27:59 +00:00
|
|
|
remove_trickle_files(project);
|
|
|
|
}
|
2010-05-13 20:18:27 +00:00
|
|
|
if (sr.send_full_workload) {
|
|
|
|
project->send_full_workload = true;
|
|
|
|
}
|
2012-03-23 16:25:19 +00:00
|
|
|
project->dont_use_dcf = sr.dont_use_dcf;
|
2007-04-13 22:55:18 +00:00
|
|
|
project->send_time_stats_log = sr.send_time_stats_log;
|
|
|
|
project->send_job_log = sr.send_job_log;
|
2005-08-15 05:08:42 +00:00
|
|
|
project->trickle_up_pending = false;
|
2005-01-24 22:13:32 +00:00
|
|
|
|
2006-01-24 21:38:03 +00:00
|
|
|
// The project returns a hostid only if it has created a new host record.
|
2006-03-09 23:40:10 +00:00
|
|
|
// In that case reset RPC seqno
|
2006-01-24 21:38:03 +00:00
|
|
|
//
|
|
|
|
if (sr.hostid) {
|
2006-03-09 23:40:10 +00:00
|
|
|
if (project->hostid) {
|
|
|
|
// if we already have a host ID for this project,
|
|
|
|
// we must have sent it a stale seqno,
|
|
|
|
// which usually means our state file was copied from another host.
|
|
|
|
// So generate a new host CPID.
|
|
|
|
//
|
|
|
|
generate_new_host_cpid();
|
|
|
|
msg_printf(project, MSG_INFO,
|
2008-06-01 03:43:47 +00:00
|
|
|
"Generated new computer cross-project ID: %s",
|
|
|
|
host_info.host_cpid
|
2006-03-09 23:40:10 +00:00
|
|
|
);
|
|
|
|
}
|
2006-03-03 20:22:25 +00:00
|
|
|
//msg_printf(project, MSG_INFO, "Changing host ID from %d to %d", project->hostid, sr.hostid);
|
2006-01-24 21:38:03 +00:00
|
|
|
project->hostid = sr.hostid;
|
|
|
|
project->rpc_seqno = 0;
|
|
|
|
}
|
|
|
|
|
2011-02-24 22:18:14 +00:00
|
|
|
#ifdef ENABLE_AUTO_UPDATE
|
2006-12-11 23:42:54 +00:00
|
|
|
if (sr.auto_update.present) {
|
2006-12-14 00:42:43 +00:00
|
|
|
if (!sr.auto_update.validate_and_link(project)) {
|
|
|
|
auto_update = sr.auto_update;
|
|
|
|
}
|
2006-12-11 23:42:54 +00:00
|
|
|
}
|
2011-02-24 22:18:14 +00:00
|
|
|
#endif
|
2006-12-11 23:42:54 +00:00
|
|
|
|
2012-04-25 05:51:26 +00:00
|
|
|
project->project_files = sr.project_files;
|
|
|
|
project->link_project_files();
|
|
|
|
project->create_project_file_symlinks();
|
2006-07-25 21:07:14 +00:00
|
|
|
|
2006-06-22 19:40:30 +00:00
|
|
|
if (log_flags.state_debug) {
|
2009-02-23 04:54:04 +00:00
|
|
|
msg_printf(project, MSG_INFO,
|
2010-04-29 20:32:51 +00:00
|
|
|
"[state] handle_scheduler_reply(): State after handle_scheduler_reply():"
|
2006-06-22 19:40:30 +00:00
|
|
|
);
|
|
|
|
print_summary();
|
|
|
|
}
|
2007-07-06 21:58:32 +00:00
|
|
|
|
2008-10-14 21:16:04 +00:00
|
|
|
// the following must precede the backoff and request_delay checks,
|
|
|
|
// since it overrides them
|
|
|
|
//
|
|
|
|
if (sr.next_rpc_delay) {
|
|
|
|
project->next_rpc_time = now + sr.next_rpc_delay;
|
|
|
|
} else {
|
|
|
|
project->next_rpc_time = 0;
|
|
|
|
}
|
|
|
|
|
2009-09-16 17:34:19 +00:00
|
|
|
work_fetch.handle_reply(project, &sr, new_results);
|
2008-12-31 23:07:59 +00:00
|
|
|
|
|
|
|
project->nrpc_failures = 0;
|
|
|
|
project->min_rpc_time = 0;
|
2007-07-06 21:58:32 +00:00
|
|
|
|
2007-10-15 22:08:24 +00:00
|
|
|
if (sr.request_delay) {
|
|
|
|
double x = now + sr.request_delay;
|
|
|
|
project->set_min_rpc_time(x, "requested by project");
|
2007-07-06 21:58:32 +00:00
|
|
|
}
|
|
|
|
|
2009-12-23 00:58:27 +00:00
|
|
|
if (sr.got_rss_feeds) {
|
2009-12-22 03:56:24 +00:00
|
|
|
handle_sr_feeds(sr.sr_feeds, project);
|
2009-12-21 17:49:28 +00:00
|
|
|
}
|
|
|
|
|
2011-09-20 18:49:38 +00:00
|
|
|
update_trickle_up_urls(project, sr.trickle_up_urls);
|
|
|
|
|
2009-03-23 01:33:17 +00:00
|
|
|
// garbage collect in case the project sent us some irrelevant FILE_INFOs;
|
|
|
|
// avoid starting transfers for them
|
|
|
|
//
|
|
|
|
gstate.garbage_collect_always();
|
|
|
|
|
2013-05-14 15:46:17 +00:00
|
|
|
// if the user provided app_config.xml for this project,
|
|
|
|
// apply it to any app versions we just got
|
|
|
|
//
|
2013-05-19 17:02:00 +00:00
|
|
|
project->app_configs.config_app_versions(project, false);
|
2013-05-14 15:46:17 +00:00
|
|
|
|
2013-05-17 17:25:03 +00:00
|
|
|
// make sure we don't set no_rsc_apps[] for all processor types
|
|
|
|
//
|
|
|
|
if (!project->anonymous_platform) {
|
|
|
|
project->check_no_rsc_apps();
|
|
|
|
}
|
|
|
|
|
2003-01-02 23:12:05 +00:00
|
|
|
return 0;
|
2002-04-30 22:22:54 +00:00
|
|
|
}
|
2004-12-08 00:40:19 +00:00
|
|
|
|
2008-12-30 19:01:25 +00:00
|
|
|
#endif // SIM
|
|
|
|
|
|
|
|
void CLIENT_STATE::check_project_timeout() {
|
2011-01-06 23:09:13 +00:00
|
|
|
unsigned int i;
|
|
|
|
for (i=0; i<projects.size(); i++) {
|
|
|
|
PROJECT* p = projects[i];
|
|
|
|
if (p->possibly_backed_off && now > p->min_rpc_time) {
|
|
|
|
p->possibly_backed_off = false;
|
2009-04-01 23:22:17 +00:00
|
|
|
char buf[256];
|
|
|
|
sprintf(buf, "Backoff ended for %s", p->get_project_name());
|
2011-01-06 23:09:13 +00:00
|
|
|
request_work_fetch(buf);
|
|
|
|
}
|
|
|
|
}
|
2008-12-30 19:01:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// find a project that needs to have its master file fetched
|
|
|
|
//
|
|
|
|
PROJECT* CLIENT_STATE::next_project_master_pending() {
|
|
|
|
unsigned int i;
|
|
|
|
PROJECT* p;
|
|
|
|
|
|
|
|
for (i=0; i<projects.size(); i++) {
|
|
|
|
p = projects[i];
|
|
|
|
if (p->waiting_until_min_rpc_time()) continue;
|
|
|
|
if (p->suspended_via_gui) continue;
|
|
|
|
if (p->master_url_fetch_pending) {
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-02-09 22:00:31 +00:00
|
|
|
// find a project for which a scheduler RPC has been requested
|
|
|
|
// - by user
|
|
|
|
// - by an account manager
|
|
|
|
// - by the project
|
|
|
|
// - because the project was just attached (for verification)
|
2008-12-30 19:01:25 +00:00
|
|
|
//
|
|
|
|
PROJECT* CLIENT_STATE::next_project_sched_rpc_pending() {
|
|
|
|
unsigned int i;
|
|
|
|
PROJECT* p;
|
|
|
|
|
|
|
|
for (i=0; i<projects.size(); i++) {
|
|
|
|
p = projects[i];
|
2009-03-04 22:55:57 +00:00
|
|
|
bool honor_backoff = true;
|
|
|
|
bool honor_suspend = true;
|
2008-12-30 19:01:25 +00:00
|
|
|
|
2011-10-07 21:20:42 +00:00
|
|
|
// is a scheduler-requested RPC due?
|
|
|
|
//
|
2009-03-04 22:55:57 +00:00
|
|
|
if (!p->sched_rpc_pending && p->next_rpc_time && p->next_rpc_time<now) {
|
2011-10-07 21:20:42 +00:00
|
|
|
// don't do it if project is set to no new work
|
|
|
|
// and has no jobs currently
|
|
|
|
//
|
|
|
|
if (!p->dont_request_more_work || p->has_results()) {
|
|
|
|
p->sched_rpc_pending = RPC_REASON_PROJECT_REQ;
|
|
|
|
}
|
2008-12-30 19:01:25 +00:00
|
|
|
}
|
2009-03-04 22:55:57 +00:00
|
|
|
|
|
|
|
switch (p->sched_rpc_pending) {
|
|
|
|
case RPC_REASON_USER_REQ:
|
|
|
|
honor_backoff = false;
|
|
|
|
honor_suspend = false;
|
|
|
|
break;
|
|
|
|
case RPC_REASON_ACCT_MGR_REQ:
|
|
|
|
// This is critical for acct mgrs, to propagate new host CPIDs
|
|
|
|
honor_suspend = false;
|
|
|
|
break;
|
|
|
|
case RPC_REASON_INIT:
|
|
|
|
break;
|
|
|
|
case RPC_REASON_PROJECT_REQ:
|
|
|
|
break;
|
2013-06-15 05:40:43 +00:00
|
|
|
default:
|
|
|
|
continue;
|
2009-03-04 22:55:57 +00:00
|
|
|
}
|
|
|
|
if (honor_backoff && p->waiting_until_min_rpc_time()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (honor_suspend && p->suspended_via_gui) {
|
|
|
|
continue;
|
|
|
|
}
|
2008-12-30 19:01:25 +00:00
|
|
|
if (p->sched_rpc_pending) {
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
PROJECT* CLIENT_STATE::next_project_trickle_up_pending() {
|
|
|
|
unsigned int i;
|
|
|
|
PROJECT* p;
|
|
|
|
|
|
|
|
for (i=0; i<projects.size(); i++) {
|
|
|
|
p = projects[i];
|
|
|
|
if (p->waiting_until_min_rpc_time()) continue;
|
|
|
|
if (p->suspended_via_gui) continue;
|
|
|
|
if (p->trickle_up_pending) {
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// find a project with finished results that should be reported.
|
|
|
|
// This means:
|
|
|
|
// - we're not backing off contacting the project
|
2012-03-20 19:37:04 +00:00
|
|
|
// - no upload for that project is active
|
2008-12-30 19:01:25 +00:00
|
|
|
// - the result is ready_to_report (compute done; files uploaded)
|
|
|
|
// - we're within a day of the report deadline,
|
|
|
|
// or at least a day has elapsed since the result was completed,
|
|
|
|
// or we have a sporadic connection
|
2011-05-21 18:24:00 +00:00
|
|
|
// or the project is in "don't request more work" state
|
2012-03-20 19:37:04 +00:00
|
|
|
// or a network suspend period is coming up soon
|
2012-11-19 20:38:54 +00:00
|
|
|
// or the project has > RESULT_REPORT_IF_AT_LEAST_N results ready to report
|
2008-12-30 19:01:25 +00:00
|
|
|
//
|
2012-03-20 19:37:04 +00:00
|
|
|
PROJECT* CLIENT_STATE::find_project_with_overdue_results(
|
|
|
|
bool network_suspend_soon
|
|
|
|
) {
|
2008-12-30 19:01:25 +00:00
|
|
|
unsigned int i;
|
|
|
|
RESULT* r;
|
|
|
|
|
2012-11-19 20:38:54 +00:00
|
|
|
for (i=0; i<projects.size(); i++) {
|
|
|
|
PROJECT* p = projects[i];
|
|
|
|
p->n_ready = 0;
|
|
|
|
p->dont_contact = false;
|
|
|
|
if (p->waiting_until_min_rpc_time()) p->dont_contact = true;
|
|
|
|
if (p->suspended_via_gui) p->dont_contact = true;
|
|
|
|
#ifndef SIM
|
|
|
|
if (actively_uploading(p)) p->dont_contact = true;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2008-12-30 19:01:25 +00:00
|
|
|
for (i=0; i<results.size(); i++) {
|
|
|
|
r = results[i];
|
|
|
|
if (!r->ready_to_report) continue;
|
|
|
|
|
|
|
|
PROJECT* p = r->project;
|
2012-11-19 20:38:54 +00:00
|
|
|
if (p->dont_contact) continue;
|
2008-12-30 19:01:25 +00:00
|
|
|
|
2011-05-21 18:24:00 +00:00
|
|
|
if (p->dont_request_more_work) {
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
2011-05-07 18:34:32 +00:00
|
|
|
if (r->report_immediately) {
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
2014-05-08 07:51:18 +00:00
|
|
|
if (cc_config.report_results_immediately) {
|
2008-12-30 19:01:25 +00:00
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (net_status.have_sporadic_connection) {
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
2012-03-20 19:37:04 +00:00
|
|
|
if (network_suspend_soon) {
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
2008-12-30 19:01:25 +00:00
|
|
|
double cushion = std::max(REPORT_DEADLINE_CUSHION, work_buf_min());
|
|
|
|
if (gstate.now > r->report_deadline - cushion) {
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
2013-12-30 17:53:35 +00:00
|
|
|
if (gstate.now > r->completed_time + MAX_REPORT_DELAY) {
|
2008-12-30 19:01:25 +00:00
|
|
|
return p;
|
|
|
|
}
|
2012-11-19 20:38:54 +00:00
|
|
|
|
|
|
|
p->n_ready++;
|
|
|
|
if (p->n_ready >= RESULT_REPORT_IF_AT_LEAST_N) {
|
|
|
|
return p;
|
|
|
|
}
|
2008-12-30 19:01:25 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-12-31 23:07:59 +00:00
|
|
|
// trigger work fetch
|
2011-01-06 23:09:13 +00:00
|
|
|
//
|
2008-12-31 23:07:59 +00:00
|
|
|
void CLIENT_STATE::request_work_fetch(const char* where) {
|
|
|
|
if (log_flags.work_fetch_debug) {
|
2010-04-29 20:45:44 +00:00
|
|
|
msg_printf(0, MSG_INFO, "[work_fetch] Request work fetch: %s", where);
|
2008-12-31 23:07:59 +00:00
|
|
|
}
|
|
|
|
must_check_work_fetch = true;
|
|
|
|
}
|
|
|
|
|