745 lines
19 KiB
C++
745 lines
19 KiB
C++
|
/******************************************************************************
|
||
|
|
||
|
Copyright (c) 2000 Microsoft Corporation
|
||
|
|
||
|
Module Name:
|
||
|
module.cpp
|
||
|
|
||
|
Abstract:
|
||
|
This file contains the implementation of the CServiceModule class, which is
|
||
|
used to handling service-related routines.
|
||
|
|
||
|
Revision History:
|
||
|
Davide Massarenti (Dmassare) 03/14/2000
|
||
|
created
|
||
|
|
||
|
******************************************************************************/
|
||
|
|
||
|
#include "stdafx.h"
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
DWORD dwTimeOut = 8*1000; // time for EXE to be idle before shutting down
|
||
|
const DWORD dwPause = 1000; // time to wait for threads to finish up
|
||
|
|
||
|
CServiceModule _Module;
|
||
|
MPC::NTEvent g_NTEvents;
|
||
|
CMPCConfig g_Config;
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
// These variables control the overriding of properties for debug.
|
||
|
//
|
||
|
bool g_Override_History = false;
|
||
|
UL_HISTORY g_Override_History_Value = UL_HISTORY_NONE;
|
||
|
|
||
|
bool g_Override_Persist = false;
|
||
|
VARIANT_BOOL g_Override_Persist_Value = false;
|
||
|
|
||
|
bool g_Override_Compressed = false;
|
||
|
VARIANT_BOOL g_Override_Compressed_Value = false;
|
||
|
|
||
|
|
||
|
////////////////////////////////////////////////////////////////////////////////
|
||
|
////////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
const WCHAR c_szMessageFile [] = L"%WINDIR%\\PCHealth\\UploadLB\\Binaries\\UploadM.exe";
|
||
|
|
||
|
const WCHAR c_szRegistryLog [] = L"SYSTEM\\CurrentControlSet\\Services\\EventLog\\Application\\UploadM";
|
||
|
const WCHAR c_szRegistryLog_File [] = L"EventMessageFile";
|
||
|
const WCHAR c_szRegistryLog_Flags[] = L"TypesSupported";
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
#ifdef DEBUG
|
||
|
#define DEBUG_REGKEY L"SOFTWARE\\Microsoft\\PCHealth\\UploadM\\Debug"
|
||
|
#define DEBUG_TIMEOUT L"TIMEOUT"
|
||
|
#define DEBUG_BREAKONSTART L"BREAKONSTART"
|
||
|
#define DEBUG_HISTORY L"HISTORY"
|
||
|
#define DEBUG_PERSIST L"PERSIST"
|
||
|
#define DEBUG_COMPRESSED L"COMPRESSED"
|
||
|
|
||
|
void CServiceModule::ReadDebugSettings()
|
||
|
{
|
||
|
__ULT_FUNC_ENTRY( "CServiceModule::ReadDebugSettings" );
|
||
|
|
||
|
HRESULT hr;
|
||
|
MPC::RegKey rkBase;
|
||
|
bool fFound;
|
||
|
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rkBase.SetRoot( HKEY_LOCAL_MACHINE ));
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rkBase.Attach ( DEBUG_REGKEY ));
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rkBase.Exists ( fFound ));
|
||
|
|
||
|
if(fFound)
|
||
|
{
|
||
|
CComVariant vValue;
|
||
|
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rkBase.get_Value( vValue, fFound, DEBUG_BREAKONSTART ));
|
||
|
if(fFound && vValue.vt == VT_I4)
|
||
|
{
|
||
|
if(vValue.lVal == 1) DebugBreak();
|
||
|
if(vValue.lVal == 2) while(vValue.lVal) ::Sleep( 100 );
|
||
|
}
|
||
|
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rkBase.get_Value( vValue, fFound, DEBUG_TIMEOUT ));
|
||
|
if(fFound && vValue.vt == VT_I4)
|
||
|
{
|
||
|
dwTimeOut = 1000 * vValue.lVal;
|
||
|
}
|
||
|
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rkBase.get_Value( vValue, fFound, DEBUG_HISTORY ));
|
||
|
if(fFound && vValue.vt == VT_I4 && vValue.lVal != -1)
|
||
|
{
|
||
|
g_Override_History = true;
|
||
|
g_Override_History_Value = (UL_HISTORY)vValue.lVal;
|
||
|
}
|
||
|
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rkBase.get_Value( vValue, fFound, DEBUG_PERSIST ));
|
||
|
if(fFound && vValue.vt == VT_I4 && vValue.lVal != -1)
|
||
|
{
|
||
|
g_Override_Persist = true;
|
||
|
g_Override_Persist_Value = vValue.lVal ? VARIANT_TRUE : VARIANT_FALSE;
|
||
|
}
|
||
|
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rkBase.get_Value( vValue, fFound, DEBUG_COMPRESSED ));
|
||
|
if(fFound && vValue.vt == VT_I4 && vValue.lVal != -1)
|
||
|
{
|
||
|
g_Override_Compressed = true;
|
||
|
g_Override_Compressed_Value = vValue.lVal ? VARIANT_TRUE : VARIANT_FALSE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
__ULT_FUNC_CLEANUP;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
static const WCHAR s_SvcHost[] = L"Software\\Microsoft\\Windows NT\\CurrentVersion\\Svchost";
|
||
|
|
||
|
static const WCHAR s_Key [] = L"System\\CurrentControlSet\\Services\\%s";
|
||
|
static const WCHAR s_Key2 [] = L"\\Parameters";
|
||
|
static const WCHAR s_Name [] = L"ServiceDll";
|
||
|
static const WCHAR s_Value[] = L"%WINDIR%\\PCHealth\\HelpCtr\\Binaries\\pchsvc.dll";
|
||
|
|
||
|
static HRESULT ServiceHost_Install( LPCWSTR szName, LPCWSTR szGroup )
|
||
|
{
|
||
|
__ULT_FUNC_ENTRY( "ServiceHost_Install" );
|
||
|
|
||
|
HRESULT hr;
|
||
|
|
||
|
|
||
|
//
|
||
|
// Register the message file into the registry.
|
||
|
//
|
||
|
{
|
||
|
MPC::wstring szPath ( c_szMessageFile ); MPC::SubstituteEnvVariables( szPath );
|
||
|
MPC::RegKey rkEventLog;
|
||
|
CComVariant vValue;
|
||
|
|
||
|
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rkEventLog.SetRoot( HKEY_LOCAL_MACHINE, KEY_ALL_ACCESS ));
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rkEventLog.Attach ( c_szRegistryLog ));
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rkEventLog.Create ( ));
|
||
|
|
||
|
vValue = szPath.c_str(); __MPC_EXIT_IF_METHOD_FAILS(hr, rkEventLog.put_Value( vValue, c_szRegistryLog_File ));
|
||
|
vValue = (long)0x1F ; __MPC_EXIT_IF_METHOD_FAILS(hr, rkEventLog.put_Value( vValue, c_szRegistryLog_Flags ));
|
||
|
}
|
||
|
|
||
|
|
||
|
{
|
||
|
WCHAR rgRegPath[_MAX_PATH]; swprintf( rgRegPath, s_Key, szName ); wcscat( rgRegPath, s_Key2 );
|
||
|
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, MPC::RegKey_Value_Write( s_Value, rgRegPath, s_Name, HKEY_LOCAL_MACHINE, true ));
|
||
|
}
|
||
|
|
||
|
{
|
||
|
MPC::RegKey rk;
|
||
|
MPC::WStringList lstValue;
|
||
|
bool fFound;
|
||
|
bool fGot = false;
|
||
|
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rk.SetRoot( HKEY_LOCAL_MACHINE, KEY_ALL_ACCESS ));
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rk.Attach ( s_SvcHost ));
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rk.Create ( ));
|
||
|
|
||
|
|
||
|
if(SUCCEEDED(rk.Read( lstValue, fFound, szGroup )))
|
||
|
{
|
||
|
for(MPC::WStringIterConst it = lstValue.begin(); it != lstValue.end(); it++)
|
||
|
{
|
||
|
if(!MPC::StrICmp( *it, szName ))
|
||
|
{
|
||
|
fGot = true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if(fGot == false)
|
||
|
{
|
||
|
lstValue.push_back( szName );
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rk.Write( lstValue, szGroup ));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
hr = S_OK;
|
||
|
|
||
|
|
||
|
__ULT_FUNC_CLEANUP;
|
||
|
|
||
|
__ULT_FUNC_EXIT(hr);
|
||
|
}
|
||
|
|
||
|
static HRESULT ServiceHost_Uninstall( LPCWSTR szName, LPCWSTR szGroup )
|
||
|
{
|
||
|
__ULT_FUNC_ENTRY( "ServiceHost_Uninstall" );
|
||
|
|
||
|
HRESULT hr;
|
||
|
|
||
|
{
|
||
|
WCHAR rgRegPath[_MAX_PATH]; swprintf( rgRegPath, s_Key, szName );
|
||
|
MPC::RegKey rk;
|
||
|
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rk.SetRoot( HKEY_LOCAL_MACHINE, KEY_ALL_ACCESS ));
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rk.Attach ( rgRegPath ));
|
||
|
(void)rk.Delete( /*fDeep*/true );
|
||
|
}
|
||
|
|
||
|
{
|
||
|
MPC::RegKey rk;
|
||
|
MPC::WStringList lstValue;
|
||
|
bool fFound;
|
||
|
|
||
|
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rk.SetRoot( HKEY_LOCAL_MACHINE, KEY_ALL_ACCESS ));
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rk.Attach ( s_SvcHost ));
|
||
|
|
||
|
if(SUCCEEDED(rk.Read( lstValue, fFound, szGroup )))
|
||
|
{
|
||
|
MPC::WStringIterConst it = lstValue.begin();
|
||
|
bool fGot = false;
|
||
|
|
||
|
while(it != lstValue.end())
|
||
|
{
|
||
|
MPC::WStringIterConst it2 = it++;
|
||
|
|
||
|
if(!MPC::StrICmp( *it2, szName ))
|
||
|
{
|
||
|
lstValue.erase( it2 );
|
||
|
fGot = true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if(fGot)
|
||
|
{
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, rk.Write( lstValue, szGroup ));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
hr = S_OK;
|
||
|
|
||
|
|
||
|
__ULT_FUNC_CLEANUP;
|
||
|
|
||
|
__ULT_FUNC_EXIT(hr);
|
||
|
}
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
CServiceModule::CServiceModule()
|
||
|
{
|
||
|
m_hEventShutdown = NULL; // HANDLE m_hEventShutdown;
|
||
|
m_dwThreadID = 0; // DWORD m_dwThreadID;
|
||
|
m_hMonitor = NULL; // HANDLE m_hMonitor;
|
||
|
m_bActivity = FALSE; // BOOL m_bActivity;
|
||
|
//
|
||
|
m_szServiceName = NULL; // LPCWSTR m_szServiceName;
|
||
|
m_iDisplayName = 0; // UINT m_iDisplayName;
|
||
|
m_iDescription = 0; // UINT m_iDescription;
|
||
|
m_hServiceStatus = NULL; // SERVICE_STATUS_HANDLE m_hServiceStatus;
|
||
|
// SERVICE_STATUS m_status;
|
||
|
m_bService = FALSE; // BOOL m_bService;
|
||
|
|
||
|
::ZeroMemory( &m_status, sizeof( m_status ) );
|
||
|
}
|
||
|
|
||
|
CServiceModule::~CServiceModule()
|
||
|
{
|
||
|
if(m_hEventShutdown) ::CloseHandle( m_hEventShutdown );
|
||
|
if(m_hMonitor ) ::CloseHandle( m_hMonitor );
|
||
|
}
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
LONG CServiceModule::Lock()
|
||
|
{
|
||
|
LONG lCount = CComModule::Lock();
|
||
|
|
||
|
return lCount;
|
||
|
}
|
||
|
|
||
|
LONG CServiceModule::Unlock()
|
||
|
{
|
||
|
LONG lCount = CComModule::Unlock();
|
||
|
|
||
|
if(lCount == 0)
|
||
|
{
|
||
|
m_bActivity = TRUE;
|
||
|
|
||
|
if(m_hEventShutdown) ::SetEvent( m_hEventShutdown ); // tell monitor that we transitioned to zero
|
||
|
}
|
||
|
|
||
|
return lCount;
|
||
|
}
|
||
|
|
||
|
void CServiceModule::MonitorShutdown()
|
||
|
{
|
||
|
while(1)
|
||
|
{
|
||
|
DWORD dwWait;
|
||
|
|
||
|
m_bActivity = FALSE;
|
||
|
dwWait = ::WaitForSingleObject( m_hEventShutdown, dwTimeOut );
|
||
|
|
||
|
if(dwWait == WAIT_OBJECT_0) continue; // We are alive...
|
||
|
|
||
|
if(g_Root.CanContinue()) continue;
|
||
|
|
||
|
//
|
||
|
// If no activity let's really bail.
|
||
|
//
|
||
|
if(m_bActivity == FALSE && m_nLockCnt <= 0)
|
||
|
{
|
||
|
::CoSuspendClassObjects();
|
||
|
|
||
|
if(m_bActivity == FALSE && m_nLockCnt <= 0) break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ForceShutdown();
|
||
|
}
|
||
|
|
||
|
void CServiceModule::ForceShutdown()
|
||
|
{
|
||
|
|
||
|
//
|
||
|
// Tell process to exit.
|
||
|
//
|
||
|
::PostThreadMessage( m_dwThreadID, WM_QUIT, 0, 0 );
|
||
|
}
|
||
|
|
||
|
BOOL CServiceModule::StartMonitor()
|
||
|
{
|
||
|
DWORD dwThreadID;
|
||
|
|
||
|
|
||
|
m_hMonitor = ::CreateThread( NULL, 0, _Monitor, this, 0, &dwThreadID );
|
||
|
if(m_hMonitor == NULL) return FALSE;
|
||
|
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
HRESULT CServiceModule::RegisterServer( BOOL bRegTypeLib, BOOL bService, LPCWSTR szSvcHostGroup )
|
||
|
{
|
||
|
HRESULT hr;
|
||
|
|
||
|
// Remove any previous service since it may point to the incorrect file
|
||
|
Uninstall( szSvcHostGroup );
|
||
|
|
||
|
if(bService)
|
||
|
{
|
||
|
// Create service
|
||
|
Install( szSvcHostGroup );
|
||
|
}
|
||
|
|
||
|
// Add object entries
|
||
|
if(FAILED(hr = CComModule::RegisterServer( TRUE ))) return hr;
|
||
|
|
||
|
if(FAILED(hr = _Module.UpdateRegistryFromResource( IDR_UPLOADMANAGER, TRUE ))) return hr;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
HRESULT CServiceModule::UnregisterServer( LPCWSTR szSvcHostGroup )
|
||
|
{
|
||
|
HRESULT hr;
|
||
|
|
||
|
// Remove service
|
||
|
Uninstall( szSvcHostGroup );
|
||
|
|
||
|
// Remove object entries
|
||
|
if(FAILED(hr = CComModule::UnregisterServer( TRUE ))) return hr;
|
||
|
|
||
|
if(FAILED(hr = _Module.UpdateRegistryFromResource( IDR_UPLOADMANAGER, FALSE ))) return hr;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
void CServiceModule::Init( _ATL_OBJMAP_ENTRY* p, HINSTANCE h, LPCWSTR szServiceName, UINT iDisplayName, UINT iDescription, const GUID* plibid )
|
||
|
{
|
||
|
CComModule::Init( p, h, plibid );
|
||
|
|
||
|
|
||
|
m_szServiceName = szServiceName;
|
||
|
m_iDisplayName = iDisplayName;
|
||
|
m_iDescription = iDescription;
|
||
|
|
||
|
// set up the initial service status
|
||
|
m_hServiceStatus = NULL;
|
||
|
|
||
|
m_status.dwServiceType = SERVICE_WIN32_OWN_PROCESS;
|
||
|
m_status.dwCurrentState = SERVICE_STOPPED;
|
||
|
m_status.dwControlsAccepted = SERVICE_ACCEPT_STOP;
|
||
|
m_status.dwWin32ExitCode = 0;
|
||
|
m_status.dwServiceSpecificExitCode = 0;
|
||
|
m_status.dwCheckPoint = 0;
|
||
|
m_status.dwWaitHint = 0;
|
||
|
}
|
||
|
|
||
|
BOOL CServiceModule::IsInstalled()
|
||
|
{
|
||
|
BOOL bResult = FALSE;
|
||
|
SC_HANDLE hSCM = ::OpenSCManager( NULL, NULL, SC_MANAGER_ALL_ACCESS );
|
||
|
|
||
|
|
||
|
if((hSCM = ::OpenSCManager( NULL, NULL, SC_MANAGER_ALL_ACCESS )))
|
||
|
{
|
||
|
SC_HANDLE hService;
|
||
|
|
||
|
if((hService = ::OpenServiceW( hSCM, m_szServiceName, SERVICE_QUERY_CONFIG )))
|
||
|
{
|
||
|
bResult = TRUE;
|
||
|
|
||
|
::CloseServiceHandle( hService );
|
||
|
}
|
||
|
|
||
|
::CloseServiceHandle( hSCM );
|
||
|
}
|
||
|
|
||
|
return bResult;
|
||
|
}
|
||
|
|
||
|
BOOL CServiceModule::Install( LPCWSTR szSvcHostGroup )
|
||
|
{
|
||
|
BOOL bResult = FALSE;
|
||
|
SC_HANDLE hSCM;
|
||
|
|
||
|
|
||
|
if((hSCM = ::OpenSCManager( NULL, NULL, SC_MANAGER_ALL_ACCESS )))
|
||
|
{
|
||
|
LPCWSTR szDisplayName;
|
||
|
WCHAR rgDisplayName[512];
|
||
|
WCHAR rgDescription[512];
|
||
|
WCHAR rgFilePath [_MAX_PATH];
|
||
|
SC_HANDLE hService;
|
||
|
DWORD dwStartType;
|
||
|
|
||
|
|
||
|
if(szSvcHostGroup)
|
||
|
{
|
||
|
dwStartType = SERVICE_WIN32_SHARE_PROCESS;
|
||
|
|
||
|
swprintf( rgFilePath, L"%%SystemRoot%%\\System32\\svchost.exe -k %s", szSvcHostGroup );
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
dwStartType = SERVICE_WIN32_OWN_PROCESS;
|
||
|
|
||
|
::GetModuleFileNameW( NULL, rgFilePath, _MAX_PATH );
|
||
|
}
|
||
|
|
||
|
if(::LoadStringW( _Module.GetResourceInstance(), m_iDisplayName, rgDisplayName, MAXSTRLEN(rgDisplayName) ) != 0)
|
||
|
{
|
||
|
szDisplayName = rgDisplayName;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
szDisplayName = m_szServiceName;
|
||
|
}
|
||
|
|
||
|
if(::LoadStringW( _Module.GetResourceInstance(), m_iDescription, rgDescription, MAXSTRLEN(rgDescription) ) == 0)
|
||
|
{
|
||
|
rgDescription[0] = 0;
|
||
|
}
|
||
|
|
||
|
hService = ::OpenServiceW( hSCM, m_szServiceName, SERVICE_QUERY_CONFIG );
|
||
|
if(hService == NULL)
|
||
|
{
|
||
|
hService = ::CreateServiceW( hSCM ,
|
||
|
m_szServiceName ,
|
||
|
szDisplayName ,
|
||
|
SERVICE_ALL_ACCESS ,
|
||
|
dwStartType ,
|
||
|
SERVICE_AUTO_START ,
|
||
|
SERVICE_ERROR_NORMAL,
|
||
|
rgFilePath ,
|
||
|
NULL ,
|
||
|
NULL ,
|
||
|
L"RPCSS\0" ,
|
||
|
NULL ,
|
||
|
NULL );
|
||
|
}
|
||
|
|
||
|
if(hService)
|
||
|
{
|
||
|
if(rgDescription[0])
|
||
|
{
|
||
|
SERVICE_DESCRIPTIONW desc; ::ZeroMemory( &desc , sizeof(desc ) );
|
||
|
SERVICE_FAILURE_ACTIONSW recovery; ::ZeroMemory( &recovery, sizeof(recovery) );
|
||
|
SC_ACTION actions[] =
|
||
|
{
|
||
|
{ SC_ACTION_RESTART, 100 },
|
||
|
{ SC_ACTION_RESTART, 100 },
|
||
|
{ SC_ACTION_NONE , 100 },
|
||
|
};
|
||
|
|
||
|
desc.lpDescription = rgDescription;
|
||
|
|
||
|
recovery.dwResetPeriod = 24 * 60 * 60; // 1 day
|
||
|
recovery.cActions = ARRAYSIZE(actions);
|
||
|
recovery.lpsaActions = actions;
|
||
|
|
||
|
::ChangeServiceConfig2W( hService, SERVICE_CONFIG_DESCRIPTION , &desc );
|
||
|
::ChangeServiceConfig2W( hService, SERVICE_CONFIG_FAILURE_ACTIONS, &recovery );
|
||
|
}
|
||
|
|
||
|
if(szSvcHostGroup)
|
||
|
{
|
||
|
if(SUCCEEDED(ServiceHost_Install( m_szServiceName, szSvcHostGroup )))
|
||
|
{
|
||
|
bResult = TRUE;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
bResult = TRUE;
|
||
|
}
|
||
|
|
||
|
::CloseServiceHandle( hService );
|
||
|
}
|
||
|
|
||
|
if(bResult == FALSE)
|
||
|
{
|
||
|
(void)g_NTEvents.LogEvent( EVENTLOG_ERROR_TYPE, UPLOADM_ERR_CANNOTCREATESERVICE, NULL );
|
||
|
}
|
||
|
|
||
|
::CloseServiceHandle( hSCM );
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
(void)g_NTEvents.LogEvent( EVENTLOG_ERROR_TYPE, UPLOADM_ERR_CANNOTOPENSCM, NULL );
|
||
|
}
|
||
|
|
||
|
return bResult;
|
||
|
}
|
||
|
|
||
|
BOOL CServiceModule::Uninstall( LPCWSTR szSvcHostGroup )
|
||
|
{
|
||
|
BOOL bResult = FALSE;
|
||
|
SC_HANDLE hSCM;
|
||
|
|
||
|
|
||
|
if((hSCM = ::OpenSCManager( NULL, NULL, SC_MANAGER_ALL_ACCESS )))
|
||
|
{
|
||
|
SC_HANDLE hService;
|
||
|
|
||
|
if((hService = ::OpenServiceW( hSCM, m_szServiceName, SERVICE_STOP | DELETE )))
|
||
|
{
|
||
|
SERVICE_STATUS status;
|
||
|
|
||
|
::ControlService( hService, SERVICE_CONTROL_STOP, &status );
|
||
|
|
||
|
bResult = ::DeleteService( hService );
|
||
|
if(bResult)
|
||
|
{
|
||
|
::Sleep( 2000 ); // Let the service stop down...
|
||
|
|
||
|
if(szSvcHostGroup)
|
||
|
{
|
||
|
if(FAILED(ServiceHost_Uninstall( m_szServiceName, szSvcHostGroup )))
|
||
|
{
|
||
|
bResult = FALSE;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if(bResult == FALSE)
|
||
|
{
|
||
|
(void)g_NTEvents.LogEvent( EVENTLOG_ERROR_TYPE, UPLOADM_ERR_CANNOTDELETESERVICE, NULL );
|
||
|
}
|
||
|
|
||
|
::CloseServiceHandle( hService );
|
||
|
}
|
||
|
|
||
|
::CloseServiceHandle( hSCM );
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
(void)g_NTEvents.LogEvent( EVENTLOG_ERROR_TYPE, UPLOADM_ERR_CANNOTOPENSCM, NULL );
|
||
|
}
|
||
|
|
||
|
|
||
|
return bResult;
|
||
|
}
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////////////////////////////////
|
||
|
// Service startup and registration
|
||
|
BOOL CServiceModule::Start( BOOL bService )
|
||
|
{
|
||
|
SERVICE_TABLE_ENTRYW st[] =
|
||
|
{
|
||
|
{ (LPWSTR)m_szServiceName, _ServiceMain },
|
||
|
{ NULL , NULL }
|
||
|
};
|
||
|
|
||
|
m_hEventShutdown = ::CreateEvent( NULL, FALSE, FALSE, NULL );
|
||
|
if(m_hEventShutdown == NULL) return FALSE;
|
||
|
|
||
|
if((m_bService = bService) && !::StartServiceCtrlDispatcherW( st ))
|
||
|
{
|
||
|
DWORD dwRes = ::GetLastError();
|
||
|
|
||
|
m_bService = FALSE;
|
||
|
}
|
||
|
|
||
|
if(m_bService == FALSE)
|
||
|
{
|
||
|
if(StartMonitor() == FALSE) return FALSE;
|
||
|
|
||
|
if(FAILED(Run())) return FALSE;
|
||
|
}
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
void CServiceModule::ServiceMain( DWORD dwArgc, LPWSTR lpszArgv[] )
|
||
|
{
|
||
|
// Register the control request handler
|
||
|
m_status.dwCurrentState = SERVICE_START_PENDING;
|
||
|
|
||
|
if((m_hServiceStatus = ::RegisterServiceCtrlHandlerW( m_szServiceName, _Handler )))
|
||
|
{
|
||
|
SetServiceStatus( SERVICE_START_PENDING );
|
||
|
|
||
|
m_status.dwWin32ExitCode = S_OK;
|
||
|
m_status.dwCheckPoint = 0;
|
||
|
m_status.dwWaitHint = 0;
|
||
|
|
||
|
// When the Run function returns, the service has stopped.
|
||
|
Run();
|
||
|
|
||
|
SetServiceStatus( SERVICE_STOPPED );
|
||
|
|
||
|
(void)g_NTEvents.LogEvent( EVENTLOG_INFORMATION_TYPE, UPLOADM_INFO_STOPPED, NULL );
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
(void)g_NTEvents.LogEvent( EVENTLOG_ERROR_TYPE, UPLOADM_ERR_REGISTERHANDLER, NULL );
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
void CServiceModule::Handler( DWORD dwOpcode )
|
||
|
{
|
||
|
switch(dwOpcode)
|
||
|
{
|
||
|
case SERVICE_CONTROL_STOP:
|
||
|
SetServiceStatus( SERVICE_STOP_PENDING );
|
||
|
ForceShutdown();
|
||
|
break;
|
||
|
|
||
|
case SERVICE_CONTROL_PAUSE:
|
||
|
break;
|
||
|
|
||
|
case SERVICE_CONTROL_CONTINUE:
|
||
|
break;
|
||
|
|
||
|
case SERVICE_CONTROL_INTERROGATE:
|
||
|
break;
|
||
|
|
||
|
case SERVICE_CONTROL_SHUTDOWN:
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
(void)g_NTEvents.LogEvent( EVENTLOG_ERROR_TYPE, UPLOADM_ERR_BADSVCREQUEST, NULL );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
HRESULT CServiceModule::Run()
|
||
|
{
|
||
|
__ULT_FUNC_ENTRY( "CServiceModule::Run" );
|
||
|
|
||
|
HRESULT hr;
|
||
|
MSG msg;
|
||
|
|
||
|
|
||
|
m_dwThreadID = ::GetCurrentThreadId();
|
||
|
|
||
|
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, ::CoInitializeEx( NULL, COINIT_MULTITHREADED )); // We need to be a multi-threaded application.
|
||
|
|
||
|
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, RegisterClassObjects( CLSCTX_LOCAL_SERVER | CLSCTX_REMOTE_SERVER, REGCLS_MULTIPLEUSE ));
|
||
|
|
||
|
(void)g_NTEvents.LogEvent( EVENTLOG_INFORMATION_TYPE, UPLOADM_INFO_STARTED, NULL );
|
||
|
if(m_bService)
|
||
|
{
|
||
|
SetServiceStatus( SERVICE_RUNNING );
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
//
|
||
|
// Load the state of the queue.
|
||
|
//
|
||
|
__MPC_EXIT_IF_METHOD_FAILS(hr, g_Root.Init());
|
||
|
|
||
|
while(::GetMessage( &msg, 0, 0, 0 ))
|
||
|
{
|
||
|
::DispatchMessage( &msg );
|
||
|
}
|
||
|
|
||
|
_Module.RevokeClassObjects();
|
||
|
::Sleep( dwPause ); //wait for any threads to finish
|
||
|
|
||
|
|
||
|
hr = S_OK;
|
||
|
|
||
|
__ULT_FUNC_CLEANUP;
|
||
|
|
||
|
__ULT_FUNC_EXIT(hr);
|
||
|
}
|
||
|
|
||
|
void CServiceModule::SetServiceStatus( DWORD dwState )
|
||
|
{
|
||
|
m_status.dwCurrentState = dwState;
|
||
|
|
||
|
::SetServiceStatus( m_hServiceStatus, &m_status );
|
||
|
}
|
||
|
|
||
|
////////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
void WINAPI CServiceModule::_ServiceMain( DWORD dwArgc, LPWSTR* lpszArgv )
|
||
|
{
|
||
|
_Module.ServiceMain( dwArgc, lpszArgv );
|
||
|
}
|
||
|
|
||
|
void WINAPI CServiceModule::_Handler( DWORD dwOpcode )
|
||
|
{
|
||
|
_Module.Handler( dwOpcode );
|
||
|
}
|
||
|
|
||
|
DWORD WINAPI CServiceModule::_Monitor( void* pv )
|
||
|
{
|
||
|
((CServiceModule*)pv)->MonitorShutdown();
|
||
|
|
||
|
return 0;
|
||
|
}
|