os/persistentdata/persistentstorage/sql/SQLite/mem2.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/*
sl@0
     2
** 2007 August 15
sl@0
     3
**
sl@0
     4
** The author disclaims copyright to this source code.  In place of
sl@0
     5
** a legal notice, here is a blessing:
sl@0
     6
**
sl@0
     7
**    May you do good and not evil.
sl@0
     8
**    May you find forgiveness for yourself and forgive others.
sl@0
     9
**    May you share freely, never taking more than you give.
sl@0
    10
**
sl@0
    11
*************************************************************************
sl@0
    12
**
sl@0
    13
** This file contains low-level memory allocation drivers for when
sl@0
    14
** SQLite will use the standard C-library malloc/realloc/free interface
sl@0
    15
** to obtain the memory it needs while adding lots of additional debugging
sl@0
    16
** information to each allocation in order to help detect and fix memory
sl@0
    17
** leaks and memory usage errors.
sl@0
    18
**
sl@0
    19
** This file contains implementations of the low-level memory allocation
sl@0
    20
** routines specified in the sqlite3_mem_methods object.
sl@0
    21
**
sl@0
    22
** $Id: mem2.c,v 1.37 2008/07/25 08:49:00 danielk1977 Exp $
sl@0
    23
*/
sl@0
    24
#include "sqliteInt.h"
sl@0
    25
sl@0
    26
/*
sl@0
    27
** This version of the memory allocator is used only if the
sl@0
    28
** SQLITE_MEMDEBUG macro is defined
sl@0
    29
*/
sl@0
    30
#ifdef SQLITE_MEMDEBUG
sl@0
    31
sl@0
    32
/*
sl@0
    33
** The backtrace functionality is only available with GLIBC
sl@0
    34
*/
sl@0
    35
#ifdef __GLIBC__
sl@0
    36
  extern int backtrace(void**,int);
sl@0
    37
  extern void backtrace_symbols_fd(void*const*,int,int);
sl@0
    38
#else
sl@0
    39
# define backtrace(A,B) 0
sl@0
    40
# define backtrace_symbols_fd(A,B,C)
sl@0
    41
#endif
sl@0
    42
#include <stdio.h>
sl@0
    43
sl@0
    44
/*
sl@0
    45
** Each memory allocation looks like this:
sl@0
    46
**
sl@0
    47
**  ------------------------------------------------------------------------
sl@0
    48
**  | Title |  backtrace pointers |  MemBlockHdr |  allocation |  EndGuard |
sl@0
    49
**  ------------------------------------------------------------------------
sl@0
    50
**
sl@0
    51
** The application code sees only a pointer to the allocation.  We have
sl@0
    52
** to back up from the allocation pointer to find the MemBlockHdr.  The
sl@0
    53
** MemBlockHdr tells us the size of the allocation and the number of
sl@0
    54
** backtrace pointers.  There is also a guard word at the end of the
sl@0
    55
** MemBlockHdr.
sl@0
    56
*/
sl@0
    57
struct MemBlockHdr {
sl@0
    58
  i64 iSize;                          /* Size of this allocation */
sl@0
    59
  struct MemBlockHdr *pNext, *pPrev;  /* Linked list of all unfreed memory */
sl@0
    60
  char nBacktrace;                    /* Number of backtraces on this alloc */
sl@0
    61
  char nBacktraceSlots;               /* Available backtrace slots */
sl@0
    62
  short nTitle;                       /* Bytes of title; includes '\0' */
sl@0
    63
  int iForeGuard;                     /* Guard word for sanity */
sl@0
    64
};
sl@0
    65
sl@0
    66
/*
sl@0
    67
** Guard words
sl@0
    68
*/
sl@0
    69
#define FOREGUARD 0x80F5E153
sl@0
    70
#define REARGUARD 0xE4676B53
sl@0
    71
sl@0
    72
/*
sl@0
    73
** Number of malloc size increments to track.
sl@0
    74
*/
sl@0
    75
#define NCSIZE  1000
sl@0
    76
sl@0
    77
