xref: /sqlite-3.40.0/src/test_malloc.c (revision dbdc4d49)
12f999a67Sdrh /*
22f999a67Sdrh ** 2007 August 15
32f999a67Sdrh **
42f999a67Sdrh ** The author disclaims copyright to this source code.  In place of
52f999a67Sdrh ** a legal notice, here is a blessing:
62f999a67Sdrh **
72f999a67Sdrh **    May you do good and not evil.
82f999a67Sdrh **    May you find forgiveness for yourself and forgive others.
92f999a67Sdrh **    May you share freely, never taking more than you give.
102f999a67Sdrh **
112f999a67Sdrh *************************************************************************
122f999a67Sdrh **
132f999a67Sdrh ** This file contains code used to implement test interfaces to the
142f999a67Sdrh ** memory allocation subsystem.
152f999a67Sdrh **
16*dbdc4d49Sdanielk1977 ** $Id: test_malloc.c,v 1.20 2008/03/28 07:42:54 danielk1977 Exp $
172f999a67Sdrh */
182f999a67Sdrh #include "sqliteInt.h"
192f999a67Sdrh #include "tcl.h"
202f999a67Sdrh #include <stdlib.h>
212f999a67Sdrh #include <string.h>
222f999a67Sdrh #include <assert.h>
232f999a67Sdrh 
242f999a67Sdrh /*
252f999a67Sdrh ** Transform pointers to text and back again
262f999a67Sdrh */
272f999a67Sdrh static void pointerToText(void *p, char *z){
282f999a67Sdrh   static const char zHex[] = "0123456789abcdef";
292f999a67Sdrh   int i, k;
304a50aac5Sdrh   unsigned int u;
314a50aac5Sdrh   sqlite3_uint64 n;
324a50aac5Sdrh   if( sizeof(n)==sizeof(p) ){
334a50aac5Sdrh     memcpy(&n, &p, sizeof(p));
344a50aac5Sdrh   }else if( sizeof(u)==sizeof(p) ){
354a50aac5Sdrh     memcpy(&u, &p, sizeof(u));
364a50aac5Sdrh     n = u;
374a50aac5Sdrh   }else{
384a50aac5Sdrh     assert( 0 );
394a50aac5Sdrh   }
402f999a67Sdrh   for(i=0, k=sizeof(p)*2-1; i<sizeof(p)*2; i++, k--){
412f999a67Sdrh     z[k] = zHex[n&0xf];
422f999a67Sdrh     n >>= 4;
432f999a67Sdrh   }
442f999a67Sdrh   z[sizeof(p)*2] = 0;
452f999a67Sdrh }
462f999a67Sdrh static int hexToInt(int h){
472f999a67Sdrh   if( h>='0' && h<='9' ){
482f999a67Sdrh     return h - '0';
492f999a67Sdrh   }else if( h>='a' && h<='f' ){
502f999a67Sdrh     return h - 'a' + 10;
512f999a67Sdrh   }else{
522f999a67Sdrh     return -1;
532f999a67Sdrh   }
542f999a67Sdrh }
552f999a67Sdrh static int textToPointer(const char *z, void **pp){
562f999a67Sdrh   sqlite3_uint64 n = 0;
572f999a67Sdrh   int i;
584a50aac5Sdrh   unsigned int u;
592f999a67Sdrh   for(i=0; i<sizeof(void*)*2 && z[0]; i++){
602f999a67Sdrh     int v;
612f999a67Sdrh     v = hexToInt(*z++);
622f999a67Sdrh     if( v<0 ) return TCL_ERROR;
632f999a67Sdrh     n = n*16 + v;
642f999a67Sdrh   }
652f999a67Sdrh   if( *z!=0 ) return TCL_ERROR;
664a50aac5Sdrh   if( sizeof(n)==sizeof(*pp) ){
674a50aac5Sdrh     memcpy(pp, &n, sizeof(n));
684a50aac5Sdrh   }else if( sizeof(u)==sizeof(*pp) ){
694a50aac5Sdrh     u = (unsigned int)n;
704a50aac5Sdrh     memcpy(pp, &u, sizeof(u));
714a50aac5Sdrh   }else{
724a50aac5Sdrh     assert( 0 );
734a50aac5Sdrh   }
742f999a67Sdrh   return TCL_OK;
752f999a67Sdrh }
762f999a67Sdrh 
772f999a67Sdrh /*
782f999a67Sdrh ** Usage:    sqlite3_malloc  NBYTES
792f999a67Sdrh **
802f999a67Sdrh ** Raw test interface for sqlite3_malloc().
812f999a67Sdrh */
822f999a67Sdrh static int test_malloc(
832f999a67Sdrh   void * clientData,
842f999a67Sdrh   Tcl_Interp *interp,
852f999a67Sdrh   int objc,
862f999a67Sdrh   Tcl_Obj *CONST objv[]
872f999a67Sdrh ){
882f999a67Sdrh   int nByte;
892f999a67Sdrh   void *p;
902f999a67Sdrh   char zOut[100];
912f999a67Sdrh   if( objc!=2 ){
922f999a67Sdrh     Tcl_WrongNumArgs(interp, 1, objv, "NBYTES");
932f999a67Sdrh     return TCL_ERROR;
942f999a67Sdrh   }
952f999a67Sdrh   if( Tcl_GetIntFromObj(interp, objv[1], &nByte) ) return TCL_ERROR;
962f999a67Sdrh   p = sqlite3_malloc((unsigned)nByte);
972f999a67Sdrh   pointerToText(p, zOut);
982f999a67Sdrh   Tcl_AppendResult(interp, zOut, NULL);
992f999a67Sdrh   return TCL_OK;
1002f999a67Sdrh }
1012f999a67Sdrh 
1022f999a67Sdrh /*
1032f999a67Sdrh ** Usage:    sqlite3_realloc  PRIOR  NBYTES
1042f999a67Sdrh **
1052f999a67Sdrh ** Raw test interface for sqlite3_realloc().
1062f999a67Sdrh */
1072f999a67Sdrh static int test_realloc(
1082f999a67Sdrh   void * clientData,
1092f999a67Sdrh   Tcl_Interp *interp,
1102f999a67Sdrh   int objc,
1112f999a67Sdrh   Tcl_Obj *CONST objv[]
1122f999a67Sdrh ){
1132f999a67Sdrh   int nByte;
1142f999a67Sdrh   void *pPrior, *p;
1152f999a67Sdrh   char zOut[100];
1162f999a67Sdrh   if( objc!=3 ){
1172f999a67Sdrh     Tcl_WrongNumArgs(interp, 1, objv, "PRIOR NBYTES");
1182f999a67Sdrh     return TCL_ERROR;
1192f999a67Sdrh   }
1202f999a67Sdrh   if( Tcl_GetIntFromObj(interp, objv[2], &nByte) ) return TCL_ERROR;
1212f999a67Sdrh   if( textToPointer(Tcl_GetString(objv[1]), &pPrior) ){
1222f999a67Sdrh     Tcl_AppendResult(interp, "bad pointer: ", Tcl_GetString(objv[1]), (char*)0);
1232f999a67Sdrh     return TCL_ERROR;
1242f999a67Sdrh   }
1252f999a67Sdrh   p = sqlite3_realloc(pPrior, (unsigned)nByte);
1262f999a67Sdrh   pointerToText(p, zOut);
1272f999a67Sdrh   Tcl_AppendResult(interp, zOut, NULL);
1282f999a67Sdrh   return TCL_OK;
1292f999a67Sdrh }
1302f999a67Sdrh 
1312f999a67Sdrh 
1322f999a67Sdrh /*
1332f999a67Sdrh ** Usage:    sqlite3_free  PRIOR
1342f999a67Sdrh **
1352f999a67Sdrh ** Raw test interface for sqlite3_free().
1362f999a67Sdrh */
1372f999a67Sdrh static int test_free(
1382f999a67Sdrh   void * clientData,
1392f999a67Sdrh   Tcl_Interp *interp,
1402f999a67Sdrh   int objc,
1412f999a67Sdrh   Tcl_Obj *CONST objv[]
1422f999a67Sdrh ){
1432f999a67Sdrh   void *pPrior;
1442f999a67Sdrh   if( objc!=2 ){
1452f999a67Sdrh     Tcl_WrongNumArgs(interp, 1, objv, "PRIOR");
1462f999a67Sdrh     return TCL_ERROR;
1472f999a67Sdrh   }
1482f999a67Sdrh   if( textToPointer(Tcl_GetString(objv[1]), &pPrior) ){
1492f999a67Sdrh     Tcl_AppendResult(interp, "bad pointer: ", Tcl_GetString(objv[1]), (char*)0);
1502f999a67Sdrh     return TCL_ERROR;
1512f999a67Sdrh   }
1522f999a67Sdrh   sqlite3_free(pPrior);
1532f999a67Sdrh   return TCL_OK;
1542f999a67Sdrh }
1552f999a67Sdrh 
1562f999a67Sdrh /*
1579c7a60dfSdrh ** These routines are in test_hexio.c
1589c7a60dfSdrh */
1599c7a60dfSdrh int sqlite3TestHexToBin(const char *, int, char *);
1609c7a60dfSdrh int sqlite3TestBinToHex(char*,int);
1619c7a60dfSdrh 
1629c7a60dfSdrh /*
1639c7a60dfSdrh ** Usage:    memset  ADDRESS  SIZE  HEX
1649c7a60dfSdrh **
1659c7a60dfSdrh ** Set a chunk of memory (obtained from malloc, probably) to a
1669c7a60dfSdrh ** specified hex pattern.
1679c7a60dfSdrh */
1689c7a60dfSdrh static int test_memset(
1699c7a60dfSdrh   void * clientData,
1709c7a60dfSdrh   Tcl_Interp *interp,
1719c7a60dfSdrh   int objc,
1729c7a60dfSdrh   Tcl_Obj *CONST objv[]
1739c7a60dfSdrh ){
1749c7a60dfSdrh   void *p;
1759c7a60dfSdrh   int size, n, i;
1769c7a60dfSdrh   char *zHex;
1779c7a60dfSdrh   char *zOut;
1789c7a60dfSdrh   char zBin[100];
1799c7a60dfSdrh 
1809c7a60dfSdrh   if( objc!=4 ){
1819c7a60dfSdrh     Tcl_WrongNumArgs(interp, 1, objv, "ADDRESS SIZE HEX");
1829c7a60dfSdrh     return TCL_ERROR;
1839c7a60dfSdrh   }
1849c7a60dfSdrh   if( textToPointer(Tcl_GetString(objv[1]), &p) ){
1859c7a60dfSdrh     Tcl_AppendResult(interp, "bad pointer: ", Tcl_GetString(objv[1]), (char*)0);
1869c7a60dfSdrh     return TCL_ERROR;
1879c7a60dfSdrh   }
1889c7a60dfSdrh   if( Tcl_GetIntFromObj(interp, objv[2], &size) ){
1899c7a60dfSdrh     return TCL_ERROR;
1909c7a60dfSdrh   }
1919c7a60dfSdrh   if( size<=0 ){
1929c7a60dfSdrh     Tcl_AppendResult(interp, "size must be positive", (char*)0);
1939c7a60dfSdrh     return TCL_ERROR;
1949c7a60dfSdrh   }
1959c7a60dfSdrh   zHex = Tcl_GetStringFromObj(objv[3], &n);
1969c7a60dfSdrh   if( n>sizeof(zBin)*2 ) n = sizeof(zBin)*2;
1979c7a60dfSdrh   n = sqlite3TestHexToBin(zHex, n, zBin);
1989c7a60dfSdrh   if( n==0 ){
1999c7a60dfSdrh     Tcl_AppendResult(interp, "no data", (char*)0);
2009c7a60dfSdrh     return TCL_ERROR;
2019c7a60dfSdrh   }
2029c7a60dfSdrh   zOut = p;
2039c7a60dfSdrh   for(i=0; i<size; i++){
2049c7a60dfSdrh     zOut[i] = zBin[i%n];
2059c7a60dfSdrh   }
2069c7a60dfSdrh   return TCL_OK;
2079c7a60dfSdrh }
2089c7a60dfSdrh 
2099c7a60dfSdrh /*
2109c7a60dfSdrh ** Usage:    memget  ADDRESS  SIZE
2119c7a60dfSdrh **
2129c7a60dfSdrh ** Return memory as hexadecimal text.
2139c7a60dfSdrh */
2149c7a60dfSdrh static int test_memget(
2159c7a60dfSdrh   void * clientData,
2169c7a60dfSdrh   Tcl_Interp *interp,
2179c7a60dfSdrh   int objc,
2189c7a60dfSdrh   Tcl_Obj *CONST objv[]
2199c7a60dfSdrh ){
2209c7a60dfSdrh   void *p;
2219c7a60dfSdrh   int size, n;
2229c7a60dfSdrh   char *zBin;
2239c7a60dfSdrh   char zHex[100];
2249c7a60dfSdrh 
2259c7a60dfSdrh   if( objc!=3 ){
2269c7a60dfSdrh     Tcl_WrongNumArgs(interp, 1, objv, "ADDRESS SIZE");
2279c7a60dfSdrh     return TCL_ERROR;
2289c7a60dfSdrh   }
2299c7a60dfSdrh   if( textToPointer(Tcl_GetString(objv[1]), &p) ){
2309c7a60dfSdrh     Tcl_AppendResult(interp, "bad pointer: ", Tcl_GetString(objv[1]), (char*)0);
2319c7a60dfSdrh     return TCL_ERROR;
2329c7a60dfSdrh   }
2339c7a60dfSdrh   if( Tcl_GetIntFromObj(interp, objv[2], &size) ){
2349c7a60dfSdrh     return TCL_ERROR;
2359c7a60dfSdrh   }
2369c7a60dfSdrh   if( size<=0 ){
2379c7a60dfSdrh     Tcl_AppendResult(interp, "size must be positive", (char*)0);
2389c7a60dfSdrh     return TCL_ERROR;
2399c7a60dfSdrh   }
2409c7a60dfSdrh   zBin = p;
2419c7a60dfSdrh   while( size>0 ){
2429c7a60dfSdrh     if( size>(sizeof(zHex)-1)/2 ){
2439c7a60dfSdrh       n = (sizeof(zHex)-1)/2;
2449c7a60dfSdrh     }else{
2459c7a60dfSdrh       n = size;
2469c7a60dfSdrh     }
2479c7a60dfSdrh     memcpy(zHex, zBin, n);
2489c7a60dfSdrh     zBin += n;
2499c7a60dfSdrh     size -= n;
2509c7a60dfSdrh     sqlite3TestBinToHex(zHex, n);
2519c7a60dfSdrh     Tcl_AppendResult(interp, zHex, (char*)0);
2529c7a60dfSdrh   }
2539c7a60dfSdrh   return TCL_OK;
2549c7a60dfSdrh }
2559c7a60dfSdrh 
2569c7a60dfSdrh /*
2572f999a67Sdrh ** Usage:    sqlite3_memory_used
2582f999a67Sdrh **
2592f999a67Sdrh ** Raw test interface for sqlite3_memory_used().
2602f999a67Sdrh */
2612f999a67Sdrh static int test_memory_used(
2622f999a67Sdrh   void * clientData,
2632f999a67Sdrh   Tcl_Interp *interp,
2642f999a67Sdrh   int objc,
2652f999a67Sdrh   Tcl_Obj *CONST objv[]
2662f999a67Sdrh ){
2672f999a67Sdrh   Tcl_SetObjResult(interp, Tcl_NewWideIntObj(sqlite3_memory_used()));
2682f999a67Sdrh   return TCL_OK;
2692f999a67Sdrh }
2702f999a67Sdrh 
2712f999a67Sdrh /*
2722f999a67Sdrh ** Usage:    sqlite3_memory_highwater ?RESETFLAG?
2732f999a67Sdrh **
2742f999a67Sdrh ** Raw test interface for sqlite3_memory_highwater().
2752f999a67Sdrh */
2762f999a67Sdrh static int test_memory_highwater(
2772f999a67Sdrh   void * clientData,
2782f999a67Sdrh   Tcl_Interp *interp,
2792f999a67Sdrh   int objc,
2802f999a67Sdrh   Tcl_Obj *CONST objv[]
2812f999a67Sdrh ){
2822f999a67Sdrh   int resetFlag = 0;
2832f999a67Sdrh   if( objc!=1 && objc!=2 ){
2842f999a67Sdrh     Tcl_WrongNumArgs(interp, 1, objv, "?RESET?");
2852f999a67Sdrh     return TCL_ERROR;
2862f999a67Sdrh   }
2872f999a67Sdrh   if( objc==2 ){
2882f999a67Sdrh     if( Tcl_GetBooleanFromObj(interp, objv[1], &resetFlag) ) return TCL_ERROR;
2892f999a67Sdrh   }
2902f999a67Sdrh   Tcl_SetObjResult(interp,
2912f999a67Sdrh      Tcl_NewWideIntObj(sqlite3_memory_highwater(resetFlag)));
2922f999a67Sdrh   return TCL_OK;
2932f999a67Sdrh }
2942f999a67Sdrh 
2952f999a67Sdrh /*
2962f999a67Sdrh ** Usage:    sqlite3_memdebug_backtrace DEPTH
2972f999a67Sdrh **
2982f999a67Sdrh ** Set the depth of backtracing.  If SQLITE_MEMDEBUG is not defined
2992f999a67Sdrh ** then this routine is a no-op.
3002f999a67Sdrh */
3012f999a67Sdrh static int test_memdebug_backtrace(
3022f999a67Sdrh   void * clientData,
3032f999a67Sdrh   Tcl_Interp *interp,
3042f999a67Sdrh   int objc,
3052f999a67Sdrh   Tcl_Obj *CONST objv[]
3062f999a67Sdrh ){
3072f999a67Sdrh   int depth;
3082f999a67Sdrh   if( objc!=2 ){
3092f999a67Sdrh     Tcl_WrongNumArgs(interp, 1, objv, "DEPT");
3102f999a67Sdrh     return TCL_ERROR;
3112f999a67Sdrh   }
3122f999a67Sdrh   if( Tcl_GetIntFromObj(interp, objv[1], &depth) ) return TCL_ERROR;
3132f999a67Sdrh #ifdef SQLITE_MEMDEBUG
3142f999a67Sdrh   {
31549e4fd71Sdrh     extern void sqlite3MemdebugBacktrace(int);
31649e4fd71Sdrh     sqlite3MemdebugBacktrace(depth);
3172f999a67Sdrh   }
3182f999a67Sdrh #endif
3192f999a67Sdrh   return TCL_OK;
3202f999a67Sdrh }
3212f999a67Sdrh 
3222f999a67Sdrh /*
3232f999a67Sdrh ** Usage:    sqlite3_memdebug_dump  FILENAME
3242f999a67Sdrh **
3252f999a67Sdrh ** Write a summary of unfreed memory to FILENAME.
3262f999a67Sdrh */
3272f999a67Sdrh static int test_memdebug_dump(
3282f999a67Sdrh   void * clientData,
3292f999a67Sdrh   Tcl_Interp *interp,
3302f999a67Sdrh   int objc,
3312f999a67Sdrh   Tcl_Obj *CONST objv[]
3322f999a67Sdrh ){
3332f999a67Sdrh   if( objc!=2 ){
3342f999a67Sdrh     Tcl_WrongNumArgs(interp, 1, objv, "FILENAME");
3352f999a67Sdrh     return TCL_ERROR;
3362f999a67Sdrh   }
3372d7636e2Sdrh #if defined(SQLITE_MEMDEBUG) || defined(SQLITE_MEMORY_SIZE) \
3382d7636e2Sdrh      || defined(SQLITE_POW2_MEMORY_SIZE)
3392f999a67Sdrh   {
34049e4fd71Sdrh     extern void sqlite3MemdebugDump(const char*);
34149e4fd71Sdrh     sqlite3MemdebugDump(Tcl_GetString(objv[1]));
3422f999a67Sdrh   }
3432f999a67Sdrh #endif
3442f999a67Sdrh   return TCL_OK;
3452f999a67Sdrh }
3462f999a67Sdrh 
347a7a8e14bSdanielk1977 /*
348a7a8e14bSdanielk1977 ** Usage:    sqlite3_memdebug_malloc_count
349a7a8e14bSdanielk1977 **
350a7a8e14bSdanielk1977 ** Return the total number of times malloc() has been called.
351a7a8e14bSdanielk1977 */
352a7a8e14bSdanielk1977 static int test_memdebug_malloc_count(
353a7a8e14bSdanielk1977   void * clientData,
354a7a8e14bSdanielk1977   Tcl_Interp *interp,
355a7a8e14bSdanielk1977   int objc,
356a7a8e14bSdanielk1977   Tcl_Obj *CONST objv[]
357a7a8e14bSdanielk1977 ){
358a7a8e14bSdanielk1977   int nMalloc = -1;
359a7a8e14bSdanielk1977   if( objc!=1 ){
360a7a8e14bSdanielk1977     Tcl_WrongNumArgs(interp, 1, objv, "");
361a7a8e14bSdanielk1977     return TCL_ERROR;
362a7a8e14bSdanielk1977   }
363a7a8e14bSdanielk1977 #if defined(SQLITE_MEMDEBUG)
364a7a8e14bSdanielk1977   {
36549e4fd71Sdrh     extern int sqlite3MemdebugMallocCount();
36649e4fd71Sdrh     nMalloc = sqlite3MemdebugMallocCount();
367a7a8e14bSdanielk1977   }
368a7a8e14bSdanielk1977 #endif
369a7a8e14bSdanielk1977   Tcl_SetObjResult(interp, Tcl_NewIntObj(nMalloc));
370a7a8e14bSdanielk1977   return TCL_OK;
371a7a8e14bSdanielk1977 }
372a7a8e14bSdanielk1977 
3732f999a67Sdrh 
3742f999a67Sdrh /*
375a1644fd8Sdanielk1977 ** Usage:    sqlite3_memdebug_fail  COUNTER  ?OPTIONS?
376a1644fd8Sdanielk1977 **
377a1644fd8Sdanielk1977 ** where options are:
378a1644fd8Sdanielk1977 **
379643167ffSdrh **     -repeat    <count>
380a1644fd8Sdanielk1977 **     -benigncnt <varname>
3810e6f1546Sdrh **
3820e6f1546Sdrh ** Arrange for a simulated malloc() failure after COUNTER successes.
383643167ffSdrh ** If a repeat count is specified, the fault is repeated that many
384643167ffSdrh ** times.
3850e6f1546Sdrh **
3860e6f1546Sdrh ** Each call to this routine overrides the prior counter value.
3870e6f1546Sdrh ** This routine returns the number of simulated failures that have
3880e6f1546Sdrh ** happened since the previous call to this routine.
3890e6f1546Sdrh **
3900e6f1546Sdrh ** To disable simulated failures, use a COUNTER of -1.
3910e6f1546Sdrh */
3920e6f1546Sdrh static int test_memdebug_fail(
3930e6f1546Sdrh   void * clientData,
3940e6f1546Sdrh   Tcl_Interp *interp,
3950e6f1546Sdrh   int objc,
3960e6f1546Sdrh   Tcl_Obj *CONST objv[]
3970e6f1546Sdrh ){
398a1644fd8Sdanielk1977   int ii;
3990e6f1546Sdrh   int iFail;
400643167ffSdrh   int nRepeat = 1;
401a1644fd8Sdanielk1977   Tcl_Obj *pBenignCnt = 0;
402643167ffSdrh   int nBenign;
4030e6f1546Sdrh   int nFail = 0;
404a1644fd8Sdanielk1977 
405a1644fd8Sdanielk1977   if( objc<2 ){
406a1644fd8Sdanielk1977     Tcl_WrongNumArgs(interp, 1, objv, "COUNTER ?OPTIONS?");
4070e6f1546Sdrh     return TCL_ERROR;
4080e6f1546Sdrh   }
4090e6f1546Sdrh   if( Tcl_GetIntFromObj(interp, objv[1], &iFail) ) return TCL_ERROR;
410a1644fd8Sdanielk1977 
411a1644fd8Sdanielk1977   for(ii=2; ii<objc; ii+=2){
412a1644fd8Sdanielk1977     int nOption;
413a1644fd8Sdanielk1977     char *zOption = Tcl_GetStringFromObj(objv[ii], &nOption);
414a1644fd8Sdanielk1977     char *zErr = 0;
415a1644fd8Sdanielk1977 
416a1644fd8Sdanielk1977     if( nOption>1 && strncmp(zOption, "-repeat", nOption)==0 ){
417a1644fd8Sdanielk1977       if( ii==(objc-1) ){
418a1644fd8Sdanielk1977         zErr = "option requires an argument: ";
419ed138fb3Sdrh       }else{
420643167ffSdrh         if( Tcl_GetIntFromObj(interp, objv[ii+1], &nRepeat) ){
421a1644fd8Sdanielk1977           return TCL_ERROR;
422ed138fb3Sdrh         }
423a1644fd8Sdanielk1977       }
424a1644fd8Sdanielk1977     }else if( nOption>1 && strncmp(zOption, "-benigncnt", nOption)==0 ){
425a1644fd8Sdanielk1977       if( ii==(objc-1) ){
426a1644fd8Sdanielk1977         zErr = "option requires an argument: ";
427a1644fd8Sdanielk1977       }else{
428a1644fd8Sdanielk1977         pBenignCnt = objv[ii+1];
429a1644fd8Sdanielk1977       }
430a1644fd8Sdanielk1977     }else{
431a1644fd8Sdanielk1977       zErr = "unknown option: ";
432a1644fd8Sdanielk1977     }
433a1644fd8Sdanielk1977 
434a1644fd8Sdanielk1977     if( zErr ){
435a1644fd8Sdanielk1977       Tcl_AppendResult(interp, zErr, zOption, 0);
436a1644fd8Sdanielk1977       return TCL_ERROR;
437a1644fd8Sdanielk1977     }
438a1644fd8Sdanielk1977   }
439a1644fd8Sdanielk1977 
44077db4c05Sdrh   sqlite3_test_control(-12345); /* Just to stress the test_control interface */
441ed13d98cSdrh   nBenign = sqlite3_test_control(SQLITE_TESTCTRL_FAULT_BENIGN_FAILURES,
442ed13d98cSdrh                                  SQLITE_FAULTINJECTOR_MALLOC);
443ed13d98cSdrh   nFail = sqlite3_test_control(SQLITE_TESTCTRL_FAULT_FAILURES,
444ed13d98cSdrh                                SQLITE_FAULTINJECTOR_MALLOC);
445ed13d98cSdrh   sqlite3_test_control(SQLITE_TESTCTRL_FAULT_CONFIG,
446ed13d98cSdrh                        SQLITE_FAULTINJECTOR_MALLOC, iFail, nRepeat);
447a1644fd8Sdanielk1977   if( pBenignCnt ){
448643167ffSdrh     Tcl_ObjSetVar2(interp, pBenignCnt, 0, Tcl_NewIntObj(nBenign), 0);
449a1644fd8Sdanielk1977   }
4500e6f1546Sdrh   Tcl_SetObjResult(interp, Tcl_NewIntObj(nFail));
4510e6f1546Sdrh   return TCL_OK;
4520e6f1546Sdrh }
4530e6f1546Sdrh 
454cd03724cSdanielk1977 /*
455cd03724cSdanielk1977 ** Usage:    sqlite3_memdebug_pending
456cd03724cSdanielk1977 **
457cd03724cSdanielk1977 ** Return the number of malloc() calls that will succeed before a
458cd03724cSdanielk1977 ** simulated failure occurs. A negative return value indicates that
459cd03724cSdanielk1977 ** no malloc() failure is scheduled.
460cd03724cSdanielk1977 */
461cd03724cSdanielk1977 static int test_memdebug_pending(
462cd03724cSdanielk1977   void * clientData,
463cd03724cSdanielk1977   Tcl_Interp *interp,
464cd03724cSdanielk1977   int objc,
465cd03724cSdanielk1977   Tcl_Obj *CONST objv[]
466cd03724cSdanielk1977 ){
4675efaf070Sdrh   int nPending;
468cd03724cSdanielk1977   if( objc!=1 ){
469cd03724cSdanielk1977     Tcl_WrongNumArgs(interp, 1, objv, "");
470cd03724cSdanielk1977     return TCL_ERROR;
471cd03724cSdanielk1977   }
4725efaf070Sdrh   nPending = sqlite3_test_control(SQLITE_TESTCTRL_FAULT_PENDING,
473ed13d98cSdrh                                   SQLITE_FAULTINJECTOR_MALLOC);
474ed13d98cSdrh   Tcl_SetObjResult(interp, Tcl_NewIntObj(nPending));
475cd03724cSdanielk1977   return TCL_OK;
476cd03724cSdanielk1977 }
477cd03724cSdanielk1977 
4780e6f1546Sdrh 
4790e6f1546Sdrh /*
4804a50aac5Sdrh ** Usage:    sqlite3_memdebug_settitle TITLE
4814a50aac5Sdrh **
4824a50aac5Sdrh ** Set a title string stored with each allocation.  The TITLE is
4834a50aac5Sdrh ** typically the name of the test that was running when the
4844a50aac5Sdrh ** allocation occurred.  The TITLE is stored with the allocation
4854a50aac5Sdrh ** and can be used to figure out which tests are leaking memory.
4864a50aac5Sdrh **
4874a50aac5Sdrh ** Each title overwrite the previous.
4884a50aac5Sdrh */
4894a50aac5Sdrh static int test_memdebug_settitle(
4904a50aac5Sdrh   void * clientData,
4914a50aac5Sdrh   Tcl_Interp *interp,
4924a50aac5Sdrh   int objc,
4934a50aac5Sdrh   Tcl_Obj *CONST objv[]
4944a50aac5Sdrh ){
4954a50aac5Sdrh   const char *zTitle;
4964a50aac5Sdrh   if( objc!=2 ){
4974a50aac5Sdrh     Tcl_WrongNumArgs(interp, 1, objv, "TITLE");
4984a50aac5Sdrh     return TCL_ERROR;
4994a50aac5Sdrh   }
5004a50aac5Sdrh   zTitle = Tcl_GetString(objv[1]);
5014a50aac5Sdrh #ifdef SQLITE_MEMDEBUG
5024a50aac5Sdrh   {
50349e4fd71Sdrh     extern int sqlite3MemdebugSettitle(const char*);
50449e4fd71Sdrh     sqlite3MemdebugSettitle(zTitle);
5054a50aac5Sdrh   }
5064a50aac5Sdrh #endif
5074a50aac5Sdrh   return TCL_OK;
5084a50aac5Sdrh }
5094a50aac5Sdrh 
510cd3e8f7cSdanielk1977 #define MALLOC_LOG_FRAMES 10
5116f332c18Sdanielk1977 static Tcl_HashTable aMallocLog;
5126f332c18Sdanielk1977 static int mallocLogEnabled = 0;
5136f332c18Sdanielk1977 
5146f332c18Sdanielk1977 typedef struct MallocLog MallocLog;
5156f332c18Sdanielk1977 struct MallocLog {
5166f332c18Sdanielk1977   int nCall;
5176f332c18Sdanielk1977   int nByte;
5186f332c18Sdanielk1977 };
5196f332c18Sdanielk1977 
5206f332c18Sdanielk1977 static void test_memdebug_callback(int nByte, int nFrame, void **aFrame){
5216f332c18Sdanielk1977   if( mallocLogEnabled ){
5226f332c18Sdanielk1977     MallocLog *pLog;
5236f332c18Sdanielk1977     Tcl_HashEntry *pEntry;
5246f332c18Sdanielk1977     int isNew;
5256f332c18Sdanielk1977 
5266f332c18Sdanielk1977     int aKey[MALLOC_LOG_FRAMES];
5276f332c18Sdanielk1977     int nKey = sizeof(int)*MALLOC_LOG_FRAMES;
5286f332c18Sdanielk1977 
5296f332c18Sdanielk1977     memset(aKey, 0, nKey);
5306f332c18Sdanielk1977     if( (sizeof(void*)*nFrame)<nKey ){
5316f332c18Sdanielk1977       nKey = nFrame*sizeof(void*);
5326f332c18Sdanielk1977     }
5336f332c18Sdanielk1977     memcpy(aKey, aFrame, nKey);
5346f332c18Sdanielk1977 
5356f332c18Sdanielk1977     pEntry = Tcl_CreateHashEntry(&aMallocLog, (const char *)aKey, &isNew);
5366f332c18Sdanielk1977     if( isNew ){
5376f332c18Sdanielk1977       pLog = (MallocLog *)Tcl_Alloc(sizeof(MallocLog));
5386f332c18Sdanielk1977       memset(pLog, 0, sizeof(MallocLog));
5396f332c18Sdanielk1977       Tcl_SetHashValue(pEntry, (ClientData)pLog);
5406f332c18Sdanielk1977     }else{
5416f332c18Sdanielk1977       pLog = (MallocLog *)Tcl_GetHashValue(pEntry);
5426f332c18Sdanielk1977     }
5436f332c18Sdanielk1977 
5446f332c18Sdanielk1977     pLog->nCall++;
5456f332c18Sdanielk1977     pLog->nByte += nByte;
5466f332c18Sdanielk1977   }
5476f332c18Sdanielk1977 }
5486f332c18Sdanielk1977 
549*dbdc4d49Sdanielk1977 static int test_memdebug_log_clear(){
550*dbdc4d49Sdanielk1977   Tcl_HashSearch search;
551*dbdc4d49Sdanielk1977   Tcl_HashEntry *pEntry;
552*dbdc4d49Sdanielk1977   for(
553*dbdc4d49Sdanielk1977     pEntry=Tcl_FirstHashEntry(&aMallocLog, &search);
554*dbdc4d49Sdanielk1977     pEntry;
555*dbdc4d49Sdanielk1977     pEntry=Tcl_NextHashEntry(&search)
556*dbdc4d49Sdanielk1977   ){
557*dbdc4d49Sdanielk1977     MallocLog *pLog = (MallocLog *)Tcl_GetHashValue(pEntry);
558*dbdc4d49Sdanielk1977     Tcl_Free((char *)pLog);
559*dbdc4d49Sdanielk1977   }
560*dbdc4d49Sdanielk1977   Tcl_DeleteHashTable(&aMallocLog);
561*dbdc4d49Sdanielk1977   Tcl_InitHashTable(&aMallocLog, MALLOC_LOG_FRAMES);
562*dbdc4d49Sdanielk1977 }
563*dbdc4d49Sdanielk1977 
5646f332c18Sdanielk1977 static int test_memdebug_log(
5656f332c18Sdanielk1977   void * clientData,
5666f332c18Sdanielk1977   Tcl_Interp *interp,
5676f332c18Sdanielk1977   int objc,
5686f332c18Sdanielk1977   Tcl_Obj *CONST objv[]
5696f332c18Sdanielk1977 ){
5706f332c18Sdanielk1977   static int isInit = 0;
5716f332c18Sdanielk1977   int iSub;
5726f332c18Sdanielk1977 
573*dbdc4d49Sdanielk1977   static const char *MB_strs[] = { "start", "stop", "dump", "clear", "sync" };
574*dbdc4d49Sdanielk1977   enum MB_enum {
575*dbdc4d49Sdanielk1977       MB_LOG_START, MB_LOG_STOP, MB_LOG_DUMP, MB_LOG_CLEAR, MB_LOG_SYNC
576*dbdc4d49Sdanielk1977   };
5776f332c18Sdanielk1977 
5786f332c18Sdanielk1977   if( !isInit ){
5796f332c18Sdanielk1977 #ifdef SQLITE_MEMDEBUG
5806f332c18Sdanielk1977     extern void sqlite3MemdebugBacktraceCallback(
5816f332c18Sdanielk1977         void (*xBacktrace)(int, int, void **));
5826f332c18Sdanielk1977     sqlite3MemdebugBacktraceCallback(test_memdebug_callback);
5836f332c18Sdanielk1977 #endif
5846f332c18Sdanielk1977     Tcl_InitHashTable(&aMallocLog, MALLOC_LOG_FRAMES);
5856f332c18Sdanielk1977     isInit = 1;
5866f332c18Sdanielk1977   }
5876f332c18Sdanielk1977 
5886f332c18Sdanielk1977   if( objc<2 ){
5896f332c18Sdanielk1977     Tcl_WrongNumArgs(interp, 1, objv, "SUB-COMMAND ...");
5906f332c18Sdanielk1977   }
5916f332c18Sdanielk1977   if( Tcl_GetIndexFromObj(interp, objv[1], MB_strs, "sub-command", 0, &iSub) ){
5926f332c18Sdanielk1977     return TCL_ERROR;
5936f332c18Sdanielk1977   }
5946f332c18Sdanielk1977 
5956f332c18Sdanielk1977   switch( (enum MB_enum)iSub ){
5966f332c18Sdanielk1977     case MB_LOG_START:
5976f332c18Sdanielk1977       mallocLogEnabled = 1;
5986f332c18Sdanielk1977       break;
5996f332c18Sdanielk1977     case MB_LOG_STOP:
6006f332c18Sdanielk1977       mallocLogEnabled = 0;
6016f332c18Sdanielk1977       break;
6026f332c18Sdanielk1977     case MB_LOG_DUMP: {
6036f332c18Sdanielk1977       Tcl_HashSearch search;
6046f332c18Sdanielk1977       Tcl_HashEntry *pEntry;
6056f332c18Sdanielk1977       Tcl_Obj *pRet = Tcl_NewObj();
6066f332c18Sdanielk1977 
6076f332c18Sdanielk1977       assert(sizeof(int)==sizeof(void*));
6086f332c18Sdanielk1977 
6096f332c18Sdanielk1977       for(
6106f332c18Sdanielk1977         pEntry=Tcl_FirstHashEntry(&aMallocLog, &search);
6116f332c18Sdanielk1977         pEntry;
6126f332c18Sdanielk1977         pEntry=Tcl_NextHashEntry(&search)
6136f332c18Sdanielk1977       ){
6146f332c18Sdanielk1977         Tcl_Obj *apElem[MALLOC_LOG_FRAMES+2];
6156f332c18Sdanielk1977         MallocLog *pLog = (MallocLog *)Tcl_GetHashValue(pEntry);
6166f332c18Sdanielk1977         int *aKey = (int *)Tcl_GetHashKey(&aMallocLog, pEntry);
6176f332c18Sdanielk1977         int ii;
6186f332c18Sdanielk1977 
6196f332c18Sdanielk1977         apElem[0] = Tcl_NewIntObj(pLog->nCall);
6206f332c18Sdanielk1977         apElem[1] = Tcl_NewIntObj(pLog->nByte);
6216f332c18Sdanielk1977         for(ii=0; ii<MALLOC_LOG_FRAMES; ii++){
6226f332c18Sdanielk1977           apElem[ii+2] = Tcl_NewIntObj(aKey[ii]);
6236f332c18Sdanielk1977         }
6246f332c18Sdanielk1977 
6256f332c18Sdanielk1977         Tcl_ListObjAppendElement(interp, pRet,
6266f332c18Sdanielk1977             Tcl_NewListObj(MALLOC_LOG_FRAMES+2, apElem)
6276f332c18Sdanielk1977         );
6286f332c18Sdanielk1977       }
6296f332c18Sdanielk1977 
6306f332c18Sdanielk1977       Tcl_SetObjResult(interp, pRet);
6316f332c18Sdanielk1977       break;
6326f332c18Sdanielk1977     }
6336f332c18Sdanielk1977     case MB_LOG_CLEAR: {
634*dbdc4d49Sdanielk1977       test_memdebug_log_clear();
635*dbdc4d49Sdanielk1977       break;
6366f332c18Sdanielk1977     }
637*dbdc4d49Sdanielk1977 
638*dbdc4d49Sdanielk1977     case MB_LOG_SYNC: {
639*dbdc4d49Sdanielk1977       extern void sqlite3MemdebugSync();
640*dbdc4d49Sdanielk1977       test_memdebug_log_clear();
641*dbdc4d49Sdanielk1977       mallocLogEnabled = 1;
642*dbdc4d49Sdanielk1977       sqlite3MemdebugSync();
643*dbdc4d49Sdanielk1977       break;
6446f332c18Sdanielk1977     }
6456f332c18Sdanielk1977   }
6466f332c18Sdanielk1977 
6476f332c18Sdanielk1977   return TCL_OK;
6486f332c18Sdanielk1977 }
6494a50aac5Sdrh 
6504a50aac5Sdrh /*
6512f999a67Sdrh ** Register commands with the TCL interpreter.
6522f999a67Sdrh */
6532f999a67Sdrh int Sqlitetest_malloc_Init(Tcl_Interp *interp){
6542f999a67Sdrh   static struct {
6552f999a67Sdrh      char *zName;
6562f999a67Sdrh      Tcl_ObjCmdProc *xProc;
6572f999a67Sdrh   } aObjCmd[] = {
6582f999a67Sdrh      { "sqlite3_malloc",             test_malloc                   },
6592f999a67Sdrh      { "sqlite3_realloc",            test_realloc                  },
6602f999a67Sdrh      { "sqlite3_free",               test_free                     },
6619c7a60dfSdrh      { "memset",                     test_memset                   },
6629c7a60dfSdrh      { "memget",                     test_memget                   },
6632f999a67Sdrh      { "sqlite3_memory_used",        test_memory_used              },
6642f999a67Sdrh      { "sqlite3_memory_highwater",   test_memory_highwater         },
6652f999a67Sdrh      { "sqlite3_memdebug_backtrace", test_memdebug_backtrace       },
6662f999a67Sdrh      { "sqlite3_memdebug_dump",      test_memdebug_dump            },
6670e6f1546Sdrh      { "sqlite3_memdebug_fail",      test_memdebug_fail            },
668cd03724cSdanielk1977      { "sqlite3_memdebug_pending",   test_memdebug_pending         },
6694a50aac5Sdrh      { "sqlite3_memdebug_settitle",  test_memdebug_settitle        },
670a7a8e14bSdanielk1977      { "sqlite3_memdebug_malloc_count", test_memdebug_malloc_count },
6716f332c18Sdanielk1977      { "sqlite3_memdebug_log",       test_memdebug_log },
6722f999a67Sdrh   };
6732f999a67Sdrh   int i;
6742f999a67Sdrh   for(i=0; i<sizeof(aObjCmd)/sizeof(aObjCmd[0]); i++){
6752f999a67Sdrh     Tcl_CreateObjCommand(interp, aObjCmd[i].zName, aObjCmd[i].xProc, 0, 0);
6762f999a67Sdrh   }
6772f999a67Sdrh   return TCL_OK;
6782f999a67Sdrh }
679