Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

744 lines
19 KiB

  1. /******************************************************************************
  2. Copyright (c) 2000 Microsoft Corporation
  3. Module Name:
  4. module.cpp
  5. Abstract:
  6. This file contains the implementation of the CServiceModule class, which is
  7. used to handling service-related routines.
  8. Revision History:
  9. Davide Massarenti (Dmassare) 03/14/2000
  10. created
  11. ******************************************************************************/
  12. #include "stdafx.h"
  13. /////////////////////////////////////////////////////////////////////////////
  14. DWORD dwTimeOut = 8*1000; // time for EXE to be idle before shutting down
  15. const DWORD dwPause = 1000; // time to wait for threads to finish up
  16. CServiceModule _Module;
  17. MPC::NTEvent g_NTEvents;
  18. CMPCConfig g_Config;
  19. /////////////////////////////////////////////////////////////////////////////
  20. //
  21. // These variables control the overriding of properties for debug.
  22. //
  23. bool g_Override_History = false;
  24. UL_HISTORY g_Override_History_Value = UL_HISTORY_NONE;
  25. bool g_Override_Persist = false;
  26. VARIANT_BOOL g_Override_Persist_Value = false;
  27. bool g_Override_Compressed = false;
  28. VARIANT_BOOL g_Override_Compressed_Value = false;
  29. ////////////////////////////////////////////////////////////////////////////////
  30. ////////////////////////////////////////////////////////////////////////////////
  31. const WCHAR c_szMessageFile [] = L"%WINDIR%\\PCHealth\\UploadLB\\Binaries\\UploadM.exe";
  32. const WCHAR c_szRegistryLog [] = L"SYSTEM\\CurrentControlSet\\Services\\EventLog\\Application\\UploadM";
  33. const WCHAR c_szRegistryLog_File [] = L"EventMessageFile";
  34. const WCHAR c_szRegistryLog_Flags[] = L"TypesSupported";
  35. /////////////////////////////////////////////////////////////////////////////
  36. /////////////////////////////////////////////////////////////////////////////
  37. #ifdef DEBUG
  38. #define DEBUG_REGKEY L"SOFTWARE\\Microsoft\\PCHealth\\UploadM\\Debug"
  39. #define DEBUG_TIMEOUT L"TIMEOUT"
  40. #define DEBUG_BREAKONSTART L"BREAKONSTART"
  41. #define DEBUG_HISTORY L"HISTORY"
  42. #define DEBUG_PERSIST L"PERSIST"
  43. #define DEBUG_COMPRESSED L"COMPRESSED"
  44. void CServiceModule::ReadDebugSettings()
  45. {
  46. __ULT_FUNC_ENTRY( "CServiceModule::ReadDebugSettings" );
  47. HRESULT hr;
  48. MPC::RegKey rkBase;
  49. bool fFound;
  50. __MPC_EXIT_IF_METHOD_FAILS(hr, rkBase.SetRoot( HKEY_LOCAL_MACHINE ));
  51. __MPC_EXIT_IF_METHOD_FAILS(hr, rkBase.Attach ( DEBUG_REGKEY ));
  52. __MPC_EXIT_IF_METHOD_FAILS(hr, rkBase.Exists ( fFound ));
  53. if(fFound)
  54. {
  55. CComVariant vValue;
  56. __MPC_EXIT_IF_METHOD_FAILS(hr, rkBase.get_Value( vValue, fFound, DEBUG_BREAKONSTART ));
  57. if(fFound && vValue.vt == VT_I4)
  58. {
  59. if(vValue.lVal == 1) DebugBreak();
  60. if(vValue.lVal == 2) while(vValue.lVal) ::Sleep( 100 );
  61. }
  62. __MPC_EXIT_IF_METHOD_FAILS(hr, rkBase.get_Value( vValue, fFound, DEBUG_TIMEOUT ));
  63. if(fFound && vValue.vt == VT_I4)
  64. {
  65. dwTimeOut = 1000 * vValue.lVal;
  66. }
  67. __MPC_EXIT_IF_METHOD_FAILS(hr, rkBase.get_Value( vValue, fFound, DEBUG_HISTORY ));
  68. if(fFound && vValue.vt == VT_I4 && vValue.lVal != -1)
  69. {
  70. g_Override_History = true;
  71. g_Override_History_Value = (UL_HISTORY)vValue.lVal;
  72. }
  73. __MPC_EXIT_IF_METHOD_FAILS(hr, rkBase.get_Value( vValue, fFound, DEBUG_PERSIST ));
  74. if(fFound && vValue.vt == VT_I4 && vValue.lVal != -1)
  75. {
  76. g_Override_Persist = true;
  77. g_Override_Persist_Value = vValue.lVal ? VARIANT_TRUE : VARIANT_FALSE;
  78. }
  79. __MPC_EXIT_IF_METHOD_FAILS(hr, rkBase.get_Value( vValue, fFound, DEBUG_COMPRESSED ));
  80. if(fFound && vValue.vt == VT_I4 && vValue.lVal != -1)
  81. {
  82. g_Override_Compressed = true;
  83. g_Override_Compressed_Value = vValue.lVal ? VARIANT_TRUE : VARIANT_FALSE;
  84. }
  85. }
  86. __ULT_FUNC_CLEANUP;
  87. }
  88. #endif
  89. /////////////////////////////////////////////////////////////////////////////
  90. static const WCHAR s_SvcHost[] = L"Software\\Microsoft\\Windows NT\\CurrentVersion\\Svchost";
  91. static const WCHAR s_Key [] = L"System\\CurrentControlSet\\Services\\%s";
  92. static const WCHAR s_Key2 [] = L"\\Parameters";
  93. static const WCHAR s_Name [] = L"ServiceDll";
  94. static const WCHAR s_Value[] = L"%WINDIR%\\PCHealth\\HelpCtr\\Binaries\\pchsvc.dll";
  95. static HRESULT ServiceHost_Install( LPCWSTR szName, LPCWSTR szGroup )
  96. {
  97. __ULT_FUNC_ENTRY( "ServiceHost_Install" );
  98. HRESULT hr;
  99. //
  100. // Register the message file into the registry.
  101. //
  102. {
  103. MPC::wstring szPath ( c_szMessageFile ); MPC::SubstituteEnvVariables( szPath );
  104. MPC::RegKey rkEventLog;
  105. CComVariant vValue;
  106. __MPC_EXIT_IF_METHOD_FAILS(hr, rkEventLog.SetRoot( HKEY_LOCAL_MACHINE, KEY_ALL_ACCESS ));
  107. __MPC_EXIT_IF_METHOD_FAILS(hr, rkEventLog.Attach ( c_szRegistryLog ));
  108. __MPC_EXIT_IF_METHOD_FAILS(hr, rkEventLog.Create ( ));
  109. vValue = szPath.c_str(); __MPC_EXIT_IF_METHOD_FAILS(hr, rkEventLog.put_Value( vValue, c_szRegistryLog_File ));
  110. vValue = (long)0x1F ; __MPC_EXIT_IF_METHOD_FAILS(hr, rkEventLog.put_Value( vValue, c_szRegistryLog_Flags ));
  111. }
  112. {
  113. WCHAR rgRegPath[_MAX_PATH]; swprintf( rgRegPath, s_Key, szName ); wcscat( rgRegPath, s_Key2 );
  114. __MPC_EXIT_IF_METHOD_FAILS(hr, MPC::RegKey_Value_Write( s_Value, rgRegPath, s_Name, HKEY_LOCAL_MACHINE, true ));
  115. }
  116. {
  117. MPC::RegKey rk;
  118. MPC::WStringList lstValue;
  119. bool fFound;
  120. bool fGot = false;
  121. __MPC_EXIT_IF_METHOD_FAILS(hr, rk.SetRoot( HKEY_LOCAL_MACHINE, KEY_ALL_ACCESS ));
  122. __MPC_EXIT_IF_METHOD_FAILS(hr, rk.Attach ( s_SvcHost ));
  123. __MPC_EXIT_IF_METHOD_FAILS(hr, rk.Create ( ));
  124. if(SUCCEEDED(rk.Read( lstValue, fFound, szGroup )))
  125. {
  126. for(MPC::WStringIterConst it = lstValue.begin(); it != lstValue.end(); it++)
  127. {
  128. if(!MPC::StrICmp( *it, szName ))
  129. {
  130. fGot = true;
  131. break;
  132. }
  133. }
  134. }
  135. if(fGot == false)
  136. {
  137. lstValue.push_back( szName );
  138. __MPC_EXIT_IF_METHOD_FAILS(hr, rk.Write( lstValue, szGroup ));
  139. }
  140. }
  141. hr = S_OK;
  142. __ULT_FUNC_CLEANUP;
  143. __ULT_FUNC_EXIT(hr);
  144. }
  145. static HRESULT ServiceHost_Uninstall( LPCWSTR szName, LPCWSTR szGroup )
  146. {
  147. __ULT_FUNC_ENTRY( "ServiceHost_Uninstall" );
  148. HRESULT hr;
  149. {
  150. WCHAR rgRegPath[_MAX_PATH]; swprintf( rgRegPath, s_Key, szName );
  151. MPC::RegKey rk;
  152. __MPC_EXIT_IF_METHOD_FAILS(hr, rk.SetRoot( HKEY_LOCAL_MACHINE, KEY_ALL_ACCESS ));
  153. __MPC_EXIT_IF_METHOD_FAILS(hr, rk.Attach ( rgRegPath ));
  154. (void)rk.Delete( /*fDeep*/true );
  155. }
  156. {
  157. MPC::RegKey rk;
  158. MPC::WStringList lstValue;
  159. bool fFound;
  160. __MPC_EXIT_IF_METHOD_FAILS(hr, rk.SetRoot( HKEY_LOCAL_MACHINE, KEY_ALL_ACCESS ));
  161. __MPC_EXIT_IF_METHOD_FAILS(hr, rk.Attach ( s_SvcHost ));
  162. if(SUCCEEDED(rk.Read( lstValue, fFound, szGroup )))
  163. {
  164. MPC::WStringIterConst it = lstValue.begin();
  165. bool fGot = false;
  166. while(it != lstValue.end())
  167. {
  168. MPC::WStringIterConst it2 = it++;
  169. if(!MPC::StrICmp( *it2, szName ))
  170. {
  171. lstValue.erase( it2 );
  172. fGot = true;
  173. }
  174. }
  175. if(fGot)
  176. {
  177. __MPC_EXIT_IF_METHOD_FAILS(hr, rk.Write( lstValue, szGroup ));
  178. }
  179. }
  180. }
  181. hr = S_OK;
  182. __ULT_FUNC_CLEANUP;
  183. __ULT_FUNC_EXIT(hr);
  184. }
  185. /////////////////////////////////////////////////////////////////////////////
  186. CServiceModule::CServiceModule()
  187. {
  188. m_hEventShutdown = NULL; // HANDLE m_hEventShutdown;
  189. m_dwThreadID = 0; // DWORD m_dwThreadID;
  190. m_hMonitor = NULL; // HANDLE m_hMonitor;
  191. m_bActivity = FALSE; // BOOL m_bActivity;
  192. //
  193. m_szServiceName = NULL; // LPCWSTR m_szServiceName;
  194. m_iDisplayName = 0; // UINT m_iDisplayName;
  195. m_iDescription = 0; // UINT m_iDescription;
  196. m_hServiceStatus = NULL; // SERVICE_STATUS_HANDLE m_hServiceStatus;
  197. // SERVICE_STATUS m_status;
  198. m_bService = FALSE; // BOOL m_bService;
  199. ::ZeroMemory( &m_status, sizeof( m_status ) );
  200. }
  201. CServiceModule::~CServiceModule()
  202. {
  203. if(m_hEventShutdown) ::CloseHandle( m_hEventShutdown );
  204. if(m_hMonitor ) ::CloseHandle( m_hMonitor );
  205. }
  206. /////////////////////////////////////////////////////////////////////////////
  207. LONG CServiceModule::Lock()
  208. {
  209. LONG lCount = CComModule::Lock();
  210. return lCount;
  211. }
  212. LONG CServiceModule::Unlock()
  213. {
  214. LONG lCount = CComModule::Unlock();
  215. if(lCount == 0)
  216. {
  217. m_bActivity = TRUE;
  218. if(m_hEventShutdown) ::SetEvent( m_hEventShutdown ); // tell monitor that we transitioned to zero
  219. }
  220. return lCount;
  221. }
  222. void CServiceModule::MonitorShutdown()
  223. {
  224. while(1)
  225. {
  226. DWORD dwWait;
  227. m_bActivity = FALSE;
  228. dwWait = ::WaitForSingleObject( m_hEventShutdown, dwTimeOut );
  229. if(dwWait == WAIT_OBJECT_0) continue; // We are alive...
  230. if(g_Root.CanContinue()) continue;
  231. //
  232. // If no activity let's really bail.
  233. //
  234. if(m_bActivity == FALSE && m_nLockCnt <= 0)
  235. {
  236. ::CoSuspendClassObjects();
  237. if(m_bActivity == FALSE && m_nLockCnt <= 0) break;
  238. }
  239. }
  240. ForceShutdown();
  241. }
  242. void CServiceModule::ForceShutdown()
  243. {
  244. //
  245. // Tell process to exit.
  246. //
  247. ::PostThreadMessage( m_dwThreadID, WM_QUIT, 0, 0 );
  248. }
  249. BOOL CServiceModule::StartMonitor()
  250. {
  251. DWORD dwThreadID;
  252. m_hMonitor = ::CreateThread( NULL, 0, _Monitor, this, 0, &dwThreadID );
  253. if(m_hMonitor == NULL) return FALSE;
  254. return TRUE;
  255. }
  256. /////////////////////////////////////////////////////////////////////////////
  257. HRESULT CServiceModule::RegisterServer( BOOL bRegTypeLib, BOOL bService, LPCWSTR szSvcHostGroup )
  258. {
  259. HRESULT hr;
  260. // Remove any previous service since it may point to the incorrect file
  261. Uninstall( szSvcHostGroup );
  262. if(bService)
  263. {
  264. // Create service
  265. Install( szSvcHostGroup );
  266. }
  267. // Add object entries
  268. if(FAILED(hr = CComModule::RegisterServer( TRUE ))) return hr;
  269. if(FAILED(hr = _Module.UpdateRegistryFromResource( IDR_UPLOADMANAGER, TRUE ))) return hr;
  270. return S_OK;
  271. }
  272. HRESULT CServiceModule::UnregisterServer( LPCWSTR szSvcHostGroup )
  273. {
  274. HRESULT hr;
  275. // Remove service
  276. Uninstall( szSvcHostGroup );
  277. // Remove object entries
  278. if(FAILED(hr = CComModule::UnregisterServer( TRUE ))) return hr;
  279. if(FAILED(hr = _Module.UpdateRegistryFromResource( IDR_UPLOADMANAGER, FALSE ))) return hr;
  280. return S_OK;
  281. }
  282. void CServiceModule::Init( _ATL_OBJMAP_ENTRY* p, HINSTANCE h, LPCWSTR szServiceName, UINT iDisplayName, UINT iDescription, const GUID* plibid )
  283. {
  284. CComModule::Init( p, h, plibid );
  285. m_szServiceName = szServiceName;
  286. m_iDisplayName = iDisplayName;
  287. m_iDescription = iDescription;
  288. // set up the initial service status
  289. m_hServiceStatus = NULL;
  290. m_status.dwServiceType = SERVICE_WIN32_OWN_PROCESS;
  291. m_status.dwCurrentState = SERVICE_STOPPED;
  292. m_status.dwControlsAccepted = SERVICE_ACCEPT_STOP;
  293. m_status.dwWin32ExitCode = 0;
  294. m_status.dwServiceSpecificExitCode = 0;
  295. m_status.dwCheckPoint = 0;
  296. m_status.dwWaitHint = 0;
  297. }
  298. BOOL CServiceModule::IsInstalled()
  299. {
  300. BOOL bResult = FALSE;
  301. SC_HANDLE hSCM = ::OpenSCManager( NULL, NULL, SC_MANAGER_ALL_ACCESS );
  302. if((hSCM = ::OpenSCManager( NULL, NULL, SC_MANAGER_ALL_ACCESS )))
  303. {
  304. SC_HANDLE hService;
  305. if((hService = ::OpenServiceW( hSCM, m_szServiceName, SERVICE_QUERY_CONFIG )))
  306. {
  307. bResult = TRUE;
  308. ::CloseServiceHandle( hService );
  309. }
  310. ::CloseServiceHandle( hSCM );
  311. }
  312. return bResult;
  313. }
  314. BOOL CServiceModule::Install( LPCWSTR szSvcHostGroup )
  315. {
  316. BOOL bResult = FALSE;
  317. SC_HANDLE hSCM;
  318. if((hSCM = ::OpenSCManager( NULL, NULL, SC_MANAGER_ALL_ACCESS )))
  319. {
  320. LPCWSTR szDisplayName;
  321. WCHAR rgDisplayName[512];
  322. WCHAR rgDescription[512];
  323. WCHAR rgFilePath [_MAX_PATH];
  324. SC_HANDLE hService;
  325. DWORD dwStartType;
  326. if(szSvcHostGroup)
  327. {
  328. dwStartType = SERVICE_WIN32_SHARE_PROCESS;
  329. swprintf( rgFilePath, L"%%SystemRoot%%\\System32\\svchost.exe -k %s", szSvcHostGroup );
  330. }
  331. else
  332. {
  333. dwStartType = SERVICE_WIN32_OWN_PROCESS;
  334. ::GetModuleFileNameW( NULL, rgFilePath, _MAX_PATH );
  335. }
  336. if(::LoadStringW( _Module.GetResourceInstance(), m_iDisplayName, rgDisplayName, MAXSTRLEN(rgDisplayName) ) != 0)
  337. {
  338. szDisplayName = rgDisplayName;
  339. }
  340. else
  341. {
  342. szDisplayName = m_szServiceName;
  343. }
  344. if(::LoadStringW( _Module.GetResourceInstance(), m_iDescription, rgDescription, MAXSTRLEN(rgDescription) ) == 0)
  345. {
  346. rgDescription[0] = 0;
  347. }
  348. hService = ::OpenServiceW( hSCM, m_szServiceName, SERVICE_QUERY_CONFIG );
  349. if(hService == NULL)
  350. {
  351. hService = ::CreateServiceW( hSCM ,
  352. m_szServiceName ,
  353. szDisplayName ,
  354. SERVICE_ALL_ACCESS ,
  355. dwStartType ,
  356. SERVICE_AUTO_START ,
  357. SERVICE_ERROR_NORMAL,
  358. rgFilePath ,
  359. NULL ,
  360. NULL ,
  361. L"RPCSS\0" ,
  362. NULL ,
  363. NULL );
  364. }
  365. if(hService)
  366. {
  367. if(rgDescription[0])
  368. {
  369. SERVICE_DESCRIPTIONW desc; ::ZeroMemory( &desc , sizeof(desc ) );
  370. SERVICE_FAILURE_ACTIONSW recovery; ::ZeroMemory( &recovery, sizeof(recovery) );
  371. SC_ACTION actions[] =
  372. {
  373. { SC_ACTION_RESTART, 100 },
  374. { SC_ACTION_RESTART, 100 },
  375. { SC_ACTION_NONE , 100 },
  376. };
  377. desc.lpDescription = rgDescription;
  378. recovery.dwResetPeriod = 24 * 60 * 60; // 1 day
  379. recovery.cActions = ARRAYSIZE(actions);
  380. recovery.lpsaActions = actions;
  381. ::ChangeServiceConfig2W( hService, SERVICE_CONFIG_DESCRIPTION , &desc );
  382. ::ChangeServiceConfig2W( hService, SERVICE_CONFIG_FAILURE_ACTIONS, &recovery );
  383. }
  384. if(szSvcHostGroup)
  385. {
  386. if(SUCCEEDED(ServiceHost_Install( m_szServiceName, szSvcHostGroup )))
  387. {
  388. bResult = TRUE;
  389. }
  390. }
  391. else
  392. {
  393. bResult = TRUE;
  394. }
  395. ::CloseServiceHandle( hService );
  396. }
  397. if(bResult == FALSE)
  398. {
  399. (void)g_NTEvents.LogEvent( EVENTLOG_ERROR_TYPE, UPLOADM_ERR_CANNOTCREATESERVICE, NULL );
  400. }
  401. ::CloseServiceHandle( hSCM );
  402. }
  403. else
  404. {
  405. (void)g_NTEvents.LogEvent( EVENTLOG_ERROR_TYPE, UPLOADM_ERR_CANNOTOPENSCM, NULL );
  406. }
  407. return bResult;
  408. }
  409. BOOL CServiceModule::Uninstall( LPCWSTR szSvcHostGroup )
  410. {
  411. BOOL bResult = FALSE;
  412. SC_HANDLE hSCM;
  413. if((hSCM = ::OpenSCManager( NULL, NULL, SC_MANAGER_ALL_ACCESS )))
  414. {
  415. SC_HANDLE hService;
  416. if((hService = ::OpenServiceW( hSCM, m_szServiceName, SERVICE_STOP | DELETE )))
  417. {
  418. SERVICE_STATUS status;
  419. ::ControlService( hService, SERVICE_CONTROL_STOP, &status );
  420. bResult = ::DeleteService( hService );
  421. if(bResult)
  422. {
  423. ::Sleep( 2000 ); // Let the service stop down...
  424. if(szSvcHostGroup)
  425. {
  426. if(FAILED(ServiceHost_Uninstall( m_szServiceName, szSvcHostGroup )))
  427. {
  428. bResult = FALSE;
  429. }
  430. }
  431. }
  432. if(bResult == FALSE)
  433. {
  434. (void)g_NTEvents.LogEvent( EVENTLOG_ERROR_TYPE, UPLOADM_ERR_CANNOTDELETESERVICE, NULL );
  435. }
  436. ::CloseServiceHandle( hService );
  437. }
  438. ::CloseServiceHandle( hSCM );
  439. }
  440. else
  441. {
  442. (void)g_NTEvents.LogEvent( EVENTLOG_ERROR_TYPE, UPLOADM_ERR_CANNOTOPENSCM, NULL );
  443. }
  444. return bResult;
  445. }
  446. //////////////////////////////////////////////////////////////////////////////////////////////
  447. // Service startup and registration
  448. BOOL CServiceModule::Start( BOOL bService )
  449. {
  450. SERVICE_TABLE_ENTRYW st[] =
  451. {
  452. { (LPWSTR)m_szServiceName, _ServiceMain },
  453. { NULL , NULL }
  454. };
  455. m_hEventShutdown = ::CreateEvent( NULL, FALSE, FALSE, NULL );
  456. if(m_hEventShutdown == NULL) return FALSE;
  457. if((m_bService = bService) && !::StartServiceCtrlDispatcherW( st ))
  458. {
  459. DWORD dwRes = ::GetLastError();
  460. m_bService = FALSE;
  461. }
  462. if(m_bService == FALSE)
  463. {
  464. if(StartMonitor() == FALSE) return FALSE;
  465. if(FAILED(Run())) return FALSE;
  466. }
  467. return TRUE;
  468. }
  469. void CServiceModule::ServiceMain( DWORD dwArgc, LPWSTR lpszArgv[] )
  470. {
  471. // Register the control request handler
  472. m_status.dwCurrentState = SERVICE_START_PENDING;
  473. if((m_hServiceStatus = ::RegisterServiceCtrlHandlerW( m_szServiceName, _Handler )))
  474. {
  475. SetServiceStatus( SERVICE_START_PENDING );
  476. m_status.dwWin32ExitCode = S_OK;
  477. m_status.dwCheckPoint = 0;
  478. m_status.dwWaitHint = 0;
  479. // When the Run function returns, the service has stopped.
  480. Run();
  481. SetServiceStatus( SERVICE_STOPPED );
  482. (void)g_NTEvents.LogEvent( EVENTLOG_INFORMATION_TYPE, UPLOADM_INFO_STOPPED, NULL );
  483. }
  484. else
  485. {
  486. (void)g_NTEvents.LogEvent( EVENTLOG_ERROR_TYPE, UPLOADM_ERR_REGISTERHANDLER, NULL );
  487. }
  488. }
  489. void CServiceModule::Handler( DWORD dwOpcode )
  490. {
  491. switch(dwOpcode)
  492. {
  493. case SERVICE_CONTROL_STOP:
  494. SetServiceStatus( SERVICE_STOP_PENDING );
  495. ForceShutdown();
  496. break;
  497. case SERVICE_CONTROL_PAUSE:
  498. break;
  499. case SERVICE_CONTROL_CONTINUE:
  500. break;
  501. case SERVICE_CONTROL_INTERROGATE:
  502. break;
  503. case SERVICE_CONTROL_SHUTDOWN:
  504. break;
  505. default:
  506. (void)g_NTEvents.LogEvent( EVENTLOG_ERROR_TYPE, UPLOADM_ERR_BADSVCREQUEST, NULL );
  507. }
  508. }
  509. HRESULT CServiceModule::Run()
  510. {
  511. __ULT_FUNC_ENTRY( "CServiceModule::Run" );
  512. HRESULT hr;
  513. MSG msg;
  514. m_dwThreadID = ::GetCurrentThreadId();
  515. __MPC_EXIT_IF_METHOD_FAILS(hr, ::CoInitializeEx( NULL, COINIT_MULTITHREADED )); // We need to be a multi-threaded application.
  516. __MPC_EXIT_IF_METHOD_FAILS(hr, RegisterClassObjects( CLSCTX_LOCAL_SERVER | CLSCTX_REMOTE_SERVER, REGCLS_MULTIPLEUSE ));
  517. (void)g_NTEvents.LogEvent( EVENTLOG_INFORMATION_TYPE, UPLOADM_INFO_STARTED, NULL );
  518. if(m_bService)
  519. {
  520. SetServiceStatus( SERVICE_RUNNING );
  521. }
  522. //
  523. // Load the state of the queue.
  524. //
  525. __MPC_EXIT_IF_METHOD_FAILS(hr, g_Root.Init());
  526. while(::GetMessage( &msg, 0, 0, 0 ))
  527. {
  528. ::DispatchMessage( &msg );
  529. }
  530. _Module.RevokeClassObjects();
  531. ::Sleep( dwPause ); //wait for any threads to finish
  532. hr = S_OK;
  533. __ULT_FUNC_CLEANUP;
  534. __ULT_FUNC_EXIT(hr);
  535. }
  536. void CServiceModule::SetServiceStatus( DWORD dwState )
  537. {
  538. m_status.dwCurrentState = dwState;
  539. ::SetServiceStatus( m_hServiceStatus, &m_status );
  540. }
  541. ////////////////////////////////////////////////////////////////////////////////
  542. void WINAPI CServiceModule::_ServiceMain( DWORD dwArgc, LPWSTR* lpszArgv )
  543. {
  544. _Module.ServiceMain( dwArgc, lpszArgv );
  545. }
  546. void WINAPI CServiceModule::_Handler( DWORD dwOpcode )
  547. {
  548. _Module.Handler( dwOpcode );
  549. }
  550. DWORD WINAPI CServiceModule::_Monitor( void* pv )
  551. {
  552. ((CServiceModule*)pv)->MonitorShutdown();
  553. return 0;
  554. }