blob: 5bbea94f8feebef091cea1619636acd5441d537f [file] [log] [blame]
/*
* Windows Service related function definitions
* By Raju Krishnappa(raju_krishnappa@yahoo.com)
*
*/
#ifdef WIN32
#include <windows.h>
#include <tchar.h>
#include <stdio.h> /* sprintf */
#include <process.h> /* beginthreadex */
#include <net-snmp/library/winservice.h>
#ifdef mingw32 /* MinGW doesn't fully support exception handling. */
#define TRY if(1)
#define LEAVE goto labelFIN
#define FINALLY do { \
labelFIN: \
} while(0); if(1)
#else
#define TRY __try
#define LEAVE __leave
#define FINALLY __finally
#endif /* mingw32 */
/*
* External global variables used here
*/
/*
* Application Name
* This should be declared by the application, which wants to register as
* windows service
*/
extern LPTSTR g_szAppName;
/*
* Declare global variable
*/
/*
* Flag to indicate whether process is running as Service
*/
BOOL g_fRunningAsService = FALSE;
/*
* Variable to maintain Current Service status
*/
static SERVICE_STATUS ServiceStatus;
/*
* Service Handle
*/
static SERVICE_STATUS_HANDLE hServiceStatus = 0L;
/*
* Service Table Entry
*/
SERVICE_TABLE_ENTRY ServiceTableEntry[] = {
{NULL, ServiceMain}, /* Service Main function */
{NULL, NULL}
};
/*
* Handle to Thread, to implement Pause, Resume and Stop functions
*/
static HANDLE hServiceThread = NULL; /* Thread Handle */
/*
* Holds calling partys Function Entry point, that should start
* when entering service mode
*/
static INT (*ServiceEntryPoint) (INT Argc, LPTSTR Argv[]) = 0L;
/*
* To hold Stop Function address, to be called when STOP request
* received from the SCM
*/
static VOID (*StopFunction) (VOID) = 0L;
/*
* To register as Windows Service with SCM(Service Control Manager)
* Input - Service Name, Service Display Name,Service Description and
* Service startup arguments
*/
VOID
RegisterService (LPCTSTR lpszServiceName, LPCTSTR lpszServiceDisplayName,
LPCTSTR lpszServiceDescription,
InputParams * StartUpArg) /* Startup argument to the service */
{
TCHAR szServicePath[MAX_PATH]; /* To hold module File name */
TCHAR MsgErrorString[MAX_STR_SIZE]; /* Message or Error string */
TCHAR szServiceCommand[MAX_PATH + 9]; /* Command to execute */
SC_HANDLE hSCManager = NULL;
SC_HANDLE hService = NULL;
TCHAR szRegAppLogKey[] =
"SYSTEM\\CurrentControlSet\\Services\\EventLog\\Application\\";
TCHAR szRegKey[512];
HKEY hKey = NULL; /* Key to registry entry */
HKEY hParamKey = NULL; /* To store startup parameters */
DWORD dwData; /* Type of logging supported */
DWORD i, j; /* Loop variables */
GetModuleFileName (NULL, szServicePath, MAX_PATH);
TRY
{
/*
* Open Service Control Manager handle
*/
hSCManager = OpenSCManager (NULL, NULL, SC_MANAGER_CREATE_SERVICE);
if (hSCManager == NULL)
{
DisplayError (_T ("Can't open SCM"));
LEAVE;
}
/*
* Generate the Command to be executed by SCM
*/
_stprintf (szServiceCommand, "%s %s", szServicePath, _T ("-service"));
/*
* Create the Desired service
*/
hService = CreateService (hSCManager, lpszServiceName, lpszServiceDisplayName,
SERVICE_ALL_ACCESS, SERVICE_WIN32_OWN_PROCESS,
SERVICE_AUTO_START, SERVICE_ERROR_NORMAL, szServiceCommand,
NULL, /* load-order group */
NULL, /* group member tag */
NULL, /* dependencies */
NULL, /* account */
NULL); /* password */
if (hService == NULL)
{
_stprintf (MsgErrorString, "%s %s",
_T ("Can't Create Service"), lpszServiceDisplayName);
DisplayError (MsgErrorString);
LEAVE;
}
/*
* Create registry entries for EventLog
*/
/*
* Create registry Application event log key
*/
_tcscpy (szRegKey, szRegAppLogKey);
_tcscat (szRegKey, lpszServiceName);
/*
* Create registry key
*/
if (RegCreateKey (HKEY_LOCAL_MACHINE, szRegKey, &hKey) != ERROR_SUCCESS)
{
_stprintf (MsgErrorString, "%s %s",
_T ("Unable to create registry entries"), lpszServiceDisplayName);
DisplayError (MsgErrorString);
LEAVE;
}
/*
* Add Event ID message file name to the 'EventMessageFile' subkey
*/
RegSetValueEx (hKey, "EventMessageFile", 0, REG_EXPAND_SZ,
(CONST BYTE *) szServicePath,
_tcslen (szServicePath) + sizeof (TCHAR));
/*
* Set the supported types flags.
*/
dwData = EVENTLOG_ERROR_TYPE | EVENTLOG_WARNING_TYPE | EVENTLOG_INFORMATION_TYPE;
RegSetValueEx (hKey, "TypesSupported", 0, REG_DWORD,
(CONST BYTE *) & dwData, sizeof (DWORD));
/*
* Close Registry key
*/
RegCloseKey (hKey);
/*
* Set Service Description String and save startup parameters if present
*/
if (lpszServiceDescription != NULL || StartUpArg->Argc > 2)
{
/*
* Create Registry Key path
*/
_tcscpy (szRegKey, _T ("SYSTEM\\CurrentControlSet\\Services\\"));
_tcscat (szRegKey, g_szAppName);
hKey = NULL;
/*
* Open Registry key using Create and Set access.
*/
if (RegOpenKeyEx (HKEY_LOCAL_MACHINE, szRegKey, 0, KEY_WRITE,
&hKey) != ERROR_SUCCESS)
{
_stprintf (MsgErrorString, "%s %s",
_T ("Unable to create registry entries"),
lpszServiceDisplayName);
DisplayError (MsgErrorString);
LEAVE;
}
/*
* Create description subkey and the set value
*/
if (lpszServiceDescription != NULL)
{
if (RegSetValueEx (hKey, "Description", 0, REG_SZ,
(CONST BYTE *) lpszServiceDescription,
_tcslen (lpszServiceDescription) +
sizeof (TCHAR)) != ERROR_SUCCESS)
{
_stprintf (MsgErrorString, "%s %s",
_T ("Unable to create registry entries"),
lpszServiceDisplayName);
DisplayError (MsgErrorString);
LEAVE;
};
}
/*
* Save startup arguments if they are present
*/
if (StartUpArg->Argc > 2)
{
/*
* Create Subkey parameters
*/
if (RegCreateKeyEx
(hKey, "Parameters", 0, NULL,
REG_OPTION_NON_VOLATILE, KEY_WRITE, NULL,
&hParamKey, NULL) != ERROR_SUCCESS)
{
_stprintf (MsgErrorString, "%s %s",
_T ("Unable to create registry entries"),
lpszServiceDisplayName);
DisplayError (MsgErrorString);
LEAVE;
}
/*
* Save parameters
*/
/*
* Loop through arguments
*/
for (i = 2, j = 1; i < StartUpArg->Argc; i++, j++)
{
_stprintf (szRegKey, "%s%d", _T ("Param"), j);
/*
* Create registry key
*/
if (RegSetValueEx
(hParamKey, szRegKey, 0, REG_SZ,
(CONST BYTE *) StartUpArg->Argv[i],
_tcslen (StartUpArg->Argv[i]) +
sizeof (TCHAR)) != ERROR_SUCCESS)
{
_stprintf (MsgErrorString, "%s %s",
_T ("Unable to create registry entries"),
lpszServiceDisplayName);
DisplayError (MsgErrorString);
LEAVE;
};
}
}
/*
* Everything is set, delete hKey
*/
RegCloseKey (hParamKey);
RegCloseKey (hKey);
}
/*
* Ready to Log messages
*/
/*
* Successfully registered as service
*/
_stprintf (MsgErrorString, "%s %s", lpszServiceName,
_T ("- Successfully registered as Service"));
/*
* Log message to eventlog
*/
WriteToEventLog (EVENTLOG_INFORMATION_TYPE, MsgErrorString);
}
FINALLY
{
if (hSCManager)
CloseServiceHandle (hSCManager);
if (hService)
CloseServiceHandle (hService);
if (hKey)
RegCloseKey (hKey);
if (hParamKey)
RegCloseKey (hParamKey);
}
}
/*
* Unregister the service with the Windows SCM
* Input - ServiceName
*/
VOID
UnregisterService (LPCSTR lpszServiceName)
{
TCHAR MsgErrorString[MAX_STR_SIZE]; /* Message or Error string */
SC_HANDLE hSCManager = NULL; /* SCM handle */
SC_HANDLE hService = NULL; /* Service Handle */
SERVICE_STATUS sStatus;
TCHAR szRegAppLogKey[] =
"SYSTEM\\CurrentControlSet\\Services\\EventLog\\Application\\";
TCHAR szRegKey[512];
/* HKEY hKey = NULL; ?* Key to registry entry */
TRY
{
/*
* Open Service Control Manager
*/
hSCManager = OpenSCManager (NULL, NULL, SC_MANAGER_CREATE_SERVICE);
if (hSCManager == NULL)
{
MessageBox (NULL, _T ("Can't open SCM"), g_szAppName, MB_ICONHAND);
LEAVE;
}
/*
* Open registered service
*/
hService = OpenService (hSCManager, lpszServiceName, SERVICE_ALL_ACCESS);
if (hService == NULL)
{
_stprintf (MsgErrorString, "%s %s", _T ("Can't open service"),
lpszServiceName);
MessageBox (NULL, MsgErrorString, g_szAppName, MB_ICONHAND);
LEAVE;
}
/*
* Query service status
* If running stop before deleting
*/
if (QueryServiceStatus (hService, &sStatus))
{
if (sStatus.dwCurrentState == SERVICE_RUNNING
|| sStatus.dwCurrentState == SERVICE_PAUSED)
{
ControlService (hService, SERVICE_CONTROL_STOP, &sStatus);
}
};
/*
* Delete the service
*/
if (DeleteService (hService) == FALSE)
{
_stprintf (MsgErrorString, "%s %s", _T ("Can't delete service"),
lpszServiceName);
/*
* Log message to eventlog
*/
WriteToEventLog (EVENTLOG_INFORMATION_TYPE, MsgErrorString);
LEAVE;
}
/*
* Log "Service deleted successfully " message to eventlog
*/
_stprintf (MsgErrorString, "%s %s", lpszServiceName, _T ("- Service deleted"));
WriteToEventLog (EVENTLOG_INFORMATION_TYPE, MsgErrorString);
/*
* Delete registry entries for EventLog
*/
_tcscpy (szRegKey, szRegAppLogKey);
_tcscat (szRegKey, lpszServiceName);
RegDeleteKey (HKEY_LOCAL_MACHINE, szRegKey);
}
/*
* Delete the handles
*/
FINALLY
{
if (hService)
CloseServiceHandle (hService);
if (hSCManager)
CloseServiceHandle (hSCManager);
}
}
/*
* To write message to Windows Event log
* Input - Event Type, Message string
*/
VOID
WriteToEventLog (WORD wType, LPCTSTR pszFormat, ...)
{
TCHAR szMessage[512];
LPTSTR LogStr[1];
va_list ArgList;
HANDLE hEventSource = NULL;
va_start (ArgList, pszFormat);
_vstprintf (szMessage, pszFormat, ArgList);
va_end (ArgList);
LogStr[0] = szMessage;
hEventSource = RegisterEventSource (NULL, g_szAppName);
if (hEventSource == NULL)
return;
ReportEvent (hEventSource, wType, 0,
DISPLAY_MSG, /* To Just output the text to event log */
NULL, 1, 0, LogStr, NULL);
DeregisterEventSource (hEventSource);
if (!g_fRunningAsService)
{
/*
* We are running in command mode, output the string
*/
_putts (szMessage);
}
}
/*
* Pre-process the second command-line argument from the user.
* Service related options are:
* -register - registers the service
* -unregister - unregisters the service
* -service - run as service
* other command-line arguments are ignored here.
*
* Return: Type indicating the option specified
*/
INT
ParseCmdLineForServiceOption (int argc, TCHAR * argv[])
{
int nReturn = RUN_AS_CONSOLE; /* Defualted to run as console */
if (argc >= 2)
{
/*
* second argument present
*/
if (lstrcmpi (_T ("-register"), argv[1]) == 0)
{
nReturn = REGISTER_SERVICE;
}
else if (lstrcmpi (_T ("-unregister"), argv[1]) == 0)
{
nReturn = UN_REGISTER_SERVICE;
}
else if (lstrcmpi (_T ("-service"), argv[1]) == 0)
{
nReturn = RUN_AS_SERVICE;
}
}
return nReturn;
}
/*
* To Display an error message describing the last system error
* message, along with a title passed as a parameter.
*/
VOID
DisplayError (LPCTSTR pszTitle)
{
LPVOID pErrorMsg;
/*
* Build Error String
*/
FormatMessage (FORMAT_MESSAGE_ALLOCATE_BUFFER |
FORMAT_MESSAGE_FROM_SYSTEM, NULL, GetLastError (),
MAKELANGID (LANG_NEUTRAL, SUBLANG_DEFAULT),
(LPTSTR) & pErrorMsg, 0, NULL);
if (g_fRunningAsService != FALSE)
{
WriteToEventLog (EVENTLOG_ERROR_TYPE, pErrorMsg);
}
else
{
MessageBox (NULL, pErrorMsg, pszTitle, MB_ICONHAND);
}
LocalFree (pErrorMsg);
}
/*
* To update current service status
* Sends the current service status to the SCM. Also updates
* the global service status structure.
*/
static BOOL
UpdateServiceStatus (DWORD dwStatus, DWORD dwErrorCode, DWORD dwWaitHint)
{
DWORD static dwCheckpoint = 1;
DWORD dwControls = SERVICE_ACCEPT_STOP | SERVICE_ACCEPT_PAUSE_CONTINUE;
if (g_fRunningAsService == FALSE)
return FALSE;
ZeroMemory (&ServiceStatus, sizeof (ServiceStatus));
ServiceStatus.dwServiceType = SERVICE_WIN32;
ServiceStatus.dwCurrentState = dwStatus;
ServiceStatus.dwWaitHint = dwWaitHint;
if (dwErrorCode)
{
ServiceStatus.dwWin32ExitCode = ERROR_SERVICE_SPECIFIC_ERROR;
ServiceStatus.dwServiceSpecificExitCode = dwErrorCode;
}
/*
* special cases that depend on the new state
*/
switch (dwStatus)
{
case SERVICE_START_PENDING:
dwControls = 0;
break;
case SERVICE_RUNNING:
case SERVICE_STOPPED:
dwCheckpoint = 0;
break;
}
ServiceStatus.dwCheckPoint = dwCheckpoint++;
ServiceStatus.dwControlsAccepted = dwControls;
return ReportCurrentServiceStatus ();
}
/*
* Reports current Service status to SCM
*/
static BOOL
ReportCurrentServiceStatus ()
{
return SetServiceStatus (hServiceStatus, &ServiceStatus);
}
/*
* The ServiceMain function to start service.
*/
VOID WINAPI
ServiceMain (DWORD argc, LPTSTR argv[])
{
SECURITY_ATTRIBUTES SecurityAttributes;
DWORD dwThreadId;
/*
* Input Arguments to function startup
*/
DWORD ArgCount = 0;
LPTSTR *ArgArray = NULL;
TCHAR szRegKey[512];
TCHAR szValue[128];
DWORD nSize;
HKEY hParamKey = NULL; /* To read startup parameters */
DWORD TotalParams = 0;
DWORD i;
InputParams ThreadInputParams;
/*
* Build the Input parameters to pass to worker thread
*/
/*
* SCM sends Service Name as first arg, increment to point
* arguments user specified while starting contorl agent
*/
/*
* Read registry parameter
*/
ArgCount = 1;
/*
* Create Registry Key path
*/
_stprintf (szRegKey, "%s%s\\%s",
_T ("SYSTEM\\CurrentControlSet\\Services\\"), g_szAppName,
"Parameters");
if (RegOpenKeyEx
(HKEY_LOCAL_MACHINE, szRegKey, 0, KEY_ALL_ACCESS, &hParamKey) == ERROR_SUCCESS)
{
/*
* Read startup Configuration information
*/
/*
* Find number of subkeys inside parameters
*/
if (RegQueryInfoKey (hParamKey, NULL, NULL, 0,
NULL, NULL, NULL, &TotalParams,
NULL, NULL, NULL, NULL) == ERROR_SUCCESS)
{
if (TotalParams != 0)
{
ArgCount += TotalParams;
/*
* Allocate memory to hold strings
*/
ArgArray = (LPTSTR *) malloc (sizeof (LPTSTR) * ArgCount);
/*
* Copy first argument
*/
ArgArray[0] = _tcsdup (argv[0]);
for (i = 1; i <= TotalParams; i++)
{
/*
* Create Subkey value name
*/
_stprintf (szRegKey, "%s%d", "Param", i);
/*
* Set size
*/
nSize = 128;
RegQueryValueEx (hParamKey, szRegKey, 0, NULL,
(LPBYTE) & szValue, &nSize);
ArgArray[i] = _tcsdup (szValue);
}
}
}
RegCloseKey (hParamKey);
}
if (ArgCount == 1)
{
/*
* No statup agrs are given
*/
ThreadInputParams.Argc = argc;
ThreadInputParams.Argv = argv;
}
else
{
ThreadInputParams.Argc = ArgCount;
ThreadInputParams.Argv = ArgArray;
}
/*
* Register Service Control Handler
*/
hServiceStatus = RegisterServiceCtrlHandler (g_szAppName, ControlHandler);
if (hServiceStatus == 0)
{
WriteToEventLog (EVENTLOG_ERROR_TYPE,
_T ("RegisterServiceCtrlHandler failed"));
return;
}
/*
* Update the service status to START_PENDING
*/
UpdateServiceStatus (SERVICE_START_PENDING, NO_ERROR, SCM_WAIT_INTERVAL);
/*
* Spin of worker thread, which does majority of the work
*/
TRY
{
if (SetSimpleSecurityAttributes (&SecurityAttributes) == FALSE)
{
WriteToEventLog (EVENTLOG_ERROR_TYPE,
_T ("Couldn't init security attributes"));
LEAVE;
}
hServiceThread =
(void *) _beginthreadex (&SecurityAttributes, 0,
ThreadFunction,
(void *) &ThreadInputParams, 0, &dwThreadId);
if (hServiceThread == NULL)
{
WriteToEventLog (EVENTLOG_ERROR_TYPE, _T ("Couldn't start worker thread"));
LEAVE;
}
/*
* Set Service Status to Running
*/
UpdateServiceStatus (SERVICE_RUNNING, NO_ERROR, SCM_WAIT_INTERVAL);
/*
* Wait for termination event and worker thread to
* * spin down.
*/
WaitForSingleObject (hServiceThread, INFINITE);
}
FINALLY
{
/*
* Release resources
*/
UpdateServiceStatus (SERVICE_STOPPED, NO_ERROR, SCM_WAIT_INTERVAL);
if (hServiceThread)
CloseHandle (hServiceThread);
FreeSecurityAttributes (&SecurityAttributes);
/*
* Delete allocated argument list
*/
if (ArgCount > 1 && ArgArray != NULL)
{
/*
* Delete all strings
*/
for (i = 0; i < ArgCount; i++)
{
free (ArgArray[i]);
}
free (ArgArray);
}
}
}
/*
* Function to start as Windows service
* The calling party should specify their entry point as input parameter
* Returns TRUE if the Service is started successfully
*/
BOOL
RunAsService (INT (*ServiceFunction) (INT, LPTSTR *))
{
/*
* Set the ServiceEntryPoint
*/
ServiceEntryPoint = ServiceFunction;
/*
* By default, mark as Running as a service
*/
g_fRunningAsService = TRUE;
/*
* Initialize ServiceTableEntry table
*/
ServiceTableEntry[0].lpServiceName = g_szAppName; /* Application Name */
/*
* Call SCM via StartServiceCtrlDispatcher to run as Service
* * If the function returns TRUE we are running as Service,
*/
if (StartServiceCtrlDispatcher (ServiceTableEntry) == FALSE)
{
g_fRunningAsService = FALSE;
/*
* Some other error has occurred.
*/
WriteToEventLog (EVENTLOG_ERROR_TYPE,
_T ("Couldn't start service - %s"), g_szAppName);
}
return g_fRunningAsService;
}
/*
* Service control handler function
* Responds to SCM commands/requests
* This service handles 4 commands
* - interrogate, pause, continue and stop.
*/
VOID WINAPI
ControlHandler (DWORD dwControl)
{
switch (dwControl)
{
case SERVICE_CONTROL_INTERROGATE:
ProcessServiceInterrogate ();
break;
case SERVICE_CONTROL_PAUSE:
ProcessServicePause ();
break;
case SERVICE_CONTROL_CONTINUE:
ProcessServiceContinue ();
break;
case SERVICE_CONTROL_STOP:
ProcessServiceStop ();
break;
}
}
/*
* To stop the service.
* If a stop function was registered, invoke it,
* otherwise terminate the worker thread.
* After stopping, Service status is set to STOP in
* main loop
*/
VOID
ProcessServiceStop (VOID)
{
UpdateServiceStatus (SERVICE_STOP_PENDING, NO_ERROR, SCM_WAIT_INTERVAL);
if (StopFunction != NULL)
{
(*StopFunction) ();
}
else
{
TerminateThread (hServiceThread, 0);
}
}
/*
* Returns the current state of the service to the SCM.
*/
VOID
ProcessServiceInterrogate (VOID)
{
ReportCurrentServiceStatus ();
}
/*
* To Create a security descriptor with a NULL ACL, which
* allows unlimited access. Returns a SECURITY_ATTRIBUTES
* structure that contains the security descriptor.
* The structure contains a dynamically allocated security
* descriptor that must be freed either manually, or by
* calling FreeSecurityAttributes
*/
BOOL
SetSimpleSecurityAttributes (SECURITY_ATTRIBUTES * pSecurityAttr)
{
BOOL fReturn = FALSE;
SECURITY_DESCRIPTOR *pSecurityDesc = NULL;
/*
* If an invalid address is passed as a parameter, return
* FALSE right away.
*/
if (!pSecurityAttr)
return FALSE;
pSecurityDesc =
(SECURITY_DESCRIPTOR *) LocalAlloc (LPTR, SECURITY_DESCRIPTOR_MIN_LENGTH);
if (!pSecurityDesc)
return FALSE;
fReturn =
InitializeSecurityDescriptor (pSecurityDesc, SECURITY_DESCRIPTOR_REVISION);
if (fReturn != FALSE)
{
fReturn = SetSecurityDescriptorDacl (pSecurityDesc, TRUE, NULL, FALSE);
}
if (fReturn != FALSE)
{
pSecurityAttr->nLength = sizeof (SECURITY_ATTRIBUTES);
pSecurityAttr->lpSecurityDescriptor = pSecurityDesc;
pSecurityAttr->bInheritHandle = TRUE;
}
else
{
/*
* Couldn't initialize or set security descriptor.
*/
LocalFree (pSecurityDesc);
}
return fReturn;
}
/*
* This function Frees the security descriptor, if any was created.
*/
VOID
FreeSecurityAttributes (SECURITY_ATTRIBUTES * pSecurityAttr)
{
if (pSecurityAttr && pSecurityAttr->lpSecurityDescriptor)
LocalFree (pSecurityAttr->lpSecurityDescriptor);
}
/*
* This function runs in the worker thread
* until an exit is forced, or until the SCM issues the STOP command.
* Invokes registered service function
* Returns when called registered function returns
*
* Input:
* lpParam contains argc and argv, pass to service main function
*/
DWORD WINAPI
ThreadFunction (LPVOID lpParam)
{
InputParams * pInputArg = (InputParams *) lpParam;
return (*ServiceEntryPoint) (pInputArg->Argc, pInputArg->Argv);
}
/*
* This function is called to register an application-specific function
* which is invoked when the SCM stops the worker thread.
*/
VOID
RegisterStopFunction (VOID (*StopFunc) (VOID))
{
StopFunction = StopFunc;
}
/*
* SCM pause command invokes this function
* If the service is not running, this function does nothing.
* Otherwise, suspend the worker thread and update the status.
*/
VOID
ProcessServicePause (VOID)
{
if (ServiceStatus.dwCurrentState == SERVICE_RUNNING)
{
UpdateServiceStatus (SERVICE_PAUSE_PENDING, NO_ERROR, SCM_WAIT_INTERVAL);
if (SuspendThread (hServiceThread) != -1)
{
UpdateServiceStatus (SERVICE_PAUSED, NO_ERROR, SCM_WAIT_INTERVAL);
}
}
}
/*
* SCM resume command invokes this function
* If the service is not paused, this function does nothing.
* Otherwise, resume the worker thread and update the status.
*/
VOID
ProcessServiceContinue (VOID)
{
if (ServiceStatus.dwCurrentState == SERVICE_PAUSED)
{
UpdateServiceStatus (SERVICE_CONTINUE_PENDING, NO_ERROR, SCM_WAIT_INTERVAL);
if (ResumeThread (hServiceThread) != -1)
{
UpdateServiceStatus (SERVICE_RUNNING, NO_ERROR, SCM_WAIT_INTERVAL);
}
}
}
#endif /* WIN32 */