mirror of https://github.com/BOINC/boinc.git
269 lines
8.5 KiB
C++
269 lines
8.5 KiB
C++
// This file is part of BOINC.
|
|
// http://boinc.berkeley.edu
|
|
// Copyright (C) 2022 University of California
|
|
//
|
|
// BOINC is free software; you can redistribute it and/or modify it
|
|
// under the terms of the GNU Lesser General Public License
|
|
// as published by the Free Software Foundation,
|
|
// either version 3 of the License, or (at your option) any later version.
|
|
//
|
|
// BOINC is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
// See the GNU Lesser General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
// along with BOINC. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
#ifndef BOINC_RESULT_H
|
|
#define BOINC_RESULT_H
|
|
|
|
#include "project.h"
|
|
|
|
struct RESULT {
|
|
char name[256];
|
|
char wu_name[256];
|
|
double received_time; // when we got this from server
|
|
double report_deadline;
|
|
int version_num; // identifies the app used
|
|
char plan_class[64];
|
|
char platform[256];
|
|
APP_VERSION* avp;
|
|
std::vector<FILE_REF> output_files;
|
|
bool ready_to_report;
|
|
// we're ready to report this result to the server;
|
|
// either computation is done and all the files have been uploaded
|
|
// or there was an error
|
|
double completed_time;
|
|
// time when ready_to_report was set
|
|
bool got_server_ack;
|
|
// we've received the ack for this result from the server
|
|
double final_cpu_time;
|
|
double final_elapsed_time;
|
|
double final_peak_working_set_size;
|
|
double final_peak_swap_size;
|
|
double final_peak_disk_usage;
|
|
double final_bytes_sent;
|
|
double final_bytes_received;
|
|
#ifdef SIM
|
|
double peak_flop_count;
|
|
double sim_flops_left;
|
|
#endif
|
|
|
|
// the following are nonzero if reported by app
|
|
double fpops_per_cpu_sec;
|
|
double fpops_cumulative;
|
|
double intops_per_cpu_sec;
|
|
double intops_cumulative;
|
|
|
|
int _state;
|
|
// state of this result: see lib/result_state.h
|
|
inline int state() { return _state; }
|
|
inline void set_ready_to_report() {
|
|
ready_to_report = true;
|
|
}
|
|
void set_state(int, const char*);
|
|
int exit_status;
|
|
// return value from the application
|
|
std::string stderr_out;
|
|
// the concatenation of:
|
|
//
|
|
// - if report_result_error() is called for this result:
|
|
// <message>x</message>
|
|
// <exit_status>x</exit_status>
|
|
// <signal>x</signal>
|
|
// - if called in FILES_DOWNLOADED state:
|
|
// <couldnt_start>x</couldnt_start>
|
|
// - if called in NEW state:
|
|
// <download_error>x</download_error> for each failed download
|
|
// - if called in COMPUTE_DONE state:
|
|
// <upload_error>x</upload_error> for each failed upload
|
|
//
|
|
// - <stderr_txt>X</stderr_txt>, where X is the app's stderr output
|
|
bool suspended_via_gui;
|
|
bool coproc_missing;
|
|
// a coproc needed by this job is missing
|
|
// (e.g. because user removed their GPU board).
|
|
bool report_immediately;
|
|
bool not_started; // temp for CPU sched
|
|
|
|
std::string name_md5; // see sort_results();
|
|
int index; // index in results vector
|
|
|
|
APP* app;
|
|
WORKUNIT* wup;
|
|
PROJECT* project;
|
|
|
|
RESULT(){
|
|
clear();
|
|
}
|
|
~RESULT(){}
|
|
void clear();
|
|
int parse_server(XML_PARSER&);
|
|
int parse_state(XML_PARSER&);
|
|
int parse_name(XML_PARSER&, const char* end_tag);
|
|
int write(MIOFILE&, bool to_server);
|
|
int write_gui(MIOFILE&, bool check_resources = false);
|
|
bool is_upload_done(); // files uploaded?
|
|
void clear_uploaded_flags();
|
|
FILE_REF* lookup_file(FILE_INFO*);
|
|
FILE_INFO* lookup_file_logical(const char*);
|
|
void abort_inactive(int);
|
|
// abort the result if it hasn't started computing yet
|
|
// Called only for results with no active task
|
|
// (otherwise you need to abort the active task)
|
|
void append_log_record();
|
|
|
|
// stuff related to CPU scheduling
|
|
|
|
bool is_not_started();
|
|
double estimated_runtime();
|
|
double estimated_runtime_uncorrected();
|
|
double estimated_runtime_remaining();
|
|
inline double estimated_flops_remaining() {
|
|
#ifdef SIM
|
|
return sim_flops_left;
|
|
#else
|
|
return estimated_runtime_remaining()*avp->flops;
|
|
#endif
|
|
}
|
|
|
|
inline bool computing_done() {
|
|
if (state() >= RESULT_COMPUTE_ERROR) return true;
|
|
if (ready_to_report) return true;
|
|
return false;
|
|
}
|
|
bool runnable();
|
|
// downloaded, not finished, not suspended, project not suspended
|
|
bool nearly_runnable();
|
|
// downloading or downloaded,
|
|
// not finished, suspended, project not suspended
|
|
bool downloading();
|
|
// downloading, not downloaded, not suspended, project not suspended
|
|
bool some_download_stalled();
|
|
// some input or app file is downloading, and backed off
|
|
// i.e. it may be a long time before we can run this result
|
|
inline bool uses_coprocs() {
|
|
return (avp->gpu_usage.rsc_type != 0);
|
|
}
|
|
inline bool uses_gpu() {
|
|
int rt = avp->gpu_usage.rsc_type;
|
|
if (!rt) return false;
|
|
if (coprocs.coprocs[rt].non_gpu) return false;
|
|
return true;
|
|
}
|
|
inline int resource_type() {
|
|
return avp->gpu_usage.rsc_type;
|
|
}
|
|
inline bool non_cpu_intensive() {
|
|
return app->non_cpu_intensive;
|
|
}
|
|
inline bool sporadic() {
|
|
return app->sporadic;
|
|
}
|
|
inline bool always_run() {
|
|
return non_cpu_intensive() || sporadic();
|
|
}
|
|
inline bool dont_throttle() {
|
|
if (non_cpu_intensive()) return true;
|
|
if (avp->dont_throttle) return true;
|
|
return false;
|
|
}
|
|
// make a string describing resource usage
|
|
inline void rsc_string(char* buf, int len) {
|
|
if (avp->gpu_usage.rsc_type) {
|
|
snprintf(buf, len,
|
|
"%.2f CPU + %.2f %s",
|
|
avp->avg_ncpus, avp->gpu_usage.usage,
|
|
rsc_name_long(avp->gpu_usage.rsc_type)
|
|
);
|
|
} else {
|
|
snprintf(buf, len, "%.2f CPU", avp->avg_ncpus);
|
|
}
|
|
}
|
|
|
|
|
|
// temporaries used in CLIENT_STATE::rr_simulation():
|
|
double rrsim_flops_left;
|
|
double rrsim_finish_delay;
|
|
double rrsim_flops;
|
|
bool rrsim_done;
|
|
|
|
bool already_selected;
|
|
// used to keep cpu scheduler from scheduling a result twice
|
|
// transient; used only within schedule_cpus()
|
|
// also used in round-robin simulation
|
|
double computation_deadline();
|
|
// report deadline - prefs.work_buf_min - time slice
|
|
bool rr_sim_misses_deadline;
|
|
|
|
// temporaries used in enforce_schedule():
|
|
bool unfinished_time_slice;
|
|
int seqno;
|
|
|
|
bool edf_scheduled;
|
|
// temporary used to tell GUI that this result is deadline-scheduled
|
|
|
|
int coproc_indices[MAX_COPROCS_PER_JOB];
|
|
// keep track of coprocessor reservations
|
|
char resources[256];
|
|
// textual description of resources used
|
|
double schedule_backoff;
|
|
// don't try to schedule until this time
|
|
// (because the app called boinc_temporary_exit())
|
|
char schedule_backoff_reason[256];
|
|
};
|
|
|
|
#define CONCURRENT_LIMIT_APP 1
|
|
#define CONCURRENT_LIMIT_PROJECT 2
|
|
|
|
// are we at or beyond a max_concurrent limit for this job?
|
|
//
|
|
inline int max_concurrent_exceeded(RESULT* rp) {
|
|
APP* app = rp->app;
|
|
if (app->max_concurrent) {
|
|
if (app->app_n_concurrent >= app->max_concurrent) {
|
|
return CONCURRENT_LIMIT_APP;
|
|
}
|
|
}
|
|
PROJECT* p = rp->project;
|
|
if (p->app_configs.project_max_concurrent) {
|
|
if (p->proj_n_concurrent >= p->app_configs.project_max_concurrent) {
|
|
return CONCURRENT_LIMIT_PROJECT;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
// we're about to run this job.
|
|
// increment counters for enforcing max_concurrent prefs
|
|
// Used in RR simulation (RR_SIM::activate())
|
|
// and in job scheduling (enforce_run_list())
|
|
//
|
|
inline void max_concurrent_inc(RESULT* rp) {
|
|
rp->app->app_n_concurrent++;
|
|
rp->project->proj_n_concurrent++;
|
|
}
|
|
|
|
// a completed result, for which the RESULT record no longer exists.
|
|
// We keep an in-memory log of these.
|
|
// Keep this consistent with lib/gui_rpc_client.h
|
|
//
|
|
struct OLD_RESULT {
|
|
char project_url[256];
|
|
char result_name[256];
|
|
char app_name[256];
|
|
int exit_status;
|
|
double elapsed_time;
|
|
double cpu_time;
|
|
double completed_time;
|
|
double create_time; // when this record was created
|
|
};
|
|
|
|
extern std::deque<OLD_RESULT> old_results;
|
|
|
|
void add_old_result(RESULT&);
|
|
extern void print_old_results(MIOFILE&);
|
|
|
|
#endif
|