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 */