/*
sl@0
    78
** All of the static variables used by this module are collected
sl@0
    79
** into a single structure named "mem".  This is to keep the
sl@0
    80
** static variables organized and to reduce namespace pollution
sl@0
    81
** when this module is combined with other in the amalgamation.
sl@0
    82
*/
sl@0
    83
static struct {
sl@0
    84
  
sl@0
    85
  /*
sl@0
    86
  ** Mutex to control access to the memory allocation subsystem.
sl@0
    87
  */
sl@0
    88
  sqlite3_mutex *mutex;
sl@0
    89
sl@0
    90
  /*
sl@0
    91
  ** Head and tail of a linked list of all outstanding allocations
sl@0
    92
  */
sl@0
    93
  struct MemBlockHdr *pFirst;
sl@0
    94
  struct MemBlockHdr *pLast;
sl@0
    95
  
sl@0
    96
  /*
sl@0
    97
  ** The number of levels of backtrace to save in new allocations.
sl@0
    98
  */
sl@0
    99
  int nBacktrace;
sl@0
   100
  void (*xBacktrace)(int, int, void **);
sl@0
   101
sl@0
   102
  /*
sl@0
   103
  ** Title text to insert in front of each block
sl@0
   104
  */
sl@0
   105
  int nTitle;        /* Bytes of zTitle to save.  Includes '\0' and padding */
sl@0
   106
  char zTitle[100];  /* The title text */
sl@0
   107
sl@0
   108
  /* 
sl@0
   109
  ** sqlite3MallocDisallow() increments the following counter.
sl@0
   110
  ** sqlite3MallocAllow() decrements it.
sl@0
   111
  */
sl@0
   112
  int disallow; /* Do not allow memory allocation */
sl@0
   113
sl@0
   114
  /*
sl@0
   115
  ** Gather statistics on the sizes of memory allocations.
sl@0
   116
  ** nAlloc[i] is the number of allocation attempts of i*8
sl@0
   117
  ** bytes.  i==NCSIZE is the number of allocation attempts for
sl@0
   118
  ** sizes more than NCSIZE*8 bytes.
sl@0
   119
  */
sl@0
   120
  int nAlloc[NCSIZE];      /* Total number of allocations */
sl@0
   121
  int nCurrent[NCSIZE];    /* Current number of allocations */
sl@0
   122
  int mxCurrent[NCSIZE];   /* Highwater mark for nCurrent */
sl@0
   123
sl@0
   124
} mem;
sl@0
   125
sl@0
   126
sl@0
   127
/*
sl@0
   128
** Adjust memory usage statistics
sl@0
   129
*/
sl@0
   130
static void adjustStats(int iSize, int increment){
sl@0
   131
  int i = ((iSize+7)&~7)/8;
sl@0
   132
  if( i>NCSIZE-1 ){
sl@0
   133
    i = NCSIZE - 1;
sl@0
   134
  }
sl@0
   135
  if( increment>0 ){
sl@0
   136
    mem.nAlloc[i]++;
sl@0
   137
    mem.nCurrent[i]++;
sl@0
   138
    if( mem.nCurrent[i]>mem.mxCurrent[i] ){
sl@0
   139
      mem.mxCurrent[i] = mem.nCurrent[i];
sl@0
   140
    }
sl@0
   141
  }else{
sl@0
   142
    mem.nCurrent[i]--;
sl@0
   143
    assert( mem.nCurrent[i]>=0 );
sl@0
   144
  }
sl@0
   145
}
sl@0
   146
sl@0
   147
/*
sl@0
   148
** Given an allocation, find the MemBlockHdr for that allocation.
sl@0
   149
**
sl@0
   150
** This routine checks the guards at either end of the allocation and
sl@0
   151
** if they are incorrect it asserts.
sl@0
   152
*/
sl@0
   153
