// The contents of this file are subject to the BOINC 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://boinc.berkeley.edu/license_1.0.txt // // 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): // DirectX 8.1 Screen Saver Framework from Microsoft. // Microsoft Knowledge Base Article - 79212 // #include "boinc_win.h" #include #include #include #include #define COMPILE_MULTIMON_STUBS #include #include #include "boinc_ss.h" #include "win_util.h" #include "win_screensaver.h" //----------------------------------------------------------------------------- // Diagnostics Support //----------------------------------------------------------------------------- #define TRACE __noop // If we are compiling for a release build noop the // trace macro, otherwise use the native debugging // support in the MFC libraries static CScreensaver* gs_pScreensaver = NULL; INT WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) { HRESULT hr; CScreensaver BOINCSS; if( FAILED( hr = BOINCSS.Create( hInstance ) ) ) { BOINCSS.DisplayErrorMsg( hr ); return 0; } return BOINCSS.Run(); } //----------------------------------------------------------------------------- // Name: CScreensaver() // Desc: Constructor //----------------------------------------------------------------------------- CScreensaver::CScreensaver() { gs_pScreensaver = this; m_bCheckingSaverPassword = FALSE; m_bIs9x = FALSE; m_dwSaverMouseMoveCount = 0; m_hWnd = NULL; m_hWndParent = NULL; m_hPasswordDLL = NULL; m_VerifySaverPassword = NULL; m_bAllScreensSame = FALSE; m_bWindowed = FALSE; m_bWaitForInputIdle = FALSE; m_bErrorMode = FALSE; m_hrError = S_OK; m_szError[0] = TEXT('\0'); LoadString( NULL, IDS_DESCRIPTION, m_strWindowTitle, 200 ); m_bPaintingInitialized = FALSE; m_bBOINCCoreNotified = FALSE; m_bLogMessagePump = FALSE; ZeroMemory( m_Monitors, sizeof(m_Monitors) ); m_dwNumMonitors = 0; } //----------------------------------------------------------------------------- // Name: Create() // Desc: Have the client program call this function before calling Run(). //----------------------------------------------------------------------------- HRESULT CScreensaver::Create( HINSTANCE hInstance ) { HRESULT hr; m_hInstance = hInstance; // Parse the command line and do the appropriate thing m_SaverMode = ParseCommandLine( GetCommandLine() ); EnumMonitors(); UtilInitOSVersion(); // Register global messages BOINC_SS_START_MSG = RegisterWindowMessage( START_SS_MSG ); BOINC_SS_STOP_MSG = RegisterWindowMessage( STOP_SS_MSG ); // Create the screen saver window(s) if( m_SaverMode == sm_preview || m_SaverMode == sm_test || m_SaverMode == sm_full ) { if( FAILED( hr = CreateSaverWindow() ) ) { m_bErrorMode = TRUE; m_hrError = hr; } } if( m_SaverMode == sm_preview ) { // In preview mode, "pause" (enter a limited message loop) briefly // before proceeding, so the display control panel knows to update itself. m_bWaitForInputIdle = TRUE; // Post a message to mark the end of the initial group of window messages PostMessage( m_hWnd, WM_USER, 0, 0 ); MSG msg; while( m_bWaitForInputIdle ) { // If GetMessage returns FALSE, it's quitting time. if( !GetMessage( &msg, m_hWnd, 0, 0 ) ) { // Post the quit message to handle it later PostQuitMessage(0); break; } TranslateMessage( &msg ); DispatchMessage( &msg ); } } return S_OK; } //----------------------------------------------------------------------------- // Name: Run() // Desc: Starts main execution of the screen saver. //----------------------------------------------------------------------------- INT CScreensaver::Run() { HRESULT hr; // Parse the command line and do the appropriate thing switch ( m_SaverMode ) { case sm_config: { if( m_bErrorMode ) { DisplayErrorMsg( m_hrError ); } else { DoConfig(); } break; } case sm_preview: case sm_test: case sm_full: { if( FAILED( hr = DoSaver() ) ) DisplayErrorMsg( hr ); break; } case sm_passwordchange: { ChangePassword(); break; } } return 0; } //----------------------------------------------------------------------------- // Name: StartupBOINC() // Desc: Notifies BOINC that it has to start the screensaver in full screen mode. //----------------------------------------------------------------------------- VOID CScreensaver::StartupBOINC() { HWND hWnd; HANDLE boinc_mutex; if( m_SaverMode != sm_preview ) { boinc_mutex = CreateMutex(NULL, false, RUN_MUTEX); if(boinc_mutex != NULL && GetLastError() != ERROR_ALREADY_EXISTS) { CloseHandle(boinc_mutex); // Create a screen saver window on the primary display if the boinc client crashes CreateSaverWindow(); if(IsConfigStartupBOINC()) { m_bErrorMode = TRUE; m_hrError = SCRAPPERR_BOINCNOTDETECTED; } else { m_bErrorMode = TRUE; m_hrError = SCRAPPERR_BOINCNOTDETECTEDSTARTUP; } m_bBOINCCoreNotified = FALSE; } else { hWnd = FindWindow(TEXT("BOINCWindowClass"), NULL); if(NULL == hWnd) { // Create a screen saver window on the primary display if the boinc client crashes CreateSaverWindow(); m_bErrorMode = TRUE; m_hrError = SCRAPPERR_BOINCNOTFOUND; m_bBOINCCoreNotified = FALSE; } else { INTERNALMONITORINFO* pMonitorInfo; pMonitorInfo = &m_Monitors[0]; if( (NULL != pMonitorInfo->hWnd) && (m_bBOINCCoreNotified == FALSE) ) { // Tell the boinc client to start the screen saver SendMessage(hWnd, BOINC_SS_START_MSG, NULL, NULL); // We have now notified the boinc client m_bBOINCCoreNotified = TRUE; } } } } } //----------------------------------------------------------------------------- // Name: RenderBOINC() // Desc: Notifies BOINC that it has to start the screensaver in full screen mode. //----------------------------------------------------------------------------- VOID CScreensaver::ShutdownBOINC() { HWND hWnd; if( m_bBOINCCoreNotified ) { hWnd = FindWindow(TEXT("BOINCWindowClass"), NULL); if(NULL != hWnd) PostMessage(hWnd, BOINC_SS_STOP_MSG, NULL, NULL); } } //----------------------------------------------------------------------------- // Name: ParseCommandLine() // Desc: Interpret command-line parameters passed to this app. //----------------------------------------------------------------------------- SaverMode CScreensaver::ParseCommandLine( TCHAR* pstrCommandLine ) { m_hWndParent = NULL; TRACE("ParseCommandLine: '%s'\n", pstrCommandLine); // Skip the first part of the command line, which is the full path // to the exe. If it contains spaces, it will be contained in quotes. if (*pstrCommandLine == TEXT('\"')) { pstrCommandLine++; while (*pstrCommandLine != TEXT('\0') && *pstrCommandLine != TEXT('\"')) pstrCommandLine++; if( *pstrCommandLine == TEXT('\"') ) pstrCommandLine++; } else { while (*pstrCommandLine != TEXT('\0') && *pstrCommandLine != TEXT(' ')) pstrCommandLine++; if( *pstrCommandLine == TEXT(' ') ) pstrCommandLine++; } // Skip along to the first option delimiter "/" or "-" while ( *pstrCommandLine != TEXT('\0') && *pstrCommandLine != TEXT('/') && *pstrCommandLine != TEXT('-') ) pstrCommandLine++; // If there wasn't one, then must be config mode if ( *pstrCommandLine == TEXT('\0') ) return sm_config; // Otherwise see what the option was switch ( *(++pstrCommandLine) ) { case 'c': case 'C': pstrCommandLine++; while ( *pstrCommandLine && !isdigit(*pstrCommandLine) ) pstrCommandLine++; if ( isdigit(*pstrCommandLine) ) { #ifdef _WIN64 m_hWndParent = HWND(_atoi64(pstrCommandLine)); #else m_hWndParent = HWND(_ttol(pstrCommandLine)); #endif } else { m_hWndParent = NULL; } return sm_config; case 't': case 'T': return sm_test; case 'p': case 'P': // Preview-mode, so option is followed by the parent HWND in decimal pstrCommandLine++; while ( *pstrCommandLine && !isdigit(*pstrCommandLine) ) pstrCommandLine++; if ( isdigit(*pstrCommandLine) ) { #ifdef _WIN64 m_hWndParent = HWND(_atoi64(pstrCommandLine)); #else m_hWndParent = HWND(_ttol(pstrCommandLine)); #endif } return sm_preview; case 'a': case 'A': // Password change mode, so option is followed by parent HWND in decimal pstrCommandLine++; while ( *pstrCommandLine && !isdigit(*pstrCommandLine) ) pstrCommandLine++; if ( isdigit(*pstrCommandLine) ) { #ifdef _WIN64 m_hWndParent = HWND(_atoi64(pstrCommandLine)); #else m_hWndParent = HWND(_ttol(pstrCommandLine)); #endif } return sm_passwordchange; default: // All other options => run the screensaver (typically this is "/s") return sm_full; } } //----------------------------------------------------------------------------- // Name: EnumMonitors() // Desc: Determine HMONITOR, desktop rect, and other info for each monitor. // Note that EnumDisplayDevices enumerates monitors in the order // indicated on the Settings page of the Display control panel, which // is the order we want to list monitors in, as opposed to the order // used by D3D's GetAdapterInfo. //----------------------------------------------------------------------------- VOID CScreensaver::EnumMonitors( VOID ) { DWORD iDevice = 0; DISPLAY_DEVICE_FULL dispdev; DISPLAY_DEVICE_FULL dispdev2; DEVMODE devmode; dispdev.cb = sizeof(dispdev); dispdev2.cb = sizeof(dispdev2); devmode.dmSize = sizeof(devmode); devmode.dmDriverExtra = 0; INTERNALMONITORINFO* pMonitorInfoNew; while( EnumDisplayDevices(NULL, iDevice, (DISPLAY_DEVICE*)&dispdev, 0) ) { // Ignore NetMeeting's mirrored displays if( (dispdev.StateFlags & DISPLAY_DEVICE_MIRRORING_DRIVER) == 0 ) { // To get monitor info for a display device, call EnumDisplayDevices // a second time, passing dispdev.DeviceName (from the first call) as // the first parameter. EnumDisplayDevices(dispdev.DeviceName, 0, (DISPLAY_DEVICE*)&dispdev2, 0); pMonitorInfoNew = &m_Monitors[m_dwNumMonitors]; ZeroMemory( pMonitorInfoNew, sizeof(INTERNALMONITORINFO) ); StringCchCopy( pMonitorInfoNew->strDeviceName, 128, dispdev.DeviceString ); StringCchCopy( pMonitorInfoNew->strMonitorName, 128, dispdev2.DeviceString ); if( dispdev.StateFlags & DISPLAY_DEVICE_ATTACHED_TO_DESKTOP ) { EnumDisplaySettings( dispdev.DeviceName, ENUM_CURRENT_SETTINGS, &devmode ); if( dispdev.StateFlags & DISPLAY_DEVICE_PRIMARY_DEVICE ) { // For some reason devmode.dmPosition is not always (0, 0) // for the primary display, so force it. pMonitorInfoNew->rcScreen.left = 0; pMonitorInfoNew->rcScreen.top = 0; } else { pMonitorInfoNew->rcScreen.left = devmode.dmPosition.x; pMonitorInfoNew->rcScreen.top = devmode.dmPosition.y; } pMonitorInfoNew->rcScreen.right = pMonitorInfoNew->rcScreen.left + devmode.dmPelsWidth; pMonitorInfoNew->rcScreen.bottom = pMonitorInfoNew->rcScreen.top + devmode.dmPelsHeight; pMonitorInfoNew->hMonitor = MonitorFromRect( &pMonitorInfoNew->rcScreen, MONITOR_DEFAULTTONULL ); } m_dwNumMonitors++; if( m_dwNumMonitors == MAX_DISPLAYS ) break; } iDevice++; } } //----------------------------------------------------------------------------- // Name: CreateSaverWindow // Desc: Register and create the appropriate window(s) //----------------------------------------------------------------------------- HRESULT CScreensaver::CreateSaverWindow() { // Register an appropriate window class for the primary display WNDCLASS cls; cls.hCursor = LoadCursor( NULL, IDC_ARROW ); cls.hIcon = LoadIcon( m_hInstance, MAKEINTRESOURCE(IDI_MAIN_ICON) ); cls.lpszMenuName = NULL; cls.lpszClassName = TEXT("BOINCPrimarySaverWndClass"); cls.hbrBackground = (HBRUSH) GetStockObject(BLACK_BRUSH); cls.hInstance = m_hInstance; cls.style = CS_VREDRAW|CS_HREDRAW; cls.lpfnWndProc = PrimarySaverProcStub; cls.cbWndExtra = 0; cls.cbClsExtra = 0; RegisterClass( &cls ); // Register an appropriate window class for the secondary display(s) WNDCLASS cls2; cls2.hCursor = LoadCursor( NULL, IDC_ARROW ); cls2.hIcon = LoadIcon( m_hInstance, MAKEINTRESOURCE(IDI_MAIN_ICON) ); cls2.lpszMenuName = NULL; cls2.lpszClassName = TEXT("BOINCGenericSaverWndClass"); cls2.hbrBackground = (HBRUSH) GetStockObject(BLACK_BRUSH); cls2.hInstance = m_hInstance; cls2.style = CS_VREDRAW|CS_HREDRAW; cls2.lpfnWndProc = GenericSaverProcStub; cls2.cbWndExtra = 0; cls2.cbClsExtra = 0; RegisterClass( &cls2 ); // Create the window RECT rc; DWORD dwStyle; switch ( m_SaverMode ) { case sm_preview: GetClientRect( m_hWndParent, &rc ); dwStyle = WS_VISIBLE | WS_CHILD; AdjustWindowRect( &rc, dwStyle, FALSE ); m_hWnd = CreateWindow( TEXT("BOINCPrimarySaverWndClass"), m_strWindowTitle, dwStyle, rc.left, rc.top, rc.right-rc.left, rc.bottom-rc.top, m_hWndParent, NULL, m_hInstance, this ); m_Monitors[0].hWnd = m_hWnd; GetClientRect( m_hWnd, &m_rcRenderTotal ); GetClientRect( m_hWnd, &m_rcRenderCurDevice ); break; case sm_test: rc.left = rc.top = 50; rc.right = rc.left+600; rc.bottom = rc.top+400; dwStyle = WS_VISIBLE | WS_OVERLAPPED | WS_CAPTION | WS_MINIMIZEBOX | WS_SYSMENU; AdjustWindowRect( &rc, dwStyle, FALSE ); m_hWnd = CreateWindow( TEXT("BOINCPrimarySaverWndClass"), m_strWindowTitle, dwStyle, rc.left, rc.top, rc.right-rc.left, rc.bottom-rc.top, NULL, NULL, m_hInstance, this ); m_Monitors[0].hWnd = m_hWnd; GetClientRect( m_hWnd, &m_rcRenderTotal ); GetClientRect( m_hWnd, &m_rcRenderCurDevice ); SetTimer(m_hWnd, 2, 1000, NULL); break; case sm_full: dwStyle = WS_VISIBLE | WS_POPUP; m_hWnd = NULL; for( DWORD iMonitor = 0; iMonitor < m_dwNumMonitors; iMonitor++ ) { INTERNALMONITORINFO* pMonitorInfo; pMonitorInfo = &m_Monitors[iMonitor]; if( pMonitorInfo->hWnd == NULL ) { if( pMonitorInfo->hMonitor == NULL ) continue; rc = pMonitorInfo->rcScreen; if( 0 == iMonitor ) { pMonitorInfo->hWnd = CreateWindowEx( WS_EX_TOPMOST, TEXT("BOINCPrimarySaverWndClass"), m_strWindowTitle, dwStyle, rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top, NULL, NULL, m_hInstance, this ); } else { pMonitorInfo->hWnd = CreateWindowEx( WS_EX_TOPMOST, TEXT("BOINCGenericSaverWndClass"), m_strWindowTitle, dwStyle, rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top, NULL, NULL, m_hInstance, this ); } if( pMonitorInfo->hWnd == NULL ) return E_FAIL; if( m_hWnd == NULL ) m_hWnd = pMonitorInfo->hWnd; SetTimer(pMonitorInfo->hWnd, 2, 1000, NULL); } } } if ( m_hWnd == NULL ) return E_FAIL; return S_OK; } //----------------------------------------------------------------------------- // Name: DoSaver() // Desc: Run the screensaver graphics - may be preview, test or full-on mode //----------------------------------------------------------------------------- HRESULT CScreensaver::DoSaver() { // Figure out if we're on Win9x OSVERSIONINFO osvi; osvi.dwOSVersionInfoSize = sizeof(osvi); GetVersionEx( &osvi ); m_bIs9x = (osvi.dwPlatformId == VER_PLATFORM_WIN32_WINDOWS); // If we're in full on mode, and on 9x, then need to load the password DLL if ( m_SaverMode == sm_full && m_bIs9x ) { // Only do this if the password is set - check registry: HKEY hKey; if ( RegOpenKey( HKEY_CURRENT_USER , REGSTR_PATH_SCREENSAVE , &hKey ) == ERROR_SUCCESS ) { DWORD dwVal; DWORD dwSize = sizeof(dwVal); if ( (RegQueryValueEx( hKey, REGSTR_VALUE_USESCRPASSWORD, NULL, NULL, (BYTE *)&dwVal, &dwSize ) == ERROR_SUCCESS) && dwVal ) { m_hPasswordDLL = LoadLibrary( TEXT("PASSWORD.CPL") ); if ( m_hPasswordDLL ) m_VerifySaverPassword = (VERIFYPWDPROC)GetProcAddress( m_hPasswordDLL, TEXT("VerifyScreenSavePwd") ); RegCloseKey( hKey ); } } } // Flag as screensaver running if in full on mode if ( m_SaverMode == sm_full ) { BOOL bUnused; SystemParametersInfo( SPI_SCREENSAVERRUNNING, TRUE, &bUnused, 0 ); } // Message pump BOOL bGotMsg; MSG msg; msg.message = WM_NULL; while ( msg.message != WM_QUIT ) { bGotMsg = PeekMessage( &msg, NULL, 0, 0, PM_REMOVE ); if( bGotMsg ) { TranslateMessage( &msg ); DispatchMessage( &msg ); } else { Sleep(10); } } return S_OK; } //----------------------------------------------------------------------------- // Name: DoConfig() // Desc: //----------------------------------------------------------------------------- VOID CScreensaver::DoConfig() { DialogBox( NULL, MAKEINTRESOURCE(DLG_CONFIG), m_hWndParent, ConfigureDialogProcStub ); } //----------------------------------------------------------------------------- // Name: ConfigureDialogProcStub() // Desc: //----------------------------------------------------------------------------- INT_PTR CALLBACK CScreensaver::ConfigureDialogProcStub( HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam ) { return gs_pScreensaver->ConfigureDialogProc( hwndDlg, uMsg, wParam, lParam ); } //----------------------------------------------------------------------------- // Name: ConfigureDialogProc() // Desc: //----------------------------------------------------------------------------- INT_PTR CScreensaver::ConfigureDialogProc(HWND hwnd,UINT msg,WPARAM wParam,LPARAM lParam) { DWORD screen_blank=0, blank_time=0; char buf[256]; int retval; switch (msg) { case WM_INITDIALOG: // make sure you check return value of registry queries // in case the item in question doesn't happen to exist. retval = UtilGetRegKey( REG_BLANK_NAME, screen_blank ); if ( retval < 0 ) { screen_blank=0; } CheckDlgButton(hwnd, IDC_BLANK, screen_blank); retval = UtilGetRegKey( REG_BLANK_TIME, blank_time ); if ( retval < 0 ) { blank_time=0; } _ltot(blank_time, buf, 10); SetDlgItemText(hwnd, IDC_BLANK_TIME, buf); return TRUE; case WM_COMMAND: int id=LOWORD(wParam); if (id==IDOK) { screen_blank=( IsDlgButtonChecked(hwnd,IDC_BLANK) == BST_CHECKED ); UtilSetRegKey( REG_BLANK_NAME, screen_blank ); GetDlgItemText(hwnd, IDC_BLANK_TIME, buf, 256 ); blank_time = atoi(buf); UtilSetRegKey( REG_BLANK_TIME, blank_time ); } if (id==IDOK || id==IDCANCEL) EndDialog(hwnd,id); break; } return FALSE; } //----------------------------------------------------------------------------- // Name: PrimarySaverProcStub() // Desc: This function forwards all window messages to SaverProc, which has // access to the "this" pointer. //----------------------------------------------------------------------------- LRESULT CALLBACK CScreensaver::PrimarySaverProcStub( HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam ) { return gs_pScreensaver->PrimarySaverProc( hWnd, uMsg, wParam, lParam ); } //----------------------------------------------------------------------------- // Name: PrimarySaverProc() // Desc: Handle window messages for main screensaver windows (one per screen). //----------------------------------------------------------------------------- LRESULT CScreensaver::PrimarySaverProc( HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam ) { switch ( uMsg ) { case WM_USER: // All initialization messages have gone through. Allow // 500ms of idle time, then proceed with initialization. SetTimer( hWnd, 1, 500, NULL ); break; case WM_TIMER: switch (wParam) { case 1: // Initial idle time is done, proceed with initialization. m_bWaitForInputIdle = FALSE; KillTimer( hWnd, 1 ); return 0; case 2: if( m_bErrorMode ) { UpdateErrorBox(); } else { StartupBOINC(); } return 0; } break; case WM_DESTROY: if( m_SaverMode == sm_preview || m_SaverMode == sm_test ) ShutdownSaver(); break; case WM_SETCURSOR: if ( m_SaverMode == sm_full && !m_bCheckingSaverPassword ) { // Hide cursor SetCursor( NULL ); return TRUE; } break; case WM_PAINT: { // Show error message, if there is one PAINTSTRUCT ps; BeginPaint( hWnd, &ps ); // In preview mode, just fill // the preview window with black. if( !m_bErrorMode && m_SaverMode == sm_preview ) { RECT rc; GetClientRect(hWnd,&rc); FillRect(ps.hdc, &rc, (HBRUSH)GetStockObject(BLACK_BRUSH) ); DrawIcon(ps.hdc, (rc.right / 2) - 16, (rc.bottom / 2) - 16, LoadIcon( m_hInstance, MAKEINTRESOURCE(IDI_MAIN_ICON) ) ); } else { DoPaint( hWnd, ps.hdc ); } EndPaint( hWnd, &ps ); return 0; } case WM_MOUSEMOVE: if( m_SaverMode != sm_test ) { static INT xPrev = -1; static INT yPrev = -1; INT xCur = GET_X_LPARAM(lParam); INT yCur = GET_Y_LPARAM(lParam); if( xCur != xPrev || yCur != yPrev ) { xPrev = xCur; yPrev = yCur; m_dwSaverMouseMoveCount++; if ( m_dwSaverMouseMoveCount > 5 ) InterruptSaver(); } } break; case WM_KEYDOWN: case WM_LBUTTONDOWN: case WM_RBUTTONDOWN: case WM_MBUTTONDOWN: if( m_SaverMode != sm_test ) InterruptSaver(); break; case WM_NCACTIVATE: case WM_ACTIVATEAPP: return 0; break; case WM_POWERBROADCAST: if( wParam == PBT_APMSUSPEND && m_VerifySaverPassword == NULL ) InterruptSaver(); break; case WM_SYSCOMMAND: if ( m_SaverMode == sm_full ) { switch ( wParam ) { case SC_NEXTWINDOW: case SC_PREVWINDOW: case SC_SCREENSAVE: case SC_CLOSE: return FALSE; }; } break; } if ( BOINC_SS_STOP_MSG == uMsg ) { if( m_SaverMode != sm_test ) InterruptSaver(); } if (m_bLogMessagePump) TRACE(TEXT("PrimarySaverProc hWnd '%d' uMsg '%X' wParam '%d' lParam '%d'\n"), hWnd, uMsg, wParam, lParam); return DefWindowProc( hWnd, uMsg, wParam, lParam ); } //----------------------------------------------------------------------------- // Name: GenericSaverProcStub() // Desc: This function forwards all window messages to SaverProc, which has // access to the "this" pointer. //----------------------------------------------------------------------------- LRESULT CALLBACK CScreensaver::GenericSaverProcStub( HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam ) { return gs_pScreensaver->GenericSaverProc( hWnd, uMsg, wParam, lParam ); } //----------------------------------------------------------------------------- // Name: GenericSaverProc() // Desc: Handle window messages for main screensaver windows (one per screen). //----------------------------------------------------------------------------- LRESULT CScreensaver::GenericSaverProc( HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam ) { switch ( uMsg ) { case WM_USER: // All initialization messages have gone through. Allow // 500ms of idle time, then proceed with initialization. SetTimer( hWnd, 1, 500, NULL ); break; case WM_TIMER: switch (wParam) { case 1: // Initial idle time is done, proceed with initialization. m_bWaitForInputIdle = FALSE; KillTimer( hWnd, 1 ); return 0; case 2: if( m_bErrorMode ) { UpdateErrorBox(); } else { StartupBOINC(); } return 0; } break; case WM_DESTROY: if( m_SaverMode == sm_preview || m_SaverMode == sm_test ) ShutdownSaver(); break; case WM_SETCURSOR: if ( m_SaverMode == sm_full && !m_bCheckingSaverPassword ) { // Hide cursor SetCursor( NULL ); return TRUE; } break; case WM_PAINT: { // Show error message, if there is one PAINTSTRUCT ps; BeginPaint( hWnd, &ps ); // In preview mode, just fill // the preview window with black. if( !m_bErrorMode && m_SaverMode == sm_preview ) { RECT rc; GetClientRect(hWnd,&rc); FillRect(ps.hdc, &rc, (HBRUSH)GetStockObject(BLACK_BRUSH) ); DrawIcon(ps.hdc, (rc.right / 2) - 16, (rc.bottom / 2) - 16, LoadIcon( m_hInstance, MAKEINTRESOURCE(IDI_MAIN_ICON) ) ); } else { DoPaint( hWnd, ps.hdc ); } EndPaint( hWnd, &ps ); return 0; } case WM_MOUSEMOVE: if( m_SaverMode != sm_test ) { static INT xPrev = -1; static INT yPrev = -1; INT xCur = GET_X_LPARAM(lParam); INT yCur = GET_Y_LPARAM(lParam); if( xCur != xPrev || yCur != yPrev ) { xPrev = xCur; yPrev = yCur; m_dwSaverMouseMoveCount++; if ( m_dwSaverMouseMoveCount > 5 ) InterruptSaver(); } } break; case WM_KEYDOWN: case WM_LBUTTONDOWN: case WM_RBUTTONDOWN: case WM_MBUTTONDOWN: if( m_SaverMode != sm_test ) InterruptSaver(); break; case WM_ACTIVATEAPP: if( wParam == FALSE && m_SaverMode != sm_test ) InterruptSaver(); break; case WM_POWERBROADCAST: if( wParam == PBT_APMSUSPEND && m_VerifySaverPassword == NULL ) InterruptSaver(); break; case WM_SYSCOMMAND: if ( m_SaverMode == sm_full ) { switch ( wParam ) { case SC_NEXTWINDOW: case SC_PREVWINDOW: case SC_SCREENSAVE: case SC_CLOSE: return FALSE; }; } break; } if ( BOINC_SS_STOP_MSG == uMsg ) { if( m_SaverMode != sm_test ) InterruptSaver(); } if (m_bLogMessagePump) TRACE(TEXT("GenericSaverProc hWnd '%d' uMsg '%X' wParam '%d' lParam '%d'\n"), hWnd, uMsg, wParam, lParam); return DefWindowProc( hWnd, uMsg, wParam, lParam ); } //----------------------------------------------------------------------------- // Name: ShutdownSaver() // Desc: //----------------------------------------------------------------------------- VOID CScreensaver::ShutdownSaver() { // Unflag screensaver running if in full on mode if ( m_SaverMode == sm_full ) { BOOL bUnused; SystemParametersInfo( SPI_SCREENSAVERRUNNING, FALSE, &bUnused, 0 ); } // Unload the password DLL (if we loaded it) if ( m_hPasswordDLL != NULL ) { FreeLibrary( m_hPasswordDLL ); m_hPasswordDLL = NULL; } ShutdownBOINC(); // Post message to drop out of message loop PostQuitMessage( 0 ); } //----------------------------------------------------------------------------- // Name: InterruptSaver() // Desc: A message was received (mouse move, keydown, etc.) that may mean // the screen saver should show the password dialog and/or shut down. //----------------------------------------------------------------------------- VOID CScreensaver::InterruptSaver() { BOOL bPasswordOkay = FALSE; if( m_SaverMode == sm_test || m_SaverMode == sm_full && !m_bCheckingSaverPassword ) { if( m_bIs9x && m_SaverMode == sm_full ) { // If no VerifyPassword function, then no password is set // or we're not on 9x. if ( m_VerifySaverPassword != NULL ) { m_bCheckingSaverPassword = TRUE; bPasswordOkay = m_VerifySaverPassword( m_hWnd ); m_bCheckingSaverPassword = FALSE; if ( !bPasswordOkay ) { // Back to screen saving... SetCursor( NULL ); m_dwSaverMouseMoveCount = 0; return; } } } ShutdownSaver(); } } //----------------------------------------------------------------------------- // Name: UpdateErrorBox() // Desc: Update the box that shows the error message //----------------------------------------------------------------------------- VOID CScreensaver::UpdateErrorBox() { INTERNALMONITORINFO* pMonitorInfo; HWND hwnd; RECT rcBounds; static DWORD dwTimeLast = 0; DWORD dwTimeNow; FLOAT fTimeDelta; // Update timing to determine how much to move error box if( dwTimeLast == 0 ) dwTimeLast = timeGetTime(); dwTimeNow = timeGetTime(); fTimeDelta = (FLOAT)(dwTimeNow - dwTimeLast) / 1000.0f; dwTimeLast = dwTimeNow; // Load error string if necessary if( m_szError[0] == TEXT('\0') ) { GetTextForError( m_hrError, m_szError, sizeof(m_szError) / sizeof(TCHAR) ); } for( DWORD iMonitor = 0; iMonitor < m_dwNumMonitors; iMonitor++ ) { pMonitorInfo = &m_Monitors[iMonitor]; hwnd = pMonitorInfo->hWnd; if( hwnd == NULL ) continue; if( m_SaverMode == sm_full ) { rcBounds = pMonitorInfo->rcScreen; ScreenToClient( hwnd, (POINT*)&rcBounds.left ); ScreenToClient( hwnd, (POINT*)&rcBounds.right ); } else { rcBounds = m_rcRenderTotal; } if( pMonitorInfo->widthError == 0 ) { if( m_SaverMode == sm_preview ) { pMonitorInfo->widthError = (float) (rcBounds.right - rcBounds.left); pMonitorInfo->heightError = (float) (rcBounds.bottom - rcBounds.top); pMonitorInfo->xError = 0.0f; pMonitorInfo->yError = 0.0f; pMonitorInfo->xVelError = 0.0f; pMonitorInfo->yVelError = 0.0f; InvalidateRect( hwnd, NULL, FALSE ); // Invalidate the hwnd so it gets drawn UpdateWindow( hwnd ); } else { pMonitorInfo->widthError = 500; pMonitorInfo->heightError = 154; pMonitorInfo->xError = (rcBounds.right + rcBounds.left - pMonitorInfo->widthError) / 2.0f; pMonitorInfo->yError = (rcBounds.bottom + rcBounds.top - pMonitorInfo->heightError) / 2.0f; pMonitorInfo->xVelError = (rcBounds.right - rcBounds.left) / 10.0f; pMonitorInfo->yVelError = (rcBounds.bottom - rcBounds.top) / 20.0f; } } else { if( m_SaverMode != sm_preview ) { RECT rcOld; RECT rcNew; SetRect( &rcOld, (INT)pMonitorInfo->xError, (INT)pMonitorInfo->yError, (INT)(pMonitorInfo->xError + pMonitorInfo->widthError), (INT)(pMonitorInfo->yError + pMonitorInfo->heightError) ); // Update rect velocity if( (pMonitorInfo->xError + pMonitorInfo->xVelError * fTimeDelta + pMonitorInfo->widthError > rcBounds.right && pMonitorInfo->xVelError > 0.0f) || (pMonitorInfo->xError + pMonitorInfo->xVelError * fTimeDelta < rcBounds.left && pMonitorInfo->xVelError < 0.0f) ) { pMonitorInfo->xVelError = -pMonitorInfo->xVelError; } if( (pMonitorInfo->yError + pMonitorInfo->yVelError * fTimeDelta + pMonitorInfo->heightError > rcBounds.bottom && pMonitorInfo->yVelError > 0.0f) || (pMonitorInfo->yError + pMonitorInfo->yVelError * fTimeDelta < rcBounds.top && pMonitorInfo->yVelError < 0.0f) ) { pMonitorInfo->yVelError = -pMonitorInfo->yVelError; } // Update rect position pMonitorInfo->xError += pMonitorInfo->xVelError * fTimeDelta; pMonitorInfo->yError += pMonitorInfo->yVelError * fTimeDelta; SetRect( &rcNew, (INT)pMonitorInfo->xError, (INT)pMonitorInfo->yError, (INT)(pMonitorInfo->xError + pMonitorInfo->widthError), (INT)(pMonitorInfo->yError + pMonitorInfo->heightError) ); if( rcOld.left != rcNew.left || rcOld.top != rcNew.top ) { InvalidateRect( hwnd, &rcOld, FALSE ); // Invalidate old rect so it gets erased InvalidateRect( hwnd, &rcNew, FALSE ); // Invalidate new rect so it gets drawn UpdateWindow( hwnd ); } } } } } //----------------------------------------------------------------------------- // Name: GetTextForError() // Desc: Translate an HRESULT error code into a string that can be displayed // to explain the error. A class derived from CD3DScreensaver can // provide its own version of this function that provides app-specific // error translation instead of or in addition to calling this function. // This function returns TRUE if a specific error was translated, or // FALSE if no specific translation for the HRESULT was found (though // it still puts a generic string into pszError). //----------------------------------------------------------------------------- BOOL CScreensaver::GetTextForError( HRESULT hr, TCHAR* pszError, DWORD dwNumChars ) { const DWORD dwErrorMap[][2] = { // HRESULT, stringID E_FAIL, IDS_ERR_GENERIC, E_OUTOFMEMORY, IDS_ERR_OUTOFMEMORY, SCRAPPERR_BOINCNOTDETECTED, IDS_ERR_BOINCNOTDETECTED, SCRAPPERR_BOINCNOTDETECTEDSTARTUP, IDS_ERR_BOINCNOTDETECTEDSTARTUP, SCRAPPERR_BOINCNOTFOUND, IDS_ERR_BOINCNOTFOUND, SCRAPPERR_NOPREVIEW, IDS_ERR_NOPREVIEW }; const DWORD dwErrorMapSize = sizeof(dwErrorMap) / sizeof(DWORD[2]); DWORD iError; DWORD resid = 0; for( iError = 0; iError < dwErrorMapSize; iError++ ) { if( hr == (HRESULT)dwErrorMap[iError][0] ) { resid = dwErrorMap[iError][1]; } } if( resid == 0 ) { resid = IDS_ERR_GENERIC; } LoadString( NULL, resid, pszError, dwNumChars ); if( resid == IDS_ERR_GENERIC ) return FALSE; else return TRUE; } //----------------------------------------------------------------------------- // Name: DoPaint() // Desc: //----------------------------------------------------------------------------- VOID CScreensaver::DoPaint(HWND hwnd, HDC hdc) { HMONITOR hMonitor = MonitorFromWindow( hwnd, MONITOR_DEFAULTTONEAREST ); INTERNALMONITORINFO* pMonitorInfo; for( DWORD iMonitor = 0; iMonitor < m_dwNumMonitors; iMonitor++) { pMonitorInfo = &m_Monitors[iMonitor]; if( pMonitorInfo->hMonitor == hMonitor ) break; } if( iMonitor == m_dwNumMonitors ) return; // Draw the error message box RECT rc; RECT rc2; RECT rcOrginal; int iTextHeight; static HBRUSH hbrushBlack = (HBRUSH)GetStockObject( BLACK_BRUSH ); static HBRUSH hbrushRed = (HBRUSH)CreateSolidBrush( RGB(255,0,0) ); static HBITMAP hbmp = LoadBitmap( m_hInstance, MAKEINTRESOURCE(IDB_BOINCSPLAT) ); SetRect( &rc, (INT)pMonitorInfo->xError, (INT)pMonitorInfo->yError, (INT)(pMonitorInfo->xError + pMonitorInfo->widthError), (INT)(pMonitorInfo->yError + pMonitorInfo->heightError) ); SetRect( &rcOrginal, (INT)pMonitorInfo->xError, (INT)pMonitorInfo->yError, (INT)(pMonitorInfo->xError + pMonitorInfo->widthError), (INT)(pMonitorInfo->yError + pMonitorInfo->heightError) ); // Draw the background as black, and put a frame around it that it red. FillRect(hdc, &rc, hbrushBlack); FrameRect(hdc, &rc, hbrushRed); // Draw the bitmap rectangle and copy the bitmap into // it. the bitmap is centered in the rectangle by adding 2 // to the left and top coordinates of the bitmap rectangle, // and subtracting 4 from the right and bottom coordinates. DrawTransparentBitmap(hdc, hbmp, (rc.left + 2), (rc.top + 2), RGB(129, 129, 129)); rc.left += 166; // Draw text in the center of the frame SetBkColor(hdc, RGB(0,0,0)); // Black SetTextColor(hdc, RGB(255,255,255)); // Red rc2 = rc; iTextHeight = DrawText(hdc, m_szError, -1, &rc, DT_CENTER | DT_CALCRECT ); rc = rc2; rc2.top = (rc.bottom + rc.top - iTextHeight) / 2; DrawText(hdc, m_szError, -1, &rc2, DT_CENTER ); // Erase everywhere except the error message box ExcludeClipRect( hdc, rcOrginal.left, rcOrginal.top, rcOrginal.right, rcOrginal.bottom ); rc = pMonitorInfo->rcScreen; ScreenToClient( hwnd, (POINT*)&rc.left ); ScreenToClient( hwnd, (POINT*)&rc.right ); FillRect(hdc, &rc, hbrushBlack ); } //----------------------------------------------------------------------------- // Name: ChangePassword() // Desc: //----------------------------------------------------------------------------- VOID CScreensaver::ChangePassword() { // Load the password change DLL HINSTANCE mpr = LoadLibrary( TEXT("MPR.DLL") ); if ( mpr != NULL ) { // Grab the password change function from it typedef DWORD (PASCAL *PWCHGPROC)( LPCSTR, HWND, DWORD, LPVOID ); PWCHGPROC pwd = (PWCHGPROC)GetProcAddress( mpr, "PwdChangePasswordA" ); // Do the password change if ( pwd != NULL ) pwd( "SCRSAVE", m_hWndParent, 0, NULL ); // Free the library FreeLibrary( mpr ); } } //----------------------------------------------------------------------------- // Name: DisplayErrorMsg() // Desc: Displays error messages in a message box //----------------------------------------------------------------------------- HRESULT CScreensaver::DisplayErrorMsg( HRESULT hr ) { TCHAR strMsg[512]; GetTextForError( hr, strMsg, 512 ); MessageBox( m_hWnd, strMsg, m_strWindowTitle, MB_ICONERROR | MB_OK ); return hr; } //----------------------------------------------------------------------------- // Name: DrawTransparentBitmap() // Desc: Draws a bitmap on the screen with a transparent background. // Code orginally from Microsoft Knowledge Base Article - 79212 //----------------------------------------------------------------------------- void CScreensaver::DrawTransparentBitmap(HDC hdc, HBITMAP hBitmap, LONG xStart, LONG yStart, COLORREF cTransparentColor) { BITMAP bm; COLORREF cColor; HBITMAP bmAndBack, bmAndObject, bmAndMem, bmSave; HBITMAP bmBackOld, bmObjectOld, bmMemOld, bmSaveOld; HDC hdcMem, hdcBack, hdcObject, hdcTemp, hdcSave; POINT ptSize; hdcTemp = CreateCompatibleDC(hdc); SelectObject(hdcTemp, hBitmap); // Select the bitmap GetObject(hBitmap, sizeof(BITMAP), (LPSTR)&bm); ptSize.x = bm.bmWidth; // Get width of bitmap ptSize.y = bm.bmHeight; // Get height of bitmap DPtoLP(hdcTemp, &ptSize, 1); // Convert from device // to logical points // Create some DCs to hold temporary data. hdcBack = CreateCompatibleDC(hdc); hdcObject = CreateCompatibleDC(hdc); hdcMem = CreateCompatibleDC(hdc); hdcSave = CreateCompatibleDC(hdc); // Create a bitmap for each DC. DCs are required for a number of // GDI functions. // Monochrome DC bmAndBack = CreateBitmap(ptSize.x, ptSize.y, 1, 1, NULL); // Monochrome DC bmAndObject = CreateBitmap(ptSize.x, ptSize.y, 1, 1, NULL); bmAndMem = CreateCompatibleBitmap(hdc, ptSize.x, ptSize.y); bmSave = CreateCompatibleBitmap(hdc, ptSize.x, ptSize.y); // Each DC must select a bitmap object to store pixel data. bmBackOld = (HBITMAP)SelectObject(hdcBack, bmAndBack); bmObjectOld = (HBITMAP)SelectObject(hdcObject, bmAndObject); bmMemOld = (HBITMAP)SelectObject(hdcMem, bmAndMem); bmSaveOld = (HBITMAP)SelectObject(hdcSave, bmSave); // Set proper mapping mode. SetMapMode(hdcTemp, GetMapMode(hdc)); // Save the bitmap sent here, because it will be overwritten. BitBlt(hdcSave, 0, 0, ptSize.x, ptSize.y, hdcTemp, 0, 0, SRCCOPY); // Set the background color of the source DC to the color. // contained in the parts of the bitmap that should be transparent cColor = SetBkColor(hdcTemp, cTransparentColor); // Create the object mask for the bitmap by performing a BitBlt // from the source bitmap to a monochrome bitmap. BitBlt(hdcObject, 0, 0, ptSize.x, ptSize.y, hdcTemp, 0, 0, SRCCOPY); // Set the background color of the source DC back to the original // color. SetBkColor(hdcTemp, cColor); // Create the inverse of the object mask. BitBlt(hdcBack, 0, 0, ptSize.x, ptSize.y, hdcObject, 0, 0, NOTSRCCOPY); // Copy the background of the main DC to the destination. BitBlt(hdcMem, 0, 0, ptSize.x, ptSize.y, hdc, xStart, yStart, SRCCOPY); // Mask out the places where the bitmap will be placed. BitBlt(hdcMem, 0, 0, ptSize.x, ptSize.y, hdcObject, 0, 0, SRCAND); // Mask out the transparent colored pixels on the bitmap. BitBlt(hdcTemp, 0, 0, ptSize.x, ptSize.y, hdcBack, 0, 0, SRCAND); // XOR the bitmap with the background on the destination DC. BitBlt(hdcMem, 0, 0, ptSize.x, ptSize.y, hdcTemp, 0, 0, SRCPAINT); // Copy the destination to the screen. BitBlt(hdc, xStart, yStart, ptSize.x, ptSize.y, hdcMem, 0, 0, SRCCOPY); // Place the original bitmap back into the bitmap sent here. BitBlt(hdcTemp, 0, 0, ptSize.x, ptSize.y, hdcSave, 0, 0, SRCCOPY); // Delete the memory bitmaps. DeleteObject(SelectObject(hdcBack, bmBackOld)); DeleteObject(SelectObject(hdcObject, bmObjectOld)); DeleteObject(SelectObject(hdcMem, bmMemOld)); DeleteObject(SelectObject(hdcSave, bmSaveOld)); // Delete the memory DCs. DeleteDC(hdcMem); DeleteDC(hdcBack); DeleteDC(hdcObject); DeleteDC(hdcSave); DeleteDC(hdcTemp); } //----------------------------------------------------------------------------- // Name: IsConfigStatupBOINC() // Desc: Determine if BOINC is configured to automatically start at logon/startup. //----------------------------------------------------------------------------- // Define dynamically linked to function typedef HRESULT (STDAPICALLTYPE* MYSHGETFOLDERPATH)(HWND hwnd, int csidl, HANDLE hToken, DWORD dwFlags, LPSTR pszPath); BOOL CScreensaver::IsConfigStartupBOINC() { BOOL bRetVal; BOOL bCheckFileExists; TCHAR szBuffer[512]; HANDLE hFileHandle; HMODULE hShell32; MYSHGETFOLDERPATH pfnMySHGetFolderPath = NULL; // Lets set the default value to FALSE bRetVal = FALSE; // Attempt to link to dynamic function if it exists hShell32 = LoadLibrary(TEXT("SHELL32.DLL")); if ( NULL != hShell32 ) pfnMySHGetFolderPath = (MYSHGETFOLDERPATH) GetProcAddress(hShell32, TEXT("SHGetFolderPathA")); // Now lets begin looking in the registry if (ERROR_SUCCESS == UtilGetRegStartupStr(REG_STARTUP_NAME, szBuffer)) { bRetVal = TRUE; } else { // It could be in the global startup group ZeroMemory( szBuffer, 512 ); bCheckFileExists = FALSE; if ( NULL != pfnMySHGetFolderPath ) { if (SUCCEEDED((pfnMySHGetFolderPath)(NULL, CSIDL_STARTUP|CSIDL_FLAG_CREATE, NULL, SHGFP_TYPE_CURRENT, szBuffer))) { TRACE(TEXT("IsConfigStartupBOINC: pfnMySHGetFolderPath - CSIDL_STARTUP - '%s'\n"), szBuffer); if (SUCCEEDED(StringCchCatN(szBuffer, 512, TEXT("\\BOINC.lnk"), 10))) { TRACE(TEXT("IsConfigStartupBOINC: Final pfnMySHGetFolderPath - CSIDL_STARTUP - '%s'\n"), szBuffer); bCheckFileExists = TRUE; } else { TRACE(TEXT("IsConfigStartupBOINC: FAILED pfnMySHGetFolderPath - CSIDL_STARTUP Append Operation\n")); } } else { TRACE(TEXT("IsConfigStartupBOINC: FAILED pfnMySHGetFolderPath - CSIDL_STARTUP\n")); } } if (bCheckFileExists) { hFileHandle = CreateFile( szBuffer, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); if (INVALID_HANDLE_VALUE != hFileHandle) { TRACE(TEXT("IsConfigStartupBOINC: CreateFile returned a valid handle '%d'\n"), hFileHandle); CloseHandle(hFileHandle); bRetVal = TRUE; } else { TRACE(TEXT("IsConfigStartupBOINC: CreateFile returned INVALID_HANDLE_VALUE - GetLastError() '%d'\n"), GetLastError()); // It could be in the global startup group ZeroMemory( szBuffer, 512 ); bCheckFileExists = FALSE; if ( NULL != pfnMySHGetFolderPath ) { if (SUCCEEDED((pfnMySHGetFolderPath)(NULL, CSIDL_COMMON_STARTUP|CSIDL_FLAG_CREATE, NULL, SHGFP_TYPE_CURRENT, szBuffer))) { TRACE(TEXT("IsConfigStartupBOINC: pfnMySHGetFolderPath - CSIDL_COMMON_STARTUP - '%s'\n"), szBuffer); if (SUCCEEDED(StringCchCatN(szBuffer, 512, TEXT("\\BOINC.lnk"), 10))) { TRACE(TEXT("IsConfigStartupBOINC: Final pfnMySHGetFolderPath - CSIDL_COMMON_STARTUP - '%s'\n"), szBuffer); bCheckFileExists = TRUE; } else { TRACE(TEXT("IsConfigStartupBOINC: FAILED pfnMySHGetFolderPath - CSIDL_COMMON_STARTUP Append Operation\n")); } } else { TRACE(TEXT("IsConfigStartupBOINC: FAILED pfnMySHGetFolderPath - CSIDL_COMMON_STARTUP\n")); } } if (bCheckFileExists) { hFileHandle = CreateFile( szBuffer, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); if (INVALID_HANDLE_VALUE != hFileHandle) { TRACE(TEXT("IsConfigStartupBOINC: CreateFile returned a valid handle '%d'\n"), hFileHandle); CloseHandle(hFileHandle); bRetVal = TRUE; } else { TRACE(TEXT("IsConfigStartupBOINC: CreateFile returned INVALID_HANDLE_VALUE - GetLastError() '%d'\n"), GetLastError()); } } } } } // Free the dynamically linked to library FreeLibrary(hShell32); TRACE(TEXT("IsConfigStartupBOINC: Returning '%d'\n"), bRetVal); return bRetVal; }