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