sl@0: /* sl@0: ** 2007 August 15 sl@0: ** sl@0: ** The author disclaims copyright to this source code. In place of sl@0: ** a legal notice, here is a blessing: sl@0: ** sl@0: ** May you do good and not evil. sl@0: ** May you find forgiveness for yourself and forgive others. sl@0: ** May you share freely, never taking more than you give. sl@0: ** sl@0: ************************************************************************* sl@0: ** sl@0: ** This file contains low-level memory allocation drivers for when sl@0: ** SQLite will use the standard C-library malloc/realloc/free interface sl@0: ** to obtain the memory it needs while adding lots of additional debugging sl@0: ** information to each allocation in order to help detect and fix memory sl@0: ** leaks and memory usage errors. sl@0: ** sl@0: ** This file contains implementations of the low-level memory allocation sl@0: ** routines specified in the sqlite3_mem_methods object. sl@0: ** sl@0: ** $Id: mem2.c,v 1.39 2008/09/01 18:34:20 danielk1977 Exp $ sl@0: */ sl@0: #include "sqliteInt.h" sl@0: sl@0: /* sl@0: ** This version of the memory allocator is used only if the sl@0: ** SQLITE_MEMDEBUG macro is defined sl@0: */ sl@0: #ifdef SQLITE_MEMDEBUG sl@0: sl@0: /* sl@0: ** The backtrace functionality is only available with GLIBC sl@0: */ sl@0: #ifdef __GLIBC__ sl@0: extern int backtrace(void**,int); sl@0: extern void backtrace_symbols_fd(void*const*,int,int); sl@0: #else sl@0: # define backtrace(A,B) 1 sl@0: # define backtrace_symbols_fd(A,B,C) sl@0: #endif sl@0: #include sl@0: sl@0: /* sl@0: ** Each memory allocation looks like this: sl@0: ** sl@0: ** ------------------------------------------------------------------------ sl@0: ** | Title | backtrace pointers | MemBlockHdr | allocation | EndGuard | sl@0: ** ------------------------------------------------------------------------ sl@0: ** sl@0: ** The application code sees only a pointer to the allocation. We have sl@0: ** to back up from the allocation pointer to find the MemBlockHdr. The sl@0: ** MemBlockHdr tells us the size of the allocation and the number of sl@0: ** backtrace pointers. There is also a guard word at the end of the sl@0: ** MemBlockHdr. sl@0: */ sl@0: struct MemBlockHdr { sl@0: i64 iSize; /* Size of this allocation */ sl@0: struct MemBlockHdr *pNext, *pPrev; /* Linked list of all unfreed memory */ sl@0: char nBacktrace; /* Number of backtraces on this alloc */ sl@0: char nBacktraceSlots; /* Available backtrace slots */ sl@0: short nTitle; /* Bytes of title; includes '\0' */ sl@0: int iForeGuard; /* Guard word for sanity */ sl@0: }; sl@0: sl@0: /* sl@0: ** Guard words sl@0: */ sl@0: #define FOREGUARD 0x80F5E153 sl@0: #define REARGUARD 0xE4676B53 sl@0: sl@0: /* sl@0: ** Number of malloc size increments to track. sl@0: */ sl@0: #define NCSIZE 1000 sl@0: sl@0: /* sl@0: ** All of the static variables used by this module are collected sl@0: ** into a single structure named "mem". This is to keep the sl@0: ** static variables organized and to reduce namespace pollution sl@0: ** when this module is combined with other in the amalgamation. sl@0: */ sl@0: static struct { sl@0: sl@0: /* sl@0: ** Mutex to control access to the memory allocation subsystem. sl@0: */ sl@0: sqlite3_mutex *mutex; sl@0: sl@0: /* sl@0: ** Head and tail of a linked list of all outstanding allocations sl@0: */ sl@0: struct MemBlockHdr *pFirst; sl@0: struct MemBlockHdr *pLast; sl@0: sl@0: /* sl@0: ** The number of levels of backtrace to save in new allocations. sl@0: */ sl@0: int nBacktrace; sl@0: void (*xBacktrace)(int, int, void **); sl@0: sl@0: /* sl@0: ** Title text to insert in front of each block sl@0: */ sl@0: int nTitle; /* Bytes of zTitle to save. Includes '\0' and padding */ sl@0: char zTitle[100]; /* The title text */ sl@0: sl@0: /* sl@0: ** sqlite3MallocDisallow() increments the following counter. sl@0: ** sqlite3MallocAllow() decrements it. sl@0: */ sl@0: int disallow; /* Do not allow memory allocation */ sl@0: sl@0: /* sl@0: ** Gather statistics on the sizes of memory allocations. sl@0: ** nAlloc[i] is the number of allocation attempts of i*8 sl@0: ** bytes. i==NCSIZE is the number of allocation attempts for sl@0: ** sizes more than NCSIZE*8 bytes. sl@0: */ sl@0: int nAlloc[NCSIZE]; /* Total number of allocations */ sl@0: int nCurrent[NCSIZE]; /* Current number of allocations */ sl@0: int mxCurrent[NCSIZE]; /* Highwater mark for nCurrent */ sl@0: sl@0: } mem; sl@0: sl@0: sl@0: /* sl@0: ** Adjust memory usage statistics sl@0: */ sl@0: static void adjustStats(int iSize, int increment){ sl@0: int i = ((iSize+7)&~7)/8; sl@0: if( i>NCSIZE-1 ){ sl@0: i = NCSIZE - 1; sl@0: } sl@0: if( increment>0 ){ sl@0: mem.nAlloc[i]++; sl@0: mem.nCurrent[i]++; sl@0: if( mem.nCurrent[i]>mem.mxCurrent[i] ){ sl@0: mem.mxCurrent[i] = mem.nCurrent[i]; sl@0: } sl@0: }else{ sl@0: mem.nCurrent[i]--; sl@0: assert( mem.nCurrent[i]>=0 ); sl@0: } sl@0: } sl@0: sl@0: /* sl@0: ** Given an allocation, find the MemBlockHdr for that allocation. sl@0: ** sl@0: ** This routine checks the guards at either end of the allocation and sl@0: ** if they are incorrect it asserts. sl@0: */ sl@0: static struct MemBlockHdr *sqlite3MemsysGetHeader(void *pAllocation){ sl@0: struct MemBlockHdr *p; sl@0: int *pInt; sl@0: u8 *pU8; sl@0: int nReserve; sl@0: sl@0: p = (struct MemBlockHdr*)pAllocation; sl@0: p--; sl@0: assert( p->iForeGuard==FOREGUARD ); sl@0: nReserve = (p->iSize+7)&~7; sl@0: pInt = (int*)pAllocation; sl@0: pU8 = (u8*)pAllocation; sl@0: assert( pInt[nReserve/sizeof(int)]==REARGUARD ); sl@0: assert( (nReserve-0)<=p->iSize || pU8[nReserve-1]==0x65 ); sl@0: assert( (nReserve-1)<=p->iSize || pU8[nReserve-2]==0x65 ); sl@0: assert( (nReserve-2)<=p->iSize || pU8[nReserve-3]==0x65 ); sl@0: return p; sl@0: } sl@0: sl@0: /* sl@0: ** Return the number of bytes currently allocated at address p. sl@0: */ sl@0: static int sqlite3MemSize(void *p){ sl@0: struct MemBlockHdr *pHdr; sl@0: if( !p ){ sl@0: return 0; sl@0: } sl@0: pHdr = sqlite3MemsysGetHeader(p); sl@0: return pHdr->iSize; sl@0: } sl@0: sl@0: /* sl@0: ** Initialize the memory allocation subsystem. sl@0: */ sl@0: static int sqlite3MemInit(void *NotUsed){ sl@0: if( !sqlite3GlobalConfig.bMemstat ){ sl@0: /* If memory status is enabled, then the malloc.c wrapper will already sl@0: ** hold the STATIC_MEM mutex when the routines here are invoked. */ sl@0: mem.mutex = sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_MEM); sl@0: } sl@0: return SQLITE_OK; sl@0: } sl@0: sl@0: /* sl@0: ** Deinitialize the memory allocation subsystem. sl@0: */ sl@0: static void sqlite3MemShutdown(void *NotUsed){ sl@0: mem.mutex = 0; sl@0: } sl@0: sl@0: /* sl@0: ** Round up a request size to the next valid allocation size. sl@0: */ sl@0: static int sqlite3MemRoundup(int n){ sl@0: return (n+7) & ~7; sl@0: } sl@0: sl@0: /* sl@0: ** Allocate nByte bytes of memory. sl@0: */ sl@0: static void *sqlite3MemMalloc(int nByte){ sl@0: struct MemBlockHdr *pHdr; sl@0: void **pBt; sl@0: char *z; sl@0: int *pInt; sl@0: void *p = 0; sl@0: int totalSize; sl@0: int nReserve; sl@0: sqlite3_mutex_enter(mem.mutex); sl@0: assert( mem.disallow==0 ); sl@0: nReserve = (nByte+7)&~7; sl@0: totalSize = nReserve + sizeof(*pHdr) + sizeof(int) + sl@0: mem.nBacktrace*sizeof(void*) + mem.nTitle; sl@0: p = malloc(totalSize); sl@0: if( p ){ sl@0: z = p; sl@0: pBt = (void**)&z[mem.nTitle]; sl@0: pHdr = (struct MemBlockHdr*)&pBt[mem.nBacktrace]; sl@0: pHdr->pNext = 0; sl@0: pHdr->pPrev = mem.pLast; sl@0: if( mem.pLast ){ sl@0: mem.pLast->pNext = pHdr; sl@0: }else{ sl@0: mem.pFirst = pHdr; sl@0: } sl@0: mem.pLast = pHdr; sl@0: pHdr->iForeGuard = FOREGUARD; sl@0: pHdr->nBacktraceSlots = mem.nBacktrace; sl@0: pHdr->nTitle = mem.nTitle; sl@0: if( mem.nBacktrace ){ sl@0: void *aAddr[40]; sl@0: pHdr->nBacktrace = backtrace(aAddr, mem.nBacktrace+1)-1; sl@0: memcpy(pBt, &aAddr[1], pHdr->nBacktrace*sizeof(void*)); sl@0: if( mem.xBacktrace ){ sl@0: mem.xBacktrace(nByte, pHdr->nBacktrace-1, &aAddr[1]); sl@0: } sl@0: }else{ sl@0: pHdr->nBacktrace = 0; sl@0: } sl@0: if( mem.nTitle ){ sl@0: memcpy(z, mem.zTitle, mem.nTitle); sl@0: } sl@0: pHdr->iSize = nByte; sl@0: adjustStats(nByte, +1); sl@0: pInt = (int*)&pHdr[1]; sl@0: pInt[nReserve/sizeof(int)] = REARGUARD; sl@0: memset(pInt, 0x65, nReserve); sl@0: p = (void*)pInt; sl@0: } sl@0: sqlite3_mutex_leave(mem.mutex); sl@0: return p; sl@0: } sl@0: sl@0: /* sl@0: ** Free memory. sl@0: */ sl@0: static void sqlite3MemFree(void *pPrior){ sl@0: struct MemBlockHdr *pHdr; sl@0: void **pBt; sl@0: char *z; sl@0: assert( sqlite3GlobalConfig.bMemstat || mem.mutex!=0 ); sl@0: pHdr = sqlite3MemsysGetHeader(pPrior); sl@0: pBt = (void**)pHdr; sl@0: pBt -= pHdr->nBacktraceSlots; sl@0: sqlite3_mutex_enter(mem.mutex); sl@0: if( pHdr->pPrev ){ sl@0: assert( pHdr->pPrev->pNext==pHdr ); sl@0: pHdr->pPrev->pNext = pHdr->pNext; sl@0: }else{ sl@0: assert( mem.pFirst==pHdr ); sl@0: mem.pFirst = pHdr->pNext; sl@0: } sl@0: if( pHdr->pNext ){ sl@0: assert( pHdr->pNext->pPrev==pHdr ); sl@0: pHdr->pNext->pPrev = pHdr->pPrev; sl@0: }else{ sl@0: assert( mem.pLast==pHdr ); sl@0: mem.pLast = pHdr->pPrev; sl@0: } sl@0: z = (char*)pBt; sl@0: z -= pHdr->nTitle; sl@0: adjustStats(pHdr->iSize, -1); sl@0: memset(z, 0x2b, sizeof(void*)*pHdr->nBacktraceSlots + sizeof(*pHdr) + sl@0: pHdr->iSize + sizeof(int) + pHdr->nTitle); sl@0: free(z); sl@0: sqlite3_mutex_leave(mem.mutex); sl@0: } sl@0: sl@0: /* sl@0: ** Change the size of an existing memory allocation. sl@0: ** sl@0: ** For this debugging implementation, we *always* make a copy of the sl@0: ** allocation into a new place in memory. In this way, if the sl@0: ** higher level code is using pointer to the old allocation, it is sl@0: ** much more likely to break and we are much more liking to find sl@0: ** the error. sl@0: */ sl@0: static void *sqlite3MemRealloc(void *pPrior, int nByte){ sl@0: struct MemBlockHdr *pOldHdr; sl@0: void *pNew; sl@0: assert( mem.disallow==0 ); sl@0: pOldHdr = sqlite3MemsysGetHeader(pPrior); sl@0: pNew = sqlite3MemMalloc(nByte); sl@0: if( pNew ){ sl@0: memcpy(pNew, pPrior, nByteiSize ? nByte : pOldHdr->iSize); sl@0: if( nByte>pOldHdr->iSize ){ sl@0: memset(&((char*)pNew)[pOldHdr->iSize], 0x2b, nByte - pOldHdr->iSize); sl@0: } sl@0: sqlite3MemFree(pPrior); sl@0: } sl@0: return pNew; sl@0: } sl@0: sl@0: sl@0: const sqlite3_mem_methods *sqlite3MemGetDefault(void){ sl@0: static const sqlite3_mem_methods defaultMethods = { sl@0: sqlite3MemMalloc, sl@0: sqlite3MemFree, sl@0: sqlite3MemRealloc, sl@0: sqlite3MemSize, sl@0: sqlite3MemRoundup, sl@0: sqlite3MemInit, sl@0: sqlite3MemShutdown, sl@0: 0 sl@0: }; sl@0: return &defaultMethods; sl@0: } sl@0: sl@0: /* sl@0: ** Populate the low-level memory allocation function pointers in sl@0: ** sqlite3GlobalConfig.m with pointers to the routines in this file. sl@0: */ sl@0: void sqlite3MemSetDefault(void){ sl@0: sqlite3_config(SQLITE_CONFIG_MALLOC, sqlite3MemGetDefault()); sl@0: } sl@0: sl@0: /* sl@0: ** Set the number of backtrace levels kept for each allocation. sl@0: ** A value of zero turns off backtracing. The number is always rounded sl@0: ** up to a multiple of 2. sl@0: */ sl@0: void sqlite3MemdebugBacktrace(int depth){ sl@0: if( depth<0 ){ depth = 0; } sl@0: if( depth>20 ){ depth = 20; } sl@0: depth = (depth+1)&0xfe; sl@0: mem.nBacktrace = depth; sl@0: } sl@0: sl@0: void sqlite3MemdebugBacktraceCallback(void (*xBacktrace)(int, int, void **)){ sl@0: mem.xBacktrace = xBacktrace; sl@0: } sl@0: sl@0: /* sl@0: ** Set the title string for subsequent allocations. sl@0: */ sl@0: void sqlite3MemdebugSettitle(const char *zTitle){ sl@0: int n = strlen(zTitle) + 1; sl@0: sqlite3_mutex_enter(mem.mutex); sl@0: if( n>=sizeof(mem.zTitle) ) n = sizeof(mem.zTitle)-1; sl@0: memcpy(mem.zTitle, zTitle, n); sl@0: mem.zTitle[n] = 0; sl@0: mem.nTitle = (n+7)&~7; sl@0: sqlite3_mutex_leave(mem.mutex); sl@0: } sl@0: sl@0: void sqlite3MemdebugSync(){ sl@0: struct MemBlockHdr *pHdr; sl@0: for(pHdr=mem.pFirst; pHdr; pHdr=pHdr->pNext){ sl@0: void **pBt = (void**)pHdr; sl@0: pBt -= pHdr->nBacktraceSlots; sl@0: mem.xBacktrace(pHdr->iSize, pHdr->nBacktrace-1, &pBt[1]); sl@0: } sl@0: } sl@0: sl@0: /* sl@0: ** Open the file indicated and write a log of all unfreed memory sl@0: ** allocations into that log. sl@0: */ sl@0: void sqlite3MemdebugDump(const char *zFilename){ sl@0: FILE *out; sl@0: struct MemBlockHdr *pHdr; sl@0: void **pBt; sl@0: int i; sl@0: out = fopen(zFilename, "w"); sl@0: if( out==0 ){ sl@0: fprintf(stderr, "** Unable to output memory debug output log: %s **\n", sl@0: zFilename); sl@0: return; sl@0: } sl@0: for(pHdr=mem.pFirst; pHdr; pHdr=pHdr->pNext){ sl@0: char *z = (char*)pHdr; sl@0: z -= pHdr->nBacktraceSlots*sizeof(void*) + pHdr->nTitle; sl@0: fprintf(out, "**** %lld bytes at %p from %s ****\n", sl@0: pHdr->iSize, &pHdr[1], pHdr->nTitle ? z : "???"); sl@0: if( pHdr->nBacktrace ){ sl@0: fflush(out); sl@0: pBt = (void**)pHdr; sl@0: pBt -= pHdr->nBacktraceSlots; sl@0: backtrace_symbols_fd(pBt, pHdr->nBacktrace, fileno(out)); sl@0: fprintf(out, "\n"); sl@0: } sl@0: } sl@0: fprintf(out, "COUNTS:\n"); sl@0: for(i=0; i