static struct MemBlockHdr *sqlite3MemsysGetHeader(void *pAllocation){
sl@0
   154
  struct MemBlockHdr *p;
sl@0
   155
  int *pInt;
sl@0
   156
  u8 *pU8;
sl@0
   157
  int nReserve;
sl@0
   158
sl@0
   159
  p = (struct MemBlockHdr*)pAllocation;
sl@0
   160
  p--;
sl@0
   161
  assert( p->iForeGuard==FOREGUARD );
sl@0
   162
  nReserve = (p->iSize+7)&~7;
sl@0
   163
  pInt = (int*)pAllocation;
sl@0
   164
  pU8 = (u8*)pAllocation;
sl@0
   165
  assert( pInt[nReserve/sizeof(int)]==REARGUARD );
sl@0
   166
  assert( (nReserve-0)<=p->iSize || pU8[nReserve-1]==0x65 );
sl@0
   167
  assert( (nReserve-1)<=p->iSize || pU8[nReserve-2]==0x65 );
sl@0
   168
  assert( (nReserve-2)<=p->iSize || pU8[nReserve-3]==0x65 );
sl@0
   169
  return p;
sl@0
   170
}
sl@0
   171
sl@0
   172
/*
sl@0
   173
** Return the number of bytes currently allocated at address p.
sl@0
   174
*/
sl@0
   175
static int sqlite3MemSize(void *p){
sl@0
   176
  struct MemBlockHdr *pHdr;
sl@0
   177
  if( !p ){
sl@0
   178
    return 0;
sl@0
   179
  }
sl@0
   180
  pHdr = sqlite3MemsysGetHeader(p);
sl@0
   181
  return pHdr->iSize;
sl@0
   182
}
sl@0
   183
sl@0
   184
/*
sl@0
   185
** Initialize the memory allocation subsystem.
sl@0
   186
*/
sl@0
   187
static int sqlite3MemInit(void *NotUsed){
sl@0
   188
  if( !sqlite3Config.bMemstat ){
sl@0
   189
    /* If memory status is enabled, then the malloc.c wrapper will already
sl@0
   190
    ** hold the STATIC_MEM mutex when the routines here are invoked. */
sl@0
   191
    mem.mutex = sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_MEM);
sl@0
   192
  }
sl@0
   193
  return SQLITE_OK;
sl@0
   194
}
sl@0
   195
sl@0
   196
/*
sl@0
   197
** Deinitialize the memory allocation subsystem.
sl@0
   198
*/
sl@0
   199
static void sqlite3MemShutdown(void *NotUsed){
sl@0
   200
  mem.mutex = 0;
sl@0
   201
}
sl@0
   202
sl@0
   203
/*
sl@0
   204
** Round up a request size to the next valid allocation size.
sl@0
   205
*/
sl@0
   206
static int sqlite3MemRoundup(int n){
sl@0
   207
  return (n+7) & ~7;
sl@0
   208
}
sl@0
   209
sl@0
   210
/*
sl@0
   211
** Allocate nByte bytes of memory.
sl@0
   212
*/
sl@0
   213
