// 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): // #include "windows_cpp.h" #ifdef _WIN32 #include #endif #if HAVE_SIGNAL_H #include #endif #include "md5_file.h" #include "log_flags.h" #include "file_names.h" #include "shmem.h" #include "client_state.h" // Make a directory for each of the available slots specified // in the client state // int CLIENT_STATE::make_slot_dirs() { int i; int retval; for (i=0; ioutput_files.size(); i++) { fip = rp->output_files[i].file_info; fip->status = FILE_PRESENT; if (!fip->upload_when_present && !fip->sticky) { fip->delete_file(); } else { get_pathname(fip, path); retval = md5_file(path, fip->md5_cksum, fip->nbytes); if (retval) { // an output file is unexpectedly absent. // fip->status = retval; } } } // Detach from shared memory. In Windows, this is the same as // destroying the shared mem since we're the last one attached // #ifdef _WIN32 if (at.app_client_shm.shm) { detach_shmem(at.shm_handle, at.app_client_shm.shm); } #else if (at.app_client_shm.shm) { detach_shmem(at.app_client_shm.shm); } destroy_shmem(at.shm_key); #endif at.result->is_active = false; at.result->state = RESULT_COMPUTE_DONE; update_avg_cpu(at.result->project); at.result->project->exp_avg_cpu += at.result->final_cpu_time; return 0; } // poll status of existing apps and and clean up after them // bool CLIENT_STATE::handle_running_apps() { unsigned int i; ACTIVE_TASK* atp; bool action = false; char buf[256]; for (i=0; istate != PROCESS_RUNNING) { sprintf(buf, "Computation for result %s finished", atp->wup->name); show_message(atp->wup->project, buf, MSG_INFO); if (log_flags.task_debug) { printf( "task finished; pid %d, status %d\n", atp->pid, atp->exit_status ); } app_finished(*atp); active_tasks.remove(atp); delete atp; set_client_state_dirty("handle_running_apps"); action = true; } } return action; } // Returns true if all the input files for a result are available // locally, false otherwise // bool CLIENT_STATE::input_files_available(RESULT* rp) { WORKUNIT* wup = rp->wup; FILE_INFO* fip; unsigned int i; APP_VERSION* avp; avp = wup->avp; for (i=0; iapp_files.size(); i++) { fip = avp->app_files[i].file_info; if (fip->status != FILE_PRESENT) return false; } for (i=0; iinput_files.size(); i++) { fip = wup->input_files[i].file_info; if (fip->status != FILE_PRESENT) return false; } return true; } // start new app if possible // bool CLIENT_STATE::start_apps() { unsigned int i; RESULT* rp; ACTIVE_TASK* atp; bool action = false; int open_slot, retval; char buf[256]; for (i=0; istate == RESULT_FILES_DOWNLOADED && !rp->is_active ) { if (log_flags.task) { sprintf(buf, "Starting computation for result %s", rp->name); show_message(rp->project, buf, MSG_INFO); } rp->is_active = true; atp = new ACTIVE_TASK; atp->slot = open_slot; atp->init(rp); retval = active_tasks.insert(atp); // couldn't start process // if (retval) { atp->state = PROCESS_COULDNT_START; atp->result->active_task_state = PROCESS_COULDNT_START; report_result_error( *(atp->result), retval, "Couldn't start the app for this result.\n" ); } action = true; set_client_state_dirty("start_apps"); app_started = time(0); } } return action; } // This is called when the client is initialized. // Try to restart any tasks that were running when we last shut down. // int CLIENT_STATE::restart_tasks() { return active_tasks.restart_tasks(); }