2018-04-21 06:56:12 +00:00
|
|
|
|
/*
|
|
|
|
|
* PROJECT: ReactOS kernel-mode tests
|
2019-08-28 20:42:52 +00:00
|
|
|
|
* LICENSE: GPL-2.0-or-later (https://spdx.org/licenses/GPL-2.0-or-later)
|
2018-04-21 06:56:12 +00:00
|
|
|
|
* PURPOSE: Test for ntstrsafe.h functions
|
|
|
|
|
* COPYRIGHT: Copyright 2018 Hern<EFBFBD>n Di Pietro <hernan.di.pietro@gmail.com>
|
2019-08-28 20:42:52 +00:00
|
|
|
|
* Copyright 2019 Colin Finck <colin@reactos.org>
|
2018-04-21 06:56:12 +00:00
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#define KMT_EMULATE_KERNEL
|
|
|
|
|
#include <kmt_test.h>
|
|
|
|
|
#include <ntstrsafe.h>
|
|
|
|
|
#include <ntdef.h>
|
|
|
|
|
#include <ndk/rtlfuncs.h>
|
|
|
|
|
|
|
|
|
|
#define TESTAPI static void
|
|
|
|
|
|
2019-08-28 20:42:52 +00:00
|
|
|
|
static const WCHAR FormatStringInts[] = L"%d %d %d";
|
|
|
|
|
static const WCHAR FormatStringIntsResult[] = L"1 2 3";
|
|
|
|
|
static const WCHAR FormatStringStrs[] = L"%s %s %s";
|
|
|
|
|
|
|
|
|
|
|
2018-04-21 06:56:12 +00:00
|
|
|
|
TESTAPI
|
|
|
|
|
Test_RtlUnicodeStringPrintf()
|
|
|
|
|
{
|
2019-08-28 20:42:52 +00:00
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
PWSTR pBuffer = NULL;
|
2019-05-30 15:12:28 +00:00
|
|
|
|
USHORT BufferSize;
|
2019-08-28 20:42:52 +00:00
|
|
|
|
size_t EqualBytes;
|
2018-04-21 06:56:12 +00:00
|
|
|
|
UNICODE_STRING UsString;
|
|
|
|
|
|
2019-08-28 20:42:52 +00:00
|
|
|
|
KmtStartSeh();
|
2018-04-21 06:56:12 +00:00
|
|
|
|
|
|
|
|
|
/* STATUS_SUCCESS test */
|
2019-08-28 20:42:52 +00:00
|
|
|
|
BufferSize = 6 * sizeof(WCHAR);
|
|
|
|
|
pBuffer = KmtAllocateGuarded(BufferSize);
|
|
|
|
|
if (!pBuffer)
|
|
|
|
|
goto Cleanup;
|
|
|
|
|
|
|
|
|
|
RtlFillMemory(pBuffer, BufferSize, 0xAA);
|
|
|
|
|
RtlInitEmptyUnicodeString(&UsString, pBuffer, BufferSize);
|
|
|
|
|
|
|
|
|
|
Status = RtlUnicodeStringPrintf(&UsString, FormatStringInts, 1, 2, 3);
|
|
|
|
|
EqualBytes = RtlCompareMemory(UsString.Buffer, FormatStringIntsResult, sizeof(FormatStringIntsResult));
|
|
|
|
|
ok_eq_hex(Status, STATUS_SUCCESS);
|
|
|
|
|
ok_eq_size(EqualBytes, sizeof(FormatStringIntsResult));
|
|
|
|
|
ok_eq_uint(UsString.Length, sizeof(FormatStringIntsResult) - sizeof(WCHAR));
|
2021-09-13 01:33:14 +00:00
|
|
|
|
ok_eq_uint(UsString.MaximumLength, BufferSize);
|
2019-08-28 20:42:52 +00:00
|
|
|
|
|
|
|
|
|
KmtFreeGuarded(pBuffer);
|
|
|
|
|
pBuffer = NULL;
|
2018-04-21 06:56:12 +00:00
|
|
|
|
|
|
|
|
|
/* STATUS_BUFFER_OVERFLOW tests */
|
2019-08-28 20:42:52 +00:00
|
|
|
|
BufferSize = 2 * sizeof(WCHAR);
|
|
|
|
|
pBuffer = KmtAllocateGuarded(BufferSize);
|
|
|
|
|
if (!pBuffer)
|
|
|
|
|
goto Cleanup;
|
2018-04-21 06:56:12 +00:00
|
|
|
|
|
2019-08-28 20:42:52 +00:00
|
|
|
|
RtlInitEmptyUnicodeString(&UsString, pBuffer, BufferSize);
|
|
|
|
|
|
|
|
|
|
Status = RtlUnicodeStringPrintf(&UsString, FormatStringStrs, L"AAA", L"BBB", L"CCC");
|
|
|
|
|
EqualBytes = RtlCompareMemory(UsString.Buffer, L"AA", BufferSize);
|
|
|
|
|
ok_eq_hex(Status, STATUS_BUFFER_OVERFLOW);
|
|
|
|
|
ok_eq_size(EqualBytes, BufferSize);
|
2018-04-21 06:56:12 +00:00
|
|
|
|
ok_eq_uint(UsString.Length, UsString.MaximumLength);
|
|
|
|
|
|
2019-08-28 20:42:52 +00:00
|
|
|
|
KmtFreeGuarded(pBuffer);
|
|
|
|
|
pBuffer = NULL;
|
|
|
|
|
|
2018-04-21 06:56:12 +00:00
|
|
|
|
|
2019-08-28 20:42:52 +00:00
|
|
|
|
BufferSize = 7 * sizeof(WCHAR);
|
|
|
|
|
pBuffer = KmtAllocateGuarded(BufferSize);
|
|
|
|
|
if (!pBuffer)
|
|
|
|
|
goto Cleanup;
|
|
|
|
|
|
|
|
|
|
RtlInitEmptyUnicodeString(&UsString, pBuffer, BufferSize);
|
|
|
|
|
|
|
|
|
|
Status = RtlUnicodeStringPrintf(&UsString, FormatStringStrs, L"0123", L"4567", L"89AB");
|
|
|
|
|
EqualBytes = RtlCompareMemory(UsString.Buffer, L"0123 45", BufferSize);
|
|
|
|
|
ok_eq_hex(Status, STATUS_BUFFER_OVERFLOW);
|
|
|
|
|
ok_eq_size(EqualBytes, BufferSize);
|
2018-04-21 06:56:12 +00:00
|
|
|
|
ok_eq_uint(UsString.Length, UsString.MaximumLength);
|
2019-08-28 20:42:52 +00:00
|
|
|
|
|
|
|
|
|
KmtFreeGuarded(pBuffer);
|
|
|
|
|
pBuffer = NULL;
|
|
|
|
|
|
|
|
|
|
// Note: RtlUnicodeStringPrintf returns STATUS_BUFFER_OVERFLOW here while RtlUnicodeStringPrintfEx returns STATUS_INVALID_PARAMETER!
|
|
|
|
|
// Documented on MSDN and verified with the Win10 version of ntstrsafe.h
|
|
|
|
|
RtlInitEmptyUnicodeString(&UsString, NULL, 0);
|
|
|
|
|
Status = RtlUnicodeStringPrintf(&UsString, FormatStringStrs, L"AAA", L"BBB", L"CCC");
|
|
|
|
|
ok_eq_hex(Status, STATUS_BUFFER_OVERFLOW);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Cleanup:
|
|
|
|
|
if (pBuffer)
|
|
|
|
|
KmtFreeGuarded(pBuffer);
|
|
|
|
|
|
|
|
|
|
// None of these functions should have crashed.
|
|
|
|
|
KmtEndSeh(STATUS_SUCCESS);
|
2018-04-21 06:56:12 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TESTAPI
|
|
|
|
|
Test_RtlUnicodeStringPrintfEx()
|
|
|
|
|
{
|
2019-08-28 20:42:52 +00:00
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
PWSTR pBuffer = NULL;
|
2019-05-30 15:12:28 +00:00
|
|
|
|
USHORT BufferSize;
|
2019-08-28 20:42:52 +00:00
|
|
|
|
size_t EqualBytes;
|
|
|
|
|
UNICODE_STRING RemString;
|
|
|
|
|
UNICODE_STRING UsString;
|
|
|
|
|
WCHAR FillResult[10];
|
2018-04-21 06:56:12 +00:00
|
|
|
|
|
2019-08-28 20:42:52 +00:00
|
|
|
|
RtlFillMemory(FillResult, sizeof(FillResult), 0xAA);
|
2018-04-21 06:56:12 +00:00
|
|
|
|
|
2019-08-28 20:42:52 +00:00
|
|
|
|
KmtStartSeh();
|
2018-04-21 06:56:12 +00:00
|
|
|
|
|
2019-08-28 20:42:52 +00:00
|
|
|
|
/* STATUS_SUCCESS test, fill behind flag: low-byte as fill character */
|
|
|
|
|
BufferSize = sizeof(FormatStringIntsResult) - sizeof(UNICODE_NULL) + sizeof(FillResult);
|
|
|
|
|
pBuffer = KmtAllocateGuarded(BufferSize);
|
|
|
|
|
if (!pBuffer)
|
|
|
|
|
goto Cleanup;
|
|
|
|
|
|
|
|
|
|
RtlInitEmptyUnicodeString(&UsString, pBuffer, BufferSize);
|
|
|
|
|
RtlInitEmptyUnicodeString(&RemString, NULL, 0);
|
|
|
|
|
|
|
|
|
|
Status = RtlUnicodeStringPrintfEx(&UsString, &RemString, STRSAFE_FILL_BEHIND | 0xAA, FormatStringInts, 1, 2, 3);
|
|
|
|
|
EqualBytes = RtlCompareMemory(UsString.Buffer, FormatStringIntsResult, sizeof(FormatStringIntsResult) - sizeof(WCHAR));
|
|
|
|
|
ok_eq_hex(Status, STATUS_SUCCESS);
|
|
|
|
|
ok_eq_size(EqualBytes, sizeof(FormatStringIntsResult) - sizeof(WCHAR));
|
|
|
|
|
ok_eq_uint(UsString.Length, sizeof(FormatStringIntsResult) - sizeof(WCHAR));
|
|
|
|
|
ok_eq_uint(UsString.MaximumLength, BufferSize);
|
|
|
|
|
|
|
|
|
|
ok_eq_pointer(RemString.Buffer, &UsString.Buffer[UsString.Length / sizeof(WCHAR)]);
|
|
|
|
|
ok_eq_uint(RemString.Length, 0);
|
|
|
|
|
ok_eq_uint(RemString.MaximumLength, UsString.MaximumLength - UsString.Length);
|
2018-04-21 06:56:12 +00:00
|
|
|
|
|
2019-08-28 20:42:52 +00:00
|
|
|
|
EqualBytes = RtlCompareMemory(RemString.Buffer, FillResult, RemString.MaximumLength);
|
|
|
|
|
ok_eq_size(EqualBytes, sizeof(FillResult));
|
|
|
|
|
|
|
|
|
|
KmtFreeGuarded(pBuffer);
|
|
|
|
|
pBuffer = NULL;
|
2018-04-21 06:56:12 +00:00
|
|
|
|
|
|
|
|
|
|
2019-08-28 20:42:52 +00:00
|
|
|
|
/* STATUS_BUFFER_OVERFLOW test */
|
|
|
|
|
BufferSize = 8 * sizeof(WCHAR);
|
|
|
|
|
pBuffer = KmtAllocateGuarded(BufferSize);
|
|
|
|
|
if (!pBuffer)
|
|
|
|
|
goto Cleanup;
|
|
|
|
|
|
|
|
|
|
RtlInitEmptyUnicodeString(&UsString, pBuffer, BufferSize);
|
|
|
|
|
RtlInitEmptyUnicodeString(&RemString, NULL, 0);
|
|
|
|
|
|
|
|
|
|
Status = RtlUnicodeStringPrintfEx(&UsString, &RemString, 0, FormatStringStrs, L"AAA", L"BBB", L"CCC");
|
|
|
|
|
EqualBytes = RtlCompareMemory(UsString.Buffer, L"AAA BBB ", UsString.Length);
|
|
|
|
|
ok_eq_hex(Status, STATUS_BUFFER_OVERFLOW);
|
|
|
|
|
ok_eq_size(EqualBytes, UsString.Length);
|
|
|
|
|
ok_eq_uint(UsString.Length, UsString.MaximumLength);
|
2018-04-21 06:56:12 +00:00
|
|
|
|
|
2019-08-28 20:42:52 +00:00
|
|
|
|
ok_eq_pointer(RemString.Buffer, &UsString.Buffer[UsString.Length / sizeof(WCHAR)]);
|
2018-04-21 06:56:12 +00:00
|
|
|
|
ok_eq_uint(RemString.Length, 0);
|
2019-08-28 20:42:52 +00:00
|
|
|
|
ok_eq_uint(RemString.MaximumLength, 0);
|
2018-04-21 06:56:12 +00:00
|
|
|
|
|
2019-08-28 20:42:52 +00:00
|
|
|
|
KmtFreeGuarded(pBuffer);
|
|
|
|
|
pBuffer = NULL;
|
2018-04-21 06:56:12 +00:00
|
|
|
|
|
|
|
|
|
|
2019-08-28 20:42:52 +00:00
|
|
|
|
// Note: RtlUnicodeStringPrintf returns STATUS_BUFFER_OVERFLOW here while RtlUnicodeStringPrintfEx returns STATUS_INVALID_PARAMETER!
|
|
|
|
|
// Documented on MSDN and verified with the Win10 version of ntstrsafe.h
|
|
|
|
|
RtlInitEmptyUnicodeString(&UsString, NULL, 0);
|
|
|
|
|
Status = RtlUnicodeStringPrintfEx(&UsString, NULL, 0, FormatStringStrs, L"AAA", L"BBB", L"CCC");
|
|
|
|
|
ok_eq_hex(Status, STATUS_INVALID_PARAMETER);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Cleanup:
|
|
|
|
|
if (pBuffer)
|
|
|
|
|
KmtFreeGuarded(pBuffer);
|
|
|
|
|
|
|
|
|
|
// None of these functions should have crashed.
|
|
|
|
|
KmtEndSeh(STATUS_SUCCESS);
|
2018-04-21 06:56:12 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
START_TEST(RtlStrSafe)
|
|
|
|
|
{
|
|
|
|
|
Test_RtlUnicodeStringPrintf();
|
|
|
|
|
Test_RtlUnicodeStringPrintfEx();
|
|
|
|
|
}
|