os/persistentdata/persistentstorage/sql/SQLite/mutex.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
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
** The implementation in this file does not provide any mutual
sl@0
    15
** exclusion and is thus suitable for use only in applications
sl@0
    16
** that use SQLite in a single thread.  But this implementation
sl@0
    17
** does do a lot of error checking on mutexes to make sure they
sl@0
    18
** are called correctly and at appropriate times.  Hence, this
sl@0
    19
** implementation is suitable for testing.
sl@0
    20
** debugging purposes
sl@0
    21
**
sl@0
    22
** $Id: mutex.c,v 1.27 2008/06/19 08:51:24 danielk1977 Exp $
sl@0
    23
*/
sl@0
    24
#include "sqliteInt.h"
sl@0
    25
sl@0
    26
#ifndef SQLITE_MUTEX_NOOP
sl@0
    27
/*
sl@0
    28
** Initialize the mutex system.
sl@0
    29
*/
sl@0
    30
int sqlite3MutexInit(void){ 
sl@0
    31
  int rc = SQLITE_OK;
sl@0
    32
  if( sqlite3Config.bCoreMutex ){
sl@0
    33
    if( !sqlite3Config.mutex.xMutexAlloc ){
sl@0
    34
      /* If the xMutexAlloc method has not been set, then the user did not
sl@0
    35
      ** install a mutex implementation via sqlite3_config() prior to 
sl@0
    36
      ** sqlite3_initialize() being called. This block copies pointers to
sl@0
    37
      ** the default implementation into the sqlite3Config structure.
sl@0
    38
      **
sl@0
    39
      ** The danger is that although sqlite3_config() is not a threadsafe
sl@0
    40
      ** API, sqlite3_initialize() is, and so multiple threads may be
sl@0
    41
      ** attempting to run this function simultaneously. To guard write
sl@0
    42
      ** access to the sqlite3Config structure, the 'MASTER' static mutex
sl@0
    43
      ** is obtained before modifying it.
sl@0
    44
      */
sl@0
    45
      sqlite3_mutex_methods *p = sqlite3DefaultMutex();
sl@0
    46
      sqlite3_mutex *pMaster = 0;
sl@0
    47
  
sl@0
    48
      rc = p->xMutexInit();
sl@0
    49
      if( rc==SQLITE_OK ){
sl@0
    50
        pMaster = p->xMutexAlloc(SQLITE_MUTEX_STATIC_MASTER);
sl@0
    51
        assert(pMaster);
sl@0
    52
        p->xMutexEnter(pMaster);
sl@0
    53
        assert( sqlite3Config.mutex.xMutexAlloc==0 
sl@0
    54
             || sqlite3Config.mutex.xMutexAlloc==p->xMutexAlloc
sl@0
    55
        );
sl@0
    56
        if( !sqlite3Config.mutex.xMutexAlloc ){
sl@0
    57
          sqlite3Config.mutex = *p;
sl@0
    58
        }
sl@0
    59
        p->xMutexLeave(pMaster);
sl@0
    60
      }
sl@0
    61
    }else{
sl@0
    62
      rc = sqlite3Config.mutex.xMutexInit();
sl@0
    63
    }
sl@0
    64
  }
sl@0
    65
sl@0
    66
  return rc;
sl@0
    67
}
sl@0
    68
sl@0
    69
/*
sl@0
    70
** Shutdown the mutex system. This call frees resources allocated by
sl@0
    71
** sqlite3MutexInit().
sl@0
    72
*/
sl@0
    73
int sqlite3MutexEnd(void){
sl@0
    74
  int rc = SQLITE_OK;
sl@0
    75
  rc = sqlite3Config.mutex.xMutexEnd();
sl@0
    76
  return rc;
sl@0
    77
}
sl@0
    78
sl@0
    79
/*
sl@0
    80
** Retrieve a pointer to a static mutex or allocate a new dynamic one.
sl@0
    81
*/
sl@0
    82
sqlite3_mutex *sqlite3_mutex_alloc(int id){
sl@0
    83
#ifndef SQLITE_OMIT_AUTOINIT
sl@0
    84
  if( sqlite3_initialize() ) return 0;
sl@0
    85
#endif
sl@0
    86
  return sqlite3Config.mutex.xMutexAlloc(id);
sl@0
    87
}
sl@0
    88
sl@0
    89
sqlite3_mutex *sqlite3MutexAlloc(int id){
sl@0
    90
  if( !sqlite3Config.bCoreMutex ){
sl@0
    91
    return 0;
sl@0
    92
  }
sl@0
    93
  return sqlite3Config.mutex.xMutexAlloc(id);
sl@0
    94
}
sl@0
    95
sl@0
    96
