mirror of
https://github.com/reactos/reactos.git
synced 2024-12-28 10:04:49 +00:00
Improved driver loading sequence with respect to the group order.
Added ability to run user-mode services. svn path=/trunk/; revision=3110
This commit is contained in:
parent
1b52f5ff18
commit
81a07ed6a6
1 changed files with 297 additions and 105 deletions
|
@ -1,4 +1,4 @@
|
||||||
/* $Id: database.c,v 1.2 2002/06/12 23:33:15 ekohl Exp $
|
/* $Id: database.c,v 1.3 2002/06/17 15:47:32 ekohl Exp $
|
||||||
*
|
*
|
||||||
* service control manager
|
* service control manager
|
||||||
*
|
*
|
||||||
|
@ -62,6 +62,7 @@ typedef struct _SERVICE
|
||||||
ULONG Tag;
|
ULONG Tag;
|
||||||
|
|
||||||
BOOLEAN ServiceRunning;
|
BOOLEAN ServiceRunning;
|
||||||
|
BOOLEAN ServiceVisited;
|
||||||
|
|
||||||
} SERVICE, *PSERVICE;
|
} SERVICE, *PSERVICE;
|
||||||
|
|
||||||
|
@ -86,7 +87,7 @@ CreateGroupListRoutine(PWSTR ValueName,
|
||||||
|
|
||||||
if (ValueType == REG_SZ)
|
if (ValueType == REG_SZ)
|
||||||
{
|
{
|
||||||
// PrintString("Data: '%S'\n", (PWCHAR)ValueData);
|
DPRINT("Data: '%S'\n", (PWCHAR)ValueData);
|
||||||
|
|
||||||
Group = (PSERVICE_GROUP)HeapAlloc(GetProcessHeap(),
|
Group = (PSERVICE_GROUP)HeapAlloc(GetProcessHeap(),
|
||||||
HEAP_ZERO_MEMORY,
|
HEAP_ZERO_MEMORY,
|
||||||
|
@ -118,7 +119,7 @@ CreateServiceListEntry(PUNICODE_STRING ServiceName)
|
||||||
PSERVICE Service = NULL;
|
PSERVICE Service = NULL;
|
||||||
NTSTATUS Status;
|
NTSTATUS Status;
|
||||||
|
|
||||||
// PrintString("Service: '%wZ'\n", ServiceName);
|
DPRINT("Service: '%wZ'\n", ServiceName);
|
||||||
|
|
||||||
/* Allocate service entry */
|
/* Allocate service entry */
|
||||||
Service = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY,
|
Service = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY,
|
||||||
|
@ -193,13 +194,11 @@ CreateServiceListEntry(PUNICODE_STRING ServiceName)
|
||||||
return(Status);
|
return(Status);
|
||||||
}
|
}
|
||||||
|
|
||||||
#if 0
|
DPRINT("ServiceName: '%wZ'\n", &Service->ServiceName);
|
||||||
PrintString("ServiceName: '%wZ'\n", &Service->ServiceName);
|
DPRINT("RegistryPath: '%wZ'\n", &Service->RegistryPath);
|
||||||
PrintString("RegistryPath: '%wZ'\n", &Service->RegistryPath);
|
DPRINT("ServiceGroup: '%wZ'\n", &Service->ServiceGroup);
|
||||||
PrintString("ServiceGroup: '%wZ'\n", &Service->ServiceGroup);
|
DPRINT("Start %lx Type %lx ErrorControl %lx\n",
|
||||||
PrintString("Start %lx Type %lx ErrorControl %lx\n",
|
Service->Start, Service->Type, Service->ErrorControl);
|
||||||
Service->Start, Service->Type, Service->ErrorControl);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* Append service entry */
|
/* Append service entry */
|
||||||
InsertTailList(&ServiceListHead,
|
InsertTailList(&ServiceListHead,
|
||||||
|
@ -213,7 +212,6 @@ NTSTATUS
|
||||||
ScmCreateServiceDataBase(VOID)
|
ScmCreateServiceDataBase(VOID)
|
||||||
{
|
{
|
||||||
RTL_QUERY_REGISTRY_TABLE QueryTable[2];
|
RTL_QUERY_REGISTRY_TABLE QueryTable[2];
|
||||||
WCHAR NameBuffer[MAX_PATH];
|
|
||||||
OBJECT_ATTRIBUTES ObjectAttributes;
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
||||||
UNICODE_STRING ServicesKeyName;
|
UNICODE_STRING ServicesKeyName;
|
||||||
UNICODE_STRING SubKeyName;
|
UNICODE_STRING SubKeyName;
|
||||||
|
@ -225,13 +223,12 @@ ScmCreateServiceDataBase(VOID)
|
||||||
ULONG KeyInfoLength = 0;
|
ULONG KeyInfoLength = 0;
|
||||||
ULONG ReturnedLength;
|
ULONG ReturnedLength;
|
||||||
|
|
||||||
// PrintString("ScmCreateServiceDataBase() called\n");
|
DPRINT("ScmCreateServiceDataBase() called\n");
|
||||||
|
|
||||||
/* Initialize basic variables */
|
/* Initialize basic variables */
|
||||||
InitializeListHead(&GroupListHead);
|
InitializeListHead(&GroupListHead);
|
||||||
InitializeListHead(&ServiceListHead);
|
InitializeListHead(&ServiceListHead);
|
||||||
|
|
||||||
|
|
||||||
/* Build group order list */
|
/* Build group order list */
|
||||||
RtlZeroMemory(&QueryTable,
|
RtlZeroMemory(&QueryTable,
|
||||||
sizeof(QueryTable));
|
sizeof(QueryTable));
|
||||||
|
@ -291,7 +288,7 @@ ScmCreateServiceDataBase(VOID)
|
||||||
SubKeyName.Buffer = KeyInfo->Name;
|
SubKeyName.Buffer = KeyInfo->Name;
|
||||||
SubKeyName.Buffer[SubKeyName.Length / sizeof(WCHAR)] = 0;
|
SubKeyName.Buffer[SubKeyName.Length / sizeof(WCHAR)] = 0;
|
||||||
|
|
||||||
// PrintString("KeyName: '%wZ'\n", &SubKeyName);
|
DPRINT("KeyName: '%wZ'\n", &SubKeyName);
|
||||||
Status = CreateServiceListEntry(&SubKeyName);
|
Status = CreateServiceListEntry(&SubKeyName);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -305,7 +302,80 @@ ScmCreateServiceDataBase(VOID)
|
||||||
HeapFree(GetProcessHeap(), 0, KeyInfo);
|
HeapFree(GetProcessHeap(), 0, KeyInfo);
|
||||||
NtClose(ServicesKey);
|
NtClose(ServicesKey);
|
||||||
|
|
||||||
// PrintString("ScmCreateServiceDataBase() done\n");
|
DPRINT("ScmCreateServiceDataBase() done\n");
|
||||||
|
|
||||||
|
return(STATUS_SUCCESS);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static NTSTATUS
|
||||||
|
ScmCheckDriver(PSERVICE_GROUP Group,
|
||||||
|
PSERVICE Service)
|
||||||
|
{
|
||||||
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
||||||
|
UNICODE_STRING DirName;
|
||||||
|
HANDLE DirHandle;
|
||||||
|
NTSTATUS Status;
|
||||||
|
|
||||||
|
DPRINT("ScmCheckDriver() called\n");
|
||||||
|
|
||||||
|
if (Service->Type == SERVICE_KERNEL_DRIVER)
|
||||||
|
{
|
||||||
|
RtlInitUnicodeString(&DirName,
|
||||||
|
L"\\Driver");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
RtlInitUnicodeString(&DirName,
|
||||||
|
L"\\FileSystem");
|
||||||
|
}
|
||||||
|
|
||||||
|
InitializeObjectAttributes(&ObjectAttributes,
|
||||||
|
&DirName,
|
||||||
|
0,
|
||||||
|
NULL,
|
||||||
|
NULL);
|
||||||
|
|
||||||
|
Status = NtOpenDirectoryObject(&DirHandle,
|
||||||
|
DIRECTORY_QUERY | DIRECTORY_TRAVERSE,
|
||||||
|
&ObjectAttributes);
|
||||||
|
if (!NT_SUCCESS(Status))
|
||||||
|
{
|
||||||
|
return(Status);
|
||||||
|
}
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
Index = 0;
|
||||||
|
while (TRUE)
|
||||||
|
{
|
||||||
|
|
||||||
|
|
||||||
|
Status = NtQueryDirectoryObject(DirHandle,
|
||||||
|
&DirInfo,
|
||||||
|
BufferLength,
|
||||||
|
TRUE,
|
||||||
|
FALSE,
|
||||||
|
&Index,
|
||||||
|
&DataLength);
|
||||||
|
if (!NT_SUCCESS(Status))
|
||||||
|
{
|
||||||
|
NtClose(DirHandle);
|
||||||
|
return(Status);
|
||||||
|
}
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
if (NT_SUCCESS(Status))
|
||||||
|
{
|
||||||
|
CurrentGroup->ServicesRunning = TRUE;
|
||||||
|
CurrentService->ServiceRunning = TRUE;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
NtClose(DirHandle);
|
||||||
|
|
||||||
return(STATUS_SUCCESS);
|
return(STATUS_SUCCESS);
|
||||||
}
|
}
|
||||||
|
@ -314,56 +384,168 @@ ScmCreateServiceDataBase(VOID)
|
||||||
VOID
|
VOID
|
||||||
ScmGetBootAndSystemDriverState(VOID)
|
ScmGetBootAndSystemDriverState(VOID)
|
||||||
{
|
{
|
||||||
|
PLIST_ENTRY GroupEntry;
|
||||||
|
PLIST_ENTRY ServiceEntry;
|
||||||
|
PSERVICE_GROUP CurrentGroup;
|
||||||
|
PSERVICE CurrentService;
|
||||||
|
NTSTATUS Status;
|
||||||
|
|
||||||
|
DPRINT("ScmGetBootAndSystemDriverState() called\n");
|
||||||
|
|
||||||
|
GroupEntry = GroupListHead.Flink;
|
||||||
|
while (GroupEntry != &GroupListHead)
|
||||||
|
{
|
||||||
|
CurrentGroup = CONTAINING_RECORD(GroupEntry, SERVICE_GROUP, GroupListEntry);
|
||||||
|
|
||||||
|
DPRINT("Checking group '%wZ'\n", &CurrentGroup->GroupName);
|
||||||
|
|
||||||
|
ServiceEntry = ServiceListHead.Flink;
|
||||||
|
while (ServiceEntry != &ServiceListHead)
|
||||||
|
{
|
||||||
|
CurrentService = CONTAINING_RECORD(ServiceEntry, SERVICE, ServiceListEntry);
|
||||||
|
|
||||||
|
if (CurrentService->Start == SERVICE_BOOT_START ||
|
||||||
|
CurrentService->Start == SERVICE_SYSTEM_START)
|
||||||
|
{
|
||||||
|
/* Check driver */
|
||||||
|
DPRINT(" Checking service: %wZ\n", &CurrentService->ServiceName);
|
||||||
|
|
||||||
|
ScmCheckDriver(CurrentGroup,
|
||||||
|
CurrentService);
|
||||||
|
}
|
||||||
|
ServiceEntry = ServiceEntry->Flink;
|
||||||
|
}
|
||||||
|
GroupEntry = GroupEntry->Flink;
|
||||||
|
}
|
||||||
|
|
||||||
|
DPRINT("ScmGetBootAndSystemDriverState() done\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static NTSTATUS
|
static NTSTATUS
|
||||||
ScmStartService(PSERVICE Service)
|
ScmStartService(PSERVICE Service,
|
||||||
|
PSERVICE_GROUP Group)
|
||||||
{
|
{
|
||||||
#if 0
|
RTL_QUERY_REGISTRY_TABLE QueryTable[3];
|
||||||
PROCESS_INFORMATION ProcessInformation;
|
PROCESS_INFORMATION ProcessInformation;
|
||||||
STARTUPINFO StartupInfo;
|
STARTUPINFOW StartupInfo;
|
||||||
WCHAR CommandLine[MAX_PATH];
|
UNICODE_STRING ImagePath;
|
||||||
|
NTSTATUS Status;
|
||||||
|
ULONG Type;
|
||||||
BOOL Result;
|
BOOL Result;
|
||||||
|
|
||||||
PrintString("ScmStartService() called\n");
|
DPRINT("ScmStartService() called\n");
|
||||||
|
|
||||||
GetSystemDirectoryW(CommandLine, MAX_PATH);
|
if (Service->Type == SERVICE_KERNEL_DRIVER ||
|
||||||
_tcscat(CommandLine, "\\");
|
Service->Type == SERVICE_FILE_SYSTEM_DRIVER ||
|
||||||
_tcscat(CommandLine, FileName);
|
Service->Type == SERVICE_RECOGNIZER_DRIVER)
|
||||||
|
|
||||||
PrintString("SCM: %s\n", CommandLine);
|
|
||||||
|
|
||||||
/* FIXME: create '\\.\pipe\net\NtControlPipe' instance */
|
|
||||||
|
|
||||||
StartupInfo.cb = sizeof(StartupInfo);
|
|
||||||
StartupInfo.lpReserved = NULL;
|
|
||||||
StartupInfo.lpDesktop = NULL;
|
|
||||||
StartupInfo.lpTitle = NULL;
|
|
||||||
StartupInfo.dwFlags = 0;
|
|
||||||
StartupInfo.cbReserved2 = 0;
|
|
||||||
StartupInfo.lpReserved2 = 0;
|
|
||||||
|
|
||||||
Result = CreateProcessW(CommandLine,
|
|
||||||
NULL,
|
|
||||||
NULL,
|
|
||||||
NULL,
|
|
||||||
FALSE,
|
|
||||||
DETACHED_PROCESS,
|
|
||||||
NULL,
|
|
||||||
NULL,
|
|
||||||
&StartupInfo,
|
|
||||||
&ProcessInformation);
|
|
||||||
if (!Result)
|
|
||||||
{
|
{
|
||||||
/* FIXME: close control pipe */
|
/* Load driver */
|
||||||
|
DPRINT(" Path: %wZ\n", &Service->RegistryPath);
|
||||||
|
Status = NtLoadDriver(&Service->RegistryPath);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
RtlInitUnicodeString(&ImagePath, NULL);
|
||||||
|
|
||||||
PrintString("SCM: Failed to start '%s'\n", FileName);
|
/* Get service data */
|
||||||
return(STATUS_UNSUCCESSFUL);
|
RtlZeroMemory(&QueryTable,
|
||||||
|
sizeof(QueryTable));
|
||||||
|
|
||||||
|
QueryTable[0].Name = L"Type";
|
||||||
|
QueryTable[0].Flags = RTL_QUERY_REGISTRY_DIRECT | RTL_QUERY_REGISTRY_REQUIRED;
|
||||||
|
QueryTable[0].EntryContext = &Type;
|
||||||
|
|
||||||
|
QueryTable[1].Name = L"ImagePath";
|
||||||
|
QueryTable[1].Flags = RTL_QUERY_REGISTRY_DIRECT | RTL_QUERY_REGISTRY_REQUIRED;
|
||||||
|
QueryTable[1].EntryContext = &ImagePath;
|
||||||
|
|
||||||
|
Status = RtlQueryRegistryValues(RTL_REGISTRY_SERVICES,
|
||||||
|
Service->ServiceName.Buffer,
|
||||||
|
QueryTable,
|
||||||
|
NULL,
|
||||||
|
NULL);
|
||||||
|
if (NT_SUCCESS(Status))
|
||||||
|
{
|
||||||
|
DPRINT("ImagePath: '%S'\n", ImagePath.Buffer);
|
||||||
|
DPRINT("Type: %lx\n", Type);
|
||||||
|
|
||||||
|
/* FIXME: create '\\.\pipe\net\NtControlPipe' instance */
|
||||||
|
|
||||||
|
StartupInfo.cb = sizeof(StartupInfo);
|
||||||
|
StartupInfo.lpReserved = NULL;
|
||||||
|
StartupInfo.lpDesktop = NULL;
|
||||||
|
StartupInfo.lpTitle = NULL;
|
||||||
|
StartupInfo.dwFlags = 0;
|
||||||
|
StartupInfo.cbReserved2 = 0;
|
||||||
|
StartupInfo.lpReserved2 = 0;
|
||||||
|
|
||||||
|
Result = CreateProcessW(ImagePath.Buffer,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
FALSE,
|
||||||
|
DETACHED_PROCESS,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
&StartupInfo,
|
||||||
|
&ProcessInformation);
|
||||||
|
|
||||||
|
RtlFreeUnicodeString(&ImagePath);
|
||||||
|
|
||||||
|
if (!Result)
|
||||||
|
{
|
||||||
|
/* FIXME: close control pipe */
|
||||||
|
|
||||||
|
DPRINT("Failed to start '%S'\n", Service->ServiceName.Buffer);
|
||||||
|
Status = STATUS_UNSUCCESSFUL;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* FIXME: connect control pipe */
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* FIXME: connect control pipe */
|
if (NT_SUCCESS(Status))
|
||||||
|
{
|
||||||
|
if (Group != NULL)
|
||||||
|
{
|
||||||
|
Group->ServicesRunning = TRUE;
|
||||||
|
}
|
||||||
|
Service->ServiceRunning = TRUE;
|
||||||
|
}
|
||||||
|
#if 0
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (CurrentService->ErrorControl == 1)
|
||||||
|
{
|
||||||
|
/* Log error */
|
||||||
|
|
||||||
|
}
|
||||||
|
else if (CurrentService->ErrorControl == 2)
|
||||||
|
{
|
||||||
|
if (IsLastKnownGood == FALSE)
|
||||||
|
{
|
||||||
|
/* Boot last known good configuration */
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (CurrentService->ErrorControl == 3)
|
||||||
|
{
|
||||||
|
if (IsLastKnownGood == FALSE)
|
||||||
|
{
|
||||||
|
/* Boot last known good configuration */
|
||||||
|
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* BSOD! */
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return(STATUS_SUCCESS);
|
return(STATUS_SUCCESS);
|
||||||
|
@ -379,77 +561,87 @@ ScmAutoStartServices(VOID)
|
||||||
PSERVICE CurrentService;
|
PSERVICE CurrentService;
|
||||||
NTSTATUS Status;
|
NTSTATUS Status;
|
||||||
|
|
||||||
|
/* Clear 'ServiceVisited' flag */
|
||||||
|
ServiceEntry = ServiceListHead.Flink;
|
||||||
|
while (ServiceEntry != &ServiceListHead)
|
||||||
|
{
|
||||||
|
CurrentService = CONTAINING_RECORD(ServiceEntry, SERVICE, ServiceListEntry);
|
||||||
|
CurrentService->ServiceVisited = FALSE;
|
||||||
|
ServiceEntry = ServiceEntry->Flink;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Start all services which are members of an existing group */
|
||||||
GroupEntry = GroupListHead.Flink;
|
GroupEntry = GroupListHead.Flink;
|
||||||
while (GroupEntry != &GroupListHead)
|
while (GroupEntry != &GroupListHead)
|
||||||
{
|
{
|
||||||
CurrentGroup = CONTAINING_RECORD(GroupEntry, SERVICE_GROUP, GroupListEntry);
|
CurrentGroup = CONTAINING_RECORD(GroupEntry, SERVICE_GROUP, GroupListEntry);
|
||||||
|
|
||||||
// PrintString("Group '%wZ'\n", &CurrentGroup->GroupName);
|
DPRINT("Group '%wZ'\n", &CurrentGroup->GroupName);
|
||||||
|
|
||||||
ServiceEntry = ServiceListHead.Flink;
|
ServiceEntry = ServiceListHead.Flink;
|
||||||
while (ServiceEntry != &ServiceListHead)
|
while (ServiceEntry != &ServiceListHead)
|
||||||
{
|
{
|
||||||
CurrentService = CONTAINING_RECORD(ServiceEntry, SERVICE, ServiceListEntry);
|
CurrentService = CONTAINING_RECORD(ServiceEntry, SERVICE, ServiceListEntry);
|
||||||
|
|
||||||
if ((RtlCompareUnicodeString(&CurrentGroup->GroupName, &CurrentService->ServiceGroup, TRUE) == 0) &&
|
if ((RtlEqualUnicodeString(&CurrentGroup->GroupName, &CurrentService->ServiceGroup, TRUE)) &&
|
||||||
(CurrentService->Start == SERVICE_AUTO_START))
|
(CurrentService->Start == SERVICE_AUTO_START) &&
|
||||||
|
(CurrentService->ServiceVisited == FALSE))
|
||||||
{
|
{
|
||||||
if (CurrentService->Type == SERVICE_KERNEL_DRIVER ||
|
CurrentService->ServiceVisited = TRUE;
|
||||||
CurrentService->Type == SERVICE_FILE_SYSTEM_DRIVER ||
|
ScmStartService(CurrentService,
|
||||||
CurrentService->Type == SERVICE_RECOGNIZER_DRIVER)
|
CurrentGroup);
|
||||||
{
|
|
||||||
/* Load driver */
|
|
||||||
// PrintString(" Path: %wZ\n", &CurrentService->RegistryPath);
|
|
||||||
Status = NtLoadDriver(&CurrentService->RegistryPath);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
/* Start service */
|
|
||||||
Status = ScmStartService(CurrentService);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (NT_SUCCESS(Status))
|
|
||||||
{
|
|
||||||
CurrentGroup->ServicesRunning = TRUE;
|
|
||||||
CurrentService->ServiceRunning = TRUE;
|
|
||||||
}
|
|
||||||
#if 0
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (CurrentService->ErrorControl == 1)
|
|
||||||
{
|
|
||||||
/* Log error */
|
|
||||||
|
|
||||||
}
|
|
||||||
else if (CurrentService->ErrorControl == 2)
|
|
||||||
{
|
|
||||||
if (IsLastKnownGood == FALSE)
|
|
||||||
{
|
|
||||||
/* Boot last known good configuration */
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (CurrentService->ErrorControl == 3)
|
|
||||||
{
|
|
||||||
if (IsLastKnownGood == FALSE)
|
|
||||||
{
|
|
||||||
/* Boot last known good configuration */
|
|
||||||
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
/* BSOD! */
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ServiceEntry = ServiceEntry->Flink;
|
ServiceEntry = ServiceEntry->Flink;
|
||||||
}
|
}
|
||||||
|
|
||||||
GroupEntry = GroupEntry->Flink;
|
GroupEntry = GroupEntry->Flink;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Start all services which are members of any non-existing group */
|
||||||
|
ServiceEntry = ServiceListHead.Flink;
|
||||||
|
while (ServiceEntry != &ServiceListHead)
|
||||||
|
{
|
||||||
|
CurrentService = CONTAINING_RECORD(ServiceEntry, SERVICE, ServiceListEntry);
|
||||||
|
|
||||||
|
if ((CurrentGroup->GroupName.Length > 0) &&
|
||||||
|
(CurrentService->Start == SERVICE_AUTO_START) &&
|
||||||
|
(CurrentService->ServiceVisited == FALSE))
|
||||||
|
{
|
||||||
|
CurrentService->ServiceVisited = TRUE;
|
||||||
|
ScmStartService(CurrentService,
|
||||||
|
NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
ServiceEntry = ServiceEntry->Flink;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Start all services which are not a member of any group */
|
||||||
|
ServiceEntry = ServiceListHead.Flink;
|
||||||
|
while (ServiceEntry != &ServiceListHead)
|
||||||
|
{
|
||||||
|
CurrentService = CONTAINING_RECORD(ServiceEntry, SERVICE, ServiceListEntry);
|
||||||
|
|
||||||
|
if ((CurrentGroup->GroupName.Length == 0) &&
|
||||||
|
(CurrentService->Start == SERVICE_AUTO_START) &&
|
||||||
|
(CurrentService->ServiceVisited == FALSE))
|
||||||
|
{
|
||||||
|
CurrentService->ServiceVisited = TRUE;
|
||||||
|
ScmStartService(CurrentService,
|
||||||
|
NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
ServiceEntry = ServiceEntry->Flink;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Clear 'ServiceVisited' flag again */
|
||||||
|
ServiceEntry = ServiceListHead.Flink;
|
||||||
|
while (ServiceEntry != &ServiceListHead)
|
||||||
|
{
|
||||||
|
CurrentService = CONTAINING_RECORD(ServiceEntry, SERVICE, ServiceListEntry);
|
||||||
|
CurrentService->ServiceVisited = FALSE;
|
||||||
|
ServiceEntry = ServiceEntry->Flink;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* EOF */
|
/* EOF */
|
||||||
|
|
Loading…
Reference in a new issue