os/persistentdata/persistentstorage/sql/SQLite364/mutex.c
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
/*
sl@0
     2
** 2007 August 14
sl@0
     3
**
sl@0
     4
** The author disclaims copyright to this source code.  In place of
sl@0
     5
** a legal notice, here is a blessing:
sl@0
     6
**
sl@0
     7
**    May you do good and not evil.
sl@0
     8
**    May you find forgiveness for yourself and forgive others.
sl@0
     9
**    May you share freely, never taking more than you give.
sl@0
    10
**
sl@0
    11
*************************************************************************
sl@0
    12
** This file contains the C functions that implement mutexes.
sl@0
    13
**
sl@0
    14
** This file contains code that is common across all mutex implementations.
sl@0
    15
sl@0
    16
**
sl@0
    17
** $Id: mutex.c,v 1.29 2008/10/07 15:25:48 drh Exp $
sl@0
    18
*/
sl@0
    19
#include "sqliteInt.h"
sl@0
    20
sl@0
    21
#ifndef SQLITE_MUTEX_OMIT
sl@0
    22
/*
sl@0
    23
** Initialize the mutex system.
sl@0
    24
*/
sl@0
    25
int sqlite3MutexInit(void){ 
sl@0
    26
  int rc = SQLITE_OK;
sl@0
    27
  if( sqlite3GlobalConfig.bCoreMutex ){
sl@0
    28
    if( !sqlite3GlobalConfig.mutex.xMutexAlloc ){
sl@0
    29
      /* If the xMutexAlloc method has not been set, then the user did not
sl@0
    30
      ** install a mutex implementation via sqlite3_config() prior to 
sl@0
    31
      ** sqlite3_initialize() being called. This block copies pointers to
sl@0
    32
      ** the default implementation into the sqlite3GlobalConfig structure.
sl@0
    33
      **
sl@0
    34
      ** The danger is that although sqlite3_config() is not a threadsafe
sl@0
    35
      ** API, sqlite3_initialize() is, and so multiple threads may be
sl@0
    36
      ** attempting to run this function simultaneously. To guard write
sl@0
    37
      ** access to the sqlite3GlobalConfig structure, the 'MASTER' static mutex
sl@0
    38
      ** is obtained before modifying it.
sl@0
    39
      */
sl@0
    40
      sqlite3_mutex_methods *p = sqlite3DefaultMutex();
sl@0
    41
      sqlite3_mutex *pMaster = 0;
sl@0
    42
  
sl@0
    43
      rc = p->xMutexInit();
sl@0
    44
      if( rc==SQLITE_OK ){
sl@0
    45
        pMaster = p->xMutexAlloc(SQLITE_MUTEX_STATIC_MASTER);
sl@0
    46
        assert(pMaster);
sl@0
    47
        p->xMutexEnter(pMaster);
sl@0
    48
        assert( sqlite3GlobalConfig.mutex.xMutexAlloc==0 
sl@0
    49
             || sqlite3GlobalConfig.mutex.xMutexAlloc==p->xMutexAlloc
sl@0
    50
        );
sl@0
    51
        if( !sqlite3GlobalConfig.mutex.xMutexAlloc ){
sl@0
    52
          sqlite3GlobalConfig.mutex = *p;
sl@0
    53
        }
sl@0
    54
        p->xMutexLeave(pMaster);
sl@0
    55
      }
sl@0
    56
    }else{
sl@0
    57
      rc = sqlite3GlobalConfig.mutex.xMutexInit();
sl@0
    58
    }
sl@0
    59
  }
sl@0
    60
sl@0
    61
  return rc;
sl@0
    62
}
sl@0
    63
sl@0
    64
/*
sl@0
    65
** Shutdown the mutex system. This call frees resources allocated by
sl@0
    66
** sqlite3MutexInit().
sl@0
    67
*/
sl@0
    68
int sqlite3MutexEnd(void){
sl@0
    69
  int rc = SQLITE_OK;
sl@0
    70
  if( sqlite3GlobalConfig.mutex.xMutexEnd ){
sl@0
    71
    rc = sqlite3GlobalConfig.mutex.xMutexEnd();
sl@0
    72
  }
sl@0
    73
  return rc;
sl@0
    74
}
sl@0
    75
