// 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 .
#include "cpp.h"
#ifdef _WIN32
#include "boinc_win.h"
#include "zlib.h"
#else
#include "config.h"
// Somehow having config.h define _FILE_OFFSET_BITS or _LARGE_FILES is
// causing open to be redefined to open64 which somehow, in some versions
// of zlib.h causes gzopen to be redefined as gzopen64 which subsequently gets
// reported as a linker error. So for this file, we compile in small files
// mode, regardless of these settings
#undef _FILE_OFFSET_BITS
#undef _LARGE_FILES
#undef _LARGEFILE_SOURCE
#undef _LARGEFILE64_SOURCE
#include
#include
#include
#include
#endif
#include "error_numbers.h"
#include "file_names.h"
#include "filesys.h"
#include "client_msgs.h"
#include "log_flags.h"
#include "parse.h"
#include "util.h"
#include "str_util.h"
#include "str_replace.h"
#include "client_state.h"
#include "pers_file_xfer.h"
#include "sandbox.h"
#include "client_types.h"
using std::string;
using std::vector;
PROJECT::PROJECT() {
init();
}
void PROJECT::init() {
strcpy(master_url, "");
strcpy(authenticator, "");
project_specific_prefs = "";
gui_urls = "";
resource_share = 100;
for (int i=0; i 0 && val_found);
}
if (parse_str(buf, "", name, sizeof(name))) {
rsc_type = rsc_index(name);
continue;
}
if (parse_double(buf, "", value)) {
val_found = true;
}
}
return false;
}
// parse project fields from client_state.xml
//
int PROJECT::parse_state(MIOFILE& in) {
char buf[256];
std::string sched_url;
string str1, str2;
int retval, rt;
double x;
bool btemp;
init();
while (in.fgets(buf, 256)) {
if (match_tag(buf, "")) {
if (cpid_time == 0) {
cpid_time = user_create_time;
}
return 0;
}
if (parse_str(buf, "", sched_url)) {
scheduler_urls.push_back(sched_url);
continue;
}
if (parse_str(buf, "", master_url, sizeof(master_url))) continue;
if (parse_str(buf, "", project_name, sizeof(project_name))) continue;
if (parse_str(buf, "", symstore, sizeof(symstore))) continue;
if (parse_str(buf, "", user_name, sizeof(user_name))) continue;
if (parse_str(buf, "", team_name, sizeof(team_name))) continue;
if (parse_str(buf, "", host_venue, sizeof(host_venue))) continue;
if (parse_str(buf, "", email_hash, sizeof(email_hash))) continue;
if (parse_str(buf, "", cross_project_id, sizeof(cross_project_id))) continue;
if (parse_double(buf, "", cpid_time)) continue;
if (parse_double(buf, "", user_total_credit)) continue;
if (parse_double(buf, "", user_expavg_credit)) continue;
if (parse_double(buf, "", user_create_time)) continue;
if (parse_int(buf, "", rpc_seqno)) continue;
if (parse_int(buf, "", userid)) continue;
if (parse_int(buf, "", teamid)) continue;
if (parse_int(buf, "", hostid)) continue;
if (parse_double(buf, "", host_total_credit)) continue;
if (parse_double(buf, "", host_expavg_credit)) continue;
if (parse_double(buf, "", host_create_time)) continue;
if (match_tag(buf, "")) {
retval = copy_element_contents(
in,
"",
code_sign_key,
sizeof(code_sign_key)
);
if (retval) return retval;
continue;
}
if (parse_int(buf, "", nrpc_failures)) continue;
if (parse_int(buf, "", master_fetch_failures)) continue;
if (parse_double(buf, "", min_rpc_time)) continue;
if (parse_bool(buf, "master_url_fetch_pending", master_url_fetch_pending)) continue;
if (parse_int(buf, "", sched_rpc_pending)) continue;
if (parse_double(buf, "", next_rpc_time)) continue;
if (parse_bool(buf, "trickle_up_pending", trickle_up_pending)) continue;
if (parse_int(buf, "", send_time_stats_log)) continue;
if (parse_int(buf, "", send_job_log)) continue;
if (parse_bool(buf, "send_full_workload", send_full_workload)) continue;
if (parse_bool(buf, "non_cpu_intensive", non_cpu_intensive)) continue;
if (parse_bool(buf, "verify_files_on_app_start", verify_files_on_app_start)) continue;
if (parse_bool(buf, "suspended_via_gui", suspended_via_gui)) continue;
if (parse_bool(buf, "dont_request_more_work", dont_request_more_work)) continue;
if (parse_bool(buf, "detach_when_done", detach_when_done)) continue;
if (parse_bool(buf, "ended", ended)) continue;
//#ifdef USE_REC
if (parse_double(buf, "", pwf.rec)) continue;
if (parse_double(buf, "", pwf.rec_time)) continue;
//#else
if (parse_double(buf, "", rsc_pwf[0].short_term_debt)) continue;
if (parse_double(buf, "", rsc_pwf[0].long_term_debt)) continue;
//#endif
if (parse_double(buf, "", rsc_pwf[0].backoff_interval)) continue;
if (parse_double(buf, "", rsc_pwf[0].backoff_time)) {
if (rsc_pwf[0].backoff_time > gstate.now + 28*SECONDS_PER_DAY) {
rsc_pwf[0].backoff_time = gstate.now + 28*SECONDS_PER_DAY;
}
continue;
}
//#ifndef USE_REC
if (match_tag(buf, "")) {
if (parse_rsc_param(in, "", rt, x)) {
rsc_pwf[rt].short_term_debt = x;
}
continue;
}
if (match_tag(buf, "")) {
if (parse_rsc_param(in, "", rt, x)) {
rsc_pwf[rt].long_term_debt = x;
}
continue;
}
//#endif
if (match_tag(buf, "")) {
if (parse_rsc_param(in, "", rt, x)) {
rsc_pwf[rt].backoff_interval = x;
}
continue;
}
if (match_tag(buf, "")) {
if (parse_rsc_param(in, "", rt, x)) {
rsc_pwf[rt].backoff_time = x;
}
continue;
}
if (parse_double(buf, "", resource_share)) continue;
// not authoritative
if (parse_double(buf, "", duration_correction_factor)) continue;
if (parse_bool(buf, "attached_via_acct_mgr", attached_via_acct_mgr)) continue;
if (parse_bool(buf, "no_cpu_apps", btemp)) {
handle_no_rsc_apps("CPU", btemp);
continue;
}
if (parse_bool(buf, "no_cuda_apps", btemp)) {
handle_no_rsc_apps("NVIDIA", btemp);
continue;
}
if (parse_bool(buf, "no_ati_apps", btemp)) {
handle_no_rsc_apps("ATI", btemp);
continue;
}
if (parse_str(buf, "", buf, sizeof(buf))) {
handle_no_rsc_apps(buf, true);
continue;
}
if (parse_bool(buf, "no_cpu_ams", btemp)) {
handle_no_rsc_ams("CPU", btemp);
continue;
}
if (parse_bool(buf, "no_cuda_ams", btemp)) {
handle_no_rsc_ams("NVIDIA", btemp);
continue;
}
if (parse_bool(buf, "no_ati_ams", btemp)) {
handle_no_rsc_ams("ATI", btemp);
continue;
}
if (parse_str(buf, "", buf, sizeof(buf))) {
handle_no_rsc_ams(buf, true);
continue;
}
// backwards compat - old state files had ams_resource_share = 0
if (parse_double(buf, "", ams_resource_share)) continue;
if (parse_double(buf, "", x)) {
if (x > 0) ams_resource_share = x;
continue;
}
if (parse_bool(buf, "scheduler_rpc_in_progress", btemp)) continue;
if (parse_bool(buf, "use_symlinks", use_symlinks)) continue;
if (parse_bool(buf, "anonymous_platform", btemp)) continue;
if (log_flags.unparsed_xml) {
msg_printf(0, MSG_INFO,
"[unparsed_xml] PROJECT::parse_state(): unrecognized: %s", buf
);
}
}
return ERR_XML_PARSE;
}
// Write project information to client state file or GUI RPC reply
//
int PROJECT::write_state(MIOFILE& out, bool gui_rpc) {
unsigned int i;
char un[2048], tn[2048];
out.printf(
"\n"
);
xml_escape(user_name, un, sizeof(un));
xml_escape(team_name, tn, sizeof(tn));
out.printf(
" %s\n"
" %s\n"
" %s\n"
" %s\n"
" %s\n"
" %s\n"
" %s\n"
" %s\n"
" %f\n"
" %f\n"
" %f\n"
" %f\n"
" %d\n"
" %d\n"
" %d\n"
" %d\n"
" %f\n"
" %f\n"
" %f\n"
" %d\n"
" %d\n"
" %f\n"
" %f\n"
//#ifdef USE_REC
" %f\n"
" %f\n"
//#endif
" %f\n"
" %f\n"
" %d\n"
" %d\n"
" %d\n"
"%s%s%s%s%s%s%s%s%s%s%s%s%s",
master_url,
project_name,
symstore,
un,
tn,
host_venue,
email_hash,
cross_project_id,
cpid_time,
user_total_credit,
user_expavg_credit,
user_create_time,
rpc_seqno,
userid,
teamid,
hostid,
host_total_credit,
host_expavg_credit,
host_create_time,
nrpc_failures,
master_fetch_failures,
min_rpc_time,
next_rpc_time,
//#ifdef USE_REC
pwf.rec,
pwf.rec_time,
//#else
resource_share,
duration_correction_factor,
sched_rpc_pending,
send_time_stats_log,
send_job_log,
anonymous_platform?" \n":"",
master_url_fetch_pending?" \n":"",
trickle_up_pending?" \n":"",
send_full_workload?" \n":"",
non_cpu_intensive?" \n":"",
verify_files_on_app_start?" \n":"",
suspended_via_gui?" \n":"",
dont_request_more_work?" \n":"",
detach_when_done?" \n":"",
ended?" \n":"",
attached_via_acct_mgr?" \n":"",
(this == gstate.scheduler_op->cur_proj)?" \n":"",
use_symlinks?" \n":""
);
for (int j=0; j\n"
" %s\n"
" %f\n"
" \n"
" \n"
" %s\n"
" %f\n"
" \n"
//#endif
" \n"
" %s\n"
" %f\n"
" \n"
" \n"
" %s\n"
" %f\n"
" \n",
//#ifndef USE_REC
rsc_name(j), rsc_pwf[j].short_term_debt,
rsc_name(j), rsc_pwf[j].long_term_debt,
//#endif
rsc_name(j), rsc_pwf[j].backoff_interval,
rsc_name(j), rsc_pwf[j].backoff_time
);
if (no_rsc_ams[j]) {
out.printf(" %s\n", rsc_name(j));
}
if (no_rsc_apps[j]) {
out.printf(" %s\n", rsc_name(j));
}
}
if (ams_resource_share >= 0) {
out.printf(" %f\n",
ams_resource_share
);
}
if (gui_rpc) {
out.printf("%s", gui_urls.c_str());
out.printf(
" %f\n"
" %f\n",
last_rpc_time,
project_files_downloaded_time
);
if (download_backoff.next_xfer_time > gstate.now) {
out.printf(
" %f\n",
download_backoff.next_xfer_time - gstate.now
);
}
if (upload_backoff.next_xfer_time > gstate.now) {
out.printf(
" %f\n",
upload_backoff.next_xfer_time - gstate.now
);
}
for (int j=0; j%s\n", rsc_name(j));
}
}
if (strlen(host_venue)) {
out.printf(" %s\n", host_venue);
}
} else {
for (i=0; i%s\n",
scheduler_urls[i].c_str()
);
}
if (strlen(code_sign_key)) {
out.printf(
" \n%s\n", code_sign_key
);
}
}
out.printf(
"\n"
);
return 0;
}
// Some project data is stored in account file, other in client_state.xml
// Copy fields that are stored in client_state.xml from "p" into "this"
//
void PROJECT::copy_state_fields(PROJECT& p) {
scheduler_urls = p.scheduler_urls;
safe_strcpy(project_name, p.project_name);
safe_strcpy(user_name, p.user_name);
safe_strcpy(team_name, p.team_name);
safe_strcpy(host_venue, p.host_venue);
safe_strcpy(email_hash, p.email_hash);
safe_strcpy(cross_project_id, p.cross_project_id);
user_total_credit = p.user_total_credit;
user_expavg_credit = p.user_expavg_credit;
user_create_time = p.user_create_time;
cpid_time = p.cpid_time;
rpc_seqno = p.rpc_seqno;
userid = p.userid;
teamid = p.teamid;
hostid = p.hostid;
host_total_credit = p.host_total_credit;
host_expavg_credit = p.host_expavg_credit;
host_create_time = p.host_create_time;
nrpc_failures = p.nrpc_failures;
master_fetch_failures = p.master_fetch_failures;
min_rpc_time = p.min_rpc_time;
next_rpc_time = p.next_rpc_time;
master_url_fetch_pending = p.master_url_fetch_pending;
sched_rpc_pending = p.sched_rpc_pending;
trickle_up_pending = p.trickle_up_pending;
safe_strcpy(code_sign_key, p.code_sign_key);
for (int i=0; i= 0) {
resource_share = ams_resource_share;
}
use_symlinks = p.use_symlinks;
}
// Write project statistic to project statistics file
//
int PROJECT::write_statistics(MIOFILE& out, bool /*gui_rpc*/) {
out.printf(
"\n"
" %s\n",
master_url
);
for (std::vector::iterator i=statistics.begin();
i!=statistics.end(); ++i
) {
out.printf(
" \n"
" %f\n"
" %f\n"
" %f\n"
" %f\n"
" %f\n"
" \n",
i->day,
i->user_total_credit,
i->user_expavg_credit,
i->host_total_credit,
i->host_expavg_credit
);
}
out.printf(
"\n"
);
return 0;
}
void PROJECT::suspend() {
suspended_via_gui = true;
gstate.request_schedule_cpus("project suspended");
gstate.request_work_fetch("project suspended");
}
void PROJECT::resume() {
suspended_via_gui = false;
gstate.request_schedule_cpus("project resumed");
gstate.request_work_fetch("project resumed");
}
void PROJECT::abort_not_started() {
for (unsigned int i=0; iproject != this) continue;
if (rp->not_started()) {
rp->abort_inactive(ERR_ABORTED_VIA_GUI);
}
}
}
void PROJECT::get_task_durs(double& not_started_dur, double& in_progress_dur) {
not_started_dur = 0;
in_progress_dur = 0;
for (unsigned int i=0; iproject != this) continue;
double d = rp->estimated_time_remaining();
if (rp->not_started()) {
not_started_dur += d;
} else {
in_progress_dur += d;
}
}
}
const char* PROJECT::get_scheduler_url(int index, double r) {
int n = (int) scheduler_urls.size();
int ir = (int)(r*n);
int i = (index + ir)%n;
return scheduler_urls[i].c_str();
}
bool FILE_XFER_BACKOFF::ok_to_transfer() {
double dt = next_xfer_time - gstate.now;
if (dt > gstate.pers_retry_delay_max) {
// must have changed the system clock
//
dt = 0;
}
return (dt <= 0);
}
void FILE_XFER_BACKOFF::file_xfer_failed(PROJECT* p) {
file_xfer_failures++;
if (file_xfer_failures < FILE_XFER_FAILURE_LIMIT) {
next_xfer_time = 0;
} else {
double backoff = calculate_exponential_backoff(
file_xfer_failures,
gstate.pers_retry_delay_min,
gstate.pers_retry_delay_max
);
if (log_flags.file_xfer_debug) {
msg_printf(p, MSG_INFO,
"[file_xfer] project-wide xfer delay for %f sec",
backoff
);
}
next_xfer_time = gstate.now + backoff;
}
}
void FILE_XFER_BACKOFF::file_xfer_succeeded() {
file_xfer_failures = 0;
next_xfer_time = 0;
}
int PROJECT::parse_project_files(MIOFILE& in, bool delete_existing_symlinks) {
char buf[256];
unsigned int i;
char project_dir[256], path[256];
if (delete_existing_symlinks) {
// delete current sym links.
// This is done when parsing scheduler reply,
// to ensure that we get rid of sym links for
// project files no longer in use
//
get_project_dir(this, project_dir, sizeof(project_dir));
for (i=0; i")) return 0;
if (match_tag(buf, "")) {
FILE_REF file_ref;
file_ref.parse(in);
project_files.push_back(file_ref);
} else {
if (log_flags.unparsed_xml) {
msg_printf(0, MSG_INFO,
"[unparsed_xml] parse_project_files(): unrecognized: %s\n", buf
);
}
}
}
return ERR_XML_PARSE;
}
// install pointers from FILE_REFs to FILE_INFOs for project files,
// and flag FILE_INFOs as being project files.
//
void PROJECT::link_project_files(bool recreate_symlink_files) {
FILE_INFO* fip;
vector::iterator fref_iter;
fref_iter = project_files.begin();
while (fref_iter != project_files.end()) {
FILE_REF& fref = *fref_iter;
fip = gstate.lookup_file_info(this, fref.file_name);
if (!fip) {
msg_printf(this, MSG_INTERNAL_ERROR,
"project file refers to non-existent %s", fref.file_name
);
fref_iter = project_files.erase(fref_iter);
continue;
}
fref.file_info = fip;
fip->is_project_file = true;
fref_iter++;
}
if (recreate_symlink_files) {
for (unsigned i=0; iproject == this && fip->is_project_file && fip->status == FILE_PRESENT) {
write_symlink_for_project_file(fip);
}
}
}
}
void PROJECT::write_project_files(MIOFILE& f) {
unsigned int i;
if (!project_files.size()) return;
f.printf("\n");
for (i=0; i\n");
}
// write symlinks for project files.
// Note: it's conceivable that one physical file
// has several logical names, so try them all
//
int PROJECT::write_symlink_for_project_file(FILE_INFO* fip) {
char project_dir[256], link_path[256], file_path[256];
unsigned int i;
get_project_dir(this, project_dir, sizeof(project_dir));
for (i=0; iname);
make_soft_link(this, link_path, file_path);
}
return 0;
}
// a project file download just finished.
// If it's the last one, update project_files_downloaded_time
//
void PROJECT::update_project_files_downloaded_time() {
unsigned int i;
for (i=0; istatus != FILE_PRESENT) continue;
}
project_files_downloaded_time = gstate.now;
}
int APP::parse(MIOFILE& in) {
char buf[256];
strcpy(name, "");
strcpy(user_friendly_name, "");
project = NULL;
while (in.fgets(buf, 256)) {
if (match_tag(buf, "")) {
if (!strlen(user_friendly_name)) {
strcpy(user_friendly_name, name);
}
return 0;
}
if (parse_str(buf, "", name, sizeof(name))) continue;
if (parse_str(buf, "", user_friendly_name, sizeof(user_friendly_name))) continue;
if (log_flags.unparsed_xml) {
msg_printf(0, MSG_INFO,
"[unparsed_xml] APP::parse(): unrecognized: %s\n", buf
);
}
#ifdef SIM
if (parse_double(buf, "", latency_bound)) continue;
if (parse_double(buf, "", fpops_est)) continue;
if (parse_double(buf, "", weight)) continue;
if (parse_double(buf, "", working_set)) continue;
if (match_tag(buf, "")) {
XML_PARSER xp(&in);
fpops.parse(xp, "/fpops");
continue;
}
if (match_tag(buf, "")) {
XML_PARSER xp(&in);
checkpoint_period.parse(xp, "/checkpoint_period");
continue;
}
#endif
}
return ERR_XML_PARSE;
}
int APP::write(MIOFILE& out) {
out.printf(
"\n"
" %s\n"
" %s\n"
"\n",
name, user_friendly_name
);
return 0;
}
FILE_INFO::FILE_INFO() {
strcpy(name, "");
strcpy(md5_cksum, "");
max_nbytes = 0;
nbytes = 0;
upload_offset = -1;
generated_locally = false;
status = FILE_NOT_PRESENT;
executable = false;
uploaded = false;
upload_when_present = false;
sticky = false;
gzip_when_done = false;
signature_required = false;
is_user_file = false;
is_project_file = false;
is_auto_update_file = false;
pers_file_xfer = NULL;
result = NULL;
project = NULL;
urls.clear();
start_url = -1;
current_url = -1;
strcpy(signed_xml, "");
strcpy(xml_signature, "");
strcpy(file_signature, "");
cert_sigs = 0;
}
void FILE_INFO::reset() {
status = FILE_NOT_PRESENT;
delete_file();
error_msg = "";
}
// Set the appropriate permissions depending on whether
// it's an executable file
// This doesn't seem to exist in Windows
//
int FILE_INFO::set_permissions() {
#ifdef _WIN32
return 0;
#else
int retval;
char pathname[256];
get_pathname(this, pathname, sizeof(pathname));
if (g_use_sandbox) {
// give exec permissions for user, group and others but give
// read permissions only for user and group to protect account keys
retval = set_to_project_group(pathname);
if (retval) return retval;
if (executable) {
retval = chmod(pathname,
S_IRUSR|S_IWUSR|S_IXUSR
|S_IRGRP|S_IWGRP|S_IXGRP
|S_IXOTH
);
} else {
retval = chmod(pathname,
S_IRUSR|S_IWUSR
|S_IRGRP|S_IWGRP
);
}
} else {
// give read/exec permissions for user, group and others
// in case someone runs BOINC from different user
if (executable) {
retval = chmod(pathname,
S_IRUSR|S_IWUSR|S_IXUSR
|S_IRGRP|S_IXGRP
|S_IROTH|S_IXOTH
);
} else {
retval = chmod(pathname,
S_IRUSR|S_IWUSR
|S_IRGRP
|S_IROTH
);
}
}
return retval;
#endif
}
// If from server, make an exact copy of everything
// except the start/end tags and the element.
//
int FILE_INFO::parse(MIOFILE& in, bool from_server) {
char buf[256], buf2[1024];
std::string url;
PERS_FILE_XFER *pfxp;
int retval;
while (in.fgets(buf, 256)) {
if (match_tag(buf, "")) {
if (!strlen(name)) return ERR_BAD_FILENAME;
if (strstr(name, "..")) return ERR_BAD_FILENAME;
if (strstr(name, "%")) return ERR_BAD_FILENAME;
return 0;
}
if (match_tag(buf, "")) {
retval = copy_element_contents(
in,
"",
xml_signature,
sizeof(xml_signature)
);
if (retval) return retval;
continue;
}
if (match_tag(buf, "")) {
retval = copy_element_contents(
in,
"",
file_signature,
sizeof(file_signature)
);
if (retval) return retval;
if (from_server) {
strcat(signed_xml, "\n");
strcat(signed_xml, file_signature);
strcat(signed_xml, "\n");
}
continue;
}
if (match_tag(buf, "")) {
if (!cert_sigs->parse_miofile_embed(in)) {
msg_printf(0, MSG_INTERNAL_ERROR,
"FILE_INFO::parse(): cannot parse \n");
return ERR_XML_PARSE;
}
continue;
}
safe_strcat(signed_xml, buf);
if (parse_str(buf, "", name, sizeof(name))) continue;
if (parse_str(buf, "", url)) {
urls.push_back(url);
continue;
}
if (parse_str(buf, "", md5_cksum, sizeof(md5_cksum))) continue;
if (parse_double(buf, "", nbytes)) continue;
if (parse_double(buf, "", max_nbytes)) continue;
if (parse_bool(buf, "generated_locally", generated_locally)) continue;
if (parse_int(buf, "", status)) continue;
if (parse_bool(buf, "executable", executable)) continue;
if (parse_bool(buf, "uploaded", uploaded)) continue;
if (parse_bool(buf, "upload_when_present", upload_when_present)) continue;
if (parse_bool(buf, "sticky", sticky)) continue;
if (parse_bool(buf, "gzip_when_done", gzip_when_done)) continue;
if (parse_bool(buf, "signature_required", signature_required)) continue;
if (parse_bool(buf, "is_project_file", is_project_file)) continue;
if (match_tag(buf, "")) {
pfxp = new PERS_FILE_XFER;
retval = pfxp->parse(in);
#ifdef SIM
delete pfxp;
continue;
#endif
if (!retval) {
pers_file_xfer = pfxp;
} else {
delete pfxp;
}
continue;
}
if (!from_server && match_tag(buf, "")) {
retval = copy_element_contents(
in,
"",
signed_xml,
sizeof(signed_xml)
);
if (retval) return retval;
continue;
}
if (match_tag(buf, "")) {
while (in.fgets(buf, 256)) {
if (match_tag(buf, "")) break;
}
continue;
}
if (match_tag(buf, "")) {
retval = copy_element_contents(
in, "", buf2, sizeof(buf2)
);
if (retval) return retval;
error_msg = buf2;
continue;
}
if (log_flags.unparsed_xml) {
msg_printf(0, MSG_INFO,
"[unparsed_xml] FILE_INFO::parse(): unrecognized: %s\n", buf
);
}
}
return ERR_XML_PARSE;
}
int FILE_INFO::write(MIOFILE& out, bool to_server) {
unsigned int i;
int retval;
char buf[1024];
out.printf(
"\n"
" %s\n"
" %f\n"
" %f\n",
name, nbytes, max_nbytes
);
if (strlen(md5_cksum)) {
out.printf(
" %s\n",
md5_cksum
);
}
if (!to_server) {
if (generated_locally) out.printf(" \n");
out.printf(" %d\n", status);
if (executable) out.printf(" \n");
if (uploaded) out.printf(" \n");
if (upload_when_present) out.printf(" \n");
if (sticky) out.printf(" \n");
if (gzip_when_done) out.printf(" \n");
if (signature_required) out.printf(" \n");
if (is_user_file) out.printf(" \n");
if (strlen(file_signature)) out.printf(" \n%s\n", file_signature);
}
for (i=0; i%s\n", buf);
}
if (!to_server && pers_file_xfer) {
retval = pers_file_xfer->write(out);
if (retval) return retval;
}
if (!to_server) {
if (strlen(signed_xml) && strlen(xml_signature)) {
out.printf(
" \n%s \n"
" \n%s \n",
signed_xml, xml_signature
);
}
}
if (!error_msg.empty()) {
strip_whitespace(error_msg);
out.printf(" \n%s\n\n", error_msg.c_str());
}
out.printf("\n");
return 0;
}
int FILE_INFO::write_gui(MIOFILE& out) {
out.printf(
"\n"
" %s\n"
" %s\n"
" %s\n"
" %f\n"
" %f\n"
" %d\n",
project->master_url,
project->project_name,
name,
nbytes,
max_nbytes,
status
);
if (generated_locally) out.printf(" \n");
if (uploaded) out.printf(" \n");
if (upload_when_present) out.printf(" \n");
if (sticky) out.printf(" \n");
if (pers_file_xfer) {
pers_file_xfer->write(out);
FILE_XFER_BACKOFF& fxb = project->file_xfer_backoff(pers_file_xfer->is_upload);
if (fxb.next_xfer_time > gstate.now) {
out.printf(" %f\n",
fxb.next_xfer_time - gstate.now
);
}
}
out.printf("\n");
return 0;
}
// delete physical underlying file associated with FILE_INFO
//
int FILE_INFO::delete_file() {
char path[256];
get_pathname(this, path, sizeof(path));
int retval = delete_project_owned_file(path, true);
if (retval && status != FILE_NOT_PRESENT) {
msg_printf(project, MSG_INTERNAL_ERROR, "Couldn't delete file %s", path);
}
status = FILE_NOT_PRESENT;
return retval;
}
// Files may have URLs for both upload and download.
// Call this to get the initial url,
// The is_upload arg says which kind you want.
// NULL return means there is no URL of the requested type
//
const char* FILE_INFO::get_init_url() {
if (!urls.size()) {
return NULL;
}
// if a project supplies multiple URLs, try them in order
// (e.g. in Einstein@home they're ordered by proximity to client).
//
current_url = 0;
start_url = current_url;
return urls[current_url].c_str();
}
// Call this to get the next URL.
// NULL return means you've tried them all.
//
const char* FILE_INFO::get_next_url() {
if (!urls.size()) return NULL;
while(1) {
current_url = (current_url + 1)%((int)urls.size());
if (current_url == start_url) {
return NULL;
}
return urls[current_url].c_str();
}
}
const char* FILE_INFO::get_current_url() {
if (current_url < 0) {
return get_init_url();
}
if (current_url >= (int)urls.size()) {
msg_printf(project, MSG_INTERNAL_ERROR,
"File %s has no URL", name
);
return NULL;
}
return urls[current_url].c_str();
}
// merges information from a new FILE_INFO that has the same name as one
// that is already present in the client state file.
// Potentially changes upload_when_present, max_nbytes, and signed_xml
//
int FILE_INFO::merge_info(FILE_INFO& new_info) {
char buf[256];
unsigned int i;
upload_when_present = new_info.upload_when_present;
if (max_nbytes <= 0 && new_info.max_nbytes) {
max_nbytes = new_info.max_nbytes;
sprintf(buf, " %.0f\n", new_info.max_nbytes);
strcat(signed_xml, buf);
}
// replace existing URLs with new ones
//
urls.clear();
for (i=0; i\n"
" %s\n"
" %d\n",
name,
status
);
s = buf;
if (error_msg.size()) {
sprintf(buf,
" %s\n",
error_msg.c_str()
);
s = s + buf;
}
s = s + "\n";
}
#define BUFSIZE 16384
int FILE_INFO::gzip() {
char buf[BUFSIZE];
char inpath[256], outpath[256];
get_pathname(this, inpath, sizeof(inpath));
strcpy(outpath, inpath);
strcat(outpath, ".gz");
FILE* in = boinc_fopen(inpath, "rb");
if (!in) return ERR_FOPEN;
gzFile out = gzopen(outpath, "wb");
while (1) {
int n = (int)fread(buf, 1, BUFSIZE, in);
if (n <= 0) break;
int m = gzwrite(out, buf, n);
if (m != n) {
fclose(in);
gzclose(out);
return ERR_WRITE;
}
}
fclose(in);
gzclose(out);
delete_project_owned_file(inpath, true);
boinc_rename(outpath, inpath);
return 0;
}
int APP_VERSION::parse(MIOFILE& in) {
char buf[256];
FILE_REF file_ref;
strcpy(app_name, "");
strcpy(api_version, "");
version_num = 0;
strcpy(platform, "");
strcpy(plan_class, "");
strcpy(cmdline, "");
avg_ncpus = 1;
max_ncpus = 1;
gpu_usage.rsc_type = 0;
gpu_usage.usage = 0;
gpu_ram = 0;
app = NULL;
project = NULL;
flops = gstate.host_info.p_fpops;
missing_coproc = false;
strcpy(missing_coproc_name, "");
while (in.fgets(buf, 256)) {
if (match_tag(buf, "")) return 0;
if (parse_str(buf, "", app_name, sizeof(app_name))) continue;
if (match_tag(buf, "")) {
file_ref.parse(in);
app_files.push_back(file_ref);
continue;
}
if (parse_int(buf, "", version_num)) continue;
if (parse_str(buf, "", api_version, sizeof(api_version))) continue;
if (parse_str(buf, "", platform, sizeof(platform))) continue;
if (parse_str(buf, "", plan_class, sizeof(plan_class))) continue;
if (parse_double(buf, "", avg_ncpus)) continue;
if (parse_double(buf, "", max_ncpus)) continue;
if (parse_double(buf, "", flops)) continue;
if (parse_str(buf, "", cmdline, sizeof(cmdline))) continue;
if (parse_double(buf, "", gpu_ram)) continue;
if (match_tag(buf, "")) {
COPROC_REQ cp;
int retval = cp.parse(in);
if (!retval) {
int rt = rsc_index(cp.type);
if (rt > 0) {
gpu_usage.rsc_type = rt;
gpu_usage.usage = cp.count;
} else {
missing_coproc = true;
missing_coproc_usage = cp.count;
strcpy(missing_coproc_name, cp.type);
}
} else {
msg_printf(0, MSG_INTERNAL_ERROR, "Error parsing ");
}
continue;
}
if (log_flags.unparsed_xml) {
msg_printf(0, MSG_INFO,
"[unparsed_xml] APP_VERSION::parse(): unrecognized: %s\n", buf
);
}
}
return ERR_XML_PARSE;
}
int APP_VERSION::write(MIOFILE& out, bool write_file_info) {
unsigned int i;
int retval;
out.printf(
"\n"
" %s\n"
" %d\n"
" %s\n"
" %f\n"
" %f\n"
" %f\n",
app_name,
version_num,
platform,
avg_ncpus,
max_ncpus,
flops
);
if (strlen(plan_class)) {
out.printf(" %s\n", plan_class);
}
if (strlen(api_version)) {
out.printf(" %s\n", api_version);
}
if (strlen(cmdline)) {
out.printf(" %s\n", cmdline);
}
if (write_file_info) {
for (i=0; i\n"
" %s\n"
" %f\n"
" \n",
rsc_name(gpu_usage.rsc_type),
gpu_usage.usage
);
}
if (missing_coproc) {
out.printf(
" \n"
" %s\n"
" %f\n"
" \n",
missing_coproc_name,
missing_coproc_usage
);
}
if (gpu_ram) {
out.printf(
" %f\n",
gpu_ram
);
}
out.printf(
"\n"
);
return 0;
}
bool APP_VERSION::had_download_failure(int& failnum) {
unsigned int i;
for (i=0; ihad_failure(failnum)) {
return true;
}
}
return false;
}
void APP_VERSION::get_file_errors(string& str) {
int errnum;
unsigned int i;
FILE_INFO* fip;
string msg;
str = "couldn't get input files:\n";
for (i=0; ihad_failure(errnum)) {
fip->failure_message(msg);
str = str + msg;
}
}
}
void APP_VERSION::clear_errors() {
int x;
unsigned int i;
for (i=0; ihad_failure(x)) {
fip->reset();
}
}
}
int APP_VERSION::api_major_version() {
int v, n;
n = sscanf(api_version, "%d", &v);
if (n != 1) return 0;
return v;
}
int FILE_REF::parse(MIOFILE& in) {
char buf[256];
bool temp;
strcpy(file_name, "");
strcpy(open_name, "");
main_program = false;
copy_file = false;
optional = false;
while (in.fgets(buf, 256)) {
if (match_tag(buf, "")) return 0;
if (parse_str(buf, "", file_name, sizeof(file_name))) continue;
if (parse_str(buf, "", open_name, sizeof(open_name))) continue;
if (parse_bool(buf, "main_program", main_program)) continue;
if (parse_bool(buf, "copy_file", copy_file)) continue;
if (parse_bool(buf, "optional", optional)) continue;
if (parse_bool(buf, "no_validate", temp)) continue;
if (log_flags.unparsed_xml) {
msg_printf(0, MSG_INFO,
"[unparsed_xml] FILE_REF::parse(): unrecognized: %s\n", buf
);
}
}
return ERR_XML_PARSE;
}
int FILE_REF::write(MIOFILE& out) {
out.printf(
" \n"
" %s\n",
file_name
);
if (strlen(open_name)) {
out.printf(" %s\n", open_name);
}
if (main_program) {
out.printf(" \n");
}
if (copy_file) {
out.printf(" \n");
}
if (optional) {
out.printf(" \n");
}
out.printf(" \n");
return 0;
}
int WORKUNIT::parse(MIOFILE& in) {
char buf[4096];
FILE_REF file_ref;
double dtemp;
strcpy(name, "");
strcpy(app_name, "");
version_num = 0;
command_line = "";
//strcpy(env_vars, "");
app = NULL;
project = NULL;
// Default these to very large values (1 week on a 1 cobblestone machine)
// so we don't keep asking the server for more work
rsc_fpops_est = 1e9*SECONDS_PER_DAY*7;
rsc_fpops_bound = 4e9*SECONDS_PER_DAY*7;
rsc_memory_bound = 1e8;
rsc_disk_bound = 1e9;
while (in.fgets(buf, sizeof(buf))) {
if (match_tag(buf, "")) return 0;
if (parse_str(buf, "", name, sizeof(name))) continue;
if (parse_str(buf, "", app_name, sizeof(app_name))) continue;
if (parse_int(buf, "", version_num)) continue;
if (match_tag(buf, "")) {
if (strstr(buf, "")) {
parse_str(buf, "", command_line);
} else {
bool found=false;
while (in.fgets(buf, sizeof(buf))) {
if (strstr(buf, "", env_vars, sizeof(env_vars))) continue;
if (parse_double(buf, "", rsc_fpops_est)) continue;
if (parse_double(buf, "", rsc_fpops_bound)) continue;
if (parse_double(buf, "", rsc_memory_bound)) continue;
if (parse_double(buf, "", rsc_disk_bound)) continue;
if (match_tag(buf, "")) {
file_ref.parse(in);
#ifndef SIM
input_files.push_back(file_ref);
#endif
continue;
}
// unused stuff
if (parse_double(buf, "", dtemp)) continue;
if (log_flags.unparsed_xml) {
msg_printf(0, MSG_INFO,
"[unparsed_xml] WORKUNIT::parse(): unrecognized: %s\n", buf
);
}
}
return ERR_XML_PARSE;
}
int WORKUNIT::write(MIOFILE& out) {
unsigned int i;
out.printf(
"\n"
" %s\n"
" %s\n"
" %d\n"
//" %s\n"
" %f\n"
" %f\n"
" %f\n"
" %f\n",
name,
app_name,
version_num,
//env_vars,
rsc_fpops_est,
rsc_fpops_bound,
rsc_memory_bound,
rsc_disk_bound
);
if (command_line.size()) {
out.printf(
" \n"
"%s\n"
" \n",
command_line.c_str()
);
}
for (i=0; i\n");
return 0;
}
bool WORKUNIT::had_download_failure(int& failnum) {
unsigned int i;
for (i=0;ihad_failure(failnum)) {
return true;
}
}
return false;
}
void WORKUNIT::get_file_errors(string& str) {
int x;
unsigned int i;
FILE_INFO* fip;
string msg;
str = "couldn't get input files:\n";
for (i=0;ihad_failure(x)) {
fip->failure_message(msg);
str = str + msg;
}
}
}
// if any input files had download error from previous WU,
// reset them to try download again
//
void WORKUNIT::clear_errors() {
int x;
unsigned int i;
for (i=0; ihad_failure(x)) {
fip->reset();
}
}
}
int RESULT::parse_name(FILE* in, const char* end_tag) {
char buf[256];
strcpy(name, "");
while (fgets(buf, 256, in)) {
if (match_tag(buf, end_tag)) return 0;
if (parse_str(buf, "", name, sizeof(name))) continue;
if (log_flags.unparsed_xml) {
msg_printf(0, MSG_INFO,
"[unparsed_xml] RESULT::parse_name(): unrecognized: %s\n", buf
);
}
}
return ERR_XML_PARSE;
}
void RESULT::clear() {
strcpy(name, "");
strcpy(wu_name, "");
report_deadline = 0;
received_time = 0;
output_files.clear();
_state = RESULT_NEW;
ready_to_report = false;
completed_time = 0;
got_server_ack = false;
final_cpu_time = 0;
final_elapsed_time = 0;
#ifdef SIM
peak_flop_count = 0;
#endif
exit_status = 0;
stderr_out = "";
suspended_via_gui = false;
rr_sim_misses_deadline = false;
fpops_per_cpu_sec = 0;
fpops_cumulative = 0;
intops_per_cpu_sec = 0;
intops_cumulative = 0;
app = NULL;
wup = NULL;
project = NULL;
version_num = 0;
strcpy(platform, "");
strcpy(plan_class, "");
strcpy(resources, "");
coproc_missing = false;
schedule_backoff = 0;
}
// parse a element from scheduling server.
//
int RESULT::parse_server(MIOFILE& in) {
char buf[256];
FILE_REF file_ref;
clear();
while (in.fgets(buf, 256)) {
if (match_tag(buf, "")) return 0;
if (parse_str(buf, "", name, sizeof(name))) continue;
if (parse_str(buf, "", wu_name, sizeof(wu_name))) continue;
if (parse_double(buf, "", report_deadline)) continue;
if (parse_str(buf, "", platform, sizeof(platform))) continue;
if (parse_str(buf, "", plan_class, sizeof(plan_class))) continue;
if (parse_int(buf, "", version_num)) continue;
if (match_tag(buf, "")) {
file_ref.parse(in);
output_files.push_back(file_ref);
continue;
}
if (log_flags.unparsed_xml) {
msg_printf(0, MSG_INFO,
"[unparsed_xml] RESULT::parse(): unrecognized: %s\n", buf
);
}
}
return ERR_XML_PARSE;
}
// parse a element from state file
//
int RESULT::parse_state(MIOFILE& in) {
char buf[256];
FILE_REF file_ref;
clear();
while (in.fgets(buf, 256)) {
if (match_tag(buf, "")) {
// set state to something reasonable in case of bad state file
//
if (got_server_ack || ready_to_report) {
switch (state()) {
case RESULT_NEW:
case RESULT_FILES_DOWNLOADING:
case RESULT_FILES_DOWNLOADED:
case RESULT_FILES_UPLOADING:
set_state(RESULT_FILES_UPLOADED, "RESULT::parse_state");
break;
}
}
return 0;
}
if (parse_str(buf, "", name, sizeof(name))) continue;
if (parse_str(buf, "", wu_name, sizeof(wu_name))) continue;
if (parse_double(buf, "", received_time)) continue;
if (parse_double(buf, "", report_deadline)) {
continue;
}
if (match_tag(buf, "")) {
file_ref.parse(in);
#ifndef SIM
output_files.push_back(file_ref);
#endif
continue;
}
if (parse_double(buf, "", final_cpu_time)) continue;
if (parse_double(buf, "", final_elapsed_time)) continue;
if (parse_int(buf, "", exit_status)) continue;
if (parse_bool(buf, "got_server_ack", got_server_ack)) continue;
if (parse_bool(buf, "ready_to_report", ready_to_report)) continue;
if (parse_double(buf, "", completed_time)) continue;
if (parse_bool(buf, "suspended_via_gui", suspended_via_gui)) continue;
if (parse_int(buf, "", _state)) continue;
if (match_tag(buf, "")) {
while (in.fgets(buf, 256)) {
if (match_tag(buf, "")) break;
if (strstr(buf, "")) continue;
stderr_out.append(buf);
}
continue;
}
if (parse_double(buf, "", fpops_per_cpu_sec)) continue;
if (parse_double(buf, "", fpops_cumulative)) continue;
if (parse_double(buf, "", intops_per_cpu_sec)) continue;
if (parse_double(buf, "", intops_cumulative)) continue;
if (parse_str(buf, "", platform, sizeof(platform))) continue;
if (parse_str(buf, "", plan_class, sizeof(plan_class))) continue;
if (parse_int(buf, "", version_num)) continue;
if (log_flags.unparsed_xml) {
msg_printf(0, MSG_INFO,
"[unparsed_xml] RESULT::parse(): unrecognized: %s\n", buf
);
}
}
return ERR_XML_PARSE;
}
int RESULT::write(MIOFILE& out, bool to_server) {
unsigned int i;
FILE_INFO* fip;
int n, retval;
out.printf(
"\n"
" %s\n"
" %f\n"
" %f\n"
" %d\n"
" %d\n"
" %s\n"
" %d\n",
name,
final_cpu_time,
final_elapsed_time,
exit_status,
state(),
platform,
version_num
);
if (strlen(plan_class)) {
out.printf(" %s\n", plan_class);
}
if (fpops_per_cpu_sec) {
out.printf(" %f\n", fpops_per_cpu_sec);
}
if (fpops_cumulative) {
out.printf(" %f\n", fpops_cumulative);
}
if (intops_per_cpu_sec) {
out.printf(" %f\n", intops_per_cpu_sec);
}
if (intops_cumulative) {
out.printf(" %f\n", intops_cumulative);
}
if (to_server) {
out.printf(
" %d\n",
wup->version_num
);
}
n = (int)stderr_out.length();
if (n || to_server) {
out.printf("\n");
// the following is here so that it gets recorded on server
// (there's no core_client_version field of result table)
//
if (to_server) {
out.printf(
"%d.%d.%d\n",
gstate.core_client_version.major,
gstate.core_client_version.minor,
gstate.core_client_version.release
);
}
if (n) {
out.printf("\n");
}
out.printf("\n");
}
if (to_server) {
for (i=0; iuploaded) {
retval = fip->write(out, true);
if (retval) return retval;
}
}
} else {
if (got_server_ack) out.printf(" \n");
if (ready_to_report) out.printf(" \n");
if (completed_time) out.printf(" %f\n", completed_time);
if (suspended_via_gui) out.printf(" \n");
out.printf(
" %s\n"
" %f\n"
" %f\n",
wu_name,
report_deadline,
received_time
);
for (i=0; i\n");
return 0;
}
#ifndef SIM
int RESULT::write_gui(MIOFILE& out) {
out.printf(
"\n"
" %s\n"
" %s\n"
" %d\n"
" %s\n"
" %s\n"
" %f\n"
" %f\n"
" %d\n"
" %d\n"
" %f\n"
" %f\n"
" %f\n",
name,
wu_name,
version_num,
plan_class,
project->master_url,
final_cpu_time,
final_elapsed_time,
exit_status,
state(),
report_deadline,
received_time,
estimated_time_remaining()
);
if (got_server_ack) out.printf(" \n");
if (ready_to_report) out.printf(" \n");
if (completed_time) out.printf(" %f\n", completed_time);
if (suspended_via_gui) out.printf(" \n");
if (project->suspended_via_gui) out.printf(" \n");
if (edf_scheduled) out.printf(" \n");
if (coproc_missing) out.printf(" \n");
if (schedule_backoff > gstate.now) out.printf(" \n");
ACTIVE_TASK* atp = gstate.active_tasks.lookup_result(this);
if (atp) {
atp->write_gui(out);
}
if (!strlen(resources)) {
// only need to compute this string once
//
if (avp->gpu_usage.rsc_type) {
sprintf(resources,
"%.2f CPUs + %.2f %s GPUs",
avp->avg_ncpus,
avp->gpu_usage.usage,
rsc_name(avp->gpu_usage.rsc_type)
);
} else if (avp->missing_coproc) {
sprintf(resources, "%.2f CPUs + %s GPU (missing)",
avp->avg_ncpus, avp->missing_coproc_name
);
} else if (avp->avg_ncpus != 1) {
sprintf(resources, "%.2f CPUs", avp->avg_ncpus);
} else {
strcpy(resources, " ");
}
}
if (strlen(resources)>1) {
char buf[256];
strcpy(buf, "");
if (atp && atp->task_state() == PROCESS_EXECUTING) {
if (avp->gpu_usage.rsc_type) {
sprintf(buf, " (device %d)", coproc_indices[0]);
}
}
out.printf(
" %s%s\n", resources, buf
);
}
out.printf("\n");
return 0;
}
#endif
// Returns true if the result's output files are all either
// successfully uploaded or have unrecoverable errors
//
bool RESULT::is_upload_done() {
unsigned int i;
FILE_INFO* fip;
int retval;
for (i=0; iupload_when_present) {
if (fip->had_failure(retval)) continue;
if (!fip->uploaded) {
return false;
}
}
}
return true;
}
// resets all FILE_INFO's in result to uploaded = false
// if upload_when_present is true.
//
void RESULT::clear_uploaded_flags() {
unsigned int i;
FILE_INFO* fip;
for (i=0; iupload_when_present) {
fip->uploaded = false;
}
}
}
bool PROJECT::some_download_stalled() {
#ifndef SIM
unsigned int i;
if (!download_backoff.ok_to_transfer()) return true;
for (i=0; ipers_file_xfers.size(); i++) {
PERS_FILE_XFER* pfx = gstate.pers_file_xfers->pers_file_xfers[i];
if (pfx->fip->project != this) continue;
if (pfx->is_upload) continue;
if (pfx->next_request_time > gstate.now) return true;
}
#endif
return false;
}
// return true if some file needed by this result (input or application)
// is downloading and backed off
//
bool RESULT::some_download_stalled() {
#ifndef SIM
unsigned int i;
FILE_INFO* fip;
PERS_FILE_XFER* pfx;
bool some_file_missing = false;
for (i=0; iinput_files.size(); i++) {
fip = wup->input_files[i].file_info;
if (fip->status != FILE_PRESENT) some_file_missing = true;
pfx = fip->pers_file_xfer;
if (pfx && pfx->next_request_time > gstate.now) {
return true;
}
}
for (i=0; iapp_files.size(); i++) {
fip = avp->app_files[i].file_info;
if (fip->status != FILE_PRESENT) some_file_missing = true;
pfx = fip->pers_file_xfer;
if (pfx && pfx->next_request_time > gstate.now) {
return true;
}
}
if (some_file_missing && !project->download_backoff.ok_to_transfer()) {
return true;
}
#endif
return false;
}
FILE_REF* RESULT::lookup_file(FILE_INFO* fip) {
for (unsigned int i=0; irsc_fpops_est, name, final_elapsed_time
);
fclose(f);
}
// abort a result that's not currently running
//
void RESULT::abort_inactive(int status) {
if (state() >= RESULT_COMPUTE_ERROR) return;
set_state(RESULT_ABORTED, "RESULT::abort_inactive");
exit_status = status;
}
MODE::MODE() {
perm_mode = 0;
temp_mode = 0;
temp_timeout = 0;
}
void MODE::set(int mode, double duration) {
if (mode == RUN_MODE_RESTORE) {
temp_timeout = 0;
temp_mode = perm_mode;
return;
}
if (duration) {
temp_mode = mode;
temp_timeout = gstate.now + duration;
} else {
temp_timeout = 0;
temp_mode = mode;
perm_mode = mode;
gstate.set_client_state_dirty("Set mode");
}
}
int MODE::get_perm() {
return perm_mode;
}
int MODE::get_current() {
if (temp_timeout > gstate.now) {
return temp_mode;
} else {
return perm_mode;
}
}
double MODE::delay() {
if (temp_timeout > gstate.now) {
return temp_timeout - gstate.now;
} else {
return 0;
}
}