1 /* 2 ** 2016-09-07 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 implements an in-memory VFS. A database is held as a contiguous 14 ** block of memory. 15 ** 16 ** This file also implements interface sqlite3_serialize() and 17 ** sqlite3_deserialize(). 18 */ 19 #include "sqliteInt.h" 20 #ifdef SQLITE_ENABLE_DESERIALIZE 21 22 /* 23 ** Forward declaration of objects used by this utility 24 */ 25 typedef struct sqlite3_vfs MemVfs; 26 typedef struct MemFile MemFile; 27 28 /* Access to a lower-level VFS that (might) implement dynamic loading, 29 ** access to randomness, etc. 30 */ 31 #define ORIGVFS(p) ((sqlite3_vfs*)((p)->pAppData)) 32 33 /* An open file */ 34 struct MemFile { 35 sqlite3_file base; /* IO methods */ 36 sqlite3_int64 sz; /* Size of the file */ 37 sqlite3_int64 szAlloc; /* Space allocated to aData */ 38 sqlite3_int64 szMax; /* Maximum allowed size of the file */ 39 unsigned char *aData; /* content of the file */ 40 int nMmap; /* Number of memory mapped pages */ 41 unsigned mFlags; /* Flags */ 42 int eLock; /* Most recent lock against this file */ 43 }; 44 45 /* The default maximum size of an in-memory database */ 46 #ifndef SQLITE_MEMDB_DEFAULT_MAXSIZE 47 # define SQLITE_MEMDB_DEFAULT_MAXSIZE 1073741824 48 #endif 49 50 /* 51 ** Methods for MemFile 52 */ 53 static int memdbClose(sqlite3_file*); 54 static int memdbRead(sqlite3_file*, void*, int iAmt, sqlite3_int64 iOfst); 55 static int memdbWrite(sqlite3_file*,const void*,int iAmt, sqlite3_int64 iOfst); 56 static int memdbTruncate(sqlite3_file*, sqlite3_int64 size); 57 static int memdbSync(sqlite3_file*, int flags); 58 static int memdbFileSize(sqlite3_file*, sqlite3_int64 *pSize); 59 static int memdbLock(sqlite3_file*, int); 60 /* static int memdbCheckReservedLock(sqlite3_file*, int *pResOut);// not used */ 61 static int memdbFileControl(sqlite3_file*, int op, void *pArg); 62 /* static int memdbSectorSize(sqlite3_file*); // not used */ 63 static int memdbDeviceCharacteristics(sqlite3_file*); 64 static int memdbFetch(sqlite3_file*, sqlite3_int64 iOfst, int iAmt, void **pp); 65 static int memdbUnfetch(sqlite3_file*, sqlite3_int64 iOfst, void *p); 66 67 /* 68 ** Methods for MemVfs 69 */ 70 static int memdbOpen(sqlite3_vfs*, const char *, sqlite3_file*, int , int *); 71 /* static int memdbDelete(sqlite3_vfs*, const char *zName, int syncDir); */ 72 static int memdbAccess(sqlite3_vfs*, const char *zName, int flags, int *); 73 static int memdbFullPathname(sqlite3_vfs*, const char *zName, int, char *zOut); 74 static void *memdbDlOpen(sqlite3_vfs*, const char *zFilename); 75 static void memdbDlError(sqlite3_vfs*, int nByte, char *zErrMsg); 76 static void (*memdbDlSym(sqlite3_vfs *pVfs, void *p, const char*zSym))(void); 77 static void memdbDlClose(sqlite3_vfs*, void*); 78 static int memdbRandomness(sqlite3_vfs*, int nByte, char *zOut); 79 static int memdbSleep(sqlite3_vfs*, int microseconds); 80 /* static int memdbCurrentTime(sqlite3_vfs*, double*); */ 81 static int memdbGetLastError(sqlite3_vfs*, int, char *); 82 static int memdbCurrentTimeInt64(sqlite3_vfs*, sqlite3_int64*); 83 84 static sqlite3_vfs memdb_vfs = { 85 2, /* iVersion */ 86 0, /* szOsFile (set when registered) */ 87 1024, /* mxPathname */ 88 0, /* pNext */ 89 "memdb", /* zName */ 90 0, /* pAppData (set when registered) */ 91 memdbOpen, /* xOpen */ 92 0, /* memdbDelete, */ /* xDelete */ 93 memdbAccess, /* xAccess */ 94 memdbFullPathname, /* xFullPathname */ 95 memdbDlOpen, /* xDlOpen */ 96 memdbDlError, /* xDlError */ 97 memdbDlSym, /* xDlSym */ 98 memdbDlClose, /* xDlClose */ 99 memdbRandomness, /* xRandomness */ 100 memdbSleep, /* xSleep */ 101 0, /* memdbCurrentTime, */ /* xCurrentTime */ 102 memdbGetLastError, /* xGetLastError */ 103 memdbCurrentTimeInt64 /* xCurrentTimeInt64 */ 104 }; 105 106 static const sqlite3_io_methods memdb_io_methods = { 107 3, /* iVersion */ 108 memdbClose, /* xClose */ 109 memdbRead, /* xRead */ 110 memdbWrite, /* xWrite */ 111 memdbTruncate, /* xTruncate */ 112 memdbSync, /* xSync */ 113 memdbFileSize, /* xFileSize */ 114 memdbLock, /* xLock */ 115 memdbLock, /* xUnlock - same as xLock in this case */ 116 0, /* memdbCheckReservedLock, */ /* xCheckReservedLock */ 117 memdbFileControl, /* xFileControl */ 118 0, /* memdbSectorSize,*/ /* xSectorSize */ 119 memdbDeviceCharacteristics, /* xDeviceCharacteristics */ 120 0, /* xShmMap */ 121 0, /* xShmLock */ 122 0, /* xShmBarrier */ 123 0, /* xShmUnmap */ 124 memdbFetch, /* xFetch */ 125 memdbUnfetch /* xUnfetch */ 126 }; 127 128 129 130 /* 131 ** Close an memdb-file. 132 ** 133 ** The pData pointer is owned by the application, so there is nothing 134 ** to free. 135 */ 136 static int memdbClose(sqlite3_file *pFile){ 137 MemFile *p = (MemFile *)pFile; 138 if( p->mFlags & SQLITE_DESERIALIZE_FREEONCLOSE ) sqlite3_free(p->aData); 139 return SQLITE_OK; 140 } 141 142 /* 143 ** Read data from an memdb-file. 144 */ 145 static int memdbRead( 146 sqlite3_file *pFile, 147 void *zBuf, 148 int iAmt, 149 sqlite_int64 iOfst 150 ){ 151 MemFile *p = (MemFile *)pFile; 152 if( iOfst+iAmt>p->sz ){ 153 memset(zBuf, 0, iAmt); 154 if( iOfst<p->sz ) memcpy(zBuf, p->aData+iOfst, p->sz - iOfst); 155 return SQLITE_IOERR_SHORT_READ; 156 } 157 memcpy(zBuf, p->aData+iOfst, iAmt); 158 return SQLITE_OK; 159 } 160 161 /* 162 ** Try to enlarge the memory allocation to hold at least sz bytes 163 */ 164 static int memdbEnlarge(MemFile *p, sqlite3_int64 newSz){ 165 unsigned char *pNew; 166 if( (p->mFlags & SQLITE_DESERIALIZE_RESIZEABLE)==0 || p->nMmap>0 ){ 167 return SQLITE_FULL; 168 } 169 if( newSz>p->szMax ){ 170 return SQLITE_FULL; 171 } 172 newSz *= 2; 173 if( newSz>p->szMax ) newSz = p->szMax; 174 pNew = sqlite3_realloc64(p->aData, newSz); 175 if( pNew==0 ) return SQLITE_NOMEM; 176 p->aData = pNew; 177 p->szAlloc = newSz; 178 return SQLITE_OK; 179 } 180 181 /* 182 ** Write data to an memdb-file. 183 */ 184 static int memdbWrite( 185 sqlite3_file *pFile, 186 const void *z, 187 int iAmt, 188 sqlite_int64 iOfst 189 ){ 190 MemFile *p = (MemFile *)pFile; 191 if( NEVER(p->mFlags & SQLITE_DESERIALIZE_READONLY) ) return SQLITE_READONLY; 192 if( iOfst+iAmt>p->sz ){ 193 int rc; 194 if( iOfst+iAmt>p->szAlloc 195 && (rc = memdbEnlarge(p, iOfst+iAmt))!=SQLITE_OK 196 ){ 197 return rc; 198 } 199 if( iOfst>p->sz ) memset(p->aData+p->sz, 0, iOfst-p->sz); 200 p->sz = iOfst+iAmt; 201 } 202 memcpy(p->aData+iOfst, z, iAmt); 203 return SQLITE_OK; 204 } 205 206 /* 207 ** Truncate an memdb-file. 208 ** 209 ** In rollback mode (which is always the case for memdb, as it does not 210 ** support WAL mode) the truncate() method is only used to reduce 211 ** the size of a file, never to increase the size. 212 */ 213 static int memdbTruncate(sqlite3_file *pFile, sqlite_int64 size){ 214 MemFile *p = (MemFile *)pFile; 215 if( NEVER(size>p->sz) ) return SQLITE_FULL; 216 p->sz = size; 217 return SQLITE_OK; 218 } 219 220 /* 221 ** Sync an memdb-file. 222 */ 223 static int memdbSync(sqlite3_file *pFile, int flags){ 224 return SQLITE_OK; 225 } 226 227 /* 228 ** Return the current file-size of an memdb-file. 229 */ 230 static int memdbFileSize(sqlite3_file *pFile, sqlite_int64 *pSize){ 231 MemFile *p = (MemFile *)pFile; 232 *pSize = p->sz; 233 return SQLITE_OK; 234 } 235 236 /* 237 ** Lock an memdb-file. 238 */ 239 static int memdbLock(sqlite3_file *pFile, int eLock){ 240 MemFile *p = (MemFile *)pFile; 241 if( eLock>SQLITE_LOCK_SHARED 242 && (p->mFlags & SQLITE_DESERIALIZE_READONLY)!=0 243 ){ 244 return SQLITE_READONLY; 245 } 246 p->eLock = eLock; 247 return SQLITE_OK; 248 } 249 250 #if 0 /* Never used because memdbAccess() always returns false */ 251 /* 252 ** Check if another file-handle holds a RESERVED lock on an memdb-file. 253 */ 254 static int memdbCheckReservedLock(sqlite3_file *pFile, int *pResOut){ 255 *pResOut = 0; 256 return SQLITE_OK; 257 } 258 #endif 259 260 /* 261 ** File control method. For custom operations on an memdb-file. 262 */ 263 static int memdbFileControl(sqlite3_file *pFile, int op, void *pArg){ 264 MemFile *p = (MemFile *)pFile; 265 int rc = SQLITE_NOTFOUND; 266 if( op==SQLITE_FCNTL_VFSNAME ){ 267 *(char**)pArg = sqlite3_mprintf("memdb(%p,%lld)", p->aData, p->sz); 268 rc = SQLITE_OK; 269 } 270 if( op==SQLITE_FCNTL_SIZE_LIMIT ){ 271 sqlite3_int64 iLimit = *(sqlite3_int64*)pArg; 272 if( iLimit<p->sz ){ 273 if( iLimit<0 ){ 274 iLimit = p->szMax; 275 }else{ 276 iLimit = p->sz; 277 } 278 } 279 p->szMax = iLimit; 280 *(sqlite3_int64*)pArg = iLimit; 281 rc = SQLITE_OK; 282 } 283 return rc; 284 } 285 286 #if 0 /* Not used because of SQLITE_IOCAP_POWERSAFE_OVERWRITE */ 287 /* 288 ** Return the sector-size in bytes for an memdb-file. 289 */ 290 static int memdbSectorSize(sqlite3_file *pFile){ 291 return 1024; 292 } 293 #endif 294 295 /* 296 ** Return the device characteristic flags supported by an memdb-file. 297 */ 298 static int memdbDeviceCharacteristics(sqlite3_file *pFile){ 299 return SQLITE_IOCAP_ATOMIC | 300 SQLITE_IOCAP_POWERSAFE_OVERWRITE | 301 SQLITE_IOCAP_SAFE_APPEND | 302 SQLITE_IOCAP_SEQUENTIAL; 303 } 304 305 /* Fetch a page of a memory-mapped file */ 306 static int memdbFetch( 307 sqlite3_file *pFile, 308 sqlite3_int64 iOfst, 309 int iAmt, 310 void **pp 311 ){ 312 MemFile *p = (MemFile *)pFile; 313 p->nMmap++; 314 *pp = (void*)(p->aData + iOfst); 315 return SQLITE_OK; 316 } 317 318 /* Release a memory-mapped page */ 319 static int memdbUnfetch(sqlite3_file *pFile, sqlite3_int64 iOfst, void *pPage){ 320 MemFile *p = (MemFile *)pFile; 321 p->nMmap--; 322 return SQLITE_OK; 323 } 324 325 /* 326 ** Open an mem file handle. 327 */ 328 static int memdbOpen( 329 sqlite3_vfs *pVfs, 330 const char *zName, 331 sqlite3_file *pFile, 332 int flags, 333 int *pOutFlags 334 ){ 335 MemFile *p = (MemFile*)pFile; 336 if( (flags & SQLITE_OPEN_MAIN_DB)==0 ){ 337 return ORIGVFS(pVfs)->xOpen(ORIGVFS(pVfs), zName, pFile, flags, pOutFlags); 338 } 339 memset(p, 0, sizeof(*p)); 340 p->mFlags = SQLITE_DESERIALIZE_RESIZEABLE | SQLITE_DESERIALIZE_FREEONCLOSE; 341 assert( pOutFlags!=0 ); /* True because flags==SQLITE_OPEN_MAIN_DB */ 342 *pOutFlags = flags | SQLITE_OPEN_MEMORY; 343 p->base.pMethods = &memdb_io_methods; 344 p->szMax = SQLITE_MEMDB_DEFAULT_MAXSIZE; 345 return SQLITE_OK; 346 } 347 348 #if 0 /* Only used to delete rollback journals, master journals, and WAL 349 ** files, none of which exist in memdb. So this routine is never used */ 350 /* 351 ** Delete the file located at zPath. If the dirSync argument is true, 352 ** ensure the file-system modifications are synced to disk before 353 ** returning. 354 */ 355 static int memdbDelete(sqlite3_vfs *pVfs, const char *zPath, int dirSync){ 356 return SQLITE_IOERR_DELETE; 357 } 358 #endif 359 360 /* 361 ** Test for access permissions. Return true if the requested permission 362 ** is available, or false otherwise. 363 ** 364 ** With memdb, no files ever exist on disk. So always return false. 365 */ 366 static int memdbAccess( 367 sqlite3_vfs *pVfs, 368 const char *zPath, 369 int flags, 370 int *pResOut 371 ){ 372 *pResOut = 0; 373 return SQLITE_OK; 374 } 375 376 /* 377 ** Populate buffer zOut with the full canonical pathname corresponding 378 ** to the pathname in zPath. zOut is guaranteed to point to a buffer 379 ** of at least (INST_MAX_PATHNAME+1) bytes. 380 */ 381 static int memdbFullPathname( 382 sqlite3_vfs *pVfs, 383 const char *zPath, 384 int nOut, 385 char *zOut 386 ){ 387 sqlite3_snprintf(nOut, zOut, "%s", zPath); 388 return SQLITE_OK; 389 } 390 391 /* 392 ** Open the dynamic library located at zPath and return a handle. 393 */ 394 static void *memdbDlOpen(sqlite3_vfs *pVfs, const char *zPath){ 395 return ORIGVFS(pVfs)->xDlOpen(ORIGVFS(pVfs), zPath); 396 } 397 398 /* 399 ** Populate the buffer zErrMsg (size nByte bytes) with a human readable 400 ** utf-8 string describing the most recent error encountered associated 401 ** with dynamic libraries. 402 */ 403 static void memdbDlError(sqlite3_vfs *pVfs, int nByte, char *zErrMsg){ 404 ORIGVFS(pVfs)->xDlError(ORIGVFS(pVfs), nByte, zErrMsg); 405 } 406 407 /* 408 ** Return a pointer to the symbol zSymbol in the dynamic library pHandle. 409 */ 410 static void (*memdbDlSym(sqlite3_vfs *pVfs, void *p, const char *zSym))(void){ 411 return ORIGVFS(pVfs)->xDlSym(ORIGVFS(pVfs), p, zSym); 412 } 413 414 /* 415 ** Close the dynamic library handle pHandle. 416 */ 417 static void memdbDlClose(sqlite3_vfs *pVfs, void *pHandle){ 418 ORIGVFS(pVfs)->xDlClose(ORIGVFS(pVfs), pHandle); 419 } 420 421 /* 422 ** Populate the buffer pointed to by zBufOut with nByte bytes of 423 ** random data. 424 */ 425 static int memdbRandomness(sqlite3_vfs *pVfs, int nByte, char *zBufOut){ 426 return ORIGVFS(pVfs)->xRandomness(ORIGVFS(pVfs), nByte, zBufOut); 427 } 428 429 /* 430 ** Sleep for nMicro microseconds. Return the number of microseconds 431 ** actually slept. 432 */ 433 static int memdbSleep(sqlite3_vfs *pVfs, int nMicro){ 434 return ORIGVFS(pVfs)->xSleep(ORIGVFS(pVfs), nMicro); 435 } 436 437 #if 0 /* Never used. Modern cores only call xCurrentTimeInt64() */ 438 /* 439 ** Return the current time as a Julian Day number in *pTimeOut. 440 */ 441 static int memdbCurrentTime(sqlite3_vfs *pVfs, double *pTimeOut){ 442 return ORIGVFS(pVfs)->xCurrentTime(ORIGVFS(pVfs), pTimeOut); 443 } 444 #endif 445 446 static int memdbGetLastError(sqlite3_vfs *pVfs, int a, char *b){ 447 return ORIGVFS(pVfs)->xGetLastError(ORIGVFS(pVfs), a, b); 448 } 449 static int memdbCurrentTimeInt64(sqlite3_vfs *pVfs, sqlite3_int64 *p){ 450 return ORIGVFS(pVfs)->xCurrentTimeInt64(ORIGVFS(pVfs), p); 451 } 452 453 /* 454 ** Translate a database connection pointer and schema name into a 455 ** MemFile pointer. 456 */ 457 static MemFile *memdbFromDbSchema(sqlite3 *db, const char *zSchema){ 458 MemFile *p = 0; 459 int rc = sqlite3_file_control(db, zSchema, SQLITE_FCNTL_FILE_POINTER, &p); 460 if( rc ) return 0; 461 if( p->base.pMethods!=&memdb_io_methods ) return 0; 462 return p; 463 } 464 465 /* 466 ** Return the serialization of a database 467 */ 468 unsigned char *sqlite3_serialize( 469 sqlite3 *db, /* The database connection */ 470 const char *zSchema, /* Which database within the connection */ 471 sqlite3_int64 *piSize, /* Write size here, if not NULL */ 472 unsigned int mFlags /* Maybe SQLITE_SERIALIZE_NOCOPY */ 473 ){ 474 MemFile *p; 475 int iDb; 476 Btree *pBt; 477 sqlite3_int64 sz; 478 int szPage = 0; 479 sqlite3_stmt *pStmt = 0; 480 unsigned char *pOut; 481 char *zSql; 482 int rc; 483 484 #ifdef SQLITE_ENABLE_API_ARMOR 485 if( !sqlite3SafetyCheckOk(db) ){ 486 (void)SQLITE_MISUSE_BKPT; 487 return 0; 488 } 489 #endif 490 491 if( zSchema==0 ) zSchema = db->aDb[0].zDbSName; 492 p = memdbFromDbSchema(db, zSchema); 493 iDb = sqlite3FindDbName(db, zSchema); 494 if( piSize ) *piSize = -1; 495 if( iDb<0 ) return 0; 496 if( p ){ 497 if( piSize ) *piSize = p->sz; 498 if( mFlags & SQLITE_SERIALIZE_NOCOPY ){ 499 pOut = p->aData; 500 }else{ 501 pOut = sqlite3_malloc64( p->sz ); 502 if( pOut ) memcpy(pOut, p->aData, p->sz); 503 } 504 return pOut; 505 } 506 pBt = db->aDb[iDb].pBt; 507 if( pBt==0 ) return 0; 508 szPage = sqlite3BtreeGetPageSize(pBt); 509 zSql = sqlite3_mprintf("PRAGMA \"%w\".page_count", zSchema); 510 rc = zSql ? sqlite3_prepare_v2(db, zSql, -1, &pStmt, 0) : SQLITE_NOMEM; 511 sqlite3_free(zSql); 512 if( rc ) return 0; 513 rc = sqlite3_step(pStmt); 514 if( rc!=SQLITE_ROW ){ 515 pOut = 0; 516 }else{ 517 sz = sqlite3_column_int64(pStmt, 0)*szPage; 518 if( piSize ) *piSize = sz; 519 if( mFlags & SQLITE_SERIALIZE_NOCOPY ){ 520 pOut = 0; 521 }else{ 522 pOut = sqlite3_malloc64( sz ); 523 if( pOut ){ 524 int nPage = sqlite3_column_int(pStmt, 0); 525 Pager *pPager = sqlite3BtreePager(pBt); 526 int pgno; 527 for(pgno=1; pgno<=nPage; pgno++){ 528 DbPage *pPage = 0; 529 unsigned char *pTo = pOut + szPage*(sqlite3_int64)(pgno-1); 530 rc = sqlite3PagerGet(pPager, pgno, (DbPage**)&pPage, 0); 531 if( rc==SQLITE_OK ){ 532 memcpy(pTo, sqlite3PagerGetData(pPage), szPage); 533 }else{ 534 memset(pTo, 0, szPage); 535 } 536 sqlite3PagerUnref(pPage); 537 } 538 } 539 } 540 } 541 sqlite3_finalize(pStmt); 542 return pOut; 543 } 544 545 /* Convert zSchema to a MemDB and initialize its content. 546 */ 547 int sqlite3_deserialize( 548 sqlite3 *db, /* The database connection */ 549 const char *zSchema, /* Which DB to reopen with the deserialization */ 550 unsigned char *pData, /* The serialized database content */ 551 sqlite3_int64 szDb, /* Number bytes in the deserialization */ 552 sqlite3_int64 szBuf, /* Total size of buffer pData[] */ 553 unsigned mFlags /* Zero or more SQLITE_DESERIALIZE_* flags */ 554 ){ 555 MemFile *p; 556 char *zSql; 557 sqlite3_stmt *pStmt = 0; 558 int rc; 559 int iDb; 560 561 #ifdef SQLITE_ENABLE_API_ARMOR 562 if( !sqlite3SafetyCheckOk(db) ){ 563 return SQLITE_MISUSE_BKPT; 564 } 565 if( szDb<0 ) return SQLITE_MISUSE_BKPT; 566 if( szBuf<0 ) return SQLITE_MISUSE_BKPT; 567 #endif 568 569 sqlite3_mutex_enter(db->mutex); 570 if( zSchema==0 ) zSchema = db->aDb[0].zDbSName; 571 iDb = sqlite3FindDbName(db, zSchema); 572 if( iDb<0 ){ 573 rc = SQLITE_ERROR; 574 goto end_deserialize; 575 } 576 zSql = sqlite3_mprintf("ATTACH x AS %Q", zSchema); 577 rc = sqlite3_prepare_v2(db, zSql, -1, &pStmt, 0); 578 sqlite3_free(zSql); 579 if( rc ) goto end_deserialize; 580 db->init.iDb = (u8)iDb; 581 db->init.reopenMemdb = 1; 582 rc = sqlite3_step(pStmt); 583 db->init.reopenMemdb = 0; 584 if( rc!=SQLITE_DONE ){ 585 rc = SQLITE_ERROR; 586 goto end_deserialize; 587 } 588 p = memdbFromDbSchema(db, zSchema); 589 if( p==0 ){ 590 rc = SQLITE_ERROR; 591 }else{ 592 p->aData = pData; 593 p->sz = szDb; 594 p->szAlloc = szBuf; 595 p->szMax = szBuf; 596 if( p->szMax<SQLITE_MEMDB_DEFAULT_MAXSIZE ){ 597 p->szMax = SQLITE_MEMDB_DEFAULT_MAXSIZE; 598 } 599 p->mFlags = mFlags; 600 rc = SQLITE_OK; 601 } 602 603 end_deserialize: 604 sqlite3_finalize(pStmt); 605 sqlite3_mutex_leave(db->mutex); 606 return rc; 607 } 608 609 /* 610 ** This routine is called when the extension is loaded. 611 ** Register the new VFS. 612 */ 613 int sqlite3MemdbInit(void){ 614 sqlite3_vfs *pLower = sqlite3_vfs_find(0); 615 int sz = pLower->szOsFile; 616 memdb_vfs.pAppData = pLower; 617 /* In all known configurations of SQLite, the size of a default 618 ** sqlite3_file is greater than the size of a memdb sqlite3_file. 619 ** Should that ever change, remove the following NEVER() */ 620 if( NEVER(sz<sizeof(MemFile)) ) sz = sizeof(MemFile); 621 memdb_vfs.szOsFile = sz; 622 return sqlite3_vfs_register(&memdb_vfs, 0); 623 } 624 #endif /* SQLITE_ENABLE_DESERIALIZE */ 625