sl@0
    76
/*
sl@0
    77
** Retrieve a pointer to a static mutex or allocate a new dynamic one.
sl@0
    78
*/
sl@0
    79
sqlite3_mutex *sqlite3_mutex_alloc(int id){
sl@0
    80
#ifndef SQLITE_OMIT_AUTOINIT
sl@0
    81
  if( sqlite3_initialize() ) return 0;
sl@0
    82
#endif
sl@0
    83
  return sqlite3GlobalConfig.mutex.xMutexAlloc(id);
sl@0
    84
}
sl@0
    85
sl@0
    86
sqlite3_mutex *sqlite3MutexAlloc(int id){
sl@0
    87
  if( !sqlite3GlobalConfig.bCoreMutex ){
sl@0
    88
    return 0;
sl@0
    89
  }
sl@0
    90
  return sqlite3GlobalConfig.mutex.xMutexAlloc(id);
sl@0
    91
}
sl@0
    92
sl@0
    93
/*
sl@0
    94
** Free a dynamic mutex.
sl@0
    95
*/
sl@0
    96
void sqlite3_mutex_free(sqlite3_mutex *p){
sl@0
    97
  if( p ){
sl@0
    98
    sqlite3GlobalConfig.mutex.xMutexFree(p);
sl@0
    99
  }
sl@0
   100
}
sl@0
   101
sl@0
   102
/*
sl@0
   103
** Obtain the mutex p. If some other thread already has the mutex, block
sl@0
   104
** until it can be obtained.
sl@0
   105
*/
sl@0
   106
void sqlite3_mutex_enter(sqlite3_mutex *p){
sl@0
   107
  if( p ){
sl@0
   108
    sqlite3GlobalConfig.mutex.xMutexEnter(p);
sl@0
   109
  }
sl@0
   110
}
sl@0
   111
sl@0
   112
/*
sl@0
   113
** Obtain the mutex p. If successful, return SQLITE_OK. Otherwise, if another
sl@0
   114
** thread holds the mutex and it cannot be obtained, return SQLITE_BUSY.
sl@0
   115
*/
sl@0
   116
int sqlite3_mutex_try(sqlite3_mutex *p){
sl@0
   117
  int rc = SQLITE_OK;
sl@0
   118
  if( p ){
sl@0
   119
    return sqlite3GlobalConfig.mutex.xMutexTry(p);
sl@0
   120
  }
sl@0
   121
  return rc;
sl@0
   122
}
sl@0
   123
sl@0
   124
/*
sl@0
   125
** The sqlite3_mutex_leave() routine exits a mutex that was previously
sl@0
   126
** entered by the same thread.  The behavior is undefined if the mutex 
sl@0
   127
** is not currently entered. If a NULL pointer is passed as an argument
sl@0
   128
** this function is a no-op.
sl@0
   129
*/
sl@0
   130
void sqlite3_mutex_leave(sqlite3_mutex *p){
sl@0
   131
  if( p ){
sl@0
   132
    sqlite3GlobalConfig.mutex.xMutexLeave(p);
sl@0
   133
  }
sl@0
   134
}
sl@0
   135
sl@0
   136
#ifndef NDEBUG
sl@0
   137
/*
sl@0
   138
** The sqlite3_mutex_held() and sqlite3_mutex_notheld() routine are
sl@0
   139
** intended for use inside assert() statements.
sl@0
   140
*/
sl@0
   141
int sqlite3_mutex_held(sqlite3_mutex *p){
sl@0
   142
  return p==0 || sqlite3GlobalConfig.mutex.xMutexHeld(p);
sl@0
   143
}
sl@0
   144
int sqlite3_mutex_notheld(sqlite3_mutex *p){
sl@0
   145
  return p==0 || sqlite3GlobalConfig.mutex.xMutexNotheld(p);
sl@0
   146
}
sl@0
   147
#endif
sl@0
   148
sl@0
   149
#endif /* SQLITE_OMIT_MUTEX */