static void *sqlite3MemMalloc(int nByte){
sl@0
   214
  struct MemBlockHdr *pHdr;
sl@0
   215
  void **pBt;
sl@0
   216
  char *z;
sl@0
   217
  int *pInt;
sl@0
   218
  void *p = 0;
sl@0
   219
  int totalSize;
sl@0
   220
  int nReserve;
sl@0
   221
  sqlite3_mutex_enter(mem.mutex);
sl@0
   222
  assert( mem.disallow==0 );
sl@0
   223
  nReserve = (nByte+7)&~7;
sl@0
   224
  totalSize = nReserve + sizeof(*pHdr) + sizeof(int) +
sl@0
   225
               mem.nBacktrace*sizeof(void*) + mem.nTitle;
sl@0
   226
  p = malloc(totalSize);
sl@0
   227
  if( p ){
sl@0
   228
    z = p;
sl@0
   229
    pBt = (void**)&z[mem.nTitle];
sl@0
   230
    pHdr = (struct MemBlockHdr*)&pBt[mem.nBacktrace];
sl@0
   231
    pHdr->pNext = 0;
sl@0
   232
    pHdr->pPrev = mem.pLast;
sl@0
   233
    if( mem.pLast ){
sl@0
   234
      mem.pLast->pNext = pHdr;
sl@0
   235
    }else{
sl@0
   236
      mem.pFirst = pHdr;
sl@0
   237
    }
sl@0
   238
    mem.pLast = pHdr;
sl@0
   239
    pHdr->iForeGuard = FOREGUARD;
sl@0
   240
    pHdr->nBacktraceSlots = mem.nBacktrace;
sl@0
   241
    pHdr->nTitle = mem.nTitle;
sl@0
   242
    if( mem.nBacktrace ){
sl@0
   243
      void *aAddr[40];
sl@0
   244
      pHdr->nBacktrace = backtrace(aAddr, mem.nBacktrace+1)-1;
sl@0
   245
      memcpy(pBt, &aAddr[1], pHdr->nBacktrace*sizeof(void*));
sl@0
   246
      if( mem.xBacktrace ){
sl@0
   247
        mem.xBacktrace(nByte, pHdr->nBacktrace-1, &aAddr[1]);
sl@0
   248
      }
sl@0
   249
    }else{
sl@0
   250
      pHdr->nBacktrace = 0;
sl@0
   251
    }
sl@0
   252
    if( mem.nTitle ){
sl@0
   253
      memcpy(z, mem.zTitle, mem.nTitle);
sl@0
   254
    }
sl@0
   255
    pHdr->iSize = nByte;
sl@0
   256
    adjustStats(nByte, +1);
sl@0
   257
    pInt = (int*)&pHdr[1];
sl@0
   258
    pInt[nReserve/sizeof(int)] = REARGUARD;
sl@0
   259
    memset(pInt, 0x65, nReserve);
sl@0
   260
    p = (void*)pInt;
sl@0
   261
  }
sl@0
   262
  sqlite3_mutex_leave(mem.mutex);
sl@0
   263
  return p; 
sl@0
   264
}
sl@0
   265
sl@0
   266
/*
sl@0
   267
** Free memory.
sl@0
   268
*/
sl@0
   269
static void sqlite3MemFree(void *pPrior){
sl@0
   270
  struct MemBlockHdr *pHdr;
sl@0
   271
  void **pBt;
sl@0
   272
  char *z;
sl@0
   273
  assert( sqlite3Config.bMemstat || mem.mutex!=0 );
sl@0
   274
  pHdr = sqlite3MemsysGetHeader(pPrior);
sl@0
   275
  pBt = (void**)pHdr;
sl@0
   276
  pBt -= pHdr->nBacktraceSlots;
sl@0
   277
  sqlite3_mutex_enter(mem.mutex);
sl@0
   278
  if( pHdr->pPrev ){
sl@0
   279
    assert( pHdr->pPrev->pNext==pHdr );
sl@0
   280
    pHdr->pPrev->pNext = pHdr->pNext;
sl@0
   281
  }else{
sl@0
   282
    assert( mem.pFirst==pHdr );
sl@0
   283
    mem.pFirst = pHdr->pNext;
sl@0
   284
  }
sl@0
   285
  if( pHdr->pNext ){
sl@0
   286
    assert( pHdr->pNext->pPrev==pHdr );
sl@0
   287
    pHdr->pNext->pPrev = pHdr->pPrev;
sl@0
   288
  }else{
sl@0
   289
    assert( mem.pLast==pHdr );
sl@0
   290
    mem.pLast = pHdr->pPrev;
sl@0
   291
  }
sl@0
   292
  z = (char*)pBt;
sl@0
   293
  z -= pHdr->nTitle;
sl@0
   294
  adjustStats(pHdr->iSize, -1);
sl@0
   295
  memset(z, 0x2b, sizeof(void*)*pHdr->nBacktraceSlots + sizeof(*pHdr) +
sl@0
   296
                  pHdr->iSize + sizeof(int) + pHdr->nTitle);
sl@0
   297
  free(z);
sl@0
   298
  sqlite3_mutex_leave(mem.mutex);  
sl@0
   299
}
sl@0
   300
sl@0
   301
