2008-08-18 22:09:22 +00:00
|
|
|
// 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.,
|
|
|
|
// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
//
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
#include "boinc_win.h"
|
|
|
|
|
|
|
|
#include <windowsx.h>
|
|
|
|
#include <mmsystem.h>
|
|
|
|
#include <regstr.h>
|
|
|
|
#include <strsafe.h>
|
|
|
|
|
|
|
|
#include "boinc_ss.h"
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "diagnostics.h"
|
|
|
|
#include "common_defs.h"
|
|
|
|
#include "util.h"
|
2009-02-05 12:32:16 +00:00
|
|
|
#include "common_defs.h"
|
|
|
|
#include "filesys.h"
|
|
|
|
|
2008-08-18 22:09:22 +00:00
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
#include "gui_rpc_client.h"
|
|
|
|
#include "screensaver_win.h"
|
2009-02-05 12:44:39 +00:00
|
|
|
#include "str_util.h"
|
2009-02-06 02:17:45 +00:00
|
|
|
#define PATH_SEPARATOR "\\"
|
|
|
|
#else
|
|
|
|
#define PATH_SEPARATOR "/"
|
2008-08-18 22:09:22 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef __APPLE__
|
|
|
|
#include <Carbon/Carbon.h>
|
|
|
|
#include <sys/wait.h>
|
|
|
|
#include "gui_rpc_client.h"
|
|
|
|
#include "Mac_Saver_Module.h"
|
|
|
|
#endif
|
|
|
|
#include "screensaver.h"
|
|
|
|
|
|
|
|
// Flags for testing & debugging
|
|
|
|
#define SIMULATE_NO_GRAPHICS 0
|
|
|
|
|
|
|
|
|
|
|
|
bool CScreensaver::is_same_task(RESULT* taska, RESULT* taskb) {
|
|
|
|
if ((taska == NULL) || (taskb == NULL)) return false;
|
|
|
|
if (taska->name != taskb->name) return false;
|
|
|
|
if (taska->project_url != taskb->project_url) return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CScreensaver::count_active_graphic_apps(RESULTS& results, RESULT* exclude) {
|
|
|
|
unsigned int i = 0;
|
|
|
|
unsigned int graphics_app_count = 0;
|
|
|
|
m_bV5_GFX_app_is_running = false;
|
|
|
|
|
|
|
|
// Count the number of active graphics-capable apps excluding the specified result.
|
|
|
|
// If exclude is NULL, don't exclude any results.
|
|
|
|
for (i = 0; i < results.results.size(); i++) {
|
|
|
|
BOINCTRACE(_T("get_random_graphics_app -- active task detected\n"));
|
|
|
|
BOINCTRACE(
|
|
|
|
_T("get_random_graphics_app -- name = '%s', path = '%s'\n"),
|
|
|
|
results.results[i]->name.c_str(), results.results[i]->graphics_exec_path.c_str()
|
|
|
|
);
|
|
|
|
if (results.results[i]->supports_graphics) m_bV5_GFX_app_is_running = true;
|
|
|
|
if ((results.results[i]->graphics_exec_path.size() == 0)
|
|
|
|
&& (state.executing_as_daemon || !(results.results[i]->supports_graphics))) continue;
|
|
|
|
BOINCTRACE(_T("get_random_graphics_app -- active task detected w/graphics\n"));
|
2009-02-06 11:17:13 +00:00
|
|
|
|
2008-08-18 22:09:22 +00:00
|
|
|
if (is_same_task(results.results[i], exclude)) continue;
|
|
|
|
graphics_app_count++;
|
|
|
|
}
|
|
|
|
return graphics_app_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Choose a random graphics application out of the vector.
|
|
|
|
// Exclude the specified result unless it is the only candidate.
|
|
|
|
// If exclude is NULL or an empty string, don't exclude any results.
|
|
|
|
//
|
|
|
|
RESULT* CScreensaver::get_random_graphics_app(RESULTS& results, RESULT* exclude) {
|
|
|
|
RESULT* rp = NULL;
|
|
|
|
unsigned int i = 0;
|
|
|
|
unsigned int graphics_app_count = 0;
|
|
|
|
unsigned int random_selection = 0;
|
|
|
|
unsigned int current_counter = 0;
|
|
|
|
RESULT *avoid = exclude;
|
|
|
|
|
|
|
|
BOINCTRACE(_T("get_random_graphics_app -- Function Start\n"));
|
|
|
|
|
|
|
|
graphics_app_count = count_active_graphic_apps(results, avoid);
|
|
|
|
BOINCTRACE(_T("get_random_graphics_app -- graphics_app_count = '%d'\n"), graphics_app_count);
|
|
|
|
|
|
|
|
// If no graphics app found other than the one excluded, count again without excluding any
|
|
|
|
if ((0 == graphics_app_count) && (avoid != NULL)) {
|
|
|
|
avoid = NULL;
|
|
|
|
graphics_app_count = count_active_graphic_apps(results, avoid);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If no graphics app was found, return NULL
|
|
|
|
if (0 == graphics_app_count) {
|
|
|
|
goto CLEANUP;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Choose which application to display.
|
|
|
|
random_selection = (rand() % graphics_app_count) + 1;
|
|
|
|
BOINCTRACE(_T("get_random_graphics_app -- random_selection = '%d'\n"), random_selection);
|
|
|
|
|
|
|
|
// Lets find the chosen graphics application.
|
|
|
|
for (i = 0; i < results.results.size(); i++) {
|
|
|
|
if ((results.results[i]->graphics_exec_path.size() == 0)
|
|
|
|
&& (state.executing_as_daemon || !(results.results[i]->supports_graphics))) continue;
|
|
|
|
if (is_same_task(results.results[i], avoid)) continue;
|
|
|
|
|
|
|
|
current_counter++;
|
|
|
|
if (current_counter == random_selection) {
|
|
|
|
rp = results.results[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CLEANUP:
|
|
|
|
BOINCTRACE(_T("get_random_graphics_app -- Function End\n"));
|
|
|
|
|
|
|
|
return rp;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Launch the graphics application
|
|
|
|
//
|
|
|
|
#ifdef _WIN32
|
|
|
|
int CScreensaver::launch_screensaver(RESULT* rp, HANDLE& graphics_application)
|
|
|
|
#else
|
|
|
|
int CScreensaver::launch_screensaver(RESULT* rp, int& graphics_application)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
int retval = 0;
|
|
|
|
if (!rp->graphics_exec_path.empty()) {
|
|
|
|
// V6 Graphics
|
|
|
|
#ifdef __APPLE__
|
|
|
|
// For sandbox security, use gfx_switcher to launch gfx app
|
|
|
|
// as user boinc_project and group boinc_project.
|
|
|
|
//
|
|
|
|
// For unknown reasons, the graphics application exits with
|
|
|
|
// "RegisterProcess failed (error = -50)" unless we pass its
|
|
|
|
// full path twice in the argument list to execv.
|
|
|
|
char* argv[5];
|
|
|
|
argv[0] = "gfx_Switcher";
|
|
|
|
argv[1] = "-launch_gfx";
|
|
|
|
argv[2] = strrchr(rp->slot_path.c_str(), '/');
|
|
|
|
if (*argv[2]) argv[2]++; // Point to the slot number in ascii
|
|
|
|
|
|
|
|
argv[3] = "--fullscreen";
|
|
|
|
argv[4] = 0;
|
|
|
|
|
|
|
|
retval = run_program(
|
|
|
|
rp->slot_path.c_str(),
|
|
|
|
m_gfx_Switcher_Path,
|
|
|
|
4,
|
|
|
|
argv,
|
|
|
|
0,
|
|
|
|
graphics_application
|
|
|
|
);
|
|
|
|
#else
|
|
|
|
char* argv[3];
|
|
|
|
argv[0] = "app_graphics"; // not used
|
|
|
|
argv[1] = "--fullscreen";
|
|
|
|
argv[2] = 0;
|
|
|
|
retval = run_program(
|
|
|
|
rp->slot_path.c_str(),
|
|
|
|
rp->graphics_exec_path.c_str(),
|
|
|
|
2,
|
|
|
|
argv,
|
|
|
|
0,
|
|
|
|
graphics_application
|
|
|
|
);
|
|
|
|
#endif
|
|
|
|
} else {
|
|
|
|
// V5 and Older
|
|
|
|
DISPLAY_INFO di;
|
|
|
|
#ifdef _WIN32
|
|
|
|
graphics_application = NULL;
|
|
|
|
|
|
|
|
memset(di.window_station, 0, sizeof(di.window_station));
|
|
|
|
memset(di.desktop, 0, sizeof(di.desktop));
|
|
|
|
memset(di.display, 0, sizeof(di.display));
|
|
|
|
|
|
|
|
if (!m_bIs9x) {
|
|
|
|
// Retrieve the current window station and desktop names
|
|
|
|
GetUserObjectInformation(
|
|
|
|
GetProcessWindowStation(),
|
|
|
|
UOI_NAME,
|
|
|
|
di.window_station,
|
|
|
|
(sizeof(di.window_station)),
|
|
|
|
NULL
|
|
|
|
);
|
|
|
|
GetUserObjectInformation(
|
|
|
|
GetThreadDesktop(GetCurrentThreadId()),
|
|
|
|
UOI_NAME,
|
|
|
|
di.desktop,
|
|
|
|
sizeof(di.desktop),
|
|
|
|
NULL
|
|
|
|
);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
char *p = getenv("DISPLAY");
|
|
|
|
if (p) strcpy(di.display, p);
|
|
|
|
|
|
|
|
graphics_application = 0;
|
|
|
|
#endif
|
|
|
|
retval = rpc->show_graphics(
|
|
|
|
rp->project_url.c_str(),
|
|
|
|
rp->name.c_str(),
|
|
|
|
MODE_FULLSCREEN,
|
|
|
|
di
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Terminate the graphics application
|
|
|
|
//
|
|
|
|
#ifdef _WIN32
|
|
|
|
int CScreensaver::terminate_screensaver(HANDLE& graphics_application, RESULT *worker_app)
|
|
|
|
#else
|
|
|
|
int CScreensaver::terminate_screensaver(int& graphics_application, RESULT *worker_app)
|
|
|
|
#endif
|
|
|
|
{
|
2009-02-10 13:45:43 +00:00
|
|
|
if (!m_bScience_gfx_running) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-08-18 22:09:22 +00:00
|
|
|
if (graphics_application) {
|
|
|
|
// V6 Graphics
|
|
|
|
kill_program(graphics_application);
|
|
|
|
#ifdef __APPLE__
|
|
|
|
// For sandbox security, use gfx_switcher to launch gfx app
|
|
|
|
// as user boinc_project and group boinc_project.
|
2009-02-10 12:35:12 +00:00
|
|
|
int retval = 0;
|
|
|
|
char current_dir[MAXPATHLEN];
|
|
|
|
char gfx_pid[16];
|
|
|
|
pid_t thePID;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
sprintf(gfx_pid, "%d", graphics_application);
|
|
|
|
getcwd( current_dir, sizeof(current_dir));
|
|
|
|
|
|
|
|
char* argv[4];
|
|
|
|
argv[0] = "gfx_switcher";
|
|
|
|
argv[1] = "-kill_gfx";
|
|
|
|
argv[2] = gfx_pid;
|
|
|
|
argv[3] = 0;
|
|
|
|
|
|
|
|
retval = run_program(
|
|
|
|
current_dir,
|
|
|
|
m_gfx_Switcher_Path,
|
|
|
|
3,
|
|
|
|
argv,
|
|
|
|
0,
|
|
|
|
thePID
|
|
|
|
);
|
|
|
|
if (retval) return retval;
|
|
|
|
|
|
|
|
for (i=0; i<200; i++) {
|
|
|
|
boinc_sleep(0.01); // Wait 2 seconds max
|
|
|
|
// Prevent gfx_switcher from becoming a zombie
|
|
|
|
if (waitpid(thePID, 0, WNOHANG) == thePID) break;
|
|
|
|
}
|
|
|
|
return retval;
|
2008-08-18 22:09:22 +00:00
|
|
|
#endif
|
2009-02-10 13:45:43 +00:00
|
|
|
graphics_application = 0;
|
2008-08-18 22:09:22 +00:00
|
|
|
} else {
|
|
|
|
// V5 and Older
|
|
|
|
DISPLAY_INFO di;
|
|
|
|
|
|
|
|
if (worker_app == NULL) return 0;
|
|
|
|
if (worker_app->name.empty()) return 0;
|
|
|
|
|
|
|
|
memset(di.window_station, 0, sizeof(di.window_station));
|
|
|
|
memset(di.desktop, 0, sizeof(di.desktop));
|
|
|
|
memset(di.display, 0, sizeof(di.display));
|
|
|
|
|
|
|
|
rpc->show_graphics(
|
|
|
|
worker_app->project_url.c_str(),
|
|
|
|
worker_app->name.c_str(),
|
|
|
|
MODE_HIDE_GRAPHICS,
|
|
|
|
di
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-02-05 12:32:16 +00:00
|
|
|
// Launch the graphics application
|
|
|
|
//
|
|
|
|
#ifdef _WIN32
|
|
|
|
int CScreensaver::launch_default_screensaver(char *dir_path, HANDLE& graphics_application)
|
|
|
|
#else
|
|
|
|
int CScreensaver::launch_default_screensaver(char *dir_path, int& graphics_application)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
int retval = 0;
|
|
|
|
char full_path[1024];
|
|
|
|
|
|
|
|
strlcpy(full_path, dir_path, sizeof(full_path));
|
2009-02-06 02:17:45 +00:00
|
|
|
strlcat(full_path, PATH_SEPARATOR, sizeof(full_path));
|
2009-02-05 12:32:16 +00:00
|
|
|
strlcat(full_path, DEFAULT_SS_EXECUTABLE, sizeof(full_path));
|
|
|
|
|
|
|
|
// For unknown reasons, the graphics application exits with
|
|
|
|
// "RegisterProcess failed (error = -50)" unless we pass its
|
|
|
|
// full path twice in the argument list to execv on Macs.
|
|
|
|
|
|
|
|
char* argv[3];
|
|
|
|
argv[0] = full_path; // not used
|
|
|
|
argv[1] = "--fullscreen";
|
|
|
|
argv[2] = 0;
|
|
|
|
retval = run_program(
|
|
|
|
dir_path,
|
|
|
|
full_path,
|
|
|
|
2,
|
|
|
|
argv,
|
|
|
|
0,
|
|
|
|
graphics_application
|
|
|
|
);
|
|
|
|
|
|
|
|
BOINCTRACE(_T("launch_default_screensaver %s returned %d\n"), full_path, retval);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-08-18 22:09:22 +00:00
|
|
|
#ifdef _WIN32
|
2009-02-10 12:35:12 +00:00
|
|
|
DWORD WINAPI CScreensaver::DataManagementProc()
|
2008-08-18 22:09:22 +00:00
|
|
|
#else
|
2009-02-10 12:35:12 +00:00
|
|
|
void *CScreensaver::DataManagementProc()
|
2008-08-18 22:09:22 +00:00
|
|
|
#endif
|
2009-02-10 12:35:12 +00:00
|
|
|
{
|
2009-02-06 11:17:13 +00:00
|
|
|
int retval = 0;
|
|
|
|
int suspend_reason = 0;
|
|
|
|
RESULT* theResult = NULL;
|
|
|
|
RESULT* graphics_app_result_ptr = NULL;
|
2008-08-18 22:09:22 +00:00
|
|
|
RESULT previous_result;
|
2008-08-18 23:39:50 +00:00
|
|
|
// previous_result_ptr = &previous_result when previous_result is valid, else NULL
|
2009-02-06 11:17:13 +00:00
|
|
|
RESULT* previous_result_ptr = NULL;
|
|
|
|
int iResultCount = 0;
|
|
|
|
int iIndex = 0;
|
|
|
|
double default_phase_start_time = 0.0;
|
|
|
|
double science_phase_start_time = 0.0;
|
|
|
|
double last_change_time = 0.0;
|
|
|
|
// If we run default screensaver during science phase because no science graphics
|
|
|
|
// are available, then shorten next default graphics phase by that much time.
|
2009-02-12 11:54:57 +00:00
|
|
|
double default_saver_start_time_in_science_phase = 0.0;
|
|
|
|
double default_saver_duration_in_science_phase = 0.0;
|
2009-02-06 11:17:13 +00:00
|
|
|
|
|
|
|
SS_PHASE ss_phase = DEFAULT_SS_PHASE;
|
|
|
|
bool switch_to_default_gfx = false;
|
2009-02-05 12:32:16 +00:00
|
|
|
char full_path[1024];
|
2008-08-18 22:09:22 +00:00
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
BOINCTRACE(_T("CScreensaver::DataManagementProc - Display screen saver loading message\n"));
|
|
|
|
SetError(TRUE, SCRAPPERR_BOINCSCREENSAVERLOADING);
|
|
|
|
m_tThreadCreateTime = time(0);
|
|
|
|
|
|
|
|
// Set the starting point for iterating through the results
|
|
|
|
m_iLastResultShown = 0;
|
|
|
|
m_tLastResultChangeTime = 0;
|
2009-02-05 12:32:16 +00:00
|
|
|
#endif
|
|
|
|
|
2009-02-06 02:17:45 +00:00
|
|
|
m_bDefault_ss_exists = false;
|
2009-02-10 12:35:12 +00:00
|
|
|
m_bScience_gfx_running = false;
|
|
|
|
m_bDefault_gfx_running = false;
|
2009-02-06 11:17:13 +00:00
|
|
|
m_fGFXDefaultPeriod = GFX_DEFAULT_PERIOD;
|
|
|
|
m_fGFxSciencePeriod = GFX_SCIENCE_PERIOD;
|
|
|
|
m_fGFXChangePeriod = GFX_CHANGE_PERIOD;
|
2009-02-06 02:17:45 +00:00
|
|
|
|
2009-02-05 12:32:16 +00:00
|
|
|
#ifdef __APPLE__
|
|
|
|
char * default_ss_dir_path = "/Library/Application Support/BOINC Data";
|
|
|
|
#else
|
|
|
|
// TODO: Obtain correct path to Windows default OpenGL screensaver executable
|
2009-02-05 12:44:39 +00:00
|
|
|
char * default_ss_dir_path = "C:\\Program Files\\BOINC";
|
2009-02-05 12:32:16 +00:00
|
|
|
#endif
|
|
|
|
strlcpy(full_path, default_ss_dir_path, sizeof(full_path));
|
2009-02-06 02:17:45 +00:00
|
|
|
strlcat(full_path, PATH_SEPARATOR, sizeof(full_path));
|
2009-02-05 12:32:16 +00:00
|
|
|
strlcat(full_path, DEFAULT_SS_EXECUTABLE, sizeof(full_path));
|
|
|
|
|
|
|
|
if (boinc_file_exists(full_path)) {
|
2009-02-06 02:17:45 +00:00
|
|
|
m_bDefault_ss_exists = true;
|
2009-02-05 12:32:16 +00:00
|
|
|
ss_phase = DEFAULT_SS_PHASE;
|
|
|
|
default_phase_start_time = dtime();
|
|
|
|
science_phase_start_time = 0;
|
|
|
|
switch_to_default_gfx = true;
|
|
|
|
} else {
|
|
|
|
ss_phase = SCIENCE_SS_PHASE;
|
|
|
|
default_phase_start_time = 0;
|
|
|
|
science_phase_start_time = dtime();
|
|
|
|
}
|
|
|
|
|
2009-02-06 11:17:13 +00:00
|
|
|
GetDisplayPeriods(default_ss_dir_path);
|
2008-08-18 22:09:22 +00:00
|
|
|
|
|
|
|
while (true) {
|
|
|
|
for (int i = 0; i < 4; i++) {
|
2008-08-29 19:06:04 +00:00
|
|
|
// ***
|
|
|
|
// *** Things that should be run frequently.
|
|
|
|
// *** 4 times per second.
|
|
|
|
// ***
|
2008-08-18 22:09:22 +00:00
|
|
|
|
2008-08-29 19:06:04 +00:00
|
|
|
// Are we supposed to exit the screensaver?
|
2008-08-18 22:09:22 +00:00
|
|
|
if (m_QuitDataManagementProc) { // If main thread has requested we exit
|
|
|
|
if (m_hGraphicsApplication || graphics_app_result_ptr) {
|
|
|
|
terminate_screensaver(m_hGraphicsApplication, graphics_app_result_ptr);
|
|
|
|
graphics_app_result_ptr = NULL;
|
|
|
|
previous_result_ptr = NULL;
|
|
|
|
m_hGraphicsApplication = 0;
|
|
|
|
}
|
|
|
|
return 0; // Exit the thread
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
|
|
|
|
// Check for keyboard and mouse activity just in case the
|
|
|
|
// user wants to blow out of the screensaver.
|
|
|
|
//
|
|
|
|
CheckKeyboardMouseActivity();
|
|
|
|
|
|
|
|
// Check to see if there are any notification windows from
|
|
|
|
// personal firewalls, virus scanners, or anything else that
|
|
|
|
// demands the users attention. If there is blow out of the
|
|
|
|
// screensaver
|
|
|
|
CheckForNotificationWindow();
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
boinc_sleep(0.25);
|
|
|
|
}
|
|
|
|
|
|
|
|
// ***
|
|
|
|
// *** Things that should be run frequently.
|
|
|
|
// *** 1 time per second.
|
|
|
|
// ***
|
|
|
|
|
|
|
|
// Blank screen saver?
|
|
|
|
if ((m_dwBlankScreen) && (time(0) > m_dwBlankTime)) {
|
|
|
|
BOINCTRACE(_T("CScreensaver::DataManagementProc - Time to blank\n"));
|
|
|
|
SetError(FALSE, SCRAPPERR_SCREENSAVERBLANKED);
|
|
|
|
m_QuitDataManagementProc = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do we need to get the core client state?
|
|
|
|
if (m_bResetCoreState) {
|
|
|
|
// Try and get the current state of the CC
|
|
|
|
retval = rpc->get_state(state);
|
|
|
|
if (retval) {
|
|
|
|
// CC may not yet be running
|
|
|
|
HandleRPCError();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_bResetCoreState = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOINCTRACE(_T("CScreensaver::DataManagementProc - ErrorMode = '%d', ErrorCode = '%x'\n"), m_bErrorMode, m_hrError);
|
|
|
|
|
|
|
|
// Update our task list
|
|
|
|
m_updating_results = true;
|
|
|
|
retval = rpc->get_screensaver_tasks(suspend_reason, results);
|
|
|
|
m_updating_results = false;
|
|
|
|
if (retval) {
|
|
|
|
// rpc call returned error
|
|
|
|
HandleRPCError();
|
|
|
|
m_bResetCoreState = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2009-02-06 11:17:13 +00:00
|
|
|
// Time to switch to default graphics phase?
|
2009-02-06 02:17:45 +00:00
|
|
|
if (m_bDefault_ss_exists && (ss_phase == SCIENCE_SS_PHASE)) {
|
2009-02-06 11:17:13 +00:00
|
|
|
if (science_phase_start_time && ((dtime() - science_phase_start_time) > m_fGFxSciencePeriod)) {
|
2009-02-05 12:32:16 +00:00
|
|
|
switch_to_default_gfx = true;
|
|
|
|
ss_phase = DEFAULT_SS_PHASE;
|
|
|
|
default_phase_start_time = dtime();
|
|
|
|
science_phase_start_time = 0;
|
2009-02-12 11:54:57 +00:00
|
|
|
if (m_bDefault_gfx_running && default_saver_start_time_in_science_phase) {
|
|
|
|
// Remember how long default graphics ran during science phase
|
|
|
|
default_saver_duration_in_science_phase += (dtime() - default_saver_start_time_in_science_phase);
|
|
|
|
}
|
|
|
|
default_saver_start_time_in_science_phase = 0;
|
2009-02-05 12:32:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (switch_to_default_gfx) {
|
2009-02-10 12:35:12 +00:00
|
|
|
if (m_bScience_gfx_running) {
|
2009-02-05 12:32:16 +00:00
|
|
|
if (m_hGraphicsApplication || previous_result_ptr) {
|
|
|
|
// use previous_result_ptr because graphics_app_result_ptr may no longer be valid
|
|
|
|
terminate_screensaver(m_hGraphicsApplication, previous_result_ptr);
|
|
|
|
if (m_hGraphicsApplication == 0) {
|
|
|
|
graphics_app_result_ptr = NULL;
|
2009-02-10 12:35:12 +00:00
|
|
|
m_bScience_gfx_running = false;
|
2009-02-05 12:32:16 +00:00
|
|
|
} else {
|
|
|
|
// waitpid test will clear m_hGraphicsApplication and graphics_app_result_ptr
|
|
|
|
}
|
|
|
|
previous_result_ptr = NULL;
|
|
|
|
}
|
|
|
|
} else {
|
2009-02-12 11:54:57 +00:00
|
|
|
if (!m_bDefault_gfx_running) {
|
2009-02-05 12:32:16 +00:00
|
|
|
switch_to_default_gfx = false;
|
|
|
|
retval = launch_default_screensaver(default_ss_dir_path, m_hGraphicsApplication);
|
|
|
|
if (retval) {
|
|
|
|
m_hGraphicsApplication = 0;
|
|
|
|
previous_result_ptr = NULL;
|
|
|
|
graphics_app_result_ptr = NULL;
|
2009-02-10 12:35:12 +00:00
|
|
|
m_bDefault_gfx_running = false;
|
2009-02-05 12:32:16 +00:00
|
|
|
} else {
|
2009-02-10 12:35:12 +00:00
|
|
|
m_bDefault_gfx_running = true;
|
2009-02-06 11:17:13 +00:00
|
|
|
if (ss_phase == SCIENCE_SS_PHASE) {
|
2009-02-12 11:54:57 +00:00
|
|
|
default_saver_start_time_in_science_phase = dtime();
|
2009-02-06 11:17:13 +00:00
|
|
|
}
|
2009-02-05 12:32:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-08-18 22:09:22 +00:00
|
|
|
// Core client suspended?
|
|
|
|
if (suspend_reason && !(suspend_reason & SUSPEND_REASON_CPU_USAGE_LIMIT)) {
|
|
|
|
SetError(TRUE, SCRAPPERR_BOINCSUSPENDED);
|
2009-02-10 12:35:12 +00:00
|
|
|
if (m_bDefault_ss_exists && !m_bDefault_gfx_running) {
|
2009-02-05 12:32:16 +00:00
|
|
|
switch_to_default_gfx = true;
|
|
|
|
}
|
2009-02-06 11:17:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Time to switch to science graphics phase?
|
|
|
|
if (ss_phase == DEFAULT_SS_PHASE) {
|
|
|
|
if (default_phase_start_time &&
|
2009-02-12 11:54:57 +00:00
|
|
|
((dtime() - default_phase_start_time + default_saver_duration_in_science_phase)
|
2009-02-06 11:17:13 +00:00
|
|
|
> m_fGFXDefaultPeriod)) {
|
2009-02-12 11:54:57 +00:00
|
|
|
// BOINCTRACE(_T("CScreensaver::Ending Default phase: now=%f, default_phase_start_time=%f, default_saver_duration_in_science_phase=%f\n"),
|
|
|
|
// dtime(), default_phase_start_time, default_saver_duration_in_science_phase);
|
2009-02-06 11:17:13 +00:00
|
|
|
ss_phase = SCIENCE_SS_PHASE;
|
|
|
|
default_phase_start_time = 0;
|
2009-02-12 11:54:57 +00:00
|
|
|
default_saver_duration_in_science_phase = 0;
|
2009-02-06 11:17:13 +00:00
|
|
|
science_phase_start_time = dtime();
|
2009-02-10 12:35:12 +00:00
|
|
|
if (m_bDefault_gfx_running) {
|
2009-02-12 11:54:57 +00:00
|
|
|
default_saver_start_time_in_science_phase = science_phase_start_time;
|
2008-08-18 22:09:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-02-05 12:32:16 +00:00
|
|
|
if (ss_phase == SCIENCE_SS_PHASE) {
|
|
|
|
|
2008-08-18 22:09:22 +00:00
|
|
|
#if SIMULATE_NO_GRAPHICS /* FOR TESTING */
|
|
|
|
|
|
|
|
SetError(TRUE, SCRAPPERR_BOINCNOGRAPHICSAPPSEXECUTING);
|
2009-02-10 12:35:12 +00:00
|
|
|
if (m_bDefault_ss_exists && !m_bDefault_gfx_running) {
|
2009-02-05 12:32:16 +00:00
|
|
|
switch_to_default_gfx = true;
|
|
|
|
}
|
2008-08-18 22:09:22 +00:00
|
|
|
|
|
|
|
#else /* NORMAL OPERATION */
|
|
|
|
|
2009-02-10 12:35:12 +00:00
|
|
|
if (m_bScience_gfx_running) {
|
2009-02-05 12:32:16 +00:00
|
|
|
// Is the current graphics app's associated task still running?
|
|
|
|
|
|
|
|
if ((m_hGraphicsApplication) || (graphics_app_result_ptr)) {
|
|
|
|
iResultCount = results.results.size();
|
|
|
|
graphics_app_result_ptr = NULL;
|
2008-08-18 22:09:22 +00:00
|
|
|
|
2009-02-05 12:32:16 +00:00
|
|
|
// Find the current task in the new results vector (if it still exists)
|
|
|
|
for (iIndex = 0; iIndex < iResultCount; iIndex++) {
|
|
|
|
theResult = results.results.at(iIndex);
|
2008-08-18 22:09:22 +00:00
|
|
|
|
2009-02-05 12:32:16 +00:00
|
|
|
if (is_same_task(theResult, previous_result_ptr)) {
|
|
|
|
graphics_app_result_ptr = theResult;
|
|
|
|
previous_result = *theResult;
|
|
|
|
previous_result_ptr = &previous_result;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2008-08-18 22:09:22 +00:00
|
|
|
|
2009-02-05 12:32:16 +00:00
|
|
|
// V6 graphics only: if worker application has stopped running, terminate_screensaver
|
|
|
|
if ((graphics_app_result_ptr == NULL) && (m_hGraphicsApplication != 0)) {
|
|
|
|
if (previous_result_ptr) {
|
|
|
|
BOINCTRACE(_T("CScreensaver::DataManagementProc - %s finished\n"),
|
|
|
|
previous_result.graphics_exec_path.c_str());
|
|
|
|
}
|
|
|
|
terminate_screensaver(m_hGraphicsApplication, previous_result_ptr);
|
|
|
|
previous_result_ptr = NULL;
|
2009-02-10 12:35:12 +00:00
|
|
|
if (m_hGraphicsApplication == 0) {
|
|
|
|
graphics_app_result_ptr = NULL;
|
|
|
|
m_bScience_gfx_running = false;
|
|
|
|
// Save previous_result and previous_result_ptr for get_random_graphics_app() call
|
|
|
|
} else {
|
|
|
|
// waitpid test will clear m_hGraphicsApplication and graphics_app_result_ptr
|
|
|
|
}
|
2009-02-05 12:32:16 +00:00
|
|
|
}
|
2008-08-18 22:09:22 +00:00
|
|
|
|
2009-02-06 11:17:13 +00:00
|
|
|
if (last_change_time && ((dtime() - last_change_time) > m_fGFXChangePeriod)) {
|
2009-02-05 12:32:16 +00:00
|
|
|
if (count_active_graphic_apps(results, previous_result_ptr) > 0) {
|
|
|
|
if (previous_result_ptr) {
|
|
|
|
BOINCTRACE(_T("CScreensaver::DataManagementProc - time to change: %s / %s\n"),
|
|
|
|
previous_result.name.c_str(), previous_result.graphics_exec_path.c_str());
|
|
|
|
}
|
|
|
|
terminate_screensaver(m_hGraphicsApplication, graphics_app_result_ptr);
|
|
|
|
if (m_hGraphicsApplication == 0) {
|
|
|
|
graphics_app_result_ptr = NULL;
|
2009-02-10 12:35:12 +00:00
|
|
|
m_bScience_gfx_running = false;
|
2009-02-05 12:32:16 +00:00
|
|
|
// Save previous_result and previous_result_ptr for get_random_graphics_app() call
|
|
|
|
} else {
|
|
|
|
// waitpid test will clear m_hGraphicsApplication and graphics_app_result_ptr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
last_change_time = dtime();
|
|
|
|
}
|
2008-08-18 22:09:22 +00:00
|
|
|
}
|
2009-02-10 12:35:12 +00:00
|
|
|
} // End if (m_bScience_gfx_running)
|
2009-02-05 12:32:16 +00:00
|
|
|
|
|
|
|
// If no current graphics app, pick an active task at random and launch its graphics app
|
2009-02-10 12:35:12 +00:00
|
|
|
if ((m_bDefault_gfx_running || (m_hGraphicsApplication == 0)) && (graphics_app_result_ptr == NULL)) {
|
2009-02-05 12:32:16 +00:00
|
|
|
graphics_app_result_ptr = get_random_graphics_app(results, previous_result_ptr);
|
2008-08-18 22:09:22 +00:00
|
|
|
previous_result_ptr = NULL;
|
2009-02-05 12:32:16 +00:00
|
|
|
|
|
|
|
if (graphics_app_result_ptr) {
|
2009-02-10 12:35:12 +00:00
|
|
|
if (m_bDefault_gfx_running) {
|
2009-02-05 12:32:16 +00:00
|
|
|
kill_program(m_hGraphicsApplication);
|
2009-02-12 11:54:57 +00:00
|
|
|
// Remember how long default graphics ran during science phase
|
|
|
|
if (default_saver_start_time_in_science_phase) {
|
|
|
|
default_saver_duration_in_science_phase += (dtime() - default_saver_start_time_in_science_phase);
|
|
|
|
//BOINCTRACE(_T("CScreensaver::During Science phase: now=%f, default_saver_start_time=%f, default_saver_duration=%f\n"),
|
|
|
|
// dtime(), default_saver_start_time_in_science_phase, default_saver_duration_in_science_phase);
|
|
|
|
}
|
|
|
|
default_saver_start_time_in_science_phase = 0;
|
2009-02-05 12:32:16 +00:00
|
|
|
// waitpid test will clear m_hGraphicsApplication and graphics_app_result_ptr
|
|
|
|
} else {
|
|
|
|
retval = launch_screensaver(graphics_app_result_ptr, m_hGraphicsApplication);
|
|
|
|
if (retval) {
|
|
|
|
m_hGraphicsApplication = 0;
|
|
|
|
previous_result_ptr = NULL;
|
|
|
|
graphics_app_result_ptr = NULL;
|
2009-02-10 12:35:12 +00:00
|
|
|
m_bScience_gfx_running = false;
|
2009-02-05 12:32:16 +00:00
|
|
|
} else {
|
|
|
|
#ifdef __APPLE__
|
|
|
|
// Show ScreenSaverAppStartingMsg for GFX_STARTING_MSG_DURATION seconds
|
|
|
|
SetError(FALSE, SCRAPPERR_BOINCAPPFOUNDGRAPHICSLOADING);
|
|
|
|
#endif
|
|
|
|
SetError(FALSE, SCRAPPERR_SCREENSAVERRUNNING);
|
|
|
|
last_change_time = dtime();
|
2009-02-10 12:35:12 +00:00
|
|
|
m_bScience_gfx_running = true;
|
2009-02-05 12:32:16 +00:00
|
|
|
// Make a local copy of current result, since original pointer
|
|
|
|
// may have been freed by the time we perform later tests
|
|
|
|
previous_result = *graphics_app_result_ptr;
|
|
|
|
previous_result_ptr = &previous_result;
|
|
|
|
if (previous_result_ptr) {
|
|
|
|
BOINCTRACE(_T("CScreensaver::DataManagementProc - launching %s\n"),
|
|
|
|
previous_result.graphics_exec_path.c_str());
|
|
|
|
}
|
|
|
|
}
|
2008-08-18 23:39:50 +00:00
|
|
|
}
|
2009-02-05 12:32:16 +00:00
|
|
|
} else {
|
2009-02-10 12:35:12 +00:00
|
|
|
if ((!suspend_reason) || (suspend_reason & SUSPEND_REASON_CPU_USAGE_LIMIT)) {
|
|
|
|
// No science graphics available
|
|
|
|
if (state.projects.size() == 0) {
|
|
|
|
// We are not attached to any projects
|
|
|
|
SetError(TRUE, SCRAPPERR_BOINCNOPROJECTSDETECTED);
|
|
|
|
} else if (results.results.size() == 0) {
|
|
|
|
// We currently do not have any applications to run
|
|
|
|
SetError(TRUE, SCRAPPERR_BOINCNOAPPSEXECUTING);
|
2009-02-05 12:32:16 +00:00
|
|
|
} else {
|
2009-02-10 12:35:12 +00:00
|
|
|
// We currently do not have any graphics capable application
|
|
|
|
if (m_bV5_GFX_app_is_running) {
|
|
|
|
SetError(TRUE, SCRAPPERR_DAEMONALLOWSNOGRAPHICS);
|
|
|
|
} else {
|
|
|
|
SetError(TRUE, SCRAPPERR_BOINCNOGRAPHICSAPPSEXECUTING);
|
|
|
|
}
|
2009-02-05 12:32:16 +00:00
|
|
|
}
|
2008-08-18 22:09:22 +00:00
|
|
|
}
|
2009-02-05 12:32:16 +00:00
|
|
|
|
|
|
|
// We can't run a science graphics app, so run the default graphics if available
|
2009-02-10 12:35:12 +00:00
|
|
|
if (m_bDefault_ss_exists && !m_bDefault_gfx_running) {
|
2009-02-05 12:32:16 +00:00
|
|
|
switch_to_default_gfx = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // End if no science graphics available
|
|
|
|
} // End if no current science graphics app is running
|
|
|
|
|
|
|
|
#endif // ! SIMULATE_NO_GRAPHICS
|
|
|
|
|
|
|
|
if (switch_to_default_gfx) {
|
|
|
|
switch_to_default_gfx = false;
|
2009-02-10 12:35:12 +00:00
|
|
|
if (m_bDefault_ss_exists && !m_bDefault_gfx_running) {
|
2009-02-06 02:17:45 +00:00
|
|
|
retval = launch_default_screensaver(default_ss_dir_path, m_hGraphicsApplication);
|
|
|
|
if (retval) {
|
|
|
|
m_hGraphicsApplication = 0;
|
|
|
|
previous_result_ptr = NULL;
|
|
|
|
graphics_app_result_ptr = NULL;
|
2009-02-10 12:35:12 +00:00
|
|
|
m_bDefault_gfx_running = false;
|
2009-02-06 02:17:45 +00:00
|
|
|
} else {
|
2009-02-10 12:35:12 +00:00
|
|
|
m_bDefault_gfx_running = true;
|
2009-02-12 11:54:57 +00:00
|
|
|
default_saver_start_time_in_science_phase = dtime();
|
2009-02-06 02:17:45 +00:00
|
|
|
}
|
2008-08-18 22:09:22 +00:00
|
|
|
}
|
|
|
|
}
|
2009-02-05 12:32:16 +00:00
|
|
|
} // End if (ss_phase == SCIENCE_SS_PHASE)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Is the graphics app still running?
|
|
|
|
if (m_hGraphicsApplication) {
|
|
|
|
if (!process_exists(m_hGraphicsApplication)) {
|
|
|
|
// Something has happened to the previously selected screensaver
|
|
|
|
// application. Start a different one.
|
|
|
|
BOINCTRACE(_T("CScreensaver::DataManagementProc - Graphics application isn't running, start a new one.\n"));
|
|
|
|
m_hGraphicsApplication = 0;
|
|
|
|
graphics_app_result_ptr = NULL;
|
2009-02-10 12:35:12 +00:00
|
|
|
m_bDefault_gfx_running = false;
|
|
|
|
m_bScience_gfx_running = false;
|
2009-02-05 12:32:16 +00:00
|
|
|
continue;
|
|
|
|
} else {
|
2008-08-29 19:06:04 +00:00
|
|
|
#ifdef _WIN32
|
2009-02-05 12:32:16 +00:00
|
|
|
CheckForegroundWindow();
|
2008-08-18 22:09:22 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
2009-02-05 12:32:16 +00:00
|
|
|
|
2008-08-18 22:09:22 +00:00
|
|
|
} // end while(true)
|
|
|
|
}
|
2009-02-05 12:32:16 +00:00
|
|
|
|
|
|
|
|
2009-02-06 11:17:13 +00:00
|
|
|
void CScreensaver::GetDisplayPeriods(char *dir_path) {
|
|
|
|
char buf[1024];
|
|
|
|
FILE* f;
|
|
|
|
MIOFILE mf;
|
|
|
|
|
|
|
|
strlcpy(buf, dir_path, sizeof(buf));
|
|
|
|
strlcat(buf, PATH_SEPARATOR, sizeof(buf));
|
|
|
|
strlcat(buf, SS_CONFIG_FILE, sizeof(buf));
|
|
|
|
|
|
|
|
f = boinc_fopen(buf, "r");
|
|
|
|
if (!f) return;
|
|
|
|
|
|
|
|
mf.init_file(f);
|
|
|
|
XML_PARSER xp(&mf);
|
|
|
|
|
|
|
|
while (mf.fgets(buf, sizeof(buf))) {
|
|
|
|
if (parse_double(buf, "<default_gfx_duration>", m_fGFXDefaultPeriod)) continue;
|
|
|
|
if (parse_double(buf, "<science_gfx_duration>", m_fGFxSciencePeriod)) continue;
|
|
|
|
if (parse_double(buf, "<science_gfx_change_interval>", m_fGFXChangePeriod)) continue;
|
|
|
|
}
|
|
|
|
fclose(f);
|
2009-02-05 12:32:16 +00:00
|
|
|
|
2009-02-06 11:17:13 +00:00
|
|
|
BOINCTRACE(_T("CScreensaver::GetDisplayPeriods: m_fGFXDefaultPeriod=%f, m_fGFxSciencePeriod=%f, m_fGFXChangePeriod=%f\n"),
|
|
|
|
m_fGFXDefaultPeriod, m_fGFxSciencePeriod, m_fGFXChangePeriod);
|
2009-02-05 12:32:16 +00:00
|
|
|
}
|