// Berkeley Open Infrastructure for Network Computing // http://boinc.berkeley.edu // Copyright (C) 2005 University of California // // This 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 2.1 of the License, or (at your option) any later version. // // This software 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. // // To view the GNU Lesser General Public License visit // http://www.gnu.org/copyleft/lesser.html // or write to the Free Software Foundation, Inc., // 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA // This is the primary sample BOINC application; // it shows most of the features of the BOINC API. // // read "in", convert to upper case, write to "out" // // command line options (use for debugging various scenarios): // -run_slow: sleep 1 second after each character; useful for debugging // -cpu_time N: use about N CPU seconds after copying files // -early_exit: exit(10) after 30 chars // -early_crash: crash after 30 chars // #ifdef _WIN32 #include "boinc_win.h" #else #include "config.h" #endif #ifndef _WIN32 #include #include #include #include #include #include #include #endif #define BOINC_APP_GRAPHICS #ifdef BOINC_APP_GRAPHICS #include "graphics_api.h" #include "graphics_lib.h" #endif #include "diagnostics.h" #include "str_util.h" #include "util.h" #include "filesys.h" #include "boinc_api.h" #include "mfile.h" using std::string; #define CHECKPOINT_FILE "upper_case_state" #define INPUT_FILENAME "in" #define OUTPUT_FILENAME "out" bool run_slow = false; bool early_exit = false; bool early_crash = false; bool early_sleep = false; double cpu_time = 20; static void use_some_cpu() { double j = 3.14159; int i, n = 0; for (i=0; i<20000000; i++) { n++; j *= n+j-3.14159; j /= (float)n; } } int do_checkpoint(MFILE& mf, int nchars) { int retval; string resolved_name; FILE* f = fopen("temp", "w"); if (!f) return 1; fprintf(f, "%d", nchars); fclose(f); fprintf(stderr, "APP: upper_case checkpointing\n"); retval = mf.flush(); if (retval) return retval; boinc_resolve_filename_s(CHECKPOINT_FILE, resolved_name); retval = boinc_rename("temp", resolved_name.c_str()); if (retval) return retval; //use_some_cpu(); fprintf(stderr, "APP: upper_case checkpoint done\n"); return 0; } void worker() { int c, nchars = 0, retval, n; double fsize; char input_path[512], output_path[512], chkpt_path[512]; MFILE out; FILE* state, *infile; // open the input file (resolve logical name first) // boinc_resolve_filename(INPUT_FILENAME, input_path, sizeof(input_path)); infile = boinc_fopen(input_path, "r"); if (!infile) { fprintf(stderr, "Couldn't find input file, resolved name %s.\n", input_path ); exit(-1); } // get size of input file (used to compute fraction done) // file_size(input_path, fsize); // open output file // boinc_resolve_filename(OUTPUT_FILENAME, output_path, sizeof(output_path)); // See if there's a valid checkpoint file. // If so seek input file and truncate output file // boinc_resolve_filename(CHECKPOINT_FILE, chkpt_path, sizeof(chkpt_path)); state = boinc_fopen(chkpt_path, "r"); if (state) { n = fscanf(state, "%d", &nchars); fclose(state); } if (state && n==1) { fseek(infile, nchars, SEEK_SET); boinc_truncate(output_path, nchars); retval = out.open(output_path, "a"); } else { retval = out.open(output_path, "w"); } if (retval) { fprintf(stderr, "APP: upper_case output open failed:\n"); fprintf(stderr, "resolved name %s, retval %d\n", output_path, retval); perror("open"); exit(1); } // main loop - read characters, convert to UC, write // for (int i=0; ; i++) { c = fgetc(infile); if (c == EOF) break; c = toupper(c); out._putchar(c); nchars++; if (run_slow) { boinc_sleep(1.); } if (early_exit && i>30) { exit(-10); } if (early_crash && i>30) { #ifdef _WIN32 DebugBreak(); #else *(int*)0 = 0; #endif } if (early_sleep && i>30) { g_sleep = true; while (1) boinc_sleep(1); } if (boinc_time_to_checkpoint()) { retval = do_checkpoint(out, nchars); if (retval) { fprintf(stderr, "APP: upper_case checkpoint failed %d\n", retval); exit(retval); } boinc_checkpoint_completed(); } double f = nchars/fsize; if (cpu_time) f /= 2; boinc_fraction_done(f); } retval = out.flush(); if (retval) { fprintf(stderr, "APP: upper_case flush failed %d\n", retval); exit(1); } // burn up some CPU time if needed // if (cpu_time) { double start = dtime(); while (1) { double e = dtime()-start; if (e > cpu_time) break; boinc_fraction_done(.5 + e/(cpu_time*2)); if (boinc_time_to_checkpoint()) { retval = do_checkpoint(out, nchars); if (retval) { fprintf(stderr, "APP: upper_case checkpoint failed %d\n", retval); exit(1); } boinc_checkpoint_completed(); } use_some_cpu(); } } boinc_finish(0); } int main(int argc, char **argv) { int i; int retval = 0; for (i=0; i