os/persistentdata/persistentstorage/sqlite3api/SQLite/btmutex.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 27
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
**
sl@0
    13
** $Id: btmutex.c,v 1.10 2008/07/14 19:39:17 drh Exp $
sl@0
    14
**
sl@0
    15
** This file contains code used to implement mutexes on Btree objects.
sl@0
    16
** This code really belongs in btree.c.  But btree.c is getting too
sl@0
    17
** big and we want to break it down some.  This packaged seemed like
sl@0
    18
** a good breakout.
sl@0
    19
*/
sl@0
    20
#include "btreeInt.h"
sl@0
    21
#if SQLITE_THREADSAFE && !defined(SQLITE_OMIT_SHARED_CACHE)
sl@0
    22
sl@0
    23
sl@0
    24
/*
sl@0
    25
** Enter a mutex on the given BTree object.
sl@0
    26
**
sl@0
    27
** If the object is not sharable, then no mutex is ever required
sl@0
    28
** and this routine is a no-op.  The underlying mutex is non-recursive.
sl@0
    29
** But we keep a reference count in Btree.wantToLock so the behavior
sl@0
    30
** of this interface is recursive.
sl@0
    31
**
sl@0
    32
** To avoid deadlocks, multiple Btrees are locked in the same order
sl@0
    33
** by all database connections.  The p->pNext is a list of other
sl@0
    34
** Btrees belonging to the same database connection as the p Btree
sl@0
    35
** which need to be locked after p.  If we cannot get a lock on
sl@0
    36
** p, then first unlock all of the others on p->pNext, then wait
sl@0
    37
** for the lock to become available on p, then relock all of the
sl@0
    38
** subsequent Btrees that desire a lock.
sl@0
    39
*/
sl@0
    40
void sqlite3BtreeEnter(Btree *p){
sl@0
    41
  Btree *pLater;
sl@0
    42
sl@0
    43
  /* Some basic sanity checking on the Btree.  The list of Btrees
sl@0
    44
  ** connected by pNext and pPrev should be in sorted order by
sl@0
    45
  ** Btree.pBt value. All elements of the list should belong to
sl@0
    46
  ** the same connection. Only shared Btrees are on the list. */
sl@0
    47
  assert( p->pNext==0 || p->pNext->pBt>p->pBt );
sl@0
    48
  assert( p->pPrev==0 || p->pPrev->pBt<p->pBt );
sl@0
    49
  assert( p->pNext==0 || p->pNext->db==p->db );
sl@0
    50
  assert( p->pPrev==0 || p->pPrev->db==p->db );
sl@0
    51
  assert( p->sharable || (p->pNext==0 && p->pPrev==0) );
sl@0
    52
sl@0
    53
  /* Check for locking consistency */
sl@0
    54
  assert( !p->locked || p->wantToLock>0 );
sl@0
    55
  assert( p->sharable || p->wantToLock==0 );
sl@0
    56
sl@0
    57
  /* We should already hold a lock on the database connection */
sl@0
    58
  assert( sqlite3_mutex_held(p->db->mutex) );
sl@0
    59
sl@0
    60
  if( !p->sharable ) return;
sl@0
    61
  p->wantToLock++;
sl@0
    62
  if( p->locked ) return;
sl@0
    63
sl@0
    64
#ifndef SQLITE_MUTEX_NOOP
sl@0
    65
  /* In most cases, we should be able to acquire the lock we
sl@0
    66
  ** want without having to go throught the ascending lock
sl@0
    67
  ** procedure that follows.  Just be sure not to block.
sl@0
    68
  */
sl@0
    69
  if( sqlite3_mutex_try(p->pBt->mutex)==SQLITE_OK ){
sl@0
    70
    p->locked = 1;
sl@0
    71
    return;
sl@0
    72
  }
sl@0
    73
sl@0
    74
  /* To avoid deadlock, first release all locks with a larger
sl@0
    75
  ** BtShared address.  Then acquire our lock.  Then reacquire
sl@0
    76
  ** the other BtShared locks that we used to hold in ascending
sl@0
    77
  ** order.
sl@0
    78
  */
sl@0
    79
  for(pLater=p->pNext; pLater; pLater=pLater->pNext){
sl@0
    80
    assert( pLater->sharable );
sl@0
    81
    assert( pLater->pNext==0 || pLater->pNext->pBt>pLater->pBt );
sl@0
    82
    assert( !pLater->locked || pLater->wantToLock>0 );
sl@0
    83
    if( pLater->locked ){
sl@0
    84
      sqlite3_mutex_leave(pLater->pBt->mutex);
sl@0
    85
      pLater->locked = 0;
sl@0
    86
    }
sl@0
    87
  }
sl@0
    88
  sqlite3_mutex_enter(p->pBt->mutex);
sl@0
    89
  p->locked = 1;
sl@0
    90
  for(pLater=p->pNext; pLater; pLater=pLater->pNext){
sl@0
    91
    if( pLater->wantToLock ){
sl@0
    92
      sqlite3_mutex_enter(pLater->pBt->mutex);
sl@0
    93
      pLater->locked = 1;
sl@0
    94
    }
sl@0
    95
  }
sl@0
    96
#endif /* SQLITE_MUTEX_NOOP */
sl@0
    97
}
sl@0
    98
