ProcessHacker/phlib/basesup.c
2025-05-13 19:45:22 +03:00

5984 lines
148 KiB
C

/*
* Process Hacker -
* base support functions
*
* Copyright (C) 2009-2016 wj32
*
* This file is part of Process Hacker.
*
* Process Hacker is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Process Hacker is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Process Hacker. If not, see <http://www.gnu.org/licenses/>.
*/
/*
* This file contains basic low-level code as well as general algorithms and data structures.
*
* Memory allocation. PhAllocate is a wrapper around RtlAllocateHeap, and always allocates from the
* phlib heap. PhAllocatePage is a wrapper around NtAllocateVirtualMemory and allocates pages.
*
* Null-terminated strings. The Ph*StringZ functions manipulate null-terminated strings. The copying
* functions provide a simple way to copy strings which may not be null-terminated, but have a
* specified limit.
*
* String. The design of the string object was chosen for maximum compatibility. As such each string
* buffer must be null-terminated, and each object contains an embedded PH_STRINGREF structure. Note
* that efficient sub-string creation (no copying, only references the parent string object) could
* not be implemented due to the mandatory null-termination. String objects must be regarded as
* immutable (for thread-safety reasons) unless the object has just been created and no references
* have been shared.
*
* String builder. This is a set of functions which allow for efficient modification of strings. For
* performance reasons, these functions modify string objects directly, even though they are
* normally immutable.
*
* List. A simple PVOID list that resizes itself when needed.
*
* Pointer list. Similar to the normal list object, but uses a free list in order to support
* constant time insertion and deletion. In order for the free list to work, normal entries have
* their lowest bit clear while free entries have their lowest bit set, with the index of the next
* free entry in the upper bits.
*
* Hashtable. A hashtable with power-of-two bucket sizes and with all entries stored in a single
* array. This improves locality but may be inefficient when resizing the hashtable. It is a good
* idea to store pointers to objects as entries, as opposed to the objects themselves.
*
* Simple hashtable. A wrapper around the normal hashtable, with PVOID keys and PVOID values.
*
* Free list. A thread-safe memory allocation method where freed blocks are stored in a S-list, and
* allocations are made from this list whenever possible.
*
* Callback. A thread-safe notification mechanism where clients can register callback functions
* which are then invoked by other code.
*/
#include <phbase.h>
#include <phintrnl.h>
#include <math.h>
#include <objbase.h>
#define PH_VECTOR_LEVEL_NONE 0
#define PH_VECTOR_LEVEL_SSE2 1
#define PH_VECTOR_LEVEL_AVX 2
typedef struct _PHP_BASE_THREAD_CONTEXT
{
PUSER_THREAD_START_ROUTINE StartAddress;
PVOID Parameter;
} PHP_BASE_THREAD_CONTEXT, *PPHP_BASE_THREAD_CONTEXT;
VOID NTAPI PhpListDeleteProcedure(
_In_ PVOID Object,
_In_ ULONG Flags
);
VOID NTAPI PhpPointerListDeleteProcedure(
_In_ PVOID Object,
_In_ ULONG Flags
);
VOID NTAPI PhpQueueDeleteProcedure(
_In_ PVOID Object,
_In_ ULONG Flags
);
VOID NTAPI PhpHashtableDeleteProcedure(
_In_ PVOID Object,
_In_ ULONG Flags
);
// Types
PPH_OBJECT_TYPE PhStringType;
PPH_OBJECT_TYPE PhBytesType;
PPH_OBJECT_TYPE PhListType;
PPH_OBJECT_TYPE PhPointerListType;
PPH_OBJECT_TYPE PhHashtableType;
// Misc.
static BOOLEAN PhpVectorLevel = PH_VECTOR_LEVEL_NONE;
static PPH_STRING PhSharedEmptyString = NULL;
// Threads
static PH_FREE_LIST PhpBaseThreadContextFreeList;
#ifdef DEBUG
ULONG PhDbgThreadDbgTlsIndex;
LIST_ENTRY PhDbgThreadListHead;
PH_QUEUED_LOCK PhDbgThreadListLock = PH_QUEUED_LOCK_INIT;
#endif
// Data
static ULONG PhpPrimeNumbers[] =
{
0x3, 0x7, 0xb, 0x11, 0x17, 0x1d, 0x25, 0x2f, 0x3b, 0x47, 0x59, 0x6b, 0x83,
0xa3, 0xc5, 0xef, 0x125, 0x161, 0x1af, 0x209, 0x277, 0x2f9, 0x397, 0x44f,
0x52f, 0x63d, 0x78b, 0x91d, 0xaf1, 0xd2b, 0xfd1, 0x12fd, 0x16cf, 0x1b65,
0x20e3, 0x2777, 0x2f6f, 0x38ff, 0x446f, 0x521f, 0x628d, 0x7655, 0x8e01,
0xaa6b, 0xcc89, 0xf583, 0x126a7, 0x1619b, 0x1a857, 0x1fd3b, 0x26315, 0x2dd67,
0x3701b, 0x42023, 0x4f361, 0x5f0ed, 0x72125, 0x88e31, 0xa443b, 0xc51eb,
0xec8c1, 0x11bdbf, 0x154a3f, 0x198c4f, 0x1ea867, 0x24ca19, 0x2c25c1, 0x34fa1b,
0x3f928f, 0x4c4987, 0x5b8b6f, 0x6dda89
};
/**
* Initializes the base support module.
*/
BOOLEAN PhBaseInitialization(
VOID
)
{
PH_OBJECT_TYPE_PARAMETERS parameters;
// The following relies on the (technically undefined) value of XState being zero before Windows 7 SP1.
// NOTE: This is unused for now.
/*if (USER_SHARED_DATA->XState.EnabledFeatures & XSTATE_MASK_AVX)
PhpVectorLevel = PH_VECTOR_LEVEL_AVX;
else*/ if (USER_SHARED_DATA->ProcessorFeatures[PF_XMMI64_INSTRUCTIONS_AVAILABLE])
PhpVectorLevel = PH_VECTOR_LEVEL_SSE2;
PhStringType = PhCreateObjectType(L"String", 0, NULL);
PhBytesType = PhCreateObjectType(L"Bytes", 0, NULL);
parameters.FreeListSize = sizeof(PH_LIST);
parameters.FreeListCount = 128;
PhListType = PhCreateObjectTypeEx(L"List", PH_OBJECT_TYPE_USE_FREE_LIST, PhpListDeleteProcedure, &parameters);
PhPointerListType = PhCreateObjectType(L"PointerList", 0, PhpPointerListDeleteProcedure);
parameters.FreeListSize = sizeof(PH_HASHTABLE);
parameters.FreeListCount = 64;
PhHashtableType = PhCreateObjectTypeEx(L"Hashtable", PH_OBJECT_TYPE_USE_FREE_LIST, PhpHashtableDeleteProcedure, &parameters);
PhInitializeFreeList(&PhpBaseThreadContextFreeList, sizeof(PHP_BASE_THREAD_CONTEXT), 16);
#ifdef DEBUG
PhDbgThreadDbgTlsIndex = TlsAlloc();
InitializeListHead(&PhDbgThreadListHead);
#endif
return TRUE;
}
NTSTATUS PhpBaseThreadStart(
_In_ PVOID Parameter
)
{
NTSTATUS status;
HRESULT result;
PHP_BASE_THREAD_CONTEXT context;
#ifdef DEBUG
PHP_BASE_THREAD_DBG dbg;
#endif
context = *(PPHP_BASE_THREAD_CONTEXT)Parameter;
PhFreeToFreeList(&PhpBaseThreadContextFreeList, Parameter);
#ifdef DEBUG
dbg.ClientId = NtCurrentTeb()->ClientId;
dbg.StartAddress = context.StartAddress;
dbg.Parameter = context.Parameter;
dbg.CurrentAutoPool = NULL;
PhAcquireQueuedLockExclusive(&PhDbgThreadListLock);
InsertTailList(&PhDbgThreadListHead, &dbg.ListEntry);
PhReleaseQueuedLockExclusive(&PhDbgThreadListLock);
TlsSetValue(PhDbgThreadDbgTlsIndex, &dbg);
#endif
// Initialization code
result = CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
// Call the user-supplied function.
status = context.StartAddress(context.Parameter);
// De-initialization code
if (result == S_OK || result == S_FALSE)
CoUninitialize();
#ifdef DEBUG
PhAcquireQueuedLockExclusive(&PhDbgThreadListLock);
RemoveEntryList(&dbg.ListEntry);
PhReleaseQueuedLockExclusive(&PhDbgThreadListLock);
#endif
return status;
}
/**
* Creates a thread.
*
* \param StackSize The initial stack size of the thread.
* \param StartAddress The function to execute in the thread.
* \param Parameter A user-defined value to pass to the function.
*/
HANDLE PhCreateThread(
_In_opt_ SIZE_T StackSize,
_In_ PUSER_THREAD_START_ROUTINE StartAddress,
_In_opt_ PVOID Parameter
)
{
HANDLE threadHandle;
PPHP_BASE_THREAD_CONTEXT context;
context = PhAllocateFromFreeList(&PhpBaseThreadContextFreeList);
context->StartAddress = StartAddress;
context->Parameter = Parameter;
threadHandle = CreateThread(
NULL,
StackSize,
PhpBaseThreadStart,
context,
0,
NULL
);
if (threadHandle)
{
PHLIB_INC_STATISTIC(BaseThreadsCreated);
return threadHandle;
}
else
{
PHLIB_INC_STATISTIC(BaseThreadsCreateFailed);
PhFreeToFreeList(&PhpBaseThreadContextFreeList, context);
return NULL;
}
}
/**
* Gets the current system time (UTC).
*
* \remarks Use this function instead of NtQuerySystemTime() because no system calls are involved.
*/
VOID PhQuerySystemTime(
_Out_ PLARGE_INTEGER SystemTime
)
{
do
{
SystemTime->HighPart = USER_SHARED_DATA->SystemTime.High1Time;
SystemTime->LowPart = USER_SHARED_DATA->SystemTime.LowPart;
} while (SystemTime->HighPart != USER_SHARED_DATA->SystemTime.High2Time);
}
/**
* Gets the offset of the current time zone from UTC.
*/
VOID PhQueryTimeZoneBias(
_Out_ PLARGE_INTEGER TimeZoneBias
)
{
do
{
TimeZoneBias->HighPart = USER_SHARED_DATA->TimeZoneBias.High1Time;
TimeZoneBias->LowPart = USER_SHARED_DATA->TimeZoneBias.LowPart;
} while (TimeZoneBias->HighPart != USER_SHARED_DATA->TimeZoneBias.High2Time);
}
/**
* Converts system time to local time.
*
* \param SystemTime A UTC time value.
* \param LocalTime A variable which receives the local time value. This may be the same variable as
* \a SystemTime.
*
* \remarks Use this function instead of RtlSystemTimeToLocalTime() because no system calls are
* involved.
*/
VOID PhSystemTimeToLocalTime(
_In_ PLARGE_INTEGER SystemTime,
_Out_ PLARGE_INTEGER LocalTime
)
{
LARGE_INTEGER timeZoneBias;
PhQueryTimeZoneBias(&timeZoneBias);
LocalTime->QuadPart = SystemTime->QuadPart - timeZoneBias.QuadPart;
}
/**
* Converts local time to system time.
*
* \param LocalTime A local time value.
* \param SystemTime A variable which receives the UTC time value. This may be the same variable as
* \a LocalTime.
*
* \remarks Use this function instead of RtlLocalTimeToSystemTime() because no system calls are
* involved.
*/
VOID PhLocalTimeToSystemTime(
_In_ PLARGE_INTEGER LocalTime,
_Out_ PLARGE_INTEGER SystemTime
)
{
LARGE_INTEGER timeZoneBias;
PhQueryTimeZoneBias(&timeZoneBias);
SystemTime->QuadPart = LocalTime->QuadPart + timeZoneBias.QuadPart;
}
/**
* Allocates a block of memory.
*
* \param Size The number of bytes to allocate.
*
* \return A pointer to the allocated block of memory.
*
* \remarks If the function fails to allocate the block of memory, it raises an exception. The block
* is guaranteed to be aligned at MEMORY_ALLOCATION_ALIGNMENT bytes.
*/
_May_raise_
_Check_return_
_Ret_notnull_
_Post_writable_byte_size_(Size)
PVOID PhAllocate(
_In_ SIZE_T Size
)
{
return RtlAllocateHeap(PhHeapHandle, HEAP_GENERATE_EXCEPTIONS, Size);
}
/**
* Allocates a block of memory.
*
* \param Size The number of bytes to allocate.
*
* \return A pointer to the allocated block of memory, or NULL if the block could not be allocated.
*/
PVOID PhAllocateSafe(
_In_ SIZE_T Size
)
{
return RtlAllocateHeap(PhHeapHandle, 0, Size);
}
/**
* Allocates a block of memory.
*
* \param Size The number of bytes to allocate.
* \param Flags Flags controlling the allocation.
*
* \return A pointer to the allocated block of memory, or NULL if the block could not be allocated.
*/
PVOID PhAllocateExSafe(
_In_ SIZE_T Size,
_In_ ULONG Flags
)
{
return RtlAllocateHeap(PhHeapHandle, Flags, Size);
}
/**
* Frees a block of memory allocated with PhAllocate().
*
* \param Memory A pointer to a block of memory.
*/
VOID PhFree(
_Frees_ptr_opt_ PVOID Memory
)
{
RtlFreeHeap(PhHeapHandle, 0, Memory);
}
/**
* Re-allocates a block of memory originally allocated with PhAllocate().
*
* \param Memory A pointer to a block of memory.
* \param Size The new size of the memory block, in bytes.
*
* \return A pointer to the new block of memory. The existing contents of the memory block are
* copied to the new block.
*
* \remarks If the function fails to allocate the block of memory, it raises an exception.
*/
_May_raise_
_Post_writable_byte_size_(Size)
PVOID PhReAllocate(
_Frees_ptr_opt_ PVOID Memory,
_In_ SIZE_T Size
)
{
return RtlReAllocateHeap(PhHeapHandle, HEAP_GENERATE_EXCEPTIONS, Memory, Size);
}
/**
* Re-allocates a block of memory originally allocated with PhAllocate().
*
* \param Memory A pointer to a block of memory.
* \param Size The new size of the memory block, in bytes.
*
* \return A pointer to the new block of memory, or NULL if the block could not be allocated. The
* existing contents of the memory block are copied to the new block.
*/
PVOID PhReAllocateSafe(
_In_ PVOID Memory,
_In_ SIZE_T Size
)
{
return RtlReAllocateHeap(PhHeapHandle, 0, Memory, Size);
}
/**
* Allocates pages of memory.
*
* \param Size The number of bytes to allocate. The number of pages allocated will be large enough
* to contain \a Size bytes.
* \param NewSize The number of bytes actually allocated. This is \a Size rounded up to the next
* multiple of PAGE_SIZE.
*
* \return A pointer to the allocated block of memory, or NULL if the block could not be allocated.
*/
_Check_return_
_Ret_maybenull_
PVOID PhAllocatePage(
_In_ SIZE_T Size,
_Out_opt_ PSIZE_T NewSize
)
{
PVOID baseAddress;
baseAddress = NULL;
if (NT_SUCCESS(NtAllocateVirtualMemory(
NtCurrentProcess(),
&baseAddress,
0,
&Size,
MEM_COMMIT,
PAGE_READWRITE
)))
{
if (NewSize)
*NewSize = Size;
return baseAddress;
}
else
{
return NULL;
}
}
/**
* Frees pages of memory allocated with PhAllocatePage().
*
* \param Memory A pointer to a block of memory.
*/
VOID PhFreePage(
_Frees_ptr_opt_ PVOID Memory
)
{
SIZE_T size;
size = 0;
NtFreeVirtualMemory(
NtCurrentProcess(),
&Memory,
&size,
MEM_RELEASE
);
}
/**
* Determines the length of the specified string, in characters.
*
* \param String The string.
*/
SIZE_T PhCountStringZ(
_In_ PWSTR String
)
{
if (PhpVectorLevel >= PH_VECTOR_LEVEL_SSE2)
{
PWSTR p;
ULONG unaligned;
__m128i b;
__m128i z;
ULONG mask;
ULONG index;
p = (PWSTR)((ULONG_PTR)String & ~0xe); // String should be 2 byte aligned
unaligned = PtrToUlong(String) & 0xf;
z = _mm_setzero_si128();
if (unaligned != 0)
{
b = _mm_load_si128((__m128i *)p);
b = _mm_cmpeq_epi16(b, z);
mask = _mm_movemask_epi8(b) >> unaligned;
if (_BitScanForward(&index, mask))
return index / sizeof(WCHAR);
p += 16 / sizeof(WCHAR);
}
while (TRUE)
{
b = _mm_load_si128((__m128i *)p);
b = _mm_cmpeq_epi16(b, z);
mask = _mm_movemask_epi8(b);
if (_BitScanForward(&index, mask))
return (SIZE_T)(p - String) + index / sizeof(WCHAR);
p += 16 / sizeof(WCHAR);
}
}
else
{
return wcslen(String);
}
}
/**
* Allocates space for and copies a byte string.
*
* \param String The string to duplicate.
*
* \return The new string, which can be freed using PhFree().
*/
PSTR PhDuplicateBytesZ(
_In_ PSTR String
)
{
PSTR newString;
SIZE_T length;
length = strlen(String) + 1; // include the null terminator
newString = PhAllocate(length);
memcpy(newString, String, length);
return newString;
}
/**
* Allocates space for and copies a byte string.
*
* \param String The string to duplicate.
*
* \return The new string, which can be freed using PhFree(), or NULL if storage could not be
* allocated.
*/
PSTR PhDuplicateBytesZSafe(
_In_ PSTR String
)
{
PSTR newString;
SIZE_T length;
length = strlen(String) + 1; // include the null terminator
newString = PhAllocateSafe(length);
if (!newString)
return NULL;
memcpy(newString, String, length);
return newString;
}
/**
* Allocates space for and copies a 16-bit string.
*
* \param String The string to duplicate.
*
* \return The new string, which can be freed using PhFree().
*/
PWSTR PhDuplicateStringZ(
_In_ PWSTR String
)
{
PWSTR newString;
SIZE_T length;
length = (PhCountStringZ(String) + 1) * sizeof(WCHAR); // include the null terminator
newString = PhAllocate(length);
memcpy(newString, String, length);
return newString;
}
/**
* Copies a string with optional null termination and a maximum number of characters.
*
* \param InputBuffer A pointer to the input string.
* \param InputCount The maximum number of characters to copy, not including the null terminator.
* Specify -1 for no limit.
* \param OutputBuffer A pointer to the output buffer.
* \param OutputCount The number of characters in the output buffer, including the null terminator.
* \param ReturnCount A variable which receives the number of characters required to contain the
* input string, including the null terminator. If the function returns TRUE, this variable contains
* the number of characters written to the output buffer.
*
* \return TRUE if the input string was copied to the output string, otherwise FALSE.
*
* \remarks The function stops copying when it encounters the first null character in the input
* string. It will also stop copying when it reaches the character count specified in \a InputCount,
* if \a InputCount is not -1.
*/
BOOLEAN PhCopyBytesZ(
_In_ PSTR InputBuffer,
_In_ SIZE_T InputCount,
_Out_writes_opt_z_(OutputCount) PSTR OutputBuffer,
_In_ SIZE_T OutputCount,
_Out_opt_ PSIZE_T ReturnCount
)
{
SIZE_T i;
BOOLEAN copied;
// Determine the length of the input string.
if (InputCount != -1)
{
i = 0;
while (i < InputCount && InputBuffer[i])
i++;
}
else
{
i = strlen(InputBuffer);
}
// Copy the string if there is enough room.
if (OutputBuffer && OutputCount >= i + 1) // need one character for null terminator
{
memcpy(OutputBuffer, InputBuffer, i);
OutputBuffer[i] = 0;
copied = TRUE;
}
else
{
copied = FALSE;
}
if (ReturnCount)
*ReturnCount = i + 1;
return copied;
}
/**
* Copies a string with optional null termination and a maximum number of characters.
*
* \param InputBuffer A pointer to the input string.
* \param InputCount The maximum number of characters to copy, not including the null terminator.
* Specify -1 for no limit.
* \param OutputBuffer A pointer to the output buffer.
* \param OutputCount The number of characters in the output buffer, including the null terminator.
* \param ReturnCount A variable which receives the number of characters required to contain the
* input string, including the null terminator. If the function returns TRUE, this variable contains
* the number of characters written to the output buffer.
*
* \return TRUE if the input string was copied to the output string, otherwise FALSE.
*
* \remarks The function stops copying when it encounters the first null character in the input
* string. It will also stop copying when it reaches the character count specified in \a InputCount,
* if \a InputCount is not -1.
*/
BOOLEAN PhCopyStringZ(
_In_ PWSTR InputBuffer,
_In_ SIZE_T InputCount,
_Out_writes_opt_z_(OutputCount) PWSTR OutputBuffer,
_In_ SIZE_T OutputCount,
_Out_opt_ PSIZE_T ReturnCount
)
{
SIZE_T i;
BOOLEAN copied;
// Determine the length of the input string.
if (InputCount != -1)
{
i = 0;
while (i < InputCount && InputBuffer[i])
i++;
}
else
{
i = PhCountStringZ(InputBuffer);
}
// Copy the string if there is enough room.
if (OutputBuffer && OutputCount >= i + 1) // need one character for null terminator
{
memcpy(OutputBuffer, InputBuffer, i * sizeof(WCHAR));
OutputBuffer[i] = 0;
copied = TRUE;
}
else
{
copied = FALSE;
}
if (ReturnCount)
*ReturnCount = i + 1;
return copied;
}
/**
* Copies a string with optional null termination and a maximum number of characters.
*
* \param InputBuffer A pointer to the input string.
* \param InputCount The maximum number of characters to copy, not including the null terminator.
* Specify -1 for no limit.
* \param OutputBuffer A pointer to the output buffer.
* \param OutputCount The number of characters in the output buffer, including the null terminator.
* \param ReturnCount A variable which receives the number of characters required to contain the
* input string, including the null terminator. If the function returns TRUE, this variable contains
* the number of characters written to the output buffer.
*
* \return TRUE if the input string was copied to the output string, otherwise FALSE.
*
* \remarks The function stops copying when it encounters the first null character in the input
* string. It will also stop copying when it reaches the character count specified in \a InputCount,
* if \a InputCount is not -1.
*/
BOOLEAN PhCopyStringZFromBytes(
_In_ PSTR InputBuffer,
_In_ SIZE_T InputCount,
_Out_writes_opt_z_(OutputCount) PWSTR OutputBuffer,
_In_ SIZE_T OutputCount,
_Out_opt_ PSIZE_T ReturnCount
)
{
SIZE_T i;
BOOLEAN copied;
// Determine the length of the input string.
if (InputCount != -1)
{
i = 0;
while (i < InputCount && InputBuffer[i])
i++;
}
else
{
i = strlen(InputBuffer);
}
// Copy the string if there is enough room.
if (OutputBuffer && OutputCount >= i + 1) // need one character for null terminator
{
PhZeroExtendToUtf16Buffer(InputBuffer, i, OutputBuffer);
OutputBuffer[i] = 0;
copied = TRUE;
}
else
{
copied = FALSE;
}
if (ReturnCount)
*ReturnCount = i + 1;
return copied;
}
/**
* Copies a string with optional null termination and a maximum number of characters.
*
* \param InputBuffer A pointer to the input string.
* \param InputCount The maximum number of characters to copy, not including the null terminator.
* Specify -1 for no limit.
* \param OutputBuffer A pointer to the output buffer.
* \param OutputCount The number of characters in the output buffer, including the null terminator.
* \param ReturnCount A variable which receives the number of characters required to contain the
* input string, including the null terminator. If the function returns TRUE, this variable contains
* the number of characters written to the output buffer.
*
* \return TRUE if the input string was copied to the output string, otherwise FALSE.
*
* \remarks The function stops copying when it encounters the first null character in the input
* string. It will also stop copying when it reaches the character count specified in \a InputCount,
* if \a InputCount is not -1.
*/
BOOLEAN PhCopyStringZFromMultiByte(
_In_ PSTR InputBuffer,
_In_ SIZE_T InputCount,
_Out_writes_opt_z_(OutputCount) PWSTR OutputBuffer,
_In_ SIZE_T OutputCount,
_Out_opt_ PSIZE_T ReturnCount
)
{
NTSTATUS status;
SIZE_T i;
ULONG unicodeBytes;
BOOLEAN copied;
// Determine the length of the input string.
if (InputCount != -1)
{
i = 0;
while (i < InputCount && InputBuffer[i])
i++;
}
else
{
i = strlen(InputBuffer);
}
// Determine the length of the output string.
status = RtlMultiByteToUnicodeSize(
&unicodeBytes,
InputBuffer,
(ULONG)i
);
if (!NT_SUCCESS(status))
{
if (ReturnCount)
*ReturnCount = -1;
return FALSE;
}
// Convert the string to Unicode if there is enough room.
if (OutputBuffer && OutputCount >= unicodeBytes / sizeof(WCHAR) + 1)
{
status = RtlMultiByteToUnicodeN(
OutputBuffer,
unicodeBytes,
NULL,
InputBuffer,
(ULONG)i
);
if (NT_SUCCESS(status))
{
// RtlMultiByteToUnicodeN doesn't null terminate the string.
*(PWCHAR)((PCHAR)OutputBuffer + unicodeBytes) = 0;
copied = TRUE;
}
else
{
copied = FALSE;
}
}
else
{
copied = FALSE;
}
if (ReturnCount)
*ReturnCount = unicodeBytes / sizeof(WCHAR) + 1;
return copied;
}
FORCEINLINE LONG PhpCompareRightNatural(
_In_ PWSTR A,
_In_ PWSTR B
)
{
LONG bias = 0;
for (; ; A++, B++)
{
if (!PhIsDigitCharacter(*A) && !PhIsDigitCharacter(*B))
{
return bias;
}
else if (!PhIsDigitCharacter(*A))
{
return -1;
}
else if (!PhIsDigitCharacter(*B))
{
return 1;
}
else if (*A < *B)
{
if (bias == 0)
bias = -1;
}
else if (*A > *B)
{
if (bias == 0)
bias = 1;
}
else if (!*A && !*B)
{
return bias;
}
}
return 0;
}
FORCEINLINE LONG PhpCompareLeftNatural(
_In_ PWSTR A,
_In_ PWSTR B
)
{
for (; ; A++, B++)
{
if (!PhIsDigitCharacter(*A) && !PhIsDigitCharacter(*B))
{
return 0;
}
else if (!PhIsDigitCharacter(*A))
{
return -1;
}
else if (!PhIsDigitCharacter(*B))
{
return 1;
}
else if (*A < *B)
{
return -1;
}
else if (*A > *B)
{
return 1;
}
}
return 0;
}
FORCEINLINE LONG PhpCompareStringZNatural(
_In_ PWSTR A,
_In_ PWSTR B,
_In_ BOOLEAN IgnoreCase
)
{
/* strnatcmp.c -- Perform 'natural order' comparisons of strings in C.
Copyright (C) 2000, 2004 by Martin Pool <mbp sourcefrog net>
This software is provided 'as-is', without any express or implied
warranty. In no event will the authors be held liable for any damages
arising from the use of this software.
Permission is granted to anyone to use this software for any purpose,
including commercial applications, and to alter it and redistribute it
freely, subject to the following restrictions:
1. The origin of this software must not be misrepresented; you must not
claim that you wrote the original software. If you use this software
in a product, an acknowledgment in the product documentation would be
appreciated but is not required.
2. Altered source versions must be plainly marked as such, and must not be
misrepresented as being the original software.
3. This notice may not be removed or altered from any source distribution.
This code has been modified for Process Hacker.
*/
ULONG ai, bi;
WCHAR ca, cb;
LONG result;
BOOLEAN fractional;
ai = 0;
bi = 0;
while (TRUE)
{
ca = A[ai];
cb = B[bi];
/* Skip over leading spaces or zeros. */
while (ca == ' ')
ca = A[++ai];
while (cb == ' ')
cb = B[++bi];
/* Process run of digits. */
if (PhIsDigitCharacter(ca) && PhIsDigitCharacter(cb))
{
fractional = (ca == '0' || cb == '0');
if (fractional)
{
if ((result = PhpCompareLeftNatural(A + ai, B + bi)) != 0)
return result;
}
else
{
if ((result = PhpCompareRightNatural(A + ai, B + bi)) != 0)
return result;
}
}
if (!ca && !cb)
{
/* Strings are considered the same. */
return 0;
}
if (IgnoreCase)
{
ca = towupper(ca);
cb = towupper(cb);
}
if (ca < cb)
return -1;
else if (ca > cb)
return 1;
ai++;
bi++;
}
}
/**
* Compares two strings in natural sort order.
*
* \param A The first string.
* \param B The second string.
* \param IgnoreCase Whether to ignore character cases.
*/
LONG PhCompareStringZNatural(
_In_ PWSTR A,
_In_ PWSTR B,
_In_ BOOLEAN IgnoreCase
)
{
if (!IgnoreCase)
return PhpCompareStringZNatural(A, B, FALSE);
else
return PhpCompareStringZNatural(A, B, TRUE);
}
/**
* Compares two strings.
*
* \param String1 The first string.
* \param String2 The second string.
* \param IgnoreCase TRUE to perform a case-insensitive comparison, otherwise FALSE.
*/
LONG PhCompareStringRef(
_In_ PPH_STRINGREF String1,
_In_ PPH_STRINGREF String2,
_In_ BOOLEAN IgnoreCase
)
{
SIZE_T l1;
SIZE_T l2;
PWCHAR s1;
PWCHAR s2;
WCHAR c1;
WCHAR c2;
PWCHAR end;
// Note: this function assumes that the difference between the lengths of the two strings can
// fit inside a LONG.
l1 = String1->Length;
l2 = String2->Length;
assert(!(l1 & 1));
assert(!(l2 & 1));
s1 = String1->Buffer;
s2 = String2->Buffer;
end = (PWCHAR)((PCHAR)s1 + (l1 <= l2 ? l1 : l2));
if (!IgnoreCase)
{
while (s1 != end)
{
c1 = *s1;
c2 = *s2;
if (c1 != c2)
return (LONG)c1 - (LONG)c2;
s1++;
s2++;
}
}
else
{
while (s1 != end)
{
c1 = *s1;
c2 = *s2;
if (c1 != c2)
{
c1 = RtlUpcaseUnicodeChar(c1);
c2 = RtlUpcaseUnicodeChar(c2);
if (c1 != c2)
return (LONG)c1 - (LONG)c2;
}
s1++;
s2++;
}
}
return (LONG)(l1 - l2);
}
/**
* Determines if two strings are equal.
*
* \param String1 The first string.
* \param String2 The second string.
* \param IgnoreCase TRUE to perform a case-insensitive comparison, otherwise FALSE.
*/
BOOLEAN PhEqualStringRef(
_In_ PPH_STRINGREF String1,
_In_ PPH_STRINGREF String2,
_In_ BOOLEAN IgnoreCase
)
{
SIZE_T l1;
SIZE_T l2;
PWSTR s1;
PWSTR s2;
WCHAR c1;
WCHAR c2;
SIZE_T length;
l1 = String1->Length;
l2 = String2->Length;
assert(!(l1 & 1));
assert(!(l2 & 1));
if (l1 != l2)
return FALSE;
s1 = String1->Buffer;
s2 = String2->Buffer;
if (PhpVectorLevel >= PH_VECTOR_LEVEL_SSE2)
{
length = l1 / 16;
if (length != 0)
{
__m128i b1;
__m128i b2;
do
{
b1 = _mm_loadu_si128((__m128i *)s1);
b2 = _mm_loadu_si128((__m128i *)s2);
b1 = _mm_cmpeq_epi32(b1, b2);
if (_mm_movemask_epi8(b1) != 0xffff)
{
if (!IgnoreCase)
{
return FALSE;
}
else
{
// Compare character-by-character to ignore case.
l1 = length * 16 + (l1 & 15);
l1 /= sizeof(WCHAR);
goto CompareCharacters;
}
}
s1 += 16 / sizeof(WCHAR);
s2 += 16 / sizeof(WCHAR);
} while (--length != 0);
}
// Compare character-by-character because we have no more 16-byte blocks to compare.
l1 = (l1 & 15) / sizeof(WCHAR);
}
else
{
length = l1 / sizeof(ULONG_PTR);
if (length != 0)
{
do
{
if (*(PULONG_PTR)s1 != *(PULONG_PTR)s2)
{
if (!IgnoreCase)
{
return FALSE;
}
else
{
// Compare character-by-character to ignore case.
l1 = length * sizeof(ULONG_PTR) + (l1 & (sizeof(ULONG_PTR) - 1));
l1 /= sizeof(WCHAR);
goto CompareCharacters;
}
}
s1 += sizeof(ULONG_PTR) / sizeof(WCHAR);
s2 += sizeof(ULONG_PTR) / sizeof(WCHAR);
} while (--length != 0);
}
// Compare character-by-character because we have no more ULONG_PTR blocks to compare.
l1 = (l1 & (sizeof(ULONG_PTR) - 1)) / sizeof(WCHAR);
}
CompareCharacters:
if (l1 != 0)
{
if (!IgnoreCase)
{
do
{
c1 = *s1;
c2 = *s2;
if (c1 != c2)
return FALSE;
s1++;
s2++;
} while (--l1 != 0);
}
else
{
do
{
c1 = *s1;
c2 = *s2;
if (c1 != c2)
{
c1 = RtlUpcaseUnicodeChar(c1);
c2 = RtlUpcaseUnicodeChar(c2);
if (c1 != c2)
return FALSE;
}
s1++;
s2++;
} while (--l1 != 0);
}
}
return TRUE;
}
/**
* Locates a character in a string.
*
* \param String The string to search.
* \param Character The character to search for.
* \param IgnoreCase TRUE to perform a case-insensitive search, otherwise FALSE.
*
* \return The index, in characters, of the first occurrence of \a Character in \a String1. If
* \a Character was not found, -1 is returned.
*/
ULONG_PTR PhFindCharInStringRef(
_In_ PPH_STRINGREF String,
_In_ WCHAR Character,
_In_ BOOLEAN IgnoreCase
)
{
PWSTR buffer;
SIZE_T length;
buffer = String->Buffer;
length = String->Length / sizeof(WCHAR);
if (!IgnoreCase)
{
if (PhpVectorLevel >= PH_VECTOR_LEVEL_SSE2)
{
SIZE_T length16;
length16 = String->Length / 16;
length &= 7;
if (length16 != 0)
{
__m128i pattern;
__m128i block;
ULONG mask;
ULONG index;
pattern = _mm_set1_epi16(Character);
do
{
block = _mm_loadu_si128((__m128i *)buffer);
block = _mm_cmpeq_epi16(block, pattern);
mask = _mm_movemask_epi8(block);
if (_BitScanForward(&index, mask))
return (String->Length - length16 * 16) / sizeof(WCHAR) - length + index / 2;
buffer += 16 / sizeof(WCHAR);
} while (--length16 != 0);
}
}
if (length != 0)
{
do
{
if (*buffer == Character)
return String->Length / sizeof(WCHAR) - length;
buffer++;
} while (--length != 0);
}
}
else
{
if (length != 0)
{
WCHAR c;
c = RtlUpcaseUnicodeChar(Character);
do
{
if (RtlUpcaseUnicodeChar(*buffer) == c)
return String->Length / sizeof(WCHAR) - length;
buffer++;
} while (--length != 0);
}
}
return -1;
}
/**
* Locates a character in a string, searching backwards.
*
* \param String The string to search.
* \param Character The character to search for.
* \param IgnoreCase TRUE to perform a case-insensitive search, otherwise FALSE.
*
* \return The index, in characters, of the last occurrence of \a Character in \a String1. If
* \a Character was not found, -1 is returned.
*/
ULONG_PTR PhFindLastCharInStringRef(
_In_ PPH_STRINGREF String,
_In_ WCHAR Character,
_In_ BOOLEAN IgnoreCase
)
{
PWCHAR buffer;
SIZE_T length;
buffer = (PWCHAR)((PCHAR)String->Buffer + String->Length);
length = String->Length / sizeof(WCHAR);
if (!IgnoreCase)
{
if (PhpVectorLevel >= PH_VECTOR_LEVEL_SSE2)
{
SIZE_T length16;
length16 = String->Length / 16;
length &= 7;
if (length16 != 0)
{
__m128i pattern;
__m128i block;
ULONG mask;
ULONG index;
pattern = _mm_set1_epi16(Character);
buffer -= 16 / sizeof(WCHAR);
do
{
block = _mm_loadu_si128((__m128i *)buffer);
block = _mm_cmpeq_epi16(block, pattern);
mask = _mm_movemask_epi8(block);
if (_BitScanReverse(&index, mask))
return (length16 - 1) * 16 / sizeof(WCHAR) + length + index / 2;
buffer -= 16 / sizeof(WCHAR);
} while (--length16 != 0);
buffer += 16 / sizeof(WCHAR);
}
}
if (length != 0)
{
buffer--;
do
{
if (*buffer == Character)
return length - 1;
buffer--;
} while (--length != 0);
}
}
else
{
if (length != 0)
{
WCHAR c;
c = RtlUpcaseUnicodeChar(Character);
buffer--;
do
{
if (RtlUpcaseUnicodeChar(*buffer) == c)
return length - 1;
buffer--;
} while (--length != 0);
}
}
return -1;
}
/**
* Locates a string in a string.
*
* \param String The string to search.
* \param SubString The string to search for.
* \param IgnoreCase TRUE to perform a case-insensitive search, otherwise FALSE.
*
* \return The index, in characters, of the first occurrence of \a SubString in \a String. If
* \a SubString was not found, -1 is returned.
*/
ULONG_PTR PhFindStringInStringRef(
_In_ PPH_STRINGREF String,
_In_ PPH_STRINGREF SubString,
_In_ BOOLEAN IgnoreCase
)
{
SIZE_T length1;
SIZE_T length2;
PH_STRINGREF sr1;
PH_STRINGREF sr2;
WCHAR c;
SIZE_T i;
length1 = String->Length / sizeof(WCHAR);
length2 = SubString->Length / sizeof(WCHAR);
// Can't be a substring if it's bigger than the first string.
if (length2 > length1)
return -1;
// We always get a match if the substring is zero-length.
if (length2 == 0)
return 0;
sr1.Buffer = String->Buffer;
sr1.Length = SubString->Length - sizeof(WCHAR);
sr2.Buffer = SubString->Buffer;
sr2.Length = SubString->Length - sizeof(WCHAR);
if (!IgnoreCase)
{
c = *sr2.Buffer++;
for (i = length1 - length2 + 1; i != 0; i--)
{
if (*sr1.Buffer++ == c && PhEqualStringRef(&sr1, &sr2, FALSE))
{
goto FoundUString;
}
}
}
else
{
c = RtlUpcaseUnicodeChar(*sr2.Buffer++);
for (i = length1 - length2 + 1; i != 0; i--)
{
if (RtlUpcaseUnicodeChar(*sr1.Buffer++) == c && PhEqualStringRef(&sr1, &sr2, TRUE))
{
goto FoundUString;
}
}
}
return -1;
FoundUString:
return (ULONG_PTR)(sr1.Buffer - String->Buffer - 1);
}
/**
* Splits a string.
*
* \param Input The input string.
* \param Separator The character to split at.
* \param FirstPart A variable which receives the part of \a Input before the separator. This may be
* the same variable as \a Input. If the separator is not found in \a Input, this variable is set to
* \a Input.
* \param SecondPart A variable which recieves the part of \a Input after the separator. This may be
* the same variable as \a Input. If the separator is not found in \a Input, this variable is set to
* an empty string.
*
* \return TRUE if \a Separator was found in \a Input, otherwise FALSE.
*/
BOOLEAN PhSplitStringRefAtChar(
_In_ PPH_STRINGREF Input,
_In_ WCHAR Separator,
_Out_ PPH_STRINGREF FirstPart,
_Out_ PPH_STRINGREF SecondPart
)
{
PH_STRINGREF input;
ULONG_PTR index;
input = *Input; // get a copy of the input because FirstPart/SecondPart may alias Input
index = PhFindCharInStringRef(Input, Separator, FALSE);
if (index == -1)
{
// The separator was not found.
FirstPart->Buffer = Input->Buffer;
FirstPart->Length = Input->Length;
SecondPart->Buffer = NULL;
SecondPart->Length = 0;
return FALSE;
}
FirstPart->Buffer = input.Buffer;
FirstPart->Length = index * sizeof(WCHAR);
SecondPart->Buffer = (PWCHAR)((PCHAR)input.Buffer + index * sizeof(WCHAR) + sizeof(WCHAR));
SecondPart->Length = input.Length - index * sizeof(WCHAR) - sizeof(WCHAR);
return TRUE;
}
/**
* Splits a string at the last occurrence of a character.
*
* \param Input The input string.
* \param Separator The character to split at.
* \param FirstPart A variable which receives the part of \a Input before the separator. This may be
* the same variable as \a Input. If the separator is not found in \a Input, this variable is set to
* \a Input.
* \param SecondPart A variable which recieves the part of \a Input after the separator. This may be
* the same variable as \a Input. If the separator is not found in \a Input, this variable is set to
* an empty string.
*
* \return TRUE if \a Separator was found in \a Input, otherwise FALSE.
*/
BOOLEAN PhSplitStringRefAtLastChar(
_In_ PPH_STRINGREF Input,
_In_ WCHAR Separator,
_Out_ PPH_STRINGREF FirstPart,
_Out_ PPH_STRINGREF SecondPart
)
{
PH_STRINGREF input;
ULONG_PTR index;
input = *Input; // get a copy of the input because FirstPart/SecondPart may alias Input
index = PhFindLastCharInStringRef(Input, Separator, FALSE);
if (index == -1)
{
// The separator was not found.
FirstPart->Buffer = Input->Buffer;
FirstPart->Length = Input->Length;
SecondPart->Buffer = NULL;
SecondPart->Length = 0;
return FALSE;
}
FirstPart->Buffer = input.Buffer;
FirstPart->Length = index * sizeof(WCHAR);
SecondPart->Buffer = (PWCHAR)((PCHAR)input.Buffer + index * sizeof(WCHAR) + sizeof(WCHAR));
SecondPart->Length = input.Length - index * sizeof(WCHAR) - sizeof(WCHAR);
return TRUE;
}
/**
* Splits a string.
*
* \param Input The input string.
* \param Separator The string to split at.
* \param IgnoreCase TRUE to perform a case-insensitive search, otherwise FALSE.
* \param FirstPart A variable which receives the part of \a Input before the separator. This may be
* the same variable as \a Input. If the separator is not found in \a Input, this variable is set to
* \a Input.
* \param SecondPart A variable which recieves the part of \a Input after the separator. This may be
* the same variable as \a Input. If the separator is not found in \a Input, this variable is set to
* an empty string.
*
* \return TRUE if \a Separator was found in \a Input, otherwise FALSE.
*/
BOOLEAN PhSplitStringRefAtString(
_In_ PPH_STRINGREF Input,
_In_ PPH_STRINGREF Separator,
_In_ BOOLEAN IgnoreCase,
_Out_ PPH_STRINGREF FirstPart,
_Out_ PPH_STRINGREF SecondPart
)
{
PH_STRINGREF input;
ULONG_PTR index;
input = *Input; // get a copy of the input because FirstPart/SecondPart may alias Input
index = PhFindStringInStringRef(Input, Separator, IgnoreCase);
if (index == -1)
{
// The separator was not found.
FirstPart->Buffer = Input->Buffer;
FirstPart->Length = Input->Length;
SecondPart->Buffer = NULL;
SecondPart->Length = 0;
return FALSE;
}
FirstPart->Buffer = input.Buffer;
FirstPart->Length = index * sizeof(WCHAR);
SecondPart->Buffer = (PWCHAR)((PCHAR)input.Buffer + index * sizeof(WCHAR) + Separator->Length);
SecondPart->Length = input.Length - index * sizeof(WCHAR) - Separator->Length;
return TRUE;
}
/**
* Splits a string.
*
* \param Input The input string.
* \param Separator The character set, string or range to split at.
* \param Flags A combination of flags.
* \li \c PH_SPLIT_AT_CHAR_SET \a Separator specifies a character set. \a Input will be split at a
* character which is contained in \a Separator.
* \li \c PH_SPLIT_AT_STRING \a Separator specifies a string. \a Input will be split an occurrence
* of \a Separator.
* \li \c PH_SPLIT_AT_RANGE \a Separator specifies a range. The \a Buffer field contains a character
* index cast to \c PWSTR and the \a Length field contains the length of the range, in bytes.
* \li \c PH_SPLIT_CASE_INSENSITIVE Specifies a case-insensitive search.
* \li \c PH_SPLIT_COMPLEMENT_CHAR_SET If used with \c PH_SPLIT_AT_CHAR_SET, the separator is a
* character which is not contained in \a Separator.
* \li \c PH_SPLIT_START_AT_END If used with \c PH_SPLIT_AT_CHAR_SET, the search is performed
* starting from the end of the string.
* \li \c PH_SPLIT_CHAR_SET_IS_UPPERCASE If used with \c PH_SPLIT_CASE_INSENSITIVE, specifies that
* the character set in \a Separator contains only uppercase characters.
* \param FirstPart A variable which receives the part of \a Input before the separator. This may be
* the same variable as \a Input. If the separator is not found in \a Input, this variable is set to
* \a Input.
* \param SecondPart A variable which recieves the part of \a Input after the separator. This may be
* the same variable as \a Input. If the separator is not found in \a Input, this variable is set to
* an empty string.
* \param SeparatorPart A variable which receives the part of \a Input that is the separator. If the
* separator is not found in \a Input, this variable is set to an empty string.
*
* \return TRUE if a separator was found in \a Input, otherwise FALSE.
*/
BOOLEAN PhSplitStringRefEx(
_In_ PPH_STRINGREF Input,
_In_ PPH_STRINGREF Separator,
_In_ ULONG Flags,
_Out_ PPH_STRINGREF FirstPart,
_Out_ PPH_STRINGREF SecondPart,
_Out_opt_ PPH_STRINGREF SeparatorPart
)
{
PH_STRINGREF input;
SIZE_T separatorIndex;
SIZE_T separatorLength;
PWCHAR charSet;
SIZE_T charSetCount;
BOOLEAN charSetTable[256];
BOOLEAN charSetTableComplete;
SIZE_T i;
SIZE_T j;
USHORT c;
PWCHAR s;
LONG_PTR direction;
input = *Input; // Get a copy of the input because FirstPart/SecondPart/SeparatorPart may alias Input
if (Flags & PH_SPLIT_AT_RANGE)
{
separatorIndex = (SIZE_T)Separator->Buffer;
separatorLength = Separator->Length;
if (separatorIndex == -1)
goto SeparatorNotFound;
goto SeparatorFound;
}
else if (Flags & PH_SPLIT_AT_STRING)
{
if (Flags & PH_SPLIT_START_AT_END)
{
// not implemented
goto SeparatorNotFound;
}
separatorIndex = PhFindStringInStringRef(Input, Separator, !!(Flags & PH_SPLIT_CASE_INSENSITIVE));
if (separatorIndex == -1)
goto SeparatorNotFound;
separatorLength = Separator->Length;
goto SeparatorFound;
}
// Special case for character sets with only one character.
if (!(Flags & PH_SPLIT_COMPLEMENT_CHAR_SET) && Separator->Length == sizeof(WCHAR))
{
if (!(Flags & PH_SPLIT_START_AT_END))
separatorIndex = PhFindCharInStringRef(Input, Separator->Buffer[0], !!(Flags & PH_SPLIT_CASE_INSENSITIVE));
else
separatorIndex = PhFindLastCharInStringRef(Input, Separator->Buffer[0], !!(Flags & PH_SPLIT_CASE_INSENSITIVE));
if (separatorIndex == -1)
goto SeparatorNotFound;
separatorLength = sizeof(WCHAR);
goto SeparatorFound;
}
if (input.Length == 0)
goto SeparatorNotFound;
// Build the character set lookup table.
charSet = Separator->Buffer;
charSetCount = Separator->Length / sizeof(WCHAR);
memset(charSetTable, 0, sizeof(charSetTable));
charSetTableComplete = TRUE;
for (i = 0; i < charSetCount; i++)
{
c = charSet[i];
if (Flags & PH_SPLIT_CASE_INSENSITIVE)
c = RtlUpcaseUnicodeChar(c);
charSetTable[c & 0xff] = TRUE;
if (c >= 256)
charSetTableComplete = FALSE;
}
// Perform the search.
i = input.Length / sizeof(WCHAR);
separatorLength = sizeof(WCHAR);
if (!(Flags & PH_SPLIT_START_AT_END))
{
s = input.Buffer;
direction = 1;
}
else
{
s = (PWCHAR)((PCHAR)input.Buffer + input.Length - sizeof(WCHAR));
direction = -1;
}
do
{
c = *s;
if (Flags & PH_SPLIT_CASE_INSENSITIVE)
c = RtlUpcaseUnicodeChar(c);
if (c < 256 && charSetTableComplete)
{
if (!(Flags & PH_SPLIT_COMPLEMENT_CHAR_SET))
{
if (charSetTable[c])
goto CharFound;
}
else
{
if (!charSetTable[c])
goto CharFound;
}
}
else
{
if (!(Flags & PH_SPLIT_COMPLEMENT_CHAR_SET))
{
if (charSetTable[c & 0xff])
{
if (!(Flags & PH_SPLIT_CASE_INSENSITIVE) || (Flags & PH_SPLIT_CHAR_SET_IS_UPPERCASE))
{
for (j = 0; j < charSetCount; j++)
{
if (charSet[j] == c)
goto CharFound;
}
}
else
{
for (j = 0; j < charSetCount; j++)
{
if (RtlUpcaseUnicodeChar(charSet[j]) == c)
goto CharFound;
}
}
}
}
else
{
if (charSetTable[c & 0xff])
{
if (!(Flags & PH_SPLIT_CASE_INSENSITIVE) || (Flags & PH_SPLIT_CHAR_SET_IS_UPPERCASE))
{
for (j = 0; j < charSetCount; j++)
{
if (charSet[j] == c)
break;
}
}
else
{
for (j = 0; j < charSetCount; j++)
{
if (RtlUpcaseUnicodeChar(charSet[j]) == c)
break;
}
}
if (j == charSetCount)
goto CharFound;
}
else
{
goto CharFound;
}
}
}
s += direction;
} while (--i != 0);
goto SeparatorNotFound;
CharFound:
separatorIndex = s - input.Buffer;
SeparatorFound:
FirstPart->Buffer = input.Buffer;
FirstPart->Length = separatorIndex * sizeof(WCHAR);
SecondPart->Buffer = (PWCHAR)((PCHAR)input.Buffer + separatorIndex * sizeof(WCHAR) + separatorLength);
SecondPart->Length = input.Length - separatorIndex * sizeof(WCHAR) - separatorLength;
if (SeparatorPart)
{
SeparatorPart->Buffer = input.Buffer + separatorIndex;
SeparatorPart->Length = separatorLength;
}
return TRUE;
SeparatorNotFound:
FirstPart->Buffer = input.Buffer;
FirstPart->Length = input.Length;
SecondPart->Buffer = NULL;
SecondPart->Length = 0;
if (SeparatorPart)
{
SeparatorPart->Buffer = NULL;
SeparatorPart->Length = 0;
}
return FALSE;
}
VOID PhTrimStringRef(
_Inout_ PPH_STRINGREF String,
_In_ PPH_STRINGREF CharSet,
_In_ ULONG Flags
)
{
PWCHAR charSet;
SIZE_T charSetCount;
BOOLEAN charSetTable[256];
BOOLEAN charSetTableComplete;
SIZE_T i;
SIZE_T j;
USHORT c;
SIZE_T trimCount;
SIZE_T count;
PWCHAR s;
if (String->Length == 0 || CharSet->Length == 0)
return;
if (CharSet->Length == sizeof(WCHAR))
{
c = CharSet->Buffer[0];
if (!(Flags & PH_TRIM_END_ONLY))
{
trimCount = 0;
count = String->Length / sizeof(WCHAR);
s = String->Buffer;
while (count-- != 0)
{
if (*s++ != c)
break;
trimCount++;
}
PhSkipStringRef(String, trimCount * sizeof(WCHAR));
}
if (!(Flags & PH_TRIM_START_ONLY))
{
trimCount = 0;
count = String->Length / sizeof(WCHAR);
s = (PWCHAR)((PCHAR)String->Buffer + String->Length - sizeof(WCHAR));
while (count-- != 0)
{
if (*s-- != c)
break;
trimCount++;
}
String->Length -= trimCount * sizeof(WCHAR);
}
return;
}
// Build the character set lookup table.
charSet = CharSet->Buffer;
charSetCount = CharSet->Length / sizeof(WCHAR);
memset(charSetTable, 0, sizeof(charSetTable));
charSetTableComplete = TRUE;
for (i = 0; i < charSetCount; i++)
{
c = charSet[i];
charSetTable[c & 0xff] = TRUE;
if (c >= 256)
charSetTableComplete = FALSE;
}
// Trim the string.
if (!(Flags & PH_TRIM_END_ONLY))
{
trimCount = 0;
count = String->Length / sizeof(WCHAR);
s = String->Buffer;
while (count-- != 0)
{
c = *s++;
if (!charSetTable[c & 0xff])
break;
if (!charSetTableComplete)
{
for (j = 0; j < charSetCount; j++)
{
if (charSet[j] == c)
goto CharFound;
}
break;
}
CharFound:
trimCount++;
}
PhSkipStringRef(String, trimCount * sizeof(WCHAR));
}
if (!(Flags & PH_TRIM_START_ONLY))
{
trimCount = 0;
count = String->Length / sizeof(WCHAR);
s = (PWCHAR)((PCHAR)String->Buffer + String->Length - sizeof(WCHAR));
while (count-- != 0)
{
c = *s--;
if (!charSetTable[c & 0xff])
break;
if (!charSetTableComplete)
{
for (j = 0; j < charSetCount; j++)
{
if (charSet[j] == c)
goto CharFound2;
}
break;
}
CharFound2:
trimCount++;
}
String->Length -= trimCount * sizeof(WCHAR);
}
}
/**
* Creates a string object from an existing null-terminated string.
*
* \param Buffer A null-terminated Unicode string.
*/
PPH_STRING PhCreateString(
_In_ PWSTR Buffer
)
{
return PhCreateStringEx(Buffer, wcslen(Buffer) * sizeof(WCHAR));
}
/**
* Creates a string object using a specified length.
*
* \param Buffer A null-terminated Unicode string.
* \param Length The length, in bytes, of the string.
*/
PPH_STRING PhCreateStringEx(
_In_opt_ PWCHAR Buffer,
_In_ SIZE_T Length
)
{
PPH_STRING string;
string = PhCreateObject(
FIELD_OFFSET(PH_STRING, Data) + Length + sizeof(WCHAR), // Null terminator
PhStringType
);
assert(!(Length & 1));
string->Length = Length;
string->Buffer = string->Data;
*(PWCHAR)((PCHAR)string->Buffer + Length) = 0;
if (Buffer)
{
memcpy(string->Buffer, Buffer, Length);
}
return string;
}
/**
* Obtains a reference to a zero-length string.
*/
PPH_STRING PhReferenceEmptyString(
VOID
)
{
PPH_STRING string;
PPH_STRING newString;
string = PhSharedEmptyString;
if (!string)
{
newString = PhCreateStringEx(NULL, 0);
string = _InterlockedCompareExchangePointer(
&PhSharedEmptyString,
newString,
NULL
);
if (!string)
{
string = newString; // success
}
else
{
PhDereferenceObject(newString);
}
}
return PhReferenceObject(string);
}
/**
* Concatenates multiple strings.
*
* \param Count The number of strings to concatenate.
*/
PPH_STRING PhConcatStrings(
_In_ ULONG Count,
...
)
{
va_list argptr;
va_start(argptr, Count);
return PhConcatStrings_V(Count, argptr);
}
/**
* Concatenates multiple strings.
*
* \param Count The number of strings to concatenate.
* \param ArgPtr A pointer to an array of strings.
*/
PPH_STRING PhConcatStrings_V(
_In_ ULONG Count,
_In_ va_list ArgPtr
)
{
va_list argptr;
ULONG i;
SIZE_T totalLength = 0;
SIZE_T stringLength;
SIZE_T cachedLengths[PH_CONCAT_STRINGS_LENGTH_CACHE_SIZE];
PWSTR arg;
PPH_STRING string;
// Compute the total length, in bytes, of the strings.
argptr = ArgPtr;
for (i = 0; i < Count; i++)
{
arg = va_arg(argptr, PWSTR);
stringLength = PhCountStringZ(arg) * sizeof(WCHAR);
totalLength += stringLength;
if (i < PH_CONCAT_STRINGS_LENGTH_CACHE_SIZE)
cachedLengths[i] = stringLength;
}
// Create the string.
string = PhCreateStringEx(NULL, totalLength);
totalLength = 0;
// Append the strings one by one.
argptr = ArgPtr;
for (i = 0; i < Count; i++)
{
arg = va_arg(argptr, PWSTR);
if (i < PH_CONCAT_STRINGS_LENGTH_CACHE_SIZE)
stringLength = cachedLengths[i];
else
stringLength = PhCountStringZ(arg) * sizeof(WCHAR);
memcpy(
(PCHAR)string->Buffer + totalLength,
arg,
stringLength
);
totalLength += stringLength;
}
return string;
}
/**
* Concatenates two strings.
*
* \param String1 The first string.
* \param String2 The second string.
*/
PPH_STRING PhConcatStrings2(
_In_ PWSTR String1,
_In_ PWSTR String2
)
{
PPH_STRING string;
SIZE_T length1;
SIZE_T length2;
length1 = PhCountStringZ(String1) * sizeof(WCHAR);
length2 = PhCountStringZ(String2) * sizeof(WCHAR);
string = PhCreateStringEx(NULL, length1 + length2);
memcpy(
string->Buffer,
String1,
length1
);
memcpy(
(PCHAR)string->Buffer + length1,
String2,
length2
);
return string;
}
/**
* Concatenates two strings.
*
* \param String1 The first string.
* \param String2 The second string.
*/
PPH_STRING PhConcatStringRef2(
_In_ PPH_STRINGREF String1,
_In_ PPH_STRINGREF String2
)
{
PPH_STRING string;
assert(!(String1->Length & 1));
assert(!(String2->Length & 1));
string = PhCreateStringEx(NULL, String1->Length + String2->Length);
memcpy(string->Buffer, String1->Buffer, String1->Length);
memcpy((PCHAR)string->Buffer + String1->Length, String2->Buffer, String2->Length);
return string;
}
/**
* Concatenates three strings.
*
* \param String1 The first string.
* \param String2 The second string.
* \param String3 The third string.
*/
PPH_STRING PhConcatStringRef3(
_In_ PPH_STRINGREF String1,
_In_ PPH_STRINGREF String2,
_In_ PPH_STRINGREF String3
)
{
PPH_STRING string;
PCHAR buffer;
assert(!(String1->Length & 1));
assert(!(String2->Length & 1));
assert(!(String3->Length & 1));
string = PhCreateStringEx(NULL, String1->Length + String2->Length + String3->Length);
buffer = (PCHAR)string->Buffer;
memcpy(buffer, String1->Buffer, String1->Length);
buffer += String1->Length;
memcpy(buffer, String2->Buffer, String2->Length);
buffer += String2->Length;
memcpy(buffer, String3->Buffer, String3->Length);
return string;
}
/**
* Creates a string using format specifiers.
*
* \param Format The format-control string.
*/
PPH_STRING PhFormatString(
_In_ _Printf_format_string_ PWSTR Format,
...
)
{
va_list argptr;
va_start(argptr, Format);
return PhFormatString_V(Format, argptr);
}
/**
* Creates a string using format specifiers.
*
* \param Format The format-control string.
* \param ArgPtr A pointer to the list of arguments.
*/
PPH_STRING PhFormatString_V(
_In_ _Printf_format_string_ PWSTR Format,
_In_ va_list ArgPtr
)
{
PPH_STRING string;
int length;
length = _vscwprintf(Format, ArgPtr);
if (length == -1)
return NULL;
string = PhCreateStringEx(NULL, length * sizeof(WCHAR));
_vsnwprintf(string->Buffer, length, Format, ArgPtr);
return string;
}
/**
* Creates a bytes object from an existing null-terminated string of bytes.
*
* \param Buffer A null-terminated byte string.
*/
PPH_BYTES PhCreateBytes(
_In_ PSTR Buffer
)
{
return PhCreateBytesEx(Buffer, strlen(Buffer) * sizeof(CHAR));
}
/**
* Creates a bytes object.
*
* \param Buffer An array of bytes.
* \param Length The length of \a Buffer, in bytes.
*/
PPH_BYTES PhCreateBytesEx(
_In_opt_ PCHAR Buffer,
_In_ SIZE_T Length
)
{
PPH_BYTES bytes;
bytes = PhCreateObject(
FIELD_OFFSET(PH_BYTES, Data) + Length + sizeof(CHAR), // Null terminator for compatibility
PhBytesType
);
bytes->Length = Length;
bytes->Buffer = bytes->Data;
bytes->Buffer[Length] = 0;
if (Buffer)
{
memcpy(bytes->Buffer, Buffer, Length);
}
return bytes;
}
BOOLEAN PhWriteUnicodeDecoder(
_Inout_ PPH_UNICODE_DECODER Decoder,
_In_ ULONG CodeUnit
)
{
switch (Decoder->Encoding)
{
case PH_UNICODE_UTF8:
if (Decoder->InputCount >= 4)
return FALSE;
Decoder->u.Utf8.Input[Decoder->InputCount] = (UCHAR)CodeUnit;
Decoder->InputCount++;
return TRUE;
case PH_UNICODE_UTF16:
if (Decoder->InputCount >= 2)
return FALSE;
Decoder->u.Utf16.Input[Decoder->InputCount] = (USHORT)CodeUnit;
Decoder->InputCount++;
return TRUE;
case PH_UNICODE_UTF32:
if (Decoder->InputCount >= 1)
return FALSE;
Decoder->u.Utf32.Input = CodeUnit;
Decoder->InputCount = 1;
return TRUE;
default:
PhRaiseStatus(STATUS_UNSUCCESSFUL);
}
}
BOOLEAN PhpReadUnicodeDecoder(
_Inout_ PPH_UNICODE_DECODER Decoder,
_Out_ PULONG CodeUnit
)
{
switch (Decoder->Encoding)
{
case PH_UNICODE_UTF8:
if (Decoder->InputCount == 0)
return FALSE;
*CodeUnit = Decoder->u.Utf8.Input[0];
Decoder->u.Utf8.Input[0] = Decoder->u.Utf8.Input[1];
Decoder->u.Utf8.Input[1] = Decoder->u.Utf8.Input[2];
Decoder->u.Utf8.Input[2] = Decoder->u.Utf8.Input[3];
Decoder->InputCount--;
return TRUE;
case PH_UNICODE_UTF16:
if (Decoder->InputCount == 0)
return FALSE;
*CodeUnit = Decoder->u.Utf16.Input[0];
Decoder->u.Utf16.Input[0] = Decoder->u.Utf16.Input[1];
Decoder->InputCount--;
return TRUE;
case PH_UNICODE_UTF32:
if (Decoder->InputCount == 0)
return FALSE;
*CodeUnit = Decoder->u.Utf32.Input;
Decoder->InputCount--;
return TRUE;
default:
PhRaiseStatus(STATUS_UNSUCCESSFUL);
}
}
VOID PhpUnreadUnicodeDecoder(
_Inout_ PPH_UNICODE_DECODER Decoder,
_In_ ULONG CodeUnit
)
{
switch (Decoder->Encoding)
{
case PH_UNICODE_UTF8:
if (Decoder->InputCount >= 4)
PhRaiseStatus(STATUS_UNSUCCESSFUL);
Decoder->u.Utf8.Input[3] = Decoder->u.Utf8.Input[2];
Decoder->u.Utf8.Input[2] = Decoder->u.Utf8.Input[1];
Decoder->u.Utf8.Input[1] = Decoder->u.Utf8.Input[0];
Decoder->u.Utf8.Input[0] = (UCHAR)CodeUnit;
Decoder->InputCount++;
break;
case PH_UNICODE_UTF16:
if (Decoder->InputCount >= 2)
PhRaiseStatus(STATUS_UNSUCCESSFUL);
Decoder->u.Utf16.Input[1] = Decoder->u.Utf16.Input[0];
Decoder->u.Utf16.Input[0] = (USHORT)CodeUnit;
Decoder->InputCount++;
break;
case PH_UNICODE_UTF32:
if (Decoder->InputCount >= 1)
PhRaiseStatus(STATUS_UNSUCCESSFUL);
Decoder->u.Utf32.Input = CodeUnit;
Decoder->InputCount = 1;
break;
default:
PhRaiseStatus(STATUS_UNSUCCESSFUL);
}
}
BOOLEAN PhpDecodeUtf8Error(
_Inout_ PPH_UNICODE_DECODER Decoder,
_Out_ PULONG CodePoint,
_In_ ULONG Which
)
{
if (Which >= 4)
PhpUnreadUnicodeDecoder(Decoder, Decoder->u.Utf8.CodeUnit4);
if (Which >= 3)
PhpUnreadUnicodeDecoder(Decoder, Decoder->u.Utf8.CodeUnit3);
if (Which >= 2)
PhpUnreadUnicodeDecoder(Decoder, Decoder->u.Utf8.CodeUnit2);
*CodePoint = (ULONG)Decoder->u.Utf8.CodeUnit1 + 0xdc00;
Decoder->State = 0;
return TRUE;
}
BOOLEAN PhDecodeUnicodeDecoder(
_Inout_ PPH_UNICODE_DECODER Decoder,
_Out_ PULONG CodePoint
)
{
ULONG codeUnit;
while (TRUE)
{
switch (Decoder->Encoding)
{
case PH_UNICODE_UTF8:
if (!PhpReadUnicodeDecoder(Decoder, &codeUnit))
return FALSE;
switch (Decoder->State)
{
case 0:
Decoder->u.Utf8.CodeUnit1 = (UCHAR)codeUnit;
if (codeUnit < 0x80)
{
*CodePoint = codeUnit;
return TRUE;
}
else if (codeUnit < 0xc2)
{
return PhpDecodeUtf8Error(Decoder, CodePoint, 1);
}
else if (codeUnit < 0xe0)
{
Decoder->State = 1; // 2 byte sequence
continue;
}
else if (codeUnit < 0xf0)
{
Decoder->State = 2; // 3 byte sequence
continue;
}
else if (codeUnit < 0xf5)
{
Decoder->State = 3; // 4 byte sequence
continue;
}
else
{
return PhpDecodeUtf8Error(Decoder, CodePoint, 1);
}
break;
case 1: // 2 byte sequence
Decoder->u.Utf8.CodeUnit2 = (UCHAR)codeUnit;
if ((codeUnit & 0xc0) == 0x80)
{
*CodePoint = ((ULONG)Decoder->u.Utf8.CodeUnit1 << 6) + codeUnit - 0x3080;
Decoder->State = 0;
return TRUE;
}
else
{
return PhpDecodeUtf8Error(Decoder, CodePoint, 2);
}
break;
case 2: // 3 byte sequence (1)
Decoder->u.Utf8.CodeUnit2 = (UCHAR)codeUnit;
if (((codeUnit & 0xc0) == 0x80) &&
(Decoder->u.Utf8.CodeUnit1 != 0xe0 || codeUnit >= 0xa0))
{
Decoder->State = 4; // 3 byte sequence (2)
continue;
}
else
{
return PhpDecodeUtf8Error(Decoder, CodePoint, 2);
}
break;
case 3: // 4 byte sequence (1)
Decoder->u.Utf8.CodeUnit2 = (UCHAR)codeUnit;
if (((codeUnit & 0xc0) == 0x80) &&
(Decoder->u.Utf8.CodeUnit1 != 0xf0 || codeUnit >= 0x90) &&
(Decoder->u.Utf8.CodeUnit1 != 0xf4 || codeUnit < 0x90))
{
Decoder->State = 5; // 4 byte sequence (2)
continue;
}
else
{
return PhpDecodeUtf8Error(Decoder, CodePoint, 2);
}
break;
case 4: // 3 byte sequence (2)
Decoder->u.Utf8.CodeUnit3 = (UCHAR)codeUnit;
if ((codeUnit & 0xc0) == 0x80)
{
*CodePoint =
((ULONG)Decoder->u.Utf8.CodeUnit1 << 12) +
((ULONG)Decoder->u.Utf8.CodeUnit2 << 6) +
codeUnit - 0xe2080;
Decoder->State = 0;
return TRUE;
}
else
{
return PhpDecodeUtf8Error(Decoder, CodePoint, 3);
}
break;
case 5: // 4 byte sequence (2)
Decoder->u.Utf8.CodeUnit3 = (UCHAR)codeUnit;
if ((codeUnit & 0xc0) == 0x80)
{
Decoder->State = 6; // 4 byte sequence (3)
continue;
}
else
{
return PhpDecodeUtf8Error(Decoder, CodePoint, 3);
}
break;
case 6: // 4 byte sequence (3)
Decoder->u.Utf8.CodeUnit4 = (UCHAR)codeUnit;
if ((codeUnit & 0xc0) == 0x80)
{
*CodePoint =
((ULONG)Decoder->u.Utf8.CodeUnit1 << 18) +
((ULONG)Decoder->u.Utf8.CodeUnit2 << 12) +
((ULONG)Decoder->u.Utf8.CodeUnit3 << 6) +
codeUnit - 0x3c82080;
Decoder->State = 0;
return TRUE;
}
else
{
return PhpDecodeUtf8Error(Decoder, CodePoint, 4);
}
break;
}
return FALSE;
case PH_UNICODE_UTF16:
if (!PhpReadUnicodeDecoder(Decoder, &codeUnit))
return FALSE;
switch (Decoder->State)
{
case 0:
if (PH_UNICODE_UTF16_IS_HIGH_SURROGATE(codeUnit))
{
Decoder->u.Utf16.CodeUnit = (USHORT)codeUnit;
Decoder->State = 1;
continue;
}
else
{
*CodePoint = codeUnit;
return TRUE;
}
break;
case 1:
if (PH_UNICODE_UTF16_IS_LOW_SURROGATE(codeUnit))
{
*CodePoint = PH_UNICODE_UTF16_TO_CODE_POINT(Decoder->u.Utf16.CodeUnit, codeUnit);
Decoder->State = 0;
return TRUE;
}
else
{
*CodePoint = Decoder->u.Utf16.CodeUnit;
PhpUnreadUnicodeDecoder(Decoder, codeUnit);
Decoder->State = 0;
return TRUE;
}
break;
}
return FALSE;
case PH_UNICODE_UTF32:
if (PhpReadUnicodeDecoder(Decoder, CodePoint))
return TRUE;
return FALSE;
default:
return FALSE;
}
}
}
BOOLEAN PhEncodeUnicode(
_In_ UCHAR Encoding,
_In_ ULONG CodePoint,
_Out_opt_ PVOID CodeUnits,
_Out_ PULONG NumberOfCodeUnits
)
{
switch (Encoding)
{
case PH_UNICODE_UTF8:
{
PUCHAR codeUnits = CodeUnits;
if (CodePoint < 0x80)
{
*NumberOfCodeUnits = 1;
if (codeUnits)
codeUnits[0] = (UCHAR)CodePoint;
}
else if (CodePoint <= 0x7ff)
{
*NumberOfCodeUnits = 2;
if (codeUnits)
{
codeUnits[0] = (UCHAR)(CodePoint >> 6) + 0xc0;
codeUnits[1] = (UCHAR)(CodePoint & 0x3f) + 0x80;
}
}
else if (CodePoint <= 0xffff)
{
*NumberOfCodeUnits = 3;
if (codeUnits)
{
codeUnits[0] = (UCHAR)(CodePoint >> 12) + 0xe0;
codeUnits[1] = (UCHAR)((CodePoint >> 6) & 0x3f) + 0x80;
codeUnits[2] = (UCHAR)(CodePoint & 0x3f) + 0x80;
}
}
else if (CodePoint <= PH_UNICODE_MAX_CODE_POINT)
{
*NumberOfCodeUnits = 4;
if (codeUnits)
{
codeUnits[0] = (UCHAR)(CodePoint >> 18) + 0xf0;
codeUnits[1] = (UCHAR)((CodePoint >> 12) & 0x3f) + 0x80;
codeUnits[2] = (UCHAR)((CodePoint >> 6) & 0x3f) + 0x80;
codeUnits[3] = (UCHAR)(CodePoint & 0x3f) + 0x80;
}
}
else
{
return FALSE;
}
}
return TRUE;
case PH_UNICODE_UTF16:
{
PUSHORT codeUnits = CodeUnits;
if (CodePoint < 0x10000)
{
*NumberOfCodeUnits = 1;
if (codeUnits)
codeUnits[0] = (USHORT)CodePoint;
}
else if (CodePoint <= PH_UNICODE_MAX_CODE_POINT)
{
*NumberOfCodeUnits = 2;
if (codeUnits)
{
codeUnits[0] = PH_UNICODE_UTF16_TO_HIGH_SURROGATE(CodePoint);
codeUnits[1] = PH_UNICODE_UTF16_TO_LOW_SURROGATE(CodePoint);
}
}
else
{
return FALSE;
}
}
return TRUE;
case PH_UNICODE_UTF32:
*NumberOfCodeUnits = 1;
if (CodeUnits)
*(PULONG)CodeUnits = CodePoint;
return TRUE;
default:
return FALSE;
}
}
/**
* Converts an ASCII string to a UTF-16 string by zero-extending each byte.
*
* \param Input The original ASCII string.
* \param InputLength The length of \a Input.
* \param Output A buffer which will contain the converted string.
*/
VOID PhZeroExtendToUtf16Buffer(
_In_reads_bytes_(InputLength) PCH Input,
_In_ SIZE_T InputLength,
_Out_writes_bytes_(InputLength * sizeof(WCHAR)) PWCH Output
)
{
SIZE_T inputLength;
inputLength = InputLength & -4;
if (inputLength)
{
do
{
Output[0] = C_1uTo2(Input[0]);
Output[1] = C_1uTo2(Input[1]);
Output[2] = C_1uTo2(Input[2]);
Output[3] = C_1uTo2(Input[3]);
Input += 4;
Output += 4;
inputLength -= 4;
} while (inputLength != 0);
}
switch (InputLength & 3)
{
case 3:
*Output++ = C_1uTo2(*Input++);
case 2:
*Output++ = C_1uTo2(*Input++);
case 1:
*Output++ = C_1uTo2(*Input++);
}
}
PPH_STRING PhZeroExtendToUtf16Ex(
_In_reads_bytes_(InputLength) PCH Input,
_In_ SIZE_T InputLength
)
{
PPH_STRING string;
string = PhCreateStringEx(NULL, InputLength * sizeof(WCHAR));
PhZeroExtendToUtf16Buffer(Input, InputLength, string->Buffer);
return string;
}
PPH_BYTES PhConvertUtf16ToAsciiEx(
_In_ PWCH Buffer,
_In_ SIZE_T Length,
_In_opt_ CHAR Replacement
)
{
PPH_BYTES bytes;
PH_UNICODE_DECODER decoder;
PWCH in;
SIZE_T inRemaining;
PCH out;
SIZE_T outLength;
ULONG codePoint;
bytes = PhCreateBytesEx(NULL, Length / sizeof(WCHAR));
PhInitializeUnicodeDecoder(&decoder, PH_UNICODE_UTF16);
in = Buffer;
inRemaining = Length / sizeof(WCHAR);
out = bytes->Buffer;
outLength = 0;
while (inRemaining != 0)
{
PhWriteUnicodeDecoder(&decoder, (USHORT)*in);
in++;
inRemaining--;
while (PhDecodeUnicodeDecoder(&decoder, &codePoint))
{
if (codePoint < 0x80)
{
*out++ = (CHAR)codePoint;
outLength++;
}
else if (Replacement)
{
*out++ = Replacement;
outLength++;
}
}
}
bytes->Length = outLength;
bytes->Buffer[outLength] = 0;
return bytes;
}
/**
* Creates a string object from an existing null-terminated multi-byte string.
*
* \param Buffer A null-terminated multi-byte string.
*/
PPH_STRING PhConvertMultiByteToUtf16(
_In_ PSTR Buffer
)
{
return PhConvertMultiByteToUtf16Ex(
Buffer,
strlen(Buffer)
);
}
/**
* Creates a string object from an existing null-terminated multi-byte string.
*
* \param Buffer A null-terminated multi-byte string.
* \param Length The number of bytes to use.
*/
PPH_STRING PhConvertMultiByteToUtf16Ex(
_In_ PCHAR Buffer,
_In_ SIZE_T Length
)
{
NTSTATUS status;
PPH_STRING string;
ULONG unicodeBytes;
status = RtlMultiByteToUnicodeSize(
&unicodeBytes,
Buffer,
(ULONG)Length
);
if (!NT_SUCCESS(status))
return NULL;
string = PhCreateStringEx(NULL, unicodeBytes);
status = RtlMultiByteToUnicodeN(
string->Buffer,
(ULONG)string->Length,
NULL,
Buffer,
(ULONG)Length
);
if (!NT_SUCCESS(status))
{
PhDereferenceObject(string);
return NULL;
}
return string;
}
/**
* Creates a multi-byte string from an existing null-terminated UTF-16 string.
*
* \param Buffer A null-terminated UTF-16 string.
*/
PPH_BYTES PhConvertUtf16ToMultiByte(
_In_ PWSTR Buffer
)
{
return PhConvertUtf16ToMultiByteEx(
Buffer,
PhCountStringZ(Buffer) * sizeof(WCHAR)
);
}
/**
* Creates a multi-byte string from an existing null-terminated UTF-16 string.
*
* \param Buffer A null-terminated UTF-16 string.
* \param Length The number of bytes to use.
*/
PPH_BYTES PhConvertUtf16ToMultiByteEx(
_In_ PWCHAR Buffer,
_In_ SIZE_T Length
)
{
NTSTATUS status;
PPH_BYTES bytes;
ULONG multiByteLength;
status = RtlUnicodeToMultiByteSize(
&multiByteLength,
Buffer,
(ULONG)Length
);
if (!NT_SUCCESS(status))
return NULL;
bytes = PhCreateBytesEx(NULL, multiByteLength);
status = RtlUnicodeToMultiByteN(
bytes->Buffer,
(ULONG)bytes->Length,
NULL,
Buffer,
(ULONG)Length
);
if (!NT_SUCCESS(status))
{
PhDereferenceObject(bytes);
return NULL;
}
return bytes;
}
BOOLEAN PhConvertUtf8ToUtf16Size(
_Out_ PSIZE_T BytesInUtf16String,
_In_reads_bytes_(BytesInUtf8String) PCH Utf8String,
_In_ SIZE_T BytesInUtf8String
)
{
BOOLEAN result;
PH_UNICODE_DECODER decoder;
PCH in;
SIZE_T inRemaining;
SIZE_T bytesInUtf16String;
ULONG codePoint;
ULONG numberOfCodeUnits;
result = TRUE;
PhInitializeUnicodeDecoder(&decoder, PH_UNICODE_UTF8);
in = Utf8String;
inRemaining = BytesInUtf8String;
bytesInUtf16String = 0;
while (inRemaining != 0)
{
PhWriteUnicodeDecoder(&decoder, (UCHAR)*in);
in++;
inRemaining--;
while (PhDecodeUnicodeDecoder(&decoder, &codePoint))
{
if (PhEncodeUnicode(PH_UNICODE_UTF16, codePoint, NULL, &numberOfCodeUnits))
bytesInUtf16String += numberOfCodeUnits * sizeof(WCHAR);
else
result = FALSE;
}
}
*BytesInUtf16String = bytesInUtf16String;
return result;
}
BOOLEAN PhConvertUtf8ToUtf16Buffer(
_Out_writes_bytes_to_(MaxBytesInUtf16String, *BytesInUtf16String) PWCH Utf16String,
_In_ SIZE_T MaxBytesInUtf16String,
_Out_opt_ PSIZE_T BytesInUtf16String,
_In_reads_bytes_(BytesInUtf8String) PCH Utf8String,
_In_ SIZE_T BytesInUtf8String
)
{
BOOLEAN result;
PH_UNICODE_DECODER decoder;
PCH in;
SIZE_T inRemaining;
PWCH out;
SIZE_T outRemaining;
SIZE_T bytesInUtf16String;
ULONG codePoint;
USHORT codeUnits[2];
ULONG numberOfCodeUnits;
result = TRUE;
PhInitializeUnicodeDecoder(&decoder, PH_UNICODE_UTF8);
in = Utf8String;
inRemaining = BytesInUtf8String;
out = Utf16String;
outRemaining = MaxBytesInUtf16String / sizeof(WCHAR);
bytesInUtf16String = 0;
while (inRemaining != 0)
{
PhWriteUnicodeDecoder(&decoder, (UCHAR)*in);
in++;
inRemaining--;
while (PhDecodeUnicodeDecoder(&decoder, &codePoint))
{
if (PhEncodeUnicode(PH_UNICODE_UTF16, codePoint, codeUnits, &numberOfCodeUnits))
{
bytesInUtf16String += numberOfCodeUnits * sizeof(WCHAR);
if (outRemaining >= numberOfCodeUnits)
{
*out++ = codeUnits[0];
if (numberOfCodeUnits >= 2)
*out++ = codeUnits[1];
outRemaining -= numberOfCodeUnits;
}
else
{
result = FALSE;
}
}
else
{
result = FALSE;
}
}
}
if (BytesInUtf16String)
*BytesInUtf16String = bytesInUtf16String;
return result;
}
PPH_STRING PhConvertUtf8ToUtf16(
_In_ PSTR Buffer
)
{
return PhConvertUtf8ToUtf16Ex(
Buffer,
strlen(Buffer)
);
}
PPH_STRING PhConvertUtf8ToUtf16Ex(
_In_ PCHAR Buffer,
_In_ SIZE_T Length
)
{
PPH_STRING string;
SIZE_T utf16Bytes;
if (!PhConvertUtf8ToUtf16Size(
&utf16Bytes,
Buffer,
Length
))
{
return NULL;
}
string = PhCreateStringEx(NULL, utf16Bytes);
if (!PhConvertUtf8ToUtf16Buffer(
string->Buffer,
string->Length,
NULL,
Buffer,
Length
))
{
PhDereferenceObject(string);
return NULL;
}
return string;
}
BOOLEAN PhConvertUtf16ToUtf8Size(
_Out_ PSIZE_T BytesInUtf8String,
_In_reads_bytes_(BytesInUtf16String) PWCH Utf16String,
_In_ SIZE_T BytesInUtf16String
)
{
BOOLEAN result;
PH_UNICODE_DECODER decoder;
PWCH in;
SIZE_T inRemaining;
SIZE_T bytesInUtf8String;
ULONG codePoint;
ULONG numberOfCodeUnits;
result = TRUE;
PhInitializeUnicodeDecoder(&decoder, PH_UNICODE_UTF16);
in = Utf16String;
inRemaining = BytesInUtf16String / sizeof(WCHAR);
bytesInUtf8String = 0;
while (inRemaining != 0)
{
PhWriteUnicodeDecoder(&decoder, (USHORT)*in);
in++;
inRemaining--;
while (PhDecodeUnicodeDecoder(&decoder, &codePoint))
{
if (PhEncodeUnicode(PH_UNICODE_UTF8, codePoint, NULL, &numberOfCodeUnits))
bytesInUtf8String += numberOfCodeUnits;
else
result = FALSE;
}
}
*BytesInUtf8String = bytesInUtf8String;
return result;
}
BOOLEAN PhConvertUtf16ToUtf8Buffer(
_Out_writes_bytes_to_(MaxBytesInUtf8String, *BytesInUtf8String) PCH Utf8String,
_In_ SIZE_T MaxBytesInUtf8String,
_Out_opt_ PSIZE_T BytesInUtf8String,
_In_reads_bytes_(BytesInUtf16String) PWCH Utf16String,
_In_ SIZE_T BytesInUtf16String
)
{
BOOLEAN result;
PH_UNICODE_DECODER decoder;
PWCH in;
SIZE_T inRemaining;
PCH out;
SIZE_T outRemaining;
SIZE_T bytesInUtf8String;
ULONG codePoint;
UCHAR codeUnits[4];
ULONG numberOfCodeUnits;
result = TRUE;
PhInitializeUnicodeDecoder(&decoder, PH_UNICODE_UTF16);
in = Utf16String;
inRemaining = BytesInUtf16String / sizeof(WCHAR);
out = Utf8String;
outRemaining = MaxBytesInUtf8String;
bytesInUtf8String = 0;
while (inRemaining != 0)
{
PhWriteUnicodeDecoder(&decoder, (USHORT)*in);
in++;
inRemaining--;
while (PhDecodeUnicodeDecoder(&decoder, &codePoint))
{
if (PhEncodeUnicode(PH_UNICODE_UTF8, codePoint, codeUnits, &numberOfCodeUnits))
{
bytesInUtf8String += numberOfCodeUnits;
if (outRemaining >= numberOfCodeUnits)
{
*out++ = codeUnits[0];
if (numberOfCodeUnits >= 2)
*out++ = codeUnits[1];
if (numberOfCodeUnits >= 3)
*out++ = codeUnits[2];
if (numberOfCodeUnits >= 4)
*out++ = codeUnits[3];
outRemaining -= numberOfCodeUnits;
}
else
{
result = FALSE;
}
}
else
{
result = FALSE;
}
}
}
if (BytesInUtf8String)
*BytesInUtf8String = bytesInUtf8String;
return result;
}
PPH_BYTES PhConvertUtf16ToUtf8(
_In_ PWSTR Buffer
)
{
return PhConvertUtf16ToUtf8Ex(
Buffer,
PhCountStringZ(Buffer) * sizeof(WCHAR)
);
}
PPH_BYTES PhConvertUtf16ToUtf8Ex(
_In_ PWCHAR Buffer,
_In_ SIZE_T Length
)
{
PPH_BYTES bytes;
SIZE_T utf8Bytes;
if (!PhConvertUtf16ToUtf8Size(
&utf8Bytes,
Buffer,
Length
))
{
return NULL;
}
bytes = PhCreateBytesEx(NULL, utf8Bytes);
if (!PhConvertUtf16ToUtf8Buffer(
bytes->Buffer,
bytes->Length,
NULL,
Buffer,
Length
))
{
PhDereferenceObject(bytes);
return NULL;
}
return bytes;
}
/**
* Initializes a string builder object.
*
* \param StringBuilder A string builder object.
* \param InitialCapacity The number of bytes to allocate initially.
*/
VOID PhInitializeStringBuilder(
_Out_ PPH_STRING_BUILDER StringBuilder,
_In_ SIZE_T InitialCapacity
)
{
// Make sure the initial capacity is even, as required for all string objects.
if (InitialCapacity & 1)
InitialCapacity++;
StringBuilder->AllocatedLength = InitialCapacity;
// Allocate a PH_STRING for the string builder.
// We will dereference it and allocate a new one when we need to resize the string.
StringBuilder->String = PhCreateStringEx(NULL, StringBuilder->AllocatedLength);
// We will keep modifying the Length field of the string so that:
// 1. We know how much of the string is used, and
// 2. The user can simply get a reference to the string and use it as-is.
StringBuilder->String->Length = 0;
// Write the null terminator.
StringBuilder->String->Buffer[0] = 0;
PHLIB_INC_STATISTIC(BaseStringBuildersCreated);
}
/**
* Frees resources used by a string builder object.
*
* \param StringBuilder A string builder object.
*/
VOID PhDeleteStringBuilder(
_Inout_ PPH_STRING_BUILDER StringBuilder
)
{
PhDereferenceObject(StringBuilder->String);
}
/**
* Obtains a reference to the string constructed by a string builder object and frees resources used
* by the object.
*
* \param StringBuilder A string builder object.
*
* \return A pointer to a string. You must free the string using PhDereferenceObject() when you no
* longer need it.
*/
PPH_STRING PhFinalStringBuilderString(
_Inout_ PPH_STRING_BUILDER StringBuilder
)
{
return StringBuilder->String;
}
VOID PhpResizeStringBuilder(
_In_ PPH_STRING_BUILDER StringBuilder,
_In_ SIZE_T NewCapacity
)
{
PPH_STRING newString;
// Double the string size. If that still isn't enough room, just use the new length.
StringBuilder->AllocatedLength *= 2;
if (StringBuilder->AllocatedLength < NewCapacity)
StringBuilder->AllocatedLength = NewCapacity;
// Allocate a new string.
newString = PhCreateStringEx(NULL, StringBuilder->AllocatedLength);
// Copy the old string to the new string.
memcpy(
newString->Buffer,
StringBuilder->String->Buffer,
StringBuilder->String->Length + sizeof(WCHAR) // Include null terminator
);
// Copy the old string length.
newString->Length = StringBuilder->String->Length;
// Dereference the old string and replace it with the new string.
PhMoveReference(&StringBuilder->String, newString);
PHLIB_INC_STATISTIC(BaseStringBuildersResized);
}
FORCEINLINE VOID PhpWriteNullTerminatorStringBuilder(
_In_ PPH_STRING_BUILDER StringBuilder
)
{
assert(!(StringBuilder->String->Length & 1));
*(PWCHAR)((PCHAR)StringBuilder->String->Buffer + StringBuilder->String->Length) = 0;
}
/**
* Appends a string to the end of a string builder string.
*
* \param StringBuilder A string builder object.
* \param String The string to append.
*/
VOID PhAppendStringBuilder(
_Inout_ PPH_STRING_BUILDER StringBuilder,
_In_ PPH_STRINGREF String
)
{
PhAppendStringBuilderEx(
StringBuilder,
String->Buffer,
String->Length
);
}
/**
* Appends a string to the end of a string builder string.
*
* \param StringBuilder A string builder object.
* \param String The string to append.
*/
VOID PhAppendStringBuilder2(
_Inout_ PPH_STRING_BUILDER StringBuilder,
_In_ PWSTR String
)
{
PhAppendStringBuilderEx(
StringBuilder,
String,
PhCountStringZ(String) * sizeof(WCHAR)
);
}
/**
* Appends a string to the end of a string builder string.
*
* \param StringBuilder A string builder object.
* \param String The string to append. Specify NULL to simply reserve \a Length bytes.
* \param Length The number of bytes to append.
*/
VOID PhAppendStringBuilderEx(
_Inout_ PPH_STRING_BUILDER StringBuilder,
_In_opt_ PWCHAR String,
_In_ SIZE_T Length
)
{
if (Length == 0)
return;
// See if we need to re-allocate the string.
if (StringBuilder->AllocatedLength < StringBuilder->String->Length + Length)
{
PhpResizeStringBuilder(StringBuilder, StringBuilder->String->Length + Length);
}
// Copy the string, add the length, then write the null terminator.
if (String)
{
memcpy(
(PCHAR)StringBuilder->String->Buffer + StringBuilder->String->Length,
String,
Length
);
}
StringBuilder->String->Length += Length;
PhpWriteNullTerminatorStringBuilder(StringBuilder);
}
/**
* Appends a character to the end of a string builder string.
*
* \param StringBuilder A string builder object.
* \param Character The character to append.
*/
VOID PhAppendCharStringBuilder(
_Inout_ PPH_STRING_BUILDER StringBuilder,
_In_ WCHAR Character
)
{
if (StringBuilder->AllocatedLength < StringBuilder->String->Length + sizeof(WCHAR))
{
PhpResizeStringBuilder(StringBuilder, StringBuilder->String->Length + sizeof(WCHAR));
}
*(PWCHAR)((PCHAR)StringBuilder->String->Buffer + StringBuilder->String->Length) = Character;
StringBuilder->String->Length += sizeof(WCHAR);
PhpWriteNullTerminatorStringBuilder(StringBuilder);
}
/**
* Appends a number of characters to the end of a string builder string.
*
* \param StringBuilder A string builder object.
* \param Character The character to append.
* \param Count The number of times to append the character.
*/
VOID PhAppendCharStringBuilder2(
_Inout_ PPH_STRING_BUILDER StringBuilder,
_In_ WCHAR Character,
_In_ SIZE_T Count
)
{
if (Count == 0)
return;
// See if we need to re-allocate the string.
if (StringBuilder->AllocatedLength < StringBuilder->String->Length + Count * sizeof(WCHAR))
{
PhpResizeStringBuilder(StringBuilder, StringBuilder->String->Length + Count * sizeof(WCHAR));
}
wmemset(
(PWCHAR)((PCHAR)StringBuilder->String->Buffer + StringBuilder->String->Length),
Character,
Count
);
StringBuilder->String->Length += Count * sizeof(WCHAR);
PhpWriteNullTerminatorStringBuilder(StringBuilder);
}
/**
* Appends a formatted string to the end of a string builder string.
*
* \param StringBuilder A string builder object.
* \param Format The format-control string.
*/
VOID PhAppendFormatStringBuilder(
_Inout_ PPH_STRING_BUILDER StringBuilder,
_In_ _Printf_format_string_ PWSTR Format,
...
)
{
va_list argptr;
va_start(argptr, Format);
PhAppendFormatStringBuilder_V(StringBuilder, Format, argptr);
}
VOID PhAppendFormatStringBuilder_V(
_Inout_ PPH_STRING_BUILDER StringBuilder,
_In_ _Printf_format_string_ PWSTR Format,
_In_ va_list ArgPtr
)
{
int length;
SIZE_T lengthInBytes;
length = _vscwprintf(Format, ArgPtr);
if (length == -1 || length == 0)
return;
lengthInBytes = length * sizeof(WCHAR);
if (StringBuilder->AllocatedLength < StringBuilder->String->Length + lengthInBytes)
PhpResizeStringBuilder(StringBuilder, StringBuilder->String->Length + lengthInBytes);
_vsnwprintf(
(PWCHAR)((PCHAR)StringBuilder->String->Buffer + StringBuilder->String->Length),
length,
Format,
ArgPtr
);
StringBuilder->String->Length += lengthInBytes;
PhpWriteNullTerminatorStringBuilder(StringBuilder);
}
/**
* Inserts a string into a string builder string.
*
* \param StringBuilder A string builder object.
* \param Index The index, in characters, at which to insert the string.
* \param String The string to insert.
*/
VOID PhInsertStringBuilder(
_Inout_ PPH_STRING_BUILDER StringBuilder,
_In_ SIZE_T Index,
_In_ PPH_STRINGREF String
)
{
PhInsertStringBuilderEx(
StringBuilder,
Index,
String->Buffer,
String->Length
);
}
/**
* Inserts a string into a string builder string.
*
* \param StringBuilder A string builder object.
* \param Index The index, in characters, at which to insert the string.
* \param String The string to insert.
*/
VOID PhInsertStringBuilder2(
_Inout_ PPH_STRING_BUILDER StringBuilder,
_In_ SIZE_T Index,
_In_ PWSTR String
)
{
PhInsertStringBuilderEx(
StringBuilder,
Index,
String,
PhCountStringZ(String) * sizeof(WCHAR)
);
}
/**
* Inserts a string into a string builder string.
*
* \param StringBuilder A string builder object.
* \param Index The index, in characters, at which to insert the string.
* \param String The string to insert. Specify NULL to simply reserve \a Length bytes at \a Index.
* \param Length The number of bytes to insert.
*/
VOID PhInsertStringBuilderEx(
_Inout_ PPH_STRING_BUILDER StringBuilder,
_In_ SIZE_T Index,
_In_opt_ PWCHAR String,
_In_ SIZE_T Length
)
{
if (Length == 0)
return;
// See if we need to re-allocate the string.
if (StringBuilder->AllocatedLength < StringBuilder->String->Length + Length)
{
PhpResizeStringBuilder(StringBuilder, StringBuilder->String->Length + Length);
}
if (Index * sizeof(WCHAR) < StringBuilder->String->Length)
{
// Create some space for the string.
memmove(
&StringBuilder->String->Buffer[Index + Length / sizeof(WCHAR)],
&StringBuilder->String->Buffer[Index],
StringBuilder->String->Length - Index * sizeof(WCHAR)
);
}
if (String)
{
// Copy the new string.
memcpy(
&StringBuilder->String->Buffer[Index],
String,
Length
);
}
StringBuilder->String->Length += Length;
PhpWriteNullTerminatorStringBuilder(StringBuilder);
}
/**
* Removes characters from a string builder string.
*
* \param StringBuilder A string builder object.
* \param StartIndex The index, in characters, at which to begin removing characters.
* \param Count The number of characters to remove.
*/
VOID PhRemoveStringBuilder(
_Inout_ PPH_STRING_BUILDER StringBuilder,
_In_ SIZE_T StartIndex,
_In_ SIZE_T Count
)
{
// Overwrite the removed part with the part
// behind it.
memmove(
&StringBuilder->String->Buffer[StartIndex],
&StringBuilder->String->Buffer[StartIndex + Count],
StringBuilder->String->Length - (Count + StartIndex) * sizeof(WCHAR)
);
StringBuilder->String->Length -= Count * sizeof(WCHAR);
PhpWriteNullTerminatorStringBuilder(StringBuilder);
}
/**
* Initializes a byte string builder object.
*
* \param BytesBuilder A byte string builder object.
* \param InitialCapacity The number of bytes to allocate initially.
*/
VOID PhInitializeBytesBuilder(
_Out_ PPH_BYTES_BUILDER BytesBuilder,
_In_ SIZE_T InitialCapacity
)
{
BytesBuilder->AllocatedLength = InitialCapacity;
BytesBuilder->Bytes = PhCreateBytesEx(NULL, BytesBuilder->AllocatedLength);
BytesBuilder->Bytes->Length = 0;
BytesBuilder->Bytes->Buffer[0] = 0;
}
/**
* Frees resources used by a byte string builder object.
*
* \param BytesBuilder A byte string builder object.
*/
VOID PhDeleteBytesBuilder(
_Inout_ PPH_BYTES_BUILDER BytesBuilder
)
{
PhDereferenceObject(BytesBuilder->Bytes);
}
/**
* Obtains a reference to the byte string constructed by a byte string builder object and frees
* resources used by the object.
*
* \param BytesBuilder A byte string builder object.
*
* \return A pointer to a byte string. You must free the byte string using PhDereferenceObject()
* when you no longer need it.
*/
PPH_BYTES PhFinalBytesBuilderBytes(
_Inout_ PPH_BYTES_BUILDER BytesBuilder
)
{
return BytesBuilder->Bytes;
}
VOID PhpResizeBytesBuilder(
_In_ PPH_BYTES_BUILDER BytesBuilder,
_In_ SIZE_T NewCapacity
)
{
PPH_BYTES newBytes;
// Double the byte string size. If that still isn't enough room, just use the new length.
BytesBuilder->AllocatedLength *= 2;
if (BytesBuilder->AllocatedLength < NewCapacity)
BytesBuilder->AllocatedLength = NewCapacity;
// Allocate a new byte string.
newBytes = PhCreateBytesEx(NULL, BytesBuilder->AllocatedLength);
// Copy the old byte string to the new byte string.
memcpy(
newBytes->Buffer,
BytesBuilder->Bytes->Buffer,
BytesBuilder->Bytes->Length + sizeof(CHAR) // Include null terminator
);
// Copy the old byte string length.
newBytes->Length = BytesBuilder->Bytes->Length;
// Dereference the old byte string and replace it with the new byte string.
PhMoveReference(&BytesBuilder->Bytes, newBytes);
}
FORCEINLINE VOID PhpWriteNullTerminatorBytesBuilder(
_In_ PPH_BYTES_BUILDER BytesBuilder
)
{
BytesBuilder->Bytes->Buffer[BytesBuilder->Bytes->Length] = 0;
}
/**
* Appends a byte string to the end of a byte string builder string.
*
* \param BytesBuilder A byte string builder object.
* \param Bytes The byte string to append.
*/
VOID PhAppendBytesBuilder(
_Inout_ PPH_BYTES_BUILDER BytesBuilder,
_In_ PPH_BYTESREF Bytes
)
{
PhAppendBytesBuilderEx(
BytesBuilder,
Bytes->Buffer,
Bytes->Length,
0,
NULL
);
}
/**
* Appends a byte string to the end of a byte string builder string.
*
* \param BytesBuilder A byte string builder object.
* \param Bytes The byte string to append.
*/
VOID PhAppendBytesBuilder2(
_Inout_ PPH_BYTES_BUILDER BytesBuilder,
_In_ PCHAR Bytes
)
{
PhAppendBytesBuilderEx(
BytesBuilder,
Bytes,
strlen(Bytes),
0,
NULL
);
}
/**
* Appends a byte string to the end of a byte string builder string.
*
* \param BytesBuilder A byte string builder object.
* \param Buffer The byte string to append. Specify NULL to simply reserve \a Length bytes.
* \param Length The number of bytes to append.
* \param Alignment The required alignment. This should not be greater than 8.
* \param Offset A variable which receives the byte offset of the appended or reserved bytes in the
* byte string builder string.
*
* \return A pointer to the appended or reserved bytes.
*/
PVOID PhAppendBytesBuilderEx(
_Inout_ PPH_BYTES_BUILDER BytesBuilder,
_In_opt_ PVOID Buffer,
_In_ SIZE_T Length,
_In_opt_ SIZE_T Alignment,
_Out_opt_ PSIZE_T Offset
)
{
SIZE_T currentLength;
currentLength = BytesBuilder->Bytes->Length;
if (Length == 0)
goto Done;
if (Alignment)
currentLength = ALIGN_UP_BY(currentLength, Alignment);
// See if we need to re-allocate the byte string.
if (BytesBuilder->AllocatedLength < currentLength + Length)
PhpResizeBytesBuilder(BytesBuilder, currentLength + Length);
// Copy the byte string, add the length, then write the null terminator.
if (Buffer)
memcpy(BytesBuilder->Bytes->Buffer + currentLength, Buffer, Length);
BytesBuilder->Bytes->Length = currentLength + Length;
PhpWriteNullTerminatorBytesBuilder(BytesBuilder);
Done:
if (Offset)
*Offset = currentLength;
return BytesBuilder->Bytes->Buffer + currentLength;
}
/**
* Creates an array object.
*
* \param Array An array object.
* \param ItemSize The size of each item, in bytes.
* \param InitialCapacity The number of elements to allocate storage for, initially.
*/
VOID PhInitializeArray(
_Out_ PPH_ARRAY Array,
_In_ SIZE_T ItemSize,
_In_ SIZE_T InitialCapacity
)
{
// Initial capacity of 0 is not allowed.
if (InitialCapacity == 0)
InitialCapacity = 1;
Array->Count = 0;
Array->AllocatedCount = InitialCapacity;
Array->ItemSize = ItemSize;
Array->Items = PhAllocate(Array->AllocatedCount * ItemSize);
}
/**
* Frees resources used by an array object.
*
* \param Array An array object.
*/
VOID PhDeleteArray(
_Inout_ PPH_ARRAY Array
)
{
PhFree(Array->Items);
}
/**
* Obtains a copy of the array constructed by an array object and frees resources used by the
* object.
*
* \param Array An array object.
*
* \return The array buffer.
*/
PVOID PhFinalArrayItems(
_Inout_ PPH_ARRAY Array
)
{
return Array->Items;
}
/**
* Resizes an array.
*
* \param Array An array object.
* \param NewCapacity The new required number of elements for which storage has been reserved. This
* must not be smaller than the current number of items in the array.
*/
VOID PhResizeArray(
_Inout_ PPH_ARRAY Array,
_In_ SIZE_T NewCapacity
)
{
if (Array->Count > NewCapacity)
PhRaiseStatus(STATUS_INVALID_PARAMETER_2);
Array->AllocatedCount = NewCapacity;
Array->Items = PhReAllocate(Array->Items, Array->AllocatedCount * Array->ItemSize);
}
/**
* Adds an item to an array.
*
* \param Array An array object.
* \param Item The item to add.
*/
VOID PhAddItemArray(
_Inout_ PPH_ARRAY Array,
_In_ PVOID Item
)
{
// See if we need to resize the list.
if (Array->Count == Array->AllocatedCount)
{
Array->AllocatedCount *= 2;
Array->Items = PhReAllocate(Array->Items, Array->AllocatedCount * Array->ItemSize);
}
memcpy(PhItemArray(Array, Array->Count), Item, Array->ItemSize);
Array->Count++;
}
/**
* Adds items to an array.
*
* \param Array An array object.
* \param Items An array containing the items to add.
* \param Count The number of items to add.
*/
VOID PhAddItemsArray(
_Inout_ PPH_ARRAY Array,
_In_ PVOID Items,
_In_ SIZE_T Count
)
{
// See if we need to resize the list.
if (Array->AllocatedCount < Array->Count + Count)
{
Array->AllocatedCount *= 2;
if (Array->AllocatedCount < Array->Count + Count)
Array->AllocatedCount = Array->Count + Count;
Array->Items = PhReAllocate(Array->Items, Array->AllocatedCount * Array->ItemSize);
}
memcpy(
PhItemArray(Array, Array->Count),
Items,
Count * Array->ItemSize
);
Array->Count += Count;
}
/**
* Clears an array.
*
* \param Array An array object.
*/
VOID PhClearArray(
_Inout_ PPH_ARRAY Array
)
{
Array->Count = 0;
}
/**
* Removes an item from an array.
*
* \param Array An array object.
* \param Index The index of the item.
*/
VOID PhRemoveItemArray(
_Inout_ PPH_ARRAY Array,
_In_ SIZE_T Index
)
{
PhRemoveItemsArray(Array, Index, 1);
}
/**
* Removes items from an array.
*
* \param Array An array object.
* \param StartIndex The index at which to begin removing items.
* \param Count The number of items to remove.
*/
VOID PhRemoveItemsArray(
_Inout_ PPH_ARRAY Array,
_In_ SIZE_T StartIndex,
_In_ SIZE_T Count
)
{
// Shift the items after the items forward.
memmove(
PhItemArray(Array, StartIndex),
PhItemArray(Array, StartIndex + Count),
(Array->Count - StartIndex - Count) * Array->ItemSize
);
Array->Count -= Count;
}
/**
* Creates a list object.
*
* \param InitialCapacity The number of elements to allocate storage for, initially.
*/
PPH_LIST PhCreateList(
_In_ ULONG InitialCapacity
)
{
PPH_LIST list;
list = PhCreateObject(sizeof(PH_LIST), PhListType);
// Initial capacity of 0 is not allowed.
if (InitialCapacity == 0)
InitialCapacity = 1;
list->Count = 0;
list->AllocatedCount = InitialCapacity;
list->Items = PhAllocate(list->AllocatedCount * sizeof(PVOID));
return list;
}
VOID PhpListDeleteProcedure(
_In_ PVOID Object,
_In_ ULONG Flags
)
{
PPH_LIST list = (PPH_LIST)Object;
PhFree(list->Items);
}
/**
* Resizes a list.
*
* \param List A list object.
* \param NewCapacity The new required number of elements for which storage has been reserved. This
* must not be smaller than the current number of items in the list.
*/
VOID PhResizeList(
_Inout_ PPH_LIST List,
_In_ ULONG NewCapacity
)
{
if (List->Count > NewCapacity)
PhRaiseStatus(STATUS_INVALID_PARAMETER_2);
List->AllocatedCount = NewCapacity;
List->Items = PhReAllocate(List->Items, List->AllocatedCount * sizeof(PVOID));
}
/**
* Adds an item to a list.
*
* \param List A list object.
* \param Item The item to add.
*/
VOID PhAddItemList(
_Inout_ PPH_LIST List,
_In_ PVOID Item
)
{
// See if we need to resize the list.
if (List->Count == List->AllocatedCount)
{
List->AllocatedCount *= 2;
List->Items = PhReAllocate(List->Items, List->AllocatedCount * sizeof(PVOID));
}
List->Items[List->Count++] = Item;
}
/**
* Adds items to a list.
*
* \param List A list object.
* \param Items An array containing the items to add.
* \param Count The number of items to add.
*/
VOID PhAddItemsList(
_Inout_ PPH_LIST List,
_In_ PVOID *Items,
_In_ ULONG Count
)
{
// See if we need to resize the list.
if (List->AllocatedCount < List->Count + Count)
{
List->AllocatedCount *= 2;
if (List->AllocatedCount < List->Count + Count)
List->AllocatedCount = List->Count + Count;
List->Items = PhReAllocate(List->Items, List->AllocatedCount * sizeof(PVOID));
}
memcpy(
&List->Items[List->Count],
Items,
Count * sizeof(PVOID)
);
List->Count += Count;
}
/**
* Clears a list.
*
* \param List A list object.
*/
VOID PhClearList(
_Inout_ PPH_LIST List
)
{
List->Count = 0;
}
_Success_(return != -1)
/**
* Locates an item in a list.
*
* \param List A list object.
* \param Item The item to search for.
*
* \return The index of the item. If the
* item was not found, -1 is returned.
*/
ULONG PhFindItemList(
_In_ PPH_LIST List,
_In_ PVOID Item
)
{
ULONG i;
for (i = 0; i < List->Count; i++)
{
if (List->Items[i] == Item)
return i;
}
return -1;
}
/**
* Inserts an item into a list.
*
* \param List A list object.
* \param Index The index at which to insert the item.
* \param Item The item to add.
*/
VOID PhInsertItemList(
_Inout_ PPH_LIST List,
_In_ ULONG Index,
_In_ PVOID Item
)
{
PhInsertItemsList(List, Index, &Item, 1);
}
/**
* Inserts items into a list.
*
* \param List A list object.
* \param Index The index at which to insert the items.
* \param Items An array containing the items to add.
* \param Count The number of items to add.
*/
VOID PhInsertItemsList(
_Inout_ PPH_LIST List,
_In_ ULONG Index,
_In_ PVOID *Items,
_In_ ULONG Count
)
{
// See if we need to resize the list.
if (List->AllocatedCount < List->Count + Count)
{
List->AllocatedCount *= 2;
if (List->AllocatedCount < List->Count + Count)
List->AllocatedCount = List->Count + Count;
List->Items = PhReAllocate(List->Items, List->AllocatedCount * sizeof(PVOID));
}
if (Index < List->Count)
{
// Shift the existing items backward.
memmove(
&List->Items[Index + Count],
&List->Items[Index],
(List->Count - Index) * sizeof(PVOID)
);
}
// Copy the new items into the list.
memcpy(
&List->Items[Index],
Items,
Count * sizeof(PVOID)
);
List->Count += Count;
}
/**
* Removes an item from a list.
*
* \param List A list object.
* \param Index The index of the item.
*/
VOID PhRemoveItemList(
_Inout_ PPH_LIST List,
_In_ ULONG Index
)
{
PhRemoveItemsList(List, Index, 1);
}
/**
* Removes items from a list.
*
* \param List A list object.
* \param StartIndex The index at which to begin removing items.
* \param Count The number of items to remove.
*/
VOID PhRemoveItemsList(
_Inout_ PPH_LIST List,
_In_ ULONG StartIndex,
_In_ ULONG Count
)
{
// Shift the items after the items forward.
memmove(
&List->Items[StartIndex],
&List->Items[StartIndex + Count],
(List->Count - StartIndex - Count) * sizeof(PVOID)
);
List->Count -= Count;
}
/**
* Creates a pointer list object.
*
* \param InitialCapacity The number of elements to allocate storage for initially.
*/
PPH_POINTER_LIST PhCreatePointerList(
_In_ ULONG InitialCapacity
)
{
PPH_POINTER_LIST pointerList;
pointerList = PhCreateObject(sizeof(PH_POINTER_LIST), PhPointerListType);
// Initial capacity of 0 is not allowed.
if (InitialCapacity == 0)
InitialCapacity = 1;
pointerList->Count = 0;
pointerList->AllocatedCount = InitialCapacity;
pointerList->FreeEntry = -1;
pointerList->NextEntry = 0;
pointerList->Items = PhAllocate(pointerList->AllocatedCount * sizeof(PVOID));
return pointerList;
}
VOID NTAPI PhpPointerListDeleteProcedure(
_In_ PVOID Object,
_In_ ULONG Flags
)
{
PPH_POINTER_LIST pointerList = (PPH_POINTER_LIST)Object;
PhFree(pointerList->Items);
}
/**
* Decodes an index stored in a free entry.
*/
FORCEINLINE ULONG PhpDecodePointerListIndex(
_In_ PVOID Index
)
{
// At least with Microsoft's compiler, shift right on a signed value preserves the sign. This is
// important because we want decode(encode(-1)) = ((-1 << 1) | 1) >> 1 = -1.
return (ULONG)((LONG_PTR)Index >> 1);
}
/**
* Encodes an index for storage in a free entry.
*/
FORCEINLINE PVOID PhpEncodePointerListIndex(
_In_ ULONG Index
)
{
return (PVOID)(((ULONG_PTR)Index << 1) | 0x1);
}
FORCEINLINE HANDLE PhpPointerListIndexToHandle(
_In_ ULONG Index
)
{
// Add one to allow NULL handles to indicate failure/an invalid index.
return UlongToHandle(Index + 1);
}
FORCEINLINE ULONG PhpPointerListHandleToIndex(
_In_ HANDLE Handle
)
{
return HandleToUlong(Handle) - 1;
}
/**
* Adds a pointer to a pointer list.
*
* \param PointerList A pointer list object.
* \param Pointer The pointer to add. The pointer must be at least 2 byte aligned.
*
* \return A handle to the pointer, valid until the pointer is removed from the pointer list.
*/
HANDLE PhAddItemPointerList(
_Inout_ PPH_POINTER_LIST PointerList,
_In_ PVOID Pointer
)
{
ULONG index;
assert(PH_IS_LIST_POINTER_VALID(Pointer));
// Use a free entry if possible.
if (PointerList->FreeEntry != -1)
{
PVOID oldPointer;
index = PointerList->FreeEntry;
oldPointer = PointerList->Items[index];
PointerList->Items[index] = Pointer;
PointerList->FreeEntry = PhpDecodePointerListIndex(oldPointer);
}
else
{
// Use the next entry.
if (PointerList->NextEntry == PointerList->AllocatedCount)
{
PointerList->AllocatedCount *= 2;
PointerList->Items = PhReAllocate(PointerList->Items, PointerList->AllocatedCount * sizeof(PVOID));
}
index = PointerList->NextEntry++;
PointerList->Items[index] = Pointer;
}
PointerList->Count++;
return PhpPointerListIndexToHandle(index);
}
BOOLEAN PhEnumPointerListEx(
_In_ PPH_POINTER_LIST PointerList,
_Inout_ PULONG EnumerationKey,
_Out_ PVOID *Pointer,
_Out_ PHANDLE PointerHandle
)
{
ULONG index;
while ((index = *EnumerationKey) < PointerList->NextEntry)
{
PVOID pointer = PointerList->Items[index];
(*EnumerationKey)++;
if (PH_IS_LIST_POINTER_VALID(pointer))
{
*Pointer = pointer;
*PointerHandle = PhpPointerListIndexToHandle(index);
return TRUE;
}
}
return FALSE;
}
/**
* Locates a pointer in a pointer list.
*
* \param PointerList A pointer list object.
* \param Pointer The pointer to find. The pointer must be at least 2 byte aligned.
*
* \return A handle to the pointer, valid until the pointer is removed from the pointer list. If the
* pointer is not contained in the pointer list, NULL is returned.
*/
HANDLE PhFindItemPointerList(
_In_ PPH_POINTER_LIST PointerList,
_In_ PVOID Pointer
)
{
ULONG i;
assert(PH_IS_LIST_POINTER_VALID(Pointer));
for (i = 0; i < PointerList->NextEntry; i++)
{
if (PointerList->Items[i] == Pointer)
return PhpPointerListIndexToHandle(i);
}
return NULL;
}
/**
* Removes a pointer from a pointer list.
*
* \param PointerList A pointer list object.
* \param PointerHandle A handle to the pointer to remove.
*
* \remarks No checking is performed on the pointer handle. Make sure the handle is valid before
* calling the function.
*/
VOID PhRemoveItemPointerList(
_Inout_ PPH_POINTER_LIST PointerList,
_In_ HANDLE PointerHandle
)
{
ULONG index;
assert(PointerHandle);
index = PhpPointerListHandleToIndex(PointerHandle);
PointerList->Items[index] = PhpEncodePointerListIndex(PointerList->FreeEntry);
PointerList->FreeEntry = index;
PointerList->Count--;
}
FORCEINLINE ULONG PhpValidateHash(
_In_ ULONG Hash
)
{
// No point in using a full hash when we're going to AND with size minus one anyway.
#if defined(PH_HASHTABLE_FULL_HASH) && !defined(PH_HASHTABLE_POWER_OF_TWO_SIZE)
if (Hash != -1)
return Hash;
else
return 0;
#else
return Hash & MAXLONG;
#endif
}
FORCEINLINE ULONG PhpIndexFromHash(
_In_ PPH_HASHTABLE Hashtable,
_In_ ULONG Hash
)
{
#ifdef PH_HASHTABLE_POWER_OF_TWO_SIZE
return Hash & (Hashtable->AllocatedBuckets - 1);
#else
return Hash % Hashtable->AllocatedBuckets;
#endif
}
FORCEINLINE ULONG PhpGetNumberOfBuckets(
_In_ ULONG Capacity
)
{
#ifdef PH_HASHTABLE_POWER_OF_TWO_SIZE
return PhRoundUpToPowerOfTwo(Capacity);
#else
return PhGetPrimeNumber(Capacity);
#endif
}
/**
* Creates a hashtable object.
*
* \param EntrySize The size of each hashtable entry, in bytes.
* \param EqualFunction A comparison function that is executed to compare two hashtable entries.
* \param HashFunction A hash function that is executed to generate a hash code for a hashtable
* entry.
* \param InitialCapacity The number of entries to allocate storage for initially.
*/
PPH_HASHTABLE PhCreateHashtable(
_In_ ULONG EntrySize,
_In_ PPH_HASHTABLE_EQUAL_FUNCTION EqualFunction,
_In_ PPH_HASHTABLE_HASH_FUNCTION HashFunction,
_In_ ULONG InitialCapacity
)
{
PPH_HASHTABLE hashtable;
hashtable = PhCreateObject(sizeof(PH_HASHTABLE), PhHashtableType);
// Initial capacity of 0 is not allowed.
if (InitialCapacity == 0)
InitialCapacity = 1;
hashtable->EntrySize = EntrySize;
hashtable->EqualFunction = EqualFunction;
hashtable->HashFunction = HashFunction;
// Allocate the buckets.
hashtable->AllocatedBuckets = PhpGetNumberOfBuckets(InitialCapacity);
hashtable->Buckets = PhAllocate(sizeof(ULONG) * hashtable->AllocatedBuckets);
// Set all bucket values to -1.
memset(hashtable->Buckets, 0xff, sizeof(ULONG) * hashtable->AllocatedBuckets);
// Allocate the entries.
hashtable->AllocatedEntries = hashtable->AllocatedBuckets;
hashtable->Entries = PhAllocate(PH_HASHTABLE_ENTRY_SIZE(EntrySize) * hashtable->AllocatedEntries);
hashtable->Count = 0;
hashtable->FreeEntry = -1;
hashtable->NextEntry = 0;
return hashtable;
}
VOID PhpHashtableDeleteProcedure(
_In_ PVOID Object,
_In_ ULONG Flags
)
{
PPH_HASHTABLE hashtable = (PPH_HASHTABLE)Object;
PhFree(hashtable->Buckets);
PhFree(hashtable->Entries);
}
VOID PhpResizeHashtable(
_Inout_ PPH_HASHTABLE Hashtable,
_In_ ULONG NewCapacity
)
{
PPH_HASHTABLE_ENTRY entry;
ULONG i;
// Re-allocate the buckets. Note that we don't need to keep the contents.
Hashtable->AllocatedBuckets = PhpGetNumberOfBuckets(NewCapacity);
PhFree(Hashtable->Buckets);
Hashtable->Buckets = PhAllocate(sizeof(ULONG) * Hashtable->AllocatedBuckets);
// Set all bucket values to -1.
memset(Hashtable->Buckets, 0xff, sizeof(ULONG) * Hashtable->AllocatedBuckets);
// Re-allocate the entries.
Hashtable->AllocatedEntries = Hashtable->AllocatedBuckets;
Hashtable->Entries = PhReAllocate(
Hashtable->Entries,
PH_HASHTABLE_ENTRY_SIZE(Hashtable->EntrySize) * Hashtable->AllocatedEntries
);
// Re-distribute the entries among the buckets.
// PH_HASHTABLE_GET_ENTRY is quite slow (it involves a multiply), so we use a pointer here.
entry = Hashtable->Entries;
for (i = 0; i < Hashtable->NextEntry; i++)
{
if (entry->HashCode != -1)
{
ULONG index = PhpIndexFromHash(Hashtable, entry->HashCode);
entry->Next = Hashtable->Buckets[index];
Hashtable->Buckets[index] = i;
}
entry = (PPH_HASHTABLE_ENTRY)((ULONG_PTR)entry + PH_HASHTABLE_ENTRY_SIZE(Hashtable->EntrySize));
}
}
FORCEINLINE PVOID PhpAddEntryHashtable(
_Inout_ PPH_HASHTABLE Hashtable,
_In_ PVOID Entry,
_In_ BOOLEAN CheckForDuplicate,
_Out_opt_ PBOOLEAN Added
)
{
ULONG hashCode; // hash code of the new entry
ULONG index; // bucket index of the new entry
ULONG freeEntry; // index of new entry in entry array
PPH_HASHTABLE_ENTRY entry; // pointer to new entry in entry array
hashCode = PhpValidateHash(Hashtable->HashFunction(Entry));
index = PhpIndexFromHash(Hashtable, hashCode);
if (CheckForDuplicate)
{
ULONG i;
for (i = Hashtable->Buckets[index]; i != -1; i = entry->Next)
{
entry = PH_HASHTABLE_GET_ENTRY(Hashtable, i);
if (entry->HashCode == hashCode && Hashtable->EqualFunction(&entry->Body, Entry))
{
if (Added)
*Added = FALSE;
return &entry->Body;
}
}
}
// Use a free entry if possible.
if (Hashtable->FreeEntry != -1)
{
freeEntry = Hashtable->FreeEntry;
entry = PH_HASHTABLE_GET_ENTRY(Hashtable, freeEntry);
Hashtable->FreeEntry = entry->Next;
}
else
{
// Use the next entry in the entry array.
if (Hashtable->NextEntry == Hashtable->AllocatedEntries)
{
// Resize the hashtable.
PhpResizeHashtable(Hashtable, Hashtable->AllocatedBuckets * 2);
index = PhpIndexFromHash(Hashtable, hashCode);
}
freeEntry = Hashtable->NextEntry++;
entry = PH_HASHTABLE_GET_ENTRY(Hashtable, freeEntry);
}
// Initialize the entry.
entry->HashCode = hashCode;
entry->Next = Hashtable->Buckets[index];
Hashtable->Buckets[index] = freeEntry;
// Copy the user-supplied data to the entry.
memcpy(&entry->Body, Entry, Hashtable->EntrySize);
Hashtable->Count++;
if (Added)
*Added = TRUE;
return &entry->Body;
}
/**
* Adds an entry to a hashtable.
*
* \param Hashtable A hashtable object.
* \param Entry The entry to add.
*
* \return A pointer to the entry as stored in the hashtable. This pointer is valid until the
* hashtable is modified. If the hashtable already contained an equal entry, NULL is returned.
*
* \remarks Entries are only guaranteed to be 8 byte aligned, even on 64-bit systems.
*/
PVOID PhAddEntryHashtable(
_Inout_ PPH_HASHTABLE Hashtable,
_In_ PVOID Entry
)
{
PVOID entry;
BOOLEAN added;
entry = PhpAddEntryHashtable(Hashtable, Entry, TRUE, &added);
if (added)
return entry;
else
return NULL;
}
/**
* Adds an entry to a hashtable or returns an existing one.
*
* \param Hashtable A hashtable object.
* \param Entry The entry to add.
* \param Added A variable which receives TRUE if a new entry was created, and FALSE if an existing
* entry was returned.
*
* \return A pointer to the entry as stored in the hashtable. This pointer is valid until the
* hashtable is modified. If the hashtable already contained an equal entry, the existing entry is
* returned. Check the value of \a Added to determine whether the returned entry is new or existing.
*
* \remarks Entries are only guaranteed to be 8 byte aligned, even on 64-bit systems.
*/
PVOID PhAddEntryHashtableEx(
_Inout_ PPH_HASHTABLE Hashtable,
_In_ PVOID Entry,
_Out_opt_ PBOOLEAN Added
)
{
return PhpAddEntryHashtable(Hashtable, Entry, TRUE, Added);
}
/**
* Clears a hashtable.
*
* \param Hashtable A hashtable object.
*/
VOID PhClearHashtable(
_Inout_ PPH_HASHTABLE Hashtable
)
{
if (Hashtable->Count > 0)
{
memset(Hashtable->Buckets, 0xff, sizeof(ULONG) * Hashtable->AllocatedBuckets);
Hashtable->Count = 0;
Hashtable->FreeEntry = -1;
Hashtable->NextEntry = 0;
}
}
/**
* Enumerates the entries in a hashtable.
*
* \param Hashtable A hashtable object.
* \param Entry A variable which receives a pointer to the hashtable entry. The pointer is valid
* until the hashtable is modified.
* \param EnumerationKey A variable which is initialized to 0 before first calling this function.
*
* \return TRUE if an entry pointer was stored in \a Entry, FALSE if there are no more entries.
*
* \remarks Do not modify the hashtable while the hashtable is being enumerated (between calls to
* this function). Otherwise, the function may behave unexpectedly. You may reset the
* \a EnumerationKey variable to 0 if you wish to restart the enumeration.
*/
BOOLEAN PhEnumHashtable(
_In_ PPH_HASHTABLE Hashtable,
_Out_ PVOID *Entry,
_Inout_ PULONG EnumerationKey
)
{
while (*EnumerationKey < Hashtable->NextEntry)
{
PPH_HASHTABLE_ENTRY entry = PH_HASHTABLE_GET_ENTRY(Hashtable, *EnumerationKey);
(*EnumerationKey)++;
if (entry->HashCode != -1)
{
*Entry = &entry->Body;
return TRUE;
}
}
return FALSE;
}
/**
* Locates an entry in a hashtable.
*
* \param Hashtable A hashtable object.
* \param Entry An entry representing the entry to find.
*
* \return A pointer to the entry as stored in the hashtable. This pointer is valid until the
* hashtable is modified. If the entry could not be found, NULL is returned.
*
* \remarks The entry specified in \a Entry can be a partial entry that is filled in enough so that
* the comparison and hash functions can work with them.
*/
PVOID PhFindEntryHashtable(
_In_ PPH_HASHTABLE Hashtable,
_In_ PVOID Entry
)
{
ULONG hashCode;
ULONG index;
ULONG i;
PPH_HASHTABLE_ENTRY entry;
hashCode = PhpValidateHash(Hashtable->HashFunction(Entry));
index = PhpIndexFromHash(Hashtable, hashCode);
for (i = Hashtable->Buckets[index]; i != -1; i = entry->Next)
{
entry = PH_HASHTABLE_GET_ENTRY(Hashtable, i);
if (entry->HashCode == hashCode && Hashtable->EqualFunction(&entry->Body, Entry))
{
return &entry->Body;
}
}
return NULL;
}
/**
* Removes an entry from a hashtable.
*
* \param Hashtable A hashtable object.
* \param Entry The entry to remove.
*
* \return TRUE if the entry was removed, FALSE if the entry could not be found.
*
* \remarks The entry specified in \a Entry can be an actual entry pointer returned by
* PhFindEntryHashtable, or a partial entry.
*/
BOOLEAN PhRemoveEntryHashtable(
_Inout_ PPH_HASHTABLE Hashtable,
_In_ PVOID Entry
)
{
ULONG hashCode;
ULONG index;
ULONG i;
ULONG previousIndex;
PPH_HASHTABLE_ENTRY entry;
hashCode = PhpValidateHash(Hashtable->HashFunction(Entry));
index = PhpIndexFromHash(Hashtable, hashCode);
previousIndex = -1;
for (i = Hashtable->Buckets[index]; i != -1; i = entry->Next)
{
entry = PH_HASHTABLE_GET_ENTRY(Hashtable, i);
if (entry->HashCode == hashCode && Hashtable->EqualFunction(&entry->Body, Entry))
{
// Unlink the entry from the bucket.
if (previousIndex == -1)
{
Hashtable->Buckets[index] = entry->Next;
}
else
{
PH_HASHTABLE_GET_ENTRY(Hashtable, previousIndex)->Next = entry->Next;
}
entry->HashCode = -1; // indicates the entry is not being used
entry->Next = Hashtable->FreeEntry;
Hashtable->FreeEntry = i;
Hashtable->Count--;
return TRUE;
}
previousIndex = i;
}
return FALSE;
}
/**
* Generates a hash code for a sequence of bytes.
*
* \param Bytes A pointer to a byte array.
* \param Length The number of bytes to hash.
*/
ULONG PhHashBytes(
_In_reads_(Length) PUCHAR Bytes,
_In_ SIZE_T Length
)
{
ULONG hash = 0;
if (Length == 0)
return hash;
// FNV-1a algorithm: http://www.isthe.com/chongo/src/fnv/hash_32a.c
do
{
hash ^= *Bytes++;
hash *= 0x01000193;
} while (--Length != 0);
return hash;
}
/**
* Generates a hash code for a string.
*
* \param String The string to hash.
* \param IgnoreCase TRUE for a case-insensitive hash function, otherwise FALSE.
*/
ULONG PhHashStringRef(
_In_ PPH_STRINGREF String,
_In_ BOOLEAN IgnoreCase
)
{
ULONG hash = 0;
SIZE_T count;
PWCHAR p;
if (String->Length == 0)
return 0;
count = String->Length / sizeof(WCHAR);
p = String->Buffer;
if (!IgnoreCase)
{
return PhHashBytes((PUCHAR)String->Buffer, String->Length);
}
else
{
do
{
hash ^= (USHORT)RtlUpcaseUnicodeChar(*p++);
hash *= 0x01000193;
} while (--count != 0);
}
return hash;
}
BOOLEAN NTAPI PhpSimpleHashtableEqualFunction(
_In_ PVOID Entry1,
_In_ PVOID Entry2
)
{
PPH_KEY_VALUE_PAIR entry1 = Entry1;
PPH_KEY_VALUE_PAIR entry2 = Entry2;
return entry1->Key == entry2->Key;
}
ULONG NTAPI PhpSimpleHashtableHashFunction(
_In_ PVOID Entry
)
{
PPH_KEY_VALUE_PAIR entry = Entry;
return PhHashIntPtr((ULONG_PTR)entry->Key);
}
PPH_HASHTABLE PhCreateSimpleHashtable(
_In_ ULONG InitialCapacity
)
{
return PhCreateHashtable(
sizeof(PH_KEY_VALUE_PAIR),
PhpSimpleHashtableEqualFunction,
PhpSimpleHashtableHashFunction,
InitialCapacity
);
}
PVOID PhAddItemSimpleHashtable(
_Inout_ PPH_HASHTABLE SimpleHashtable,
_In_opt_ PVOID Key,
_In_opt_ PVOID Value
)
{
PH_KEY_VALUE_PAIR entry;
entry.Key = Key;
entry.Value = Value;
if (PhAddEntryHashtable(SimpleHashtable, &entry))
return Value;
else
return NULL;
}
PVOID *PhFindItemSimpleHashtable(
_In_ PPH_HASHTABLE SimpleHashtable,
_In_opt_ PVOID Key
)
{
PH_KEY_VALUE_PAIR lookupEntry;
PPH_KEY_VALUE_PAIR entry;
lookupEntry.Key = Key;
entry = PhFindEntryHashtable(SimpleHashtable, &lookupEntry);
if (entry)
return &entry->Value;
else
return NULL;
}
BOOLEAN PhRemoveItemSimpleHashtable(
_Inout_ PPH_HASHTABLE SimpleHashtable,
_In_opt_ PVOID Key
)
{
PH_KEY_VALUE_PAIR lookupEntry;
lookupEntry.Key = Key;
return PhRemoveEntryHashtable(SimpleHashtable, &lookupEntry);
}
/**
* Initializes a free list object.
*
* \param FreeList A pointer to the free list object.
* \param Size The number of bytes in each allocation.
* \param MaximumCount The number of unused allocations to store.
*/
VOID PhInitializeFreeList(
_Out_ PPH_FREE_LIST FreeList,
_In_ SIZE_T Size,
_In_ ULONG MaximumCount
)
{
RtlInitializeSListHead(&FreeList->ListHead);
FreeList->Count = 0;
FreeList->MaximumCount = MaximumCount;
FreeList->Size = Size;
}
/**
* Frees resources used by a free list object.
*
* \param FreeList A pointer to the free list object.
*/
VOID PhDeleteFreeList(
_Inout_ PPH_FREE_LIST FreeList
)
{
PPH_FREE_LIST_ENTRY entry;
PSLIST_ENTRY listEntry;
listEntry = RtlInterlockedFlushSList(&FreeList->ListHead);
while (listEntry)
{
entry = CONTAINING_RECORD(listEntry, PH_FREE_LIST_ENTRY, ListEntry);
listEntry = listEntry->Next;
PhFree(entry);
}
}
/**
* Allocates a block of memory from a free list.
*
* \param FreeList A pointer to a free list object.
*
* \return A pointer to the allocated block of memory. The memory must be freed using
* PhFreeToFreeList(). The block is guaranteed to be aligned at MEMORY_ALLOCATION_ALIGNMENT bytes.
*/
PVOID PhAllocateFromFreeList(
_Inout_ PPH_FREE_LIST FreeList
)
{
PPH_FREE_LIST_ENTRY entry;
PSLIST_ENTRY listEntry;
listEntry = RtlInterlockedPopEntrySList(&FreeList->ListHead);
if (listEntry)
{
_InterlockedDecrement((PLONG)&FreeList->Count);
entry = CONTAINING_RECORD(listEntry, PH_FREE_LIST_ENTRY, ListEntry);
}
else
{
entry = PhAllocate(FIELD_OFFSET(PH_FREE_LIST_ENTRY, Body) + FreeList->Size);
}
return &entry->Body;
}
/**
* Frees a block of memory to a free list.
*
* \param FreeList A pointer to a free list object.
* \param Memory A pointer to a block of memory.
*/
VOID PhFreeToFreeList(
_Inout_ PPH_FREE_LIST FreeList,
_In_ PVOID Memory
)
{
PPH_FREE_LIST_ENTRY entry;
entry = CONTAINING_RECORD(Memory, PH_FREE_LIST_ENTRY, Body);
// We don't enforce Count <= MaximumCount (that would require locking),
// but we do check it.
if (FreeList->Count < FreeList->MaximumCount)
{
RtlInterlockedPushEntrySList(&FreeList->ListHead, &entry->ListEntry);
_InterlockedIncrement((PLONG)&FreeList->Count);
}
else
{
PhFree(entry);
}
}
/**
* Initializes a callback object.
*
* \param Callback A pointer to a callback object.
*/
VOID PhInitializeCallback(
_Out_ PPH_CALLBACK Callback
)
{
InitializeListHead(&Callback->ListHead);
PhInitializeQueuedLock(&Callback->ListLock);
PhInitializeCondition(&Callback->BusyCondition);
}
/**
* Frees resources used by a callback object.
*
* \param Callback A pointer to a callback object.
*/
VOID PhDeleteCallback(
_Inout_ PPH_CALLBACK Callback
)
{
// Nothing for now
}
/**
* Registers a callback function to be notified.
*
* \param Callback A pointer to a callback object.
* \param Function The callback function.
* \param Context A user-defined value to pass to the callback function.
* \param Registration A variable which receives registration information for the callback. Do not
* modify the contents of this structure and do not free the storage for this structure until you
* have unregistered the callback.
*/
VOID PhRegisterCallback(
_Inout_ PPH_CALLBACK Callback,
_In_ PPH_CALLBACK_FUNCTION Function,
_In_opt_ PVOID Context,
_Out_ PPH_CALLBACK_REGISTRATION Registration
)
{
PhRegisterCallbackEx(
Callback,
Function,
Context,
0,
Registration
);
}
/**
* Registers a callback function to be notified.
*
* \param Callback A pointer to a callback object.
* \param Function The callback function.
* \param Context A user-defined value to pass to the callback function.
* \param Flags A combination of flags controlling the callback. Set this parameter to 0.
* \param Registration A variable which receives registration information for the callback. Do not
* modify the contents of this structure and do not free the storage for this structure until you
* have unregistered the callback.
*/
VOID PhRegisterCallbackEx(
_Inout_ PPH_CALLBACK Callback,
_In_ PPH_CALLBACK_FUNCTION Function,
_In_opt_ PVOID Context,
_In_ USHORT Flags,
_Out_ PPH_CALLBACK_REGISTRATION Registration
)
{
Registration->Function = Function;
Registration->Context = Context;
Registration->Busy = 0;
Registration->Unregistering = FALSE;
Registration->Flags = Flags;
PhAcquireQueuedLockExclusive(&Callback->ListLock);
InsertTailList(&Callback->ListHead, &Registration->ListEntry);
PhReleaseQueuedLockExclusive(&Callback->ListLock);
}
/**
* Unregisters a callback function.
*
* \param Callback A pointer to a callback object.
* \param Registration The structure returned by PhRegisterCallback().
*
* \remarks It is guaranteed that the callback function will not be in execution once this function
* returns. Attempting to unregister a callback function from within the same function will result
* in a deadlock.
*/
VOID PhUnregisterCallback(
_Inout_ PPH_CALLBACK Callback,
_Inout_ PPH_CALLBACK_REGISTRATION Registration
)
{
Registration->Unregistering = TRUE;
PhAcquireQueuedLockExclusive(&Callback->ListLock);
// Wait for the callback to be unbusy.
while (Registration->Busy)
PhWaitForCondition(&Callback->BusyCondition, &Callback->ListLock, NULL);
RemoveEntryList(&Registration->ListEntry);
PhReleaseQueuedLockExclusive(&Callback->ListLock);
}
/**
* Notifies all registered callback functions.
*
* \param Callback A pointer to a callback object.
* \param Parameter A value to pass to all callback functions.
*/
VOID PhInvokeCallback(
_In_ PPH_CALLBACK Callback,
_In_opt_ PVOID Parameter
)
{
PLIST_ENTRY listEntry;
PhAcquireQueuedLockShared(&Callback->ListLock);
listEntry = Callback->ListHead.Flink;
while (listEntry != &Callback->ListHead)
{
PPH_CALLBACK_REGISTRATION registration;
LONG busy;
registration = CONTAINING_RECORD(listEntry, PH_CALLBACK_REGISTRATION, ListEntry);
// Don't bother executing the callback function if it is being unregistered.
if (registration->Unregistering)
continue;
_InterlockedIncrement(&registration->Busy);
// Execute the callback function.
PhReleaseQueuedLockShared(&Callback->ListLock);
registration->Function(
Parameter,
registration->Context
);
PhAcquireQueuedLockShared(&Callback->ListLock);
busy = _InterlockedDecrement(&registration->Busy);
if (registration->Unregistering && busy == 0)
{
// Someone started unregistering while the callback function was executing, and we must
// wake them.
PhPulseAllCondition(&Callback->BusyCondition);
}
listEntry = listEntry->Flink;
}
PhReleaseQueuedLockShared(&Callback->ListLock);
}
/**
* Retrieves a prime number bigger than or equal to the specified number.
*/
ULONG PhGetPrimeNumber(
_In_ ULONG Minimum
)
{
ULONG i, j;
for (i = 0; i < sizeof(PhpPrimeNumbers) / sizeof(ULONG); i++)
{
if (PhpPrimeNumbers[i] >= Minimum)
return PhpPrimeNumbers[i];
}
for (i = Minimum | 1; i < MAXLONG; i += 2)
{
ULONG sqrtI = (ULONG)sqrt(i);
for (j = 3; j <= sqrtI; j += 2)
{
if (i % j == 0)
{
// Not a prime.
goto NextPrime;
}
}
// Success.
return i;
NextPrime:
NOTHING;
}
return Minimum;
}
/**
* Rounds up a number to the next power of two.
*/
ULONG PhRoundUpToPowerOfTwo(
_In_ ULONG Number
)
{
Number--;
Number |= Number >> 1;
Number |= Number >> 2;
Number |= Number >> 4;
Number |= Number >> 8;
Number |= Number >> 16;
Number++;
return Number;
}
/**
* Performs exponentiation.
*/
ULONG PhExponentiate(
_In_ ULONG Base,
_In_ ULONG Exponent
)
{
ULONG result = 1;
while (Exponent)
{
if (Exponent & 1)
result *= Base;
Exponent >>= 1;
Base *= Base;
}
return result;
}
/**
* Performs 64-bit exponentiation.
*/
ULONG64 PhExponentiate64(
_In_ ULONG64 Base,
_In_ ULONG Exponent
)
{
ULONG64 result = 1;
while (Exponent)
{
if (Exponent & 1)
result *= Base;
Exponent >>= 1;
Base *= Base;
}
return result;
}
/**
* Converts a sequence of hexadecimal digits into a byte array.
*
* \param String A string containing hexadecimal digits to convert. The string must have an even
* number of digits, because each pair of hexadecimal digits represents one byte. Example:
* "129a2eff5c0b".
* \param Buffer The output buffer.
*
* \return TRUE if the string was successfully converted, otherwise FALSE.
*/
BOOLEAN PhHexStringToBuffer(
_In_ PPH_STRINGREF String,
_Out_writes_bytes_(String->Length / sizeof(WCHAR) / 2) PUCHAR Buffer
)
{
SIZE_T i;
SIZE_T length;
// The string must have an even length.
if ((String->Length / sizeof(WCHAR)) & 1)
return FALSE;
length = String->Length / sizeof(WCHAR) / 2;
for (i = 0; i < length; i++)
{
Buffer[i] =
(UCHAR)(PhCharToInteger[(UCHAR)String->Buffer[i * 2]] << 4) +
(UCHAR)PhCharToInteger[(UCHAR)String->Buffer[i * 2 + 1]];
}
return TRUE;
}
/**
* Converts a byte array into a sequence of hexadecimal digits.
*
* \param Buffer The input buffer.
* \param Length The number of bytes to convert.
*
* \return A string containing a sequence of hexadecimal digits.
*/
PPH_STRING PhBufferToHexString(
_In_reads_bytes_(Length) PUCHAR Buffer,
_In_ ULONG Length
)
{
return PhBufferToHexStringEx(Buffer, Length, FALSE);
}
/**
* Converts a byte array into a sequence of hexadecimal digits.
*
* \param Buffer The input buffer.
* \param Length The number of bytes to convert.
* \param UpperCase TRUE to use uppercase characters, otherwise FALSE.
*
* \return A string containing a sequence of hexadecimal digits.
*/
PPH_STRING PhBufferToHexStringEx(
_In_reads_bytes_(Length) PUCHAR Buffer,
_In_ ULONG Length,
_In_ BOOLEAN UpperCase
)
{
PCHAR table;
PPH_STRING string;
ULONG i;
if (UpperCase)
table = PhIntegerToCharUpper;
else
table = PhIntegerToChar;
string = PhCreateStringEx(NULL, Length * 2 * sizeof(WCHAR));
for (i = 0; i < Length; i++)
{
string->Buffer[i * 2] = table[Buffer[i] >> 4];
string->Buffer[i * 2 + 1] = table[Buffer[i] & 0xf];
}
return string;
}
/**
* Converts a string to an integer.
*
* \param String The string to process.
* \param Base The base which the string uses to represent the integer. The maximum value is 69.
* \param Integer The resulting integer.
*/
BOOLEAN PhpStringToInteger64(
_In_ PPH_STRINGREF String,
_In_ ULONG Base,
_Out_ PULONG64 Integer
)
{
BOOLEAN valid = TRUE;
ULONG64 result;
SIZE_T length;
SIZE_T i;
length = String->Length / sizeof(WCHAR);
result = 0;
for (i = 0; i < length; i++)
{
ULONG value;
value = PhCharToInteger[(UCHAR)String->Buffer[i]];
if (value < Base)
result = result * Base + value;
else
valid = FALSE;
}
*Integer = result;
return valid;
}
/**
* Converts a string to an integer.
*
* \param String The string to process.
* \param Base The base which the string uses to represent the integer. The maximum value is 69. If
* the parameter is 0, the base is inferred from the string.
* \param Integer The resulting integer.
*
* \remarks If \a Base is 0, the following prefixes may be used to indicate bases:
*
* \li \c 0x Base 16.
* \li \c 0o Base 8.
* \li \c 0b Base 2.
* \li \c 0t Base 3.
* \li \c 0q Base 4.
* \li \c 0w Base 12.
* \li \c 0r Base 32.
*
* If there is no recognized prefix, base 10 is used.
*/
BOOLEAN PhStringToInteger64(
_In_ PPH_STRINGREF String,
_In_opt_ ULONG Base,
_Out_opt_ PLONG64 Integer
)
{
BOOLEAN valid;
ULONG64 result;
PH_STRINGREF string;
BOOLEAN negative;
ULONG base;
if (Base > 69)
return FALSE;
string = *String;
negative = FALSE;
if (string.Length != 0 && (string.Buffer[0] == '-' || string.Buffer[0] == '+'))
{
if (string.Buffer[0] == '-')
negative = TRUE;
PhSkipStringRef(&string, sizeof(WCHAR));
}
// If the caller specified a base, don't perform any additional processing.
if (Base)
{
base = Base;
}
else
{
base = 10;
if (string.Length >= 2 * sizeof(WCHAR) && string.Buffer[0] == '0')
{
switch (string.Buffer[1])
{
case 'x':
case 'X':
base = 16;
break;
case 'o':
case 'O':
base = 8;
break;
case 'b':
case 'B':
base = 2;
break;
case 't': // ternary
case 'T':
base = 3;
break;
case 'q': // quaternary
case 'Q':
base = 4;
break;
case 'w': // base 12
case 'W':
base = 12;
break;
case 'r': // base 32
case 'R':
base = 32;
break;
}
if (base != 10)
PhSkipStringRef(&string, 2 * sizeof(WCHAR));
}
}
valid = PhpStringToInteger64(&string, base, &result);
if (Integer)
*Integer = negative ? -(LONG64)result : result;
return valid;
}
BOOLEAN PhpStringToDouble(
_In_ PPH_STRINGREF String,
_In_ ULONG Base,
_Out_ DOUBLE *Double
)
{
BOOLEAN valid = TRUE;
BOOLEAN dotSeen = FALSE;
DOUBLE result;
DOUBLE fraction;
SIZE_T length;
SIZE_T i;
length = String->Length / sizeof(WCHAR);
result = 0;
fraction = 1;
for (i = 0; i < length; i++)
{
if (String->Buffer[i] == '.')
{
if (!dotSeen)
dotSeen = TRUE;
else
valid = FALSE;
}
else
{
ULONG value;
value = PhCharToInteger[(UCHAR)String->Buffer[i]];
if (value < Base)
{
if (!dotSeen)
{
result = result * Base + value;
}
else
{
fraction /= Base;
result = result + value * fraction;
}
}
else
{
valid = FALSE;
}
}
}
*Double = result;
return valid;
}
/**
* Converts a string to a double-precision floating point value.
*
* \param String The string to process.
* \param Base Reserved.
* \param Double The resulting double value.
*/
BOOLEAN PhStringToDouble(
_In_ PPH_STRINGREF String,
_Reserved_ ULONG Base,
_Out_opt_ DOUBLE *Double
)
{
BOOLEAN valid;
DOUBLE result;
PH_STRINGREF string;
BOOLEAN negative;
string = *String;
negative = FALSE;
if (string.Length != 0 && (string.Buffer[0] == '-' || string.Buffer[0] == '+'))
{
if (string.Buffer[0] == '-')
negative = TRUE;
PhSkipStringRef(&string, sizeof(WCHAR));
}
valid = PhpStringToDouble(&string, 10, &result);
if (Double)
*Double = negative ? -result : result;
return valid;
}
/**
* Converts an integer to a string.
*
* \param Integer The integer to process.
* \param Base The base which the integer is represented with. The maximum value is 69. The base
* cannot be 1. If the parameter is 0, the base used is 10.
* \param Signed TRUE if \a Integer is a signed value, otherwise FALSE.
*
* \return The resulting string, or NULL if an error occurred.
*/
PPH_STRING PhIntegerToString64(
_In_ LONG64 Integer,
_In_opt_ ULONG Base,
_In_ BOOLEAN Signed
)
{
PH_FORMAT format;
if (Base == 1 || Base > 69)
return NULL;
if (Signed)
PhInitFormatI64D(&format, Integer);
else
PhInitFormatI64U(&format, Integer);
if (Base != 0)
{
format.Type |= FormatUseRadix;
format.Radix = (UCHAR)Base;
}
return PhFormat(&format, 1, 0);
}
VOID PhPrintTimeSpan(
_Out_writes_(PH_TIMESPAN_STR_LEN_1) PWSTR Destination,
_In_ ULONG64 Ticks,
_In_opt_ ULONG Mode
)
{
switch (Mode)
{
case PH_TIMESPAN_HMSM:
_snwprintf(
Destination,
PH_TIMESPAN_STR_LEN,
L"%02I64u:%02I64u:%02I64u.%03I64u",
PH_TICKS_PARTIAL_HOURS(Ticks),
PH_TICKS_PARTIAL_MIN(Ticks),
PH_TICKS_PARTIAL_SEC(Ticks),
PH_TICKS_PARTIAL_MS(Ticks)
);
break;
case PH_TIMESPAN_DHMS:
_snwprintf(
Destination,
PH_TIMESPAN_STR_LEN,
L"%I64u:%02I64u:%02I64u:%02I64u",
PH_TICKS_PARTIAL_DAYS(Ticks),
PH_TICKS_PARTIAL_HOURS(Ticks),
PH_TICKS_PARTIAL_MIN(Ticks),
PH_TICKS_PARTIAL_SEC(Ticks)
);
break;
default:
_snwprintf(
Destination,
PH_TIMESPAN_STR_LEN,
L"%02I64u:%02I64u:%02I64u",
PH_TICKS_PARTIAL_HOURS(Ticks),
PH_TICKS_PARTIAL_MIN(Ticks),
PH_TICKS_PARTIAL_SEC(Ticks)
);
break;
}
}
/**
* Fills a memory block with a ULONG pattern.
*
* \param Memory The memory block. The block must be 4 byte aligned.
* \param Value The ULONG pattern.
* \param Count The number of elements.
*/
VOID PhFillMemoryUlong(
_Inout_updates_(Count) _Needs_align_(4) PULONG Memory,
_In_ ULONG Value,
_In_ SIZE_T Count
)
{
__m128i pattern;
SIZE_T count;
if (PhpVectorLevel < PH_VECTOR_LEVEL_SSE2)
{
if (Count != 0)
{
do
{
*Memory++ = Value;
} while (--Count != 0);
}
return;
}
if ((ULONG_PTR)Memory & 0xf)
{
switch ((ULONG_PTR)Memory & 0xf)
{
case 0x4:
if (Count >= 1)
{
*Memory++ = Value;
Count--;
}
__fallthrough;
case 0x8:
if (Count >= 1)
{
*Memory++ = Value;
Count--;
}
__fallthrough;
case 0xc:
if (Count >= 1)
{
*Memory++ = Value;
Count--;
}
break;
}
}
pattern = _mm_set1_epi32(Value);
count = Count / 4;
if (count != 0)
{
do
{
_mm_store_si128((__m128i *)Memory, pattern);
Memory += 4;
} while (--count != 0);
}
switch (Count & 0x3)
{
case 0x3:
*Memory++ = Value;
__fallthrough;
case 0x2:
*Memory++ = Value;
__fallthrough;
case 0x1:
*Memory++ = Value;
break;
}
}
/**
* Divides an array of numbers by a number.
*
* \param A The destination array, divided by \a B.
* \param B The number.
* \param Count The number of elements.
*/
VOID PhDivideSinglesBySingle(
_Inout_updates_(Count) PFLOAT A,
_In_ FLOAT B,
_In_ SIZE_T Count
)
{
PFLOAT endA;
__m128 b;
if (PhpVectorLevel < PH_VECTOR_LEVEL_SSE2)
{
while (Count--)
*A++ /= B;
return;
}
if ((ULONG_PTR)A & 0xf)
{
switch ((ULONG_PTR)A & 0xf)
{
case 0x4:
if (Count >= 1)
{
*A++ /= B;
Count--;
}
__fallthrough;
case 0x8:
if (Count >= 1)
{
*A++ /= B;
Count--;
}
__fallthrough;
case 0xc:
if (Count >= 1)
{
*A++ /= B;
Count--;
}
else
{
return; // essential; A may not be aligned properly
}
break;
}
}
endA = (PFLOAT)((ULONG_PTR)(A + Count) & ~0xf);
b = _mm_load1_ps(&B);
while (A != endA)
{
__m128 a;
a = _mm_load_ps(A);
a = _mm_div_ps(a, b);
_mm_store_ps(A, a);
A += 4;
}
switch (Count & 0x3)
{
case 0x3:
*A++ /= B;
__fallthrough;
case 0x2:
*A++ /= B;
__fallthrough;
case 0x1:
*A++ /= B;
break;
}
}