2020-09-30 16:53:55 +02:00

879 lines
16 KiB
C

#include <nt.h>
#include <ntrtl.h>
#include <nturtl.h>
#include <windows.h>
#include <malloc.h>
#include <stdio.h>
#include <suballoc.h>
typedef struct _MemTrack *PMEMTRACK;
typedef struct _MemTrack {
PMEMTRACK Next;
ULONG Address;
ULONG Size;
} MEMTRACK;
VOID
SimpleTest1(
PVOID SA
);
VOID
SimpleTest2(
PVOID SA
);
VOID
SimpleTest3(
PVOID SA
);
VOID
SimpleTest4(
PVOID
);
VOID
SimpleTest5(
PVOID
);
VOID
SimpleTest6(
PVOID
);
NTSTATUS
DecommitMem(
ULONG BaseAddress,
ULONG Size
);
NTSTATUS
CommitMem(
ULONG BaseAddress,
ULONG Size
);
VOID
MoveMem(
ULONG Destination,
ULONG Source,
ULONG Size
);
main(argv, argc)
char **argv;
int argc;
{
NTSTATUS Status;
PVOID BaseAddress, SA;
ULONG Size;
ULONG TotalFree, LargestFree;
ULONG i,j;
BOOL Success;
//
// Allocate a region of memory for SA to manage
//
BaseAddress = NULL;
Size = 15*1024*1024 - 64 * 1024;
Status = NtAllocateVirtualMemory(
NtCurrentProcess(),
&BaseAddress,
0,
&Size,
MEM_RESERVE,
PAGE_READWRITE
);
if (!NT_SUCCESS(Status)) {
printf("Couldn't allocate memory, %lx\n", Status);
return(0);
}
//
// Initialize the suballocation stuff
//
SA = SAInitialize(
(ULONG)BaseAddress,
Size,
CommitMem,
DecommitMem,
MoveMem
);
if (SA == NULL) {
printf("SAInitialize Failed\n");
}
//
// Find out how much free memory we have
//
Success = SAQueryFree(
SA,
&TotalFree,
&LargestFree
);
if (!Success) {
printf("Could not query free\n");
return;
}
//
// Run some tests
//
SimpleTest1(SA);
//
// Allocate blocks spanning two commit blocks
//
SimpleTest2(SA);
//
// Allocate and free all of memory (twice)
//
SimpleTest3(SA);
//
// More complex alloc and free test
//
SimpleTest4(SA);
//
// Test realloc
//
SimpleTest5(SA);
//
// Test Queryfree
//
SimpleTest6(SA);
//
// Make sure we didn't leak
//
Success = SAQueryFree(
SA,
&i,
&j
);
if (!Success){
printf("Couldn't requery free\n");
return;
}
if ((i != TotalFree) || (j != LargestFree)) {
printf("We leaked\n");
}
}
VOID
SimpleTest1(
PVOID SA
)
{
ULONG Address;
BOOL Success;
Success = SAAllocate(
SA,
50,
&Address
);
if (!Success) {
printf("SimpleTest1: Failed to allocate\n");
return;
}
try {
*((PULONG)(Address + 4)) = 1;
} except (EXCEPTION_EXECUTE_HANDLER) {
printf("SimpleTest1: Faulted accessing memory\n");
return;
}
Success = SAFree(
SA,
50,
Address
);
if (!Success) {
printf("SimpleTest1: Failed to free\n");
return;
}
Success = TRUE;
try {
*((PULONG)Address + 4) = 1;
} except (EXCEPTION_EXECUTE_HANDLER) {
Success = FALSE;
}
if (Success) {
printf("SimpleTest1: memory not decommited\n");
}
}
VOID
SimpleTest2(
PVOID SA
)
{
ULONG SmallAddress, LargeAddress;
BOOL Success;
Success = SAAllocate(
SA,
50,
&SmallAddress
);
if (!Success) {
printf("SimpleTest2: Could not alloc small block\n");
return;
}
try {
*((PULONG)(SmallAddress + 4)) = 1;
} except (EXCEPTION_EXECUTE_HANDLER) {
printf("SimpleTest2: small block not committed\n");
return;
}
Success = SAAllocate(
SA,
COMMIT_GRANULARITY + 50,
&LargeAddress
);
if (!Success) {
printf("SimpleTest2: Could not alloc large block\n");
return;
}
try {
*((PULONG)(LargeAddress + 4)) = 1;
*((PULONG)(LargeAddress + COMMIT_GRANULARITY)) = 1;
} except (EXCEPTION_EXECUTE_HANDLER) {
printf("SimpleTest2: Large block not committed\n");
return;
}
Success = SAFree(
SA,
50,
SmallAddress
);
if (!Success) {
printf("SimpleTest2: failed to free small block\n");
return;
}
try {
*((PULONG)(LargeAddress + 4)) = 1;
} except (EXCEPTION_EXECUTE_HANDLER) {
printf("SimpleTest2: LargeBlock decommited!!\n");
return;
}
Success = SAFree(
SA,
COMMIT_GRANULARITY + 50,
LargeAddress
);
if (!Success) {
printf("SimpleTest2: failed to free Large block\n");
return;
}
Success = FALSE;
try {
*((PULONG)(LargeAddress + 4)) = 1;
} except (EXCEPTION_EXECUTE_HANDLER) {
Success = TRUE;
}
if (!Success) {
printf("SimpleTest2: First block not decommited\n");
}
Success = FALSE;
try {
*((PULONG)(LargeAddress + COMMIT_GRANULARITY + 4)) = 1;
} except (EXCEPTION_EXECUTE_HANDLER) {
Success = TRUE;
}
if (!Success) {
printf("SimpleTest2: Last block not decommited\n");
}
}
VOID
SimpleTest3(
PVOID SA
)
{
ULONG BlockCount, i, Address;
PMEMTRACK p, Head, f;
BOOL Success;
Head = NULL;
BlockCount = 0;
do {
Success = SAAllocate(
SA,
3072,
&Address
);
if (Success) {
p = malloc(sizeof(MEMTRACK));
if (p == NULL) {
printf("SimpleTest3: malloc error\n");
return;
}
p->Next = Head;
p->Size = 3072;
p->Address = Address;
Head = p;
BlockCount++;
}
} while (Success);
try {
p = Head;
while (p != NULL) {
*((PULONG)(p->Address + 100)) = 1;
p = p->Next;
}
} except (EXCEPTION_EXECUTE_HANDLER) {
printf("SimpleTest3: failed first access test\n");
return;
}
p = Head;
while (p != NULL) {
Success = SAFree(
SA,
3072,
p->Address
);
if (!Success) {
printf("SimpleTest3: Failed first free test\n");
return;
}
f = p;
p = p->Next;
free(f);
}
Head = NULL;
for (i = 0; i < BlockCount; i++) {
Success = SAAllocate(
SA,
3072,
&Address
);
if (!Success) {
printf("SimpleTest3: Failed second alloc test\n");
return;
}
p = malloc(sizeof(MEMTRACK));
if (p == NULL) {
printf("SimpleTest3: malloc error\n");
return;
}
p->Next = Head;
p->Size = 3072;
p->Address = Address;
Head = p;
}
try {
p = Head;
while (p != NULL) {
*((PULONG)(p->Address + 100)) = 1;
p = p->Next;
}
} except (EXCEPTION_EXECUTE_HANDLER) {
printf("SimpleTest3: failed second access test\n");
return;
}
p = Head;
while (p != NULL) {
Success = SAFree(
SA,
3072,
p->Address
);
if (!Success) {
printf("SimpleTest3: Failed second free test\n");
return;
}
f = p;
p = p->Next;
free(f);
}
}
VOID
SimpleTest4(
PVOID SA
)
{
ULONG i, Address;
PMEMTRACK p, Head, f;
BOOL Success;
Head = NULL;
do {
Success = SAAllocate(
SA,
3072,
&Address
);
if (Success) {
p = malloc(sizeof(MEMTRACK));
if (p == NULL) {
printf("SimpleTest4: malloc error\n");
return;
}
p->Next = Head;
p->Size = 3072;
p->Address = Address;
Head = p;
}
} while (Success);
p = Head;
p = p->Next;
p = p->Next;
Success = SAFree(
SA,
3072,
p->Next->Address
);
if (!Success) {
printf("SimpleTest4: could not free\n");
return;
}
f = p->Next;
p->Next = p->Next->Next;
free (f);
p = p->Next;
p = p->Next;
Success = SAFree(
SA,
3072,
p->Next->Address
);
if (!Success) {
printf("SimpleTest4: could not free\n");
return;
}
f = p->Next;
p->Next = p->Next->Next;
free (f);
Success = SAFree(
SA,
3072,
p->Next->Address
);
if (!Success) {
printf("SimpleTest4: could not free\n");
return;
}
f = p->Next;
p->Next = p->Next->Next;
free (f);
Success = SAAllocate(
SA,
3072,
&Address
);
try {
*((PULONG)(Address + 4)) = 1;
} except (EXCEPTION_EXECUTE_HANDLER) {
printf("SimpleTest4: failed to access\n");
return;
}
if (!Success) {
printf("SimpleTest4: could not allocate\n");
return;
}
p = malloc(sizeof(MEMTRACK));
if (!p) {
printf("SimpleTest4: could not malloc\n");
return;
}
p->Next = Head;
p->Size = 3072;
p->Address = Address;
Head = p;
Success = SAAllocate(
SA,
3072,
&Address
);
try {
*((PULONG)(Address + 4)) = 1;
} except (EXCEPTION_EXECUTE_HANDLER) {
printf("SimpleTest4: failed to access\n");
return;
}
if (!Success) {
printf("SimpleTest4: could not allocate\n");
return;
}
p = malloc(sizeof(MEMTRACK));
if (!p) {
printf("SimpleTest4: could not malloc\n");
return;
}
p->Next = Head;
p->Size = 3072;
p->Address = Address;
Head = p;
p = Head;
while (p != NULL) {
Success = SAFree(
SA,
3072,
p->Address
);
if (!Success) {
printf("SimpleTest3: Failed second free test\n");
return;
}
f = p;
p = p->Next;
free(f);
}
}
VOID
SimpleTest5(
PVOID SA
)
{
ULONG Address, NewAddress;
ULONG Address1, Address2, Address3;
ULONG Size1, Size2, Size3;
BOOL Success;
Success = SAAllocate(
SA,
3072,
&Address
);
if (!Success) {
printf("SimpleTest5: failed to allocate\n");
return;
}
Success = SAReallocate(
SA,
3072,
Address,
3000,
&NewAddress
);
if (!Success) {
printf("SimpleTest5: failed to reallocate\n");
return;
}
if (NewAddress != Address) {
printf("SimpleTest5: Realloc in place failed\n");
}
Success = SAReallocate(
SA,
3072,
Address,
200,
&NewAddress
);
if (!Success) {
printf("SimpleTest5: failed to reallocate\n");
return;
}
if (NewAddress != Address) {
printf("SimpleTest5: Realloc in place failed\n");
return;
}
Success = SAReallocate(
SA,
200,
Address,
6000,
&NewAddress
);
if (!Success) {
printf("SimpleTest5: failed to reallocate\n");
return;
}
if (NewAddress != Address) {
printf("SimpleTest5: realloc in place failed\n");
return;
}
Success = SAAllocate(
SA,
1500,
&Address
);
if (!Success) {
printf("SimpleTest5: failed to allocate\n");
return;
}
Address1 = NewAddress;
Size1 = 6000;
Address2 = Address;
Size2 = 1500;
Success = SAReallocate(
SA,
Size1,
Address1,
3000,
&NewAddress
);
if (!Success) {
printf("SimpleTest5: failed to reallocate\n");
return;
}
if (Address1 != NewAddress) {
printf("SimpleTest5: realloc in place failed\n");
return;
}
Size1= 3000;
Success = SAAllocate(
SA,
2000,
&Address
);
if (!Success) {
printf("SimpleTest5: failed to allocate\n");
return;
}
Address3 = Address;
Size3 = 2000;
Success = SAFree(
SA,
Size1,
Address1
);
if (!Success) {
printf("SimpleTest5: failed to free\n");
return;
}
Success = SAReallocate(
SA,
Size3,
Address3,
5000,
&Address
);
if (!Success) {
printf("SimpleTest5: failed to reallocate\n");
return;
}
Address3 = Address;
Size3 = 5000;
Success = SAReallocate(
SA,
Size3,
Address3,
10000,
&Address
);
if (!Success) {
printf("SimpleTest5: failed to reallocate\n");
return;
}
Address3 = Address;
Size3 = 10000;
Success = SAFree(
SA,
Size2,
Address2
);
if (!Success) {
printf("SimpleTest5: failed to free\n");
return;
}
Success = SAFree(
SA,
Size3,
Address3
);
if (!Success) {
printf("SimpleTest5: failed to free\n");
return;
}
}
VOID
SimpleTest6(
PVOID SA
)
{
ULONG LargestFree, TotalFree;
BOOL Success;
Success = SAQueryFree(
SA,
&TotalFree,
&LargestFree
);
}
NTSTATUS
CommitMem(
ULONG BaseAddress,
ULONG Size
)
{
NTSTATUS Status;
PVOID Address;
ULONG size;
Address = (PVOID)BaseAddress;
size = Size;
Status = NtAllocateVirtualMemory(
NtCurrentProcess(),
&Address,
0L,
&size,
MEM_COMMIT,
PAGE_READWRITE
);
if (!NT_SUCCESS(Status)) {
printf(
"CommitMem failed to commit %lx, %lx, error %lx\n",
Address,
size,
Status
);
}
return Status;
}
NTSTATUS
DecommitMem(
ULONG BaseAddress,
ULONG Size
)
{
NTSTATUS Status;
PVOID Address;
ULONG size;
Address = (PVOID)BaseAddress;
size = Size;
Status = NtFreeVirtualMemory(
NtCurrentProcess(),
&Address,
&size,
MEM_DECOMMIT
);
if (!NT_SUCCESS(Status)) {
printf(
"DecommitMem failed to decommit %lx, %lx, error %lx\n",
Address,
size,
Status
);
}
return Status;
}
VOID
MoveMem(
ULONG Destination,
ULONG Source,
ULONG Size
)
{
RtlMoveMemory(
(PVOID)Destination,
(PVOID)Source,
Size
);
}