sl@0
|
1 |
/*
|
sl@0
|
2 |
** 2004 May 22
|
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 |
** This file contains code that modified the OS layer in order to simulate
|
sl@0
|
14 |
** the effect on the database file of an OS crash or power failure. This
|
sl@0
|
15 |
** is used to test the ability of SQLite to recover from those situations.
|
sl@0
|
16 |
**
|
sl@0
|
17 |
** $Id: test6.c,v 1.39 2008/06/06 11:11:26 danielk1977 Exp $
|
sl@0
|
18 |
*/
|
sl@0
|
19 |
#if SQLITE_TEST /* This file is used for testing only */
|
sl@0
|
20 |
#include "sqliteInt.h"
|
sl@0
|
21 |
#include "tcl.h"
|
sl@0
|
22 |
|
sl@0
|
23 |
#ifndef SQLITE_OMIT_DISKIO /* This file is a no-op if disk I/O is disabled */
|
sl@0
|
24 |
|
sl@0
|
25 |
/* #define TRACE_CRASHTEST */
|
sl@0
|
26 |
|
sl@0
|
27 |
typedef struct CrashFile CrashFile;
|
sl@0
|
28 |
typedef struct CrashGlobal CrashGlobal;
|
sl@0
|
29 |
typedef struct WriteBuffer WriteBuffer;
|
sl@0
|
30 |
|
sl@0
|
31 |
/*
|
sl@0
|
32 |
** Method:
|
sl@0
|
33 |
**
|
sl@0
|
34 |
** This layer is implemented as a wrapper around the "real"
|
sl@0
|
35 |
** sqlite3_file object for the host system. Each time data is
|
sl@0
|
36 |
** written to the file object, instead of being written to the
|
sl@0
|
37 |
** underlying file, the write operation is stored in an in-memory
|
sl@0
|
38 |
** structure (type WriteBuffer). This structure is placed at the
|
sl@0
|
39 |
** end of a global ordered list (the write-list).
|
sl@0
|
40 |
**
|
sl@0
|
41 |
** When data is read from a file object, the requested region is
|
sl@0
|
42 |
** first retrieved from the real file. The write-list is then
|
sl@0
|
43 |
** traversed and data copied from any overlapping WriteBuffer
|
sl@0
|
44 |
** structures to the output buffer. i.e. a read() operation following
|
sl@0
|
45 |
** one or more write() operations works as expected, even if no
|
sl@0
|
46 |
** data has actually been written out to the real file.
|
sl@0
|
47 |
**
|
sl@0
|
48 |
** When a fsync() operation is performed, an operating system crash
|
sl@0
|
49 |
** may be simulated, in which case exit(-1) is called (the call to
|
sl@0
|
50 |
** xSync() never returns). Whether or not a crash is simulated,
|
sl@0
|
51 |
** the data associated with a subset of the WriteBuffer structures
|
sl@0
|
52 |
** stored in the write-list is written to the real underlying files
|
sl@0
|
53 |
** and the entries removed from the write-list. If a crash is simulated,
|
sl@0
|
54 |
** a subset of the buffers may be corrupted before the data is written.
|
sl@0
|
55 |
**
|
sl@0
|
56 |
** The exact subset of the write-list written and/or corrupted is
|
sl@0
|
57 |
** determined by the simulated device characteristics and sector-size.
|
sl@0
|
58 |
**
|
sl@0
|
59 |
** "Normal" mode:
|
sl@0
|
60 |
**
|
sl@0
|
61 |
** Normal mode is used when the simulated device has none of the
|
sl@0
|
62 |
** SQLITE_IOCAP_XXX flags set.
|
sl@0
|
63 |
**
|
sl@0
|
64 |
** In normal mode, if the fsync() is not a simulated crash, the
|
sl@0
|
65 |
** write-list is traversed from beginning to end. Each WriteBuffer
|
sl@0
|
66 |
** structure associated with the file handle used to call xSync()
|
sl@0
|
67 |
** is written to the real file and removed from the write-list.
|
sl@0
|
68 |
**
|
sl@0
|
69 |
** If a crash is simulated, one of the following takes place for
|
sl@0
|
70 |
** each WriteBuffer in the write-list, regardless of which
|
sl@0
|
71 |
** file-handle it is associated with:
|
sl@0
|
72 |
**
|
sl@0
|
73 |
** 1. The buffer is correctly written to the file, just as if
|
sl@0
|
74 |
** a crash were not being simulated.
|
sl@0
|
75 |
**
|
sl@0
|
76 |
** 2. Nothing is done.
|
sl@0
|
77 |
**
|
sl@0
|
78 |
** 3. Garbage data is written to all sectors of the file that
|
sl@0
|
79 |
** overlap the region specified by the WriteBuffer. Or garbage
|
sl@0
|
80 |
** data is written to some contiguous section within the
|
sl@0
|
81 |
** overlapped sectors.
|
sl@0
|
82 |
**
|
sl@0
|
83 |
** Device Characteristic flag handling:
|
sl@0
|
84 |
**
|
sl@0
|
85 |
** If the IOCAP_ATOMIC flag is set, then option (3) above is
|
sl@0
|
86 |
** never selected.
|
sl@0
|
87 |
**
|
sl@0
|
88 |
** If the IOCAP_ATOMIC512 flag is set, and the WriteBuffer represents
|
sl@0
|
89 |
** an aligned write() of an integer number of 512 byte regions, then
|
sl@0
|
90 |
** option (3) above is never selected. Instead, each 512 byte region
|
sl@0
|
91 |
** is either correctly written or left completely untouched. Similar
|
sl@0
|
92 |
** logic governs the behaviour if any of the other ATOMICXXX flags
|
sl@0
|
93 |
** is set.
|
sl@0
|
94 |
**
|
sl@0
|
95 |
** If either the IOCAP_SAFEAPPEND or IOCAP_SEQUENTIAL flags are set
|
sl@0
|
96 |
** and a crash is being simulated, then an entry of the write-list is
|
sl@0
|
97 |
** selected at random. Everything in the list after the selected entry
|
sl@0
|
98 |
** is discarded before processing begins.
|
sl@0
|
99 |
**
|
sl@0
|
100 |
** If IOCAP_SEQUENTIAL is set and a crash is being simulated, option
|
sl@0
|
101 |
** (1) is selected for all write-list entries except the last. If a
|
sl@0
|
102 |
** crash is not being simulated, then all entries in the write-list
|
sl@0
|
103 |
** that occur before at least one write() on the file-handle specified
|
sl@0
|
104 |
** as part of the xSync() are written to their associated real files.
|
sl@0
|
105 |
**
|
sl@0
|
106 |
** If IOCAP_SAFEAPPEND is set and the first byte written by the write()
|
sl@0
|
107 |
** operation is one byte past the current end of the file, then option
|
sl@0
|
108 |
** (1) is always selected.
|
sl@0
|
109 |
*/
|
sl@0
|
110 |
|
sl@0
|
111 |
/*
|
sl@0
|
112 |
** Each write operation in the write-list is represented by an instance
|
sl@0
|
113 |
** of the following structure.
|
sl@0
|
114 |
**
|
sl@0
|
115 |
** If zBuf is 0, then this structure represents a call to xTruncate(),
|
sl@0
|
116 |
** not xWrite(). In that case, iOffset is the size that the file is
|
sl@0
|
117 |
** truncated to.
|
sl@0
|
118 |
*/
|
sl@0
|
119 |
struct WriteBuffer {
|
sl@0
|
120 |
i64 iOffset; /* Byte offset of the start of this write() */
|
sl@0
|
121 |
int nBuf; /* Number of bytes written */
|
sl@0
|
122 |
u8 *zBuf; /* Pointer to copy of written data */
|
sl@0
|
123 |
CrashFile *pFile; /* File this write() applies to */
|
sl@0
|
124 |
|
sl@0
|
125 |
WriteBuffer *pNext; /* Next in CrashGlobal.pWriteList */
|
sl@0
|
126 |
};
|
sl@0
|
127 |
|
sl@0
|
128 |
struct CrashFile {
|
sl@0
|
129 |
const sqlite3_io_methods *pMethod; /* Must be first */
|
sl@0
|
130 |
sqlite3_file *pRealFile; /* Underlying "real" file handle */
|
sl@0
|
131 |
char *zName;
|
sl@0
|
132 |
|
sl@0
|
133 |
/* Cache of the entire file. This is used to speed up OsRead() and
|
sl@0
|
134 |
** OsFileSize() calls. Although both could be done by traversing the
|
sl@0
|
135 |
** write-list, in practice this is impractically slow.
|
sl@0
|
136 |
*/
|
sl@0
|
137 |
int iSize; /* Size of file in bytes */
|
sl@0
|
138 |
int nData; /* Size of buffer allocated at zData */
|
sl@0
|
139 |
u8 *zData; /* Buffer containing file contents */
|
sl@0
|
140 |
};
|
sl@0
|
141 |
|
sl@0
|
142 |
struct CrashGlobal {
|
sl@0
|
143 |
WriteBuffer *pWriteList; /* Head of write-list */
|
sl@0
|
144 |
WriteBuffer *pWriteListEnd; /* End of write-list */
|
sl@0
|
145 |
|
sl@0
|
146 |
int iSectorSize; /* Value of simulated sector size */
|
sl@0
|
147 |
int iDeviceCharacteristics; /* Value of simulated device characteristics */
|
sl@0
|
148 |
|
sl@0
|
149 |
int iCrash; /* Crash on the iCrash'th call to xSync() */
|
sl@0
|
150 |
char zCrashFile[500]; /* Crash during an xSync() on this file */
|
sl@0
|
151 |
};
|
sl@0
|
152 |
|
sl@0
|
153 |
static CrashGlobal g = {0, 0, SQLITE_DEFAULT_SECTOR_SIZE, 0, 0};
|
sl@0
|
154 |
|
sl@0
|
155 |
/*
|
sl@0
|
156 |
** Set this global variable to 1 to enable crash testing.
|
sl@0
|
157 |
*/
|
sl@0
|
158 |
static int sqlite3CrashTestEnable = 0;
|
sl@0
|
159 |
|
sl@0
|
160 |
static void *crash_malloc(int nByte){
|
sl@0
|
161 |
return (void *)Tcl_Alloc((size_t)nByte);
|
sl@0
|
162 |
}
|
sl@0
|
163 |
static void crash_free(void *p){
|
sl@0
|
164 |
Tcl_Free(p);
|
sl@0
|
165 |
}
|
sl@0
|
166 |
static void *crash_realloc(void *p, int n){
|
sl@0
|
167 |
return (void *)Tcl_Realloc(p, (size_t)n);
|
sl@0
|
168 |
}
|
sl@0
|
169 |
|
sl@0
|
170 |
/*
|
sl@0
|
171 |
** Flush the write-list as if xSync() had been called on file handle
|
sl@0
|
172 |
** pFile. If isCrash is true, simulate a crash.
|
sl@0
|
173 |
*/
|
sl@0
|
174 |
static int writeListSync(CrashFile *pFile, int isCrash){
|
sl@0
|
175 |
int rc = SQLITE_OK;
|
sl@0
|
176 |
int iDc = g.iDeviceCharacteristics;
|
sl@0
|
177 |
|
sl@0
|
178 |
WriteBuffer *pWrite;
|
sl@0
|
179 |
WriteBuffer **ppPtr;
|
sl@0
|
180 |
|
sl@0
|
181 |
/* If this is not a crash simulation, set pFinal to point to the
|
sl@0
|
182 |
** last element of the write-list that is associated with file handle
|
sl@0
|
183 |
** pFile.
|
sl@0
|
184 |
**
|
sl@0
|
185 |
** If this is a crash simulation, set pFinal to an arbitrarily selected
|
sl@0
|
186 |
** element of the write-list.
|
sl@0
|
187 |
*/
|
sl@0
|
188 |
WriteBuffer *pFinal = 0;
|
sl@0
|
189 |
if( !isCrash ){
|
sl@0
|
190 |
for(pWrite=g.pWriteList; pWrite; pWrite=pWrite->pNext){
|
sl@0
|
191 |
if( pWrite->pFile==pFile ){
|
sl@0
|
192 |
pFinal = pWrite;
|
sl@0
|
193 |
}
|
sl@0
|
194 |
}
|
sl@0
|
195 |
}else if( iDc&(SQLITE_IOCAP_SEQUENTIAL|SQLITE_IOCAP_SAFE_APPEND) ){
|
sl@0
|
196 |
int nWrite = 0;
|
sl@0
|
197 |
int iFinal;
|
sl@0
|
198 |
for(pWrite=g.pWriteList; pWrite; pWrite=pWrite->pNext) nWrite++;
|
sl@0
|
199 |
sqlite3_randomness(sizeof(int), &iFinal);
|
sl@0
|
200 |
iFinal = ((iFinal<0)?-1*iFinal:iFinal)%nWrite;
|
sl@0
|
201 |
for(pWrite=g.pWriteList; iFinal>0; pWrite=pWrite->pNext) iFinal--;
|
sl@0
|
202 |
pFinal = pWrite;
|
sl@0
|
203 |
}
|
sl@0
|
204 |
|
sl@0
|
205 |
#ifdef TRACE_CRASHTEST
|
sl@0
|
206 |
printf("Sync %s (is %s crash)\n", pFile->zName, (isCrash?"a":"not a"));
|
sl@0
|
207 |
#endif
|
sl@0
|
208 |
|
sl@0
|
209 |
ppPtr = &g.pWriteList;
|
sl@0
|
210 |
for(pWrite=*ppPtr; rc==SQLITE_OK && pWrite; pWrite=*ppPtr){
|
sl@0
|
211 |
sqlite3_file *pRealFile = pWrite->pFile->pRealFile;
|
sl@0
|
212 |
|
sl@0
|
213 |
/* (eAction==1) -> write block out normally,
|
sl@0
|
214 |
** (eAction==2) -> do nothing,
|
sl@0
|
215 |
** (eAction==3) -> trash sectors.
|
sl@0
|
216 |
*/
|
sl@0
|
217 |
int eAction = 0;
|
sl@0
|
218 |
if( !isCrash ){
|
sl@0
|
219 |
eAction = 2;
|
sl@0
|
220 |
if( (pWrite->pFile==pFile || iDc&SQLITE_IOCAP_SEQUENTIAL) ){
|
sl@0
|
221 |
eAction = 1;
|
sl@0
|
222 |
}
|
sl@0
|
223 |
}else{
|
sl@0
|
224 |
char random;
|
sl@0
|
225 |
sqlite3_randomness(1, &random);
|
sl@0
|
226 |
|
sl@0
|
227 |
/* Do not select option 3 (sector trashing) if the IOCAP_ATOMIC flag
|
sl@0
|
228 |
** is set or this is an OsTruncate(), not an Oswrite().
|
sl@0
|
229 |
*/
|
sl@0
|
230 |
if( (iDc&SQLITE_IOCAP_ATOMIC) || (pWrite->zBuf==0) ){
|
sl@0
|
231 |
random &= 0x01;
|
sl@0
|
232 |
}
|
sl@0
|
233 |
|
sl@0
|
234 |
/* If IOCAP_SEQUENTIAL is set and this is not the final entry
|
sl@0
|
235 |
** in the truncated write-list, always select option 1 (write
|
sl@0
|
236 |
** out correctly).
|
sl@0
|
237 |
*/
|
sl@0
|
238 |
if( (iDc&SQLITE_IOCAP_SEQUENTIAL && pWrite!=pFinal) ){
|
sl@0
|
239 |
random = 0;
|
sl@0
|
240 |
}
|
sl@0
|
241 |
|
sl@0
|
242 |
/* If IOCAP_SAFE_APPEND is set and this OsWrite() operation is
|
sl@0
|
243 |
** an append (first byte of the written region is 1 byte past the
|
sl@0
|
244 |
** current EOF), always select option 1 (write out correctly).
|
sl@0
|
245 |
*/
|
sl@0
|
246 |
if( iDc&SQLITE_IOCAP_SAFE_APPEND && pWrite->zBuf ){
|
sl@0
|
247 |
i64 iSize;
|
sl@0
|
248 |
sqlite3OsFileSize(pRealFile, &iSize);
|
sl@0
|
249 |
if( iSize==pWrite->iOffset ){
|
sl@0
|
250 |
random = 0;
|
sl@0
|
251 |
}
|
sl@0
|
252 |
}
|
sl@0
|
253 |
|
sl@0
|
254 |
if( (random&0x06)==0x06 ){
|
sl@0
|
255 |
eAction = 3;
|
sl@0
|
256 |
}else{
|
sl@0
|
257 |
eAction = ((random&0x01)?2:1);
|
sl@0
|
258 |
}
|
sl@0
|
259 |
}
|
sl@0
|
260 |
|
sl@0
|
261 |
switch( eAction ){
|
sl@0
|
262 |
case 1: { /* Write out correctly */
|
sl@0
|
263 |
if( pWrite->zBuf ){
|
sl@0
|
264 |
rc = sqlite3OsWrite(
|
sl@0
|
265 |
pRealFile, pWrite->zBuf, pWrite->nBuf, pWrite->iOffset
|
sl@0
|
266 |
);
|
sl@0
|
267 |
}else{
|
sl@0
|
268 |
rc = sqlite3OsTruncate(pRealFile, pWrite->iOffset);
|
sl@0
|
269 |
}
|
sl@0
|
270 |
*ppPtr = pWrite->pNext;
|
sl@0
|
271 |
#ifdef TRACE_CRASHTEST
|
sl@0
|
272 |
if( isCrash ){
|
sl@0
|
273 |
printf("Writing %d bytes @ %d (%s)\n",
|
sl@0
|
274 |
pWrite->nBuf, (int)pWrite->iOffset, pWrite->pFile->zName
|
sl@0
|
275 |
);
|
sl@0
|
276 |
}
|
sl@0
|
277 |
#endif
|
sl@0
|
278 |
crash_free(pWrite);
|
sl@0
|
279 |
break;
|
sl@0
|
280 |
}
|
sl@0
|
281 |
case 2: { /* Do nothing */
|
sl@0
|
282 |
ppPtr = &pWrite->pNext;
|
sl@0
|
283 |
#ifdef TRACE_CRASHTEST
|
sl@0
|
284 |
if( isCrash ){
|
sl@0
|
285 |
printf("Omiting %d bytes @ %d (%s)\n",
|
sl@0
|
286 |
pWrite->nBuf, (int)pWrite->iOffset, pWrite->pFile->zName
|
sl@0
|
287 |
);
|
sl@0
|
288 |
}
|
sl@0
|
289 |
#endif
|
sl@0
|
290 |
break;
|
sl@0
|
291 |
}
|
sl@0
|
292 |
case 3: { /* Trash sectors */
|
sl@0
|
293 |
u8 *zGarbage;
|
sl@0
|
294 |
int iFirst = (pWrite->iOffset/g.iSectorSize);
|
sl@0
|
295 |
int iLast = (pWrite->iOffset+pWrite->nBuf-1)/g.iSectorSize;
|
sl@0
|
296 |
|
sl@0
|
297 |
assert(pWrite->zBuf);
|
sl@0
|
298 |
|
sl@0
|
299 |
#ifdef TRACE_CRASHTEST
|
sl@0
|
300 |
printf("Trashing %d sectors @ sector %d (%s)\n",
|
sl@0
|
301 |
1+iLast-iFirst, iFirst, pWrite->pFile->zName
|
sl@0
|
302 |
);
|
sl@0
|
303 |
#endif
|
sl@0
|
304 |
|
sl@0
|
305 |
zGarbage = crash_malloc(g.iSectorSize);
|
sl@0
|
306 |
if( zGarbage ){
|
sl@0
|
307 |
sqlite3_int64 i;
|
sl@0
|
308 |
for(i=iFirst; rc==SQLITE_OK && i<=iLast; i++){
|
sl@0
|
309 |
sqlite3_randomness(g.iSectorSize, zGarbage);
|
sl@0
|
310 |
rc = sqlite3OsWrite(
|
sl@0
|
311 |
pRealFile, zGarbage, g.iSectorSize, i*g.iSectorSize
|
sl@0
|
312 |
);
|
sl@0
|
313 |
}
|
sl@0
|
314 |
crash_free(zGarbage);
|
sl@0
|
315 |
}else{
|
sl@0
|
316 |
rc = SQLITE_NOMEM;
|
sl@0
|
317 |
}
|
sl@0
|
318 |
|
sl@0
|
319 |
ppPtr = &pWrite->pNext;
|
sl@0
|
320 |
break;
|
sl@0
|
321 |
}
|
sl@0
|
322 |
|
sl@0
|
323 |
default:
|
sl@0
|
324 |
assert(0); /* Cannot happen */
|
sl@0
|
325 |
}
|
sl@0
|
326 |
|
sl@0
|
327 |
if( pWrite==pFinal ) break;
|
sl@0
|
328 |
}
|
sl@0
|
329 |
|
sl@0
|
330 |
if( rc==SQLITE_OK && isCrash ){
|
sl@0
|
331 |
exit(-1);
|
sl@0
|
332 |
}
|
sl@0
|
333 |
|
sl@0
|
334 |
for(pWrite=g.pWriteList; pWrite && pWrite->pNext; pWrite=pWrite->pNext);
|
sl@0
|
335 |
g.pWriteListEnd = pWrite;
|
sl@0
|
336 |
|
sl@0
|
337 |
return rc;
|
sl@0
|
338 |
}
|
sl@0
|
339 |
|
sl@0
|
340 |
/*
|
sl@0
|
341 |
** Add an entry to the end of the write-list.
|
sl@0
|
342 |
*/
|
sl@0
|
343 |
static int writeListAppend(
|
sl@0
|
344 |
sqlite3_file *pFile,
|
sl@0
|
345 |
sqlite3_int64 iOffset,
|
sl@0
|
346 |
const u8 *zBuf,
|
sl@0
|
347 |
int nBuf
|
sl@0
|
348 |
){
|
sl@0
|
349 |
WriteBuffer *pNew;
|
sl@0
|
350 |
|
sl@0
|
351 |
assert((zBuf && nBuf) || (!nBuf && !zBuf));
|
sl@0
|
352 |
|
sl@0
|
353 |
pNew = (WriteBuffer *)crash_malloc(sizeof(WriteBuffer) + nBuf);
|
sl@0
|
354 |
if( pNew==0 ){
|
sl@0
|
355 |
fprintf(stderr, "out of memory in the crash simulator\n");
|
sl@0
|
356 |
}
|
sl@0
|
357 |
memset(pNew, 0, sizeof(WriteBuffer)+nBuf);
|
sl@0
|
358 |
pNew->iOffset = iOffset;
|
sl@0
|
359 |
pNew->nBuf = nBuf;
|
sl@0
|
360 |
pNew->pFile = (CrashFile *)pFile;
|
sl@0
|
361 |
if( zBuf ){
|
sl@0
|
362 |
pNew->zBuf = (u8 *)&pNew[1];
|
sl@0
|
363 |
memcpy(pNew->zBuf, zBuf, nBuf);
|
sl@0
|
364 |
}
|
sl@0
|
365 |
|
sl@0
|
366 |
if( g.pWriteList ){
|
sl@0
|
367 |
assert(g.pWriteListEnd);
|
sl@0
|
368 |
g.pWriteListEnd->pNext = pNew;
|
sl@0
|
369 |
}else{
|
sl@0
|
370 |
g.pWriteList = pNew;
|
sl@0
|
371 |
}
|
sl@0
|
372 |
g.pWriteListEnd = pNew;
|
sl@0
|
373 |
|
sl@0
|
374 |
return SQLITE_OK;
|
sl@0
|
375 |
}
|
sl@0
|
376 |
|
sl@0
|
377 |
/*
|
sl@0
|
378 |
** Close a crash-file.
|
sl@0
|
379 |
*/
|
sl@0
|
380 |
static int cfClose(sqlite3_file *pFile){
|
sl@0
|
381 |
CrashFile *pCrash = (CrashFile *)pFile;
|
sl@0
|
382 |
writeListSync(pCrash, 0);
|
sl@0
|
383 |
sqlite3OsClose(pCrash->pRealFile);
|
sl@0
|
384 |
return SQLITE_OK;
|
sl@0
|
385 |
}
|
sl@0
|
386 |
|
sl@0
|
387 |
/*
|
sl@0
|
388 |
** Read data from a crash-file.
|
sl@0
|
389 |
*/
|
sl@0
|
390 |
static int cfRead(
|
sl@0
|
391 |
sqlite3_file *pFile,
|
sl@0
|
392 |
void *zBuf,
|
sl@0
|
393 |
int iAmt,
|
sl@0
|
394 |
sqlite_int64 iOfst
|
sl@0
|
395 |
){
|
sl@0
|
396 |
CrashFile *pCrash = (CrashFile *)pFile;
|
sl@0
|
397 |
|
sl@0
|
398 |
/* Check the file-size to see if this is a short-read */
|
sl@0
|
399 |
if( pCrash->iSize<(iOfst+iAmt) ){
|
sl@0
|
400 |
return SQLITE_IOERR_SHORT_READ;
|
sl@0
|
401 |
}
|
sl@0
|
402 |
|
sl@0
|
403 |
memcpy(zBuf, &pCrash->zData[iOfst], iAmt);
|
sl@0
|
404 |
return SQLITE_OK;
|
sl@0
|
405 |
}
|
sl@0
|
406 |
|
sl@0
|
407 |
/*
|
sl@0
|
408 |
** Write data to a crash-file.
|
sl@0
|
409 |
*/
|
sl@0
|
410 |
static int cfWrite(
|
sl@0
|
411 |
sqlite3_file *pFile,
|
sl@0
|
412 |
const void *zBuf,
|
sl@0
|
413 |
int iAmt,
|
sl@0
|
414 |
sqlite_int64 iOfst
|
sl@0
|
415 |
){
|
sl@0
|
416 |
CrashFile *pCrash = (CrashFile *)pFile;
|
sl@0
|
417 |
if( iAmt+iOfst>pCrash->iSize ){
|
sl@0
|
418 |
pCrash->iSize = iAmt+iOfst;
|
sl@0
|
419 |
}
|
sl@0
|
420 |
while( pCrash->iSize>pCrash->nData ){
|
sl@0
|
421 |
u8 *zNew;
|
sl@0
|
422 |
int nNew = (pCrash->nData*2) + 4096;
|
sl@0
|
423 |
zNew = crash_realloc(pCrash->zData, nNew);
|
sl@0
|
424 |
if( !zNew ){
|
sl@0
|
425 |
return SQLITE_NOMEM;
|
sl@0
|
426 |
}
|
sl@0
|
427 |
memset(&zNew[pCrash->nData], 0, nNew-pCrash->nData);
|
sl@0
|
428 |
pCrash->nData = nNew;
|
sl@0
|
429 |
pCrash->zData = zNew;
|
sl@0
|
430 |
}
|
sl@0
|
431 |
memcpy(&pCrash->zData[iOfst], zBuf, iAmt);
|
sl@0
|
432 |
return writeListAppend(pFile, iOfst, zBuf, iAmt);
|
sl@0
|
433 |
}
|
sl@0
|
434 |
|
sl@0
|
435 |
/*
|
sl@0
|
436 |
** Truncate a crash-file.
|
sl@0
|
437 |
*/
|
sl@0
|
438 |
static int cfTruncate(sqlite3_file *pFile, sqlite_int64 size){
|
sl@0
|
439 |
CrashFile *pCrash = (CrashFile *)pFile;
|
sl@0
|
440 |
assert(size>=0);
|
sl@0
|
441 |
if( pCrash->iSize>size ){
|
sl@0
|
442 |
pCrash->iSize = size;
|
sl@0
|
443 |
}
|
sl@0
|
444 |
return writeListAppend(pFile, size, 0, 0);
|
sl@0
|
445 |
}
|
sl@0
|
446 |
|
sl@0
|
447 |
/*
|
sl@0
|
448 |
** Sync a crash-file.
|
sl@0
|
449 |
*/
|
sl@0
|
450 |
static int cfSync(sqlite3_file *pFile, int flags){
|
sl@0
|
451 |
CrashFile *pCrash = (CrashFile *)pFile;
|
sl@0
|
452 |
int isCrash = 0;
|
sl@0
|
453 |
|
sl@0
|
454 |
const char *zName = pCrash->zName;
|
sl@0
|
455 |
const char *zCrashFile = g.zCrashFile;
|
sl@0
|
456 |
int nName = strlen(zName);
|
sl@0
|
457 |
int nCrashFile = strlen(zCrashFile);
|
sl@0
|
458 |
|
sl@0
|
459 |
if( nCrashFile>0 && zCrashFile[nCrashFile-1]=='*' ){
|
sl@0
|
460 |
nCrashFile--;
|
sl@0
|
461 |
if( nName>nCrashFile ) nName = nCrashFile;
|
sl@0
|
462 |
}
|
sl@0
|
463 |
|
sl@0
|
464 |
if( nName==nCrashFile && 0==memcmp(zName, zCrashFile, nName) ){
|
sl@0
|
465 |
if( (--g.iCrash)==0 ) isCrash = 1;
|
sl@0
|
466 |
}
|
sl@0
|
467 |
|
sl@0
|
468 |
return writeListSync(pCrash, isCrash);
|
sl@0
|
469 |
}
|
sl@0
|
470 |
|
sl@0
|
471 |
/*
|
sl@0
|
472 |
** Return the current file-size of the crash-file.
|
sl@0
|
473 |
*/
|
sl@0
|
474 |
static int cfFileSize(sqlite3_file *pFile, sqlite_int64 *pSize){
|
sl@0
|
475 |
CrashFile *pCrash = (CrashFile *)pFile;
|
sl@0
|
476 |
*pSize = (i64)pCrash->iSize;
|
sl@0
|
477 |
return SQLITE_OK;
|
sl@0
|
478 |
}
|
sl@0
|
479 |
|
sl@0
|
480 |
/*
|
sl@0
|
481 |
** Calls related to file-locks are passed on to the real file handle.
|
sl@0
|
482 |
*/
|
sl@0
|
483 |
static int cfLock(sqlite3_file *pFile, int eLock){
|
sl@0
|
484 |
return sqlite3OsLock(((CrashFile *)pFile)->pRealFile, eLock);
|
sl@0
|
485 |
}
|
sl@0
|
486 |
static int cfUnlock(sqlite3_file *pFile, int eLock){
|
sl@0
|
487 |
return sqlite3OsUnlock(((CrashFile *)pFile)->pRealFile, eLock);
|
sl@0
|
488 |
}
|
sl@0
|
489 |
static int cfCheckReservedLock(sqlite3_file *pFile, int *pResOut){
|
sl@0
|
490 |
return sqlite3OsCheckReservedLock(((CrashFile *)pFile)->pRealFile, pResOut);
|
sl@0
|
491 |
}
|
sl@0
|
492 |
static int cfFileControl(sqlite3_file *pFile, int op, void *pArg){
|
sl@0
|
493 |
return sqlite3OsFileControl(((CrashFile *)pFile)->pRealFile, op, pArg);
|
sl@0
|
494 |
}
|
sl@0
|
495 |
|
sl@0
|
496 |
/*
|
sl@0
|
497 |
** The xSectorSize() and xDeviceCharacteristics() functions return
|
sl@0
|
498 |
** the global values configured by the [sqlite_crashparams] tcl
|
sl@0
|
499 |
* interface.
|
sl@0
|
500 |
*/
|
sl@0
|
501 |
static int cfSectorSize(sqlite3_file *pFile){
|
sl@0
|
502 |
return g.iSectorSize;
|
sl@0
|
503 |
}
|
sl@0
|
504 |
static int cfDeviceCharacteristics(sqlite3_file *pFile){
|
sl@0
|
505 |
return g.iDeviceCharacteristics;
|
sl@0
|
506 |
}
|
sl@0
|
507 |
|
sl@0
|
508 |
static const sqlite3_io_methods CrashFileVtab = {
|
sl@0
|
509 |
1, /* iVersion */
|
sl@0
|
510 |
cfClose, /* xClose */
|
sl@0
|
511 |
cfRead, /* xRead */
|
sl@0
|
512 |
cfWrite, /* xWrite */
|
sl@0
|
513 |
cfTruncate, /* xTruncate */
|
sl@0
|
514 |
cfSync, /* xSync */
|
sl@0
|
515 |
cfFileSize, /* xFileSize */
|
sl@0
|
516 |
cfLock, /* xLock */
|
sl@0
|
517 |
cfUnlock, /* xUnlock */
|
sl@0
|
518 |
cfCheckReservedLock, /* xCheckReservedLock */
|
sl@0
|
519 |
cfFileControl, /* xFileControl */
|
sl@0
|
520 |
cfSectorSize, /* xSectorSize */
|
sl@0
|
521 |
cfDeviceCharacteristics /* xDeviceCharacteristics */
|
sl@0
|
522 |
};
|
sl@0
|
523 |
|
sl@0
|
524 |
/*
|
sl@0
|
525 |
** Application data for the crash VFS
|
sl@0
|
526 |
*/
|
sl@0
|
527 |
struct crashAppData {
|
sl@0
|
528 |
sqlite3_vfs *pOrig; /* Wrapped vfs structure */
|
sl@0
|
529 |
};
|
sl@0
|
530 |
|
sl@0
|
531 |
/*
|
sl@0
|
532 |
** Open a crash-file file handle.
|
sl@0
|
533 |
**
|
sl@0
|
534 |
** The caller will have allocated pVfs->szOsFile bytes of space
|
sl@0
|
535 |
** at pFile. This file uses this space for the CrashFile structure
|
sl@0
|
536 |
** and allocates space for the "real" file structure using
|
sl@0
|
537 |
** sqlite3_malloc(). The assumption here is (pVfs->szOsFile) is
|
sl@0
|
538 |
** equal or greater than sizeof(CrashFile).
|
sl@0
|
539 |
*/
|
sl@0
|
540 |
static int cfOpen(
|
sl@0
|
541 |
sqlite3_vfs *pCfVfs,
|
sl@0
|
542 |
const char *zName,
|
sl@0
|
543 |
sqlite3_file *pFile,
|
sl@0
|
544 |
int flags,
|
sl@0
|
545 |
int *pOutFlags
|
sl@0
|
546 |
){
|
sl@0
|
547 |
sqlite3_vfs *pVfs = (sqlite3_vfs *)pCfVfs->pAppData;
|
sl@0
|
548 |
int rc;
|
sl@0
|
549 |
CrashFile *pWrapper = (CrashFile *)pFile;
|
sl@0
|
550 |
sqlite3_file *pReal = (sqlite3_file*)&pWrapper[1];
|
sl@0
|
551 |
|
sl@0
|
552 |
memset(pWrapper, 0, sizeof(CrashFile));
|
sl@0
|
553 |
rc = sqlite3OsOpen(pVfs, zName, pReal, flags, pOutFlags);
|
sl@0
|
554 |
|
sl@0
|
555 |
if( rc==SQLITE_OK ){
|
sl@0
|
556 |
i64 iSize;
|
sl@0
|
557 |
pWrapper->pMethod = &CrashFileVtab;
|
sl@0
|
558 |
pWrapper->zName = (char *)zName;
|
sl@0
|
559 |
pWrapper->pRealFile = pReal;
|
sl@0
|
560 |
rc = sqlite3OsFileSize(pReal, &iSize);
|
sl@0
|
561 |
pWrapper->iSize = (int)iSize;
|
sl@0
|
562 |
}
|
sl@0
|
563 |
if( rc==SQLITE_OK ){
|
sl@0
|
564 |
pWrapper->nData = (4096 + pWrapper->iSize);
|
sl@0
|
565 |
pWrapper->zData = crash_malloc(pWrapper->nData);
|
sl@0
|
566 |
if( pWrapper->zData ){
|
sl@0
|
567 |
memset(pWrapper->zData, 0, pWrapper->nData);
|
sl@0
|
568 |
rc = sqlite3OsRead(pReal, pWrapper->zData, pWrapper->iSize, 0);
|
sl@0
|
569 |
}else{
|
sl@0
|
570 |
rc = SQLITE_NOMEM;
|
sl@0
|
571 |
}
|
sl@0
|
572 |
}
|
sl@0
|
573 |
if( rc!=SQLITE_OK && pWrapper->pMethod ){
|
sl@0
|
574 |
sqlite3OsClose(pFile);
|
sl@0
|
575 |
}
|
sl@0
|
576 |
return rc;
|
sl@0
|
577 |
}
|
sl@0
|
578 |
|
sl@0
|
579 |
static int cfDelete(sqlite3_vfs *pCfVfs, const char *zPath, int dirSync){
|
sl@0
|
580 |
sqlite3_vfs *pVfs = (sqlite3_vfs *)pCfVfs->pAppData;
|
sl@0
|
581 |
return pVfs->xDelete(pVfs, zPath, dirSync);
|
sl@0
|
582 |
}
|
sl@0
|
583 |
static int cfAccess(
|
sl@0
|
584 |
sqlite3_vfs *pCfVfs,
|
sl@0
|
585 |
const char *zPath,
|
sl@0
|
586 |
int flags,
|
sl@0
|
587 |
int *pResOut
|
sl@0
|
588 |
){
|
sl@0
|
589 |
sqlite3_vfs *pVfs = (sqlite3_vfs *)pCfVfs->pAppData;
|
sl@0
|
590 |
return pVfs->xAccess(pVfs, zPath, flags, pResOut);
|
sl@0
|
591 |
}
|
sl@0
|
592 |
static int cfFullPathname(
|
sl@0
|
593 |
sqlite3_vfs *pCfVfs,
|
sl@0
|
594 |
const char *zPath,
|
sl@0
|
595 |
int nPathOut,
|
sl@0
|
596 |
char *zPathOut
|
sl@0
|
597 |
){
|
sl@0
|
598 |
sqlite3_vfs *pVfs = (sqlite3_vfs *)pCfVfs->pAppData;
|
sl@0
|
599 |
return pVfs->xFullPathname(pVfs, zPath, nPathOut, zPathOut);
|
sl@0
|
600 |
}
|
sl@0
|
601 |
static void *cfDlOpen(sqlite3_vfs *pCfVfs, const char *zPath){
|
sl@0
|
602 |
sqlite3_vfs *pVfs = (sqlite3_vfs *)pCfVfs->pAppData;
|
sl@0
|
603 |
return pVfs->xDlOpen(pVfs, zPath);
|
sl@0
|
604 |
}
|
sl@0
|
605 |
static void cfDlError(sqlite3_vfs *pCfVfs, int nByte, char *zErrMsg){
|
sl@0
|
606 |
sqlite3_vfs *pVfs = (sqlite3_vfs *)pCfVfs->pAppData;
|
sl@0
|
607 |
pVfs->xDlError(pVfs, nByte, zErrMsg);
|
sl@0
|
608 |
}
|
sl@0
|
609 |
static void *cfDlSym(sqlite3_vfs *pCfVfs, void *pHandle, const char *zSymbol){
|
sl@0
|
610 |
sqlite3_vfs *pVfs = (sqlite3_vfs *)pCfVfs->pAppData;
|
sl@0
|
611 |
return pVfs->xDlSym(pVfs, pHandle, zSymbol);
|
sl@0
|
612 |
}
|
sl@0
|
613 |
static void cfDlClose(sqlite3_vfs *pCfVfs, void *pHandle){
|
sl@0
|
614 |
sqlite3_vfs *pVfs = (sqlite3_vfs *)pCfVfs->pAppData;
|
sl@0
|
615 |
pVfs->xDlClose(pVfs, pHandle);
|
sl@0
|
616 |
}
|
sl@0
|
617 |
static int cfRandomness(sqlite3_vfs *pCfVfs, int nByte, char *zBufOut){
|
sl@0
|
618 |
sqlite3_vfs *pVfs = (sqlite3_vfs *)pCfVfs->pAppData;
|
sl@0
|
619 |
return pVfs->xRandomness(pVfs, nByte, zBufOut);
|
sl@0
|
620 |
}
|
sl@0
|
621 |
static int cfSleep(sqlite3_vfs *pCfVfs, int nMicro){
|
sl@0
|
622 |
sqlite3_vfs *pVfs = (sqlite3_vfs *)pCfVfs->pAppData;
|
sl@0
|
623 |
return pVfs->xSleep(pVfs, nMicro);
|
sl@0
|
624 |
}
|
sl@0
|
625 |
static int cfCurrentTime(sqlite3_vfs *pCfVfs, double *pTimeOut){
|
sl@0
|
626 |
sqlite3_vfs *pVfs = (sqlite3_vfs *)pCfVfs->pAppData;
|
sl@0
|
627 |
return pVfs->xCurrentTime(pVfs, pTimeOut);
|
sl@0
|
628 |
}
|
sl@0
|
629 |
|
sl@0
|
630 |
static int processDevSymArgs(
|
sl@0
|
631 |
Tcl_Interp *interp,
|
sl@0
|
632 |
int objc,
|
sl@0
|
633 |
Tcl_Obj *CONST objv[],
|
sl@0
|
634 |
int *piDeviceChar,
|
sl@0
|
635 |
int *piSectorSize
|
sl@0
|
636 |
){
|
sl@0
|
637 |
struct DeviceFlag {
|
sl@0
|
638 |
char *zName;
|
sl@0
|
639 |
int iValue;
|
sl@0
|
640 |
} aFlag[] = {
|
sl@0
|
641 |
{ "atomic", SQLITE_IOCAP_ATOMIC },
|
sl@0
|
642 |
{ "atomic512", SQLITE_IOCAP_ATOMIC512 },
|
sl@0
|
643 |
{ "atomic1k", SQLITE_IOCAP_ATOMIC1K },
|
sl@0
|
644 |
{ "atomic2k", SQLITE_IOCAP_ATOMIC2K },
|
sl@0
|
645 |
{ "atomic4k", SQLITE_IOCAP_ATOMIC4K },
|
sl@0
|
646 |
{ "atomic8k", SQLITE_IOCAP_ATOMIC8K },
|
sl@0
|
647 |
{ "atomic16k", SQLITE_IOCAP_ATOMIC16K },
|
sl@0
|
648 |
{ "atomic32k", SQLITE_IOCAP_ATOMIC32K },
|
sl@0
|
649 |
{ "atomic64k", SQLITE_IOCAP_ATOMIC64K },
|
sl@0
|
650 |
{ "sequential", SQLITE_IOCAP_SEQUENTIAL },
|
sl@0
|
651 |
{ "safe_append", SQLITE_IOCAP_SAFE_APPEND },
|
sl@0
|
652 |
{ 0, 0 }
|
sl@0
|
653 |
};
|
sl@0
|
654 |
|
sl@0
|
655 |
int i;
|
sl@0
|
656 |
int iDc = 0;
|
sl@0
|
657 |
int iSectorSize = 0;
|
sl@0
|
658 |
int setSectorsize = 0;
|
sl@0
|
659 |
int setDeviceChar = 0;
|
sl@0
|
660 |
|
sl@0
|
661 |
for(i=0; i<objc; i+=2){
|
sl@0
|
662 |
int nOpt;
|
sl@0
|
663 |
char *zOpt = Tcl_GetStringFromObj(objv[i], &nOpt);
|
sl@0
|
664 |
|
sl@0
|
665 |
if( (nOpt>11 || nOpt<2 || strncmp("-sectorsize", zOpt, nOpt))
|
sl@0
|
666 |
&& (nOpt>16 || nOpt<2 || strncmp("-characteristics", zOpt, nOpt))
|
sl@0
|
667 |
){
|
sl@0
|
668 |
Tcl_AppendResult(interp,
|
sl@0
|
669 |
"Bad option: \"", zOpt,
|
sl@0
|
670 |
"\" - must be \"-characteristics\" or \"-sectorsize\"", 0
|
sl@0
|
671 |
);
|
sl@0
|
672 |
return TCL_ERROR;
|
sl@0
|
673 |
}
|
sl@0
|
674 |
if( i==objc-1 ){
|
sl@0
|
675 |
Tcl_AppendResult(interp, "Option requires an argument: \"", zOpt, "\"",0);
|
sl@0
|
676 |
return TCL_ERROR;
|
sl@0
|
677 |
}
|
sl@0
|
678 |
|
sl@0
|
679 |
if( zOpt[1]=='s' ){
|
sl@0
|
680 |
if( Tcl_GetIntFromObj(interp, objv[i+1], &iSectorSize) ){
|
sl@0
|
681 |
return TCL_ERROR;
|
sl@0
|
682 |
}
|
sl@0
|
683 |
setSectorsize = 1;
|
sl@0
|
684 |
}else{
|
sl@0
|
685 |
int j;
|
sl@0
|
686 |
Tcl_Obj **apObj;
|
sl@0
|
687 |
int nObj;
|
sl@0
|
688 |
if( Tcl_ListObjGetElements(interp, objv[i+1], &nObj, &apObj) ){
|
sl@0
|
689 |
return TCL_ERROR;
|
sl@0
|
690 |
}
|
sl@0
|
691 |
for(j=0; j<nObj; j++){
|
sl@0
|
692 |
int rc;
|
sl@0
|
693 |
int iChoice;
|
sl@0
|
694 |
Tcl_Obj *pFlag = Tcl_DuplicateObj(apObj[j]);
|
sl@0
|
695 |
Tcl_IncrRefCount(pFlag);
|
sl@0
|
696 |
Tcl_UtfToLower(Tcl_GetString(pFlag));
|
sl@0
|
697 |
|
sl@0
|
698 |
rc = Tcl_GetIndexFromObjStruct(
|
sl@0
|
699 |
interp, pFlag, aFlag, sizeof(aFlag[0]), "no such flag", 0, &iChoice
|
sl@0
|
700 |
);
|
sl@0
|
701 |
Tcl_DecrRefCount(pFlag);
|
sl@0
|
702 |
if( rc ){
|
sl@0
|
703 |
return TCL_ERROR;
|
sl@0
|
704 |
}
|
sl@0
|
705 |
|
sl@0
|
706 |
iDc |= aFlag[iChoice].iValue;
|
sl@0
|
707 |
}
|
sl@0
|
708 |
setDeviceChar = 1;
|
sl@0
|
709 |
}
|
sl@0
|
710 |
}
|
sl@0
|
711 |
|
sl@0
|
712 |
if( setDeviceChar ){
|
sl@0
|
713 |
*piDeviceChar = iDc;
|
sl@0
|
714 |
}
|
sl@0
|
715 |
if( setSectorsize ){
|
sl@0
|
716 |
*piSectorSize = iSectorSize;
|
sl@0
|
717 |
}
|
sl@0
|
718 |
|
sl@0
|
719 |
return TCL_OK;
|
sl@0
|
720 |
}
|
sl@0
|
721 |
|
sl@0
|
722 |
/*
|
sl@0
|
723 |
** tclcmd: sqlite_crash_enable ENABLE
|
sl@0
|
724 |
**
|
sl@0
|
725 |
** Parameter ENABLE must be a boolean value. If true, then the "crash"
|
sl@0
|
726 |
** vfs is added to the system. If false, it is removed.
|
sl@0
|
727 |
*/
|
sl@0
|
728 |
static int crashEnableCmd(
|
sl@0
|
729 |
void * clientData,
|
sl@0
|
730 |
Tcl_Interp *interp,
|
sl@0
|
731 |
int objc,
|
sl@0
|
732 |
Tcl_Obj *CONST objv[]
|
sl@0
|
733 |
){
|
sl@0
|
734 |
int isEnable;
|
sl@0
|
735 |
static sqlite3_vfs crashVfs = {
|
sl@0
|
736 |
1, /* iVersion */
|
sl@0
|
737 |
0, /* szOsFile */
|
sl@0
|
738 |
0, /* mxPathname */
|
sl@0
|
739 |
0, /* pNext */
|
sl@0
|
740 |
"crash", /* zName */
|
sl@0
|
741 |
0, /* pAppData */
|
sl@0
|
742 |
|
sl@0
|
743 |
cfOpen, /* xOpen */
|
sl@0
|
744 |
cfDelete, /* xDelete */
|
sl@0
|
745 |
cfAccess, /* xAccess */
|
sl@0
|
746 |
cfFullPathname, /* xFullPathname */
|
sl@0
|
747 |
cfDlOpen, /* xDlOpen */
|
sl@0
|
748 |
cfDlError, /* xDlError */
|
sl@0
|
749 |
cfDlSym, /* xDlSym */
|
sl@0
|
750 |
cfDlClose, /* xDlClose */
|
sl@0
|
751 |
cfRandomness, /* xRandomness */
|
sl@0
|
752 |
cfSleep, /* xSleep */
|
sl@0
|
753 |
cfCurrentTime /* xCurrentTime */
|
sl@0
|
754 |
};
|
sl@0
|
755 |
|
sl@0
|
756 |
if( objc!=2 ){
|
sl@0
|
757 |
Tcl_WrongNumArgs(interp, 1, objv, "ENABLE");
|
sl@0
|
758 |
return TCL_ERROR;
|
sl@0
|
759 |
}
|
sl@0
|
760 |
|
sl@0
|
761 |
if( Tcl_GetBooleanFromObj(interp, objv[1], &isEnable) ){
|
sl@0
|
762 |
return TCL_ERROR;
|
sl@0
|
763 |
}
|
sl@0
|
764 |
|
sl@0
|
765 |
if( (isEnable && crashVfs.pAppData) || (!isEnable && !crashVfs.pAppData) ){
|
sl@0
|
766 |
return TCL_OK;
|
sl@0
|
767 |
}
|
sl@0
|
768 |
|
sl@0
|
769 |
if( crashVfs.pAppData==0 ){
|
sl@0
|
770 |
sqlite3_vfs *pOriginalVfs = sqlite3_vfs_find(0);
|
sl@0
|
771 |
crashVfs.mxPathname = pOriginalVfs->mxPathname;
|
sl@0
|
772 |
crashVfs.pAppData = (void *)pOriginalVfs;
|
sl@0
|
773 |
crashVfs.szOsFile = sizeof(CrashFile) + pOriginalVfs->szOsFile;
|
sl@0
|
774 |
sqlite3_vfs_register(&crashVfs, 0);
|
sl@0
|
775 |
}else{
|
sl@0
|
776 |
crashVfs.pAppData = 0;
|
sl@0
|
777 |
sqlite3_vfs_unregister(&crashVfs);
|
sl@0
|
778 |
}
|
sl@0
|
779 |
|
sl@0
|
780 |
return TCL_OK;
|
sl@0
|
781 |
}
|
sl@0
|
782 |
|
sl@0
|
783 |
/*
|
sl@0
|
784 |
** tclcmd: sqlite_crashparams ?OPTIONS? DELAY CRASHFILE
|
sl@0
|
785 |
**
|
sl@0
|
786 |
** This procedure implements a TCL command that enables crash testing
|
sl@0
|
787 |
** in testfixture. Once enabled, crash testing cannot be disabled.
|
sl@0
|
788 |
**
|
sl@0
|
789 |
** Available options are "-characteristics" and "-sectorsize". Both require
|
sl@0
|
790 |
** an argument. For -sectorsize, this is the simulated sector size in
|
sl@0
|
791 |
** bytes. For -characteristics, the argument must be a list of io-capability
|
sl@0
|
792 |
** flags to simulate. Valid flags are "atomic", "atomic512", "atomic1K",
|
sl@0
|
793 |
** "atomic2K", "atomic4K", "atomic8K", "atomic16K", "atomic32K",
|
sl@0
|
794 |
** "atomic64K", "sequential" and "safe_append".
|
sl@0
|
795 |
**
|
sl@0
|
796 |
** Example:
|
sl@0
|
797 |
**
|
sl@0
|
798 |
** sqlite_crashparams -sect 1024 -char {atomic sequential} ./test.db 1
|
sl@0
|
799 |
**
|
sl@0
|
800 |
*/
|
sl@0
|
801 |
static int crashParamsObjCmd(
|
sl@0
|
802 |
void * clientData,
|
sl@0
|
803 |
Tcl_Interp *interp,
|
sl@0
|
804 |
int objc,
|
sl@0
|
805 |
Tcl_Obj *CONST objv[]
|
sl@0
|
806 |
){
|
sl@0
|
807 |
int iDelay;
|
sl@0
|
808 |
const char *zCrashFile;
|
sl@0
|
809 |
int nCrashFile, iDc, iSectorSize;
|
sl@0
|
810 |
|
sl@0
|
811 |
iDc = -1;
|
sl@0
|
812 |
iSectorSize = -1;
|
sl@0
|
813 |
|
sl@0
|
814 |
if( objc<3 ){
|
sl@0
|
815 |
Tcl_WrongNumArgs(interp, 1, objv, "?OPTIONS? DELAY CRASHFILE");
|
sl@0
|
816 |
goto error;
|
sl@0
|
817 |
}
|
sl@0
|
818 |
|
sl@0
|
819 |
zCrashFile = Tcl_GetStringFromObj(objv[objc-1], &nCrashFile);
|
sl@0
|
820 |
if( nCrashFile>=sizeof(g.zCrashFile) ){
|
sl@0
|
821 |
Tcl_AppendResult(interp, "Filename is too long: \"", zCrashFile, "\"", 0);
|
sl@0
|
822 |
goto error;
|
sl@0
|
823 |
}
|
sl@0
|
824 |
if( Tcl_GetIntFromObj(interp, objv[objc-2], &iDelay) ){
|
sl@0
|
825 |
goto error;
|
sl@0
|
826 |
}
|
sl@0
|
827 |
|
sl@0
|
828 |
if( processDevSymArgs(interp, objc-3, &objv[1], &iDc, &iSectorSize) ){
|
sl@0
|
829 |
return TCL_ERROR;
|
sl@0
|
830 |
}
|
sl@0
|
831 |
|
sl@0
|
832 |
if( iDc>=0 ){
|
sl@0
|
833 |
g.iDeviceCharacteristics = iDc;
|
sl@0
|
834 |
}
|
sl@0
|
835 |
if( iSectorSize>=0 ){
|
sl@0
|
836 |
g.iSectorSize = iSectorSize;
|
sl@0
|
837 |
}
|
sl@0
|
838 |
|
sl@0
|
839 |
g.iCrash = iDelay;
|
sl@0
|
840 |
memcpy(g.zCrashFile, zCrashFile, nCrashFile+1);
|
sl@0
|
841 |
sqlite3CrashTestEnable = 1;
|
sl@0
|
842 |
return TCL_OK;
|
sl@0
|
843 |
|
sl@0
|
844 |
error:
|
sl@0
|
845 |
return TCL_ERROR;
|
sl@0
|
846 |
}
|
sl@0
|
847 |
|
sl@0
|
848 |
static int devSymObjCmd(
|
sl@0
|
849 |
void * clientData,
|
sl@0
|
850 |
Tcl_Interp *interp,
|
sl@0
|
851 |
int objc,
|
sl@0
|
852 |
Tcl_Obj *CONST objv[]
|
sl@0
|
853 |
){
|
sl@0
|
854 |
void devsym_register(int iDeviceChar, int iSectorSize);
|
sl@0
|
855 |
|
sl@0
|
856 |
int iDc = -1;
|
sl@0
|
857 |
int iSectorSize = -1;
|
sl@0
|
858 |
|
sl@0
|
859 |
if( processDevSymArgs(interp, objc-1, &objv[1], &iDc, &iSectorSize) ){
|
sl@0
|
860 |
return TCL_ERROR;
|
sl@0
|
861 |
}
|
sl@0
|
862 |
devsym_register(iDc, iSectorSize);
|
sl@0
|
863 |
|
sl@0
|
864 |
return TCL_OK;
|
sl@0
|
865 |
}
|
sl@0
|
866 |
|
sl@0
|
867 |
#endif /* SQLITE_OMIT_DISKIO */
|
sl@0
|
868 |
|
sl@0
|
869 |
/*
|
sl@0
|
870 |
** This procedure registers the TCL procedures defined in this file.
|
sl@0
|
871 |
*/
|
sl@0
|
872 |
int Sqlitetest6_Init(Tcl_Interp *interp){
|
sl@0
|
873 |
#ifndef SQLITE_OMIT_DISKIO
|
sl@0
|
874 |
Tcl_CreateObjCommand(interp, "sqlite3_crash_enable", crashEnableCmd, 0, 0);
|
sl@0
|
875 |
Tcl_CreateObjCommand(interp, "sqlite3_crashparams", crashParamsObjCmd, 0, 0);
|
sl@0
|
876 |
Tcl_CreateObjCommand(interp, "sqlite3_simulate_device", devSymObjCmd, 0, 0);
|
sl@0
|
877 |
#endif
|
sl@0
|
878 |
return TCL_OK;
|
sl@0
|
879 |
}
|
sl@0
|
880 |
|
sl@0
|
881 |
#endif /* SQLITE_TEST */
|