sl@0
    99
/*
sl@0
   100
** Exit the recursive mutex on a Btree.
sl@0
   101
*/
sl@0
   102
void sqlite3BtreeLeave(Btree *p){
sl@0
   103
  if( p->sharable ){
sl@0
   104
    assert( p->wantToLock>0 );
sl@0
   105
    p->wantToLock--;
sl@0
   106
    if( p->wantToLock==0 ){
sl@0
   107
      assert( p->locked );
sl@0
   108
      sqlite3_mutex_leave(p->pBt->mutex);
sl@0
   109
      p->locked = 0;
sl@0
   110
    }
sl@0
   111
  }
sl@0
   112
}
sl@0
   113
sl@0
   114
#ifndef NDEBUG
sl@0
   115
/*
sl@0
   116
** Return true if the BtShared mutex is held on the btree.  
sl@0
   117
**
sl@0
   118
** This routine makes no determination one why or another if the
sl@0
   119
** database connection mutex is held.
sl@0
   120
**
sl@0
   121
** This routine is used only from within assert() statements.
sl@0
   122
*/
sl@0
   123
int sqlite3BtreeHoldsMutex(Btree *p){
sl@0
   124
  return (p->sharable==0 ||
sl@0
   125
             (p->locked && p->wantToLock && sqlite3_mutex_held(p->pBt->mutex)));
sl@0
   126
}
sl@0
   127
#endif
sl@0
   128
sl@0
   129
sl@0
   130
#ifndef SQLITE_OMIT_INCRBLOB
sl@0
   131
/*
sl@0
   132
** Enter and leave a mutex on a Btree given a cursor owned by that
sl@0
   133
** Btree.  These entry points are used by incremental I/O and can be
sl@0
   134
** omitted if that module is not used.
sl@0
   135
*/
sl@0
   136
void sqlite3BtreeEnterCursor(BtCursor *pCur){
sl@0
   137
  sqlite3BtreeEnter(pCur->pBtree);
sl@0
   138
}
sl@0
   139
void sqlite3BtreeLeaveCursor(BtCursor *pCur){
sl@0
   140
  sqlite3BtreeLeave(pCur->pBtree);
sl@0
   141
}
sl@0
   142
#endif /* SQLITE_OMIT_INCRBLOB */
sl@0
   143
sl@0
   144
sl@0
   145
/*
sl@0
   146
** Enter the mutex on every Btree associated with a database
sl@0
   147
** connection.  This is needed (for example) prior to parsing
sl@0
   148
** a statement since we will be comparing table and column names
sl@0
   149
** against all schemas and we do not want those schemas being
sl@0
   150
** reset out from under us.
sl@0
   151
**
sl@0
   152
** There is a corresponding leave-all procedures.
sl@0
   153
**
sl@0
   154
** Enter the mutexes in accending order by BtShared pointer address
sl@0
   155
** to avoid the possibility of deadlock when two threads with
sl@0
   156
** two or more btrees in common both try to lock all their btrees
sl@0
   157
** at the same instant.
sl@0
   158
*/
sl@0
   159
