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