mirror of
https://github.com/reactos/reactos.git
synced 2025-01-03 21:09:19 +00:00
f61e14f554
- For NT6.0+, NTDLL exports those hash APIs that ADVAPI32 thunks forward to - Fix undocumented context structure layout and content - Add apitests to verify both of hash result and structure content
303 lines
9 KiB
C
303 lines
9 KiB
C
/*
|
|
* PROJECT: ReactOS API tests
|
|
* LICENSE: GPL-2.0-or-later (https://spdx.org/licenses/GPL-2.0-or-later)
|
|
* PURPOSE: Tests for basic hash APIs
|
|
* COPYRIGHT: Copyright 2023 Ratin Gao <ratin@knsoft.org>
|
|
*/
|
|
|
|
#include "precomp.h"
|
|
|
|
typedef struct _A_SHA_CTX
|
|
{
|
|
UCHAR Buffer[64];
|
|
ULONG State[5];
|
|
ULONG Count[2];
|
|
} A_SHA_CTX, *PA_SHA_CTX;
|
|
|
|
#define A_SHA_DIGEST_LEN 20
|
|
|
|
typedef struct _MD5_CTX
|
|
{
|
|
ULONG Count[2];
|
|
ULONG State[4];
|
|
UCHAR Buffer[64];
|
|
UCHAR Hash[16];
|
|
} MD5_CTX, *PMD5_CTX;
|
|
|
|
#define MD5_DIGEST_LEN 16
|
|
|
|
typedef struct _MD4_CTX
|
|
{
|
|
ULONG State[4];
|
|
ULONG Count[2];
|
|
UCHAR Buffer[64];
|
|
UCHAR Hash[16];
|
|
} MD4_CTX, *PMD4_CTX;
|
|
|
|
#define MD4_DIGEST_LEN 16
|
|
|
|
#ifndef RSA32API
|
|
#define RSA32API __stdcall
|
|
#endif
|
|
|
|
typedef
|
|
VOID
|
|
RSA32API
|
|
FN_A_SHAInit(
|
|
_Out_ PA_SHA_CTX Context);
|
|
|
|
typedef
|
|
VOID
|
|
RSA32API
|
|
FN_A_SHAUpdate(
|
|
_Inout_ PA_SHA_CTX Context,
|
|
_In_reads_(BufferSize) PUCHAR Buffer,
|
|
_In_ ULONG BufferSize);
|
|
|
|
typedef
|
|
VOID
|
|
RSA32API
|
|
FN_A_SHAFinal(
|
|
_Inout_ PA_SHA_CTX Context,
|
|
_Out_ PUCHAR Result);
|
|
|
|
typedef
|
|
VOID
|
|
RSA32API
|
|
FN_MD5Init(
|
|
_Out_ PMD5_CTX Context);
|
|
|
|
typedef
|
|
VOID
|
|
RSA32API
|
|
FN_MD5Update(
|
|
_Inout_ PMD5_CTX Context,
|
|
_In_reads_(BufferSize) PUCHAR Buffer,
|
|
_In_ ULONG BufferSize);
|
|
|
|
typedef
|
|
VOID
|
|
RSA32API
|
|
FN_MD5Final(
|
|
_Inout_ PMD5_CTX Context);
|
|
|
|
typedef
|
|
VOID
|
|
RSA32API
|
|
FN_MD4Init(
|
|
_Out_ PMD4_CTX Context);
|
|
|
|
typedef
|
|
VOID
|
|
RSA32API
|
|
FN_MD4Update(
|
|
_Inout_ PMD4_CTX Context,
|
|
_In_reads_(BufferSize) PUCHAR Buffer,
|
|
_In_ ULONG BufferSize);
|
|
|
|
typedef
|
|
VOID
|
|
RSA32API
|
|
FN_MD4Final(
|
|
_Inout_ PMD4_CTX Context);
|
|
|
|
static HMODULE g_hAdvapi32 = NULL;
|
|
static ANSI_STRING g_TestString = RTL_CONSTANT_STRING("ReactOS Hash API Test String");
|
|
|
|
static ULONG g_ctxSHA1StateInit[] = { 0x67452301, 0xEFCDAB89, 0x98BADCFE, 0x10325476, 0xC3D2E1F0 };
|
|
static UCHAR g_aucSHA1Result[A_SHA_DIGEST_LEN] = {
|
|
0xEC, 0x05, 0x43, 0xE7, 0xDE, 0x8A, 0xEE, 0xFF,
|
|
0xAD, 0x72, 0x2B, 0x9D, 0x55, 0x4F, 0xCA, 0x6A,
|
|
0x8D, 0x81, 0xF1, 0xC7
|
|
};
|
|
|
|
static ULONG g_aulMD5Or4StateInit[] = { 0x67452301, 0xEFCDAB89, 0x98BADCFE, 0x10325476 };
|
|
static UCHAR g_aucMD5Result[MD5_DIGEST_LEN] = {
|
|
0x3D, 0xE8, 0x23, 0x8B, 0x9D, 0xE0, 0xCE, 0x48,
|
|
0xB1, 0x1B, 0xDD, 0xD9, 0xC6, 0x86, 0xB2, 0xDE
|
|
};
|
|
static UCHAR g_aucMD4Result[MD4_DIGEST_LEN] = {
|
|
0xE0, 0xE8, 0x50, 0x8A, 0x4D, 0x11, 0x02, 0xA6,
|
|
0x6A, 0xF0, 0xA7, 0xAB, 0xD8, 0xC4, 0x40, 0xED
|
|
};
|
|
|
|
static void Test_SHA1(void)
|
|
{
|
|
FN_A_SHAInit* pfnA_SHAInit;
|
|
FN_A_SHAUpdate* pfnA_SHAUpdate;
|
|
FN_A_SHAFinal* pfnA_SHAFinal;
|
|
DECLSPEC_ALIGN(4) A_SHA_CTX ctx;
|
|
UCHAR Result[A_SHA_DIGEST_LEN];
|
|
SIZE_T ComparedSize;
|
|
|
|
/* Load functions */
|
|
pfnA_SHAInit = (FN_A_SHAInit*)GetProcAddress(g_hAdvapi32, "A_SHAInit");
|
|
pfnA_SHAUpdate = (FN_A_SHAUpdate*)GetProcAddress(g_hAdvapi32, "A_SHAUpdate");
|
|
pfnA_SHAFinal = (FN_A_SHAFinal*)GetProcAddress(g_hAdvapi32, "A_SHAFinal");
|
|
|
|
if (!pfnA_SHAInit || !pfnA_SHAUpdate || !pfnA_SHAFinal)
|
|
{
|
|
skip("advapi32.dll!A_SHA*** not found\n");
|
|
return;
|
|
}
|
|
|
|
/* Test A_SHAInit */
|
|
pfnA_SHAInit(&ctx);
|
|
ComparedSize = RtlCompareMemory(ctx.State,
|
|
g_ctxSHA1StateInit,
|
|
RTL_FIELD_SIZE(A_SHA_CTX, State));
|
|
ok_eq_size(ComparedSize, RTL_FIELD_SIZE(A_SHA_CTX, State));
|
|
ok_eq_ulong(ctx.Count[0], 0UL);
|
|
ok_eq_ulong(ctx.Count[1], 0UL);
|
|
|
|
/* Test A_SHAUpdate */
|
|
pfnA_SHAUpdate(&ctx, (PUCHAR)g_TestString.Buffer, g_TestString.Length);
|
|
ComparedSize = RtlCompareMemory(ctx.Buffer,
|
|
g_TestString.Buffer,
|
|
g_TestString.Length);
|
|
ok_eq_size(ComparedSize, g_TestString.Length);
|
|
ComparedSize = RtlCompareMemory(ctx.State,
|
|
g_ctxSHA1StateInit,
|
|
RTL_FIELD_SIZE(A_SHA_CTX, State));
|
|
ok_eq_size(ComparedSize, RTL_FIELD_SIZE(A_SHA_CTX, State));
|
|
ok_eq_ulong(ctx.Count[0], 0UL);
|
|
ok_eq_ulong(ctx.Count[1], (ULONG)g_TestString.Length);
|
|
|
|
/* Test A_SHAFinal */
|
|
pfnA_SHAFinal(&ctx, Result);
|
|
ComparedSize = RtlCompareMemoryUlong(ctx.Buffer, sizeof(ctx.Buffer), 0);
|
|
ok_eq_size(ComparedSize, sizeof(ctx.Buffer));
|
|
ComparedSize = RtlCompareMemory(ctx.State,
|
|
g_ctxSHA1StateInit,
|
|
RTL_FIELD_SIZE(A_SHA_CTX, State));
|
|
ok_eq_size(ComparedSize, RTL_FIELD_SIZE(A_SHA_CTX, State));
|
|
ok_eq_ulong(ctx.Count[0], 0UL);
|
|
ok_eq_ulong(ctx.Count[1], 0UL);
|
|
ComparedSize = RtlCompareMemory(Result, g_aucSHA1Result, A_SHA_DIGEST_LEN);
|
|
ok_eq_size(ComparedSize, A_SHA_DIGEST_LEN);
|
|
}
|
|
|
|
static void Test_MD5(void)
|
|
{
|
|
FN_MD5Init* pfnMD5Init;
|
|
FN_MD5Update* pfnMD5Update;
|
|
FN_MD5Final* pfnMD5Final;
|
|
DECLSPEC_ALIGN(4) MD5_CTX ctx;
|
|
SIZE_T ComparedSize;
|
|
|
|
/* Load functions */
|
|
pfnMD5Init = (FN_MD5Init*)GetProcAddress(g_hAdvapi32, "MD5Init");
|
|
pfnMD5Update = (FN_MD5Update*)GetProcAddress(g_hAdvapi32, "MD5Update");
|
|
pfnMD5Final = (FN_MD5Final*)GetProcAddress(g_hAdvapi32, "MD5Final");
|
|
|
|
if (!pfnMD5Init || !pfnMD5Update || !pfnMD5Final)
|
|
{
|
|
skip("advapi32.dll!MD5*** not found\n");
|
|
return;
|
|
}
|
|
|
|
/* Test MD5Init */
|
|
pfnMD5Init(&ctx);
|
|
ok_eq_ulong(ctx.Count[0], 0UL);
|
|
ok_eq_ulong(ctx.Count[1], 0UL);
|
|
ComparedSize = RtlCompareMemory(ctx.State,
|
|
g_aulMD5Or4StateInit,
|
|
RTL_FIELD_SIZE(MD5_CTX, State));
|
|
ok_eq_size(ComparedSize, RTL_FIELD_SIZE(MD5_CTX, State));
|
|
|
|
/* Test MD5Update */
|
|
pfnMD5Update(&ctx, (PUCHAR)g_TestString.Buffer, g_TestString.Length);
|
|
ComparedSize = RtlCompareMemory(ctx.Buffer,
|
|
g_TestString.Buffer,
|
|
g_TestString.Length);
|
|
ok_eq_size(ComparedSize, g_TestString.Length);
|
|
ComparedSize = RtlCompareMemory(ctx.State,
|
|
g_aulMD5Or4StateInit,
|
|
RTL_FIELD_SIZE(MD5_CTX, State));
|
|
ok_eq_size(ComparedSize, RTL_FIELD_SIZE(MD5_CTX, State));
|
|
ok_eq_ulong(ctx.Count[0], (ULONG)g_TestString.Length * CHAR_BIT);
|
|
ok_eq_ulong(ctx.Count[1], 0UL);
|
|
|
|
/* Test MD5Final */
|
|
pfnMD5Final(&ctx);
|
|
ok_eq_ulong(ctx.Count[0], 0x200UL);
|
|
ok_eq_ulong(ctx.Count[1], 0UL);
|
|
ok_eq_ulong(ctx.State[0], 0x8B23E83DUL);
|
|
ok_eq_ulong(ctx.State[1], 0x48CEE09DUL);
|
|
ok_eq_ulong(ctx.State[2], 0xD9DD1BB1UL);
|
|
ok_eq_ulong(ctx.State[3], 0xDEB286C6UL);
|
|
ComparedSize = RtlCompareMemoryUlong(ctx.Buffer, sizeof(ctx.Buffer), 0);
|
|
ok_eq_size(ComparedSize, sizeof(ctx.Buffer));
|
|
ComparedSize = RtlCompareMemory(ctx.Hash, &g_aucMD5Result, MD5_DIGEST_LEN);
|
|
ok_eq_size(ComparedSize, MD5_DIGEST_LEN);
|
|
}
|
|
|
|
static void Test_MD4(void)
|
|
{
|
|
FN_MD4Init* pfnMD4Init;
|
|
FN_MD4Update* pfnMD4Update;
|
|
FN_MD4Final* pfnMD4Final;
|
|
DECLSPEC_ALIGN(4) MD4_CTX ctx;
|
|
SIZE_T ComparedSize;
|
|
|
|
/* Load functions */
|
|
pfnMD4Init = (FN_MD4Init*)GetProcAddress(g_hAdvapi32, "MD4Init");
|
|
pfnMD4Update = (FN_MD4Update*)GetProcAddress(g_hAdvapi32, "MD4Update");
|
|
pfnMD4Final = (FN_MD4Final*)GetProcAddress(g_hAdvapi32, "MD4Final");
|
|
|
|
if (!pfnMD4Init || !pfnMD4Update || !pfnMD4Final)
|
|
{
|
|
skip("advapi32.dll!MD4*** not found\n");
|
|
return;
|
|
}
|
|
|
|
/* Test MD4Init */
|
|
pfnMD4Init(&ctx);
|
|
ok_eq_ulong(ctx.Count[0], 0UL);
|
|
ok_eq_ulong(ctx.Count[1], 0UL);
|
|
ComparedSize = RtlCompareMemory(ctx.State,
|
|
g_aulMD5Or4StateInit,
|
|
RTL_FIELD_SIZE(MD4_CTX, State));
|
|
ok_eq_size(ComparedSize, RTL_FIELD_SIZE(MD4_CTX, State));
|
|
|
|
/* Test MD4Update */
|
|
pfnMD4Update(&ctx, (PUCHAR)g_TestString.Buffer, g_TestString.Length);
|
|
ComparedSize = RtlCompareMemory(ctx.Buffer,
|
|
g_TestString.Buffer,
|
|
g_TestString.Length);
|
|
ok_eq_size(ComparedSize, g_TestString.Length);
|
|
ComparedSize = RtlCompareMemory(ctx.State,
|
|
g_aulMD5Or4StateInit,
|
|
RTL_FIELD_SIZE(MD4_CTX, State));
|
|
ok_eq_size(ComparedSize, RTL_FIELD_SIZE(MD4_CTX, State));
|
|
ok_eq_ulong(ctx.Count[0], (ULONG)g_TestString.Length * CHAR_BIT);
|
|
ok_eq_ulong(ctx.Count[1], 0UL);
|
|
|
|
/* Test MD4Final */
|
|
pfnMD4Final(&ctx);
|
|
ok_eq_ulong(ctx.Count[0], 0x200UL);
|
|
ok_eq_ulong(ctx.Count[1], 0UL);
|
|
ok_eq_ulong(ctx.State[0], 0x8A50E8E0UL);
|
|
ok_eq_ulong(ctx.State[1], 0xA602114DUL);
|
|
ok_eq_ulong(ctx.State[2], 0xABA7F06AUL);
|
|
ok_eq_ulong(ctx.State[3], 0xED40C4D8UL);
|
|
ComparedSize = RtlCompareMemoryUlong(ctx.Buffer, sizeof(ctx.Buffer), 0);
|
|
ok_eq_size(ComparedSize, sizeof(ctx.Buffer));
|
|
ComparedSize = RtlCompareMemory(ctx.Hash, &g_aucMD4Result, MD4_DIGEST_LEN);
|
|
ok_eq_size(ComparedSize, MD4_DIGEST_LEN);
|
|
}
|
|
|
|
START_TEST(Hash)
|
|
{
|
|
/* Load advapi32.dll */
|
|
g_hAdvapi32 = GetModuleHandleW(L"advapi32.dll");
|
|
if (!g_hAdvapi32)
|
|
{
|
|
skip("Module advapi32 not found\n");
|
|
return;
|
|
}
|
|
|
|
Test_SHA1();
|
|
Test_MD5();
|
|
Test_MD4();
|
|
}
|