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: test_onefile.c,v 1.9 2008/06/26 10:54:12 danielk1977 Exp $
17 ** This file contains some example code demonstrating how the SQLite
18 ** vfs feature can be used to have SQLite operate directly on an
19 ** embedded media, without using an intermediate file system.
21 ** Because this is only a demo designed to run on a workstation, the
22 ** underlying media is simulated using a regular file-system file. The
23 ** size of the file is fixed when it is first created (default size 10 MB).
24 ** From SQLite's point of view, this space is used to store a single
25 ** database file and the journal file.
27 ** Any statement journal created is stored in volatile memory obtained
28 ** from sqlite3_malloc(). Any attempt to create a temporary database file
29 ** will fail (SQLITE_IOERR). To prevent SQLite from attempting this,
30 ** it should be configured to store all temporary database files in
31 ** main memory (see pragma "temp_store" or the SQLITE_TEMP_STORE compile
36 ** After it has been created, the blob file is accessed using the
37 ** following three functions only:
39 ** mediaRead(); - Read a 512 byte block from the file.
40 ** mediaWrite(); - Write a 512 byte block to the file.
41 ** mediaSync(); - Tell the media hardware to sync.
43 ** It is assumed that these can be easily implemented by any "real"
44 ** media vfs driver adapting this code.
48 ** The basic principle is that the "database file" is stored at the
49 ** beginning of the 10 MB blob and grows in a forward direction. The
50 ** "journal file" is stored at the end of the 10MB blob and grows
51 ** in the reverse direction. If, during a transaction, insufficient
52 ** space is available to expand either the journal or database file,
53 ** an SQLITE_FULL error is returned. The database file is never allowed
54 ** to consume more than 90% of the blob space. If SQLite tries to
55 ** create a file larger than this, SQLITE_FULL is returned.
57 ** No allowance is made for "wear-leveling", as is required by.
58 ** embedded devices in the absence of equivalent hardware features.
60 ** The first 512 block byte of the file is reserved for storing the
61 ** size of the "database file". It is updated as part of the sync()
62 ** operation. On startup, it can only be trusted if no journal file
63 ** exists. If a journal-file does exist, then it stores the real size
64 ** of the database region. The second and subsequent blocks store the
65 ** actual database content.
67 ** The size of the "journal file" is not stored persistently in the
68 ** file. When the system is running, the size of the journal file is
69 ** stored in volatile memory. When recovering from a crash, this vfs
70 ** reports a very large size for the journal file. The normal journal
71 ** header and checksum mechanisms serve to prevent SQLite from
72 ** processing any data that lies past the logical end of the journal.
74 ** When SQLite calls OsDelete() to delete the journal file, the final
75 ** 512 bytes of the blob (the area containing the first journal header)
80 ** File locking is a no-op. Only one connection may be open at any one
81 ** time using this demo vfs.
89 ** Maximum pathname length supported by the fs backend.
92 #define BLOBSIZE 10485760
95 ** Name used to identify this VFS.
97 #define FS_VFS_NAME "fs"
99 typedef struct fs_real_file fs_real_file;
100 struct fs_real_file {
103 int nDatabase; /* Current size of database region */
104 int nJournal; /* Current size of journal region */
105 int nBlob; /* Total size of allocated blob */
106 int nRef; /* Number of pointers to this structure */
108 fs_real_file **ppThis;
111 typedef struct fs_file fs_file;
118 typedef struct tmp_file tmp_file;
126 /* Values for fs_file.eType. */
127 #define DATABASE_FILE 1
128 #define JOURNAL_FILE 2
131 ** Method declarations for fs_file.
133 static int fsClose(sqlite3_file*);
134 static int fsRead(sqlite3_file*, void*, int iAmt, sqlite3_int64 iOfst);
135 static int fsWrite(sqlite3_file*, const void*, int iAmt, sqlite3_int64 iOfst);
136 static int fsTruncate(sqlite3_file*, sqlite3_int64 size);
137 static int fsSync(sqlite3_file*, int flags);
138 static int fsFileSize(sqlite3_file*, sqlite3_int64 *pSize);
139 static int fsLock(sqlite3_file*, int);
140 static int fsUnlock(sqlite3_file*, int);
141 static int fsCheckReservedLock(sqlite3_file*, int *pResOut);
142 static int fsFileControl(sqlite3_file*, int op, void *pArg);
143 static int fsSectorSize(sqlite3_file*);
144 static int fsDeviceCharacteristics(sqlite3_file*);
147 ** Method declarations for tmp_file.
149 static int tmpClose(sqlite3_file*);
150 static int tmpRead(sqlite3_file*, void*, int iAmt, sqlite3_int64 iOfst);
151 static int tmpWrite(sqlite3_file*, const void*, int iAmt, sqlite3_int64 iOfst);
152 static int tmpTruncate(sqlite3_file*, sqlite3_int64 size);
153 static int tmpSync(sqlite3_file*, int flags);
154 static int tmpFileSize(sqlite3_file*, sqlite3_int64 *pSize);
155 static int tmpLock(sqlite3_file*, int);
156 static int tmpUnlock(sqlite3_file*, int);
157 static int tmpCheckReservedLock(sqlite3_file*, int *pResOut);
158 static int tmpFileControl(sqlite3_file*, int op, void *pArg);
159 static int tmpSectorSize(sqlite3_file*);
160 static int tmpDeviceCharacteristics(sqlite3_file*);
163 ** Method declarations for fs_vfs.
165 static int fsOpen(sqlite3_vfs*, const char *, sqlite3_file*, int , int *);
166 static int fsDelete(sqlite3_vfs*, const char *zName, int syncDir);
167 static int fsAccess(sqlite3_vfs*, const char *zName, int flags, int *);
168 static int fsFullPathname(sqlite3_vfs*, const char *zName, int nOut,char *zOut);
169 static void *fsDlOpen(sqlite3_vfs*, const char *zFilename);
170 static void fsDlError(sqlite3_vfs*, int nByte, char *zErrMsg);
171 static void *fsDlSym(sqlite3_vfs*,void*, const char *zSymbol);
172 static void fsDlClose(sqlite3_vfs*, void*);
173 static int fsRandomness(sqlite3_vfs*, int nByte, char *zOut);
174 static int fsSleep(sqlite3_vfs*, int microseconds);
175 static int fsCurrentTime(sqlite3_vfs*, double*);
178 typedef struct fs_vfs_t fs_vfs_t;
181 fs_real_file *pFileList;
182 sqlite3_vfs *pParent;
185 static fs_vfs_t fs_vfs = {
191 FS_VFS_NAME, /* zName */
194 fsDelete, /* xDelete */
195 fsAccess, /* xAccess */
196 fsFullPathname, /* xFullPathname */
197 fsDlOpen, /* xDlOpen */
198 fsDlError, /* xDlError */
199 fsDlSym, /* xDlSym */
200 fsDlClose, /* xDlClose */
201 fsRandomness, /* xRandomness */
202 fsSleep, /* xSleep */
203 fsCurrentTime /* xCurrentTime */
209 static sqlite3_io_methods fs_io_methods = {
211 fsClose, /* xClose */
213 fsWrite, /* xWrite */
214 fsTruncate, /* xTruncate */
216 fsFileSize, /* xFileSize */
218 fsUnlock, /* xUnlock */
219 fsCheckReservedLock, /* xCheckReservedLock */
220 fsFileControl, /* xFileControl */
221 fsSectorSize, /* xSectorSize */
222 fsDeviceCharacteristics /* xDeviceCharacteristics */
226 static sqlite3_io_methods tmp_io_methods = {
228 tmpClose, /* xClose */
230 tmpWrite, /* xWrite */
231 tmpTruncate, /* xTruncate */
233 tmpFileSize, /* xFileSize */
235 tmpUnlock, /* xUnlock */
236 tmpCheckReservedLock, /* xCheckReservedLock */
237 tmpFileControl, /* xFileControl */
238 tmpSectorSize, /* xSectorSize */
239 tmpDeviceCharacteristics /* xDeviceCharacteristics */
242 /* Useful macros used in several places */
243 #define MIN(x,y) ((x)<(y)?(x):(y))
244 #define MAX(x,y) ((x)>(y)?(x):(y))
250 static int tmpClose(sqlite3_file *pFile){
251 tmp_file *pTmp = (tmp_file *)pFile;
252 sqlite3_free(pTmp->zAlloc);
257 ** Read data from a tmp-file.
265 tmp_file *pTmp = (tmp_file *)pFile;
266 if( (iAmt+iOfst)>pTmp->nSize ){
267 return SQLITE_IOERR_SHORT_READ;
269 memcpy(zBuf, &pTmp->zAlloc[iOfst], iAmt);
274 ** Write data to a tmp-file.
282 tmp_file *pTmp = (tmp_file *)pFile;
283 if( (iAmt+iOfst)>pTmp->nAlloc ){
284 int nNew = 2*(iAmt+iOfst+pTmp->nAlloc);
285 char *zNew = sqlite3_realloc(pTmp->zAlloc, nNew);
292 memcpy(&pTmp->zAlloc[iOfst], zBuf, iAmt);
293 pTmp->nSize = MAX(pTmp->nSize, iOfst+iAmt);
298 ** Truncate a tmp-file.
300 static int tmpTruncate(sqlite3_file *pFile, sqlite_int64 size){
301 tmp_file *pTmp = (tmp_file *)pFile;
302 pTmp->nSize = MIN(pTmp->nSize, size);
309 static int tmpSync(sqlite3_file *pFile, int flags){
314 ** Return the current file-size of a tmp-file.
316 static int tmpFileSize(sqlite3_file *pFile, sqlite_int64 *pSize){
317 tmp_file *pTmp = (tmp_file *)pFile;
318 *pSize = pTmp->nSize;
325 static int tmpLock(sqlite3_file *pFile, int eLock){
330 ** Unlock a tmp-file.
332 static int tmpUnlock(sqlite3_file *pFile, int eLock){
337 ** Check if another file-handle holds a RESERVED lock on a tmp-file.
339 static int tmpCheckReservedLock(sqlite3_file *pFile, int *pResOut){
345 ** File control method. For custom operations on a tmp-file.
347 static int tmpFileControl(sqlite3_file *pFile, int op, void *pArg){
352 ** Return the sector-size in bytes for a tmp-file.
354 static int tmpSectorSize(sqlite3_file *pFile){
359 ** Return the device characteristic flags supported by a tmp-file.
361 static int tmpDeviceCharacteristics(sqlite3_file *pFile){
368 static int fsClose(sqlite3_file *pFile){
370 fs_file *p = (fs_file *)pFile;
371 fs_real_file *pReal = p->pReal;
373 /* Decrement the real_file ref-count. */
375 assert(pReal->nRef>=0);
377 /* When the ref-count reaches 0, destroy the structure */
378 if( pReal->nRef==0 ){
379 *pReal->ppThis = pReal->pNext;
381 pReal->pNext->ppThis = pReal->ppThis;
383 rc = pReal->pFile->pMethods->xClose(pReal->pFile);
391 ** Read data from an fs-file.
400 fs_file *p = (fs_file *)pFile;
401 fs_real_file *pReal = p->pReal;
402 sqlite3_file *pF = pReal->pFile;
404 if( (p->eType==DATABASE_FILE && (iAmt+iOfst)>pReal->nDatabase)
405 || (p->eType==JOURNAL_FILE && (iAmt+iOfst)>pReal->nJournal)
407 rc = SQLITE_IOERR_SHORT_READ;
408 }else if( p->eType==DATABASE_FILE ){
409 rc = pF->pMethods->xRead(pF, zBuf, iAmt, iOfst+BLOCKSIZE);
415 while( iRem>0 && rc==SQLITE_OK ){
416 int iRealOff = pReal->nBlob - BLOCKSIZE*((ii/BLOCKSIZE)+1) + ii%BLOCKSIZE;
417 int iRealAmt = MIN(iRem, BLOCKSIZE - (iRealOff%BLOCKSIZE));
419 rc = pF->pMethods->xRead(pF, &((char *)zBuf)[iBuf], iRealAmt, iRealOff);
430 ** Write data to an fs-file.
439 fs_file *p = (fs_file *)pFile;
440 fs_real_file *pReal = p->pReal;
441 sqlite3_file *pF = pReal->pFile;
443 if( p->eType==DATABASE_FILE ){
444 if( (iAmt+iOfst+BLOCKSIZE)>(pReal->nBlob-pReal->nJournal) ){
447 rc = pF->pMethods->xWrite(pF, zBuf, iAmt, iOfst+BLOCKSIZE);
449 pReal->nDatabase = MAX(pReal->nDatabase, iAmt+iOfst);
457 while( iRem>0 && rc==SQLITE_OK ){
458 int iRealOff = pReal->nBlob - BLOCKSIZE*((ii/BLOCKSIZE)+1) + ii%BLOCKSIZE;
459 int iRealAmt = MIN(iRem, BLOCKSIZE - (iRealOff%BLOCKSIZE));
461 if( iRealOff<(pReal->nDatabase+BLOCKSIZE) ){
464 rc = pF->pMethods->xWrite(pF, &((char *)zBuf)[iBuf], iRealAmt,iRealOff);
471 pReal->nJournal = MAX(pReal->nJournal, iAmt+iOfst);
479 ** Truncate an fs-file.
481 static int fsTruncate(sqlite3_file *pFile, sqlite_int64 size){
482 fs_file *p = (fs_file *)pFile;
483 fs_real_file *pReal = p->pReal;
484 if( p->eType==DATABASE_FILE ){
485 pReal->nDatabase = MIN(pReal->nDatabase, size);
487 pReal->nJournal = MIN(pReal->nJournal, size);
495 static int fsSync(sqlite3_file *pFile, int flags){
496 fs_file *p = (fs_file *)pFile;
497 fs_real_file *pReal = p->pReal;
498 sqlite3_file *pRealFile = pReal->pFile;
501 if( p->eType==DATABASE_FILE ){
502 unsigned char zSize[4];
503 zSize[0] = (pReal->nDatabase&0xFF000000)>>24;
504 zSize[1] = (pReal->nDatabase&0x00FF0000)>>16;
505 zSize[2] = (pReal->nDatabase&0x0000FF00)>>8;
506 zSize[3] = (pReal->nDatabase&0x000000FF);
507 rc = pRealFile->pMethods->xWrite(pRealFile, zSize, 4, 0);
510 rc = pRealFile->pMethods->xSync(pRealFile, flags&(~SQLITE_SYNC_DATAONLY));
517 ** Return the current file-size of an fs-file.
519 static int fsFileSize(sqlite3_file *pFile, sqlite_int64 *pSize){
520 fs_file *p = (fs_file *)pFile;
521 fs_real_file *pReal = p->pReal;
522 if( p->eType==DATABASE_FILE ){
523 *pSize = pReal->nDatabase;
525 *pSize = pReal->nJournal;
533 static int fsLock(sqlite3_file *pFile, int eLock){
538 ** Unlock an fs-file.
540 static int fsUnlock(sqlite3_file *pFile, int eLock){
545 ** Check if another file-handle holds a RESERVED lock on an fs-file.
547 static int fsCheckReservedLock(sqlite3_file *pFile, int *pResOut){
553 ** File control method. For custom operations on an fs-file.
555 static int fsFileControl(sqlite3_file *pFile, int op, void *pArg){
560 ** Return the sector-size in bytes for an fs-file.
562 static int fsSectorSize(sqlite3_file *pFile){
567 ** Return the device characteristic flags supported by an fs-file.
569 static int fsDeviceCharacteristics(sqlite3_file *pFile){
574 ** Open an fs file handle.
583 fs_vfs_t *pFsVfs = (fs_vfs_t *)pVfs;
584 fs_file *p = (fs_file *)pFile;
585 fs_real_file *pReal = 0;
590 if( 0==(flags&(SQLITE_OPEN_MAIN_DB|SQLITE_OPEN_MAIN_JOURNAL)) ){
591 tmp_file *p = (tmp_file *)pFile;
592 memset(p, 0, sizeof(*p));
593 p->base.pMethods = &tmp_io_methods;
597 eType = ((flags&(SQLITE_OPEN_MAIN_DB))?DATABASE_FILE:JOURNAL_FILE);
598 p->base.pMethods = &fs_io_methods;
601 assert(strlen("-journal")==8);
602 nName = strlen(zName)-((eType==JOURNAL_FILE)?8:0);
603 pReal=pFsVfs->pFileList;
604 for(; pReal && strncmp(pReal->zName, zName, nName); pReal=pReal->pNext);
608 sqlite3_file *pRealFile;
609 sqlite3_vfs *pParent = pFsVfs->pParent;
610 assert(eType==DATABASE_FILE);
612 pReal = (fs_real_file *)sqlite3_malloc(sizeof(*pReal)+pParent->szOsFile);
617 memset(pReal, 0, sizeof(*pReal)+pParent->szOsFile);
618 pReal->zName = zName;
619 pReal->pFile = (sqlite3_file *)(&pReal[1]);
621 rc = pParent->xOpen(pParent, zName, pReal->pFile, flags, pOutFlags);
625 pRealFile = pReal->pFile;
627 rc = pRealFile->pMethods->xFileSize(pRealFile, &size);
632 rc = pRealFile->pMethods->xWrite(pRealFile, "\0", 1, BLOBSIZE-1);
633 pReal->nBlob = BLOBSIZE;
637 rc = pRealFile->pMethods->xRead(pRealFile, zS, 4, 0);
638 pReal->nDatabase = (zS[0]<<24)+(zS[1]<<16)+(zS[2]<<8)+zS[3];
640 rc = pRealFile->pMethods->xRead(pRealFile, zS, 4, pReal->nBlob-4);
641 if( zS[0] || zS[1] || zS[2] || zS[3] ){
642 pReal->nJournal = pReal->nBlob;
648 pReal->pNext = pFsVfs->pFileList;
650 pReal->pNext->ppThis = &pReal->pNext;
652 pReal->ppThis = &pFsVfs->pFileList;
653 pFsVfs->pFileList = pReal;
663 if( pReal->pFile->pMethods ){
664 pReal->pFile->pMethods->xClose(pReal->pFile);
673 ** Delete the file located at zPath. If the dirSync argument is true,
674 ** ensure the file-system modifications are synced to disk before
677 static int fsDelete(sqlite3_vfs *pVfs, const char *zPath, int dirSync){
679 fs_vfs_t *pFsVfs = (fs_vfs_t *)pVfs;
682 int nName = strlen(zPath) - 8;
684 assert(strlen("-journal")==8);
685 assert(strcmp("-journal", &zPath[nName])==0);
687 pReal = pFsVfs->pFileList;
688 for(; pReal && strncmp(pReal->zName, zPath, nName); pReal=pReal->pNext);
691 rc = pF->pMethods->xWrite(pF, "\0\0\0\0", 4, pReal->nBlob-BLOCKSIZE);
700 ** Test for access permissions. Return true if the requested permission
701 ** is available, or false otherwise.
709 fs_vfs_t *pFsVfs = (fs_vfs_t *)pVfs;
712 int nName = strlen(zPath);
714 if( flags!=SQLITE_ACCESS_EXISTS ){
715 sqlite3_vfs *pParent = ((fs_vfs_t *)pVfs)->pParent;
716 return pParent->xAccess(pParent, zPath, flags, pResOut);
719 assert(strlen("-journal")==8);
720 if( nName>8 && strcmp("-journal", &zPath[nName-8])==0 ){
725 pReal = pFsVfs->pFileList;
726 for(; pReal && strncmp(pReal->zName, zPath, nName); pReal=pReal->pNext);
728 *pResOut = (pReal && (!isJournal || pReal->nJournal>0));
733 ** Populate buffer zOut with the full canonical pathname corresponding
734 ** to the pathname in zPath. zOut is guaranteed to point to a buffer
735 ** of at least (FS_MAX_PATHNAME+1) bytes.
737 static int fsFullPathname(
738 sqlite3_vfs *pVfs, /* Pointer to vfs object */
739 const char *zPath, /* Possibly relative input path */
740 int nOut, /* Size of output buffer in bytes */
741 char *zOut /* Output buffer */
743 sqlite3_vfs *pParent = ((fs_vfs_t *)pVfs)->pParent;
744 return pParent->xFullPathname(pParent, zPath, nOut, zOut);
748 ** Open the dynamic library located at zPath and return a handle.
750 static void *fsDlOpen(sqlite3_vfs *pVfs, const char *zPath){
751 sqlite3_vfs *pParent = ((fs_vfs_t *)pVfs)->pParent;
752 return pParent->xDlOpen(pParent, zPath);
756 ** Populate the buffer zErrMsg (size nByte bytes) with a human readable
757 ** utf-8 string describing the most recent error encountered associated
758 ** with dynamic libraries.
760 static void fsDlError(sqlite3_vfs *pVfs, int nByte, char *zErrMsg){
761 sqlite3_vfs *pParent = ((fs_vfs_t *)pVfs)->pParent;
762 pParent->xDlError(pParent, nByte, zErrMsg);
766 ** Return a pointer to the symbol zSymbol in the dynamic library pHandle.
768 static void *fsDlSym(sqlite3_vfs *pVfs, void *pHandle, const char *zSymbol){
769 sqlite3_vfs *pParent = ((fs_vfs_t *)pVfs)->pParent;
770 return pParent->xDlSym(pParent, pHandle, zSymbol);
774 ** Close the dynamic library handle pHandle.
776 static void fsDlClose(sqlite3_vfs *pVfs, void *pHandle){
777 sqlite3_vfs *pParent = ((fs_vfs_t *)pVfs)->pParent;
778 pParent->xDlClose(pParent, pHandle);
782 ** Populate the buffer pointed to by zBufOut with nByte bytes of
785 static int fsRandomness(sqlite3_vfs *pVfs, int nByte, char *zBufOut){
786 sqlite3_vfs *pParent = ((fs_vfs_t *)pVfs)->pParent;
787 return pParent->xRandomness(pParent, nByte, zBufOut);
791 ** Sleep for nMicro microseconds. Return the number of microseconds
794 static int fsSleep(sqlite3_vfs *pVfs, int nMicro){
795 sqlite3_vfs *pParent = ((fs_vfs_t *)pVfs)->pParent;
796 return pParent->xSleep(pParent, nMicro);
800 ** Return the current time as a Julian Day number in *pTimeOut.
802 static int fsCurrentTime(sqlite3_vfs *pVfs, double *pTimeOut){
803 sqlite3_vfs *pParent = ((fs_vfs_t *)pVfs)->pParent;
804 return pParent->xCurrentTime(pParent, pTimeOut);
808 ** This procedure registers the fs vfs with SQLite. If the argument is
809 ** true, the fs vfs becomes the new default vfs. It is the only publicly
810 ** available function in this file.
813 if( fs_vfs.pParent ) return SQLITE_OK;
814 fs_vfs.pParent = sqlite3_vfs_find(0);
815 fs_vfs.base.mxPathname = fs_vfs.pParent->mxPathname;
816 fs_vfs.base.szOsFile = MAX(sizeof(tmp_file), sizeof(fs_file));
817 return sqlite3_vfs_register(&fs_vfs.base, 0);
821 int SqlitetestOnefile_Init() {return fs_register();}