/*
sl@0
    97
** Free a dynamic mutex.
sl@0
    98
*/
sl@0
    99
void sqlite3_mutex_free(sqlite3_mutex *p){
sl@0
   100
  if( p ){
sl@0
   101
    sqlite3Config.mutex.xMutexFree(p);
sl@0
   102
  }
sl@0
   103
}
sl@0
   104
sl@0
   105
/*
sl@0
   106
** Obtain the mutex p. If some other thread already has the mutex, block
sl@0
   107
** until it can be obtained.
sl@0
   108
*/
sl@0
   109
void sqlite3_mutex_enter(sqlite3_mutex *p){
sl@0
   110
  if( p ){
sl@0
   111
    sqlite3Config.mutex.xMutexEnter(p);
sl@0
   112
  }
sl@0
   113
}
sl@0
   114
sl@0
   115
/*
sl@0
   116
** Obtain the mutex p. If successful, return SQLITE_OK. Otherwise, if another
sl@0
   117
** thread holds the mutex and it cannot be obtained, return SQLITE_BUSY.
sl@0
   118
*/
sl@0
   119
int sqlite3_mutex_try(sqlite3_mutex *p){
sl@0
   120
  int rc = SQLITE_OK;
sl@0
   121
  if( p ){
sl@0
   122
    return sqlite3Config.mutex.xMutexTry(p);
sl@0
   123
  }
sl@0
   124
  return rc;
sl@0
   125
}
sl@0
   126
sl@0
   127
/*
sl@0
   128
** The sqlite3_mutex_leave() routine exits a mutex that was previously
sl@0
   129
** entered by the same thread.  The behavior is undefined if the mutex 
sl@0
   130
** is not currently entered. If a NULL pointer is passed as an argument
sl@0
   131
** this function is a no-op.
sl@0
   132
*/
sl@0
   133
void sqlite3_mutex_leave(sqlite3_mutex *p){
sl@0
   134
  if( p ){
sl@0
   135
    sqlite3Config.mutex.xMutexLeave(p);
sl@0
   136
  }
sl@0
   137
}
sl@0
   138
sl@0
   139
#ifndef NDEBUG
sl@0
   140
/*
sl@0
   141
** The sqlite3_mutex_held() and sqlite3_mutex_notheld() routine are
sl@0
   142
** intended for use inside assert() statements.
sl@0
   143
*/
sl@0
   144
int sqlite3_mutex_held(sqlite3_mutex *p){
sl@0
   145
  return p==0 || sqlite3Config.mutex.xMutexHeld(p);
sl@0
   146
}
sl@0
   147
int sqlite3_mutex_notheld(sqlite3_mutex *p){
sl@0
   148
  return p==0 || sqlite3Config.mutex.xMutexNotheld(p);
sl@0
   149
}
sl@0
   150
#endif
sl@0
   151
sl@0
   152
#endif
sl@0
   153
sl@0
   154
#ifdef SQLITE_MUTEX_NOOP_DEBUG
sl@0
   155
/*
sl@0
   156
** In this implementation, mutexes do not provide any mutual exclusion.
sl@0
   157
** But the error checking is provided.  This implementation is useful
sl@0
   158
** for test purposes.
sl@0
   159
*/
sl@0
   160
sl@0
   161
/*
sl@0
   162
** The mutex object
sl@0
   163
*/
sl@0
   164
struct sqlite3_mutex {
sl@0
   165
  int id;     /* The mutex type */
sl@0
   166
  int cnt;    /* Number of entries without a matching leave */
sl@0
   167
};
sl@0
   168
sl@0
   169
/*
sl@0
   170
** The sqlite3_mutex_held() and sqlite3_mutex_notheld() routine are
sl@0
   171
** intended for use inside assert() statements.
sl@0
   172
*/
sl@0
   173
static int noopMutexHeld(sqlite3_mutex *p){
sl@0
   174
  return p==0 || p->cnt>0;
sl@0
   175
}
sl@0
   176
static int noopMutexNotheld(sqlite3_mutex *p){
sl@0
   177
  return p==0 || p->cnt==0;
sl@0
   178
}
sl@0
   179
sl@0
   180
/*
sl@0
   181
** Initialize and deinitialize the mutex subsystem.
sl@0
   182
*/
sl@0
   183
static int noopMutexInit(void){ return SQLITE_OK; }
sl@0
   184
static int noopMutexEnd(void){ return SQLITE_OK; }
sl@0
   185
sl@0
   186
/*
sl@0
   187
** The sqlite3_mutex_alloc() routine allocates a new
sl@0
   188
** mutex and returns a pointer to it.  If it returns NULL
sl@0
   189
** that means that a mutex could not be allocated. 
sl@0
   190
*/
sl@0
   191
