// The contents of this file are subject to the Mozilla Public License // Version 1.0 (the "License"); you may not use this file except in // compliance with the License. You may obtain a copy of the License at // http://www.mozilla.org/MPL/ // // Software distributed under the License is distributed on an "AS IS" // basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the // License for the specific language governing rights and limitations // under the License. // // The Original Code is the Berkeley Open Infrastructure for Network Computing. // // The Initial Developer of the Original Code is the SETI@home project. // Portions created by the SETI@home project are Copyright (C) 2002 // University of California at Berkeley. All Rights Reserved. // // Contributor(s): // // API_IGNORE_CLIENT will make the app ignore the core client // this is useful for debugging just the application //#define API_IGNORE_CLIENT #include #include #include #ifdef WIN32 #include #include #include #include #include // for timing HANDLE hGlobalDrawEvent; #endif #if HAVE_UNISTD_H #include #endif #ifdef HAVE_SYS_TIME_H #include #endif #include #include #include #ifdef BOINC_APP_GRAPHICS #ifdef __APPLE_CC__ #include #endif #ifdef _WIN32 #include // Header File For The OpenGL32 Library #include // Header File For The GLu32 Library #include // Header File For The Glaux Library #endif #endif #include "parse.h" #include "error_numbers.h" #include "graphics_api.h" #ifdef __APPLE_CC__ #include #include "mac_app_opengl.h" MPQueueID drawQueue; #endif #include "boinc_api.h" static APP_INIT_DATA aid; GRAPHICS_INFO gi; static double timer_period = 1.0/50.0; // 50 Hz timer static double time_until_checkpoint; static double time_until_redraw; static double time_until_fraction_done_update; static double fraction_done; static bool ready_to_checkpoint = false; static bool ready_to_redraw = false; static bool this_process_active; int ok_to_draw = 0; // read the INIT_DATA and FD_INIT files // int boinc_init() { #ifdef _WIN32 freopen(STDERR_FILE, "a", stderr); fprintf(stderr, "in boinc_init()\n"); #endif #ifndef API_IGNORE_CLIENT FILE* f; int retval; f = fopen(INIT_DATA_FILE, "r"); if (!f) { fprintf(stderr, "boinc_init(): can't open init data file\n"); return ERR_FOPEN; } retval = parse_init_data_file(f, aid); if (retval) { fprintf(stderr, "boinc_init(): can't parse init data file\n"); return retval; } fclose(f); f = fopen(GRAPHICS_DATA_FILE, "r"); if (!f) { fprintf(stderr, "boinc_init(): can't open graphics data file\n"); return ERR_FOPEN; } retval = parse_graphics_file(f, &gi); if (retval) { fprintf(stderr, "boinc_init(): can't parse graphics data file\n"); return retval; } fclose(f); f = fopen(FD_INIT_FILE, "r"); if (f) { parse_fd_init_file(f); fclose(f); } #else strcpy(aid.app_preferences, ""); strcpy(aid.user_name, "John Smith"); strcpy(aid.team_name, "The A-Team"); aid.wu_cpu_time = 1000; aid.total_cobblestones = 1000; aid.recent_avg_cobblestones = 500; aid.checkpoint_period = DEFAULT_CHECKPOINT_PERIOD; aid.fraction_done_update_period = DEFAULT_FRACTION_DONE_UPDATE_PERIOD; gi.graphics_mode = MODE_WINDOW; gi.refresh_period = 0.1; // 1/10th of a second gi.xsize = 640; gi.ysize = 480; #endif time_until_checkpoint = aid.checkpoint_period; time_until_fraction_done_update = aid.fraction_done_update_period; time_until_redraw = gi.refresh_period; this_process_active = true; set_timer(timer_period); boinc_install_signal_handlers(); return 0; } // Install signal handlers to aid in debugging // TODO: write Windows equivalent error handlers? // int boinc_install_signal_handlers() { #ifdef HAVE_SIGNAL_H // terminal line hangup signal( SIGHUP, boinc_catch_signal ); // interrupt program signal( SIGINT, boinc_catch_signal ); // quit program signal( SIGQUIT, boinc_catch_signal ); // illegal instruction signal( SIGILL, boinc_catch_signal ); // abort(2) call signal( SIGABRT, boinc_catch_signal ); // bus error signal( SIGBUS, boinc_catch_signal ); // segmentation violation signal( SIGSEGV, boinc_catch_signal ); // system call given invalid argument signal( SIGSYS, boinc_catch_signal ); // write on a pipe with no reader signal( SIGPIPE, boinc_catch_signal ); #endif return 0; } void boinc_catch_signal(int signal) { #ifdef HAVE_SIGNAL_H switch(signal) { case SIGHUP: // terminal line hangup fprintf( stderr, "SIGHUP: terminal line hangup" ); break; case SIGINT: // interrupt program fprintf( stderr, "SIGINT: interrupt program" ); break; case SIGQUIT: // quit program fprintf( stderr, "SIGQUIT: quit program" ); break; case SIGILL: // illegal instruction fprintf( stderr, "SIGILL: illegal instruction" ); break; case SIGABRT: // abort(2) call fprintf( stderr, "SIGABRT: abort called" ); break; case SIGBUS: // bus error fprintf( stderr, "SIGBUS: bus error" ); break; case SIGSEGV: // segmentation violation fprintf( stderr, "SIGSEGV: segmentation violation" ); break; case SIGSYS: // system call given invalid argument fprintf( stderr, "SIGSYS: system call given invalid argument" ); break; case SIGPIPE: // write on a pipe with no reader fprintf( stderr, "SIGPIPE: write on a pipe with no reader" ); break; default: fprintf( stderr, "unknown signal %d", signal ); break; } fprintf( stderr, "\nExiting...\n" ); #endif exit(ERR_SIGNAL_CATCH); } int boinc_finish(int status) { write_checkpoint_cpu_file(boinc_cpu_time()); exit(status); return 0; } int boinc_get_init_data(APP_INIT_DATA& app_init_data) { app_init_data = aid; return 0; } // resolve XML soft links // int boinc_resolve_filename(char *virtual_name, char *physical_name, int len) { FILE *fp; char buf[512]; strcpy(physical_name, virtual_name); // Open the file and load the first line fp = fopen(virtual_name, "r"); if (!fp) return 0; fgets(buf, 512, fp); fclose(fp); // If it's the XML tag, return its value, // otherwise, return the original file name // parse_str(buf, "", physical_name, len); return 0; } bool boinc_time_to_checkpoint() { // Tell the graphics thread it's OK to draw now #ifdef BOINC_APP_GRAPHICS if (ready_to_redraw) { ok_to_draw = 1; // And wait for the graphics thread to notify us that it's done drawing #ifdef _WIN32 ResetEvent(hGlobalDrawEvent); while (ok_to_draw) { // Wait for drawing to finish. We don't do an infinite wait here to avoid the // possibility of deadlock (which was happening with an infinite wait value) WaitForSingleObject( hGlobalDrawEvent, 10 ); } #endif _WIN32 #ifdef __APPLE_CC__ while (ok_to_draw) { MPWaitOnQueue( drawQueue, NULL, NULL, NULL, kDurationImmediate ); YieldToAnyThread(); } #endif // __APPLE_CC__ // Reset the refresh counter time_until_redraw = gi.refresh_period; ready_to_redraw = false; } #endif // BOINC_APP_GRAPHICS return ready_to_checkpoint; } int boinc_checkpoint_completed() { write_checkpoint_cpu_file(boinc_cpu_time()); ready_to_checkpoint = false; time_until_checkpoint = aid.checkpoint_period; return 0; } int boinc_fraction_done(double x) { fraction_done = x; return 0; } int boinc_child_start() { this_process_active = false; return 0; } int boinc_child_done(double cpu) { this_process_active = true; return 0; } double boinc_cpu_time() { #ifdef HAVE_SYS_RESOURCE_H int retval, pid = getpid(); struct rusage ru; double cpu_secs; retval = getrusage(RUSAGE_SELF, &ru); if(retval) fprintf(stderr, "error: could not get cpu time for %d\n", pid); // Sum the user and system time spent in this process cpu_secs = (double)ru.ru_utime.tv_sec + (((double)ru.ru_utime.tv_usec) / ((double)1000000.0)); cpu_secs += (double)ru.ru_stime.tv_sec + (((double)ru.ru_stime.tv_usec) / ((double)1000000.0)); return cpu_secs; #else #ifdef _WIN32 #ifdef WINNT_CLOCK HANDLE hProcess; FILETIME creationTime,exitTime,kernelTime,userTime; // TODO: Could we speed this up by retaining the process handle? hProcess = OpenProcess(PROCESS_QUERY_INFORMATION, 0, GetCurrentProcessId()); if (GetProcessTimes( hProcess, &creationTime, &exitTime, &kernelTime, &userTime) ) { ULARGE_INTEGER tKernel, tUser; LONGLONG totTime; CloseHandle(hProcess); tKernel.LowPart = kernelTime.dwLowDateTime; tKernel.HighPart = kernelTime.dwHighDateTime; tUser.LowPart = userTime.dwLowDateTime; tUser.HighPart = userTime.dwHighDateTime; // Runtimes in 100-nanosecond units totTime = tKernel.QuadPart + tUser.QuadPart; // Convert to seconds and return return(totTime / 10000000.0); } CloseHandle(hProcess); // ... fall through #endif // WINNT_CLOCK static bool first=true; static DWORD last_count = 0; if (first) { last_count = GetTickCount(); first = true; } DWORD cur = GetTickCount(); double x = (cur - last_count)/1000.; last_count = cur; return x; #endif // _WIN32 #endif #ifdef macintosh return (double) GetCPUTime() / CLOCKS_PER_SEC; #endif fprintf(stderr, "boinc_cpu_time(): not implemented\n"); } #ifdef _WIN32 void CALLBACK on_timer(UINT uTimerID, UINT uMsg, DWORD dwUser, DWORD dw1, DWORD dw2) { #else void on_timer(int a) { #endif if (!ready_to_checkpoint) { time_until_checkpoint -= timer_period; if (time_until_checkpoint <= 0) { ready_to_checkpoint = true; } } if (!ready_to_redraw) { time_until_redraw -= timer_period; if (time_until_redraw <= 0) { ready_to_redraw = true; } } if (this_process_active) { time_until_fraction_done_update -= timer_period; if (time_until_fraction_done_update < 0) { FILE* f = fopen(FRACTION_DONE_FILE, "w"); write_fraction_done_file(f, fraction_done, boinc_cpu_time()); fclose(f); time_until_fraction_done_update = aid.fraction_done_update_period; } } #if defined __APPLE_CC__ && defined BOINC_APP_GRAPHICS // Yield to the graphics thread to let it draw if needed YieldToAnyThread(); #endif } int set_timer(double period) { int retval=0; #ifdef _WIN32 // Use Windows multimedia timer, since it is more accurate // than SetTimer and doesn't require an associated event loop retval = timeSetEvent( (int)(period*1000), // uDelay (int)(period*1000), // uResolution on_timer, // lpTimeProc NULL, // dwUser TIME_PERIODIC // fuEvent ); // Create the event object used to signal between the // worker and event threads hGlobalDrawEvent = CreateEvent( NULL, // no security attributes TRUE, // manual reset event TRUE, // initial state is signaled NULL); // object not named #endif #if defined BOINC_APP_GRAPHICS && defined __APPLE_CC__ // Create notification queue for drawing MPCreateQueue( &drawQueue ); #endif #if HAVE_SIGNAL_H #if HAVE_SYS_TIME_H struct sigaction sa; sa.sa_handler = on_timer; sa.sa_flags = 0; sigaction(SIGALRM, &sa, NULL); itimerval value; value.it_value.tv_sec = (int)period; value.it_value.tv_usec = ((int)(period*1000000))%1000000; value.it_interval = value.it_value; retval = setitimer(ITIMER_REAL, &value, NULL); #endif #endif return retval; } int write_init_data_file(FILE* f, APP_INIT_DATA& ai) { if (strlen(ai.app_preferences)) { fprintf(f, "\n%s\n", ai.app_preferences); } if (strlen(ai.team_name)) { fprintf(f, "\n%s\n", ai.team_name); } if (strlen(ai.user_name)) { fprintf(f, "\n%s\n", ai.user_name); } fprintf(f, "%f\n" "%f\n" "%f\n" "%f\n" "%f\n", ai.wu_cpu_time, ai.total_cobblestones, ai.recent_avg_cobblestones, ai.checkpoint_period, ai.fraction_done_update_period ); return 0; } int parse_init_data_file(FILE* f, APP_INIT_DATA& ai) { char buf[256]; memset(&ai, 0, sizeof(ai)); while (fgets(buf, 256, f)) { if (match_tag(buf, "")) { strcpy(ai.app_preferences, ""); while (fgets(buf, 256, f)) { if (match_tag(buf, "")) break; strcat(ai.app_preferences, buf); } continue; } else if (parse_str(buf, "", ai.user_name, sizeof(ai.user_name))) continue; else if (parse_str(buf, "", ai.team_name, sizeof(ai.team_name))) continue; else if (parse_double(buf, "", ai.total_cobblestones)) continue; else if (parse_double(buf, "", ai.recent_avg_cobblestones)) continue; else if (parse_double(buf, "", ai.wu_cpu_time)) continue; else if (parse_double(buf, "", ai.checkpoint_period)) continue; else if (parse_double(buf, "", ai.fraction_done_update_period)) continue; else fprintf(stderr, "parse_init_data_file: unrecognized %s", buf); } return 0; } int write_checkpoint_cpu_file(double cpu) { FILE *f = fopen(CHECKPOINT_CPU_FILE, "w"); if (!f) return ERR_FOPEN; fprintf(f, "%f\n", cpu); fclose(f); return 0; } int parse_checkpoint_cpu_file(FILE* f, double& checkpoint_cpu) { char buf[256]; while (fgets(buf, 256, f)) { if (parse_double(buf, "", checkpoint_cpu)) continue; } return 0; } int write_fraction_done_file(FILE* f, double pct, double cpu) { fprintf(f, "%f\n" "%f\n", pct, cpu ); return 0; } int parse_fraction_done_file(FILE* f, double& pct, double& cpu) { char buf[256]; while (fgets(buf, 256, f)) { if (parse_double(buf, "", pct)) continue; else if (parse_double(buf, "", cpu)) continue; else fprintf(stderr, "parse_fraction_done_file: unrecognized %s", buf); } return 0; } // TODO: this should handle arbitrarily many fd/filename pairs. // Also, give the tags better names int write_fd_init_file(FILE* f, char *file_name, int fdesc, int input_file ) { if (input_file) { fprintf(f, "%s\n", file_name); fprintf(f, "%d\n", fdesc); } else { fprintf(f, "%s\n", file_name); fprintf(f, "%d\n", fdesc); } return 0; } // TODO: this should handle arbitrarily many fd/filename pairs. // Also, this shouldn't be doing the actual duping! // int parse_fd_init_file(FILE* f) { char buf[256],filename[256]; int filedesc; while (fgets(buf, 256, f)) { if (parse_str(buf, "", filename, sizeof(filename))) { if (fgets(buf, 256, f)) { if (parse_int(buf, "", filedesc)) { freopen(filename, "r", stdin); fprintf(stderr, "opened input file %s\n", filename); } } } else if (parse_str(buf, "", filename, sizeof(filename))) { if (fgets(buf, 256, f)) { if (parse_int(buf, "", filedesc)) { freopen(filename, "w", stdout); fprintf(stderr, "opened output file %s\n", filename); } } } else fprintf(stderr, "parse_fd_init_file: unrecognized %s", buf); } return 0; }