Update contrib.
4 ** The author disclaims copyright to this source code. In place of
5 ** a legal notice, here is a blessing:
7 ** May you do good and not evil.
8 ** May you find forgiveness for yourself and forgive others.
9 ** May you share freely, never taking more than you give.
11 *************************************************************************
13 ** $Id: test_mutex.c,v 1.11 2008/07/19 13:43:24 danielk1977 Exp $
18 #include "sqliteInt.h"
23 /* defined in test1.c */
24 const char *sqlite3TestErrorName(int);
26 /* A countable mutex */
27 struct sqlite3_mutex {
33 static struct test_mutex_globals {
34 int isInstalled; /* True if installed */
35 int disableInit; /* True to cause sqlite3_initalize() to fail */
36 int disableTry; /* True to force sqlite3_mutex_try() to fail */
37 int isInit; /* True if initialized */
38 sqlite3_mutex_methods m; /* Interface to "real" mutex system */
39 int aCounter[8]; /* Number of grabs of each type of mutex */
40 sqlite3_mutex aStatic[6]; /* The six static mutexes */
43 /* Return true if the countable mutex is currently held */
44 static int counterMutexHeld(sqlite3_mutex *p){
45 return g.m.xMutexHeld(p->pReal);
48 /* Return true if the countable mutex is not currently held */
49 static int counterMutexNotheld(sqlite3_mutex *p){
50 return g.m.xMutexNotheld(p->pReal);
53 /* Initialize the countable mutex interface
54 ** Or, if g.disableInit is non-zero, then do not initialize but instead
55 ** return the value of g.disableInit as the result code. This can be used
56 ** to simulate an initialization failure.
58 static int counterMutexInit(void){
60 if( g.disableInit ) return g.disableInit;
61 rc = g.m.xMutexInit();
67 ** Uninitialize the mutex subsystem
69 static int counterMutexEnd(void){
71 return g.m.xMutexEnd();
75 ** Allocate a countable mutex
77 static sqlite3_mutex *counterMutexAlloc(int eType){
79 sqlite3_mutex *pRet = 0;
82 assert(eType<8 && eType>=0);
84 pReal = g.m.xMutexAlloc(eType);
85 if( !pReal ) return 0;
87 if( eType==SQLITE_MUTEX_FAST || eType==SQLITE_MUTEX_RECURSIVE ){
88 pRet = (sqlite3_mutex *)malloc(sizeof(sqlite3_mutex));
90 pRet = &g.aStatic[eType-2];
99 ** Free a countable mutex
101 static void counterMutexFree(sqlite3_mutex *p){
103 g.m.xMutexFree(p->pReal);
104 if( p->eType==SQLITE_MUTEX_FAST || p->eType==SQLITE_MUTEX_RECURSIVE ){
110 ** Enter a countable mutex. Block until entry is safe.
112 static void counterMutexEnter(sqlite3_mutex *p){
114 g.aCounter[p->eType]++;
115 g.m.xMutexEnter(p->pReal);
119 ** Try to enter a mutex. Return true on success.
121 static int counterMutexTry(sqlite3_mutex *p){
123 g.aCounter[p->eType]++;
124 if( g.disableTry ) return SQLITE_BUSY;
125 return g.m.xMutexTry(p->pReal);
130 static void counterMutexLeave(sqlite3_mutex *p){
132 g.m.xMutexLeave(p->pReal);
138 static int test_shutdown(
142 Tcl_Obj *CONST objv[]
147 Tcl_WrongNumArgs(interp, 1, objv, "");
151 rc = sqlite3_shutdown();
152 Tcl_SetResult(interp, (char *)sqlite3TestErrorName(rc), TCL_VOLATILE);
157 ** sqlite3_initialize
159 static int test_initialize(
163 Tcl_Obj *CONST objv[]
168 Tcl_WrongNumArgs(interp, 1, objv, "");
172 rc = sqlite3_initialize();
173 Tcl_SetResult(interp, (char *)sqlite3TestErrorName(rc), TCL_VOLATILE);
178 ** install_mutex_counters BOOLEAN
180 static int test_install_mutex_counters(
184 Tcl_Obj *CONST objv[]
189 sqlite3_mutex_methods counter_methods = {
202 Tcl_WrongNumArgs(interp, 1, objv, "BOOLEAN");
205 if( TCL_OK!=Tcl_GetBooleanFromObj(interp, objv[1], &isInstall) ){
209 assert(isInstall==0 || isInstall==1);
210 assert(g.isInstalled==0 || g.isInstalled==1);
211 if( isInstall==g.isInstalled ){
212 Tcl_AppendResult(interp, "mutex counters are ", 0);
213 Tcl_AppendResult(interp, isInstall?"already installed":"not installed", 0);
218 assert( g.m.xMutexAlloc==0 );
219 rc = sqlite3_config(SQLITE_CONFIG_GETMUTEX, &g.m);
221 sqlite3_config(SQLITE_CONFIG_MUTEX, &counter_methods);
225 assert( g.m.xMutexAlloc );
226 rc = sqlite3_config(SQLITE_CONFIG_MUTEX, &g.m);
227 memset(&g.m, 0, sizeof(sqlite3_mutex_methods));
231 g.isInstalled = isInstall;
234 Tcl_SetResult(interp, (char *)sqlite3TestErrorName(rc), TCL_VOLATILE);
239 ** read_mutex_counters
241 static int test_read_mutex_counters(
245 Tcl_Obj *CONST objv[]
250 "fast", "recursive", "static_master", "static_mem",
251 "static_mem2", "static_prng", "static_lru", "static_lru2"
255 Tcl_WrongNumArgs(interp, 1, objv, "");
260 Tcl_IncrRefCount(pRet);
261 for(ii=0; ii<8; ii++){
262 Tcl_ListObjAppendElement(interp, pRet, Tcl_NewStringObj(aName[ii], -1));
263 Tcl_ListObjAppendElement(interp, pRet, Tcl_NewIntObj(g.aCounter[ii]));
265 Tcl_SetObjResult(interp, pRet);
266 Tcl_DecrRefCount(pRet);
272 ** clear_mutex_counters
274 static int test_clear_mutex_counters(
278 Tcl_Obj *CONST objv[]
283 Tcl_WrongNumArgs(interp, 1, objv, "");
287 for(ii=0; ii<8; ii++){
294 ** Create and free a mutex. Return the mutex pointer. The pointer
295 ** will be invalid since the mutex has already been freed. The
296 ** return pointer just checks to see if the mutex really was allocated.
298 static int test_alloc_mutex(
302 Tcl_Obj *CONST objv[]
304 #if SQLITE_THREADSAFE
305 sqlite3_mutex *p = sqlite3_mutex_alloc(SQLITE_MUTEX_FAST);
307 sqlite3_mutex_free(p);
308 sqlite3_snprintf(sizeof(zBuf), zBuf, "%p", p);
309 Tcl_AppendResult(interp, zBuf, (char*)0);
315 ** sqlite3_config OPTION
317 ** OPTION can be either one of the keywords:
319 ** SQLITE_CONFIG_SINGLETHREAD
320 ** SQLITE_CONFIG_MULTITHREAD
321 ** SQLITE_CONFIG_SERIALIZED
323 ** Or OPTION can be an raw integer.
325 static int test_config(
329 Tcl_Obj *CONST objv[]
331 struct ConfigOption {
335 {"singlethread", SQLITE_CONFIG_SINGLETHREAD},
336 {"multithread", SQLITE_CONFIG_MULTITHREAD},
337 {"serialized", SQLITE_CONFIG_SERIALIZED},
340 int s = sizeof(struct ConfigOption);
345 Tcl_WrongNumArgs(interp, 1, objv, "");
349 if( Tcl_GetIndexFromObjStruct(interp, objv[1], aOpt, s, "flag", 0, &i) ){
350 if( Tcl_GetIntFromObj(interp, objv[1], &i) ){
357 rc = sqlite3_config(i);
358 Tcl_SetResult(interp, (char *)sqlite3TestErrorName(rc), TCL_VOLATILE);
362 int Sqlitetest_mutex_Init(Tcl_Interp *interp){
365 Tcl_ObjCmdProc *xProc;
367 { "sqlite3_shutdown", (Tcl_ObjCmdProc*)test_shutdown },
368 { "sqlite3_initialize", (Tcl_ObjCmdProc*)test_initialize },
369 { "sqlite3_config", (Tcl_ObjCmdProc*)test_config },
371 { "alloc_dealloc_mutex", (Tcl_ObjCmdProc*)test_alloc_mutex },
372 { "install_mutex_counters", (Tcl_ObjCmdProc*)test_install_mutex_counters },
373 { "read_mutex_counters", (Tcl_ObjCmdProc*)test_read_mutex_counters },
374 { "clear_mutex_counters", (Tcl_ObjCmdProc*)test_clear_mutex_counters },
377 for(i=0; i<sizeof(aCmd)/sizeof(aCmd[0]); i++){
378 Tcl_CreateObjCommand(interp, aCmd[i].zName, aCmd[i].xProc, 0, 0);
380 memset(&g, 0, sizeof(g));
382 Tcl_LinkVar(interp, "disable_mutex_init",
383 (char*)&g.disableInit, TCL_LINK_INT);
384 Tcl_LinkVar(interp, "disable_mutex_try",
385 (char*)&g.disableTry, TCL_LINK_INT);