os/persistentdata/persistentstorage/sqlite3api/SQLite/mutex.c
changeset 0 bde4ae8d615e
     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 */