1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/persistentdata/persistentstorage/sqlite3api/TEST/SRC/test_mutex.c Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,387 @@
1.4 +/*
1.5 +** 2008 June 18
1.6 +**
1.7 +** The author disclaims copyright to this source code. In place of
1.8 +** a legal notice, here is a blessing:
1.9 +**
1.10 +** May you do good and not evil.
1.11 +** May you find forgiveness for yourself and forgive others.
1.12 +** May you share freely, never taking more than you give.
1.13 +**
1.14 +*************************************************************************
1.15 +**
1.16 +** $Id: test_mutex.c,v 1.11 2008/07/19 13:43:24 danielk1977 Exp $
1.17 +*/
1.18 +
1.19 +#include "tcl.h"
1.20 +#include "sqlite3.h"
1.21 +#include "sqliteInt.h"
1.22 +#include <stdlib.h>
1.23 +#include <assert.h>
1.24 +#include <string.h>
1.25 +
1.26 +/* defined in test1.c */
1.27 +const char *sqlite3TestErrorName(int);
1.28 +
1.29 +/* A countable mutex */
1.30 +struct sqlite3_mutex {
1.31 + sqlite3_mutex *pReal;
1.32 + int eType;
1.33 +};
1.34 +
1.35 +/* State variables */
1.36 +static struct test_mutex_globals {
1.37 + int isInstalled; /* True if installed */
1.38 + int disableInit; /* True to cause sqlite3_initalize() to fail */
1.39 + int disableTry; /* True to force sqlite3_mutex_try() to fail */
1.40 + int isInit; /* True if initialized */
1.41 + sqlite3_mutex_methods m; /* Interface to "real" mutex system */
1.42 + int aCounter[8]; /* Number of grabs of each type of mutex */
1.43 + sqlite3_mutex aStatic[6]; /* The six static mutexes */
1.44 +} g;
1.45 +
1.46 +/* Return true if the countable mutex is currently held */
1.47 +static int counterMutexHeld(sqlite3_mutex *p){
1.48 + return g.m.xMutexHeld(p->pReal);
1.49 +}
1.50 +
1.51 +/* Return true if the countable mutex is not currently held */
1.52 +static int counterMutexNotheld(sqlite3_mutex *p){
1.53 + return g.m.xMutexNotheld(p->pReal);
1.54 +}
1.55 +
1.56 +/* Initialize the countable mutex interface
1.57 +** Or, if g.disableInit is non-zero, then do not initialize but instead
1.58 +** return the value of g.disableInit as the result code. This can be used
1.59 +** to simulate an initialization failure.
1.60 +*/
1.61 +static int counterMutexInit(void){
1.62 + int rc;
1.63 + if( g.disableInit ) return g.disableInit;
1.64 + rc = g.m.xMutexInit();
1.65 + g.isInit = 1;
1.66 + return rc;
1.67 +}
1.68 +
1.69 +/*
1.70 +** Uninitialize the mutex subsystem
1.71 +*/
1.72 +static int counterMutexEnd(void){
1.73 + g.isInit = 0;
1.74 + return g.m.xMutexEnd();
1.75 +}
1.76 +
1.77 +/*
1.78 +** Allocate a countable mutex
1.79 +*/
1.80 +static sqlite3_mutex *counterMutexAlloc(int eType){
1.81 + sqlite3_mutex *pReal;
1.82 + sqlite3_mutex *pRet = 0;
1.83 +
1.84 + assert( g.isInit );
1.85 + assert(eType<8 && eType>=0);
1.86 +
1.87 + pReal = g.m.xMutexAlloc(eType);
1.88 + if( !pReal ) return 0;
1.89 +
1.90 + if( eType==SQLITE_MUTEX_FAST || eType==SQLITE_MUTEX_RECURSIVE ){
1.91 + pRet = (sqlite3_mutex *)malloc(sizeof(sqlite3_mutex));
1.92 + }else{
1.93 + pRet = &g.aStatic[eType-2];
1.94 + }
1.95 +
1.96 + pRet->eType = eType;
1.97 + pRet->pReal = pReal;
1.98 + return pRet;
1.99 +}
1.100 +
1.101 +/*
1.102 +** Free a countable mutex
1.103 +*/
1.104 +static void counterMutexFree(sqlite3_mutex *p){
1.105 + assert( g.isInit );
1.106 + g.m.xMutexFree(p->pReal);
1.107 + if( p->eType==SQLITE_MUTEX_FAST || p->eType==SQLITE_MUTEX_RECURSIVE ){
1.108 + free(p);
1.109 + }
1.110 +}
1.111 +
1.112 +/*
1.113 +** Enter a countable mutex. Block until entry is safe.
1.114 +*/
1.115 +static void counterMutexEnter(sqlite3_mutex *p){
1.116 + assert( g.isInit );
1.117 + g.aCounter[p->eType]++;
1.118 + g.m.xMutexEnter(p->pReal);
1.119 +}
1.120 +
1.121 +/*
1.122 +** Try to enter a mutex. Return true on success.
1.123 +*/
1.124 +static int counterMutexTry(sqlite3_mutex *p){
1.125 + assert( g.isInit );
1.126 + g.aCounter[p->eType]++;
1.127 + if( g.disableTry ) return SQLITE_BUSY;
1.128 + return g.m.xMutexTry(p->pReal);
1.129 +}
1.130 +
1.131 +/* Leave a mutex
1.132 +*/
1.133 +static void counterMutexLeave(sqlite3_mutex *p){
1.134 + assert( g.isInit );
1.135 + g.m.xMutexLeave(p->pReal);
1.136 +}
1.137 +
1.138 +/*
1.139 +** sqlite3_shutdown
1.140 +*/
1.141 +static int test_shutdown(
1.142 + void * clientData,
1.143 + Tcl_Interp *interp,
1.144 + int objc,
1.145 + Tcl_Obj *CONST objv[]
1.146 +){
1.147 + int rc;
1.148 +
1.149 + if( objc!=1 ){
1.150 + Tcl_WrongNumArgs(interp, 1, objv, "");
1.151 + return TCL_ERROR;
1.152 + }
1.153 +
1.154 + rc = sqlite3_shutdown();
1.155 + Tcl_SetResult(interp, (char *)sqlite3TestErrorName(rc), TCL_VOLATILE);
1.156 + return TCL_OK;
1.157 +}
1.158 +
1.159 +/*
1.160 +** sqlite3_initialize
1.161 +*/
1.162 +static int test_initialize(
1.163 + void * clientData,
1.164 + Tcl_Interp *interp,
1.165 + int objc,
1.166 + Tcl_Obj *CONST objv[]
1.167 +){
1.168 + int rc;
1.169 +
1.170 + if( objc!=1 ){
1.171 + Tcl_WrongNumArgs(interp, 1, objv, "");
1.172 + return TCL_ERROR;
1.173 + }
1.174 +
1.175 + rc = sqlite3_initialize();
1.176 + Tcl_SetResult(interp, (char *)sqlite3TestErrorName(rc), TCL_VOLATILE);
1.177 + return TCL_OK;
1.178 +}
1.179 +
1.180 +/*
1.181 +** install_mutex_counters BOOLEAN
1.182 +*/
1.183 +static int test_install_mutex_counters(
1.184 + void * clientData,
1.185 + Tcl_Interp *interp,
1.186 + int objc,
1.187 + Tcl_Obj *CONST objv[]
1.188 +){
1.189 + int rc = SQLITE_OK;
1.190 + int isInstall;
1.191 +
1.192 + sqlite3_mutex_methods counter_methods = {
1.193 + counterMutexInit,
1.194 + counterMutexEnd,
1.195 + counterMutexAlloc,
1.196 + counterMutexFree,
1.197 + counterMutexEnter,
1.198 + counterMutexTry,
1.199 + counterMutexLeave,
1.200 + counterMutexHeld,
1.201 + counterMutexNotheld
1.202 + };
1.203 +
1.204 + if( objc!=2 ){
1.205 + Tcl_WrongNumArgs(interp, 1, objv, "BOOLEAN");
1.206 + return TCL_ERROR;
1.207 + }
1.208 + if( TCL_OK!=Tcl_GetBooleanFromObj(interp, objv[1], &isInstall) ){
1.209 + return TCL_ERROR;
1.210 + }
1.211 +
1.212 + assert(isInstall==0 || isInstall==1);
1.213 + assert(g.isInstalled==0 || g.isInstalled==1);
1.214 + if( isInstall==g.isInstalled ){
1.215 + Tcl_AppendResult(interp, "mutex counters are ", 0);
1.216 + Tcl_AppendResult(interp, isInstall?"already installed":"not installed", 0);
1.217 + return TCL_ERROR;
1.218 + }
1.219 +
1.220 + if( isInstall ){
1.221 + assert( g.m.xMutexAlloc==0 );
1.222 + rc = sqlite3_config(SQLITE_CONFIG_GETMUTEX, &g.m);
1.223 + if( rc==SQLITE_OK ){
1.224 + sqlite3_config(SQLITE_CONFIG_MUTEX, &counter_methods);
1.225 + }
1.226 + g.disableTry = 0;
1.227 + }else{
1.228 + assert( g.m.xMutexAlloc );
1.229 + rc = sqlite3_config(SQLITE_CONFIG_MUTEX, &g.m);
1.230 + memset(&g.m, 0, sizeof(sqlite3_mutex_methods));
1.231 + }
1.232 +
1.233 + if( rc==SQLITE_OK ){
1.234 + g.isInstalled = isInstall;
1.235 + }
1.236 +
1.237 + Tcl_SetResult(interp, (char *)sqlite3TestErrorName(rc), TCL_VOLATILE);
1.238 + return TCL_OK;
1.239 +}
1.240 +
1.241 +/*
1.242 +** read_mutex_counters
1.243 +*/
1.244 +static int test_read_mutex_counters(
1.245 + void * clientData,
1.246 + Tcl_Interp *interp,
1.247 + int objc,
1.248 + Tcl_Obj *CONST objv[]
1.249 +){
1.250 + Tcl_Obj *pRet;
1.251 + int ii;
1.252 + char *aName[8] = {
1.253 + "fast", "recursive", "static_master", "static_mem",
1.254 + "static_mem2", "static_prng", "static_lru", "static_lru2"
1.255 + };
1.256 +
1.257 + if( objc!=1 ){
1.258 + Tcl_WrongNumArgs(interp, 1, objv, "");
1.259 + return TCL_ERROR;
1.260 + }
1.261 +
1.262 + pRet = Tcl_NewObj();
1.263 + Tcl_IncrRefCount(pRet);
1.264 + for(ii=0; ii<8; ii++){
1.265 + Tcl_ListObjAppendElement(interp, pRet, Tcl_NewStringObj(aName[ii], -1));
1.266 + Tcl_ListObjAppendElement(interp, pRet, Tcl_NewIntObj(g.aCounter[ii]));
1.267 + }
1.268 + Tcl_SetObjResult(interp, pRet);
1.269 + Tcl_DecrRefCount(pRet);
1.270 +
1.271 + return TCL_OK;
1.272 +}
1.273 +
1.274 +/*
1.275 +** clear_mutex_counters
1.276 +*/
1.277 +static int test_clear_mutex_counters(
1.278 + void * clientData,
1.279 + Tcl_Interp *interp,
1.280 + int objc,
1.281 + Tcl_Obj *CONST objv[]
1.282 +){
1.283 + int ii;
1.284 +
1.285 + if( objc!=1 ){
1.286 + Tcl_WrongNumArgs(interp, 1, objv, "");
1.287 + return TCL_ERROR;
1.288 + }
1.289 +
1.290 + for(ii=0; ii<8; ii++){
1.291 + g.aCounter[ii] = 0;
1.292 + }
1.293 + return TCL_OK;
1.294 +}
1.295 +
1.296 +/*
1.297 +** Create and free a mutex. Return the mutex pointer. The pointer
1.298 +** will be invalid since the mutex has already been freed. The
1.299 +** return pointer just checks to see if the mutex really was allocated.
1.300 +*/
1.301 +static int test_alloc_mutex(
1.302 + void * clientData,
1.303 + Tcl_Interp *interp,
1.304 + int objc,
1.305 + Tcl_Obj *CONST objv[]
1.306 +){
1.307 +#if SQLITE_THREADSAFE
1.308 + sqlite3_mutex *p = sqlite3_mutex_alloc(SQLITE_MUTEX_FAST);
1.309 + char zBuf[100];
1.310 + sqlite3_mutex_free(p);
1.311 + sqlite3_snprintf(sizeof(zBuf), zBuf, "%p", p);
1.312 + Tcl_AppendResult(interp, zBuf, (char*)0);
1.313 +#endif
1.314 + return TCL_OK;
1.315 +}
1.316 +
1.317 +/*
1.318 +** sqlite3_config OPTION
1.319 +**
1.320 +** OPTION can be either one of the keywords:
1.321 +**
1.322 +** SQLITE_CONFIG_SINGLETHREAD
1.323 +** SQLITE_CONFIG_MULTITHREAD
1.324 +** SQLITE_CONFIG_SERIALIZED
1.325 +**
1.326 +** Or OPTION can be an raw integer.
1.327 +*/
1.328 +static int test_config(
1.329 + void * clientData,
1.330 + Tcl_Interp *interp,
1.331 + int objc,
1.332 + Tcl_Obj *CONST objv[]
1.333 +){
1.334 + struct ConfigOption {
1.335 + const char *zName;
1.336 + int iValue;
1.337 + } aOpt[] = {
1.338 + {"singlethread", SQLITE_CONFIG_SINGLETHREAD},
1.339 + {"multithread", SQLITE_CONFIG_MULTITHREAD},
1.340 + {"serialized", SQLITE_CONFIG_SERIALIZED},
1.341 + {0, 0}
1.342 + };
1.343 + int s = sizeof(struct ConfigOption);
1.344 + int i;
1.345 + int rc;
1.346 +
1.347 + if( objc!=2 ){
1.348 + Tcl_WrongNumArgs(interp, 1, objv, "");
1.349 + return TCL_ERROR;
1.350 + }
1.351 +
1.352 + if( Tcl_GetIndexFromObjStruct(interp, objv[1], aOpt, s, "flag", 0, &i) ){
1.353 + if( Tcl_GetIntFromObj(interp, objv[1], &i) ){
1.354 + return TCL_ERROR;
1.355 + }
1.356 + }else{
1.357 + i = aOpt[i].iValue;
1.358 + }
1.359 +
1.360 + rc = sqlite3_config(i);
1.361 + Tcl_SetResult(interp, (char *)sqlite3TestErrorName(rc), TCL_VOLATILE);
1.362 + return TCL_OK;
1.363 +}
1.364 +
1.365 +int Sqlitetest_mutex_Init(Tcl_Interp *interp){
1.366 + static struct {
1.367 + char *zName;
1.368 + Tcl_ObjCmdProc *xProc;
1.369 + } aCmd[] = {
1.370 + { "sqlite3_shutdown", (Tcl_ObjCmdProc*)test_shutdown },
1.371 + { "sqlite3_initialize", (Tcl_ObjCmdProc*)test_initialize },
1.372 + { "sqlite3_config", (Tcl_ObjCmdProc*)test_config },
1.373 +
1.374 + { "alloc_dealloc_mutex", (Tcl_ObjCmdProc*)test_alloc_mutex },
1.375 + { "install_mutex_counters", (Tcl_ObjCmdProc*)test_install_mutex_counters },
1.376 + { "read_mutex_counters", (Tcl_ObjCmdProc*)test_read_mutex_counters },
1.377 + { "clear_mutex_counters", (Tcl_ObjCmdProc*)test_clear_mutex_counters },
1.378 + };
1.379 + int i;
1.380 + for(i=0; i<sizeof(aCmd)/sizeof(aCmd[0]); i++){
1.381 + Tcl_CreateObjCommand(interp, aCmd[i].zName, aCmd[i].xProc, 0, 0);
1.382 + }
1.383 + memset(&g, 0, sizeof(g));
1.384 +
1.385 + Tcl_LinkVar(interp, "disable_mutex_init",
1.386 + (char*)&g.disableInit, TCL_LINK_INT);
1.387 + Tcl_LinkVar(interp, "disable_mutex_try",
1.388 + (char*)&g.disableTry, TCL_LINK_INT);
1.389 + return SQLITE_OK;
1.390 +}