os/persistentdata/persistentstorage/sqlite3api/SQLite/mutex_os2.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_os2.c	Fri Jun 15 03:10:57 2012 +0200
     1.3 @@ -0,0 +1,273 @@
     1.4 +/*
     1.5 +** 2007 August 28
     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 for OS/2
    1.16 +**
    1.17 +** $Id: mutex_os2.c,v 1.10 2008/06/23 22:13:28 pweilbacher Exp $
    1.18 +*/
    1.19 +#include "sqliteInt.h"
    1.20 +
    1.21 +/*
    1.22 +** The code in this file is only used if SQLITE_MUTEX_OS2 is defined.
    1.23 +** See the mutex.h file for details.
    1.24 +*/
    1.25 +#ifdef SQLITE_MUTEX_OS2
    1.26 +
    1.27 +/********************** OS/2 Mutex Implementation **********************
    1.28 +**
    1.29 +** This implementation of mutexes is built using the OS/2 API.
    1.30 +*/
    1.31 +
    1.32 +/*
    1.33 +** The mutex object
    1.34 +** Each recursive mutex is an instance of the following structure.
    1.35 +*/
    1.36 +struct sqlite3_mutex {
    1.37 +  HMTX mutex;       /* Mutex controlling the lock */
    1.38 +  int  id;          /* Mutex type */
    1.39 +  int  nRef;        /* Number of references */
    1.40 +  TID  owner;       /* Thread holding this mutex */
    1.41 +};
    1.42 +
    1.43 +#define OS2_MUTEX_INITIALIZER   0,0,0,0
    1.44 +
    1.45 +/*
    1.46 +** Initialize and deinitialize the mutex subsystem.
    1.47 +*/
    1.48 +static int os2MutexInit(void){ return SQLITE_OK; }
    1.49 +static int os2MutexEnd(void){ return SQLITE_OK; }
    1.50 +
    1.51 +/*
    1.52 +** The sqlite3_mutex_alloc() routine allocates a new
    1.53 +** mutex and returns a pointer to it.  If it returns NULL
    1.54 +** that means that a mutex could not be allocated. 
    1.55 +** SQLite will unwind its stack and return an error.  The argument
    1.56 +** to sqlite3_mutex_alloc() is one of these integer constants:
    1.57 +**
    1.58 +** <ul>
    1.59 +** <li>  SQLITE_MUTEX_FAST               0
    1.60 +** <li>  SQLITE_MUTEX_RECURSIVE          1
    1.61 +** <li>  SQLITE_MUTEX_STATIC_MASTER      2
    1.62 +** <li>  SQLITE_MUTEX_STATIC_MEM         3
    1.63 +** <li>  SQLITE_MUTEX_STATIC_PRNG        4
    1.64 +** </ul>
    1.65 +**
    1.66 +** The first two constants cause sqlite3_mutex_alloc() to create
    1.67 +** a new mutex.  The new mutex is recursive when SQLITE_MUTEX_RECURSIVE
    1.68 +** is used but not necessarily so when SQLITE_MUTEX_FAST is used.
    1.69 +** The mutex implementation does not need to make a distinction
    1.70 +** between SQLITE_MUTEX_RECURSIVE and SQLITE_MUTEX_FAST if it does
    1.71 +** not want to.  But SQLite will only request a recursive mutex in
    1.72 +** cases where it really needs one.  If a faster non-recursive mutex
    1.73 +** implementation is available on the host platform, the mutex subsystem
    1.74 +** might return such a mutex in response to SQLITE_MUTEX_FAST.
    1.75 +**
    1.76 +** The other allowed parameters to sqlite3_mutex_alloc() each return
    1.77 +** a pointer to a static preexisting mutex.  Three static mutexes are
    1.78 +** used by the current version of SQLite.  Future versions of SQLite
    1.79 +** may add additional static mutexes.  Static mutexes are for internal
    1.80 +** use by SQLite only.  Applications that use SQLite mutexes should
    1.81 +** use only the dynamic mutexes returned by SQLITE_MUTEX_FAST or
    1.82 +** SQLITE_MUTEX_RECURSIVE.
    1.83 +**
    1.84 +** Note that if one of the dynamic mutex parameters (SQLITE_MUTEX_FAST
    1.85 +** or SQLITE_MUTEX_RECURSIVE) is used then sqlite3_mutex_alloc()
    1.86 +** returns a different mutex on every call.  But for the static
    1.87 +** mutex types, the same mutex is returned on every call that has
    1.88 +** the same type number.
    1.89 +*/
    1.90 +static sqlite3_mutex *os2MutexAlloc(int iType){
    1.91 +  sqlite3_mutex *p = NULL;
    1.92 +  switch( iType ){
    1.93 +    case SQLITE_MUTEX_FAST:
    1.94 +    case SQLITE_MUTEX_RECURSIVE: {
    1.95 +      p = sqlite3MallocZero( sizeof(*p) );
    1.96 +      if( p ){
    1.97 +        p->id = iType;
    1.98 +        if( DosCreateMutexSem( 0, &p->mutex, 0, FALSE ) != NO_ERROR ){
    1.99 +          sqlite3_free( p );
   1.100 +          p = NULL;
   1.101 +        }
   1.102 +      }
   1.103 +      break;
   1.104 +    }
   1.105 +    default: {
   1.106 +      static volatile int isInit = 0;
   1.107 +      static sqlite3_mutex staticMutexes[] = {
   1.108 +        { OS2_MUTEX_INITIALIZER, },
   1.109 +        { OS2_MUTEX_INITIALIZER, },
   1.110 +        { OS2_MUTEX_INITIALIZER, },
   1.111 +        { OS2_MUTEX_INITIALIZER, },
   1.112 +        { OS2_MUTEX_INITIALIZER, },
   1.113 +        { OS2_MUTEX_INITIALIZER, },
   1.114 +      };
   1.115 +      if ( !isInit ){
   1.116 +        APIRET rc;
   1.117 +        PTIB ptib;
   1.118 +        PPIB ppib;
   1.119 +        HMTX mutex;
   1.120 +        char name[32];
   1.121 +        DosGetInfoBlocks( &ptib, &ppib );
   1.122 +        sqlite3_snprintf( sizeof(name), name, "\\SEM32\\SQLITE%04x",
   1.123 +                          ppib->pib_ulpid );
   1.124 +        while( !isInit ){
   1.125 +          mutex = 0;
   1.126 +          rc = DosCreateMutexSem( name, &mutex, 0, FALSE);
   1.127 +          if( rc == NO_ERROR ){
   1.128 +            int i;
   1.129 +            if( !isInit ){
   1.130 +              for( i = 0; i < sizeof(staticMutexes)/sizeof(staticMutexes[0]); i++ ){
   1.131 +                DosCreateMutexSem( 0, &staticMutexes[i].mutex, 0, FALSE );
   1.132 +              }
   1.133 +              isInit = 1;
   1.134 +            }
   1.135 +            DosCloseMutexSem( mutex );
   1.136 +          }else if( rc == ERROR_DUPLICATE_NAME ){
   1.137 +            DosSleep( 1 );
   1.138 +          }else{
   1.139 +            return p;
   1.140 +          }
   1.141 +        }
   1.142 +      }
   1.143 +      assert( iType-2 >= 0 );
   1.144 +      assert( iType-2 < sizeof(staticMutexes)/sizeof(staticMutexes[0]) );
   1.145 +      p = &staticMutexes[iType-2];
   1.146 +      p->id = iType;
   1.147 +      break;
   1.148 +    }
   1.149 +  }
   1.150 +  return p;
   1.151 +}
   1.152 +
   1.153 +
   1.154 +/*
   1.155 +** This routine deallocates a previously allocated mutex.
   1.156 +** SQLite is careful to deallocate every mutex that it allocates.
   1.157 +*/
   1.158 +static void os2MutexFree(sqlite3_mutex *p){
   1.159 +  if( p==0 ) return;
   1.160 +  assert( p->nRef==0 );
   1.161 +  assert( p->id==SQLITE_MUTEX_FAST || p->id==SQLITE_MUTEX_RECURSIVE );
   1.162 +  DosCloseMutexSem( p->mutex );
   1.163 +  sqlite3_free( p );
   1.164 +}
   1.165 +
   1.166 +/*
   1.167 +** The sqlite3_mutex_enter() and sqlite3_mutex_try() routines attempt
   1.168 +** to enter a mutex.  If another thread is already within the mutex,
   1.169 +** sqlite3_mutex_enter() will block and sqlite3_mutex_try() will return
   1.170 +** SQLITE_BUSY.  The sqlite3_mutex_try() interface returns SQLITE_OK
   1.171 +** upon successful entry.  Mutexes created using SQLITE_MUTEX_RECURSIVE can
   1.172 +** be entered multiple times by the same thread.  In such cases the,
   1.173 +** mutex must be exited an equal number of times before another thread
   1.174 +** can enter.  If the same thread tries to enter any other kind of mutex
   1.175 +** more than once, the behavior is undefined.
   1.176 +*/
   1.177 +static void os2MutexEnter(sqlite3_mutex *p){
   1.178 +  TID tid;
   1.179 +  PID holder1;
   1.180 +  ULONG holder2;
   1.181 +  if( p==0 ) return;
   1.182 +  assert( p->id==SQLITE_MUTEX_RECURSIVE || os2MutexNotheld(p) );
   1.183 +  DosRequestMutexSem(p->mutex, SEM_INDEFINITE_WAIT);
   1.184 +  DosQueryMutexSem(p->mutex, &holder1, &tid, &holder2);
   1.185 +  p->owner = tid;
   1.186 +  p->nRef++;
   1.187 +}
   1.188 +static int os2MutexTry(sqlite3_mutex *p){
   1.189 +  int rc;
   1.190 +  TID tid;
   1.191 +  PID holder1;
   1.192 +  ULONG holder2;
   1.193 +  if( p==0 ) return SQLITE_OK;
   1.194 +  assert( p->id==SQLITE_MUTEX_RECURSIVE || os2MutexNotheld(p) );
   1.195 +  if( DosRequestMutexSem(p->mutex, SEM_IMMEDIATE_RETURN) == NO_ERROR) {
   1.196 +    DosQueryMutexSem(p->mutex, &holder1, &tid, &holder2);
   1.197 +    p->owner = tid;
   1.198 +    p->nRef++;
   1.199 +    rc = SQLITE_OK;
   1.200 +  } else {
   1.201 +    rc = SQLITE_BUSY;
   1.202 +  }
   1.203 +
   1.204 +  return rc;
   1.205 +}
   1.206 +
   1.207 +/*
   1.208 +** The sqlite3_mutex_leave() routine exits a mutex that was
   1.209 +** previously entered by the same thread.  The behavior
   1.210 +** is undefined if the mutex is not currently entered or
   1.211 +** is not currently allocated.  SQLite will never do either.
   1.212 +*/
   1.213 +static void os2MutexLeave(sqlite3_mutex *p){
   1.214 +  TID tid;
   1.215 +  PID holder1;
   1.216 +  ULONG holder2;
   1.217 +  if( p==0 ) return;
   1.218 +  assert( p->nRef>0 );
   1.219 +  DosQueryMutexSem(p->mutex, &holder1, &tid, &holder2);
   1.220 +  assert( p->owner==tid );
   1.221 +  p->nRef--;
   1.222 +  assert( p->nRef==0 || p->id==SQLITE_MUTEX_RECURSIVE );
   1.223 +  DosReleaseMutexSem(p->mutex);
   1.224 +}
   1.225 +
   1.226 +#ifdef SQLITE_DEBUG
   1.227 +/*
   1.228 +** The sqlite3_mutex_held() and sqlite3_mutex_notheld() routine are
   1.229 +** intended for use inside assert() statements.
   1.230 +*/
   1.231 +static int os2MutexHeld(sqlite3_mutex *p){
   1.232 +  TID tid;
   1.233 +  PID pid;
   1.234 +  ULONG ulCount;
   1.235 +  PTIB ptib;
   1.236 +  if( p!=0 ) {
   1.237 +    DosQueryMutexSem(p->mutex, &pid, &tid, &ulCount);
   1.238 +  } else {
   1.239 +    DosGetInfoBlocks(&ptib, NULL);
   1.240 +    tid = ptib->tib_ptib2->tib2_ultid;
   1.241 +  }
   1.242 +  return p==0 || (p->nRef!=0 && p->owner==tid);
   1.243 +}
   1.244 +static int os2MutexNotheld(sqlite3_mutex *p){
   1.245 +  TID tid;
   1.246 +  PID pid;
   1.247 +  ULONG ulCount;
   1.248 +  PTIB ptib;
   1.249 +  if( p!= 0 ) {
   1.250 +    DosQueryMutexSem(p->mutex, &pid, &tid, &ulCount);
   1.251 +  } else {
   1.252 +    DosGetInfoBlocks(&ptib, NULL);
   1.253 +    tid = ptib->tib_ptib2->tib2_ultid;
   1.254 +  }
   1.255 +  return p==0 || p->nRef==0 || p->owner!=tid;
   1.256 +}
   1.257 +#endif
   1.258 +
   1.259 +sqlite3_mutex_methods *sqlite3DefaultMutex(void){
   1.260 +  static sqlite3_mutex_methods sMutex = {
   1.261 +    os2MutexInit,
   1.262 +    os2MutexEnd,
   1.263 +    os2MutexAlloc,
   1.264 +    os2MutexFree,
   1.265 +    os2MutexEnter,
   1.266 +    os2MutexTry,
   1.267 +    os2MutexLeave,
   1.268 +#ifdef SQLITE_DEBUG
   1.269 +    os2MutexHeld,
   1.270 +    os2MutexNotheld
   1.271 +#endif
   1.272 +  };
   1.273 +
   1.274 +  return &sMutex;
   1.275 +}
   1.276 +#endif /* SQLITE_MUTEX_OS2 */