/*
sl@0
   302
** Change the size of an existing memory allocation.
sl@0
   303
**
sl@0
   304
** For this debugging implementation, we *always* make a copy of the
sl@0
   305
** allocation into a new place in memory.  In this way, if the 
sl@0
   306
** higher level code is using pointer to the old allocation, it is 
sl@0
   307
** much more likely to break and we are much more liking to find
sl@0
   308
** the error.
sl@0
   309
*/
sl@0
   310
static void *sqlite3MemRealloc(void *pPrior, int nByte){
sl@0
   311
  struct MemBlockHdr *pOldHdr;
sl@0
   312
  void *pNew;
sl@0
   313
  assert( mem.disallow==0 );
sl@0
   314
  pOldHdr = sqlite3MemsysGetHeader(pPrior);
sl@0
   315
  pNew = sqlite3MemMalloc(nByte);
sl@0
   316
  if( pNew ){
sl@0
   317
    memcpy(pNew, pPrior, nByte<pOldHdr->iSize ? nByte : pOldHdr->iSize);
sl@0
   318
    if( nByte>pOldHdr->iSize ){
sl@0
   319
      memset(&((char*)pNew)[pOldHdr->iSize], 0x2b, nByte - pOldHdr->iSize);
sl@0
   320
    }
sl@0
   321
    sqlite3MemFree(pPrior);
sl@0
   322
  }
sl@0
   323
  return pNew;
sl@0
   324
}
sl@0
   325
sl@0
   326
sl@0
   327
const sqlite3_mem_methods *sqlite3MemGetDefault(void){
sl@0
   328
  static const sqlite3_mem_methods defaultMethods = {
sl@0
   329
     sqlite3MemMalloc,
sl@0
   330
     sqlite3MemFree,
sl@0
   331
     sqlite3MemRealloc,
sl@0
   332
     sqlite3MemSize,
sl@0
   333
     sqlite3MemRoundup,
sl@0
   334
     sqlite3MemInit,
sl@0
   335
     sqlite3MemShutdown,
sl@0
   336
     0
sl@0
   337
  };
sl@0
   338
  return &defaultMethods;
sl@0
   339
}
sl@0
   340
sl@0
   341
/*
sl@0
   342
** Populate the low-level memory allocation function pointers in
sl@0
   343
** sqlite3Config.m with pointers to the routines in this file.
sl@0
   344
*/
sl@0
   345
void sqlite3MemSetDefault(void){
sl@0
   346
  sqlite3_config(SQLITE_CONFIG_MALLOC, sqlite3MemGetDefault());
sl@0
   347
}
sl@0
   348
sl@0
   349
/*
sl@0
   350
** Set the number of backtrace levels kept for each allocation.
sl@0
   351
** A value of zero turns off backtracing.  The number is always rounded
sl@0
   352
** up to a multiple of 2.
sl@0
   353
*/
sl@0
   354
void sqlite3MemdebugBacktrace(int depth){
sl@0
   355
  if( depth<0 ){ depth = 0; }
sl@0
   356
  if( depth>20 ){ depth = 20; }
sl@0
   357
  depth = (depth+1)&0xfe;
sl@0
   358
  mem.nBacktrace = depth;
sl@0
   359
}
sl@0
   360
sl@0
   361
void sqlite3MemdebugBacktraceCallback(void (*xBacktrace)(int, int, void **)){
sl@0
   362
  mem.xBacktrace = xBacktrace;
sl@0
   363
}
sl@0
   364
sl@0
   365
/*
sl@0
   366
** Set the title string for subsequent allocations.
sl@0
   367
*/
sl@0
   368
void sqlite3MemdebugSettitle(const char *zTitle){
sl@0
   369
  int n = strlen(zTitle) + 1;
sl@0
   370
  sqlite3_mutex_enter(mem.mutex);
sl@0
   371
  if( n>=sizeof(mem.zTitle) ) n = sizeof(mem.zTitle)-1;
sl@0
   372
  memcpy(mem.zTitle, zTitle, n);
sl@0
   373
  mem.zTitle[n] = 0;
sl@0
   374
  mem.nTitle = (n+7)&~7;
sl@0
   375
  sqlite3_mutex_leave(mem.mutex);
sl@0
   376
}
sl@0
   377
