mirror of
https://github.com/reactos/reactos.git
synced 2025-01-04 05:20:54 +00:00
f87fb4e3df
RtlGetNtProductType comes into two variants: one in user-mode that is exported for use from NTDLL layer and the kernel-mode that is used exclusively by the NT kernel. The kernel-mode variant of the function is not exported.
225 lines
7 KiB
C
225 lines
7 KiB
C
/*
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
* PROJECT: ReactOS kernel
|
|
* FILE: ntoskrnl/rtl/misc.c
|
|
* PURPOSE: Various functions
|
|
*
|
|
* PROGRAMMERS:
|
|
*/
|
|
|
|
/* INCLUDES *****************************************************************/
|
|
|
|
#include <ntoskrnl.h>
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
/* GLOBALS *******************************************************************/
|
|
|
|
extern ULONG NtGlobalFlag;
|
|
extern ULONG NtMajorVersion;
|
|
extern ULONG NtMinorVersion;
|
|
extern ULONG NtOSCSDVersion;
|
|
#define PRODUCT_TAG 'iPtR'
|
|
|
|
/* FUNCTIONS *****************************************************************/
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
ULONG
|
|
NTAPI
|
|
RtlGetNtGlobalFlags(VOID)
|
|
{
|
|
return NtGlobalFlag;
|
|
}
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
NTSTATUS
|
|
NTAPI
|
|
RtlGetVersion(IN OUT PRTL_OSVERSIONINFOW lpVersionInformation)
|
|
{
|
|
PAGED_CODE();
|
|
|
|
/* Return the basics */
|
|
lpVersionInformation->dwMajorVersion = NtMajorVersion;
|
|
lpVersionInformation->dwMinorVersion = NtMinorVersion;
|
|
lpVersionInformation->dwBuildNumber = NtBuildNumber & 0x3FFF;
|
|
lpVersionInformation->dwPlatformId = VER_PLATFORM_WIN32_NT;
|
|
|
|
/* Check if this is the extended version */
|
|
if (lpVersionInformation->dwOSVersionInfoSize == sizeof(RTL_OSVERSIONINFOEXW))
|
|
{
|
|
PRTL_OSVERSIONINFOEXW InfoEx = (PRTL_OSVERSIONINFOEXW)lpVersionInformation;
|
|
InfoEx->wServicePackMajor = (USHORT)(CmNtCSDVersion >> 8) & 0xFF;
|
|
InfoEx->wServicePackMinor = (USHORT)(CmNtCSDVersion & 0xFF);
|
|
InfoEx->wSuiteMask = (USHORT)(SharedUserData->SuiteMask & 0xFFFF);
|
|
InfoEx->wProductType = SharedUserData->NtProductType;
|
|
InfoEx->wReserved = 0;
|
|
}
|
|
|
|
/* Always succeed */
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief
|
|
* Retrieves the NT type product of the operating system. This is the kernel-mode variant
|
|
* of this function.
|
|
*
|
|
* @param[out] ProductType
|
|
* The NT type product enumeration value returned by the call.
|
|
*
|
|
* @return
|
|
* The function returns TRUE when the call successfully returned the type product of the system.
|
|
* It'll return FALSE on failure otherwise. In the latter case the function will return WinNT
|
|
* as the default product type.
|
|
*
|
|
* @remarks
|
|
* The call expects to be called at PASSIVE_LEVEL. The function firstly checks if the product type is
|
|
* actually valid by checking the "ProductTypeIsValid" member of _KUSER_SHARED_DATA structure.
|
|
* Currently we do not implement code that is responsible for the management of this member, yet.
|
|
*
|
|
*/
|
|
BOOLEAN
|
|
NTAPI
|
|
RtlGetNtProductType(OUT PNT_PRODUCT_TYPE ProductType)
|
|
{
|
|
HANDLE Key;
|
|
BOOLEAN Success;
|
|
ULONG ReturnedLength;
|
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
|
NTSTATUS Status;
|
|
PKEY_VALUE_PARTIAL_INFORMATION BufferKey;
|
|
ULONG BufferKeyLength = sizeof(PKEY_VALUE_PARTIAL_INFORMATION) + (256 * sizeof(WCHAR));
|
|
UNICODE_STRING NtProductType;
|
|
static UNICODE_STRING WorkstationProduct = RTL_CONSTANT_STRING(L"WinNT");
|
|
static UNICODE_STRING LanManProduct = RTL_CONSTANT_STRING(L"LanmanNT");
|
|
static UNICODE_STRING ServerProduct = RTL_CONSTANT_STRING(L"ServerNT");
|
|
static UNICODE_STRING KeyProduct = RTL_CONSTANT_STRING(L"\\Registry\\Machine\\SYSTEM\\CurrentControlSet\\Control\\ProductOptions");
|
|
static UNICODE_STRING ValueNtProduct = RTL_CONSTANT_STRING(L"ProductType");
|
|
|
|
PAGED_CODE();
|
|
|
|
/* Before doing anything else we must allocate some buffer space in the pool */
|
|
BufferKey = ExAllocatePoolWithTag(PagedPool, BufferKeyLength, PRODUCT_TAG);
|
|
if (!BufferKey)
|
|
{
|
|
/* We failed to allocate pool memory, bail out */
|
|
DPRINT1("RtlGetNtProductType(): Memory pool allocation has failed!\n");
|
|
Success = FALSE;
|
|
goto Exit;
|
|
}
|
|
|
|
/* Initialize the object attributes to open our key */
|
|
InitializeObjectAttributes(&ObjectAttributes,
|
|
&KeyProduct,
|
|
OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE,
|
|
NULL,
|
|
NULL);
|
|
|
|
/* Open the key */
|
|
Status = ZwOpenKey(&Key, KEY_QUERY_VALUE, &ObjectAttributes);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("RtlGetNtProductType(): The ZwOpenKey() function has failed! (Status: 0x%lx)\n", Status);
|
|
Success = FALSE;
|
|
goto Exit;
|
|
}
|
|
|
|
/* Now it's time to query the value from the key to check what kind of product the OS is */
|
|
Status = ZwQueryValueKey(Key,
|
|
&ValueNtProduct,
|
|
KeyValuePartialInformation,
|
|
BufferKey,
|
|
BufferKeyLength,
|
|
&ReturnedLength);
|
|
|
|
/* Free the key from the memory */
|
|
ZwClose(Key);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("RtlGetNtProductType(): The ZwQueryValueKey() function has failed! (Status: 0x%lx)\n", Status);
|
|
Success = FALSE;
|
|
goto Exit;
|
|
}
|
|
|
|
/*
|
|
* Do a sanity check before we get the product type of the operating system
|
|
* so that we're sure the type of the value we've got is actually correct.
|
|
*/
|
|
if (BufferKey->Type != REG_SZ)
|
|
{
|
|
/* We've got something else, so bail out */
|
|
DPRINT1("RtlGetNtProductType(): An invalid value type has been found!\n");
|
|
Success = FALSE;
|
|
goto Exit;
|
|
}
|
|
|
|
/* Initialise the Unicode string with the data from the registry value */
|
|
NtProductType.Length = NtProductType.MaximumLength = BufferKey->DataLength;
|
|
NtProductType.Buffer = (PWCHAR)BufferKey->Data;
|
|
|
|
if (NtProductType.Length > sizeof(WCHAR) && NtProductType.Buffer[NtProductType.Length /
|
|
sizeof(WCHAR) - 1] == UNICODE_NULL)
|
|
{
|
|
NtProductType.Length -= sizeof(WCHAR);
|
|
}
|
|
|
|
/* Now it's time to get the product based on the value data */
|
|
if (RtlCompareUnicodeString(&NtProductType, &WorkstationProduct, TRUE) == 0)
|
|
{
|
|
/* The following product is an OS Workstation */
|
|
*ProductType = NtProductWinNt;
|
|
Success = TRUE;
|
|
}
|
|
else if (RtlCompareUnicodeString(&NtProductType, &LanManProduct, TRUE) == 0)
|
|
{
|
|
/* The following product is an OS Advanced Server */
|
|
*ProductType = NtProductLanManNt;
|
|
Success = TRUE;
|
|
}
|
|
else if (RtlCompareUnicodeString(&NtProductType, &ServerProduct, TRUE) == 0)
|
|
{
|
|
/* The following product is an OS Server */
|
|
*ProductType = NtProductServer;
|
|
Success = TRUE;
|
|
}
|
|
else
|
|
{
|
|
/* None of the product types match so bail out */
|
|
DPRINT1("RtlGetNtProductType(): Couldn't find a valid product type! Defaulting to WinNT...\n");
|
|
Success = FALSE;
|
|
goto Exit;
|
|
}
|
|
|
|
Exit:
|
|
if (!Success)
|
|
{
|
|
*ProductType = NtProductWinNt;
|
|
}
|
|
|
|
ExFreePoolWithTag(BufferKey, PRODUCT_TAG);
|
|
return Success;
|
|
}
|
|
|
|
#if !defined(_M_IX86)
|
|
//
|
|
// Stub for architectures which don't have this implemented
|
|
//
|
|
VOID
|
|
FASTCALL
|
|
RtlPrefetchMemoryNonTemporal(IN PVOID Source,
|
|
IN SIZE_T Length)
|
|
{
|
|
//
|
|
// Do nothing
|
|
//
|
|
UNREFERENCED_PARAMETER(Source);
|
|
UNREFERENCED_PARAMETER(Length);
|
|
}
|
|
#endif
|
|
|
|
/* EOF */
|