123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512 |
- /*******************************************************************************
- * Copyright (c) 2009, 2018 IBM Corp.
- *
- * All rights reserved. This program and the accompanying materials
- * are made available under the terms of the Eclipse Public License v1.0
- * and Eclipse Distribution License v1.0 which accompany this distribution.
- *
- * The Eclipse Public License is available at
- * http://www.eclipse.org/legal/epl-v10.html
- * and the Eclipse Distribution License is available at
- * http://www.eclipse.org/org/documents/edl-v10.php.
- *
- * Contributors:
- * Ian Craggs - initial API and implementation and/or initial documentation
- * Ian Craggs - use tree data structure instead of list
- * Ian Craggs - change roundup to Heap_roundup to avoid macro name clash on MacOSX
- *******************************************************************************/
- /**
- * @file
- * \brief functions to manage the heap with the goal of eliminating memory leaks
- *
- * For any module to use these functions transparently, simply include the Heap.h
- * header file. Malloc and free will be redefined, but will behave in exactly the same
- * way as normal, so no recoding is necessary.
- *
- * */
- #include "Tree.h"
- #include "Log.h"
- #include "StackTrace.h"
- #include "Thread.h"
- #if defined(HEAP_UNIT_TESTS)
- char* Broker_recordFFDC(char* symptoms);
- #endif /* HEAP_UNIT_TESTS */
- #include <stdlib.h>
- #include <string.h>
- #include <stdio.h>
- #include <stddef.h>
- #include "Heap.h"
- #undef malloc
- #undef realloc
- #undef free
- #if defined(WIN32) || defined(WIN64)
- mutex_type heap_mutex;
- #else
- static pthread_mutex_t heap_mutex_store = PTHREAD_MUTEX_INITIALIZER;
- static mutex_type heap_mutex = &heap_mutex_store;
- #endif
- static heap_info state = {0, 0}; /**< global heap state information */
- static int eyecatcher = 0x88888888;
- /*#define HEAP_STACK 1 */
- /**
- * Each item on the heap is recorded with this structure.
- */
- typedef struct
- {
- char* file; /**< the name of the source file where the storage was allocated */
- int line; /**< the line no in the source file where it was allocated */
- void* ptr; /**< pointer to the allocated storage */
- size_t size; /**< size of the allocated storage */
- #if defined(HEAP_STACK)
- char* stack;
- #endif
- } storageElement;
- static Tree heap; /**< Tree that holds the allocation records */
- static const char *errmsg = "Memory allocation error";
- static size_t Heap_roundup(size_t size);
- static int ptrCompare(void* a, void* b, int value);
- /*static void Heap_check(char* string, void* ptr);*/
- static void checkEyecatchers(char* file, int line, void* p, size_t size);
- static int Internal_heap_unlink(char* file, int line, void* p);
- static void HeapScan(enum LOG_LEVELS log_level);
- /**
- * Round allocation size up to a multiple of the size of an int. Apart from possibly reducing fragmentation,
- * on the old v3 gcc compilers I was hitting some weird behaviour, which might have been errors in
- * sizeof() used on structures and related to packing. In any case, this fixes that too.
- * @param size the size actually needed
- * @return the rounded up size
- */
- static size_t Heap_roundup(size_t size)
- {
- static int multsize = 4*sizeof(int);
- if (size % multsize != 0)
- size += multsize - (size % multsize);
- return size;
- }
- /**
- * List callback function for comparing storage elements
- * @param a pointer to the current content in the tree (storageElement*)
- * @param b pointer to the memory to free
- * @return boolean indicating whether a and b are equal
- */
- static int ptrCompare(void* a, void* b, int value)
- {
- a = ((storageElement*)a)->ptr;
- if (value)
- b = ((storageElement*)b)->ptr;
- return (a > b) ? -1 : (a == b) ? 0 : 1;
- }
- /*
- static void Heap_check(char* string, void* ptr)
- {
- Node* curnode = NULL;
- storageElement* prev, *s = NULL;
- printf("Heap_check start %p\n", ptr);
- while ((curnode = TreeNextElement(&heap, curnode)) != NULL)
- {
- prev = s;
- s = (storageElement*)(curnode->content);
- if (prev)
- {
- if (ptrCompare(s, prev, 1) != -1)
- {
- printf("%s: heap order error %d %p %p\n", string, ptrCompare(s, prev, 1), prev->ptr, s->ptr);
- exit(99);
- }
- else
- printf("%s: heap order good %d %p %p\n", string, ptrCompare(s, prev, 1), prev->ptr, s->ptr);
- }
- }
- }*/
- /**
- * Allocates a block of memory. A direct replacement for malloc, but keeps track of items
- * allocated in a list, so that free can check that a item is being freed correctly and that
- * we can check that all memory is freed at shutdown.
- * @param file use the __FILE__ macro to indicate which file this item was allocated in
- * @param line use the __LINE__ macro to indicate which line this item was allocated at
- * @param size the size of the item to be allocated
- * @return pointer to the allocated item, or NULL if there was an error
- */
- void* mymalloc(char* file, int line, size_t size)
- {
- storageElement* s = NULL;
- size_t space = sizeof(storageElement);
- size_t filenamelen = strlen(file)+1;
- Thread_lock_mutex(heap_mutex);
- size = Heap_roundup(size);
- if ((s = malloc(sizeof(storageElement))) == NULL)
- {
- Log(LOG_ERROR, 13, errmsg);
- return NULL;
- }
- s->size = size; /* size without eyecatchers */
- if ((s->file = malloc(filenamelen)) == NULL)
- {
- Log(LOG_ERROR, 13, errmsg);
- free(s);
- return NULL;
- }
- space += filenamelen;
- strcpy(s->file, file);
- #if defined(HEAP_STACK)
- #define STACK_LEN 300
- if ((s->stack = malloc(STACK_LEN)) == NULL)
- {
- Log(LOG_ERROR, 13, errmsg);
- free(s->file);
- free(s);
- return NULL;
- }
- StackTrace_get(Thread_getid(), s->stack, STACK_LEN);
- #endif
- s->line = line;
- /* Add space for eyecatcher at each end */
- if ((s->ptr = malloc(size + 2*sizeof(int))) == NULL)
- {
- Log(LOG_ERROR, 13, errmsg);
- free(s->file);
- free(s);
- return NULL;
- }
- space += size + 2*sizeof(int);
- *(int*)(s->ptr) = eyecatcher; /* start eyecatcher */
- *(int*)(((char*)(s->ptr)) + (sizeof(int) + size)) = eyecatcher; /* end eyecatcher */
- Log(TRACE_MAX, -1, "Allocating %d bytes in heap at file %s line %d ptr %p\n", (int)size, file, line, s->ptr);
- TreeAdd(&heap, s, space);
- state.current_size += size;
- if (state.current_size > state.max_size)
- state.max_size = state.current_size;
- Thread_unlock_mutex(heap_mutex);
- return ((int*)(s->ptr)) + 1; /* skip start eyecatcher */
- }
- static void checkEyecatchers(char* file, int line, void* p, size_t size)
- {
- int *sp = (int*)p;
- char *cp = (char*)p;
- int us;
- static const char *msg = "Invalid %s eyecatcher %d in heap item at file %s line %d";
- if ((us = *--sp) != eyecatcher)
- Log(LOG_ERROR, 13, msg, "start", us, file, line);
- cp += size;
- if ((us = *(int*)cp) != eyecatcher)
- Log(LOG_ERROR, 13, msg, "end", us, file, line);
- }
- /**
- * Remove an item from the recorded heap without actually freeing it.
- * Use sparingly!
- * @param file use the __FILE__ macro to indicate which file this item was allocated in
- * @param line use the __LINE__ macro to indicate which line this item was allocated at
- * @param p pointer to the item to be removed
- */
- static int Internal_heap_unlink(char* file, int line, void* p)
- {
- Node* e = NULL;
- int rc = 0;
- e = TreeFind(&heap, ((int*)p)-1);
- if (e == NULL)
- Log(LOG_ERROR, 13, "Failed to remove heap item at file %s line %d", file, line);
- else
- {
- storageElement* s = (storageElement*)(e->content);
- Log(TRACE_MAX, -1, "Freeing %d bytes in heap at file %s line %d, heap use now %d bytes\n",
- (int)s->size, file, line, (int)state.current_size);
- checkEyecatchers(file, line, p, s->size);
- /* free(s->ptr); */
- free(s->file);
- state.current_size -= s->size;
- TreeRemoveNodeIndex(&heap, e, 0);
- free(s);
- rc = 1;
- }
- return rc;
- }
- /**
- * Frees a block of memory. A direct replacement for free, but checks that a item is in
- * the allocates list first.
- * @param file use the __FILE__ macro to indicate which file this item was allocated in
- * @param line use the __LINE__ macro to indicate which line this item was allocated at
- * @param p pointer to the item to be freed
- */
- void myfree(char* file, int line, void* p)
- {
- if (p) /* it is legal und usual to call free(NULL) */
- {
- Thread_lock_mutex(heap_mutex);
- if (Internal_heap_unlink(file, line, p))
- free(((int*)p)-1);
- Thread_unlock_mutex(heap_mutex);
- }
- else
- {
- Log(LOG_ERROR, -1, "Call of free(NULL) in %s,%d",file,line);
- }
- }
- /**
- * Remove an item from the recorded heap without actually freeing it.
- * Use sparingly!
- * @param file use the __FILE__ macro to indicate which file this item was allocated in
- * @param line use the __LINE__ macro to indicate which line this item was allocated at
- * @param p pointer to the item to be removed
- */
- void Heap_unlink(char* file, int line, void* p)
- {
- Thread_lock_mutex(heap_mutex);
- Internal_heap_unlink(file, line, p);
- Thread_unlock_mutex(heap_mutex);
- }
- /**
- * Reallocates a block of memory. A direct replacement for realloc, but keeps track of items
- * allocated in a list, so that free can check that a item is being freed correctly and that
- * we can check that all memory is freed at shutdown.
- * We have to remove the item from the tree, as the memory is in order and so it needs to
- * be reinserted in the correct place.
- * @param file use the __FILE__ macro to indicate which file this item was reallocated in
- * @param line use the __LINE__ macro to indicate which line this item was reallocated at
- * @param p pointer to the item to be reallocated
- * @param size the new size of the item
- * @return pointer to the allocated item, or NULL if there was an error
- */
- void *myrealloc(char* file, int line, void* p, size_t size)
- {
- void* rc = NULL;
- storageElement* s = NULL;
- Thread_lock_mutex(heap_mutex);
- s = TreeRemoveKey(&heap, ((int*)p)-1);
- if (s == NULL)
- Log(LOG_ERROR, 13, "Failed to reallocate heap item at file %s line %d", file, line);
- else
- {
- size_t space = sizeof(storageElement);
- size_t filenamelen = strlen(file)+1;
- checkEyecatchers(file, line, p, s->size);
- size = Heap_roundup(size);
- state.current_size += size - s->size;
- if (state.current_size > state.max_size)
- state.max_size = state.current_size;
- if ((s->ptr = realloc(s->ptr, size + 2*sizeof(int))) == NULL)
- {
- Log(LOG_ERROR, 13, errmsg);
- return NULL;
- }
- space += size + 2*sizeof(int) - s->size;
- *(int*)(s->ptr) = eyecatcher; /* start eyecatcher */
- *(int*)(((char*)(s->ptr)) + (sizeof(int) + size)) = eyecatcher; /* end eyecatcher */
- s->size = size;
- space -= strlen(s->file);
- s->file = realloc(s->file, filenamelen);
- space += filenamelen;
- strcpy(s->file, file);
- s->line = line;
- rc = s->ptr;
- TreeAdd(&heap, s, space);
- }
- Thread_unlock_mutex(heap_mutex);
- return (rc == NULL) ? NULL : ((int*)(rc)) + 1; /* skip start eyecatcher */
- }
- /**
- * Utility to find an item in the heap. Lets you know if the heap already contains
- * the memory location in question.
- * @param p pointer to a memory location
- * @return pointer to the storage element if found, or NULL
- */
- void* Heap_findItem(void* p)
- {
- Node* e = NULL;
- Thread_lock_mutex(heap_mutex);
- e = TreeFind(&heap, ((int*)p)-1);
- Thread_unlock_mutex(heap_mutex);
- return (e == NULL) ? NULL : e->content;
- }
- /**
- * Scans the heap and reports any items currently allocated.
- * To be used at shutdown if any heap items have not been freed.
- */
- static void HeapScan(enum LOG_LEVELS log_level)
- {
- Node* current = NULL;
- Thread_lock_mutex(heap_mutex);
- Log(log_level, -1, "Heap scan start, total %d bytes", (int)state.current_size);
- while ((current = TreeNextElement(&heap, current)) != NULL)
- {
- storageElement* s = (storageElement*)(current->content);
- Log(log_level, -1, "Heap element size %d, line %d, file %s, ptr %p", (int)s->size, s->line, s->file, s->ptr);
- Log(log_level, -1, " Content %.*s", (10 > current->size) ? (int)s->size : 10, (char*)(((int*)s->ptr) + 1));
- #if defined(HEAP_STACK)
- Log(log_level, -1, " Stack:\n%s", s->stack);
- #endif
- }
- Log(log_level, -1, "Heap scan end");
- Thread_unlock_mutex(heap_mutex);
- }
- /**
- * Heap initialization.
- */
- int Heap_initialize(void)
- {
- TreeInitializeNoMalloc(&heap, ptrCompare);
- heap.heap_tracking = 0; /* no recursive heap tracking! */
- return 0;
- }
- /**
- * Heap termination.
- */
- void Heap_terminate(void)
- {
- Log(TRACE_MIN, -1, "Maximum heap use was %d bytes", (int)state.max_size);
- if (state.current_size > 20) /* One log list is freed after this function is called */
- {
- Log(LOG_ERROR, -1, "Some memory not freed at shutdown, possible memory leak");
- HeapScan(LOG_ERROR);
- }
- }
- /**
- * Access to heap state
- * @return pointer to the heap state structure
- */
- heap_info* Heap_get_info(void)
- {
- return &state;
- }
- /**
- * Dump a string from the heap so that it can be displayed conveniently
- * @param file file handle to dump the heap contents to
- * @param str the string to dump, could be NULL
- */
- int HeapDumpString(FILE* file, char* str)
- {
- int rc = 0;
- size_t len = str ? strlen(str) + 1 : 0; /* include the trailing null */
- if (fwrite(&(str), sizeof(char*), 1, file) != 1)
- rc = -1;
- else if (fwrite(&(len), sizeof(int), 1 ,file) != 1)
- rc = -1;
- else if (len > 0 && fwrite(str, len, 1, file) != 1)
- rc = -1;
- return rc;
- }
- /**
- * Dump the state of the heap
- * @param file file handle to dump the heap contents to
- */
- int HeapDump(FILE* file)
- {
- int rc = 0;
- Node* current = NULL;
- while (rc == 0 && (current = TreeNextElement(&heap, current)))
- {
- storageElement* s = (storageElement*)(current->content);
- if (fwrite(&(s->ptr), sizeof(s->ptr), 1, file) != 1)
- rc = -1;
- else if (fwrite(&(current->size), sizeof(current->size), 1, file) != 1)
- rc = -1;
- else if (fwrite(s->ptr, current->size, 1, file) != 1)
- rc = -1;
- }
- return rc;
- }
- #if defined(HEAP_UNIT_TESTS)
- void Log(enum LOG_LEVELS log_level, int msgno, char* format, ...)
- {
- printf("Log %s", format);
- }
- char* Broker_recordFFDC(char* symptoms)
- {
- printf("recordFFDC");
- return "";
- }
- #define malloc(x) mymalloc(__FILE__, __LINE__, x)
- #define realloc(a, b) myrealloc(__FILE__, __LINE__, a, b)
- #define free(x) myfree(__FILE__, __LINE__, x)
- int main(int argc, char *argv[])
- {
- char* h = NULL;
- Heap_initialize();
- h = malloc(12);
- free(h);
- printf("freed h\n");
- h = malloc(12);
- h = realloc(h, 14);
- h = realloc(h, 25);
- h = realloc(h, 255);
- h = realloc(h, 2225);
- h = realloc(h, 22225);
- printf("freeing h\n");
- free(h);
- Heap_terminate();
- printf("Finishing\n");
- return 0;
- }
- #endif /* HEAP_UNIT_TESTS */
- /* Local Variables: */
- /* indent-tabs-mode: t */
- /* c-basic-offset: 8 */
- /* End: */
|