sl@0
   378
void sqlite3MemdebugSync(){
sl@0
   379
  struct MemBlockHdr *pHdr;
sl@0
   380
  for(pHdr=mem.pFirst; pHdr; pHdr=pHdr->pNext){
sl@0
   381
    void **pBt = (void**)pHdr;
sl@0
   382
    pBt -= pHdr->nBacktraceSlots;
sl@0
   383
    mem.xBacktrace(pHdr->iSize, pHdr->nBacktrace-1, &pBt[1]);
sl@0
   384
  }
sl@0
   385
}
sl@0
   386
sl@0
   387
/*
sl@0
   388
** Open the file indicated and write a log of all unfreed memory 
sl@0
   389
** allocations into that log.
sl@0
   390
*/
sl@0
   391
void sqlite3MemdebugDump(const char *zFilename){
sl@0
   392
  FILE *out;
sl@0
   393
  struct MemBlockHdr *pHdr;
sl@0
   394
  void **pBt;
sl@0
   395
  int i;
sl@0
   396
  out = fopen(zFilename, "w");
sl@0
   397
  if( out==0 ){
sl@0
   398
    fprintf(stderr, "** Unable to output memory debug output log: %s **\n",
sl@0
   399
                    zFilename);
sl@0
   400
    return;
sl@0
   401
  }
sl@0
   402
  for(pHdr=mem.pFirst; pHdr; pHdr=pHdr->pNext){
sl@0
   403
    char *z = (char*)pHdr;
sl@0
   404
    z -= pHdr->nBacktraceSlots*sizeof(void*) + pHdr->nTitle;
sl@0
   405
    fprintf(out, "**** %lld bytes at %p from %s ****\n", 
sl@0
   406
            pHdr->iSize, &pHdr[1], pHdr->nTitle ? z : "???");
sl@0
   407
    if( pHdr->nBacktrace ){
sl@0
   408
      fflush(out);
sl@0
   409
      pBt = (void**)pHdr;
sl@0
   410
      pBt -= pHdr->nBacktraceSlots;
sl@0
   411
      backtrace_symbols_fd(pBt, pHdr->nBacktrace, fileno(out));
sl@0
   412
      fprintf(out, "\n");
sl@0
   413
    }
sl@0
   414
  }
sl@0
   415
  fprintf(out, "COUNTS:\n");
sl@0
   416
  for(i=0; i<NCSIZE-1; i++){
sl@0
   417
    if( mem.nAlloc[i] ){
sl@0
   418
      fprintf(out, "   %5d: %10d %10d %10d\n", 
sl@0
   419
            i*8, mem.nAlloc[i], mem.nCurrent[i], mem.mxCurrent[i]);
sl@0
   420
    }
sl@0
   421
  }
sl@0
   422
  if( mem.nAlloc[NCSIZE-1] ){
sl@0
   423
    fprintf(out, "   %5d: %10d %10d %10d\n",
sl@0
   424
             NCSIZE*8-8, mem.nAlloc[NCSIZE-1],
sl@0
   425
             mem.nCurrent[NCSIZE-1], mem.mxCurrent[NCSIZE-1]);
sl@0
   426
  }
sl@0
   427
  fclose(out);
sl@0
   428
}
sl@0
   429
sl@0
   430
/*
sl@0
   431
** Return the number of times sqlite3MemMalloc() has been called.
sl@0
   432
*/
sl@0
   433
int sqlite3MemdebugMallocCount(){
sl@0
   434
  int i;
sl@0
   435
  int nTotal = 0;
sl@0
   436
  for(i=0; i<NCSIZE; i++){
sl@0
   437
    nTotal += mem.nAlloc[i];
sl@0
   438
  }
sl@0
   439
  return nTotal;
sl@0
   440
}
sl@0
   441
sl@0
   442
sl@0
   443
#endif /* SQLITE_MEMDEBUG */