2005-12-31 19:02:48 +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/groupdb.c
|
|
|
|
* PURPOSE: Service group control interface
|
|
|
|
* COPYRIGHT: Copyright 2005 Eric Kohl
|
|
|
|
*
|
2005-12-31 19:02:48 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/* INCLUDES *****************************************************************/
|
|
|
|
|
|
|
|
#include "services.h"
|
|
|
|
|
|
|
|
#define NDEBUG
|
|
|
|
#include <debug.h>
|
|
|
|
|
|
|
|
/* GLOBALS *******************************************************************/
|
|
|
|
|
|
|
|
LIST_ENTRY GroupListHead;
|
2005-12-31 22:42:41 +00:00
|
|
|
LIST_ENTRY UnknownGroupListHead;
|
2005-12-31 19:02:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
/* FUNCTIONS *****************************************************************/
|
|
|
|
|
2016-12-03 12:48:44 +00:00
|
|
|
PSERVICE_GROUP
|
|
|
|
ScmGetServiceGroupByName(
|
|
|
|
_In_ LPCWSTR lpGroupName)
|
|
|
|
{
|
|
|
|
PLIST_ENTRY GroupEntry;
|
|
|
|
PSERVICE_GROUP lpGroup;
|
|
|
|
|
|
|
|
DPRINT("ScmGetServiceGroupByName(%S)\n", lpGroupName);
|
|
|
|
|
|
|
|
GroupEntry = GroupListHead.Flink;
|
|
|
|
while (GroupEntry != &GroupListHead)
|
|
|
|
{
|
|
|
|
lpGroup = CONTAINING_RECORD(GroupEntry, SERVICE_GROUP, GroupListEntry);
|
|
|
|
|
|
|
|
if (!_wcsicmp(lpGroup->lpGroupName, lpGroupName))
|
|
|
|
return lpGroup;
|
|
|
|
|
|
|
|
GroupEntry = GroupEntry->Flink;
|
|
|
|
}
|
|
|
|
|
|
|
|
GroupEntry = UnknownGroupListHead.Flink;
|
|
|
|
while (GroupEntry != &UnknownGroupListHead)
|
|
|
|
{
|
|
|
|
lpGroup = CONTAINING_RECORD(GroupEntry, SERVICE_GROUP, GroupListEntry);
|
|
|
|
|
|
|
|
if (!_wcsicmp(lpGroup->lpGroupName, lpGroupName))
|
|
|
|
return lpGroup;
|
|
|
|
|
|
|
|
GroupEntry = GroupEntry->Flink;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-31 22:42:41 +00:00
|
|
|
DWORD
|
|
|
|
ScmSetServiceGroup(PSERVICE lpService,
|
2009-01-06 07:53:04 +00:00
|
|
|
LPCWSTR lpGroupName)
|
2005-12-31 22:42:41 +00:00
|
|
|
{
|
|
|
|
PLIST_ENTRY GroupEntry;
|
|
|
|
PSERVICE_GROUP lpGroup;
|
|
|
|
|
2009-12-23 14:32:15 +00:00
|
|
|
DPRINT("ScmSetServiceGroup(%S)\n", lpGroupName);
|
|
|
|
|
|
|
|
if (lpService->lpGroup != NULL)
|
|
|
|
{
|
2015-04-05 08:58:08 +00:00
|
|
|
ASSERT(lpService->lpGroup->dwRefCount != 0);
|
|
|
|
ASSERT(lpService->lpGroup->dwRefCount == (DWORD)-1 ||
|
|
|
|
lpService->lpGroup->dwRefCount < 10000);
|
|
|
|
if (lpService->lpGroup->dwRefCount != (DWORD)-1)
|
|
|
|
{
|
|
|
|
lpService->lpGroup->dwRefCount--;
|
|
|
|
if (lpService->lpGroup->dwRefCount == 0)
|
|
|
|
{
|
|
|
|
ASSERT(lpService->lpGroup->TagCount == 0);
|
|
|
|
ASSERT(lpService->lpGroup->TagArray == NULL);
|
|
|
|
RemoveEntryList(&lpService->lpGroup->GroupListEntry);
|
|
|
|
HeapFree(GetProcessHeap(), 0, lpService->lpGroup);
|
|
|
|
lpService->lpGroup = NULL;
|
|
|
|
}
|
|
|
|
}
|
2009-12-23 14:32:15 +00:00
|
|
|
}
|
|
|
|
|
2015-04-05 08:58:08 +00:00
|
|
|
if (lpGroupName == NULL)
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
|
2005-12-31 22:42:41 +00:00
|
|
|
GroupEntry = GroupListHead.Flink;
|
|
|
|
while (GroupEntry != &GroupListHead)
|
|
|
|
{
|
|
|
|
lpGroup = CONTAINING_RECORD(GroupEntry, SERVICE_GROUP, GroupListEntry);
|
|
|
|
|
|
|
|
if (!_wcsicmp(lpGroup->lpGroupName, lpGroupName))
|
|
|
|
{
|
|
|
|
lpService->lpGroup = lpGroup;
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
GroupEntry = GroupEntry->Flink;
|
|
|
|
}
|
|
|
|
|
|
|
|
GroupEntry = UnknownGroupListHead.Flink;
|
|
|
|
while (GroupEntry != &UnknownGroupListHead)
|
|
|
|
{
|
|
|
|
lpGroup = CONTAINING_RECORD(GroupEntry, SERVICE_GROUP, GroupListEntry);
|
|
|
|
|
|
|
|
if (!_wcsicmp(lpGroup->lpGroupName, lpGroupName))
|
|
|
|
{
|
|
|
|
lpGroup->dwRefCount++;
|
|
|
|
lpService->lpGroup = lpGroup;
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
GroupEntry = GroupEntry->Flink;
|
|
|
|
}
|
|
|
|
|
|
|
|
lpGroup = (PSERVICE_GROUP)HeapAlloc(GetProcessHeap(),
|
|
|
|
HEAP_ZERO_MEMORY,
|
2008-09-11 08:31:52 +00:00
|
|
|
sizeof(SERVICE_GROUP) + ((wcslen(lpGroupName) + 1)* sizeof(WCHAR)));
|
2005-12-31 22:42:41 +00:00
|
|
|
if (lpGroup == NULL)
|
|
|
|
return ERROR_NOT_ENOUGH_MEMORY;
|
|
|
|
|
|
|
|
wcscpy(lpGroup->szGroupName, lpGroupName);
|
|
|
|
lpGroup->lpGroupName = lpGroup->szGroupName;
|
|
|
|
lpGroup->dwRefCount = 1;
|
2006-06-03 11:47:04 +00:00
|
|
|
lpService->lpGroup = lpGroup;
|
2005-12-31 22:42:41 +00:00
|
|
|
|
|
|
|
InsertTailList(&UnknownGroupListHead,
|
|
|
|
&lpGroup->GroupListEntry);
|
|
|
|
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-30 11:16:55 +00:00
|
|
|
static NTSTATUS WINAPI
|
2005-12-31 19:02:48 +00:00
|
|
|
CreateGroupOrderListRoutine(PWSTR ValueName,
|
|
|
|
ULONG ValueType,
|
|
|
|
PVOID ValueData,
|
|
|
|
ULONG ValueLength,
|
|
|
|
PVOID Context,
|
|
|
|
PVOID EntryContext)
|
|
|
|
{
|
|
|
|
PSERVICE_GROUP Group;
|
|
|
|
|
2013-03-08 20:49:03 +00:00
|
|
|
DPRINT("CreateGroupOrderListRoutine(%S, %x, %p, %x, %p, %p)\n",
|
2005-12-31 19:02:48 +00:00
|
|
|
ValueName, ValueType, ValueData, ValueLength, Context, EntryContext);
|
|
|
|
|
|
|
|
if (ValueType == REG_BINARY &&
|
|
|
|
ValueData != NULL &&
|
|
|
|
ValueLength >= sizeof(DWORD) &&
|
|
|
|
ValueLength >= (*(PULONG)ValueData + 1) * sizeof(DWORD))
|
|
|
|
{
|
|
|
|
Group = (PSERVICE_GROUP)Context;
|
|
|
|
Group->TagCount = ((PULONG)ValueData)[0];
|
|
|
|
if (Group->TagCount > 0)
|
|
|
|
{
|
|
|
|
if (ValueLength >= (Group->TagCount + 1) * sizeof(DWORD))
|
|
|
|
{
|
|
|
|
Group->TagArray = (PULONG)HeapAlloc(GetProcessHeap(),
|
|
|
|
HEAP_ZERO_MEMORY,
|
|
|
|
Group->TagCount * sizeof(DWORD));
|
|
|
|
if (Group->TagArray == NULL)
|
|
|
|
{
|
|
|
|
Group->TagCount = 0;
|
|
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
RtlCopyMemory(Group->TagArray,
|
|
|
|
(PULONG)ValueData + 1,
|
|
|
|
Group->TagCount * sizeof(DWORD));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Group->TagCount = 0;
|
|
|
|
return STATUS_UNSUCCESSFUL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-30 11:16:55 +00:00
|
|
|
static NTSTATUS WINAPI
|
2005-12-31 19:02:48 +00:00
|
|
|
CreateGroupListRoutine(PWSTR ValueName,
|
|
|
|
ULONG ValueType,
|
|
|
|
PVOID ValueData,
|
|
|
|
ULONG ValueLength,
|
|
|
|
PVOID Context,
|
|
|
|
PVOID EntryContext)
|
|
|
|
{
|
|
|
|
PSERVICE_GROUP Group;
|
|
|
|
RTL_QUERY_REGISTRY_TABLE QueryTable[2];
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
|
|
|
if (ValueType == REG_SZ)
|
|
|
|
{
|
|
|
|
DPRINT("Data: '%S'\n", (PWCHAR)ValueData);
|
|
|
|
|
|
|
|
Group = (PSERVICE_GROUP)HeapAlloc(GetProcessHeap(),
|
|
|
|
HEAP_ZERO_MEMORY,
|
2008-09-11 08:31:52 +00:00
|
|
|
sizeof(SERVICE_GROUP) + ((wcslen((const wchar_t*) ValueData) + 1) * sizeof(WCHAR)));
|
2005-12-31 19:02:48 +00:00
|
|
|
if (Group == NULL)
|
|
|
|
{
|
|
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
|
|
}
|
|
|
|
|
2007-01-05 20:22:44 +00:00
|
|
|
wcscpy(Group->szGroupName, (const wchar_t*) ValueData);
|
2005-12-31 19:02:48 +00:00
|
|
|
Group->lpGroupName = Group->szGroupName;
|
|
|
|
Group->dwRefCount = (DWORD)-1;
|
|
|
|
|
|
|
|
RtlZeroMemory(&QueryTable, sizeof(QueryTable));
|
|
|
|
QueryTable[0].Name = (PWSTR)ValueData;
|
|
|
|
QueryTable[0].QueryRoutine = CreateGroupOrderListRoutine;
|
|
|
|
|
|
|
|
Status = RtlQueryRegistryValues(RTL_REGISTRY_CONTROL,
|
|
|
|
L"GroupOrderList",
|
|
|
|
QueryTable,
|
|
|
|
(PVOID)Group,
|
|
|
|
NULL);
|
2013-03-08 20:49:03 +00:00
|
|
|
DPRINT("%x %lu %S\n", Status, Group->TagCount, (PWSTR)ValueData);
|
2005-12-31 19:02:48 +00:00
|
|
|
|
|
|
|
InsertTailList(&GroupListHead,
|
|
|
|
&Group->GroupListEntry);
|
|
|
|
}
|
|
|
|
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DWORD
|
|
|
|
ScmCreateGroupList(VOID)
|
|
|
|
{
|
|
|
|
RTL_QUERY_REGISTRY_TABLE QueryTable[2];
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
|
|
|
InitializeListHead(&GroupListHead);
|
2005-12-31 22:42:41 +00:00
|
|
|
InitializeListHead(&UnknownGroupListHead);
|
2005-12-31 19:02:48 +00:00
|
|
|
|
|
|
|
/* Build group order list */
|
|
|
|
RtlZeroMemory(&QueryTable,
|
|
|
|
sizeof(QueryTable));
|
|
|
|
|
|
|
|
QueryTable[0].Name = L"List";
|
|
|
|
QueryTable[0].QueryRoutine = CreateGroupListRoutine;
|
|
|
|
|
|
|
|
Status = RtlQueryRegistryValues(RTL_REGISTRY_CONTROL,
|
|
|
|
L"ServiceGroupOrder",
|
|
|
|
QueryTable,
|
|
|
|
NULL,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
return RtlNtStatusToDosError(Status);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* EOF */
|