First public contribution.
4 ** The author disclaims copyright to this source code. In place of
5 ** a legal notice, here is a blessing:
7 ** May you do good and not evil.
8 ** May you find forgiveness for yourself and forgive others.
9 ** May you share freely, never taking more than you give.
11 *************************************************************************
13 ** $Id: btmutex.c,v 1.10 2008/07/14 19:39:17 drh Exp $
15 ** This file contains code used to implement mutexes on Btree objects.
16 ** This code really belongs in btree.c. But btree.c is getting too
17 ** big and we want to break it down some. This packaged seemed like
21 #if SQLITE_THREADSAFE && !defined(SQLITE_OMIT_SHARED_CACHE)
25 ** Enter a mutex on the given BTree object.
27 ** If the object is not sharable, then no mutex is ever required
28 ** and this routine is a no-op. The underlying mutex is non-recursive.
29 ** But we keep a reference count in Btree.wantToLock so the behavior
30 ** of this interface is recursive.
32 ** To avoid deadlocks, multiple Btrees are locked in the same order
33 ** by all database connections. The p->pNext is a list of other
34 ** Btrees belonging to the same database connection as the p Btree
35 ** which need to be locked after p. If we cannot get a lock on
36 ** p, then first unlock all of the others on p->pNext, then wait
37 ** for the lock to become available on p, then relock all of the
38 ** subsequent Btrees that desire a lock.
40 void sqlite3BtreeEnter(Btree *p){
43 /* Some basic sanity checking on the Btree. The list of Btrees
44 ** connected by pNext and pPrev should be in sorted order by
45 ** Btree.pBt value. All elements of the list should belong to
46 ** the same connection. Only shared Btrees are on the list. */
47 assert( p->pNext==0 || p->pNext->pBt>p->pBt );
48 assert( p->pPrev==0 || p->pPrev->pBt<p->pBt );
49 assert( p->pNext==0 || p->pNext->db==p->db );
50 assert( p->pPrev==0 || p->pPrev->db==p->db );
51 assert( p->sharable || (p->pNext==0 && p->pPrev==0) );
53 /* Check for locking consistency */
54 assert( !p->locked || p->wantToLock>0 );
55 assert( p->sharable || p->wantToLock==0 );
57 /* We should already hold a lock on the database connection */
58 assert( sqlite3_mutex_held(p->db->mutex) );
60 if( !p->sharable ) return;
62 if( p->locked ) return;
64 #ifndef SQLITE_MUTEX_NOOP
65 /* In most cases, we should be able to acquire the lock we
66 ** want without having to go throught the ascending lock
67 ** procedure that follows. Just be sure not to block.
69 if( sqlite3_mutex_try(p->pBt->mutex)==SQLITE_OK ){
74 /* To avoid deadlock, first release all locks with a larger
75 ** BtShared address. Then acquire our lock. Then reacquire
76 ** the other BtShared locks that we used to hold in ascending
79 for(pLater=p->pNext; pLater; pLater=pLater->pNext){
80 assert( pLater->sharable );
81 assert( pLater->pNext==0 || pLater->pNext->pBt>pLater->pBt );
82 assert( !pLater->locked || pLater->wantToLock>0 );
84 sqlite3_mutex_leave(pLater->pBt->mutex);
88 sqlite3_mutex_enter(p->pBt->mutex);
90 for(pLater=p->pNext; pLater; pLater=pLater->pNext){
91 if( pLater->wantToLock ){
92 sqlite3_mutex_enter(pLater->pBt->mutex);
96 #endif /* SQLITE_MUTEX_NOOP */
100 ** Exit the recursive mutex on a Btree.
102 void sqlite3BtreeLeave(Btree *p){
104 assert( p->wantToLock>0 );
106 if( p->wantToLock==0 ){
108 sqlite3_mutex_leave(p->pBt->mutex);
116 ** Return true if the BtShared mutex is held on the btree.
118 ** This routine makes no determination one why or another if the
119 ** database connection mutex is held.
121 ** This routine is used only from within assert() statements.
123 int sqlite3BtreeHoldsMutex(Btree *p){
124 return (p->sharable==0 ||
125 (p->locked && p->wantToLock && sqlite3_mutex_held(p->pBt->mutex)));
130 #ifndef SQLITE_OMIT_INCRBLOB
132 ** Enter and leave a mutex on a Btree given a cursor owned by that
133 ** Btree. These entry points are used by incremental I/O and can be
134 ** omitted if that module is not used.
136 void sqlite3BtreeEnterCursor(BtCursor *pCur){
137 sqlite3BtreeEnter(pCur->pBtree);
139 void sqlite3BtreeLeaveCursor(BtCursor *pCur){
140 sqlite3BtreeLeave(pCur->pBtree);
142 #endif /* SQLITE_OMIT_INCRBLOB */
146 ** Enter the mutex on every Btree associated with a database
147 ** connection. This is needed (for example) prior to parsing
148 ** a statement since we will be comparing table and column names
149 ** against all schemas and we do not want those schemas being
150 ** reset out from under us.
152 ** There is a corresponding leave-all procedures.
154 ** Enter the mutexes in accending order by BtShared pointer address
155 ** to avoid the possibility of deadlock when two threads with
156 ** two or more btrees in common both try to lock all their btrees
157 ** at the same instant.
159 void sqlite3BtreeEnterAll(sqlite3 *db){
162 assert( sqlite3_mutex_held(db->mutex) );
163 for(i=0; i<db->nDb; i++){
165 if( p && p->sharable ){
168 assert( p->wantToLock==1 );
169 while( p->pPrev ) p = p->pPrev;
170 while( p->locked && p->pNext ) p = p->pNext;
171 for(pLater = p->pNext; pLater; pLater=pLater->pNext){
172 if( pLater->locked ){
173 sqlite3_mutex_leave(pLater->pBt->mutex);
178 sqlite3_mutex_enter(p->pBt->mutex);
186 void sqlite3BtreeLeaveAll(sqlite3 *db){
189 assert( sqlite3_mutex_held(db->mutex) );
190 for(i=0; i<db->nDb; i++){
192 if( p && p->sharable ){
193 assert( p->wantToLock>0 );
195 if( p->wantToLock==0 ){
197 sqlite3_mutex_leave(p->pBt->mutex);
206 ** Return true if the current thread holds the database connection
207 ** mutex and all required BtShared mutexes.
209 ** This routine is used inside assert() statements only.
211 int sqlite3BtreeHoldsAllMutexes(sqlite3 *db){
213 if( !sqlite3_mutex_held(db->mutex) ){
216 for(i=0; i<db->nDb; i++){
219 if( p && p->sharable &&
220 (p->wantToLock==0 || !sqlite3_mutex_held(p->pBt->mutex)) ){
229 ** Add a new Btree pointer to a BtreeMutexArray.
230 ** if the pointer can possibly be shared with
231 ** another database connection.
233 ** The pointers are kept in sorted order by pBtree->pBt. That
234 ** way when we go to enter all the mutexes, we can enter them
235 ** in order without every having to backup and retry and without
236 ** worrying about deadlock.
238 ** The number of shared btrees will always be small (usually 0 or 1)
239 ** so an insertion sort is an adequate algorithm here.
241 void sqlite3BtreeMutexArrayInsert(BtreeMutexArray *pArray, Btree *pBtree){
244 if( pBtree==0 || pBtree->sharable==0 ) return;
247 for(i=0; i<pArray->nMutex; i++){
248 assert( pArray->aBtree[i]!=pBtree );
252 assert( pArray->nMutex>=0 );
253 assert( pArray->nMutex<sizeof(pArray->aBtree)/sizeof(pArray->aBtree[0])-1 );
255 for(i=0; i<pArray->nMutex; i++){
256 assert( pArray->aBtree[i]!=pBtree );
257 if( pArray->aBtree[i]->pBt>pBt ){
258 for(j=pArray->nMutex; j>i; j--){
259 pArray->aBtree[j] = pArray->aBtree[j-1];
261 pArray->aBtree[i] = pBtree;
266 pArray->aBtree[pArray->nMutex++] = pBtree;
270 ** Enter the mutex of every btree in the array. This routine is
271 ** called at the beginning of sqlite3VdbeExec(). The mutexes are
272 ** exited at the end of the same function.
274 void sqlite3BtreeMutexArrayEnter(BtreeMutexArray *pArray){
276 for(i=0; i<pArray->nMutex; i++){
277 Btree *p = pArray->aBtree[i];
278 /* Some basic sanity checking */
279 assert( i==0 || pArray->aBtree[i-1]->pBt<p->pBt );
280 assert( !p->locked || p->wantToLock>0 );
282 /* We should already hold a lock on the database connection */
283 assert( sqlite3_mutex_held(p->db->mutex) );
286 if( !p->locked && p->sharable ){
287 sqlite3_mutex_enter(p->pBt->mutex);
294 ** Leave the mutex of every btree in the group.
296 void sqlite3BtreeMutexArrayLeave(BtreeMutexArray *pArray){
298 for(i=0; i<pArray->nMutex; i++){
299 Btree *p = pArray->aBtree[i];
300 /* Some basic sanity checking */
301 assert( i==0 || pArray->aBtree[i-1]->pBt<p->pBt );
302 assert( p->locked || !p->sharable );
303 assert( p->wantToLock>0 );
305 /* We should already hold a lock on the database connection */
306 assert( sqlite3_mutex_held(p->db->mutex) );
309 if( p->wantToLock==0 && p->locked ){
310 sqlite3_mutex_leave(p->pBt->mutex);
317 #endif /* SQLITE_THREADSAFE && !SQLITE_OMIT_SHARED_CACHE */