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.
 
 
 
 
 
 

1566 lines
46 KiB

#include "shsrvice.h"
#include "service.h"
#include "mischlpr.h"
#include "sfstr.h"
#include "reg.h"
#include "resource.h"
#include "dbg.h"
#include "tfids.h"
#include <dbt.h>
#include <initguid.h>
#include <ioevent.h>
#include <shlwapi.h>
#include <shlwapip.h>
#define ARRAYSIZE(a) (sizeof((a))/sizeof((a)[0]))
#define MAX_EVENTNAME 100
struct CTRLEVENT
{
CTRLEVENT* peventNext;
DWORD dwControl;
DWORD dwEventType;
BYTE rgbEventData[1];
};
const LPWSTR pszSVCHostGroup = TEXT("netsvcs");
///////////////////////////////////////////////////////////////////////////////
//
SERVICE_TABLE_ENTRY CGenericServiceManager::_rgste[] =
{
{ TEXT("ShellHWDetection"), CGenericServiceManager::_ServiceMain },
{ NULL, NULL },
};
CGenericServiceManager::SUBSERVICE CGenericServiceManager::_rgsubservice[] =
{
{ TEXT("Shell.HWEventDetector"), IDS_SHELLHWDETECTION_FRIENDLYNAME,
TEXT("RpcSs\0"), TEXT("ShellSvcGroup"), IDS_SHELLHWDETECTION_DESCRIPTION,
{0} },
};
// "- 1": Last entry of both arrays are NULL terminators
DWORD CGenericServiceManager::_cste =
ARRAYSIZE(CGenericServiceManager::_rgste) - 1;
CRITICAL_SECTION CGenericServiceManager::_cs = {0};
BOOL CGenericServiceManager::_fInitializationDone = FALSE;
ULONG CGenericServiceManager::_cRefCS = 0;
HANDLE CGenericServiceManager::_hEventInitCS = NULL;
BOOL CGenericServiceManager::_fSVCHosted = FALSE;
#ifdef DEBUG
BOOL CGenericServiceManager::_fRunAsService = TRUE;
#endif
///////////////////////////////////////////////////////////////////////////////
//
HRESULT _GetServiceEventName(LPCWSTR pszServiceName, LPWSTR pszEventName,
DWORD cchEventName);
///////////////////////////////////////////////////////////////////////////////
//
// static
BOOL _IsAlreadyInstalled(LPCWSTR pszRegStr, LPCWSTR pszServiceName)
{
LPCWSTR psz = pszRegStr;
BOOL fThere = FALSE;
do
{
if (!lstrcmp(psz, pszServiceName))
{
fThere = TRUE;
break;
}
psz += lstrlen(psz) + 1;
}
while (*psz);
return fThere;
}
HRESULT _UnInstall(LPCWSTR pszServiceName)
{
HRESULT hres = E_FAIL;
SC_HANDLE hSCM = OpenSCManager(NULL, NULL, SC_MANAGER_CONNECT);
// Need to do it for all services
if (hSCM)
{
hres = S_OK;
SC_HANDLE hService = OpenService(hSCM, pszServiceName,
DELETE);
if (hService)
{
DeleteService(hService);
CloseServiceHandle(hService);
}
else
{
// Don't fail, if somebody manually removed the service n
// from the reg, then all n + x services will not uninstall.
hres = S_FALSE;
}
CloseServiceHandle(hSCM);
}
return hres;
}
HRESULT _InstSetSVCHostInfo(LPWSTR pszServiceName)
{
HRESULT hres = E_FAIL;
HKEY hkey;
DWORD dwDisp;
BOOL fAlreadyThere = FALSE;
if (ERROR_SUCCESS == RegCreateKeyEx(HKEY_LOCAL_MACHINE,
TEXT("Software\\Microsoft\\Windows NT\\CurrentVersion\\Svchost"),
0, NULL, REG_OPTION_NON_VOLATILE, MAXIMUM_ALLOWED, NULL, &hkey,
&dwDisp))
{
DWORD cbSize;
DWORD cbSizeNew;
BOOL fEmpty = FALSE;
LPWSTR pszNew;
if (ERROR_SUCCESS != RegQueryValueEx(hkey, pszSVCHostGroup, 0, NULL,
NULL, &cbSize))
{
fEmpty = TRUE;
// Set cbSize to the size of the 2nd NULL terminator
cbSize = sizeof(WCHAR);
}
cbSizeNew = cbSize + (lstrlen(pszServiceName) + 1) * sizeof(WCHAR);
pszNew = (LPWSTR)LocalAlloc(LPTR, cbSizeNew);
if (pszNew)
{
DWORD cbSize2 = cbSizeNew;
hres = S_OK;
if (!fEmpty)
{
if (ERROR_SUCCESS == RegQueryValueEx(hkey, pszSVCHostGroup, 0,
NULL, (PBYTE)pszNew, &cbSize2))
{
if (cbSize2 == cbSize)
{
fAlreadyThere = _IsAlreadyInstalled(pszNew,
pszServiceName);
}
else
{
hres = E_FAIL;
}
}
else
{
hres = E_FAIL;
}
}
else
{
cbSize2 = sizeof(WCHAR);
}
if (SUCCEEDED(hres) && !fAlreadyThere)
{
lstrcpy(pszNew + (cbSize2 / sizeof(WCHAR)) - 1,
pszServiceName);
if (ERROR_SUCCESS != RegSetValueEx(hkey, pszSVCHostGroup, 0,
REG_MULTI_SZ, (PBYTE)pszNew, cbSizeNew))
{
hres = E_FAIL;
}
}
LocalFree((HLOCAL)pszNew);
}
else
{
hres = E_OUTOFMEMORY;
}
// We should have an entry in the SUBSERVICE array for this...
if (SUCCEEDED(hres))
{
HKEY hkey2;
if (ERROR_SUCCESS == RegCreateKeyEx(hkey, pszSVCHostGroup,
0, NULL, REG_OPTION_NON_VOLATILE, MAXIMUM_ALLOWED, NULL,
&hkey2, &dwDisp))
{
DWORD dwSec = 0x00000001;
DWORD cbSec = sizeof(dwSec);
// We set this magic value of 1 and svchost.exe will
// call CoInitializeSecurity for us
if (ERROR_SUCCESS != RegSetValueEx(hkey2,
TEXT("CoInitializeSecurityParam"), 0, REG_DWORD,
(PBYTE)&dwSec, cbSec))
{
hres = E_FAIL;
}
}
RegCloseKey(hkey2);
}
RegCloseKey(hkey);
}
return hres;
}
HRESULT _InstSetParameters(LPWSTR pszServiceName)
{
HKEY hkeyServices;
HRESULT hres = E_FAIL;
if (ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE,
TEXT("System\\CurrentControlSet\\Services\\"), 0,
MAXIMUM_ALLOWED, &hkeyServices))
{
HKEY hkeySvc;
if (ERROR_SUCCESS == RegOpenKeyEx(hkeyServices, pszServiceName, 0,
MAXIMUM_ALLOWED, &hkeySvc))
{
HKEY hkeyParam;
if (ERROR_SUCCESS == RegCreateKeyEx(hkeySvc,
TEXT("Parameters"), 0, NULL,
REG_OPTION_NON_VOLATILE,
MAXIMUM_ALLOWED, NULL, &hkeyParam, NULL))
{
// Watch out! Hard coded path and filename!
WCHAR szServiceDll[] =
TEXT("%SystemRoot%\\System32\\shsvcs.dll");
if (ERROR_SUCCESS == RegSetValueEx(hkeyParam,
TEXT("ServiceDll"), 0, REG_EXPAND_SZ,
(PBYTE)szServiceDll, sizeof(szServiceDll)))
{
WCHAR szServiceMain[] =
TEXT("HardwareDetectionServiceMain");
if (ERROR_SUCCESS == RegSetValueEx(
hkeyParam, TEXT("ServiceMain"), 0,
REG_SZ, (PBYTE)szServiceMain,
sizeof(szServiceMain)))
{
hres = S_OK;
}
}
RegCloseKey(hkeyParam);
}
RegCloseKey(hkeySvc);
}
RegCloseKey(hkeyServices);
}
return hres;
}
// static
HRESULT CGenericServiceManager::UnInstall()
{
HRESULT hr = S_FALSE;
for (DWORD dw = 0; SUCCEEDED(hr) && (dw < _cste); ++dw)
{
hr = _UnInstall(_rgste[dw].lpServiceName);
}
return hr;
}
HRESULT _GetFriendlyStrings(CGenericServiceManager::SUBSERVICE* psubservice,
LPWSTR pszFriendlyName, DWORD cchFriendlyName, LPWSTR pszDescription,
DWORD cchDescription)
{
*pszFriendlyName = 0;
*pszDescription = 0;
HMODULE hmodule = GetModuleHandle(TEXT("shsvcs.dll"));
if (hmodule)
{
LoadString(hmodule, psubservice->uFriendlyName, pszFriendlyName,
cchFriendlyName);
LoadString(hmodule, psubservice->uDescription, pszDescription,
cchDescription);
}
return S_OK;
}
// static
HRESULT CGenericServiceManager::Install()
{
HRESULT hres = S_FALSE;
if (!IsOS(OS_WOW6432))
{
hres = E_FAIL;
SC_HANDLE hSCM = OpenSCManager(NULL, NULL, SC_MANAGER_CREATE_SERVICE);
WCHAR szFileName[MAX_PATH];
DWORD dwStartType = SERVICE_AUTO_START;
if (hSCM)
{
WCHAR szFriendlyName[200];
// Doc says limit is 1024 bytes
WCHAR szDescription[1024 / sizeof(WCHAR)];
// Need to do it for all services
hres = S_OK;
if (!_fSVCHosted)
{
if (GetModuleFileName(NULL, szFileName, ARRAYSIZE(szFileName)))
{
for (DWORD dw = 0; SUCCEEDED(hres) && (dw < _cste); ++dw)
{
_GetFriendlyStrings(&(_rgsubservice[dw]), szFriendlyName,
ARRAYSIZE(szFriendlyName), szDescription,
ARRAYSIZE(szDescription));
SC_HANDLE hService = CreateService(hSCM,
_rgste[dw].lpServiceName,
szFriendlyName,
0,
SERVICE_WIN32_SHARE_PROCESS,
dwStartType, SERVICE_ERROR_IGNORE,
szFileName,
_rgsubservice[dw].pszLoadOrderGroup,
NULL, _rgsubservice[dw].pszDependencies,
NULL, NULL);
if (hService)
{
SERVICE_DESCRIPTION sd;
sd.lpDescription = szDescription;
ChangeServiceConfig2(hService,
SERVICE_CONFIG_DESCRIPTION, &sd);
CloseServiceHandle(hService);
}
else
{
hres = E_FAIL;
}
}
}
}
else
{
for (DWORD dw = 0; SUCCEEDED(hres) && (dw < _cste); ++dw)
{
WCHAR szCmd[MAX_PATH] =
TEXT("%SystemRoot%\\System32\\svchost.exe -k ");
hres = SafeStrCatN(szCmd, pszSVCHostGroup, ARRAYSIZE(szCmd));
if (SUCCEEDED(hres))
{
_GetFriendlyStrings(&(_rgsubservice[dw]), szFriendlyName,
ARRAYSIZE(szFriendlyName), szDescription,
ARRAYSIZE(szDescription));
SC_HANDLE hService = CreateService(hSCM,
_rgste[dw].lpServiceName,
szFriendlyName,
0,
SERVICE_WIN32_SHARE_PROCESS,
dwStartType, SERVICE_ERROR_IGNORE,
szCmd,
_rgsubservice[dw].pszLoadOrderGroup,
NULL, _rgsubservice[dw].pszDependencies,
NULL, NULL);
if (hService)
{
SERVICE_DESCRIPTION sd;
sd.lpDescription = szDescription;
ChangeServiceConfig2(hService,
SERVICE_CONFIG_DESCRIPTION, &sd);
CloseServiceHandle(hService);
hres = _InstSetParameters(_rgste[dw].lpServiceName);
if (SUCCEEDED(hres))
{
hres = _InstSetSVCHostInfo(
_rgste[dw].lpServiceName);
}
}
else
{
if (ERROR_SERVICE_EXISTS == GetLastError())
{
// We had this problem on upgrade. The service is
// already there, so CreateService fails. As a result the
// StartType was not switched from Demand Start to Auto Start.
// The following lines will do just that.
// This code should be expanded for general upgraded cases.
// All the other values in the structure should be passed here.
hService = OpenService(hSCM,
_rgste[dw].lpServiceName, SERVICE_CHANGE_CONFIG);
if (hService)
{
if (ChangeServiceConfig(
hService, // handle to service
SERVICE_NO_CHANGE, // type of service
dwStartType, // when to start service
SERVICE_NO_CHANGE, // severity of start failure
szCmd, // service binary file name
_rgsubservice[dw].pszLoadOrderGroup, // load ordering group name
NULL, // tag identifier
NULL, // array of dependency names
NULL, // account name
NULL, // account password
NULL // display name
))
{
// Do this un upgrade too
hres = _InstSetSVCHostInfo(
_rgste[dw].lpServiceName);
if (SUCCEEDED(hres))
{
hres = _InstSetParameters(_rgste[dw].lpServiceName);
}
}
else
{
hres = E_FAIL;
}
SERVICE_DESCRIPTION sd;
sd.lpDescription = szDescription;
ChangeServiceConfig2(hService,
SERVICE_CONFIG_DESCRIPTION, &sd);
CloseServiceHandle(hService);
}
}
else
{
hres = E_FAIL;
}
}
}
}
}
CloseServiceHandle(hSCM);
}
// We don't need the ShellCOMServer anymore, so let's nuke it away on upgrades
_UnInstall(TEXT("ShellCOMServer"));
// Also remove the following reg entry on upgrade
_RegDeleteValue(HKEY_LOCAL_MACHINE,
TEXT("Software\\Microsoft\\Windows NT\\CurrentVersion\\SvcHost"), TEXT("shsvc"));
}
return hres;
}
// static
HRESULT CGenericServiceManager::Init()
{
return _Init();
}
// static
HRESULT CGenericServiceManager::Cleanup()
{
return _Cleanup();
}
///////////////////////////////////////////////////////////////////////////////
// Private
// static
HRESULT CGenericServiceManager::_Init()
{
HRESULT hres = E_FAIL;
ASSERT(ARRAYSIZE(_rgste) == (ARRAYSIZE(_rgsubservice) + 1));
if (!InterlockedCompareExchange((LONG*)&_fInitializationDone, TRUE,
FALSE))
{
_hEventInitCS = CreateEvent(NULL, TRUE, FALSE,
TEXT("CGenericServiceManager__Init"));
if (_hEventInitCS)
{
InitializeCriticalSection(&_cs);
SetEvent(_hEventInitCS);
hres = S_OK;
}
}
else
{
HANDLE hEvent = CreateEvent(NULL, TRUE, FALSE,
TEXT("CGenericServiceManager__Init"));
if (hEvent)
{
WaitForSingleObject(hEvent, INFINITE);
CloseHandle(hEvent);
hres = S_OK;
}
}
if (SUCCEEDED(hres))
{
::InterlockedIncrement((LONG*)&_cRefCS);
// Per thread
hres = CoInitializeEx(NULL, COINIT_MULTITHREADED | COINIT_DISABLE_OLE1DDE);
}
return hres;
}
// static
HRESULT CGenericServiceManager::_Cleanup()
{
if (_fInitializationDone)
{
ULONG cRef = ::InterlockedDecrement((LONG*)&_cRefCS);
if (!cRef)
{
DeleteCriticalSection(&_cs);
CloseHandle(_hEventInitCS);
}
// Per thread
CoUninitialize();
InterlockedExchange((LONG*)&_fInitializationDone, FALSE);
}
return S_OK;
}
__inline void _TraceServiceCode(DWORD
#ifdef DEBUG
dwControl
#endif
)
{
#ifdef DEBUG
LPWSTR pszControl = TEXT("Unknown");
switch (dwControl)
{
case SERVICE_CONTROL_STOP: pszControl = TEXT("SERVICE_CONTROL_STOP"); break;
case SERVICE_CONTROL_PAUSE: pszControl = TEXT("SERVICE_CONTROL_PAUSE"); break;
case SERVICE_CONTROL_CONTINUE: pszControl = TEXT("SERVICE_CONTROL_CONTINUE"); break;
case SERVICE_CONTROL_INTERROGATE: pszControl = TEXT("SERVICE_CONTROL_INTERROGATE"); break;
case SERVICE_CONTROL_SHUTDOWN: pszControl = TEXT("SERVICE_CONTROL_SHUTDOWN"); break;
case SERVICE_CONTROL_PARAMCHANGE: pszControl = TEXT("SERVICE_CONTROL_PARAMCHANGE"); break;
case SERVICE_CONTROL_NETBINDADD: pszControl = TEXT("SERVICE_CONTROL_NETBINDADD"); break;
case SERVICE_CONTROL_NETBINDREMOVE: pszControl = TEXT("SERVICE_CONTROL_NETBINDREMOVE"); break;
case SERVICE_CONTROL_NETBINDENABLE: pszControl = TEXT("SERVICE_CONTROL_NETBINDENABLE"); break;
case SERVICE_CONTROL_NETBINDDISABLE: pszControl = TEXT("SERVICE_CONTROL_NETBINDDISABLE"); break;
case SERVICE_CONTROL_DEVICEEVENT: pszControl = TEXT("SERVICE_CONTROL_DEVICEEVENT"); break;
case SERVICE_CONTROL_HARDWAREPROFILECHANGE: pszControl = TEXT("SERVICE_CONTROL_HARDWAREPROFILECHANGE"); break;
case SERVICE_CONTROL_POWEREVENT: pszControl = TEXT("SERVICE_CONTROL_POWEREVENT"); break;
case SERVICE_CONTROL_SESSIONCHANGE: pszControl = TEXT("SERVICE_CONTROL_SESSIONCHANGE"); break;
}
TRACE(TF_SERVICE, TEXT("Received Service Control code: %s (0x%08X)"),
pszControl, dwControl);
#endif
}
///////////////////////////////////////////////////////////////////////////////
// Private
//static
DWORD WINAPI CGenericServiceManager::_ServiceHandler(DWORD dwControl,
DWORD dwEventType, LPVOID pvEventData, LPVOID lpContext)
{
// We don't want to deny any request, so return NO_ERROR whatever happens
DWORD dwRet = NO_ERROR;
// This is called on the main thread not the thread of the specific
// service, so keep it short and sweet
SERVICEENTRY* pse = (SERVICEENTRY*)lpContext;
if (pse)
{
BOOL fProcess = FALSE;
BOOL fSynch = FALSE;
HRESULT hres = S_OK;
switch (dwControl)
{
case SERVICE_CONTROL_STOP:
case SERVICE_CONTROL_SHUTDOWN:
TRACE(TF_SERVICE, TEXT("Received SERVICE_CONTROL_SHUTDOWN or STOP, will skip all other terminating events"));
if (!pse->_fSkipTerminatingEvents)
{
fProcess = TRUE;
pse->_fSkipTerminatingEvents = TRUE;
}
else
{
TRACE(TF_SERVICE, TEXT("Skipping terminating event"));
}
break;
case SERVICE_CONTROL_INTERROGATE:
// Special case SERVICE_CONTROL_INTERROGATE. We don't really need the
// IService impl to process this. The service will also be more
// responsive this way. The state can be queried in the middle of
// execution.
TRACE(TF_SERVICE, TEXT("Received SERVICE_CONTROL_INTERROGATE"));
_SetServiceStatus(pse);
break;
default:
if (!pse->_fSkipTerminatingEvents)
{
fProcess = TRUE;
hres = _EventNeedsToBeProcessedSynchronously(dwControl,
dwEventType, pvEventData, pse, &fSynch);
}
break;
}
if (SUCCEEDED(hres) && fProcess)
{
_TraceServiceCode(dwControl);
EnterCriticalSection(&_cs);
hres = _QueueEvent(pse, dwControl, dwEventType, pvEventData);
if (SUCCEEDED(hres))
{
// Let the service process events
SetEvent(pse->_hEventRelinquishControl);
ResetEvent(pse->_hEventSynchProcessing);
}
LeaveCriticalSection(&_cs);
if (SUCCEEDED(hres) && fSynch)
{
Sleep(0);
TRACE(TF_SERVICE,
TEXT("=========== Processing SYNCHRONOUSLY ==========="));
// We have to wait before we return... (at most 20 sec)
DWORD dwWait = WaitForSingleObject(pse->_hEventSynchProcessing,
20000);
if (WAIT_TIMEOUT == dwWait)
{
TRACE(TF_SERVICE,
TEXT("=========== WAIT TIMED OUT ==========="));
}
TRACE(TF_SERVICE,
TEXT("=========== FINISHED processing SYNCHRONOUSLY ==========="));
#ifdef DEBUG
// If we get the notifs from a windowproc, return TRUE,
// or else we'll deny any request to remove, lock, ...
if (!_fRunAsService)
{
if (SERVICE_CONTROL_DEVICEEVENT == dwControl)
{
dwRet = TRUE;
}
}
#endif
}
}
}
return dwRet;
}
//static
void WINAPI CGenericServiceManager::_ServiceMain(DWORD cArg, LPWSTR* ppszArgs)
{
SERVICEENTRY* pse;
LPCWSTR pszServiceName = *ppszArgs;
HRESULT hres = _Init();
if (SUCCEEDED(hres))
{
hres = _InitServiceEntry(pszServiceName, &pse);
if (SUCCEEDED(hres))
{
hres = _RegisterServiceCtrlHandler(pszServiceName, pse);
if (SUCCEEDED(hres))
{
pse->_servicestatus.dwServiceType = SERVICE_WIN32_SHARE_PROCESS;
pse->_servicestatus.dwCurrentState = SERVICE_START_PENDING;
_SetServiceStatus(pse);
hres = _CreateIService(pszServiceName, &(pse->_pservice));
if (SUCCEEDED(hres))
{
WCHAR szEventName[MAX_EVENTNAME];
hres = _GetServiceEventName(pszServiceName, szEventName,
ARRAYSIZE(szEventName));
if (SUCCEEDED(hres))
{
hres = pse->_pservice->InitMinimum(cArg, ppszArgs,
szEventName, &(pse->_servicestatus.dwControlsAccepted),
&(pse->_fWantsDeviceEvents));
if (SUCCEEDED(hres))
{
hres = _HandleWantsDeviceEvents(pszServiceName,
pse->_fWantsDeviceEvents);
if (SUCCEEDED(hres))
{
if (pse->_fWantsDeviceEvents)
{
hres = pse->_pservice->InitDeviceEventHandler(
pse->_ssh);
}
if (SUCCEEDED(hres))
{
pse->_servicestatus.dwCurrentState =
SERVICE_RUNNING;
_SetServiceStatus(pse);
hres = pse->_pservice->InitFinal();
if (SUCCEEDED(hres))
{
do
{
hres = pse->_pservice->Run();
if (SUCCEEDED(hres))
{
// The service has finished its business or it
// relinquished control because
// _hEventRelinquishControl is set
//
// If it relinquished control because of a
// service control event, then let's process it
DWORD dwWait = WaitForSingleObject(
pse->_hEventRelinquishControl,
INFINITE);
TRACE(TF_SERVICEDETAILED,
TEXT("WaitForSingleObj returned with: 0x%08X"),
dwWait);
if (WAIT_OBJECT_0 == dwWait)
{
// Process all Service Control codes
// received.
hres = _ProcessServiceControlCodes(
pse);
}
else
{
hres = E_FAIL;
}
}
}
while (SUCCEEDED(hres) &&
(SERVICE_STOPPED !=
pse->_servicestatus.dwCurrentState));
}
}
}
// What do we do with hres?
}
}
}
}
else
{
#ifdef DEBUG
TRACE(TF_SERVICEDETAILED,
TEXT("%s: _RegisterServiceCtrlHandler FAILED: 0x%08X"),
pse->_szServiceName, hres);
#endif
}
_CleanupServiceEntry(pse);
}
if (SUCCEEDED(hres) &&
(SERVICE_STOPPED == pse->_servicestatus.dwCurrentState))
{
_SetServiceStatus(pse);
}
_Cleanup();
}
TRACE(TF_SERVICE, TEXT("Exiting _ServiceMain for Service: %s"), pszServiceName);
}
//static
HRESULT CGenericServiceManager::_ProcessServiceControlCodes(SERVICEENTRY* pse)
{
HRESULT hres;
BOOL fEndLoop = FALSE;
TRACE(TF_SERVICEDETAILED, TEXT("Entered _ProcessServiceControlCodes"));
do
{
CTRLEVENT* pevent;
EnterCriticalSection(&_cs);
TRACE(TF_SERVICEDETAILED, TEXT("Entered _ProcessServiceControlCodes' Critical Section"));
hres = _DeQueueEvent(pse, &pevent);
TRACE(TF_SERVICE, TEXT("DeQueued Event: 0x%08X"), hres);
if (!pse->_peventQueueHead)
{
ASSERT(!pse->_cEvents);
fEndLoop = TRUE;
ResetEvent(pse->_hEventRelinquishControl);
}
LeaveCriticalSection(&_cs);
if (SUCCEEDED(hres))
{
///////////////////////////////////////////////////////////////////
// If we're here then we should have received some service control,
// or the IService decided it had nothing more to process
//
TRACE(TF_SERVICEDETAILED, TEXT("Will call _HandleServiceControls (dwControl = 0x%08X)"),
pevent->dwControl);
hres = _HandleServiceControls(pse, pevent->dwControl,
pevent->dwEventType, (PVOID)(pevent->rgbEventData));
if (SERVICE_CONTROL_DEVICEEVENT == pevent->dwControl)
{
// Never ever return something else than NO_ERROR (S_OK) for these
hres = NO_ERROR;
}
TRACE(TF_SERVICE, TEXT("_HandleServiceControls returned: 0x%08X"), hres);
LocalFree((HLOCAL)pevent);
}
if (fEndLoop)
{
TRACE(TF_SERVICEDETAILED, TEXT("Resetting RelinquishEvent"));
SetEvent(pse->_hEventSynchProcessing);
}
}
while (!fEndLoop && SUCCEEDED(hres));
TRACE(TF_SERVICEDETAILED, TEXT("Exiting _ProcessServiceControlCodes"));
return hres;
}
#pragma warning(push)
// FALSE positive below: fPending
#pragma warning(disable : 4701)
//static
HRESULT CGenericServiceManager::_HandleServiceControls(SERVICEENTRY* pse,
DWORD dwControl, DWORD dwEventType, PVOID pvEventData)
{
HRESULT hres = _HandlePreState(pse, dwControl);
TRACE(TF_SERVICEDETAILED, TEXT("_HandlePreState returned: 0x%08X, status: 0x%08X"), hres,
pse->_servicestatus.dwCurrentState);
if (SUCCEEDED(hres))
{
if (S_OK == hres)
{
switch (dwControl)
{
case SERVICE_CONTROL_STOP:
case SERVICE_CONTROL_PAUSE:
case SERVICE_CONTROL_CONTINUE:
case SERVICE_CONTROL_SHUTDOWN:
{
BOOL fPending;
do
{
// This will return S_FALSE if it's pending
hres = pse->_pservice->HandleServiceControl(dwControl,
&(pse->_servicestatus.dwWaitHint));
if (SUCCEEDED(hres))
{
if (S_FALSE == hres)
{
ASSERT(pse->_servicestatus.dwWaitHint);
fPending = TRUE;
}
else
{
fPending = FALSE;
}
TRACE(TF_SERVICE, TEXT("Will call _HandlePostState (fPending = %d)"),
fPending);
hres = _HandlePostState(pse, dwControl, fPending);
TRACE(TF_SERVICE, TEXT("_HandlePostState returned: 0x%08X, status: 0x%08X"),
hres, pse->_servicestatus.dwCurrentState);
}
}
while (SUCCEEDED(hres) && fPending);
break;
}
case SERVICE_CONTROL_DEVICEEVENT:
TRACE(TF_SERVICE, TEXT("Received SERVICE_CONTROL_DEVICEEVENT"));
if (!(pse->_fSkipTerminatingEvents))
{
hres = pse->_pservice->HandleDeviceEvent(dwEventType,
pvEventData);
}
else
{
hres = S_OK;
}
break;
case SERVICE_CONTROL_SESSIONCHANGE:
TRACE(TF_SERVICE, TEXT("Received: SERVICE_CONTROL_SESSIONCHANGE"));
if (!(pse->_fSkipTerminatingEvents))
{
hres = pse->_pservice->HandleSessionChange(dwEventType, pvEventData);
}
else
{
hres = S_OK;
}
break;
default:
TRACE(TF_SERVICE, TEXT("Received unhandled service control"));
hres = S_FALSE;
break;
}
}
}
return hres;
}
#pragma warning(pop)
// static
HRESULT CGenericServiceManager::_GetServiceIndex(LPCWSTR pszServiceName,
DWORD* pdw)
{
HRESULT hres = E_FAIL;
ASSERT(pszServiceName);
ASSERT(pdw);
for (DWORD dw = 0; FAILED(hres) && (dw < _cste); ++dw)
{
if (!lstrcmp(pszServiceName, _rgste[dw].lpServiceName))
{
// Found it
*pdw = dw;
hres = S_OK;
}
}
return hres;
}
HRESULT CGenericServiceManager::_GetServiceCLSID(LPCWSTR pszServiceName,
CLSID* pclsid)
{
ASSERT(pszServiceName);
ASSERT(pclsid);
DWORD dw;
HRESULT hres = _GetServiceIndex(pszServiceName, &dw);
if (SUCCEEDED(hres))
{
// Found it
hres = CLSIDFromProgID(_rgsubservice[dw].pszProgID, pclsid);
}
return hres;
}
HRESULT CGenericServiceManager::_CreateIService(LPCWSTR pszServiceName,
IService** ppservice)
{
CLSID clsid;
HRESULT hres = _GetServiceCLSID(pszServiceName, &clsid);
*ppservice = NULL;
if (SUCCEEDED(hres))
{
hres = CoCreateInstance(clsid, NULL, CLSCTX_INPROC_SERVER,
IID_PPV_ARG(IService, ppservice));
}
return hres;
}
HRESULT CGenericServiceManager::_InitServiceEntry(LPCWSTR pszServiceName,
SERVICEENTRY** ppse)
{
DWORD dw;
HRESULT hres = _GetServiceIndex(pszServiceName, &dw);
if (SUCCEEDED(hres))
{
WCHAR szEventName[MAX_EVENTNAME];
*ppse = &(_rgsubservice[dw].se);
ZeroMemory(*ppse, sizeof(**ppse));
hres = _GetServiceEventName(pszServiceName, szEventName,
ARRAYSIZE(szEventName));
if (SUCCEEDED(hres))
{
(*ppse)->_hEventRelinquishControl = CreateEvent(NULL, TRUE, FALSE,
szEventName);
if (!((*ppse)->_hEventRelinquishControl))
{
hres = E_FAIL;
}
if (SUCCEEDED(hres))
{
(*ppse)->_hEventSynchProcessing = CreateEvent(NULL, TRUE, TRUE,
NULL);
if (!((*ppse)->_hEventSynchProcessing))
{
hres = E_FAIL;
}
}
if (FAILED(hres))
{
_CleanupServiceEntry(*ppse);
}
}
}
return hres;
}
HRESULT CGenericServiceManager::_CleanupServiceEntry(SERVICEENTRY* pse)
{
if (pse->_pservice)
{
pse->_pservice->Release();
}
if (pse->_hEventRelinquishControl)
{
CloseHandle(pse->_hEventRelinquishControl);
}
if (pse->_hEventSynchProcessing)
{
CloseHandle(pse->_hEventSynchProcessing);
}
return S_OK;
}
//static
HRESULT CGenericServiceManager::_HandlePreState(SERVICEENTRY* pse,
DWORD dwControl)
{
HRESULT hres;
BOOL fSetServiceStatus = TRUE;
// _HandleServiceControls will loop until we are not in a pending state.
// All incoming ctrl events are queued, and will be processed on this same
// thread, so we should never enter this fct in a pending state.
ASSERT(SERVICE_STOP_PENDING != pse->_servicestatus.dwCurrentState);
ASSERT(SERVICE_START_PENDING != pse->_servicestatus.dwCurrentState);
ASSERT(SERVICE_CONTINUE_PENDING != pse->_servicestatus.dwCurrentState);
ASSERT(SERVICE_PAUSE_PENDING != pse->_servicestatus.dwCurrentState);
// Should have been processed in _ServiceHandler
ASSERT(SERVICE_CONTROL_INTERROGATE != dwControl);
// We cleanup a bit. If the request is incompatible with the current state
// then we return S_FALSE to instruct _HandleServiceControls to not call
// the IService impl for nothing.
switch (dwControl)
{
case SERVICE_CONTROL_STOP:
switch (pse->_servicestatus.dwCurrentState)
{
case SERVICE_STOPPED:
hres = S_FALSE;
break;
case SERVICE_RUNNING:
case SERVICE_PAUSED:
default:
pse->_servicestatus.dwCurrentState = SERVICE_STOP_PENDING;
hres = S_OK;
break;
}
break;
case SERVICE_CONTROL_PAUSE:
ASSERT(SERVICE_STOPPED != pse->_servicestatus.dwCurrentState);
switch (pse->_servicestatus.dwCurrentState)
{
case SERVICE_PAUSED:
hres = S_FALSE;
break;
case SERVICE_STOPPED:
// Weird, think about it...
hres = S_FALSE;
break;
case SERVICE_RUNNING:
default:
pse->_servicestatus.dwCurrentState = SERVICE_PAUSE_PENDING;
hres = S_OK;
break;
}
break;
case SERVICE_CONTROL_CONTINUE:
ASSERT(SERVICE_STOPPED != pse->_servicestatus.dwCurrentState);
switch (pse->_servicestatus.dwCurrentState)
{
case SERVICE_RUNNING:
hres = S_FALSE;
break;
case SERVICE_STOPPED:
// Weird, think about it...
hres = S_FALSE;
break;
case SERVICE_PAUSED:
default:
pse->_servicestatus.dwCurrentState = SERVICE_CONTINUE_PENDING;
hres = S_OK;
break;
}
break;
case SERVICE_CONTROL_SHUTDOWN:
fSetServiceStatus = FALSE;
hres = S_OK;
break;
case SERVICE_CONTROL_DEVICEEVENT:
fSetServiceStatus = FALSE;
if (pse->_fWantsDeviceEvents)
{
hres = S_OK;
}
else
{
hres = S_FALSE;
}
break;
case SERVICE_CONTROL_SESSIONCHANGE:
fSetServiceStatus = FALSE;
hres = S_OK;
break;
default:
hres = S_FALSE;
break;
}
if (fSetServiceStatus)
{
_SetServiceStatus(pse);
}
return hres;
}
//static
HRESULT CGenericServiceManager::_HandlePostState(SERVICEENTRY* pse,
DWORD dwControl, BOOL fPending)
{
HRESULT hres = S_FALSE;
// All incoming ctrl events are queued, and will be processed on this same
// thread, so if we are pending, the dwControl should be compatible with
// our current pending state. We call _SetServiceStatus to update the
// dwWaitHint.
// We should already be in a pending state. This should have been set
// by _HandlePreState. Just make sure of this.
// Should have been processed in _ServiceHandler
ASSERT(SERVICE_CONTROL_INTERROGATE != dwControl);
switch (dwControl)
{
case SERVICE_CONTROL_STOP:
ASSERT(SERVICE_STOP_PENDING == pse->_servicestatus.dwCurrentState);
if (!fPending)
{
pse->_servicestatus.dwCurrentState = SERVICE_STOPPED;
}
break;
case SERVICE_CONTROL_PAUSE:
ASSERT(SERVICE_PAUSE_PENDING ==
pse->_servicestatus.dwCurrentState);
if (!fPending)
{
pse->_servicestatus.dwCurrentState = SERVICE_PAUSED;
}
break;
case SERVICE_CONTROL_CONTINUE:
ASSERT(SERVICE_CONTINUE_PENDING ==
pse->_servicestatus.dwCurrentState);
if (!fPending)
{
pse->_servicestatus.dwCurrentState = SERVICE_RUNNING;
}
break;
case SERVICE_CONTROL_SHUTDOWN:
ASSERT(!fPending);
pse->_servicestatus.dwCurrentState = SERVICE_STOPPED;
break;
}
if (SERVICE_STOPPED != pse->_servicestatus.dwCurrentState)
{
_SetServiceStatus(pse);
}
return hres;
}
// static
HRESULT CGenericServiceManager::_EventNeedsToBeProcessedSynchronously(
DWORD dwControl, DWORD dwEventType, LPVOID pvEventData, SERVICEENTRY*,
BOOL* pfBool)
{
*pfBool = FALSE;
if (SERVICE_CONTROL_DEVICEEVENT == dwControl)
{
if (pvEventData)
{
DEV_BROADCAST_HDR* dbhdr = (DEV_BROADCAST_HDR*)pvEventData;
if (DBT_DEVTYP_HANDLE == dbhdr->dbch_devicetype)
{
if (DBT_DEVICEQUERYREMOVE == dwEventType)
{
TRACE(TF_SERVICE, TEXT("Received DBT_DEVICEQUERYREMOVE"));
*pfBool = TRUE;
}
else
{
if (DBT_CUSTOMEVENT == dwEventType)
{
DEV_BROADCAST_HANDLE* pdbh =
(DEV_BROADCAST_HANDLE*)dbhdr;
if ((GUID_IO_VOLUME_LOCK == pdbh->dbch_eventguid))
{
TRACE(TF_SERVICE, TEXT("------------Received GUID_IO_VOLUME_LOCK------------"));
}
if ((GUID_IO_VOLUME_LOCK_FAILED == pdbh->dbch_eventguid))
{
TRACE(TF_SERVICE, TEXT("------------Received GUID_IO_VOLUME_LOCK_FAILED------------"));
}
if ((GUID_IO_VOLUME_UNLOCK == pdbh->dbch_eventguid))
{
TRACE(TF_SERVICE, TEXT("------------Received GUID_IO_VOLUME_UNLOCK------------"));
}
if ((GUID_IO_VOLUME_LOCK == pdbh->dbch_eventguid) ||
(GUID_IO_VOLUME_LOCK_FAILED == pdbh->dbch_eventguid) ||
(GUID_IO_VOLUME_UNLOCK == pdbh->dbch_eventguid))
{
*pfBool = TRUE;
}
}
}
}
}
}
return S_OK;
}
// static
HRESULT CGenericServiceManager::_MakeEvent(DWORD dwControl, DWORD dwEventType,
PVOID pvEventData, CTRLEVENT** ppevent)
{
HRESULT hres = S_OK;
DWORD cbSize = sizeof(CTRLEVENT);
CTRLEVENT* pevent;
if (SERVICE_CONTROL_DEVICEEVENT == dwControl)
{
if (pvEventData)
{
cbSize += ((DEV_BROADCAST_HDR*)pvEventData)->dbch_size;
}
}
pevent = (CTRLEVENT*)LocalAlloc(LPTR, cbSize);
if (pevent)
{
// Payload
pevent->dwControl = dwControl;
pevent->dwEventType = dwEventType;
*ppevent = pevent;
if (cbSize > sizeof(CTRLEVENT))
{
if (pvEventData)
{
CopyMemory(pevent->rgbEventData, pvEventData,
cbSize - sizeof(CTRLEVENT));
}
}
}
else
{
hres = E_OUTOFMEMORY;
}
return hres;
}
// static
HRESULT CGenericServiceManager::_QueueEvent(SERVICEENTRY* pse, DWORD dwControl,
DWORD dwEventType, PVOID pvEventData)
{
CTRLEVENT* pevent;
HRESULT hres = _MakeEvent(dwControl, dwEventType, pvEventData, &pevent);
if (SUCCEEDED(hres))
{
// We add at tail, remove at head
// Prev: closer to head
// Next: closer to tail
pevent->peventNext = NULL;
if (pse->_peventQueueTail)
{
ASSERT(!(pse->_peventQueueTail->peventNext));
pse->_peventQueueTail->peventNext = pevent;
}
pse->_peventQueueTail = pevent;
if (!pse->_peventQueueHead)
{
pse->_peventQueueHead = pse->_peventQueueTail;
}
#ifdef DEBUG
++(pse->_cEvents);
if (1 == pse->_cEvents)
{
ASSERT(pse->_peventQueueHead == pse->_peventQueueTail);
}
else
{
if (0 == pse->_cEvents)
{
ASSERT(!pse->_peventQueueHead && !pse->_peventQueueTail);
}
else
{
ASSERT(pse->_peventQueueHead && pse->_peventQueueTail &&
(pse->_peventQueueHead != pse->_peventQueueTail));
}
}
#endif
}
else
{
hres = E_OUTOFMEMORY;
}
return hres;
}
// static
HRESULT CGenericServiceManager::_DeQueueEvent(SERVICEENTRY* pse,
CTRLEVENT** ppevent)
{
ASSERT(pse->_peventQueueHead);
// We add at tail, remove at head
// Prev: closer to head
// Next: closer to tail
CTRLEVENT* peventRet = pse->_peventQueueHead;
CTRLEVENT* peventNewHead = peventRet->peventNext;
// Any elem left after removing head?
if (!peventNewHead)
{
// No
pse->_peventQueueTail = NULL;
}
pse->_peventQueueHead = peventNewHead;
peventRet->peventNext = NULL;
*ppevent = peventRet;
#ifdef DEBUG
--(pse->_cEvents);
if (1 == pse->_cEvents)
{
ASSERT(pse->_peventQueueHead == pse->_peventQueueTail);
}
else
{
if (0 == pse->_cEvents)
{
ASSERT(!pse->_peventQueueHead && !pse->_peventQueueTail);
}
else
{
ASSERT(pse->_peventQueueHead && pse->_peventQueueTail &&
(pse->_peventQueueHead != pse->_peventQueueTail));
}
}
#endif
return S_OK;
}
HRESULT _GetServiceEventName(LPCWSTR pszServiceName, LPWSTR pszEventName,
DWORD cchEventName)
{
LPWSTR pszNext;
DWORD cchLeft;
HRESULT hres = SafeStrCpyNEx(pszEventName, pszServiceName, cchEventName,
&pszNext, &cchLeft);
if (SUCCEEDED(hres))
{
hres = SafeStrCpyN(pszNext, TEXT("'sEvent"), cchLeft);
}
return hres;
}