/* * Copyright (c) 2000, 2001, 2002, 2003, 2004, 2005, 2008, 2009 * The President and Fellows of Harvard College. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. Neither the name of the University nor the names of its contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE UNIVERSITY AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE UNIVERSITY OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ /* * malloctest.c * * This program contains a variety of tests for malloc and free. * XXX most tests leak on error. * * These tests (subject to restrictions and limitations noted below) should * work once user-level malloc is implemented. */ #include #include #include #include #include #include #define _PATH_RANDOM "random:" #define SMALLSIZE 72 #define MEDIUMSIZE 896 #define BIGSIZE 16384 #define HUGESIZE (1024 * 1024 * 1024) /* Maximum amount of space per block we allow for indexing structures */ #define OVERHEAD 32 /* Point past which we assume something else is going on */ #define ABSURD_OVERHEAD 256 static int geti(void) { int val=0; int ch, digits=0; while (1) { ch = getchar(); if (ch=='\n' || ch=='\r') { putchar('\n'); break; } else if ((ch=='\b' || ch==127) && digits>0) { printf("\b \b"); val = val/10; digits--; } else if (ch>='0' && ch<='9') { putchar(ch); val = val*10 + (ch-'0'); digits++; } else { putchar('\a'); } } if (digits==0) { return -1; } return val; } //////////////////////////////////////////////////////////// /* * Fill a block of memory with a test pattern. */ static void markblock(volatile void *ptr, size_t size, unsigned bias, int doprint) { size_t n, i; unsigned long *pl; unsigned long val; pl = (unsigned long *)ptr; n = size / sizeof(unsigned long); for (i=0; inext = list; list = tmp; tot += sizeof(struct test3); markblock(list->junk, sizeof(list->junk), (uintptr_t)list, 0); ct++; if (ct%128==0) { printf("."); } } printf("Allocated %lu bytes\n", (unsigned long) tot); printf("Trying some more allocations which I expect to fail...\n"); x = malloc(SMALLSIZE); if (x != NULL) { printf("FAILED: malloc(%u) succeeded\n", SMALLSIZE); return; } x = malloc(MEDIUMSIZE); if (x != NULL) { printf("FAILED: malloc(%u) succeeded\n", MEDIUMSIZE); return; } x = malloc(BIGSIZE); if (x != NULL) { printf("FAILED: malloc(%u) succeeded\n", BIGSIZE); return; } printf("Ok, now I'm going to free everything...\n"); while (list != NULL) { tmp = list->next; if (checkblock(list->junk, sizeof(list->junk), (uintptr_t)list, 0)) { failed = 1; } free(list); list = tmp; } if (failed) { printf("FAILED: data corruption\n"); return; } printf("Let me see if I can allocate some more now...\n"); x = malloc(MEDIUMSIZE); if (x == NULL) { printf("FAIL: Nope, I couldn't.\n"); return; } free(x); printf("Passed malloc test 3\n"); } //////////////////////////////////////////////////////////// /* * Test 4 * * Tries to test in detail if malloc coalesces the free list properly. * * This test will likely fail if something other than a basic first-fit/ * next-fit/best-fit algorithm is used. */ static void test4(void) { void *x, *y, *z; unsigned long lx, ly, lz, overhead, zsize; printf("Entering malloc test 4.\n"); printf("This test is intended for first/best-fit based mallocs.\n"); printf("This test may not work correctly if run after other tests.\n"); printf("Testing free list coalescing:\n"); x = malloc(SMALLSIZE); if (x==NULL) { printf("FAILED: malloc(%u) failed\n", SMALLSIZE); return; } y = malloc(MEDIUMSIZE); if (y==NULL) { printf("FAILED: malloc(%u) failed\n", MEDIUMSIZE); return; } if (sizeof(unsigned long) < sizeof(void *)) { printf("Buh? I can't fit a void * in an unsigned long\n"); printf("ENVIRONMENT FAILED...\n"); return; } lx = (unsigned long)x; ly = (unsigned long)y; printf("x is 0x%lx; y is 0x%lx\n", lx, ly); /* * The memory should look something like this: * * OXXXOYYYYYYYYYYY * * where O are optional overhead (indexing) blocks. */ /* This is obviously wrong. */ if (lx == ly) { printf("FAIL: x == y\n"); return; } /* * Check for overlap. It is sufficient to check if the start * of each block is within the other block. (If the end of a * block is within the other block, either the start is too, * or the other block's start is within the first block.) */ if (lx < ly && lx + SMALLSIZE > ly) { printf("FAIL: y starts within x\n"); return; } if (ly < lx && ly + MEDIUMSIZE > lx) { printf("FAIL: x starts within y\n"); return; } /* * If y is lower than x, some memory scheme we don't * understand is in use, or else there's already stuff on the * free list. */ if (ly < lx) { printf("TEST UNSUITABLE: y is below x\n"); return; } /* * Compute the space used by index structures. */ overhead = ly - (lx + SMALLSIZE); printf("Apparent block overhead: %lu\n", overhead); if (overhead > ABSURD_OVERHEAD) { printf("TEST UNSUITABLE: block overhead absurdly large.\n"); return; } if (overhead > OVERHEAD) { printf("FAIL: block overhead is too large.\n"); return; } printf("Freeing blocks...\n"); free(x); free(y); zsize = SMALLSIZE + MEDIUMSIZE + overhead; printf("Now allocating %lu bytes... should reuse the space.\n", zsize); z = malloc(zsize); if (z == NULL) { printf("FAIL: Allocation failed...\n"); return; } lz = (unsigned long) z; printf("z is 0x%lx (x was 0x%lx, y 0x%lx)\n", lz, lx, ly); if (lz==lx) { printf("Passed.\n"); } else { printf("Failed.\n"); } free(z); } //////////////////////////////////////////////////////////// /* * Test 5/6/7 * * Generally beats on malloc/free. * * Test 5 uses random seed 0. * Test 6 seeds the random number generator from random:. * Test 7 asks for a seed. */ static void test567(int testno, unsigned long seed) { static const int sizes[8] = { 13, 17, 69, 176, 433, 871, 1150, 6060 }; void *ptrs[32]; int psizes[32]; int i, n, size, failed=0; srandom(seed); printf("Seeded random number generator with %lu.\n", seed); for (i=0; i<32; i++) { ptrs[i] = NULL; psizes[i] = 0; } for (i=0; i<100000; i++) { n = random()%32; if (ptrs[n] == NULL) { size = sizes[random()%8]; ptrs[n] = malloc(size); psizes[n] = size; if (ptrs[n] == NULL) { printf("\nmalloc %u failed\n", size); failed = 1; break; } markblock(ptrs[n], size, n, 0); } else { size = psizes[n]; if (checkblock(ptrs[n], size, n, 0)) { failed = 1; break; } free(ptrs[n]); ptrs[n] = NULL; psizes[n] = 0; } if (i%256==0) { printf("."); } } printf("\n"); for (i=0; i<32; i++) { if (ptrs[i] != NULL) { free(ptrs[i]); } } if (failed) { printf("FAILED malloc test %d\n", testno); } else { printf("Passed malloc test %d\n", testno); } } static void test5(void) { printf("Beginning malloc test 5\n"); test567(5, 0); } static void test6(void) { int fd, len; unsigned long seed; printf("Beginning malloc test 6\n"); fd = open(_PATH_RANDOM, O_RDONLY); if (fd < 0) { err(1, "%s", _PATH_RANDOM); } len = read(fd, &seed, sizeof(seed)); if (len < 0) { err(1, "%s", _PATH_RANDOM); } else if (len < (int)sizeof(seed)) { errx(1, "%s: Short read", _PATH_RANDOM); } close(fd); test567(6, seed); } static void test7(void) { unsigned long seed; printf("Beginning malloc test 7\n"); printf("Enter random seed: "); seed = geti(); test567(7, seed); } //////////////////////////////////////////////////////////// static struct { int num; const char *desc; void (*func)(void); } tests[] = { { 1, "Simple allocation test", test1 }, { 2, "Allocate all memory in a big chunk", test2 }, { 3, "Allocate all memory in small chunks", test3 }, { 4, "Free list coalescing test (first/next/best-fit only)", test4 }, { 5, "Stress test", test5 }, { 6, "Randomized stress test", test6 }, { 7, "Stress test with particular seed", test7 }, { -1, NULL, NULL } }; static int dotest(int tn) { int i; for (i=0; tests[i].num>=0; i++) { if (tests[i].num == tn) { tests[i].func(); return 0; } } return -1; } int main(int argc, char *argv[]) { int i, tn, menu=1; if (argc > 1) { for (i=1; i=0; i++) { printf(" %2d %s\n", tests[i].num, tests[i].desc); } menu = 0; } printf("malloctest: "); tn = geti(); if (tn < 0) { break; } if (dotest(tn)) { menu = 1; } } return 0; }