mirror of https://github.com/BOINC/boinc.git
558 lines
16 KiB
C++
558 lines
16 KiB
C++
// This file is part of BOINC.
|
|
// http://boinc.berkeley.edu
|
|
// Copyright (C) 2012 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/>.
|
|
|
|
// Code that's shared by the simulator, vda, and vdad
|
|
|
|
#include <algorithm>
|
|
#include <limits.h>
|
|
#include <cmath>
|
|
#include <set>
|
|
#include <stdio.h>
|
|
#include <vector>
|
|
|
|
#include "vda_lib.h"
|
|
|
|
using std::vector;
|
|
using std::set;
|
|
|
|
bool debug_status = false;
|
|
bool debug_ft = false;
|
|
|
|
///////////////// Utility functions ///////////////////////
|
|
|
|
// sort by increasing cost
|
|
//
|
|
bool compare_cost(const DATA_UNIT* d1, const DATA_UNIT* d2) {
|
|
return d1->cost < d2->cost;
|
|
}
|
|
|
|
// sort by increase min_failures
|
|
//
|
|
bool compare_min_failures(const DATA_UNIT* d1, const DATA_UNIT* d2) {
|
|
return d1->min_failures < d2->min_failures;
|
|
}
|
|
|
|
char* time_str(double t) {
|
|
static char buf[256];
|
|
int nsec = (int)fmod(t, 60);
|
|
t /= 60;
|
|
int nmin = (int)fmod(t, 60);
|
|
t /= 60;
|
|
int nhour = (int)fmod(t, 24);
|
|
t /= 24;
|
|
sprintf(buf, "%4d days %02d:%02d:%02d", (int)t, nhour, nmin, nsec);
|
|
return buf;
|
|
}
|
|
|
|
const char* status_str(int status) {
|
|
switch (status) {
|
|
case PRESENT: return "present";
|
|
case RECOVERABLE: return "recoverable";
|
|
case UNRECOVERABLE: return "unrecoverable";
|
|
}
|
|
return "unknown";
|
|
}
|
|
|
|
///////////////// META_CHUNK ///////////////////////
|
|
|
|
META_CHUNK::META_CHUNK(
|
|
VDA_FILE_AUX* d, META_CHUNK* par, double size,
|
|
int coding_level, int index
|
|
) {
|
|
dfile = d;
|
|
parent = par;
|
|
coding = d->policy.codings[coding_level];
|
|
if (parent) {
|
|
sprintf(name, "%.64s.%d", parent->name, index);
|
|
} else {
|
|
sprintf(name, "%d", index);
|
|
}
|
|
if (coding_level<d->policy.coding_levels-1) {
|
|
for (int j=0; j<coding.m; j++) {
|
|
children.push_back(new META_CHUNK(
|
|
d,
|
|
this,
|
|
size/coding.n,
|
|
coding_level+1,
|
|
j
|
|
));
|
|
}
|
|
bottom_level = false;
|
|
} else {
|
|
for (int j=0; j<coding.m; j++) {
|
|
children.push_back(
|
|
new CHUNK(this, size/coding.n, j)
|
|
);
|
|
}
|
|
bottom_level = true;
|
|
}
|
|
}
|
|
|
|
// Recovery logic: decide what to do in response to
|
|
// host failures and upload/download completions.
|
|
//
|
|
// One way to do this would be to store a bunch of state info
|
|
// with each node in the file's tree,
|
|
// and do things by local tree traversal.
|
|
//
|
|
// However, it's simpler to store minimal state info,
|
|
// and to reconstruct state info using
|
|
// a top-down tree traversal in response to each event.
|
|
// Actually we do 2 traversals:
|
|
// 1) recovery_plan()
|
|
// We see whether each node is recoverable,
|
|
// and if so compute its "recovery set":
|
|
// the set of children from which it can be recovered
|
|
// with minimal cost (i.e. network traffic).
|
|
// Decide whether each chunk currently on the server needs to remain.
|
|
// 2) decide_reconstruct()
|
|
// Decide which meta-chunks should be reconstructed
|
|
// 3) reconstruct_and_cleanup()
|
|
// Reconstruct meta-chunks and chunks, then delete meta-chunk files
|
|
// 4) recovery_action()
|
|
// decide whether to start upload/download of chunks,
|
|
// and whether to delete chunk data currently on server.
|
|
// Also compute min_failures
|
|
//
|
|
void META_CHUNK::recovery_plan() {
|
|
vector<DATA_UNIT*> recoverable;
|
|
vector<DATA_UNIT*> present;
|
|
|
|
unsigned int i;
|
|
have_unrecoverable_children = false;
|
|
need_reconstruct = false;
|
|
needed_by_parent = false;
|
|
data_now_present = false;
|
|
keep_present = false;
|
|
|
|
// make lists of children in various states
|
|
//
|
|
for (i=0; i<children.size(); i++) {
|
|
DATA_UNIT* c = children[i];
|
|
c->in_recovery_set = false;
|
|
c->data_needed = false;
|
|
c->data_now_present = false;
|
|
c->recovery_plan();
|
|
switch (c->status) {
|
|
case PRESENT:
|
|
present.push_back(c);
|
|
break;
|
|
case RECOVERABLE:
|
|
recoverable.push_back(c);
|
|
break;
|
|
case UNRECOVERABLE:
|
|
have_unrecoverable_children = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// based on states of children, decide what state we're in
|
|
//
|
|
if ((int)(present.size()) >= coding.n) {
|
|
status = PRESENT;
|
|
sort(present.begin(), present.end(), compare_cost);
|
|
present.resize(coding.n);
|
|
cost = 0;
|
|
for (i=0; i<present.size(); i++) {
|
|
DATA_UNIT* c= present[i];
|
|
cost += c->cost;
|
|
c->in_recovery_set = true;
|
|
}
|
|
} else if ((int)(present.size() + recoverable.size()) >= coding.n) {
|
|
status = RECOVERABLE;
|
|
unsigned int j = coding.n - present.size();
|
|
sort(recoverable.begin(), recoverable.end(), compare_cost);
|
|
cost = 0;
|
|
for (i=0; i<present.size(); i++) {
|
|
DATA_UNIT* c= present[i];
|
|
c->in_recovery_set = true;
|
|
}
|
|
for (i=0; i<j; i++) {
|
|
DATA_UNIT* c= recoverable[i];
|
|
c->in_recovery_set = true;
|
|
cost += c->cost;
|
|
}
|
|
|
|
} else {
|
|
status = UNRECOVERABLE;
|
|
}
|
|
}
|
|
|
|
int META_CHUNK::recovery_action(double now) {
|
|
unsigned int i;
|
|
int retval;
|
|
|
|
if (data_now_present) {
|
|
status = PRESENT;
|
|
}
|
|
if (debug_status) {
|
|
printf(" meta chunk %s: status %s have_unrec_children %d\n",
|
|
name, status_str(status), have_unrecoverable_children
|
|
);
|
|
}
|
|
for (i=0; i<children.size(); i++) {
|
|
DATA_UNIT* c = children[i];
|
|
if (debug_status) {
|
|
printf(" child %s status %s in rec set %d\n",
|
|
c->name, status_str(c->status), c->in_recovery_set
|
|
);
|
|
}
|
|
switch (status) {
|
|
case PRESENT:
|
|
if (c->status == UNRECOVERABLE) {
|
|
c->data_now_present = true;
|
|
}
|
|
break;
|
|
case RECOVERABLE:
|
|
if (c->in_recovery_set && have_unrecoverable_children) {
|
|
c->data_needed = true;
|
|
}
|
|
break;
|
|
case UNRECOVERABLE:
|
|
break;
|
|
}
|
|
retval = c->recovery_action(now);
|
|
if (retval) return retval;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
// Compute min_failures: the smallest # of host failures
|
|
// that would make this unit unrecoverable.
|
|
//
|
|
int META_CHUNK::compute_min_failures() {
|
|
unsigned int i;
|
|
for (i=0; i<children.size(); i++) {
|
|
DATA_UNIT* c = children[i];
|
|
c->compute_min_failures();
|
|
}
|
|
|
|
// Because of recovery action,
|
|
// some of our children may have changed status and fault tolerance,
|
|
// so ours may have changed too.
|
|
// Recompute them.
|
|
//
|
|
vector<DATA_UNIT*> recoverable;
|
|
vector<DATA_UNIT*> present;
|
|
for (i=0; i<children.size(); i++) {
|
|
DATA_UNIT* c = children[i];
|
|
switch (c->status) {
|
|
case PRESENT:
|
|
present.push_back(c);
|
|
break;
|
|
case RECOVERABLE:
|
|
recoverable.push_back(c);
|
|
break;
|
|
}
|
|
}
|
|
if ((int)(present.size()) >= coding.n) {
|
|
status = PRESENT;
|
|
min_failures = dfile->policy.max_ft;
|
|
} else if ((int)(present.size() + recoverable.size()) >= coding.n) {
|
|
status = RECOVERABLE;
|
|
|
|
// our min_failures is the least X such that some X host failures
|
|
// would make this node unrecoverable
|
|
//
|
|
sort(recoverable.begin(), recoverable.end(), compare_min_failures);
|
|
min_failures = 0;
|
|
unsigned int k = coding.n - present.size();
|
|
// we'd need to recover K recoverable children
|
|
unsigned int j = recoverable.size() - k + 1;
|
|
// a loss of J recoverable children would make this impossible
|
|
|
|
// the loss of J recoverable children would make us unrecoverable
|
|
// Sum the min_failures of the J children with smallest min_failures
|
|
//
|
|
for (i=0; i<j; i++) {
|
|
DATA_UNIT* c = recoverable[i];
|
|
if (debug_ft) {
|
|
printf(" Min failures of %s: %d\n", c->name, c->min_failures);
|
|
};
|
|
min_failures += c->min_failures;
|
|
}
|
|
if (debug_ft) {
|
|
printf(" our min failures: %d\n", min_failures);
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
// set the following:
|
|
// need_reconstruct: if we should reconstruct this from children
|
|
// need_present: not present, but needs to be present in the future
|
|
// keep_present: present, and needs to remain so
|
|
//
|
|
// Also set in children:
|
|
// needed_by_parent: ?
|
|
// keep_present
|
|
//
|
|
int META_CHUNK::decide_reconstruct() {
|
|
unsigned int i;
|
|
|
|
need_reconstruct = false;
|
|
if (some_child_is_unrecoverable()) {
|
|
if (status == PRESENT) {
|
|
need_reconstruct = true;
|
|
} else if (status == RECOVERABLE) {
|
|
need_present = true;
|
|
for (i=0; i<children.size(); i++) {
|
|
DATA_UNIT& c = *(children[i]);
|
|
if (c.in_recovery_set) {
|
|
if (c.status == PRESENT) {
|
|
c.keep_present = true;
|
|
} else {
|
|
c.need_present = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (needed_by_parent) {
|
|
need_reconstruct = true;
|
|
}
|
|
if (need_reconstruct and !bottom_level) {
|
|
// mark n PRESENT children as needed_by_parent
|
|
int n = 0;
|
|
for (i=0; i<children.size(); i++) {
|
|
META_CHUNK& c = *(META_CHUNK*)children[i];
|
|
if (c.status == PRESENT) {
|
|
c.needed_by_parent = true;
|
|
n++;
|
|
if (n == coding.n) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// if we need to stay present, so do a quorum of our present children
|
|
//
|
|
if (keep_present) {
|
|
int n = 0;
|
|
for (i=0; i<children.size(); i++) {
|
|
DATA_UNIT& c = *(children[i]);
|
|
if (c.status == PRESENT) {
|
|
c.keep_present = true;
|
|
n++;
|
|
if (n == coding.n) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// recurse
|
|
//
|
|
if (!bottom_level) {
|
|
for (i=0; i<children.size(); i++) {
|
|
META_CHUNK& c = *(META_CHUNK*)children[i];
|
|
c.decide_reconstruct();
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
// Recurse first, so children will be available
|
|
// If needed, reconstruct this unit (from present children)
|
|
// and then expand() (for unrecoverable children)
|
|
//
|
|
int META_CHUNK::reconstruct_and_cleanup() {
|
|
unsigned int i;
|
|
int retval;
|
|
|
|
if (!bottom_level) {
|
|
for (i=0; i<children.size(); i++) {
|
|
META_CHUNK& c = *(META_CHUNK*)children[i];
|
|
retval = c.reconstruct_and_cleanup();
|
|
if (retval) return retval;
|
|
}
|
|
}
|
|
if (need_reconstruct) {
|
|
retval = decode();
|
|
if (retval) return retval;
|
|
retval = expand();
|
|
if (retval) return retval;
|
|
if (!needed_by_parent) {
|
|
delete_file();
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
// We just decoded this chunk from a subset of its children.
|
|
// Decide whether we should now encode it (to create all of its children)
|
|
//
|
|
int META_CHUNK::expand() {
|
|
unsigned int i;
|
|
int retval;
|
|
|
|
if (bottom_level) {
|
|
bool do_encode = false;
|
|
for (i=0; i<children.size(); i++) {
|
|
CHUNK& c = *(CHUNK*)children[i];
|
|
if (c.status != PRESENT && c.need_more_replicas()) {
|
|
do_encode = true;
|
|
break;
|
|
}
|
|
}
|
|
if (do_encode) {
|
|
retval = encode(false);
|
|
if (retval) return retval;
|
|
for (i=0; i<children.size(); i++) {
|
|
CHUNK& c = *(CHUNK*)children[i];
|
|
c.present_on_server = true;
|
|
}
|
|
}
|
|
} else {
|
|
bool do_encode = false;
|
|
for (i=0; i<children.size(); i++) {
|
|
META_CHUNK& c = *(META_CHUNK*)children[i];
|
|
if (c.status == UNRECOVERABLE) {
|
|
do_encode = true;
|
|
break;
|
|
}
|
|
}
|
|
if (do_encode) {
|
|
retval = encode(false);
|
|
if (retval) return retval;
|
|
for (i=0; i<children.size(); i++) {
|
|
META_CHUNK& c = *(META_CHUNK*)children[i];
|
|
if (c.status != UNRECOVERABLE) {
|
|
c.delete_file();
|
|
}
|
|
}
|
|
for (i=0; i<children.size(); i++) {
|
|
META_CHUNK& c = *(META_CHUNK*)children[i];
|
|
if (c.status == UNRECOVERABLE) {
|
|
retval = c.expand();
|
|
if (retval) return retval;
|
|
c.delete_file();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
///////////////// CHUNK ///////////////////////
|
|
|
|
void CHUNK::recovery_plan() {
|
|
keep_present = false;
|
|
if (present_on_server) {
|
|
status = PRESENT;
|
|
cost = 0;
|
|
} else if (!hosts.empty()) {
|
|
// if file is not present on server, assume that it's present
|
|
// on all hosts (otherwise we wouldn't have downloaded it).
|
|
//
|
|
status = RECOVERABLE;
|
|
cost = size;
|
|
if ((int)(hosts.size()) < parent->dfile->policy.replication) {
|
|
data_needed = true;
|
|
}
|
|
} else {
|
|
status = UNRECOVERABLE;
|
|
}
|
|
if (debug_status) {
|
|
printf(" chunk %s: status %s\n", name, status_str(status));
|
|
}
|
|
}
|
|
|
|
int CHUNK::compute_min_failures() {
|
|
if (present_on_server) {
|
|
min_failures = parent->dfile->policy.max_ft;
|
|
return 0;
|
|
}
|
|
int nreplicas = 0;
|
|
set<VDA_CHUNK_HOST*>::iterator i;
|
|
for (i=hosts.begin(); i!=hosts.end(); ++i) {
|
|
VDA_CHUNK_HOST* ch = *i;
|
|
if (ch->present_on_host) {
|
|
nreplicas++;
|
|
}
|
|
}
|
|
min_failures = nreplicas;
|
|
return 0;
|
|
}
|
|
|
|
bool CHUNK::download_in_progress() {
|
|
set<VDA_CHUNK_HOST*>::iterator i;
|
|
for (i=hosts.begin(); i!=hosts.end(); ++i) {
|
|
VDA_CHUNK_HOST* ch = *i;
|
|
if (ch->download_in_progress()) return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
int CHUNK::recovery_action(double now) {
|
|
int retval;
|
|
char buf[1024];
|
|
|
|
VDA_FILE_AUX* fp = parent->dfile;
|
|
if (data_now_present) {
|
|
present_on_server = true;
|
|
#if 0
|
|
fp->disk_usage.sample_inc(
|
|
size,
|
|
fp->collecting_stats(),
|
|
now,
|
|
"recovery_action: now present"
|
|
);
|
|
#endif
|
|
status = PRESENT;
|
|
}
|
|
if (status == PRESENT && (int)(hosts.size()) < fp->policy.replication) {
|
|
retval = assign();
|
|
if (retval) return retval;
|
|
keep_present = true;
|
|
}
|
|
if (download_in_progress()) {
|
|
keep_present = true;
|
|
}
|
|
if (debug_status) {
|
|
printf(" chunk %s: data_needed %d present_on_server %d keep_present %d\n",
|
|
name, data_needed, present_on_server, keep_present
|
|
);
|
|
}
|
|
if (present_on_server) {
|
|
if (!keep_present) {
|
|
sprintf(buf,
|
|
" chunk %s: not needed, removing from server\n", name
|
|
);
|
|
show_msg(buf);
|
|
retval = delete_file();
|
|
if (retval) return retval;
|
|
present_on_server = false;
|
|
status = RECOVERABLE;
|
|
min_failures = fp->policy.replication;
|
|
parent->dfile->disk_usage.sample_inc(
|
|
-size,
|
|
fp->collecting_stats(),
|
|
now,
|
|
"recovery_action: don't need"
|
|
);
|
|
}
|
|
} else {
|
|
if (data_needed) {
|
|
retval = start_upload();
|
|
if (retval) return retval;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|