mirror of
https://github.com/reactos/reactos.git
synced 2025-01-07 14:51:00 +00:00
264 lines
9.2 KiB
C
264 lines
9.2 KiB
C
/*
|
|
* PROJECT: ReactOS api tests
|
|
* LICENSE: GPL - See COPYING in the top level directory
|
|
* PURPOSE: Test for NtContinue
|
|
* PROGRAMMER:
|
|
*/
|
|
|
|
#include "precomp.h"
|
|
|
|
#include <setjmp.h>
|
|
#include <time.h>
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma runtime_checks("s", off)
|
|
#endif
|
|
|
|
#define ok_eq_print(value, expected, spec) ok((value) == (expected), #value " = " spec ", expected " spec "\n", value, expected)
|
|
#define ok_eq_hex(value, expected) ok_eq_print(value, expected, "%lx")
|
|
#define ok_eq_hex64(value, expected) ok_eq_print(value, expected, "%I64x")
|
|
#define ok_eq_xmm(value, expected) ok((value).Low == (expected).Low, #value " = %I64x'%08I64x, expected %I64x'%08I64x\n", (value).Low, (value).High, (expected).Low, (expected).High)
|
|
|
|
#ifdef _M_IX86
|
|
#define NTC_SEGMENT_BITS (0xFFFF)
|
|
#define NTC_EFLAGS_BITS (0x3C0CD5)
|
|
#endif
|
|
|
|
void continuePoint(void);
|
|
|
|
static jmp_buf jmpbuf;
|
|
static CONTEXT continueContext;
|
|
static unsigned int nRandBytes;
|
|
|
|
static int initrand(void)
|
|
{
|
|
unsigned int nRandMax;
|
|
unsigned int nRandMaxBits;
|
|
time_t tLoc;
|
|
|
|
nRandMax = RAND_MAX;
|
|
for(nRandMaxBits = 0; nRandMax != 0; nRandMax >>= 1, ++ nRandMaxBits);
|
|
nRandBytes = nRandMaxBits / CHAR_BIT;
|
|
//assert(nRandBytes != 0);
|
|
srand((unsigned)(time(&tLoc) & UINT_MAX));
|
|
return 1;
|
|
}
|
|
|
|
static void randbytes(void * p, size_t n)
|
|
{
|
|
unsigned char * b;
|
|
size_t i;
|
|
int r = rand();
|
|
|
|
b = (unsigned char *)p;
|
|
for(i = 0; i < n; ++ i)
|
|
{
|
|
if(i % nRandBytes == 0)
|
|
r = rand();
|
|
b[i] = (unsigned char)(r & UCHAR_MAX);
|
|
r >>= CHAR_BIT;
|
|
}
|
|
}
|
|
|
|
static ULONG randULONG(void)
|
|
{
|
|
ULONG n;
|
|
randbytes(&n, sizeof(n));
|
|
return n;
|
|
}
|
|
|
|
#ifdef _M_AMD64
|
|
static ULONG64 randULONG64(void)
|
|
{
|
|
return (ULONG64)randULONG() << 32 | randULONG();
|
|
}
|
|
#endif
|
|
|
|
void check(CONTEXT * pContext)
|
|
{
|
|
#ifdef _M_IX86
|
|
ok(pContext->ContextFlags == CONTEXT_FULL,
|
|
"ContextFlags=0x%lx\n", pContext->ContextFlags);
|
|
|
|
/* Random data segments */
|
|
ok((pContext->SegGs & NTC_SEGMENT_BITS) ==
|
|
(continueContext.SegGs & NTC_SEGMENT_BITS),
|
|
"SegGs=0x%lx / 0x%lx\n", pContext->SegGs, continueContext.SegGs);
|
|
|
|
ok((pContext->SegFs & NTC_SEGMENT_BITS) ==
|
|
(continueContext.SegFs & NTC_SEGMENT_BITS),
|
|
"SegFs=0x%lx / 0x%lx\n", pContext->SegFs, continueContext.SegFs);
|
|
|
|
ok((pContext->SegEs & NTC_SEGMENT_BITS) ==
|
|
(continueContext.SegEs & NTC_SEGMENT_BITS),
|
|
"SegEs=0x%lx / 0x%lx\n", pContext->SegEs, continueContext.SegEs);
|
|
|
|
ok((pContext->SegDs & NTC_SEGMENT_BITS) ==
|
|
(continueContext.SegDs & NTC_SEGMENT_BITS),
|
|
"SegDs=0x%lx / 0x%lx\n", pContext->SegDs, continueContext.SegDs);
|
|
|
|
/* Integer registers */
|
|
ok(pContext->Edi == continueContext.Edi,
|
|
"Edi: 0x%lx != 0x%lx\n", pContext->Edi, continueContext.Edi);
|
|
ok(pContext->Esi == continueContext.Esi,
|
|
"Esi: 0x%lx != 0x%lx\n", pContext->Esi, continueContext.Esi);
|
|
ok(pContext->Ebx == continueContext.Ebx,
|
|
"Ebx: 0x%lx != 0x%lx\n", pContext->Ebx, continueContext.Ebx);
|
|
ok(pContext->Edx == continueContext.Edx,
|
|
"Edx: 0x%lx != 0x%lx\n", pContext->Edx, continueContext.Edx);
|
|
ok(pContext->Ecx == continueContext.Ecx,
|
|
"Ecx: 0x%lx != 0x%lx\n", pContext->Ecx, continueContext.Ecx);
|
|
ok(pContext->Eax == continueContext.Eax,
|
|
"Eax: 0x%lx != 0x%lx\n", pContext->Eax, continueContext.Eax);
|
|
|
|
/* Control registers and segments */
|
|
ok(pContext->Ebp == continueContext.Ebp,
|
|
"Ebp: 0x%lx != 0x%lx\n", pContext->Ebp, continueContext.Ebp);
|
|
ok(pContext->Eip == continueContext.Eip,
|
|
"Eip: 0x%lx != 0x%lx\n", pContext->Eip, continueContext.Eip);
|
|
ok(pContext->Esp == continueContext.Esp,
|
|
"Esp: 0x%lx != 0x%lx\n", pContext->Esp, continueContext.Esp);
|
|
|
|
ok((pContext->SegCs & NTC_SEGMENT_BITS) ==
|
|
(continueContext.SegCs & NTC_SEGMENT_BITS),
|
|
"SegCs: 0x%lx != 0x%lx\n", pContext->SegCs, continueContext.SegCs);
|
|
|
|
ok((pContext->EFlags & NTC_EFLAGS_BITS) ==
|
|
(continueContext.EFlags & NTC_EFLAGS_BITS),
|
|
"EFlags: 0x%lx != 0x%lx\n", pContext->EFlags, continueContext.EFlags);
|
|
|
|
ok((pContext->SegSs & NTC_SEGMENT_BITS) ==
|
|
(continueContext.SegSs & NTC_SEGMENT_BITS),
|
|
"SegSs: 0x%lx != 0x%lx\n", pContext->SegSs, continueContext.SegSs);
|
|
#else
|
|
ok_eq_hex64(pContext->ContextFlags, CONTEXT_FULL | CONTEXT_SEGMENTS);
|
|
ok_eq_hex(pContext->MxCsr, continueContext.MxCsr);
|
|
ok_eq_hex(pContext->SegCs, continueContext.SegCs);
|
|
ok_eq_hex(pContext->SegDs, 0x2B);
|
|
ok_eq_hex(pContext->SegEs, 0x2B);
|
|
ok_eq_hex(pContext->SegFs, 0x53);
|
|
ok_eq_hex(pContext->SegGs, 0x2B);
|
|
ok_eq_hex(pContext->SegSs, continueContext.SegSs);
|
|
ok_eq_hex(pContext->EFlags, (continueContext.EFlags & ~0x1C0000) | 0x202);
|
|
|
|
ok_eq_hex64(pContext->Rax, continueContext.Rax);
|
|
ok_eq_hex64(pContext->Rdx, continueContext.Rdx);
|
|
ok_eq_hex64(pContext->Rbx, continueContext.Rbx);
|
|
ok_eq_hex64(pContext->Rsp, continueContext.Rsp);
|
|
ok_eq_hex64(pContext->Rbp, continueContext.Rbp);
|
|
ok_eq_hex64(pContext->Rsi, continueContext.Rsi);
|
|
ok_eq_hex64(pContext->Rdi, continueContext.Rdi);
|
|
ok_eq_hex64(pContext->R8, continueContext.R8);
|
|
ok_eq_hex64(pContext->R9, continueContext.R9);
|
|
ok_eq_hex64(pContext->R10, continueContext.R10);
|
|
ok_eq_hex64(pContext->R11, continueContext.R11);
|
|
ok_eq_hex64(pContext->R12, continueContext.R12);
|
|
ok_eq_hex64(pContext->R13, continueContext.R13);
|
|
ok_eq_hex64(pContext->R14, continueContext.R14);
|
|
ok_eq_hex64(pContext->R15, continueContext.R15);
|
|
ok_eq_xmm(pContext->Xmm0, continueContext.Xmm0);
|
|
ok_eq_xmm(pContext->Xmm1, continueContext.Xmm1);
|
|
ok_eq_xmm(pContext->Xmm2, continueContext.Xmm2);
|
|
ok_eq_xmm(pContext->Xmm3, continueContext.Xmm3);
|
|
ok_eq_xmm(pContext->Xmm4, continueContext.Xmm4);
|
|
ok_eq_xmm(pContext->Xmm5, continueContext.Xmm5);
|
|
ok_eq_xmm(pContext->Xmm6, continueContext.Xmm6);
|
|
ok_eq_xmm(pContext->Xmm7, continueContext.Xmm7);
|
|
ok_eq_xmm(pContext->Xmm8, continueContext.Xmm8);
|
|
ok_eq_xmm(pContext->Xmm9, continueContext.Xmm9);
|
|
ok_eq_xmm(pContext->Xmm10, continueContext.Xmm10);
|
|
ok_eq_xmm(pContext->Xmm11, continueContext.Xmm11);
|
|
ok_eq_xmm(pContext->Xmm12, continueContext.Xmm12);
|
|
ok_eq_xmm(pContext->Xmm13, continueContext.Xmm13);
|
|
ok_eq_xmm(pContext->Xmm14, continueContext.Xmm14);
|
|
ok_eq_xmm(pContext->Xmm15, continueContext.Xmm15);
|
|
|
|
// Clear the frame register to prevent unwinding, which is broken
|
|
((_JUMP_BUFFER*)&jmpbuf)->Frame = 0;
|
|
#endif
|
|
|
|
/* Return where we came from */
|
|
longjmp(jmpbuf, 1);
|
|
}
|
|
|
|
START_TEST(NtContinue)
|
|
{
|
|
initrand();
|
|
|
|
RtlFillMemory(&continueContext, sizeof(continueContext), 0xBBBBBBBB);
|
|
|
|
/* First time */
|
|
if(setjmp(jmpbuf) == 0)
|
|
{
|
|
CONTEXT bogus[2];
|
|
|
|
RtlFillMemory(&bogus, sizeof(bogus), 0xCCCCCCCC);
|
|
|
|
continueContext.ContextFlags = CONTEXT_FULL;
|
|
GetThreadContext(GetCurrentThread(), &continueContext);
|
|
|
|
#ifdef _M_IX86
|
|
continueContext.ContextFlags = CONTEXT_FULL;
|
|
|
|
/* Fill the integer registers with random values */
|
|
continueContext.Edi = randULONG();
|
|
continueContext.Esi = randULONG();
|
|
continueContext.Ebx = randULONG();
|
|
continueContext.Edx = randULONG();
|
|
continueContext.Ecx = randULONG();
|
|
continueContext.Eax = randULONG();
|
|
continueContext.Ebp = randULONG();
|
|
|
|
/* Randomize all the allowed flags (determined experimentally with WinDbg) */
|
|
continueContext.EFlags = randULONG() & 0x3C0CD5;
|
|
|
|
/* Randomize the stack pointer as much as possible */
|
|
continueContext.Esp = (ULONG)(((ULONG_PTR)&bogus) & 0xFFFFFFFF) +
|
|
sizeof(bogus) - (randULONG() & 0xF) * 4;
|
|
|
|
/* continuePoint() is implemented in assembler */
|
|
continueContext.Eip = (ULONG)((ULONG_PTR)continuePoint & 0xFFFFFFF);
|
|
|
|
/* Can't do a lot about segments */
|
|
#elif defined(_M_AMD64)
|
|
continueContext.ContextFlags = CONTEXT_FULL | CONTEXT_DEBUG_REGISTERS;
|
|
|
|
/* Fill the integer registers with random values */
|
|
PULONG64 Registers = &continueContext.Rax;
|
|
for (ULONG i = 0; i < 16; i++)
|
|
{
|
|
Registers[i] = randULONG64();
|
|
}
|
|
|
|
/* Fill the XMM registers with random values */
|
|
Registers = (PULONG64)&continueContext.Xmm0;
|
|
for (ULONG i = 0; i < 32; i++)
|
|
{
|
|
Registers[i] = randULONG64();
|
|
}
|
|
|
|
continueContext.Dr0 = randULONG64() & 0xFFFF;
|
|
continueContext.Dr1 = randULONG64() & 0xFFFF;
|
|
continueContext.Dr2 = randULONG64() & 0xFFFF;
|
|
continueContext.Dr3 = randULONG64() & 0xFFFF;
|
|
continueContext.Dr6 = randULONG64() & 0xFFFF;
|
|
continueContext.Dr7 = randULONG64() & 0xFFFF;
|
|
|
|
/* Randomize all the allowed flags (determined experimentally with WinDbg) */
|
|
continueContext.EFlags = randULONG64() & 0x3C0CD5;
|
|
|
|
/* Randomize the stack pointer as much as possible */
|
|
continueContext.Rsp = (((ULONG_PTR)&bogus)) + (randULONG() & 0xF) * 16;
|
|
continueContext.Rsp = ALIGN_DOWN_BY(continueContext.Rsp, 16);
|
|
|
|
/* continuePoint() is implemented in assembler */
|
|
continueContext.Rip = ((ULONG_PTR)continuePoint);
|
|
#endif
|
|
|
|
NtContinue(&continueContext, FALSE);
|
|
ok(0, "should never get here\n");
|
|
}
|
|
|
|
/* Second time */
|
|
return;
|
|
}
|