2005-10-09 20:24:00 +00:00
|
|
|
/*
|
2007-09-06 14:51:38 +00:00
|
|
|
* PROJECT: ReactOS Service Control Manager
|
|
|
|
* LICENSE: GPL - See COPYING in the top level directory
|
|
|
|
* FILE: base/system/services/services.c
|
|
|
|
* PURPOSE: Main SCM controller
|
|
|
|
* COPYRIGHT: Copyright 2001-2005 Eric Kohl
|
|
|
|
* Copyright 2007 Ged Murphy <gedmurphy@reactos.org>
|
2000-12-05 02:40:51 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* INCLUDES *****************************************************************/
|
|
|
|
|
2002-06-07 20:11:03 +00:00
|
|
|
#include "services.h"
|
|
|
|
|
2014-01-13 13:10:28 +00:00
|
|
|
#include <wincon.h>
|
|
|
|
|
2004-04-11 16:10:05 +00:00
|
|
|
#define NDEBUG
|
2001-10-21 19:09:06 +00:00
|
|
|
#include <debug.h>
|
2000-12-05 02:40:51 +00:00
|
|
|
|
2005-04-02 05:37:20 +00:00
|
|
|
int WINAPI RegisterServicesProcess(DWORD ServicesProcessId);
|
2002-06-07 20:11:03 +00:00
|
|
|
|
2000-12-05 02:40:51 +00:00
|
|
|
/* GLOBALS ******************************************************************/
|
|
|
|
|
2013-03-17 12:39:44 +00:00
|
|
|
/* Defined in include/reactos/services/services.h */
|
|
|
|
// #define SCM_START_EVENT L"SvcctrlStartEvent_A3752DX"
|
|
|
|
#define SCM_AUTOSTARTCOMPLETE_EVENT L"SC_AutoStartComplete"
|
|
|
|
|
|
|
|
BOOL ScmInitialize = FALSE;
|
2005-10-30 13:13:53 +00:00
|
|
|
BOOL ScmShutdown = FALSE;
|
2020-01-07 23:14:50 +00:00
|
|
|
BOOL ScmLiveSetup = FALSE;
|
2022-02-21 10:12:48 +00:00
|
|
|
BOOL ScmSetupInProgress = FALSE;
|
2012-04-08 21:06:32 +00:00
|
|
|
static HANDLE hScmShutdownEvent = NULL;
|
2019-08-31 15:24:47 +00:00
|
|
|
static HANDLE hScmSecurityServicesEvent = NULL;
|
2005-10-30 13:13:53 +00:00
|
|
|
|
2000-12-05 02:40:51 +00:00
|
|
|
|
|
|
|
/* FUNCTIONS *****************************************************************/
|
|
|
|
|
2005-04-15 22:02:37 +00:00
|
|
|
VOID
|
|
|
|
PrintString(LPCSTR fmt, ...)
|
2000-12-05 02:40:51 +00:00
|
|
|
{
|
2009-06-17 12:44:05 +00:00
|
|
|
#if DBG
|
2005-10-22 19:46:00 +00:00
|
|
|
CHAR buffer[512];
|
|
|
|
va_list ap;
|
2000-12-05 02:40:51 +00:00
|
|
|
|
2005-10-22 19:46:00 +00:00
|
|
|
va_start(ap, fmt);
|
|
|
|
vsprintf(buffer, fmt, ap);
|
|
|
|
va_end(ap);
|
2000-12-05 02:40:51 +00:00
|
|
|
|
2005-10-22 19:46:00 +00:00
|
|
|
OutputDebugStringA(buffer);
|
2002-02-08 02:57:10 +00:00
|
|
|
#endif
|
2000-12-05 02:40:51 +00:00
|
|
|
}
|
|
|
|
|
2020-01-07 23:14:50 +00:00
|
|
|
DWORD
|
|
|
|
CheckForLiveCD(VOID)
|
|
|
|
{
|
|
|
|
WCHAR CommandLine[MAX_PATH];
|
|
|
|
HKEY hSetupKey;
|
|
|
|
DWORD dwSetupType;
|
2022-02-21 10:12:48 +00:00
|
|
|
DWORD dwSetupInProgress;
|
2020-01-07 23:14:50 +00:00
|
|
|
DWORD dwType;
|
|
|
|
DWORD dwSize;
|
|
|
|
DWORD dwError;
|
|
|
|
|
|
|
|
DPRINT1("CheckSetup()\n");
|
|
|
|
|
|
|
|
/* Open the Setup key */
|
|
|
|
dwError = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
|
|
|
|
L"SYSTEM\\Setup",
|
|
|
|
0,
|
|
|
|
KEY_QUERY_VALUE,
|
|
|
|
&hSetupKey);
|
|
|
|
if (dwError != ERROR_SUCCESS)
|
|
|
|
return dwError;
|
|
|
|
|
|
|
|
/* Read the SetupType value */
|
|
|
|
dwSize = sizeof(DWORD);
|
|
|
|
dwError = RegQueryValueExW(hSetupKey,
|
|
|
|
L"SetupType",
|
|
|
|
NULL,
|
|
|
|
&dwType,
|
|
|
|
(LPBYTE)&dwSetupType,
|
|
|
|
&dwSize);
|
|
|
|
|
|
|
|
if (dwError != ERROR_SUCCESS ||
|
|
|
|
dwType != REG_DWORD ||
|
|
|
|
dwSize != sizeof(DWORD) ||
|
|
|
|
dwSetupType == 0)
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
/* Read the CmdLine value */
|
|
|
|
dwSize = sizeof(CommandLine);
|
|
|
|
dwError = RegQueryValueExW(hSetupKey,
|
|
|
|
L"CmdLine",
|
|
|
|
NULL,
|
|
|
|
&dwType,
|
|
|
|
(LPBYTE)CommandLine,
|
|
|
|
&dwSize);
|
|
|
|
|
|
|
|
if (dwError != ERROR_SUCCESS ||
|
|
|
|
(dwType != REG_SZ &&
|
|
|
|
dwType != REG_EXPAND_SZ &&
|
|
|
|
dwType != REG_MULTI_SZ))
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
/* Check for the '-mini' option */
|
|
|
|
if (wcsstr(CommandLine, L" -mini") != NULL)
|
|
|
|
{
|
2022-03-20 19:12:56 +00:00
|
|
|
DPRINT1("Running on LiveCD\n");
|
2020-01-07 23:14:50 +00:00
|
|
|
ScmLiveSetup = TRUE;
|
|
|
|
}
|
|
|
|
|
2022-02-21 10:12:48 +00:00
|
|
|
/* Read the SystemSetupInProgress value */
|
|
|
|
dwSize = sizeof(DWORD);
|
|
|
|
dwError = RegQueryValueExW(hSetupKey,
|
|
|
|
L"SystemSetupInProgress",
|
|
|
|
NULL,
|
|
|
|
&dwType,
|
|
|
|
(LPBYTE)&dwSetupInProgress,
|
|
|
|
&dwSize);
|
|
|
|
if (dwError != ERROR_SUCCESS ||
|
|
|
|
dwType != REG_DWORD ||
|
|
|
|
dwSize != sizeof(DWORD) ||
|
|
|
|
dwSetupType == 0)
|
|
|
|
{
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dwSetupInProgress == 1)
|
|
|
|
{
|
|
|
|
DPRINT1("ReactOS Setup currently in progress!\n");
|
|
|
|
ScmSetupInProgress = TRUE;
|
|
|
|
}
|
|
|
|
|
2020-01-07 23:14:50 +00:00
|
|
|
done:
|
|
|
|
RegCloseKey(hSetupKey);
|
|
|
|
|
|
|
|
return dwError;
|
|
|
|
}
|
|
|
|
|
2000-12-05 02:40:51 +00:00
|
|
|
|
2019-08-31 15:24:47 +00:00
|
|
|
DWORD
|
|
|
|
SetSecurityServicesEvent(VOID)
|
|
|
|
{
|
|
|
|
DWORD dwError;
|
|
|
|
|
|
|
|
if (hScmSecurityServicesEvent != NULL)
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
|
|
|
|
/* Create or open the SECURITY_SERVICES_STARTED event */
|
|
|
|
hScmSecurityServicesEvent = CreateEventW(NULL,
|
|
|
|
TRUE,
|
|
|
|
FALSE,
|
|
|
|
L"SECURITY_SERVICES_STARTED");
|
|
|
|
if (hScmSecurityServicesEvent == NULL)
|
|
|
|
{
|
|
|
|
dwError = GetLastError();
|
|
|
|
if (dwError != ERROR_ALREADY_EXISTS)
|
|
|
|
return dwError;
|
|
|
|
|
|
|
|
hScmSecurityServicesEvent = OpenEventW(EVENT_MODIFY_STATE,
|
|
|
|
FALSE,
|
|
|
|
L"SECURITY_SERVICES_STARTED");
|
|
|
|
if (hScmSecurityServicesEvent == NULL)
|
|
|
|
return GetLastError();
|
|
|
|
}
|
|
|
|
|
|
|
|
SetEvent(hScmSecurityServicesEvent);
|
|
|
|
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-26 10:50:26 +00:00
|
|
|
VOID
|
2014-09-28 14:10:16 +00:00
|
|
|
ScmLogEvent(DWORD dwEventId,
|
|
|
|
WORD wType,
|
2011-05-26 10:50:26 +00:00
|
|
|
WORD wStrings,
|
|
|
|
LPCWSTR *lpStrings)
|
|
|
|
{
|
|
|
|
HANDLE hLog;
|
|
|
|
|
|
|
|
hLog = RegisterEventSourceW(NULL,
|
|
|
|
L"Service Control Manager");
|
|
|
|
if (hLog == NULL)
|
|
|
|
{
|
2013-03-08 20:49:03 +00:00
|
|
|
DPRINT1("ScmLogEvent: RegisterEventSourceW failed %lu\n", GetLastError());
|
2011-05-26 10:50:26 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ReportEventW(hLog,
|
2014-09-28 14:10:16 +00:00
|
|
|
wType,
|
2011-05-26 10:50:26 +00:00
|
|
|
0,
|
|
|
|
dwEventId,
|
2014-09-28 14:10:16 +00:00
|
|
|
NULL,
|
2011-05-26 10:50:26 +00:00
|
|
|
wStrings,
|
|
|
|
0,
|
|
|
|
lpStrings,
|
|
|
|
NULL))
|
|
|
|
{
|
2013-03-08 20:49:03 +00:00
|
|
|
DPRINT1("ScmLogEvent: ReportEventW failed %lu\n", GetLastError());
|
2011-05-26 10:50:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
DeregisterEventSource(hLog);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-06-11 19:02:00 +00:00
|
|
|
VOID
|
|
|
|
ScmWaitForLsa(VOID)
|
2010-09-12 16:29:43 +00:00
|
|
|
{
|
2018-06-28 20:32:28 +00:00
|
|
|
HANDLE hEvent = CreateEventW(NULL, TRUE, FALSE, L"LSA_RPC_SERVER_ACTIVE");
|
2013-04-25 23:13:58 +00:00
|
|
|
if (hEvent == NULL)
|
2010-09-12 16:29:43 +00:00
|
|
|
{
|
2018-06-28 20:32:28 +00:00
|
|
|
DPRINT1("Failed to create or open the notification event (Error %lu)\n", GetLastError());
|
2010-09-12 16:29:43 +00:00
|
|
|
}
|
2013-03-17 12:39:44 +00:00
|
|
|
else
|
|
|
|
{
|
2022-03-20 19:12:56 +00:00
|
|
|
DPRINT("Wait for the LSA server\n");
|
2013-03-17 12:39:44 +00:00
|
|
|
WaitForSingleObject(hEvent, INFINITE);
|
2022-03-20 19:12:56 +00:00
|
|
|
DPRINT("LSA server running\n");
|
2013-03-17 12:39:44 +00:00
|
|
|
CloseHandle(hEvent);
|
|
|
|
}
|
2012-06-11 19:02:00 +00:00
|
|
|
|
|
|
|
DPRINT("ScmWaitForLsa() done\n");
|
2010-09-12 16:29:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-10-30 13:13:53 +00:00
|
|
|
BOOL WINAPI
|
|
|
|
ShutdownHandlerRoutine(DWORD dwCtrlType)
|
|
|
|
{
|
|
|
|
DPRINT1("ShutdownHandlerRoutine() called\n");
|
|
|
|
|
2010-03-10 18:28:45 +00:00
|
|
|
if (dwCtrlType & (CTRL_SHUTDOWN_EVENT | CTRL_LOGOFF_EVENT))
|
2005-10-30 13:13:53 +00:00
|
|
|
{
|
2022-03-20 19:12:56 +00:00
|
|
|
DPRINT1("Shutdown event received\n");
|
2005-10-30 13:13:53 +00:00
|
|
|
ScmShutdown = TRUE;
|
|
|
|
|
2007-09-06 14:51:38 +00:00
|
|
|
ScmAutoShutdownServices();
|
|
|
|
ScmShutdownServiceDatabase();
|
2012-04-08 21:06:32 +00:00
|
|
|
|
2016-11-05 14:55:55 +00:00
|
|
|
/* Set the shutdown event */
|
2012-04-08 21:06:32 +00:00
|
|
|
SetEvent(hScmShutdownEvent);
|
2005-10-30 13:13:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-30 11:16:55 +00:00
|
|
|
int WINAPI
|
2008-05-08 18:42:29 +00:00
|
|
|
wWinMain(HINSTANCE hInstance,
|
|
|
|
HINSTANCE hPrevInstance,
|
|
|
|
LPWSTR lpCmdLine,
|
|
|
|
int nShowCmd)
|
2000-12-05 02:40:51 +00:00
|
|
|
{
|
2012-07-07 15:35:40 +00:00
|
|
|
HANDLE hScmStartEvent = NULL;
|
2013-03-17 12:39:44 +00:00
|
|
|
HANDLE hScmAutoStartCompleteEvent = NULL;
|
2012-07-07 15:35:40 +00:00
|
|
|
SC_RPC_LOCK Lock = NULL;
|
2012-11-24 00:53:01 +00:00
|
|
|
BOOL bCanDeleteNamedPipeCriticalSection = FALSE;
|
2005-11-06 11:56:56 +00:00
|
|
|
DWORD dwError;
|
2002-06-07 20:11:03 +00:00
|
|
|
|
2005-10-22 19:46:00 +00:00
|
|
|
DPRINT("SERVICES: Service Control Manager\n");
|
2002-06-07 20:11:03 +00:00
|
|
|
|
2020-01-07 23:14:50 +00:00
|
|
|
dwError = CheckForLiveCD();
|
|
|
|
if (dwError != ERROR_SUCCESS)
|
|
|
|
{
|
|
|
|
DPRINT1("SERVICES: Failed to check for LiveCD (Error %lu)\n", dwError);
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
2014-11-25 23:34:14 +00:00
|
|
|
/* Make us critical */
|
|
|
|
RtlSetProcessIsCritical(TRUE, NULL, TRUE);
|
|
|
|
|
2013-03-17 12:39:44 +00:00
|
|
|
/* We are initializing ourselves */
|
|
|
|
ScmInitialize = TRUE;
|
|
|
|
|
|
|
|
/* Create the start event */
|
2013-04-25 23:13:58 +00:00
|
|
|
hScmStartEvent = CreateEventW(NULL, TRUE, FALSE, SCM_START_EVENT);
|
|
|
|
if (hScmStartEvent == NULL)
|
2005-10-22 19:46:00 +00:00
|
|
|
{
|
2013-03-17 12:39:44 +00:00
|
|
|
DPRINT1("SERVICES: Failed to create the start event\n");
|
2012-07-07 15:35:40 +00:00
|
|
|
goto done;
|
2005-10-22 19:46:00 +00:00
|
|
|
}
|
2022-03-20 19:12:56 +00:00
|
|
|
DPRINT("SERVICES: Created start event with handle %p\n", hScmStartEvent);
|
2000-12-05 02:40:51 +00:00
|
|
|
|
2013-03-17 12:39:44 +00:00
|
|
|
/* Create the auto-start complete event */
|
2013-04-25 23:13:58 +00:00
|
|
|
hScmAutoStartCompleteEvent = CreateEventW(NULL, TRUE, FALSE, SCM_AUTOSTARTCOMPLETE_EVENT);
|
2013-04-25 23:16:19 +00:00
|
|
|
if (hScmAutoStartCompleteEvent == NULL)
|
2013-03-17 12:39:44 +00:00
|
|
|
{
|
|
|
|
DPRINT1("SERVICES: Failed to create the auto-start complete event\n");
|
|
|
|
goto done;
|
|
|
|
}
|
2022-03-20 19:12:56 +00:00
|
|
|
DPRINT("SERVICES: created auto-start complete event with handle %p\n", hScmAutoStartCompleteEvent);
|
2000-12-05 02:40:51 +00:00
|
|
|
|
2012-07-07 15:35:40 +00:00
|
|
|
/* Create the shutdown event */
|
2013-03-17 12:39:44 +00:00
|
|
|
hScmShutdownEvent = CreateEventW(NULL, TRUE, FALSE, NULL);
|
2012-07-07 15:35:40 +00:00
|
|
|
if (hScmShutdownEvent == NULL)
|
|
|
|
{
|
2013-04-25 23:13:58 +00:00
|
|
|
DPRINT1("SERVICES: Failed to create the shutdown event\n");
|
2012-07-07 15:35:40 +00:00
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
2012-11-24 00:53:01 +00:00
|
|
|
/* Initialize our communication named pipe's critical section */
|
|
|
|
ScmInitNamedPipeCriticalSection();
|
|
|
|
bCanDeleteNamedPipeCriticalSection = TRUE;
|
|
|
|
|
2005-10-22 19:46:00 +00:00
|
|
|
// ScmInitThreadManager();
|
2005-04-15 22:02:37 +00:00
|
|
|
|
2016-06-09 21:30:48 +00:00
|
|
|
ScmInitializeSecurity();
|
|
|
|
|
2005-10-22 19:46:00 +00:00
|
|
|
/* FIXME: more initialization */
|
2000-12-05 02:40:51 +00:00
|
|
|
|
2018-09-28 23:31:02 +00:00
|
|
|
/* Create the 'Last Known Good' control set */
|
|
|
|
dwError = ScmCreateLastKnownGoodControlSet();
|
|
|
|
if (dwError != ERROR_SUCCESS)
|
2012-07-07 18:17:59 +00:00
|
|
|
{
|
2018-09-28 23:31:02 +00:00
|
|
|
DPRINT1("SERVICES: Failed to create the 'Last Known Good' control set (Error %lu)\n", dwError);
|
2012-07-07 18:17:59 +00:00
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
2012-12-18 20:38:54 +00:00
|
|
|
/* Create the services database */
|
2005-11-06 11:56:56 +00:00
|
|
|
dwError = ScmCreateServiceDatabase();
|
|
|
|
if (dwError != ERROR_SUCCESS)
|
2005-10-22 19:46:00 +00:00
|
|
|
{
|
2013-03-17 12:39:44 +00:00
|
|
|
DPRINT1("SERVICES: Failed to create SCM database (Error %lu)\n", dwError);
|
2012-07-07 15:35:40 +00:00
|
|
|
goto done;
|
2005-10-22 19:46:00 +00:00
|
|
|
}
|
2002-06-07 20:11:03 +00:00
|
|
|
|
2012-12-18 20:38:54 +00:00
|
|
|
/* Update the services database */
|
2005-10-22 19:46:00 +00:00
|
|
|
ScmGetBootAndSystemDriverState();
|
2000-12-05 02:40:51 +00:00
|
|
|
|
2013-03-17 12:39:44 +00:00
|
|
|
/* Register the Service Control Manager process with the ReactOS Subsystem */
|
2012-12-18 20:38:54 +00:00
|
|
|
if (!RegisterServicesProcess(GetCurrentProcessId()))
|
|
|
|
{
|
|
|
|
DPRINT1("SERVICES: Could not register SCM process\n");
|
|
|
|
goto done;
|
|
|
|
}
|
2000-12-05 02:40:51 +00:00
|
|
|
|
2013-03-17 12:39:44 +00:00
|
|
|
/*
|
|
|
|
* Acquire the user service start lock until
|
|
|
|
* auto-start services have been started.
|
|
|
|
*/
|
2012-11-24 00:53:01 +00:00
|
|
|
dwError = ScmAcquireServiceStartLock(TRUE, &Lock);
|
|
|
|
if (dwError != ERROR_SUCCESS)
|
|
|
|
{
|
2022-03-20 19:12:56 +00:00
|
|
|
DPRINT1("SERVICES: Failed to acquire service start lock (Error %lu)\n", dwError);
|
2012-11-24 00:53:01 +00:00
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Start the RPC server */
|
|
|
|
ScmStartRpcServer();
|
|
|
|
|
2005-10-22 19:46:00 +00:00
|
|
|
/* Signal start event */
|
|
|
|
SetEvent(hScmStartEvent);
|
2001-01-20 18:40:27 +00:00
|
|
|
|
2022-03-20 19:12:56 +00:00
|
|
|
DPRINT("SERVICES: Initialized\n");
|
2013-03-17 12:39:44 +00:00
|
|
|
|
2005-10-30 13:13:53 +00:00
|
|
|
/* Register event handler (used for system shutdown) */
|
|
|
|
SetConsoleCtrlHandler(ShutdownHandlerRoutine, TRUE);
|
2000-12-05 02:40:51 +00:00
|
|
|
|
2014-06-16 00:49:28 +00:00
|
|
|
/*
|
|
|
|
* Set our shutdown parameters: we want to shutdown after the maintained
|
|
|
|
* services (that inherit the default shutdown level of 640).
|
|
|
|
*/
|
|
|
|
SetProcessShutdownParameters(480, SHUTDOWN_NORETRY);
|
|
|
|
|
2005-10-22 19:46:00 +00:00
|
|
|
/* Start auto-start services */
|
|
|
|
ScmAutoStartServices();
|
2000-12-05 02:40:51 +00:00
|
|
|
|
2013-03-17 12:39:44 +00:00
|
|
|
/* Signal auto-start complete event */
|
|
|
|
SetEvent(hScmAutoStartCompleteEvent);
|
|
|
|
|
2005-10-22 19:46:00 +00:00
|
|
|
/* FIXME: more to do ? */
|
2000-12-05 02:40:51 +00:00
|
|
|
|
2012-07-07 15:35:40 +00:00
|
|
|
/* Release the service start lock */
|
|
|
|
ScmReleaseServiceStartLock(&Lock);
|
2002-06-07 20:11:03 +00:00
|
|
|
|
2013-03-17 12:39:44 +00:00
|
|
|
/* Initialization finished */
|
|
|
|
ScmInitialize = FALSE;
|
|
|
|
|
2022-03-20 19:12:56 +00:00
|
|
|
DPRINT("SERVICES: Running\n");
|
2002-06-07 20:11:03 +00:00
|
|
|
|
2012-07-07 15:35:40 +00:00
|
|
|
/* Wait until the shutdown event gets signaled */
|
|
|
|
WaitForSingleObject(hScmShutdownEvent, INFINITE);
|
2001-01-20 18:40:27 +00:00
|
|
|
|
2012-07-07 15:35:40 +00:00
|
|
|
done:
|
2016-06-09 21:30:48 +00:00
|
|
|
ScmShutdownSecurity();
|
|
|
|
|
2012-11-24 00:53:01 +00:00
|
|
|
/* Delete our communication named pipe's critical section */
|
2017-10-01 14:42:04 +00:00
|
|
|
if (bCanDeleteNamedPipeCriticalSection != FALSE)
|
2012-07-08 23:08:19 +00:00
|
|
|
ScmDeleteNamedPipeCriticalSection();
|
2011-02-27 19:25:14 +00:00
|
|
|
|
2019-08-31 15:24:47 +00:00
|
|
|
if (hScmSecurityServicesEvent != NULL)
|
|
|
|
CloseHandle(hScmSecurityServicesEvent);
|
|
|
|
|
2012-04-08 21:06:32 +00:00
|
|
|
/* Close the shutdown event */
|
2012-07-07 15:35:40 +00:00
|
|
|
if (hScmShutdownEvent != NULL)
|
|
|
|
CloseHandle(hScmShutdownEvent);
|
2012-04-08 21:06:32 +00:00
|
|
|
|
2013-03-17 12:39:44 +00:00
|
|
|
/* Close the auto-start complete event */
|
|
|
|
if (hScmAutoStartCompleteEvent != NULL)
|
|
|
|
CloseHandle(hScmAutoStartCompleteEvent);
|
|
|
|
|
2012-04-08 21:06:32 +00:00
|
|
|
/* Close the start event */
|
2012-07-07 15:35:40 +00:00
|
|
|
if (hScmStartEvent != NULL)
|
|
|
|
CloseHandle(hScmStartEvent);
|
2007-09-06 14:51:38 +00:00
|
|
|
|
2022-03-20 19:12:56 +00:00
|
|
|
DPRINT("SERVICES: Finished\n");
|
2001-01-20 18:40:27 +00:00
|
|
|
|
2005-10-22 19:46:00 +00:00
|
|
|
ExitThread(0);
|
|
|
|
return 0;
|
2000-12-05 02:40:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* EOF */
|