os/persistentdata/persistentstorage/sqlite3api/SQLite/mutex_w32.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_w32.c	Fri Jun 15 03:10:57 2012 +0200
     1.3 @@ -0,0 +1,244 @@
     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 for win32
    1.16 +**
    1.17 +** $Id: mutex_w32.c,v 1.11 2008/06/26 10:41:19 danielk1977 Exp $
    1.18 +*/
    1.19 +#include "sqliteInt.h"
    1.20 +
    1.21 +/*
    1.22 +** The code in this file is only used if we are compiling multithreaded
    1.23 +** on a win32 system.
    1.24 +*/
    1.25 +#ifdef SQLITE_MUTEX_W32
    1.26 +
    1.27 +/*
    1.28 +** Each recursive mutex is an instance of the following structure.
    1.29 +*/
    1.30 +struct sqlite3_mutex {
    1.31 +  CRITICAL_SECTION mutex;    /* Mutex controlling the lock */
    1.32 +  int id;                    /* Mutex type */
    1.33 +  int nRef;                  /* Number of enterances */
    1.34 +  DWORD owner;               /* Thread holding this mutex */
    1.35 +};
    1.36 +
    1.37 +/*
    1.38 +** Return true (non-zero) if we are running under WinNT, Win2K, WinXP,
    1.39 +** or WinCE.  Return false (zero) for Win95, Win98, or WinME.
    1.40 +**
    1.41 +** Here is an interesting observation:  Win95, Win98, and WinME lack
    1.42 +** the LockFileEx() API.  But we can still statically link against that
    1.43 +** API as long as we don't call it win running Win95/98/ME.  A call to
    1.44 +** this routine is used to determine if the host is Win95/98/ME or
    1.45 +** WinNT/2K/XP so that we will know whether or not we can safely call
    1.46 +** the LockFileEx() API.
    1.47 +*/
    1.48 +#if SQLITE_OS_WINCE
    1.49 +# define mutexIsNT()  (1)
    1.50 +#else
    1.51 +  static int mutexIsNT(void){
    1.52 +    static int osType = 0;
    1.53 +    if( osType==0 ){
    1.54 +      OSVERSIONINFO sInfo;
    1.55 +      sInfo.dwOSVersionInfoSize = sizeof(sInfo);
    1.56 +      GetVersionEx(&sInfo);
    1.57 +      osType = sInfo.dwPlatformId==VER_PLATFORM_WIN32_NT ? 2 : 1;
    1.58 +    }
    1.59 +    return osType==2;
    1.60 +  }
    1.61 +#endif /* SQLITE_OS_WINCE */
    1.62 +
    1.63 +
    1.64 +#ifdef SQLITE_DEBUG
    1.65 +/*
    1.66 +** The sqlite3_mutex_held() and sqlite3_mutex_notheld() routine are
    1.67 +** intended for use only inside assert() statements.
    1.68 +*/
    1.69 +static int winMutexHeld(sqlite3_mutex *p){
    1.70 +  return p->nRef!=0 && p->owner==GetCurrentThreadId();
    1.71 +}
    1.72 +static int winMutexNotheld(sqlite3_mutex *p){
    1.73 +  return p->nRef==0 || p->owner!=GetCurrentThreadId();
    1.74 +}
    1.75 +#endif
    1.76 +
    1.77 +
    1.78 +/*
    1.79 +** Initialize and deinitialize the mutex subsystem.
    1.80 +*/
    1.81 +static int winMutexInit(void){ return SQLITE_OK; }
    1.82 +static int winMutexEnd(void){ return SQLITE_OK; }
    1.83 +
    1.84 +/*
    1.85 +** The sqlite3_mutex_alloc() routine allocates a new
    1.86 +** mutex and returns a pointer to it.  If it returns NULL
    1.87 +** that means that a mutex could not be allocated.  SQLite
    1.88 +** will unwind its stack and return an error.  The argument
    1.89 +** to sqlite3_mutex_alloc() is one of these integer constants:
    1.90 +**
    1.91 +** <ul>
    1.92 +** <li>  SQLITE_MUTEX_FAST               0
    1.93 +** <li>  SQLITE_MUTEX_RECURSIVE          1
    1.94 +** <li>  SQLITE_MUTEX_STATIC_MASTER      2
    1.95 +** <li>  SQLITE_MUTEX_STATIC_MEM         3
    1.96 +** <li>  SQLITE_MUTEX_STATIC_PRNG        4
    1.97 +** </ul>
    1.98 +**
    1.99 +** The first two constants cause sqlite3_mutex_alloc() to create
   1.100 +** a new mutex.  The new mutex is recursive when SQLITE_MUTEX_RECURSIVE
   1.101 +** is used but not necessarily so when SQLITE_MUTEX_FAST is used.
   1.102 +** The mutex implementation does not need to make a distinction
   1.103 +** between SQLITE_MUTEX_RECURSIVE and SQLITE_MUTEX_FAST if it does
   1.104 +** not want to.  But SQLite will only request a recursive mutex in
   1.105 +** cases where it really needs one.  If a faster non-recursive mutex
   1.106 +** implementation is available on the host platform, the mutex subsystem
   1.107 +** might return such a mutex in response to SQLITE_MUTEX_FAST.
   1.108 +**
   1.109 +** The other allowed parameters to sqlite3_mutex_alloc() each return
   1.110 +** a pointer to a static preexisting mutex.  Three static mutexes are
   1.111 +** used by the current version of SQLite.  Future versions of SQLite
   1.112 +** may add additional static mutexes.  Static mutexes are for internal
   1.113 +** use by SQLite only.  Applications that use SQLite mutexes should
   1.114 +** use only the dynamic mutexes returned by SQLITE_MUTEX_FAST or
   1.115 +** SQLITE_MUTEX_RECURSIVE.
   1.116 +**
   1.117 +** Note that if one of the dynamic mutex parameters (SQLITE_MUTEX_FAST
   1.118 +** or SQLITE_MUTEX_RECURSIVE) is used then sqlite3_mutex_alloc()
   1.119 +** returns a different mutex on every call.  But for the static 
   1.120 +** mutex types, the same mutex is returned on every call that has
   1.121 +** the same type number.
   1.122 +*/
   1.123 +static sqlite3_mutex *winMutexAlloc(int iType){
   1.124 +  sqlite3_mutex *p;
   1.125 +
   1.126 +  switch( iType ){
   1.127 +    case SQLITE_MUTEX_FAST:
   1.128 +    case SQLITE_MUTEX_RECURSIVE: {
   1.129 +      p = sqlite3MallocZero( sizeof(*p) );
   1.130 +      if( p ){
   1.131 +        p->id = iType;
   1.132 +        InitializeCriticalSection(&p->mutex);
   1.133 +      }
   1.134 +      break;
   1.135 +    }
   1.136 +    default: {
   1.137 +      static sqlite3_mutex staticMutexes[6];
   1.138 +      static int isInit = 0;
   1.139 +      while( !isInit ){
   1.140 +        static long lock = 0;
   1.141 +        if( InterlockedIncrement(&lock)==1 ){
   1.142 +          int i;
   1.143 +          for(i=0; i<sizeof(staticMutexes)/sizeof(staticMutexes[0]); i++){
   1.144 +            InitializeCriticalSection(&staticMutexes[i].mutex);
   1.145 +          }
   1.146 +          isInit = 1;
   1.147 +        }else{
   1.148 +          Sleep(1);
   1.149 +        }
   1.150 +      }
   1.151 +      assert( iType-2 >= 0 );
   1.152 +      assert( iType-2 < sizeof(staticMutexes)/sizeof(staticMutexes[0]) );
   1.153 +      p = &staticMutexes[iType-2];
   1.154 +      p->id = iType;
   1.155 +      break;
   1.156 +    }
   1.157 +  }
   1.158 +  return p;
   1.159 +}
   1.160 +
   1.161 +
   1.162 +/*
   1.163 +** This routine deallocates a previously
   1.164 +** allocated mutex.  SQLite is careful to deallocate every
   1.165 +** mutex that it allocates.
   1.166 +*/
   1.167 +static void winMutexFree(sqlite3_mutex *p){
   1.168 +  assert( p );
   1.169 +  assert( p->nRef==0 );
   1.170 +  assert( p->id==SQLITE_MUTEX_FAST || p->id==SQLITE_MUTEX_RECURSIVE );
   1.171 +  DeleteCriticalSection(&p->mutex);
   1.172 +  sqlite3_free(p);
   1.173 +}
   1.174 +
   1.175 +/*
   1.176 +** The sqlite3_mutex_enter() and sqlite3_mutex_try() routines attempt
   1.177 +** to enter a mutex.  If another thread is already within the mutex,
   1.178 +** sqlite3_mutex_enter() will block and sqlite3_mutex_try() will return
   1.179 +** SQLITE_BUSY.  The sqlite3_mutex_try() interface returns SQLITE_OK
   1.180 +** upon successful entry.  Mutexes created using SQLITE_MUTEX_RECURSIVE can
   1.181 +** be entered multiple times by the same thread.  In such cases the,
   1.182 +** mutex must be exited an equal number of times before another thread
   1.183 +** can enter.  If the same thread tries to enter any other kind of mutex
   1.184 +** more than once, the behavior is undefined.
   1.185 +*/
   1.186 +static void winMutexEnter(sqlite3_mutex *p){
   1.187 +  assert( p->id==SQLITE_MUTEX_RECURSIVE || winMutexNotheld(p) );
   1.188 +  EnterCriticalSection(&p->mutex);
   1.189 +  p->owner = GetCurrentThreadId(); 
   1.190 +  p->nRef++;
   1.191 +}
   1.192 +static int winMutexTry(sqlite3_mutex *p){
   1.193 +  int rc = SQLITE_BUSY;
   1.194 +  assert( p->id==SQLITE_MUTEX_RECURSIVE || winMutexNotheld(p) );
   1.195 +  /*
   1.196 +  ** The sqlite3_mutex_try() routine is very rarely used, and when it
   1.197 +  ** is used it is merely an optimization.  So it is OK for it to always
   1.198 +  ** fail.  
   1.199 +  **
   1.200 +  ** The TryEnterCriticalSection() interface is only available on WinNT.
   1.201 +  ** And some windows compilers complain if you try to use it without
   1.202 +  ** first doing some #defines that prevent SQLite from building on Win98.
   1.203 +  ** For that reason, we will omit this optimization for now.  See
   1.204 +  ** ticket #2685.
   1.205 +  */
   1.206 +#if 0
   1.207 +  if( mutexIsNT() && TryEnterCriticalSection(&p->mutex) ){
   1.208 +    p->owner = GetCurrentThreadId();
   1.209 +    p->nRef++;
   1.210 +    rc = SQLITE_OK;
   1.211 +  }
   1.212 +#endif
   1.213 +  return rc;
   1.214 +}
   1.215 +
   1.216 +/*
   1.217 +** The sqlite3_mutex_leave() routine exits a mutex that was
   1.218 +** previously entered by the same thread.  The behavior
   1.219 +** is undefined if the mutex is not currently entered or
   1.220 +** is not currently allocated.  SQLite will never do either.
   1.221 +*/
   1.222 +static void winMutexLeave(sqlite3_mutex *p){
   1.223 +  assert( p->nRef>0 );
   1.224 +  assert( p->owner==GetCurrentThreadId() );
   1.225 +  p->nRef--;
   1.226 +  assert( p->nRef==0 || p->id==SQLITE_MUTEX_RECURSIVE );
   1.227 +  LeaveCriticalSection(&p->mutex);
   1.228 +}
   1.229 +
   1.230 +sqlite3_mutex_methods *sqlite3DefaultMutex(void){
   1.231 +  static sqlite3_mutex_methods sMutex = {
   1.232 +    winMutexInit,
   1.233 +    winMutexEnd,
   1.234 +    winMutexAlloc,
   1.235 +    winMutexFree,
   1.236 +    winMutexEnter,
   1.237 +    winMutexTry,
   1.238 +    winMutexLeave,
   1.239 +#ifdef SQLITE_DEBUG
   1.240 +    winMutexHeld,
   1.241 +    winMutexNotheld
   1.242 +#endif
   1.243 +  };
   1.244 +
   1.245 +  return &sMutex;
   1.246 +}
   1.247 +#endif /* SQLITE_MUTEX_W32 */