void sqlite3BtreeEnterAll(sqlite3 *db){
sl@0
   160
  int i;
sl@0
   161
  Btree *p, *pLater;
sl@0
   162
  assert( sqlite3_mutex_held(db->mutex) );
sl@0
   163
  for(i=0; i<db->nDb; i++){
sl@0
   164
    p = db->aDb[i].pBt;
sl@0
   165
    if( p && p->sharable ){
sl@0
   166
      p->wantToLock++;
sl@0
   167
      if( !p->locked ){
sl@0
   168
        assert( p->wantToLock==1 );
sl@0
   169
        while( p->pPrev ) p = p->pPrev;
sl@0
   170
        while( p->locked && p->pNext ) p = p->pNext;
sl@0
   171
        for(pLater = p->pNext; pLater; pLater=pLater->pNext){
sl@0
   172
          if( pLater->locked ){
sl@0
   173
            sqlite3_mutex_leave(pLater->pBt->mutex);
sl@0
   174
            pLater->locked = 0;
sl@0
   175
          }
sl@0
   176
        }
sl@0
   177
        while( p ){
sl@0
   178
          sqlite3_mutex_enter(p->pBt->mutex);
sl@0
   179
          p->locked++;
sl@0
   180
          p = p->pNext;
sl@0
   181
        }
sl@0
   182
      }
sl@0
   183
    }
sl@0
   184
  }
sl@0
   185
}
sl@0
   186
void sqlite3BtreeLeaveAll(sqlite3 *db){
sl@0
   187
  int i;
sl@0
   188
  Btree *p;
sl@0
   189
  assert( sqlite3_mutex_held(db->mutex) );
sl@0
   190
  for(i=0; i<db->nDb; i++){
sl@0
   191
    p = db->aDb[i].pBt;
sl@0
   192
    if( p && p->sharable ){
sl@0
   193
      assert( p->wantToLock>0 );
sl@0
   194
      p->wantToLock--;
sl@0
   195
      if( p->wantToLock==0 ){
sl@0
   196
        assert( p->locked );
sl@0
   197
        sqlite3_mutex_leave(p->pBt->mutex);
sl@0
   198
        p->locked = 0;
sl@0
   199
      }
sl@0
   200
    }
sl@0
   201
  }
sl@0
   202
}
sl@0
   203
sl@0
   204
#ifndef NDEBUG
sl@0
   205
/*
sl@0
   206
** Return true if the current thread holds the database connection
sl@0
   207
** mutex and all required BtShared mutexes.
sl@0
   208
**
sl@0
   209
** This routine is used inside assert() statements only.
sl@0
   210
*/
sl@0
   211
int sqlite3BtreeHoldsAllMutexes(sqlite3 *db){
sl@0
   212
  int i;
sl@0
   213
  if( !sqlite3_mutex_held(db->mutex) ){
sl@0
   214
    return 0;
sl@0
   215
  }
sl@0
   216
  for(i=0; i<db->nDb; i++){
sl@0
   217
    Btree *p;
sl@0
   218
    p = db->aDb[i].pBt;
sl@0
   219
    if( p && p->sharable &&
sl@0
   220
         (p->wantToLock==0 || !sqlite3_mutex_held(p->pBt->mutex)) ){
sl@0
   221
      return 0;
sl@0
   222
    }
sl@0
   223
  }
sl@0
   224
  return 1;
sl@0
   225
}
sl@0
   226
#endif /* NDEBUG */
sl@0
   227
sl@0
   228
/*
sl@0
   229
** Add a new Btree pointer to a BtreeMutexArray. 
sl@0
   230
** if the pointer can possibly be shared with
sl@0
   231
** another database connection.
sl@0
   232
**
sl@0
   233
** The pointers are kept in sorted order by pBtree->pBt.  That
sl@0
   234
** way when we go to enter all the mutexes, we can enter them
sl@0
   235
** in order without every having to backup and retry and without
sl@0
   236
** worrying about deadlock.
sl@0
   237
**
sl@0
   238
** The number of shared btrees will always be small (usually 0 or 1)
sl@0
   239
** so an insertion sort is an adequate algorithm here.
sl@0
   240
*/
sl@0
   241
