1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/persistentdata/persistentstorage/sqlite3api/SQLite/mutex.c Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,277 @@
1.4 +/*
1.5 +** 2007 August 14
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 +** This file contains the C functions that implement mutexes.
1.16 +**
1.17 +** The implementation in this file does not provide any mutual
1.18 +** exclusion and is thus suitable for use only in applications
1.19 +** that use SQLite in a single thread. But this implementation
1.20 +** does do a lot of error checking on mutexes to make sure they
1.21 +** are called correctly and at appropriate times. Hence, this
1.22 +** implementation is suitable for testing.
1.23 +** debugging purposes
1.24 +**
1.25 +** $Id: mutex.c,v 1.28 2008/09/01 18:34:20 danielk1977 Exp $
1.26 +*/
1.27 +#include "sqliteInt.h"
1.28 +
1.29 +#ifndef SQLITE_MUTEX_NOOP
1.30 +/*
1.31 +** Initialize the mutex system.
1.32 +*/
1.33 +int sqlite3MutexInit(void){
1.34 + int rc = SQLITE_OK;
1.35 + if( sqlite3GlobalConfig.bCoreMutex ){
1.36 + if( !sqlite3GlobalConfig.mutex.xMutexAlloc ){
1.37 + /* If the xMutexAlloc method has not been set, then the user did not
1.38 + ** install a mutex implementation via sqlite3_config() prior to
1.39 + ** sqlite3_initialize() being called. This block copies pointers to
1.40 + ** the default implementation into the sqlite3GlobalConfig structure.
1.41 + **
1.42 + ** The danger is that although sqlite3_config() is not a threadsafe
1.43 + ** API, sqlite3_initialize() is, and so multiple threads may be
1.44 + ** attempting to run this function simultaneously. To guard write
1.45 + ** access to the sqlite3GlobalConfig structure, the 'MASTER' static mutex
1.46 + ** is obtained before modifying it.
1.47 + */
1.48 + sqlite3_mutex_methods *p = sqlite3DefaultMutex();
1.49 + sqlite3_mutex *pMaster = 0;
1.50 +
1.51 + rc = p->xMutexInit();
1.52 + if( rc==SQLITE_OK ){
1.53 + pMaster = p->xMutexAlloc(SQLITE_MUTEX_STATIC_MASTER);
1.54 + assert(pMaster);
1.55 + p->xMutexEnter(pMaster);
1.56 + assert( sqlite3GlobalConfig.mutex.xMutexAlloc==0
1.57 + || sqlite3GlobalConfig.mutex.xMutexAlloc==p->xMutexAlloc
1.58 + );
1.59 + if( !sqlite3GlobalConfig.mutex.xMutexAlloc ){
1.60 + sqlite3GlobalConfig.mutex = *p;
1.61 + }
1.62 + p->xMutexLeave(pMaster);
1.63 + }
1.64 + }else{
1.65 + rc = sqlite3GlobalConfig.mutex.xMutexInit();
1.66 + }
1.67 + }
1.68 +
1.69 + return rc;
1.70 +}
1.71 +
1.72 +/*
1.73 +** Shutdown the mutex system. This call frees resources allocated by
1.74 +** sqlite3MutexInit().
1.75 +*/
1.76 +int sqlite3MutexEnd(void){
1.77 + int rc = SQLITE_OK;
1.78 + rc = sqlite3GlobalConfig.mutex.xMutexEnd();
1.79 + return rc;
1.80 +}
1.81 +
1.82 +/*
1.83 +** Retrieve a pointer to a static mutex or allocate a new dynamic one.
1.84 +*/
1.85 +SQLITE_EXPORT sqlite3_mutex *sqlite3_mutex_alloc(int id){
1.86 +#ifndef SQLITE_OMIT_AUTOINIT
1.87 + if( sqlite3_initialize() ) return 0;
1.88 +#endif
1.89 + return sqlite3GlobalConfig.mutex.xMutexAlloc(id);
1.90 +}
1.91 +
1.92 +sqlite3_mutex *sqlite3MutexAlloc(int id){
1.93 + if( !sqlite3GlobalConfig.bCoreMutex ){
1.94 + return 0;
1.95 + }
1.96 + return sqlite3GlobalConfig.mutex.xMutexAlloc(id);
1.97 +}
1.98 +
1.99 +/*
1.100 +** Free a dynamic mutex.
1.101 +*/
1.102 +SQLITE_EXPORT void sqlite3_mutex_free(sqlite3_mutex *p){
1.103 + if( p ){
1.104 + sqlite3GlobalConfig.mutex.xMutexFree(p);
1.105 + }
1.106 +}
1.107 +
1.108 +/*
1.109 +** Obtain the mutex p. If some other thread already has the mutex, block
1.110 +** until it can be obtained.
1.111 +*/
1.112 +SQLITE_EXPORT void sqlite3_mutex_enter(sqlite3_mutex *p){
1.113 + if( p ){
1.114 + sqlite3GlobalConfig.mutex.xMutexEnter(p);
1.115 + }
1.116 +}
1.117 +
1.118 +/*
1.119 +** Obtain the mutex p. If successful, return SQLITE_OK. Otherwise, if another
1.120 +** thread holds the mutex and it cannot be obtained, return SQLITE_BUSY.
1.121 +*/
1.122 +SQLITE_EXPORT int sqlite3_mutex_try(sqlite3_mutex *p){
1.123 + int rc = SQLITE_OK;
1.124 + if( p ){
1.125 + return sqlite3GlobalConfig.mutex.xMutexTry(p);
1.126 + }
1.127 + return rc;
1.128 +}
1.129 +
1.130 +/*
1.131 +** The sqlite3_mutex_leave() routine exits a mutex that was previously
1.132 +** entered by the same thread. The behavior is undefined if the mutex
1.133 +** is not currently entered. If a NULL pointer is passed as an argument
1.134 +** this function is a no-op.
1.135 +*/
1.136 +SQLITE_EXPORT void sqlite3_mutex_leave(sqlite3_mutex *p){
1.137 + if( p ){
1.138 + sqlite3GlobalConfig.mutex.xMutexLeave(p);
1.139 + }
1.140 +}
1.141 +
1.142 +#ifndef NDEBUG
1.143 +/*
1.144 +** The sqlite3_mutex_held() and sqlite3_mutex_notheld() routine are
1.145 +** intended for use inside assert() statements.
1.146 +*/
1.147 +SQLITE_EXPORT int sqlite3_mutex_held(sqlite3_mutex *p){
1.148 + return p==0 || sqlite3GlobalConfig.mutex.xMutexHeld(p);
1.149 +}
1.150 +SQLITE_EXPORT int sqlite3_mutex_notheld(sqlite3_mutex *p){
1.151 + return p==0 || sqlite3GlobalConfig.mutex.xMutexNotheld(p);
1.152 +}
1.153 +#else
1.154 +
1.155 +SQLITE_EXPORT int sqlite3_mutex_held(sqlite3_mutex *p){return 1;}
1.156 +SQLITE_EXPORT int sqlite3_mutex_notheld(sqlite3_mutex *p){return 1;}
1.157 +
1.158 +#endif
1.159 +#endif
1.160 +
1.161 +#ifdef SQLITE_MUTEX_NOOP_DEBUG
1.162 +/*
1.163 +** In this implementation, mutexes do not provide any mutual exclusion.
1.164 +** But the error checking is provided. This implementation is useful
1.165 +** for test purposes.
1.166 +*/
1.167 +
1.168 +/*
1.169 +** The mutex object
1.170 +*/
1.171 +struct sqlite3_mutex {
1.172 + int id; /* The mutex type */
1.173 + int cnt; /* Number of entries without a matching leave */
1.174 +};
1.175 +
1.176 +/*
1.177 +** The sqlite3_mutex_held() and sqlite3_mutex_notheld() routine are
1.178 +** intended for use inside assert() statements.
1.179 +*/
1.180 +static int noopMutexHeld(sqlite3_mutex *p){
1.181 + return p==0 || p->cnt>0;
1.182 +}
1.183 +static int noopMutexNotheld(sqlite3_mutex *p){
1.184 + return p==0 || p->cnt==0;
1.185 +}
1.186 +
1.187 +/*
1.188 +** Initialize and deinitialize the mutex subsystem.
1.189 +*/
1.190 +static int noopMutexInit(void){ return SQLITE_OK; }
1.191 +static int noopMutexEnd(void){ return SQLITE_OK; }
1.192 +
1.193 +/*
1.194 +** The sqlite3_mutex_alloc() routine allocates a new
1.195 +** mutex and returns a pointer to it. If it returns NULL
1.196 +** that means that a mutex could not be allocated.
1.197 +*/
1.198 +static sqlite3_mutex *noopMutexAlloc(int id){
1.199 + static sqlite3_mutex aStatic[6];
1.200 + sqlite3_mutex *pNew = 0;
1.201 + switch( id ){
1.202 + case SQLITE_MUTEX_FAST:
1.203 + case SQLITE_MUTEX_RECURSIVE: {
1.204 + pNew = sqlite3Malloc(sizeof(*pNew));
1.205 + if( pNew ){
1.206 + pNew->id = id;
1.207 + pNew->cnt = 0;
1.208 + }
1.209 + break;
1.210 + }
1.211 + default: {
1.212 + assert( id-2 >= 0 );
1.213 + assert( id-2 < sizeof(aStatic)/sizeof(aStatic[0]) );
1.214 + pNew = &aStatic[id-2];
1.215 + pNew->id = id;
1.216 + break;
1.217 + }
1.218 + }
1.219 + return pNew;
1.220 +}
1.221 +
1.222 +/*
1.223 +** This routine deallocates a previously allocated mutex.
1.224 +*/
1.225 +static void noopMutexFree(sqlite3_mutex *p){
1.226 + assert( p->cnt==0 );
1.227 + assert( p->id==SQLITE_MUTEX_FAST || p->id==SQLITE_MUTEX_RECURSIVE );
1.228 + sqlite3_free(p);
1.229 +}
1.230 +
1.231 +/*
1.232 +** The sqlite3_mutex_enter() and sqlite3_mutex_try() routines attempt
1.233 +** to enter a mutex. If another thread is already within the mutex,
1.234 +** sqlite3_mutex_enter() will block and sqlite3_mutex_try() will return
1.235 +** SQLITE_BUSY. The sqlite3_mutex_try() interface returns SQLITE_OK
1.236 +** upon successful entry. Mutexes created using SQLITE_MUTEX_RECURSIVE can
1.237 +** be entered multiple times by the same thread. In such cases the,
1.238 +** mutex must be exited an equal number of times before another thread
1.239 +** can enter. If the same thread tries to enter any other kind of mutex
1.240 +** more than once, the behavior is undefined.
1.241 +*/
1.242 +static void noopMutexEnter(sqlite3_mutex *p){
1.243 + assert( p->id==SQLITE_MUTEX_RECURSIVE || noopMutexNotheld(p) );
1.244 + p->cnt++;
1.245 +}
1.246 +static int noopMutexTry(sqlite3_mutex *p){
1.247 + assert( p->id==SQLITE_MUTEX_RECURSIVE || noopMutexNotheld(p) );
1.248 + p->cnt++;
1.249 + return SQLITE_OK;
1.250 +}
1.251 +
1.252 +/*
1.253 +** The sqlite3_mutex_leave() routine exits a mutex that was
1.254 +** previously entered by the same thread. The behavior
1.255 +** is undefined if the mutex is not currently entered or
1.256 +** is not currently allocated. SQLite will never do either.
1.257 +*/
1.258 +static void noopMutexLeave(sqlite3_mutex *p){
1.259 + assert( noopMutexHeld(p) );
1.260 + p->cnt--;
1.261 + assert( p->id==SQLITE_MUTEX_RECURSIVE || noopMutexNotheld(p) );
1.262 +}
1.263 +
1.264 +sqlite3_mutex_methods *sqlite3DefaultMutex(void){
1.265 + static sqlite3_mutex_methods sMutex = {
1.266 + noopMutexInit,
1.267 + noopMutexEnd,
1.268 + noopMutexAlloc,
1.269 + noopMutexFree,
1.270 + noopMutexEnter,
1.271 + noopMutexTry,
1.272 + noopMutexLeave,
1.273 +
1.274 + noopMutexHeld,
1.275 + noopMutexNotheld
1.276 + };
1.277 +
1.278 + return &sMutex;
1.279 +}
1.280 +#endif /* SQLITE_MUTEX_NOOP_DEBUG */