mirror of
https://github.com/reactos/reactos.git
synced 2025-01-04 21:38:43 +00:00
545 lines
16 KiB
C
545 lines
16 KiB
C
/*
|
|
* PROJECT: ReactOS API Tests
|
|
* LICENSE: GPL-2.0+ (https://spdx.org/licenses/GPL-2.0+)
|
|
* PURPOSE: Tests for System Firmware functions
|
|
* COPYRIGHT: Copyright 2018 Stanislav Motylkov
|
|
* Copyright 2018 Mark Jansen
|
|
*/
|
|
|
|
#include "precomp.h"
|
|
|
|
#include <pseh/pseh2.h>
|
|
|
|
static UINT (WINAPI * pEnumSystemFirmwareTables)(DWORD, PVOID, DWORD);
|
|
static UINT (WINAPI * pGetSystemFirmwareTable)(DWORD, DWORD, PVOID, DWORD);
|
|
|
|
typedef struct ENTRY
|
|
{
|
|
DWORD Signature;
|
|
DWORD ErrInsuff;
|
|
DWORD ErrSuccess;
|
|
} ENTRY;
|
|
|
|
static UINT
|
|
CallNt(IN DWORD FirmwareTableProviderSignature,
|
|
IN DWORD FirmwareTableID,
|
|
OUT PVOID pFirmwareTableBuffer,
|
|
IN DWORD BufferSize,
|
|
IN SYSTEM_FIRMWARE_TABLE_ACTION Action)
|
|
{
|
|
SYSTEM_FIRMWARE_TABLE_INFORMATION* SysFirmwareInfo;
|
|
ULONG Result = 0, ReturnedSize;
|
|
ULONG TotalSize = BufferSize + sizeof(SYSTEM_FIRMWARE_TABLE_INFORMATION);
|
|
NTSTATUS Status;
|
|
|
|
SysFirmwareInfo = RtlAllocateHeap(RtlGetProcessHeap(), HEAP_ZERO_MEMORY, TotalSize);
|
|
if (!SysFirmwareInfo)
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return 0;
|
|
}
|
|
_SEH2_TRY
|
|
{
|
|
SysFirmwareInfo->ProviderSignature = FirmwareTableProviderSignature;
|
|
SysFirmwareInfo->TableID = FirmwareTableID;
|
|
SysFirmwareInfo->Action = Action;
|
|
SysFirmwareInfo->TableBufferLength = BufferSize;
|
|
|
|
Status = NtQuerySystemInformation(SystemFirmwareTableInformation, SysFirmwareInfo, TotalSize, &ReturnedSize);
|
|
|
|
if (NT_SUCCESS(Status) || Status == STATUS_BUFFER_TOO_SMALL)
|
|
Result = SysFirmwareInfo->TableBufferLength;
|
|
|
|
if (NT_SUCCESS(Status) && pFirmwareTableBuffer)
|
|
{
|
|
memcpy(pFirmwareTableBuffer, SysFirmwareInfo->TableBuffer, SysFirmwareInfo->TableBufferLength);
|
|
}
|
|
}
|
|
_SEH2_FINALLY
|
|
{
|
|
RtlFreeHeap(RtlGetProcessHeap(), 0, SysFirmwareInfo);
|
|
}
|
|
_SEH2_END;
|
|
|
|
SetLastError(RtlNtStatusToDosError(Status));
|
|
return Result;
|
|
}
|
|
|
|
UINT
|
|
WINAPI
|
|
fEnumSystemFirmwareTables(IN DWORD FirmwareTableProviderSignature,
|
|
OUT PVOID pFirmwareTableBuffer,
|
|
IN DWORD BufferSize)
|
|
{
|
|
return CallNt(FirmwareTableProviderSignature, 0, pFirmwareTableBuffer, BufferSize, SystemFirmwareTable_Enumerate);
|
|
}
|
|
|
|
UINT
|
|
WINAPI
|
|
fGetSystemFirmwareTable(IN DWORD FirmwareTableProviderSignature,
|
|
IN DWORD FirmwareTableID,
|
|
OUT PVOID pFirmwareTableBuffer,
|
|
IN DWORD BufferSize)
|
|
{
|
|
return CallNt(FirmwareTableProviderSignature, FirmwareTableID, pFirmwareTableBuffer, BufferSize, SystemFirmwareTable_Get);
|
|
}
|
|
|
|
static
|
|
VOID
|
|
test_EnumBuffer(
|
|
DWORD Signature,
|
|
PVOID Buffer,
|
|
DWORD dwSize,
|
|
UINT * pTableCount,
|
|
DWORD * pFirstTableID,
|
|
DWORD ErrInsuff,
|
|
DWORD ErrSuccess
|
|
)
|
|
{
|
|
DWORD dwError;
|
|
DWORD dwBufferSize;
|
|
DWORD dwException;
|
|
UINT uResultSize;
|
|
|
|
dwException = Buffer && IsBadWritePtr(Buffer, dwSize) ? STATUS_ACCESS_VIOLATION : STATUS_SUCCESS;
|
|
|
|
// Test size = 0
|
|
if (Buffer && dwException == STATUS_SUCCESS)
|
|
{
|
|
FillMemory(Buffer, dwSize, 0xFF);
|
|
}
|
|
SetLastError(0xbeeffeed);
|
|
dwError = GetLastError();
|
|
dwBufferSize = 0;
|
|
uResultSize = 0;
|
|
StartSeh()
|
|
uResultSize = pEnumSystemFirmwareTables(Signature, Buffer, dwBufferSize);
|
|
dwError = GetLastError();
|
|
EndSeh(STATUS_SUCCESS);
|
|
|
|
if (uResultSize > 0)
|
|
{
|
|
ok(dwError == ErrInsuff,
|
|
"GetLastError() returned %ld, expected %ld\n",
|
|
dwError, ErrInsuff);
|
|
}
|
|
else
|
|
{
|
|
ok(dwError == ErrSuccess,
|
|
"GetLastError() returned %ld, expected %ld\n",
|
|
dwError, ErrSuccess);
|
|
}
|
|
if (ErrSuccess == ERROR_SUCCESS)
|
|
{
|
|
ok(uResultSize % sizeof(DWORD) == 0,
|
|
"uResultSize is %u, expected %% sizeof(DWORD)\n",
|
|
uResultSize);
|
|
}
|
|
else
|
|
{
|
|
ok(uResultSize == 0,
|
|
"uResultSize is %u, expected == 0\n",
|
|
uResultSize);
|
|
}
|
|
if (Buffer && dwException == STATUS_SUCCESS)
|
|
{
|
|
ok(*(BYTE *)Buffer == 0xFF,
|
|
"Buffer should be clean at offset 0, got %x\n",
|
|
*(BYTE *)Buffer);
|
|
}
|
|
|
|
// Test size = 2
|
|
if (Buffer && dwException == STATUS_SUCCESS)
|
|
{
|
|
FillMemory(Buffer, dwSize, 0xFF);
|
|
}
|
|
SetLastError(0xbeeffeed);
|
|
dwError = GetLastError();
|
|
dwBufferSize = 2;
|
|
uResultSize = 0;
|
|
StartSeh()
|
|
uResultSize = pEnumSystemFirmwareTables(Signature, Buffer, dwBufferSize);
|
|
dwError = GetLastError();
|
|
EndSeh(STATUS_SUCCESS);
|
|
|
|
if (uResultSize > 0)
|
|
{
|
|
ok(dwError == ErrInsuff,
|
|
"GetLastError() returned %ld, expected %ld\n",
|
|
dwError, ErrInsuff);
|
|
}
|
|
else
|
|
{
|
|
ok(dwError == ErrSuccess,
|
|
"GetLastError() returned %ld, expected %ld\n",
|
|
dwError, ErrSuccess);
|
|
}
|
|
if (ErrSuccess == ERROR_SUCCESS)
|
|
{
|
|
ok(uResultSize % sizeof(DWORD) == 0,
|
|
"uResultSize is %u, expected %% sizeof(DWORD)\n",
|
|
uResultSize);
|
|
}
|
|
else
|
|
{
|
|
ok(uResultSize == 0,
|
|
"uResultSize is %u, expected == 0\n",
|
|
uResultSize);
|
|
}
|
|
if (Buffer && dwException == STATUS_SUCCESS)
|
|
{
|
|
ok(*(WORD *)Buffer == 0xFFFF,
|
|
"Buffer should be clean at offset 0, got %x\n",
|
|
*(WORD *)Buffer);
|
|
}
|
|
|
|
// Test full size
|
|
if (Buffer && dwException == STATUS_SUCCESS)
|
|
{
|
|
FillMemory(Buffer, dwSize, 0xFF);
|
|
}
|
|
if (uResultSize > 0)
|
|
{
|
|
SetLastError(0xbeeffeed);
|
|
dwError = GetLastError();
|
|
dwBufferSize = uResultSize;
|
|
uResultSize = 0;
|
|
StartSeh()
|
|
uResultSize = pEnumSystemFirmwareTables(Signature, Buffer, dwBufferSize);
|
|
dwError = GetLastError();
|
|
EndSeh(ErrSuccess == ERROR_SUCCESS ? dwException : STATUS_SUCCESS);
|
|
// Windows 7: does not throw exception here
|
|
|
|
if (dwException == STATUS_SUCCESS || ErrSuccess == ERROR_INVALID_FUNCTION)
|
|
{
|
|
ok(dwError == ErrSuccess,
|
|
"GetLastError() returned %ld, expected %ld\n",
|
|
dwError, ErrSuccess);
|
|
if (ErrSuccess == ERROR_SUCCESS)
|
|
{
|
|
ok(uResultSize == dwBufferSize,
|
|
"uResultSize is not equal dwBufferSize, expected %ld\n",
|
|
dwBufferSize);
|
|
}
|
|
else
|
|
{
|
|
ok(uResultSize == 0,
|
|
"uResultSize is %u, expected == 0\n",
|
|
uResultSize);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Windows 7: returns ERROR_NOACCESS here
|
|
ok(dwError == 0xbeeffeed,
|
|
"GetLastError() returned %ld, expected %u\n",
|
|
dwError, 0xbeeffeed);
|
|
// Windows 7: returns correct size here
|
|
ok(uResultSize == 0,
|
|
"uResultSize is %u, expected == 0\n",
|
|
uResultSize);
|
|
}
|
|
}
|
|
|
|
if (pTableCount && pFirstTableID)
|
|
{
|
|
if (uResultSize > 0)
|
|
{
|
|
if (Signature == 'RSMB')
|
|
{
|
|
// Raw SMBIOS have only one table with ID 0
|
|
ok(*(DWORD *)Buffer == 0,
|
|
"Buffer should be filled at offset 0, got %lx\n",
|
|
*(DWORD *)Buffer);
|
|
}
|
|
else
|
|
{
|
|
// In other cases ID can be different
|
|
if (ErrSuccess == ERROR_SUCCESS)
|
|
{
|
|
ok(*(DWORD *)Buffer != 0xFFFFFFFF,
|
|
"Buffer should be filled at offset 0\n");
|
|
}
|
|
else
|
|
{
|
|
ok(*(DWORD *)Buffer == 0xFFFFFFFF,
|
|
"Buffer should be clean at offset 0\n");
|
|
}
|
|
}
|
|
}
|
|
*pTableCount = uResultSize / sizeof(DWORD);
|
|
*pFirstTableID = *(DWORD *)Buffer;
|
|
}
|
|
}
|
|
|
|
static
|
|
VOID
|
|
test_GetBuffer(
|
|
DWORD Signature,
|
|
DWORD TableID,
|
|
PVOID Buffer,
|
|
DWORD dwSize,
|
|
BOOL TestFakeID,
|
|
DWORD ErrInsuff,
|
|
DWORD ErrSuccess
|
|
)
|
|
{
|
|
DWORD dwError;
|
|
DWORD dwBufferSize;
|
|
DWORD dwException;
|
|
DWORD dwErrCase;
|
|
UINT uResultSize;
|
|
|
|
dwException = Buffer && IsBadWritePtr(Buffer, dwSize) ? STATUS_ACCESS_VIOLATION : STATUS_SUCCESS;
|
|
switch (Signature)
|
|
{
|
|
case 'ACPI':
|
|
{
|
|
dwErrCase = ERROR_NOT_FOUND;
|
|
break;
|
|
}
|
|
case 'FIRM':
|
|
{
|
|
dwErrCase = ERROR_INVALID_PARAMETER;
|
|
break;
|
|
}
|
|
default:
|
|
{
|
|
dwErrCase = ErrInsuff;
|
|
}
|
|
}
|
|
|
|
// Test size = 0
|
|
if (Buffer && dwException == STATUS_SUCCESS)
|
|
{
|
|
FillMemory(Buffer, dwSize, 0xFF);
|
|
}
|
|
SetLastError(0xbeeffeed);
|
|
dwError = GetLastError();
|
|
dwBufferSize = 0;
|
|
uResultSize = 0;
|
|
StartSeh()
|
|
uResultSize = pGetSystemFirmwareTable(Signature, TableID, Buffer, dwBufferSize);
|
|
dwError = GetLastError();
|
|
EndSeh(STATUS_SUCCESS);
|
|
|
|
ok(dwError == (TestFakeID ? dwErrCase : ErrInsuff),
|
|
"GetLastError() returned %ld, expected %ld\n",
|
|
dwError, (TestFakeID ? dwErrCase : ErrInsuff));
|
|
if (ErrSuccess == ERROR_SUCCESS && (!TestFakeID || dwErrCase == ErrInsuff))
|
|
{
|
|
ok(uResultSize > 0,
|
|
"uResultSize is %u, expected > 0\n",
|
|
uResultSize);
|
|
}
|
|
else
|
|
{
|
|
ok(uResultSize == 0,
|
|
"uResultSize is %u, expected == 0\n",
|
|
uResultSize);
|
|
}
|
|
if (Buffer && dwException == STATUS_SUCCESS)
|
|
{
|
|
ok(*(BYTE *)Buffer == 0xFF,
|
|
"Buffer should be clean at offset 0, got %x\n",
|
|
*(BYTE *)Buffer);
|
|
}
|
|
|
|
// Test size = 2
|
|
if (Buffer && dwException == STATUS_SUCCESS)
|
|
{
|
|
FillMemory(Buffer, dwSize, 0xFF);
|
|
}
|
|
SetLastError(0xbeeffeed);
|
|
dwError = GetLastError();
|
|
dwBufferSize = 2;
|
|
uResultSize = 0;
|
|
StartSeh()
|
|
uResultSize = pGetSystemFirmwareTable(Signature, TableID, Buffer, dwBufferSize);
|
|
dwError = GetLastError();
|
|
EndSeh(STATUS_SUCCESS);
|
|
|
|
ok(dwError == (TestFakeID ? dwErrCase : ErrInsuff),
|
|
"GetLastError() returned %ld, expected %ld\n",
|
|
dwError, (TestFakeID ? dwErrCase : ErrInsuff));
|
|
if (ErrSuccess == ERROR_SUCCESS && (!TestFakeID || dwErrCase == ErrInsuff))
|
|
{
|
|
ok(uResultSize > 0,
|
|
"uResultSize is %u, expected > 0\n",
|
|
uResultSize);
|
|
}
|
|
else
|
|
{
|
|
ok(uResultSize == 0,
|
|
"uResultSize is %u, expected == 0\n",
|
|
uResultSize);
|
|
}
|
|
if (Buffer && dwException == STATUS_SUCCESS)
|
|
{
|
|
ok(*(WORD *)Buffer == 0xFFFF,
|
|
"Buffer should be clean at offset 0, got %x\n",
|
|
*(WORD *)Buffer);
|
|
}
|
|
|
|
// Test full size
|
|
if (Buffer && dwException == STATUS_SUCCESS)
|
|
{
|
|
FillMemory(Buffer, dwSize, 0xFF);
|
|
}
|
|
if (uResultSize == 0)
|
|
{
|
|
return;
|
|
}
|
|
SetLastError(0xbeeffeed);
|
|
dwError = GetLastError();
|
|
dwBufferSize = uResultSize;
|
|
uResultSize = 0;
|
|
StartSeh()
|
|
uResultSize = pGetSystemFirmwareTable(Signature, TableID, Buffer, dwBufferSize);
|
|
dwError = GetLastError();
|
|
EndSeh(ErrSuccess == ERROR_SUCCESS ? dwException : STATUS_SUCCESS);
|
|
// Windows 7: does not throw exception here
|
|
|
|
if (dwException == STATUS_SUCCESS || ErrSuccess == ERROR_INVALID_FUNCTION)
|
|
{
|
|
ok(dwError == ErrSuccess,
|
|
"GetLastError() returned %ld, expected %ld\n",
|
|
dwError, ErrSuccess);
|
|
if (ErrSuccess == ERROR_SUCCESS)
|
|
{
|
|
ok(uResultSize == dwBufferSize,
|
|
"uResultSize is not equal dwBufferSize, expected %ld\n",
|
|
dwBufferSize);
|
|
}
|
|
else
|
|
{
|
|
ok(uResultSize == 0,
|
|
"uResultSize is %u, expected == 0\n",
|
|
uResultSize);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Windows 7: returns ERROR_NOACCESS here
|
|
ok(dwError == 0xbeeffeed,
|
|
"GetLastError() returned %ld, expected %u\n",
|
|
dwError, 0xbeeffeed);
|
|
// Windows 7: returns correct size here
|
|
ok(uResultSize == 0,
|
|
"uResultSize is %u, expected == 0\n",
|
|
uResultSize);
|
|
}
|
|
|
|
if (Buffer && dwException == STATUS_SUCCESS)
|
|
{
|
|
if (ErrSuccess == ERROR_SUCCESS)
|
|
{
|
|
ok(*(DWORD *)Buffer != 0xFFFFFFFF,
|
|
"Buffer should be filled at offset 0\n");
|
|
}
|
|
else
|
|
{
|
|
ok(*(DWORD *)Buffer == 0xFFFFFFFF,
|
|
"Buffer should be clean at offset 0\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
static
|
|
VOID
|
|
test_Functions()
|
|
{
|
|
static const ENTRY Entries[] =
|
|
{
|
|
{ 'ACPI', ERROR_INSUFFICIENT_BUFFER, ERROR_SUCCESS },
|
|
{ 'FIRM', ERROR_INSUFFICIENT_BUFFER, ERROR_SUCCESS },
|
|
{ 'RSMB', ERROR_INSUFFICIENT_BUFFER, ERROR_SUCCESS },
|
|
/* This entry should be last */
|
|
{ 0xDEAD, ERROR_INVALID_FUNCTION, ERROR_INVALID_FUNCTION },
|
|
};
|
|
CHAR Buffer[262144]; // 256 KiB should be enough
|
|
CHAR Sign[sizeof(DWORD) + 1];
|
|
UINT TableCount[_countof(Entries)];
|
|
DWORD FirstTableID[_countof(Entries)];
|
|
int i;
|
|
|
|
// Test EnumSystemFirmwareTables
|
|
for (i = 0; i < _countof(Entries); i++)
|
|
{
|
|
// Test with NULL buffer
|
|
test_EnumBuffer(Entries[i].Signature, NULL, sizeof(Buffer), NULL, NULL,
|
|
Entries[i].ErrInsuff, Entries[i].ErrSuccess);
|
|
// Test with wrong buffer
|
|
test_EnumBuffer(Entries[i].Signature, (PVOID *)(LONG_PTR)0xbeeffeed, sizeof(Buffer), NULL, NULL,
|
|
Entries[i].ErrInsuff, Entries[i].ErrSuccess);
|
|
// Test with correct buffer
|
|
test_EnumBuffer(Entries[i].Signature, &Buffer, sizeof(Buffer), &TableCount[i], &FirstTableID[i],
|
|
Entries[i].ErrInsuff, Entries[i].ErrSuccess);
|
|
}
|
|
|
|
// Test GetSystemFirmwareTable
|
|
for (i = 0; i < _countof(Entries); i++)
|
|
{
|
|
// Test with fake ID and NULL buffer
|
|
test_GetBuffer(Entries[i].Signature, 0xbeeffeed, NULL, sizeof(Buffer),
|
|
TRUE, Entries[i].ErrInsuff, Entries[i].ErrSuccess);
|
|
// Test with fake ID and wrong buffer
|
|
test_GetBuffer(Entries[i].Signature, 0xbeeffeed, (PVOID *)(LONG_PTR)0xbeeffeed, sizeof(Buffer),
|
|
TRUE, Entries[i].ErrInsuff, Entries[i].ErrSuccess);
|
|
// Test with fake ID and correct buffer
|
|
test_GetBuffer(Entries[i].Signature, 0xbeeffeed, &Buffer, sizeof(Buffer),
|
|
TRUE, Entries[i].ErrInsuff, Entries[i].ErrSuccess);
|
|
if (TableCount[i] == 0)
|
|
{
|
|
if (i < _countof(Entries) - 1)
|
|
{
|
|
ZeroMemory(&Sign, sizeof(Sign));
|
|
*(DWORD *)&Sign = _byteswap_ulong(Entries[i].Signature);
|
|
skip("No tables for %s found. Skipping\n",
|
|
Sign);
|
|
}
|
|
continue;
|
|
}
|
|
// Test with correct ID and NULL buffer
|
|
test_GetBuffer(Entries[i].Signature, FirstTableID[i], NULL, sizeof(Buffer),
|
|
FALSE, Entries[i].ErrInsuff, Entries[i].ErrSuccess);
|
|
// Test with correct ID and wrong buffer
|
|
test_GetBuffer(Entries[i].Signature, FirstTableID[i], (PVOID *)(LONG_PTR)0xbeeffeed, sizeof(Buffer),
|
|
FALSE, Entries[i].ErrInsuff, Entries[i].ErrSuccess);
|
|
// Test with correct ID and correct buffer
|
|
test_GetBuffer(Entries[i].Signature, FirstTableID[i], &Buffer, sizeof(Buffer),
|
|
FALSE, Entries[i].ErrInsuff, Entries[i].ErrSuccess);
|
|
}
|
|
}
|
|
|
|
START_TEST(SystemFirmware)
|
|
{
|
|
HANDLE hKernel;
|
|
|
|
hKernel = GetModuleHandleW(L"kernel32.dll");
|
|
if (!hKernel)
|
|
{
|
|
skip("kernel32.dll module not found. Can't proceed\n");
|
|
return;
|
|
}
|
|
|
|
pEnumSystemFirmwareTables = (void *)fEnumSystemFirmwareTables;
|
|
pGetSystemFirmwareTable = (void *)fGetSystemFirmwareTable;
|
|
|
|
test_Functions();
|
|
|
|
pEnumSystemFirmwareTables = (void *)GetProcAddress(hKernel, "EnumSystemFirmwareTables");
|
|
pGetSystemFirmwareTable = (void *)GetProcAddress(hKernel, "GetSystemFirmwareTable");
|
|
|
|
if (!pEnumSystemFirmwareTables)
|
|
{
|
|
skip("EnumSystemFirmwareTables not found. Can't proceed\n");
|
|
return;
|
|
}
|
|
if (!pGetSystemFirmwareTable)
|
|
{
|
|
skip("GetSystemFirmwareTable not found. Can't proceed\n");
|
|
return;
|
|
}
|
|
test_Functions();
|
|
}
|