void sqlite3BtreeMutexArrayInsert(BtreeMutexArray *pArray, Btree *pBtree){
sl@0
   242
  int i, j;
sl@0
   243
  BtShared *pBt;
sl@0
   244
  if( pBtree==0 || pBtree->sharable==0 ) return;
sl@0
   245
#ifndef NDEBUG
sl@0
   246
  {
sl@0
   247
    for(i=0; i<pArray->nMutex; i++){
sl@0
   248
      assert( pArray->aBtree[i]!=pBtree );
sl@0
   249
    }
sl@0
   250
  }
sl@0
   251
#endif
sl@0
   252
  assert( pArray->nMutex>=0 );
sl@0
   253
  assert( pArray->nMutex<sizeof(pArray->aBtree)/sizeof(pArray->aBtree[0])-1 );
sl@0
   254
  pBt = pBtree->pBt;
sl@0
   255
  for(i=0; i<pArray->nMutex; i++){
sl@0
   256
    assert( pArray->aBtree[i]!=pBtree );
sl@0
   257
    if( pArray->aBtree[i]->pBt>pBt ){
sl@0
   258
      for(j=pArray->nMutex; j>i; j--){
sl@0
   259
        pArray->aBtree[j] = pArray->aBtree[j-1];
sl@0
   260
      }
sl@0
   261
      pArray->aBtree[i] = pBtree;
sl@0
   262
      pArray->nMutex++;
sl@0
   263
      return;
sl@0
   264
    }
sl@0
   265
  }
sl@0
   266
  pArray->aBtree[pArray->nMutex++] = pBtree;
sl@0
   267
}
sl@0
   268
sl@0
   269
/*
sl@0
   270
** Enter the mutex of every btree in the array.  This routine is
sl@0
   271
** called at the beginning of sqlite3VdbeExec().  The mutexes are
sl@0
   272
** exited at the end of the same function.
sl@0
   273
*/
sl@0
   274
void sqlite3BtreeMutexArrayEnter(BtreeMutexArray *pArray){
sl@0
   275
  int i;
sl@0
   276
  for(i=0; i<pArray->nMutex; i++){
sl@0
   277
    Btree *p = pArray->aBtree[i];
sl@0
   278
    /* Some basic sanity checking */
sl@0
   279
    assert( i==0 || pArray->aBtree[i-1]->pBt<p->pBt );
sl@0
   280
    assert( !p->locked || p->wantToLock>0 );
sl@0
   281
sl@0
   282
    /* We should already hold a lock on the database connection */
sl@0
   283
    assert( sqlite3_mutex_held(p->db->mutex) );
sl@0
   284
sl@0
   285
    p->wantToLock++;
sl@0
   286
    if( !p->locked && p->sharable ){
sl@0
   287
      sqlite3_mutex_enter(p->pBt->mutex);
sl@0
   288
      p->locked = 1;
sl@0
   289
    }
sl@0
   290
  }
sl@0
   291
}
sl@0
   292
sl@0
   293
/*
sl@0
   294
** Leave the mutex of every btree in the group.
sl@0
   295
*/
sl@0
   296
void sqlite3BtreeMutexArrayLeave(BtreeMutexArray *pArray){
sl@0
   297
  int i;
sl@0
   298
  for(i=0; i<pArray->nMutex; i++){
sl@0
   299
    Btree *p = pArray->aBtree[i];
sl@0
   300
    /* Some basic sanity checking */
sl@0
   301
    assert( i==0 || pArray->aBtree[i-1]->pBt<p->pBt );
sl@0
   302
    assert( p->locked || !p->sharable );
sl@0
   303
    assert( p->wantToLock>0 );
sl@0
   304
sl@0
   305
    /* We should already hold a lock on the database connection */
sl@0
   306
    assert( sqlite3_mutex_held(p->db->mutex) );
sl@0
   307
sl@0
   308
    p->wantToLock--;
sl@0
   309
    if( p->wantToLock==0 && p->locked ){
sl@0
   310
      sqlite3_mutex_leave(p->pBt->mutex);
sl@0
   311
      p->locked = 0;
sl@0
   312
    }
sl@0
   313
  }
sl@0
   314
}
sl@0
   315
sl@0
   316
sl@0
   317
#endif  /* SQLITE_THREADSAFE && !SQLITE_OMIT_SHARED_CACHE */