kernel32/tests: Cleanup heap allocation functions tests.

Signed-off-by: Rémi Bernon <rbernon@codeweavers.com>
Signed-off-by: Alexandre Julliard <julliard@winehq.org>
This commit is contained in:
Rémi Bernon 2022-04-07 21:50:18 +02:00 committed by Alexandre Julliard
parent 60e0ae2f4c
commit 8eb9cbd6e7
1 changed files with 217 additions and 145 deletions

View File

@ -81,33 +81,9 @@ static SIZE_T resize_9x(SIZE_T size)
return max(dwSizeAligned, 12); /* at least 12 bytes */ return max(dwSizeAligned, 12); /* at least 12 bytes */
} }
static void test_sized_HeapAlloc(int nbytes)
{
BOOL success;
char *buf = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, nbytes);
ok(buf != NULL, "allocate failed\n");
ok(buf[0] == 0, "buffer not zeroed\n");
success = HeapFree(GetProcessHeap(), 0, buf);
ok(success, "free failed\n");
}
static void test_sized_HeapReAlloc(int nbytes1, int nbytes2)
{
BOOL success;
char *buf = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, nbytes1);
ok(buf != NULL, "allocate failed\n");
ok(buf[0] == 0, "buffer not zeroed\n");
buf = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, buf, nbytes2);
ok(buf != NULL, "reallocate failed\n");
ok(buf[nbytes2-1] == 0, "buffer not zeroed\n");
success = HeapFree(GetProcessHeap(), 0, buf);
ok(success, "free failed\n");
}
static void test_heap(void) static void test_heap(void)
{ {
LPVOID mem; LPVOID mem;
LPVOID msecond;
SIZE_T size; SIZE_T size;
/* Heap*() functions */ /* Heap*() functions */
@ -115,9 +91,6 @@ static void test_heap(void)
ok(mem != NULL, "memory not allocated for size 0\n"); ok(mem != NULL, "memory not allocated for size 0\n");
HeapFree(GetProcessHeap(), 0, mem); HeapFree(GetProcessHeap(), 0, mem);
mem = HeapReAlloc(GetProcessHeap(), 0, NULL, 10);
ok(mem == NULL, "memory allocated by HeapReAlloc\n");
for (size = 0; size <= 256; size++) for (size = 0; size <= 256; size++)
{ {
SIZE_T heap_size; SIZE_T heap_size;
@ -128,24 +101,6 @@ static void test_heap(void)
HeapFree(GetProcessHeap(), 0, mem); HeapFree(GetProcessHeap(), 0, mem);
} }
/* test some border cases of HeapAlloc and HeapReAlloc */
mem = HeapAlloc(GetProcessHeap(), 0, 0);
ok(mem != NULL, "memory not allocated for size 0\n");
msecond = pHeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, mem, ~(SIZE_T)0 - 7);
ok(msecond == NULL, "HeapReAlloc(~0 - 7) should have failed\n");
msecond = pHeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, mem, ~(SIZE_T)0);
ok(msecond == NULL, "HeapReAlloc(~0) should have failed\n");
HeapFree(GetProcessHeap(), 0, mem);
mem = pHeapAlloc(GetProcessHeap(), 0, ~(SIZE_T)0);
ok(mem == NULL, "memory allocated for size ~0\n");
mem = HeapAlloc(GetProcessHeap(), 0, 17);
msecond = HeapReAlloc(GetProcessHeap(), 0, mem, 0);
ok(msecond != NULL, "HeapReAlloc(0) should have succeeded\n");
size = HeapSize(GetProcessHeap(), 0, msecond);
ok(size == 0 || broken(size == 1) /* some vista and win7 */,
"HeapSize should have returned 0 instead of %Iu\n", size);
HeapFree(GetProcessHeap(), 0, msecond);
/* large blocks must be 16-byte aligned */ /* large blocks must be 16-byte aligned */
mem = HeapAlloc(GetProcessHeap(), 0, 512 * 1024); mem = HeapAlloc(GetProcessHeap(), 0, 512 * 1024);
ok( mem != NULL, "failed for size 512K\n" ); ok( mem != NULL, "failed for size 512K\n" );
@ -157,110 +112,235 @@ static void test_heap(void)
static void test_HeapCreate(void) static void test_HeapCreate(void)
{ {
SYSTEM_INFO sysInfo; SIZE_T alloc_size = 0x8000 * sizeof(void *), size;
ULONG memchunk; HANDLE heap, heap1;
HANDLE heap; BYTE *ptr, *ptr1;
LPVOID mem1,mem1a,mem3; BOOL ret;
UCHAR *mem2,*mem2a;
UINT i;
BOOL error;
DWORD dwSize;
/* Retrieve the page size for this system */ /* check heap alignment */
GetSystemInfo(&sysInfo);
ok(sysInfo.dwPageSize>0,"GetSystemInfo should return a valid page size\n");
/* Create a Heap with a minimum and maximum size */ heap = HeapCreate( 0, 0, 0 );
/* Note that Windows and Wine seem to behave a bit differently with respect ok( !!heap, "HeapCreate failed, error %lu\n", GetLastError() );
to memory allocation. In Windows, you can't access all the memory ok( !((ULONG_PTR)heap & 0xffff), "wrong heap alignment\n" );
specified in the heap (due to overhead), so choosing a reasonable maximum heap1 = HeapCreate( 0, 0, 0 );
size for the heap was done mostly by trial-and-error on Win2k. It may need ok( !!heap, "HeapCreate failed, error %lu\n", GetLastError() );
more tweaking for otherWindows variants. ok( !((ULONG_PTR)heap1 & 0xffff), "wrong heap alignment\n" );
*/ ret = HeapDestroy( heap1 );
memchunk=10*sysInfo.dwPageSize; ok( ret, "HeapDestroy failed, error %lu\n", GetLastError() );
heap=HeapCreate(0,2*memchunk,5*memchunk); ret = HeapDestroy( heap );
ok( !((ULONG_PTR)heap & 0xffff), "heap %p not 64K aligned\n", heap ); ok( ret, "HeapDestroy failed, error %lu\n", GetLastError() );
/* Check that HeapCreate allocated the right amount of ram */ /* growable heap */
mem1=HeapAlloc(heap,0,5*memchunk+1);
ok(mem1==NULL,"HeapCreate allocated more Ram than it should have\n");
HeapFree(heap,0,mem1);
/* Check that a normal alloc works */ heap = HeapCreate( 0, 0, 0 );
mem1=HeapAlloc(heap,0,memchunk); ok( !!heap, "HeapCreate failed, error %lu\n", GetLastError() );
ok(mem1!=NULL,"HeapAlloc failed\n"); ok( !((ULONG_PTR)heap & 0xffff), "wrong heap alignment\n" );
if(mem1) {
ok(HeapSize(heap,0,mem1)>=memchunk, "HeapAlloc should return a big enough memory block\n");
}
/* Check that a 'zeroing' alloc works */ /* test some border cases */
mem2=HeapAlloc(heap,HEAP_ZERO_MEMORY,memchunk);
ok(mem2!=NULL,"HeapAlloc failed\n");
if(mem2) {
ok(HeapSize(heap,0,mem2)>=memchunk,"HeapAlloc should return a big enough memory block\n");
error=FALSE;
for(i=0;i<memchunk;i++) {
if(mem2[i]!=0) {
error=TRUE;
}
}
ok(!error,"HeapAlloc should have zeroed out its allocated memory\n");
}
/* Check that HeapAlloc returns NULL when requested way too much memory */ ret = HeapFree( heap, 0, NULL );
mem3=HeapAlloc(heap,0,5*memchunk); ok( ret, "HeapFree failed, error %lu\n", GetLastError() );
ok(mem3==NULL,"HeapAlloc should return NULL\n");
if(mem3) {
ok(HeapFree(heap,0,mem3),"HeapFree didn't pass successfully\n");
}
/* Check that HeapReAlloc works */ SetLastError( 0xdeadbeef );
mem2a=HeapReAlloc(heap,HEAP_ZERO_MEMORY,mem2,memchunk+5*sysInfo.dwPageSize); ptr = HeapReAlloc( heap, 0, NULL, 1 );
ok(mem2a!=NULL,"HeapReAlloc failed\n"); ok( !ptr, "HeapReAlloc succeeded\n" );
if(mem2a) { todo_wine
ok(HeapSize(heap,0,mem2a)>=memchunk+5*sysInfo.dwPageSize,"HeapReAlloc failed\n"); ok( GetLastError() == NO_ERROR, "got error %lu\n", GetLastError() );
error=FALSE;
for(i=0;i<5*sysInfo.dwPageSize;i++) {
if(mem2a[memchunk+i]!=0) {
error=TRUE;
}
}
ok(!error,"HeapReAlloc should have zeroed out its allocated memory\n");
}
/* Check that HeapReAlloc honours HEAP_REALLOC_IN_PLACE_ONLY */ ptr = HeapAlloc( heap, 0, 0 );
error=FALSE; ok( !!ptr, "HeapAlloc failed, error %lu\n", GetLastError() );
mem1a=HeapReAlloc(heap,HEAP_REALLOC_IN_PLACE_ONLY,mem1,memchunk+sysInfo.dwPageSize); size = HeapSize( heap, 0, ptr );
if(mem1a!=NULL) { ok( size == 0, "HeapSize returned %#Ix, error %lu\n", size, GetLastError() );
if(mem1a!=mem1) { ptr1 = pHeapReAlloc( heap, 0, ptr, ~(SIZE_T)0 - 7 );
error=TRUE; ok( !ptr1, "HeapReAlloc succeeded\n" );
} ptr1 = pHeapReAlloc( heap, 0, ptr, ~(SIZE_T)0 );
} ok( !ptr1, "HeapReAlloc succeeded\n" );
ok(mem1a==NULL || !error,"HeapReAlloc didn't honour HEAP_REALLOC_IN_PLACE_ONLY\n"); ret = HeapFree( heap, 0, ptr );
ok( ret, "HeapFree failed, error %lu\n", GetLastError() );
/* Check that HeapFree works correctly */ ptr = pHeapAlloc( heap, 0, ~(SIZE_T)0 );
if(mem1a) { ok( !ptr, "HeapAlloc succeeded\n" );
ok(HeapFree(heap,0,mem1a),"HeapFree failed\n");
} else {
ok(HeapFree(heap,0,mem1),"HeapFree failed\n");
}
if(mem2a) {
ok(HeapFree(heap,0,mem2a),"HeapFree failed\n");
} else {
ok(HeapFree(heap,0,mem2),"HeapFree failed\n");
}
/* 0-length buffer */ ptr = HeapAlloc( heap, 0, 1 );
mem1 = HeapAlloc(heap, 0, 0); ok( !!ptr, "HeapAlloc failed, error %lu\n", GetLastError() );
ok(mem1 != NULL, "Reserved memory\n"); ptr1 = HeapReAlloc( heap, 0, ptr, 0 );
ok( !!ptr1, "HeapReAlloc failed, error %lu\n", GetLastError() );
size = HeapSize( heap, 0, ptr1 );
ok( size == 0, "HeapSize returned %#Ix, error %lu\n", size, GetLastError() );
ret = HeapFree( heap, 0, ptr );
ok( ret, "HeapFree failed, error %lu\n", GetLastError() );
dwSize = HeapSize(heap, 0, mem1); ptr = HeapAlloc( heap, 0, 5 * alloc_size + 1 );
/* should work with 0-length buffer */ ok( !!ptr, "HeapAlloc failed, error %lu\n", GetLastError() );
ok(dwSize < 0xFFFFFFFF, "The size of the 0-length buffer\n"); ret = HeapFree( heap, 0, ptr );
ok(HeapFree(heap, 0, mem1), "Freed the 0-length buffer\n"); ok( ret, "HeapFree failed, error %lu\n", GetLastError() );
/* Check that HeapDestroy works */ ptr = HeapAlloc( heap, 0, alloc_size );
ok(HeapDestroy(heap),"HeapDestroy failed\n"); ok( !!ptr, "HeapAlloc failed, error %lu\n", GetLastError() );
size = HeapSize( heap, 0, ptr );
ok( size == alloc_size, "HeapSize returned %#Ix, error %lu\n", size, GetLastError() );
ptr1 = HeapAlloc( heap, 0, 4 * alloc_size );
ok( !!ptr1, "HeapAlloc failed, error %lu\n", GetLastError() );
ret = HeapFree( heap, 0, ptr1 );
ok( ret, "HeapFree failed, error %lu\n", GetLastError() );
ret = HeapFree( heap, 0, ptr );
ok( ret, "HeapFree failed, error %lu\n", GetLastError() );
/* test HEAP_ZERO_MEMORY */
ptr = HeapAlloc( heap, HEAP_ZERO_MEMORY, 1 );
ok( !!ptr, "HeapAlloc failed, error %lu\n", GetLastError() );
size = HeapSize( heap, 0, ptr );
ok( size == 1, "HeapSize returned %#Ix, error %lu\n", size, GetLastError() );
while (size) if (ptr[--size]) break;
ok( !size && !ptr[0], "memory wasn't zeroed\n" );
ret = HeapFree( heap, 0, ptr );
ok( ret, "HeapFree failed, error %lu\n", GetLastError() );
ptr = HeapAlloc( heap, HEAP_ZERO_MEMORY, (1 << 20) );
ok( !!ptr, "HeapAlloc failed, error %lu\n", GetLastError() );
size = HeapSize( heap, 0, ptr );
ok( size == (1 << 20), "HeapSize returned %#Ix, error %lu\n", size, GetLastError() );
while (size) if (ptr[--size]) break;
ok( !size && !ptr[0], "memory wasn't zeroed\n" );
ret = HeapFree( heap, 0, ptr );
ok( ret, "HeapFree failed, error %lu\n", GetLastError() );
ptr = HeapAlloc( heap, HEAP_ZERO_MEMORY, alloc_size );
ok( !!ptr, "HeapAlloc failed, error %lu\n", GetLastError() );
size = HeapSize( heap, 0, ptr );
ok( size == alloc_size, "HeapSize returned %#Ix, error %lu\n", size, GetLastError() );
while (size) if (ptr[--size]) break;
ok( !size && !ptr[0], "memory wasn't zeroed\n" );
ptr = HeapReAlloc( heap, HEAP_ZERO_MEMORY, ptr, 3 * alloc_size );
ok( !!ptr, "HeapReAlloc failed, error %lu\n", GetLastError() );
size = HeapSize( heap, 0, ptr );
ok( size == 3 * alloc_size, "HeapSize returned %#Ix, error %lu\n", size, GetLastError() );
while (size) if (ptr[--size]) break;
ok( !size && !ptr[0], "memory wasn't zeroed\n" );
/* shrinking a small-ish block in place and growing back is okay */
ptr1 = HeapReAlloc( heap, HEAP_REALLOC_IN_PLACE_ONLY, ptr, alloc_size * 3 / 2 );
ok( ptr1 == ptr, "HeapReAlloc HEAP_REALLOC_IN_PLACE_ONLY failed, error %lu\n", GetLastError() );
ptr1 = HeapReAlloc( heap, HEAP_REALLOC_IN_PLACE_ONLY, ptr, 2 * alloc_size );
ok( ptr1 == ptr, "HeapReAlloc HEAP_REALLOC_IN_PLACE_ONLY failed, error %lu\n", GetLastError() );
ptr = HeapReAlloc( heap, HEAP_ZERO_MEMORY, ptr, 1 );
ok( !!ptr, "HeapReAlloc failed, error %lu\n", GetLastError() );
size = HeapSize( heap, 0, ptr );
ok( size == 1, "HeapSize returned %#Ix, error %lu\n", size, GetLastError() );
while (size) if (ptr[--size]) break;
ok( !size && !ptr[0], "memory wasn't zeroed\n" );
ptr = HeapReAlloc( heap, HEAP_ZERO_MEMORY, ptr, (1 << 20) );
ok( !!ptr, "HeapReAlloc failed, error %lu\n", GetLastError() );
size = HeapSize( heap, 0, ptr );
ok( size == (1 << 20), "HeapSize returned %#Ix, error %lu\n", size, GetLastError() );
while (size) if (ptr[--size]) break;
ok( !size && !ptr[0], "memory wasn't zeroed\n" );
/* shrinking a very large block decommits pages and fail to grow in place */
ptr1 = HeapReAlloc( heap, HEAP_REALLOC_IN_PLACE_ONLY, ptr, alloc_size * 3 / 2 );
ok( ptr1 == ptr, "HeapReAlloc HEAP_REALLOC_IN_PLACE_ONLY failed, error %lu\n", GetLastError() );
ptr1 = HeapReAlloc( heap, HEAP_REALLOC_IN_PLACE_ONLY, ptr, 2 * alloc_size );
todo_wine
ok( ptr1 != ptr, "HeapReAlloc HEAP_REALLOC_IN_PLACE_ONLY succeeded\n" );
ok( GetLastError() == ERROR_NOT_ENOUGH_MEMORY, "got error %lu\n", GetLastError() );
ret = HeapFree( heap, 0, ptr1 );
ok( ret, "HeapFree failed, error %lu\n", GetLastError() );
ret = HeapDestroy( heap );
ok( ret, "HeapDestroy failed, error %lu\n", GetLastError() );
/* fixed size heaps */
heap = HeapCreate( 0, alloc_size, alloc_size );
ok( !!heap, "HeapCreate failed, error %lu\n", GetLastError() );
ok( !((ULONG_PTR)heap & 0xffff), "wrong heap alignment\n" );
ptr = HeapAlloc( heap, 0, alloc_size - (0x400 + 0x80 * sizeof(void *)) );
ok( !!ptr, "HeapAlloc failed, error %lu\n", GetLastError() );
size = HeapSize( heap, 0, ptr );
ok( size == alloc_size - (0x400 + 0x80 * sizeof(void *)),
"HeapSize returned %#Ix, error %lu\n", size, GetLastError() );
ret = HeapFree( heap, 0, ptr );
ok( ret, "HeapFree failed, error %lu\n", GetLastError() );
SetLastError( 0xdeadbeef );
ptr1 = HeapAlloc( heap, 0, alloc_size - (0x200 + 0x80 * sizeof(void *)) );
todo_wine
ok( !ptr1, "HeapAlloc succeeded\n" );
todo_wine
ok( GetLastError() == ERROR_NOT_ENOUGH_MEMORY, "got error %lu\n", GetLastError() );
ret = HeapFree( heap, 0, ptr1 );
ok( ret, "HeapFree failed, error %lu\n", GetLastError() );
ret = HeapDestroy( heap );
ok( ret, "HeapDestroy failed, error %lu\n", GetLastError() );
heap = HeapCreate( 0, 2 * alloc_size, 5 * alloc_size );
ok( !!heap, "HeapCreate failed, error %lu\n", GetLastError() );
ok( !((ULONG_PTR)heap & 0xffff), "wrong heap alignment\n" );
ptr = HeapAlloc( heap, 0, 0 );
ok( !!ptr, "HeapAlloc failed, error %lu\n", GetLastError() );
size = HeapSize( heap, 0, ptr );
ok( size == 0, "HeapSize returned %#Ix, error %lu\n", size, GetLastError() );
ret = HeapFree( heap, 0, ptr );
ok( ret, "HeapFree failed, error %lu\n", GetLastError() );
/* cannot allocate large blocks from fixed size heap */
SetLastError( 0xdeadbeef );
ptr1 = HeapAlloc( heap, 0, 3 * alloc_size );
todo_wine
ok( !ptr1, "HeapAlloc succeeded\n" );
todo_wine
ok( GetLastError() == ERROR_NOT_ENOUGH_MEMORY, "got error %lu\n", GetLastError() );
ret = HeapFree( heap, 0, ptr1 );
ok( ret, "HeapFree failed, error %lu\n", GetLastError() );
ptr = HeapAlloc( heap, 0, alloc_size );
ok( !!ptr, "HeapAlloc failed, error %lu\n", GetLastError() );
size = HeapSize( heap, 0, ptr );
ok( size == alloc_size, "HeapSize returned %#Ix, error %lu\n", size, GetLastError() );
SetLastError( 0xdeadbeef );
ptr1 = HeapAlloc( heap, 0, 4 * alloc_size );
ok( !ptr1, "HeapAlloc succeeded\n" );
todo_wine
ok( GetLastError() == ERROR_NOT_ENOUGH_MEMORY, "got error %lu\n", GetLastError() );
ret = HeapFree( heap, 0, ptr1 );
ok( ret, "HeapFree failed, error %lu\n", GetLastError() );
ret = HeapFree( heap, 0, ptr );
ok( ret, "HeapFree failed, error %lu\n", GetLastError() );
ptr = HeapAlloc( heap, HEAP_ZERO_MEMORY, alloc_size );
ok( !!ptr, "HeapAlloc failed, error %lu\n", GetLastError() );
size = HeapSize( heap, 0, ptr );
ok( size == alloc_size, "HeapSize returned %#Ix, error %lu\n", size, GetLastError() );
while (size) if (ptr[--size]) break;
ok( !size && !ptr[0], "memory wasn't zeroed\n" );
ptr = HeapReAlloc( heap, HEAP_ZERO_MEMORY, ptr, 2 * alloc_size );
ok( !!ptr, "HeapReAlloc failed, error %lu\n", GetLastError() );
size = HeapSize( heap, 0, ptr );
ok( size == 2 * alloc_size, "HeapSize returned %#Ix, error %lu\n", size, GetLastError() );
while (size) if (ptr[--size]) break;
ok( !size && !ptr[0], "memory wasn't zeroed\n" );
ptr1 = HeapReAlloc( heap, HEAP_REALLOC_IN_PLACE_ONLY, ptr, alloc_size * 3 / 2 );
ok( ptr1 == ptr, "HeapReAlloc HEAP_REALLOC_IN_PLACE_ONLY failed, error %lu\n", GetLastError() );
ptr1 = HeapReAlloc( heap, HEAP_REALLOC_IN_PLACE_ONLY, ptr, 2 * alloc_size );
ok( ptr1 == ptr, "HeapReAlloc HEAP_REALLOC_IN_PLACE_ONLY failed, error %lu\n", GetLastError() );
ret = HeapFree( heap, 0, ptr1 );
ok( ret, "HeapFree failed, error %lu\n", GetLastError() );
ret = HeapDestroy( heap );
ok( ret, "HeapDestroy failed, error %lu\n", GetLastError() );
} }
@ -1605,14 +1685,6 @@ START_TEST(heap)
test_GlobalAlloc(); test_GlobalAlloc();
test_LocalAlloc(); test_LocalAlloc();
/* Test both short and very long blocks */
test_sized_HeapAlloc(1);
test_sized_HeapAlloc(1 << 20);
test_sized_HeapReAlloc(1, 100);
test_sized_HeapReAlloc(1, (1 << 20));
test_sized_HeapReAlloc((1 << 20), (2 << 20));
test_sized_HeapReAlloc((1 << 20), 1);
test_HeapQueryInformation(); test_HeapQueryInformation();
test_GetPhysicallyInstalledSystemMemory(); test_GetPhysicallyInstalledSystemMemory();
test_GlobalMemoryStatus(); test_GlobalMemoryStatus();