Update contrib.
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: journal.c,v 1.8 2008/05/01 18:01:47 drh Exp $
16 #ifdef SQLITE_ENABLE_ATOMIC_WRITE
19 ** This file implements a special kind of sqlite3_file object used
20 ** by SQLite to create journal files if the atomic-write optimization
23 ** The distinctive characteristic of this sqlite3_file is that the
24 ** actual on disk file is created lazily. When the file is created,
25 ** the caller specifies a buffer size for an in-memory buffer to
26 ** be used to service read() and write() requests. The actual file
27 ** on disk is not created or populated until either:
29 ** 1) The in-memory representation grows too large for the allocated
31 ** 2) The xSync() method is called.
34 #include "sqliteInt.h"
38 ** A JournalFile object is a subclass of sqlite3_file used by
39 ** as an open file handle for journal files.
42 sqlite3_io_methods *pMethod; /* I/O methods on journal files */
43 int nBuf; /* Size of zBuf[] in bytes */
44 char *zBuf; /* Space to buffer journal writes */
45 int iSize; /* Amount of zBuf[] currently used */
46 int flags; /* xOpen flags */
47 sqlite3_vfs *pVfs; /* The "real" underlying VFS */
48 sqlite3_file *pReal; /* The "real" underlying file descriptor */
49 const char *zJournal; /* Name of the journal file */
51 typedef struct JournalFile JournalFile;
54 ** If it does not already exists, create and populate the on-disk file
57 static int createFile(JournalFile *p){
60 sqlite3_file *pReal = (sqlite3_file *)&p[1];
61 rc = sqlite3OsOpen(p->pVfs, p->zJournal, pReal, p->flags, 0);
65 assert(p->iSize<=p->nBuf);
66 rc = sqlite3OsWrite(p->pReal, p->zBuf, p->iSize, 0);
76 static int jrnlClose(sqlite3_file *pJfd){
77 JournalFile *p = (JournalFile *)pJfd;
79 sqlite3OsClose(p->pReal);
81 sqlite3_free(p->zBuf);
86 ** Read data from the file.
89 sqlite3_file *pJfd, /* The journal file from which to read */
90 void *zBuf, /* Put the results here */
91 int iAmt, /* Number of bytes to read */
92 sqlite_int64 iOfst /* Begin reading at this offset */
95 JournalFile *p = (JournalFile *)pJfd;
97 rc = sqlite3OsRead(p->pReal, zBuf, iAmt, iOfst);
99 assert( iAmt+iOfst<=p->iSize );
100 memcpy(zBuf, &p->zBuf[iOfst], iAmt);
106 ** Write data to the file.
108 static int jrnlWrite(
109 sqlite3_file *pJfd, /* The journal file into which to write */
110 const void *zBuf, /* Take data to be written from here */
111 int iAmt, /* Number of bytes to write */
112 sqlite_int64 iOfst /* Begin writing at this offset into the file */
115 JournalFile *p = (JournalFile *)pJfd;
116 if( !p->pReal && (iOfst+iAmt)>p->nBuf ){
121 rc = sqlite3OsWrite(p->pReal, zBuf, iAmt, iOfst);
123 memcpy(&p->zBuf[iOfst], zBuf, iAmt);
124 if( p->iSize<(iOfst+iAmt) ){
125 p->iSize = (iOfst+iAmt);
133 ** Truncate the file.
135 static int jrnlTruncate(sqlite3_file *pJfd, sqlite_int64 size){
137 JournalFile *p = (JournalFile *)pJfd;
139 rc = sqlite3OsTruncate(p->pReal, size);
140 }else if( size<p->iSize ){
149 static int jrnlSync(sqlite3_file *pJfd, int flags){
151 JournalFile *p = (JournalFile *)pJfd;
153 rc = sqlite3OsSync(p->pReal, flags);
161 ** Query the size of the file in bytes.
163 static int jrnlFileSize(sqlite3_file *pJfd, sqlite_int64 *pSize){
165 JournalFile *p = (JournalFile *)pJfd;
167 rc = sqlite3OsFileSize(p->pReal, pSize);
169 *pSize = (sqlite_int64) p->iSize;
175 ** Table of methods for JournalFile sqlite3_file object.
177 static struct sqlite3_io_methods JournalFileMethods = {
179 jrnlClose, /* xClose */
180 jrnlRead, /* xRead */
181 jrnlWrite, /* xWrite */
182 jrnlTruncate, /* xTruncate */
183 jrnlSync, /* xSync */
184 jrnlFileSize, /* xFileSize */
187 0, /* xCheckReservedLock */
188 0, /* xFileControl */
190 0 /* xDeviceCharacteristics */
194 ** Open a journal file.
196 int sqlite3JournalOpen(
197 sqlite3_vfs *pVfs, /* The VFS to use for actual file I/O */
198 const char *zName, /* Name of the journal file */
199 sqlite3_file *pJfd, /* Preallocated, blank file handle */
200 int flags, /* Opening flags */
201 int nBuf /* Bytes buffered before opening the file */
203 JournalFile *p = (JournalFile *)pJfd;
204 memset(p, 0, sqlite3JournalSize(pVfs));
206 p->zBuf = sqlite3MallocZero(nBuf);
211 return sqlite3OsOpen(pVfs, zName, pJfd, flags, 0);
213 p->pMethod = &JournalFileMethods;
222 ** If the argument p points to a JournalFile structure, and the underlying
223 ** file has not yet been created, create it now.
225 int sqlite3JournalCreate(sqlite3_file *p){
226 if( p->pMethods!=&JournalFileMethods ){
229 return createFile((JournalFile *)p);
233 ** Return the number of bytes required to store a JournalFile that uses vfs
234 ** pVfs to create the underlying on-disk files.
236 int sqlite3JournalSize(sqlite3_vfs *pVfs){
237 return (pVfs->szOsFile+sizeof(JournalFile));