| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530 | /******************************************************************************* * Copyright (c) 2009, 2020 IBM Corp. * * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v2.0 * and Eclipse Distribution License v1.0 which accompany this distribution. * * The Eclipse Public License is available at *    https://www.eclipse.org/legal/epl-2.0/ * 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 <inttypes.h>#include "Heap.h"#if !defined(NO_HEAP_TRACKING)#undef malloc#undef realloc#undef free#if defined(_WIN32) || defined(_WIN64)mutex_type heap_mutex;#elsestatic pthread_mutex_t heap_mutex_store = PTHREAD_MUTEX_INITIALIZER;static mutex_type heap_mutex = &heap_mutex_store;#endifstatic heap_info state = {0, 0}; /**< global heap state information */typedef uint64_t eyecatcherType;static eyecatcherType eyecatcher = (eyecatcherType)0x8888888888888888;#define PRIeyecatcher PRIx64  /**< print eyecatcher in HEX notation *//*#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;	void* rc = NULL;	Thread_lock_mutex(heap_mutex);	size = Heap_roundup(size);	if ((s = malloc(sizeof(storageElement))) == NULL)	{		Log(LOG_ERROR, 13, errmsg);		goto exit;	}	memset(s, 0, sizeof(storageElement));	s->size = size; /* size without eyecatchers */	if ((s->file = malloc(filenamelen)) == NULL)	{		Log(LOG_ERROR, 13, errmsg);		free(s);		goto exit;	}	memset(s->file, 0, sizeof(filenamelen));	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);		goto exit;	}	memset(s->stack, 0, sizeof(filenamelen));	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(eyecatcherType))) == NULL)	{		Log(LOG_ERROR, 13, errmsg);		free(s->file);		free(s);		goto exit;	}	memset(s->ptr, 0, size + 2*sizeof(eyecatcherType));	space += size + 2*sizeof(eyecatcherType);	*(eyecatcherType*)(s->ptr) = eyecatcher; /* start eyecatcher */	*(eyecatcherType*)(((char*)(s->ptr)) + (sizeof(eyecatcherType) + 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;	rc = ((eyecatcherType*)(s->ptr)) + 1;	/* skip start eyecatcher */exit:	Thread_unlock_mutex(heap_mutex);	return rc;}static void checkEyecatchers(char* file, int line, void* p, size_t size){	eyecatcherType *sp = (eyecatcherType*)p;	char *cp = (char*)p;	eyecatcherType us;	static const char *msg = "Invalid %s eyecatcher %" PRIeyecatcher " 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 = *(eyecatcherType*)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, ((eyecatcherType*)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(((eyecatcherType*)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, ((eyecatcherType*)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(eyecatcherType))) == NULL)		{			Log(LOG_ERROR, 13, errmsg);			goto exit;		}		space += size + 2*sizeof(eyecatcherType) - s->size;		*(eyecatcherType*)(s->ptr) = eyecatcher; /* start eyecatcher */		*(eyecatcherType*)(((char*)(s->ptr)) + (sizeof(eyecatcherType) + 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);	}exit:	Thread_unlock_mutex(heap_mutex);	return (rc == NULL) ? NULL : ((eyecatcherType*)(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, ((eyecatcherType*)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*)(((eyecatcherType*)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;}#endif#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: */
 |