static sqlite3_mutex *noopMutexAlloc(int id){
sl@0
   192
  static sqlite3_mutex aStatic[6];
sl@0
   193
  sqlite3_mutex *pNew = 0;
sl@0
   194
  switch( id ){
sl@0
   195
    case SQLITE_MUTEX_FAST:
sl@0
   196
    case SQLITE_MUTEX_RECURSIVE: {
sl@0
   197
      pNew = sqlite3Malloc(sizeof(*pNew));
sl@0
   198
      if( pNew ){
sl@0
   199
        pNew->id = id;
sl@0
   200
        pNew->cnt = 0;
sl@0
   201
      }
sl@0
   202
      break;
sl@0
   203
    }
sl@0
   204
    default: {
sl@0
   205
      assert( id-2 >= 0 );
sl@0
   206
      assert( id-2 < sizeof(aStatic)/sizeof(aStatic[0]) );
sl@0
   207
      pNew = &aStatic[id-2];
sl@0
   208
      pNew->id = id;
sl@0
   209
      break;
sl@0
   210
    }
sl@0
   211
  }
sl@0
   212
  return pNew;
sl@0
   213
}
sl@0
   214
sl@0
   215
/*
sl@0
   216
** This routine deallocates a previously allocated mutex.
sl@0
   217
*/
sl@0
   218
static void noopMutexFree(sqlite3_mutex *p){
sl@0
   219
  assert( p->cnt==0 );
sl@0
   220
  assert( p->id==SQLITE_MUTEX_FAST || p->id==SQLITE_MUTEX_RECURSIVE );
sl@0
   221
  sqlite3_free(p);
sl@0
   222
}
sl@0
   223
sl@0
   224
/*
sl@0
   225
** The sqlite3_mutex_enter() and sqlite3_mutex_try() routines attempt
sl@0
   226
** to enter a mutex.  If another thread is already within the mutex,
sl@0
   227
** sqlite3_mutex_enter() will block and sqlite3_mutex_try() will return
sl@0
   228
** SQLITE_BUSY.  The sqlite3_mutex_try() interface returns SQLITE_OK
sl@0
   229
** upon successful entry.  Mutexes created using SQLITE_MUTEX_RECURSIVE can
sl@0
   230
** be entered multiple times by the same thread.  In such cases the,
sl@0
   231
** mutex must be exited an equal number of times before another thread
sl@0
   232
** can enter.  If the same thread tries to enter any other kind of mutex
sl@0
   233
** more than once, the behavior is undefined.
sl@0
   234
*/
sl@0
   235
static void noopMutexEnter(sqlite3_mutex *p){
sl@0
   236
  assert( p->id==SQLITE_MUTEX_RECURSIVE || noopMutexNotheld(p) );
sl@0
   237
  p->cnt++;
sl@0
   238
}
sl@0
   239
static int noopMutexTry(sqlite3_mutex *p){
sl@0
   240
  assert( p->id==SQLITE_MUTEX_RECURSIVE || noopMutexNotheld(p) );
sl@0
   241
  p->cnt++;
sl@0
   242
  return SQLITE_OK;
sl@0
   243
}
sl@0
   244
sl@0
   245
/*
sl@0
   246
** The sqlite3_mutex_leave() routine exits a mutex that was
sl@0
   247
** previously entered by the same thread.  The behavior
sl@0
   248
** is undefined if the mutex is not currently entered or
sl@0
   249
** is not currently allocated.  SQLite will never do either.
sl@0
   250
*/
sl@0
   251
static void noopMutexLeave(sqlite3_mutex *p){
sl@0
   252
  assert( noopMutexHeld(p) );
sl@0
   253
  p->cnt--;
sl@0
   254
  assert( p->id==SQLITE_MUTEX_RECURSIVE || noopMutexNotheld(p) );
sl@0
   255
}
sl@0
   256
sl@0
   257
sqlite3_mutex_methods *sqlite3DefaultMutex(void){
sl@0
   258
  static sqlite3_mutex_methods sMutex = {
sl@0
   259
    noopMutexInit,
sl@0
   260
    noopMutexEnd,
sl@0
   261
    noopMutexAlloc,
sl@0
   262
    noopMutexFree,
sl@0
   263
    noopMutexEnter,
sl@0
   264
    noopMutexTry,
sl@0
   265
    noopMutexLeave,
sl@0
   266
sl@0
   267
    noopMutexHeld,
sl@0
   268
    noopMutexNotheld
sl@0
   269
  };
sl@0
   270
sl@0
   271
  return &sMutex;
sl@0
   272
}
sl@0
   273
#endif /* SQLITE_MUTEX_NOOP_DEBUG */