xref: /sqlite-3.40.0/src/test_journal.c (revision 60176fa9)
1 /*
2 ** 2008 Jan 22
3 **
4 ** The author disclaims copyright to this source code.  In place of
5 ** a legal notice, here is a blessing:
6 **
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.
10 **
11 ******************************************************************************
12 **
13 ** This file contains code for a VFS layer that acts as a wrapper around
14 ** an existing VFS. The code in this file attempts to verify that SQLite
15 ** correctly populates and syncs a journal file before writing to a
16 ** corresponding database file.
17 */
18 #if SQLITE_TEST          /* This file is used for testing only */
19 
20 #include "sqlite3.h"
21 #include "sqliteInt.h"
22 
23 /*
24 ** INTERFACE
25 **
26 **   The public interface to this wrapper VFS is two functions:
27 **
28 **     jt_register()
29 **     jt_unregister()
30 **
31 **   See header comments associated with those two functions below for
32 **   details.
33 **
34 ** LIMITATIONS
35 **
36 **   This wrapper will not work if "PRAGMA synchronous = off" is used.
37 **
38 ** OPERATION
39 **
40 **  Starting a Transaction:
41 **
42 **   When a write-transaction is started, the contents of the database is
43 **   inspected and the following data stored as part of the database file
44 **   handle (type struct jt_file):
45 **
46 **     a) The page-size of the database file.
47 **     b) The number of pages that are in the database file.
48 **     c) The set of page numbers corresponding to free-list leaf pages.
49 **     d) A check-sum for every page in the database file.
50 **
51 **   The start of a write-transaction is deemed to have occurred when a
52 **   28-byte journal header is written to byte offset 0 of the journal
53 **   file.
54 **
55 **  Syncing the Journal File:
56 **
57 **   Whenever the xSync method is invoked to sync a journal-file, the
58 **   contents of the journal file are read. For each page written to
59 **   the journal file, a check-sum is calculated and compared to the
60 **   check-sum calculated for the corresponding database page when the
61 **   write-transaction was initialized. The success of the comparison
62 **   is assert()ed. So if SQLite has written something other than the
63 **   original content to the database file, an assert() will fail.
64 **
65 **   Additionally, the set of page numbers for which records exist in
66 **   the journal file is added to (unioned with) the set of page numbers
67 **   corresponding to free-list leaf pages collected when the
68 **   write-transaction was initialized. This set comprises the page-numbers
69 **   corresponding to those pages that SQLite may now safely modify.
70 **
71 **  Writing to the Database File:
72 **
73 **   When a block of data is written to a database file, the following
74 **   invariants are asserted:
75 **
76 **     a) That the block of data is an aligned block of page-size bytes.
77 **
78 **     b) That if the page being written did not exist when the
79 **        transaction was started (i.e. the database file is growing), then
80 **        the journal-file must have been synced at least once since
81 **        the start of the transaction.
82 **
83 **     c) That if the page being written did exist when the transaction
84 **        was started, then the page must have either been a free-list
85 **        leaf page at the start of the transaction, or else must have
86 **        been stored in the journal file prior to the most recent sync.
87 **
88 **  Closing a Transaction:
89 **
90 **   When a transaction is closed, all data collected at the start of
91 **   the transaction, or following an xSync of a journal-file, is
92 **   discarded. The end of a transaction is recognized when any one
93 **   of the following occur:
94 **
95 **     a) A block of zeroes (or anything else that is not a valid
96 **        journal-header) is written to the start of the journal file.
97 **
98 **     b) A journal file is truncated to zero bytes in size using xTruncate.
99 **
100 **     c) The journal file is deleted using xDelete.
101 */
102 
103 /*
104 ** Maximum pathname length supported by the jt backend.
105 */
106 #define JT_MAX_PATHNAME 512
107 
108 /*
109 ** Name used to identify this VFS.
110 */
111 #define JT_VFS_NAME "jt"
112 
113 typedef struct jt_file jt_file;
114 struct jt_file {
115   sqlite3_file base;
116   const char *zName;       /* Name of open file */
117   int flags;               /* Flags the file was opened with */
118 
119   /* The following are only used by database file file handles */
120   int eLock;               /* Current lock held on the file */
121   u32 nPage;               /* Size of file in pages when transaction started */
122   u32 nPagesize;           /* Page size when transaction started */
123   Bitvec *pWritable;       /* Bitvec of pages that may be written to the file */
124   u32 *aCksum;             /* Checksum for first nPage pages */
125   int nSync;               /* Number of times journal file has been synced */
126 
127   /* Only used by journal file-handles */
128   sqlite3_int64 iMaxOff;   /* Maximum offset written to this transaction */
129 
130   jt_file *pNext;          /* All files are stored in a linked list */
131   sqlite3_file *pReal;     /* The file handle for the underlying vfs */
132 };
133 
134 /*
135 ** Method declarations for jt_file.
136 */
137 static int jtClose(sqlite3_file*);
138 static int jtRead(sqlite3_file*, void*, int iAmt, sqlite3_int64 iOfst);
139 static int jtWrite(sqlite3_file*,const void*,int iAmt, sqlite3_int64 iOfst);
140 static int jtTruncate(sqlite3_file*, sqlite3_int64 size);
141 static int jtSync(sqlite3_file*, int flags);
142 static int jtFileSize(sqlite3_file*, sqlite3_int64 *pSize);
143 static int jtLock(sqlite3_file*, int);
144 static int jtUnlock(sqlite3_file*, int);
145 static int jtCheckReservedLock(sqlite3_file*, int *);
146 static int jtFileControl(sqlite3_file*, int op, void *pArg);
147 static int jtSectorSize(sqlite3_file*);
148 static int jtDeviceCharacteristics(sqlite3_file*);
149 
150 /*
151 ** Method declarations for jt_vfs.
152 */
153 static int jtOpen(sqlite3_vfs*, const char *, sqlite3_file*, int , int *);
154 static int jtDelete(sqlite3_vfs*, const char *zName, int syncDir);
155 static int jtAccess(sqlite3_vfs*, const char *zName, int flags, int *);
156 static int jtFullPathname(sqlite3_vfs*, const char *zName, int, char *zOut);
157 static void *jtDlOpen(sqlite3_vfs*, const char *zFilename);
158 static void jtDlError(sqlite3_vfs*, int nByte, char *zErrMsg);
159 static void (*jtDlSym(sqlite3_vfs*,void*, const char *zSymbol))(void);
160 static void jtDlClose(sqlite3_vfs*, void*);
161 static int jtRandomness(sqlite3_vfs*, int nByte, char *zOut);
162 static int jtSleep(sqlite3_vfs*, int microseconds);
163 static int jtCurrentTime(sqlite3_vfs*, double*);
164 static int jtCurrentTimeInt64(sqlite3_vfs*, sqlite3_int64*);
165 
166 static sqlite3_vfs jt_vfs = {
167   2,                             /* iVersion */
168   sizeof(jt_file),               /* szOsFile */
169   JT_MAX_PATHNAME,               /* mxPathname */
170   0,                             /* pNext */
171   JT_VFS_NAME,                   /* zName */
172   0,                             /* pAppData */
173   jtOpen,                        /* xOpen */
174   jtDelete,                      /* xDelete */
175   jtAccess,                      /* xAccess */
176   jtFullPathname,                /* xFullPathname */
177   jtDlOpen,                      /* xDlOpen */
178   jtDlError,                     /* xDlError */
179   jtDlSym,                       /* xDlSym */
180   jtDlClose,                     /* xDlClose */
181   jtRandomness,                  /* xRandomness */
182   jtSleep,                       /* xSleep */
183   jtCurrentTime,                 /* xCurrentTime */
184   0,                             /* xGetLastError */
185   jtCurrentTimeInt64             /* xCurrentTimeInt64 */
186 };
187 
188 static sqlite3_io_methods jt_io_methods = {
189   1,                             /* iVersion */
190   jtClose,                       /* xClose */
191   jtRead,                        /* xRead */
192   jtWrite,                       /* xWrite */
193   jtTruncate,                    /* xTruncate */
194   jtSync,                        /* xSync */
195   jtFileSize,                    /* xFileSize */
196   jtLock,                        /* xLock */
197   jtUnlock,                      /* xUnlock */
198   jtCheckReservedLock,           /* xCheckReservedLock */
199   jtFileControl,                 /* xFileControl */
200   jtSectorSize,                  /* xSectorSize */
201   jtDeviceCharacteristics        /* xDeviceCharacteristics */
202 };
203 
204 struct JtGlobal {
205   sqlite3_vfs *pVfs;             /* Parent VFS */
206   jt_file *pList;                /* List of all open files */
207 };
208 static struct JtGlobal g = {0, 0};
209 
210 /*
211 ** Functions to obtain and relinquish a mutex to protect g.pList. The
212 ** STATIC_PRNG mutex is reused, purely for the sake of convenience.
213 */
214 static void enterJtMutex(void){
215   sqlite3_mutex_enter(sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_PRNG));
216 }
217 static void leaveJtMutex(void){
218   sqlite3_mutex_leave(sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_PRNG));
219 }
220 
221 extern int sqlite3_io_error_pending;
222 extern int sqlite3_io_error_hit;
223 static void stop_ioerr_simulation(int *piSave, int *piSave2){
224   *piSave = sqlite3_io_error_pending;
225   *piSave2 = sqlite3_io_error_hit;
226   sqlite3_io_error_pending = -1;
227   sqlite3_io_error_hit = 0;
228 }
229 static void start_ioerr_simulation(int iSave, int iSave2){
230   sqlite3_io_error_pending = iSave;
231   sqlite3_io_error_hit = iSave2;
232 }
233 
234 /*
235 ** The jt_file pointed to by the argument may or may not be a file-handle
236 ** open on a main database file. If it is, and a transaction is currently
237 ** opened on the file, then discard all transaction related data.
238 */
239 static void closeTransaction(jt_file *p){
240   sqlite3BitvecDestroy(p->pWritable);
241   sqlite3_free(p->aCksum);
242   p->pWritable = 0;
243   p->aCksum = 0;
244   p->nSync = 0;
245 }
246 
247 /*
248 ** Close an jt-file.
249 */
250 static int jtClose(sqlite3_file *pFile){
251   jt_file **pp;
252   jt_file *p = (jt_file *)pFile;
253 
254   closeTransaction(p);
255   enterJtMutex();
256   if( p->zName ){
257     for(pp=&g.pList; *pp!=p; pp=&(*pp)->pNext);
258     *pp = p->pNext;
259   }
260   leaveJtMutex();
261   return sqlite3OsClose(p->pReal);
262 }
263 
264 /*
265 ** Read data from an jt-file.
266 */
267 static int jtRead(
268   sqlite3_file *pFile,
269   void *zBuf,
270   int iAmt,
271   sqlite_int64 iOfst
272 ){
273   jt_file *p = (jt_file *)pFile;
274   return sqlite3OsRead(p->pReal, zBuf, iAmt, iOfst);
275 }
276 
277 /*
278 ** Parameter zJournal is the name of a journal file that is currently
279 ** open. This function locates and returns the handle opened on the
280 ** corresponding database file by the pager that currently has the
281 ** journal file opened. This file-handle is identified by the
282 ** following properties:
283 **
284 **   a) SQLITE_OPEN_MAIN_DB was specified when the file was opened.
285 **
286 **   b) The file-name specified when the file was opened matches
287 **      all but the final 8 characters of the journal file name.
288 **
289 **   c) There is currently a reserved lock on the file.
290 **/
291 static jt_file *locateDatabaseHandle(const char *zJournal){
292   jt_file *pMain = 0;
293   enterJtMutex();
294   for(pMain=g.pList; pMain; pMain=pMain->pNext){
295     int nName = strlen(zJournal) - strlen("-journal");
296     if( (pMain->flags&SQLITE_OPEN_MAIN_DB)
297      && (strlen(pMain->zName)==nName)
298      && 0==memcmp(pMain->zName, zJournal, nName)
299      && (pMain->eLock>=SQLITE_LOCK_RESERVED)
300     ){
301       break;
302     }
303   }
304   leaveJtMutex();
305   return pMain;
306 }
307 
308 /*
309 ** Parameter z points to a buffer of 4 bytes in size containing a
310 ** unsigned 32-bit integer stored in big-endian format. Decode the
311 ** integer and return its value.
312 */
313 static u32 decodeUint32(const unsigned char *z){
314   return (z[0]<<24) + (z[1]<<16) + (z[2]<<8) + z[3];
315 }
316 
317 /*
318 ** Calculate a checksum from the buffer of length n bytes pointed to
319 ** by parameter z.
320 */
321 static u32 genCksum(const unsigned char *z, int n){
322   int i;
323   u32 cksum = 0;
324   for(i=0; i<n; i++){
325     cksum = cksum + z[i] + (cksum<<3);
326   }
327   return cksum;
328 }
329 
330 /*
331 ** The first argument, zBuf, points to a buffer containing a 28 byte
332 ** serialized journal header. This function deserializes four of the
333 ** integer fields contained in the journal header and writes their
334 ** values to the output variables.
335 **
336 ** SQLITE_OK is returned if the journal-header is successfully
337 ** decoded. Otherwise, SQLITE_ERROR.
338 */
339 static int decodeJournalHdr(
340   const unsigned char *zBuf,         /* Input: 28 byte journal header */
341   u32 *pnRec,                        /* Out: Number of journalled records */
342   u32 *pnPage,                       /* Out: Original database page count */
343   u32 *pnSector,                     /* Out: Sector size in bytes */
344   u32 *pnPagesize                    /* Out: Page size in bytes */
345 ){
346   unsigned char aMagic[] = { 0xd9, 0xd5, 0x05, 0xf9, 0x20, 0xa1, 0x63, 0xd7 };
347   if( memcmp(aMagic, zBuf, 8) ) return SQLITE_ERROR;
348   if( pnRec ) *pnRec = decodeUint32(&zBuf[8]);
349   if( pnPage ) *pnPage = decodeUint32(&zBuf[16]);
350   if( pnSector ) *pnSector = decodeUint32(&zBuf[20]);
351   if( pnPagesize ) *pnPagesize = decodeUint32(&zBuf[24]);
352   return SQLITE_OK;
353 }
354 
355 /*
356 ** This function is called when a new transaction is opened, just after
357 ** the first journal-header is written to the journal file.
358 */
359 static int openTransaction(jt_file *pMain, jt_file *pJournal){
360   unsigned char *aData;
361   sqlite3_file *p = pMain->pReal;
362   int rc = SQLITE_OK;
363 
364   aData = sqlite3_malloc(pMain->nPagesize);
365   pMain->pWritable = sqlite3BitvecCreate(pMain->nPage);
366   pMain->aCksum = sqlite3_malloc(sizeof(u32) * (pMain->nPage + 1));
367   pJournal->iMaxOff = 0;
368 
369   if( !pMain->pWritable || !pMain->aCksum || !aData ){
370     rc = SQLITE_IOERR_NOMEM;
371   }else if( pMain->nPage>0 ){
372     u32 iTrunk;
373     int iSave;
374     int iSave2;
375 
376     stop_ioerr_simulation(&iSave, &iSave2);
377 
378     /* Read the database free-list. Add the page-number for each free-list
379     ** leaf to the jt_file.pWritable bitvec.
380     */
381     rc = sqlite3OsRead(p, aData, pMain->nPagesize, 0);
382     iTrunk = decodeUint32(&aData[32]);
383     while( rc==SQLITE_OK && iTrunk>0 ){
384       u32 nLeaf;
385       u32 iLeaf;
386       sqlite3_int64 iOff = (iTrunk-1)*pMain->nPagesize;
387       rc = sqlite3OsRead(p, aData, pMain->nPagesize, iOff);
388       nLeaf = decodeUint32(&aData[4]);
389       for(iLeaf=0; rc==SQLITE_OK && iLeaf<nLeaf; iLeaf++){
390         u32 pgno = decodeUint32(&aData[8+4*iLeaf]);
391         sqlite3BitvecSet(pMain->pWritable, pgno);
392       }
393       iTrunk = decodeUint32(aData);
394     }
395 
396     /* Calculate and store a checksum for each page in the database file. */
397     if( rc==SQLITE_OK ){
398       int ii;
399       for(ii=0; rc==SQLITE_OK && ii<pMain->nPage; ii++){
400         i64 iOff = (i64)(pMain->nPagesize) * (i64)ii;
401         if( iOff==PENDING_BYTE ) continue;
402         rc = sqlite3OsRead(pMain->pReal, aData, pMain->nPagesize, iOff);
403         pMain->aCksum[ii] = genCksum(aData, pMain->nPagesize);
404       }
405     }
406 
407     start_ioerr_simulation(iSave, iSave2);
408   }
409 
410   sqlite3_free(aData);
411   return rc;
412 }
413 
414 /*
415 ** The first argument to this function is a handle open on a journal file.
416 ** This function reads the journal file and adds the page number for each
417 ** page in the journal to the Bitvec object passed as the second argument.
418 */
419 static int readJournalFile(jt_file *p, jt_file *pMain){
420   int rc = SQLITE_OK;
421   unsigned char zBuf[28];
422   sqlite3_file *pReal = p->pReal;
423   sqlite3_int64 iOff = 0;
424   sqlite3_int64 iSize = p->iMaxOff;
425   unsigned char *aPage;
426   int iSave;
427   int iSave2;
428 
429   aPage = sqlite3_malloc(pMain->nPagesize);
430   if( !aPage ){
431     return SQLITE_IOERR_NOMEM;
432   }
433 
434   stop_ioerr_simulation(&iSave, &iSave2);
435 
436   while( rc==SQLITE_OK && iOff<iSize ){
437     u32 nRec, nPage, nSector, nPagesize;
438     u32 ii;
439 
440     /* Read and decode the next journal-header from the journal file. */
441     rc = sqlite3OsRead(pReal, zBuf, 28, iOff);
442     if( rc!=SQLITE_OK
443      || decodeJournalHdr(zBuf, &nRec, &nPage, &nSector, &nPagesize)
444     ){
445       goto finish_rjf;
446     }
447     iOff += nSector;
448 
449     if( nRec==0 ){
450       /* A trick. There might be another journal-header immediately
451       ** following this one. In this case, 0 records means 0 records,
452       ** not "read until the end of the file". See also ticket #2565.
453       */
454       if( iSize>=(iOff+nSector) ){
455         rc = sqlite3OsRead(pReal, zBuf, 28, iOff);
456         if( rc!=SQLITE_OK || 0==decodeJournalHdr(zBuf, 0, 0, 0, 0) ){
457           continue;
458         }
459       }
460       nRec = (iSize-iOff) / (pMain->nPagesize+8);
461     }
462 
463     /* Read all the records that follow the journal-header just read. */
464     for(ii=0; rc==SQLITE_OK && ii<nRec && iOff<iSize; ii++){
465       u32 pgno;
466       rc = sqlite3OsRead(pReal, zBuf, 4, iOff);
467       if( rc==SQLITE_OK ){
468         pgno = decodeUint32(zBuf);
469         if( pgno>0 && pgno<=pMain->nPage ){
470           if( 0==sqlite3BitvecTest(pMain->pWritable, pgno) ){
471             rc = sqlite3OsRead(pReal, aPage, pMain->nPagesize, iOff+4);
472             if( rc==SQLITE_OK ){
473               u32 cksum = genCksum(aPage, pMain->nPagesize);
474               assert( cksum==pMain->aCksum[pgno-1] );
475             }
476           }
477           sqlite3BitvecSet(pMain->pWritable, pgno);
478         }
479         iOff += (8 + pMain->nPagesize);
480       }
481     }
482 
483     iOff = ((iOff + (nSector-1)) / nSector) * nSector;
484   }
485 
486 finish_rjf:
487   start_ioerr_simulation(iSave, iSave2);
488   sqlite3_free(aPage);
489   if( rc==SQLITE_IOERR_SHORT_READ ){
490     rc = SQLITE_OK;
491   }
492   return rc;
493 }
494 
495 
496 /*
497 ** Write data to an jt-file.
498 */
499 static int jtWrite(
500   sqlite3_file *pFile,
501   const void *zBuf,
502   int iAmt,
503   sqlite_int64 iOfst
504 ){
505   int rc;
506   jt_file *p = (jt_file *)pFile;
507   if( p->flags&SQLITE_OPEN_MAIN_JOURNAL ){
508     if( iOfst==0 ){
509       jt_file *pMain = locateDatabaseHandle(p->zName);
510       assert( pMain );
511 
512       if( iAmt==28 ){
513         /* Zeroing the first journal-file header. This is the end of a
514         ** transaction. */
515         closeTransaction(pMain);
516       }else if( iAmt!=12 ){
517         /* Writing the first journal header to a journal file. This happens
518         ** when a transaction is first started.  */
519         u8 *z = (u8 *)zBuf;
520         pMain->nPage = decodeUint32(&z[16]);
521         pMain->nPagesize = decodeUint32(&z[24]);
522         if( SQLITE_OK!=(rc=openTransaction(pMain, p)) ){
523           return rc;
524         }
525       }
526     }
527     if( p->iMaxOff<(iOfst + iAmt) ){
528       p->iMaxOff = iOfst + iAmt;
529     }
530   }
531 
532   if( p->flags&SQLITE_OPEN_MAIN_DB && p->pWritable ){
533     if( iAmt<p->nPagesize
534      && p->nPagesize%iAmt==0
535      && iOfst>=(PENDING_BYTE+512)
536      && iOfst+iAmt<=PENDING_BYTE+p->nPagesize
537     ){
538       /* No-op. This special case is hit when the backup code is copying a
539       ** to a database with a larger page-size than the source database and
540       ** it needs to fill in the non-locking-region part of the original
541       ** pending-byte page.
542       */
543     }else{
544       u32 pgno = iOfst/p->nPagesize + 1;
545       assert( (iAmt==1||iAmt==p->nPagesize) && ((iOfst+iAmt)%p->nPagesize)==0 );
546       assert( pgno<=p->nPage || p->nSync>0 );
547       assert( pgno>p->nPage || sqlite3BitvecTest(p->pWritable, pgno) );
548     }
549   }
550 
551   rc = sqlite3OsWrite(p->pReal, zBuf, iAmt, iOfst);
552   if( (p->flags&SQLITE_OPEN_MAIN_JOURNAL) && iAmt==12 ){
553     jt_file *pMain = locateDatabaseHandle(p->zName);
554     int rc2 = readJournalFile(p, pMain);
555     if( rc==SQLITE_OK ) rc = rc2;
556   }
557   return rc;
558 }
559 
560 /*
561 ** Truncate an jt-file.
562 */
563 static int jtTruncate(sqlite3_file *pFile, sqlite_int64 size){
564   jt_file *p = (jt_file *)pFile;
565   if( p->flags&SQLITE_OPEN_MAIN_JOURNAL && size==0 ){
566     /* Truncating a journal file. This is the end of a transaction. */
567     jt_file *pMain = locateDatabaseHandle(p->zName);
568     closeTransaction(pMain);
569   }
570   if( p->flags&SQLITE_OPEN_MAIN_DB && p->pWritable ){
571     u32 pgno;
572     u32 locking_page = (u32)(PENDING_BYTE/p->nPagesize+1);
573     for(pgno=size/p->nPagesize+1; pgno<=p->nPage; pgno++){
574       assert( pgno==locking_page || sqlite3BitvecTest(p->pWritable, pgno) );
575     }
576   }
577   return sqlite3OsTruncate(p->pReal, size);
578 }
579 
580 /*
581 ** Sync an jt-file.
582 */
583 static int jtSync(sqlite3_file *pFile, int flags){
584   jt_file *p = (jt_file *)pFile;
585 
586   if( p->flags&SQLITE_OPEN_MAIN_JOURNAL ){
587     int rc;
588     jt_file *pMain;                   /* The associated database file */
589 
590     /* The journal file is being synced. At this point, we inspect the
591     ** contents of the file up to this point and set each bit in the
592     ** jt_file.pWritable bitvec of the main database file associated with
593     ** this journal file.
594     */
595     pMain = locateDatabaseHandle(p->zName);
596     assert(pMain);
597 
598     /* Set the bitvec values */
599     if( pMain->pWritable ){
600       pMain->nSync++;
601       rc = readJournalFile(p, pMain);
602       if( rc!=SQLITE_OK ){
603         return rc;
604       }
605     }
606   }
607 
608   return sqlite3OsSync(p->pReal, flags);
609 }
610 
611 /*
612 ** Return the current file-size of an jt-file.
613 */
614 static int jtFileSize(sqlite3_file *pFile, sqlite_int64 *pSize){
615   jt_file *p = (jt_file *)pFile;
616   return sqlite3OsFileSize(p->pReal, pSize);
617 }
618 
619 /*
620 ** Lock an jt-file.
621 */
622 static int jtLock(sqlite3_file *pFile, int eLock){
623   int rc;
624   jt_file *p = (jt_file *)pFile;
625   rc = sqlite3OsLock(p->pReal, eLock);
626   if( rc==SQLITE_OK && eLock>p->eLock ){
627     p->eLock = eLock;
628   }
629   return rc;
630 }
631 
632 /*
633 ** Unlock an jt-file.
634 */
635 static int jtUnlock(sqlite3_file *pFile, int eLock){
636   int rc;
637   jt_file *p = (jt_file *)pFile;
638   rc = sqlite3OsUnlock(p->pReal, eLock);
639   if( rc==SQLITE_OK && eLock<p->eLock ){
640     p->eLock = eLock;
641   }
642   return rc;
643 }
644 
645 /*
646 ** Check if another file-handle holds a RESERVED lock on an jt-file.
647 */
648 static int jtCheckReservedLock(sqlite3_file *pFile, int *pResOut){
649   jt_file *p = (jt_file *)pFile;
650   return sqlite3OsCheckReservedLock(p->pReal, pResOut);
651 }
652 
653 /*
654 ** File control method. For custom operations on an jt-file.
655 */
656 static int jtFileControl(sqlite3_file *pFile, int op, void *pArg){
657   jt_file *p = (jt_file *)pFile;
658   return sqlite3OsFileControl(p->pReal, op, pArg);
659 }
660 
661 /*
662 ** Return the sector-size in bytes for an jt-file.
663 */
664 static int jtSectorSize(sqlite3_file *pFile){
665   jt_file *p = (jt_file *)pFile;
666   return sqlite3OsSectorSize(p->pReal);
667 }
668 
669 /*
670 ** Return the device characteristic flags supported by an jt-file.
671 */
672 static int jtDeviceCharacteristics(sqlite3_file *pFile){
673   jt_file *p = (jt_file *)pFile;
674   return sqlite3OsDeviceCharacteristics(p->pReal);
675 }
676 
677 /*
678 ** Open an jt file handle.
679 */
680 static int jtOpen(
681   sqlite3_vfs *pVfs,
682   const char *zName,
683   sqlite3_file *pFile,
684   int flags,
685   int *pOutFlags
686 ){
687   int rc;
688   jt_file *p = (jt_file *)pFile;
689   pFile->pMethods = 0;
690   p->pReal = (sqlite3_file *)&p[1];
691   p->pReal->pMethods = 0;
692   rc = sqlite3OsOpen(g.pVfs, zName, p->pReal, flags, pOutFlags);
693   assert( rc==SQLITE_OK || p->pReal->pMethods==0 );
694   if( rc==SQLITE_OK ){
695     pFile->pMethods = &jt_io_methods;
696     p->eLock = 0;
697     p->zName = zName;
698     p->flags = flags;
699     p->pNext = 0;
700     p->pWritable = 0;
701     p->aCksum = 0;
702     enterJtMutex();
703     if( zName ){
704       p->pNext = g.pList;
705       g.pList = p;
706     }
707     leaveJtMutex();
708   }
709   return rc;
710 }
711 
712 /*
713 ** Delete the file located at zPath. If the dirSync argument is true,
714 ** ensure the file-system modifications are synced to disk before
715 ** returning.
716 */
717 static int jtDelete(sqlite3_vfs *pVfs, const char *zPath, int dirSync){
718   int nPath = strlen(zPath);
719   if( nPath>8 && 0==strcmp("-journal", &zPath[nPath-8]) ){
720     /* Deleting a journal file. The end of a transaction. */
721     jt_file *pMain = locateDatabaseHandle(zPath);
722     if( pMain ){
723       closeTransaction(pMain);
724     }
725   }
726 
727   return sqlite3OsDelete(g.pVfs, zPath, dirSync);
728 }
729 
730 /*
731 ** Test for access permissions. Return true if the requested permission
732 ** is available, or false otherwise.
733 */
734 static int jtAccess(
735   sqlite3_vfs *pVfs,
736   const char *zPath,
737   int flags,
738   int *pResOut
739 ){
740   return sqlite3OsAccess(g.pVfs, zPath, flags, pResOut);
741 }
742 
743 /*
744 ** Populate buffer zOut with the full canonical pathname corresponding
745 ** to the pathname in zPath. zOut is guaranteed to point to a buffer
746 ** of at least (JT_MAX_PATHNAME+1) bytes.
747 */
748 static int jtFullPathname(
749   sqlite3_vfs *pVfs,
750   const char *zPath,
751   int nOut,
752   char *zOut
753 ){
754   return sqlite3OsFullPathname(g.pVfs, zPath, nOut, zOut);
755 }
756 
757 /*
758 ** Open the dynamic library located at zPath and return a handle.
759 */
760 static void *jtDlOpen(sqlite3_vfs *pVfs, const char *zPath){
761   return g.pVfs->xDlOpen(g.pVfs, zPath);
762 }
763 
764 /*
765 ** Populate the buffer zErrMsg (size nByte bytes) with a human readable
766 ** utf-8 string describing the most recent error encountered associated
767 ** with dynamic libraries.
768 */
769 static void jtDlError(sqlite3_vfs *pVfs, int nByte, char *zErrMsg){
770   g.pVfs->xDlError(g.pVfs, nByte, zErrMsg);
771 }
772 
773 /*
774 ** Return a pointer to the symbol zSymbol in the dynamic library pHandle.
775 */
776 static void (*jtDlSym(sqlite3_vfs *pVfs, void *p, const char *zSym))(void){
777   return g.pVfs->xDlSym(g.pVfs, p, zSym);
778 }
779 
780 /*
781 ** Close the dynamic library handle pHandle.
782 */
783 static void jtDlClose(sqlite3_vfs *pVfs, void *pHandle){
784   g.pVfs->xDlClose(g.pVfs, pHandle);
785 }
786 
787 /*
788 ** Populate the buffer pointed to by zBufOut with nByte bytes of
789 ** random data.
790 */
791 static int jtRandomness(sqlite3_vfs *pVfs, int nByte, char *zBufOut){
792   return sqlite3OsRandomness(g.pVfs, nByte, zBufOut);
793 }
794 
795 /*
796 ** Sleep for nMicro microseconds. Return the number of microseconds
797 ** actually slept.
798 */
799 static int jtSleep(sqlite3_vfs *pVfs, int nMicro){
800   return sqlite3OsSleep(g.pVfs, nMicro);
801 }
802 
803 /*
804 ** Return the current time as a Julian Day number in *pTimeOut.
805 */
806 static int jtCurrentTime(sqlite3_vfs *pVfs, double *pTimeOut){
807   return g.pVfs->xCurrentTime(g.pVfs, pTimeOut);
808 }
809 /*
810 ** Return the current time as a Julian Day number in *pTimeOut.
811 */
812 static int jtCurrentTimeInt64(sqlite3_vfs *pVfs, sqlite3_int64 *pTimeOut){
813   return g.pVfs->xCurrentTimeInt64(g.pVfs, pTimeOut);
814 }
815 
816 /**************************************************************************
817 ** Start of public API.
818 */
819 
820 /*
821 ** Configure the jt VFS as a wrapper around the VFS named by parameter
822 ** zWrap. If the isDefault parameter is true, then the jt VFS is installed
823 ** as the new default VFS for SQLite connections. If isDefault is not
824 ** true, then the jt VFS is installed as non-default. In this case it
825 ** is available via its name, "jt".
826 */
827 int jt_register(char *zWrap, int isDefault){
828   g.pVfs = sqlite3_vfs_find(zWrap);
829   if( g.pVfs==0 ){
830     return SQLITE_ERROR;
831   }
832   jt_vfs.szOsFile = sizeof(jt_file) + g.pVfs->szOsFile;
833   if( g.pVfs->iVersion==1 ){
834     jt_vfs.iVersion = 1;
835   }else if( g.pVfs->xCurrentTimeInt64==0 ){
836     jt_vfs.xCurrentTimeInt64 = 0;
837   }
838   sqlite3_vfs_register(&jt_vfs, isDefault);
839   return SQLITE_OK;
840 }
841 
842 /*
843 ** Uninstall the jt VFS, if it is installed.
844 */
845 void jt_unregister(void){
846   sqlite3_vfs_unregister(&jt_vfs);
847 }
848 
849 #endif
850