xref: /sqlite-3.40.0/src/memdb.c (revision 2d344f94)
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 #ifndef SQLITE_OMIT_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 typedef struct MemStore MemStore;
28 
29 /* Access to a lower-level VFS that (might) implement dynamic loading,
30 ** access to randomness, etc.
31 */
32 #define ORIGVFS(p) ((sqlite3_vfs*)((p)->pAppData))
33 
34 /* Storage for a memdb file.
35 **
36 ** An memdb object can be shared or separate.  Shared memdb objects can be
37 ** used by more than one database connection.  Mutexes are used by shared
38 ** memdb objects to coordinate access.  Separate memdb objects are only
39 ** connected to a single database connection and do not require additional
40 ** mutexes.
41 **
42 ** Shared memdb objects have .zFName!=0 and .pMutex!=0.  They are created
43 ** using "file:/name?vfs=memdb".  The first character of the name must be
44 ** "/" or else the object will be a separate memdb object.  All shared
45 ** memdb objects are stored in memdb_g.apMemStore[] in an arbitrary order.
46 **
47 ** Separate memdb objects are created using a name that does not begin
48 ** with "/" or using sqlite3_deserialize().
49 **
50 ** Access rules for shared MemStore objects:
51 **
52 **   *  .zFName is initialized when the object is created and afterwards
53 **      is unchanged until the object is destroyed.  So it can be accessed
54 **      at any time as long as we know the object is not being destroyed,
55 **      which means while either the SQLITE_MUTEX_STATIC_VFS1 or
56 **      .pMutex is held or the object is not part of memdb_g.apMemStore[].
57 **
58 **   *  Can .pMutex can only be changed while holding the
59 **      SQLITE_MUTEX_STATIC_VFS1 mutex or while the object is not part
60 **      of memdb_g.apMemStore[].
61 **
62 **   *  Other fields can only be changed while holding the .pMutex mutex
63 **      or when the .nRef is less than zero and the object is not part of
64 **      memdb_g.apMemStore[].
65 **
66 **   *  The .aData pointer has the added requirement that it can can only
67 **      be changed (for resizing) when nMmap is zero.
68 **
69 */
70 struct MemStore {
71   sqlite3_int64 sz;               /* Size of the file */
72   sqlite3_int64 szAlloc;          /* Space allocated to aData */
73   sqlite3_int64 szMax;            /* Maximum allowed size of the file */
74   unsigned char *aData;           /* content of the file */
75   sqlite3_mutex *pMutex;          /* Used by shared stores only */
76   int nMmap;                      /* Number of memory mapped pages */
77   unsigned mFlags;                /* Flags */
78   int nRdLock;                    /* Number of readers */
79   int nWrLock;                    /* Number of writers.  (Always 0 or 1) */
80   int nRef;                       /* Number of users of this MemStore */
81   char *zFName;                   /* The filename for shared stores */
82 };
83 
84 /* An open file */
85 struct MemFile {
86   sqlite3_file base;              /* IO methods */
87   MemStore *pStore;               /* The storage */
88   int eLock;                      /* Most recent lock against this file */
89 };
90 
91 /*
92 ** Global variables for holding the memdb files that are accessible
93 ** to multiple database connections in separate threads.
94 **
95 ** Must hold SQLITE_MUTEX_STATIC_VFS1 to access any part of this object.
96 */
97 struct MemFS {
98   int nMemStore;                  /* Number of shared MemStore objects */
99   MemStore **apMemStore;          /* Array of all shared MemStore objects */
100 } memdb_g;
101 
102 /*
103 ** Methods for MemFile
104 */
105 static int memdbClose(sqlite3_file*);
106 static int memdbRead(sqlite3_file*, void*, int iAmt, sqlite3_int64 iOfst);
107 static int memdbWrite(sqlite3_file*,const void*,int iAmt, sqlite3_int64 iOfst);
108 static int memdbTruncate(sqlite3_file*, sqlite3_int64 size);
109 static int memdbSync(sqlite3_file*, int flags);
110 static int memdbFileSize(sqlite3_file*, sqlite3_int64 *pSize);
111 static int memdbLock(sqlite3_file*, int);
112 /* static int memdbCheckReservedLock(sqlite3_file*, int *pResOut);// not used */
113 static int memdbFileControl(sqlite3_file*, int op, void *pArg);
114 /* static int memdbSectorSize(sqlite3_file*); // not used */
115 static int memdbDeviceCharacteristics(sqlite3_file*);
116 static int memdbFetch(sqlite3_file*, sqlite3_int64 iOfst, int iAmt, void **pp);
117 static int memdbUnfetch(sqlite3_file*, sqlite3_int64 iOfst, void *p);
118 
119 /*
120 ** Methods for MemVfs
121 */
122 static int memdbOpen(sqlite3_vfs*, const char *, sqlite3_file*, int , int *);
123 /* static int memdbDelete(sqlite3_vfs*, const char *zName, int syncDir); */
124 static int memdbAccess(sqlite3_vfs*, const char *zName, int flags, int *);
125 static int memdbFullPathname(sqlite3_vfs*, const char *zName, int, char *zOut);
126 static void *memdbDlOpen(sqlite3_vfs*, const char *zFilename);
127 static void memdbDlError(sqlite3_vfs*, int nByte, char *zErrMsg);
128 static void (*memdbDlSym(sqlite3_vfs *pVfs, void *p, const char*zSym))(void);
129 static void memdbDlClose(sqlite3_vfs*, void*);
130 static int memdbRandomness(sqlite3_vfs*, int nByte, char *zOut);
131 static int memdbSleep(sqlite3_vfs*, int microseconds);
132 /* static int memdbCurrentTime(sqlite3_vfs*, double*); */
133 static int memdbGetLastError(sqlite3_vfs*, int, char *);
134 static int memdbCurrentTimeInt64(sqlite3_vfs*, sqlite3_int64*);
135 
136 static sqlite3_vfs memdb_vfs = {
137   2,                           /* iVersion */
138   0,                           /* szOsFile (set when registered) */
139   1024,                        /* mxPathname */
140   0,                           /* pNext */
141   "memdb",                     /* zName */
142   0,                           /* pAppData (set when registered) */
143   memdbOpen,                   /* xOpen */
144   0, /* memdbDelete, */        /* xDelete */
145   memdbAccess,                 /* xAccess */
146   memdbFullPathname,           /* xFullPathname */
147   memdbDlOpen,                 /* xDlOpen */
148   memdbDlError,                /* xDlError */
149   memdbDlSym,                  /* xDlSym */
150   memdbDlClose,                /* xDlClose */
151   memdbRandomness,             /* xRandomness */
152   memdbSleep,                  /* xSleep */
153   0, /* memdbCurrentTime, */   /* xCurrentTime */
154   memdbGetLastError,           /* xGetLastError */
155   memdbCurrentTimeInt64        /* xCurrentTimeInt64 */
156 };
157 
158 static const sqlite3_io_methods memdb_io_methods = {
159   3,                              /* iVersion */
160   memdbClose,                      /* xClose */
161   memdbRead,                       /* xRead */
162   memdbWrite,                      /* xWrite */
163   memdbTruncate,                   /* xTruncate */
164   memdbSync,                       /* xSync */
165   memdbFileSize,                   /* xFileSize */
166   memdbLock,                       /* xLock */
167   memdbLock,                       /* xUnlock - same as xLock in this case */
168   0, /* memdbCheckReservedLock, */ /* xCheckReservedLock */
169   memdbFileControl,                /* xFileControl */
170   0, /* memdbSectorSize,*/         /* xSectorSize */
171   memdbDeviceCharacteristics,      /* xDeviceCharacteristics */
172   0,                               /* xShmMap */
173   0,                               /* xShmLock */
174   0,                               /* xShmBarrier */
175   0,                               /* xShmUnmap */
176   memdbFetch,                      /* xFetch */
177   memdbUnfetch                     /* xUnfetch */
178 };
179 
180 /*
181 ** Enter/leave the mutex on a MemStore
182 */
183 static void memdbEnter(MemStore *p){
184   sqlite3_mutex_enter(p->pMutex);
185 }
186 static void memdbLeave(MemStore *p){
187   sqlite3_mutex_leave(p->pMutex);
188 }
189 
190 
191 
192 /*
193 ** Close an memdb-file.
194 ** Free the underlying MemStore object when its refcount drops to zero
195 ** or less.
196 */
197 static int memdbClose(sqlite3_file *pFile){
198   MemStore *p = ((MemFile*)pFile)->pStore;
199   memdbEnter(p);
200   p->nRef--;
201   if( p->nRef<=0 ){
202     if( p->mFlags & SQLITE_DESERIALIZE_FREEONCLOSE ){
203       sqlite3_free(p->aData);
204     }
205     if( p->zFName ){
206       int i;
207 #ifndef SQLITE_MUTEX_OMIT
208       sqlite3_mutex *pVfsMutex = sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_VFS1);
209 #endif
210       sqlite3_mutex_enter(pVfsMutex);
211       for(i=0; ALWAYS(i<memdb_g.nMemStore); i++){
212         if( memdb_g.apMemStore[i]==p ){
213           memdb_g.apMemStore[i] = memdb_g.apMemStore[--memdb_g.nMemStore];
214           if( memdb_g.nMemStore==0 ){
215             sqlite3_free(memdb_g.apMemStore);
216             memdb_g.apMemStore = 0;
217           }
218           break;
219         }
220       }
221       sqlite3_mutex_leave(pVfsMutex);
222     }
223     memdbLeave(p);
224     sqlite3_mutex_free(p->pMutex);
225     sqlite3_free(p);
226   }else{
227     memdbLeave(p);
228   }
229   return SQLITE_OK;
230 }
231 
232 /*
233 ** Read data from an memdb-file.
234 */
235 static int memdbRead(
236   sqlite3_file *pFile,
237   void *zBuf,
238   int iAmt,
239   sqlite_int64 iOfst
240 ){
241   MemStore *p = ((MemFile*)pFile)->pStore;
242   memdbEnter(p);
243   if( iOfst+iAmt>p->sz ){
244     memset(zBuf, 0, iAmt);
245     if( iOfst<p->sz ) memcpy(zBuf, p->aData+iOfst, p->sz - iOfst);
246     memdbLeave(p);
247     return SQLITE_IOERR_SHORT_READ;
248   }
249   memcpy(zBuf, p->aData+iOfst, iAmt);
250   memdbLeave(p);
251   return SQLITE_OK;
252 }
253 
254 /*
255 ** Try to enlarge the memory allocation to hold at least sz bytes
256 */
257 static int memdbEnlarge(MemStore *p, sqlite3_int64 newSz){
258   unsigned char *pNew;
259   if( (p->mFlags & SQLITE_DESERIALIZE_RESIZEABLE)==0 || p->nMmap>0 ){
260     return SQLITE_FULL;
261   }
262   if( newSz>p->szMax ){
263     return SQLITE_FULL;
264   }
265   newSz *= 2;
266   if( newSz>p->szMax ) newSz = p->szMax;
267   pNew = sqlite3Realloc(p->aData, newSz);
268   if( pNew==0 ) return SQLITE_IOERR_NOMEM;
269   p->aData = pNew;
270   p->szAlloc = newSz;
271   return SQLITE_OK;
272 }
273 
274 /*
275 ** Write data to an memdb-file.
276 */
277 static int memdbWrite(
278   sqlite3_file *pFile,
279   const void *z,
280   int iAmt,
281   sqlite_int64 iOfst
282 ){
283   MemStore *p = ((MemFile*)pFile)->pStore;
284   memdbEnter(p);
285   if( NEVER(p->mFlags & SQLITE_DESERIALIZE_READONLY) ){
286     /* Can't happen: memdbLock() will return SQLITE_READONLY before
287     ** reaching this point */
288     memdbLeave(p);
289     return SQLITE_IOERR_WRITE;
290   }
291   if( iOfst+iAmt>p->sz ){
292     int rc;
293     if( iOfst+iAmt>p->szAlloc
294      && (rc = memdbEnlarge(p, iOfst+iAmt))!=SQLITE_OK
295     ){
296       memdbLeave(p);
297       return rc;
298     }
299     if( iOfst>p->sz ) memset(p->aData+p->sz, 0, iOfst-p->sz);
300     p->sz = iOfst+iAmt;
301   }
302   memcpy(p->aData+iOfst, z, iAmt);
303   memdbLeave(p);
304   return SQLITE_OK;
305 }
306 
307 /*
308 ** Truncate an memdb-file.
309 **
310 ** In rollback mode (which is always the case for memdb, as it does not
311 ** support WAL mode) the truncate() method is only used to reduce
312 ** the size of a file, never to increase the size.
313 */
314 static int memdbTruncate(sqlite3_file *pFile, sqlite_int64 size){
315   MemStore *p = ((MemFile*)pFile)->pStore;
316   int rc = SQLITE_OK;
317   memdbEnter(p);
318   if( NEVER(size>p->sz) ){
319     rc = SQLITE_FULL;
320   }else{
321     p->sz = size;
322   }
323   memdbLeave(p);
324   return rc;
325 }
326 
327 /*
328 ** Sync an memdb-file.
329 */
330 static int memdbSync(sqlite3_file *pFile, int flags){
331   return SQLITE_OK;
332 }
333 
334 /*
335 ** Return the current file-size of an memdb-file.
336 */
337 static int memdbFileSize(sqlite3_file *pFile, sqlite_int64 *pSize){
338   MemStore *p = ((MemFile*)pFile)->pStore;
339   memdbEnter(p);
340   *pSize = p->sz;
341   memdbLeave(p);
342   return SQLITE_OK;
343 }
344 
345 /*
346 ** Lock an memdb-file.
347 */
348 static int memdbLock(sqlite3_file *pFile, int eLock){
349   MemFile *pThis = (MemFile*)pFile;
350   MemStore *p = pThis->pStore;
351   int rc = SQLITE_OK;
352   if( eLock==pThis->eLock ) return SQLITE_OK;
353   memdbEnter(p);
354   if( eLock>SQLITE_LOCK_SHARED ){
355     if( p->mFlags & SQLITE_DESERIALIZE_READONLY ){
356       rc = SQLITE_READONLY;
357     }else if( pThis->eLock<=SQLITE_LOCK_SHARED ){
358       if( p->nWrLock ){
359         rc = SQLITE_BUSY;
360       }else{
361         p->nWrLock = 1;
362       }
363     }
364   }else if( eLock==SQLITE_LOCK_SHARED ){
365     if( pThis->eLock > SQLITE_LOCK_SHARED ){
366       assert( p->nWrLock==1 );
367       p->nWrLock = 0;
368     }else if( p->nWrLock ){
369       rc = SQLITE_BUSY;
370     }else{
371       p->nRdLock++;
372     }
373   }else{
374     assert( eLock==SQLITE_LOCK_NONE );
375     if( pThis->eLock>SQLITE_LOCK_SHARED ){
376       assert( p->nWrLock==1 );
377       p->nWrLock = 0;
378     }
379     assert( p->nRdLock>0 );
380     p->nRdLock--;
381   }
382   if( rc==SQLITE_OK ) pThis->eLock = eLock;
383   memdbLeave(p);
384   return rc;
385 }
386 
387 #if 0
388 /*
389 ** This interface is only used for crash recovery, which does not
390 ** occur on an in-memory database.
391 */
392 static int memdbCheckReservedLock(sqlite3_file *pFile, int *pResOut){
393   *pResOut = 0;
394   return SQLITE_OK;
395 }
396 #endif
397 
398 
399 /*
400 ** File control method. For custom operations on an memdb-file.
401 */
402 static int memdbFileControl(sqlite3_file *pFile, int op, void *pArg){
403   MemStore *p = ((MemFile*)pFile)->pStore;
404   int rc = SQLITE_NOTFOUND;
405   memdbEnter(p);
406   if( op==SQLITE_FCNTL_VFSNAME ){
407     *(char**)pArg = sqlite3_mprintf("memdb(%p,%lld)", p->aData, p->sz);
408     rc = SQLITE_OK;
409   }
410   if( op==SQLITE_FCNTL_SIZE_LIMIT ){
411     sqlite3_int64 iLimit = *(sqlite3_int64*)pArg;
412     if( iLimit<p->sz ){
413       if( iLimit<0 ){
414         iLimit = p->szMax;
415       }else{
416         iLimit = p->sz;
417       }
418     }
419     p->szMax = iLimit;
420     *(sqlite3_int64*)pArg = iLimit;
421     rc = SQLITE_OK;
422   }
423   memdbLeave(p);
424   return rc;
425 }
426 
427 #if 0  /* Not used because of SQLITE_IOCAP_POWERSAFE_OVERWRITE */
428 /*
429 ** Return the sector-size in bytes for an memdb-file.
430 */
431 static int memdbSectorSize(sqlite3_file *pFile){
432   return 1024;
433 }
434 #endif
435 
436 /*
437 ** Return the device characteristic flags supported by an memdb-file.
438 */
439 static int memdbDeviceCharacteristics(sqlite3_file *pFile){
440   return SQLITE_IOCAP_ATOMIC |
441          SQLITE_IOCAP_POWERSAFE_OVERWRITE |
442          SQLITE_IOCAP_SAFE_APPEND |
443          SQLITE_IOCAP_SEQUENTIAL;
444 }
445 
446 /* Fetch a page of a memory-mapped file */
447 static int memdbFetch(
448   sqlite3_file *pFile,
449   sqlite3_int64 iOfst,
450   int iAmt,
451   void **pp
452 ){
453   MemStore *p = ((MemFile*)pFile)->pStore;
454   memdbEnter(p);
455   if( iOfst+iAmt>p->sz ){
456     *pp = 0;
457   }else{
458     p->nMmap++;
459     *pp = (void*)(p->aData + iOfst);
460   }
461   memdbLeave(p);
462   return SQLITE_OK;
463 }
464 
465 /* Release a memory-mapped page */
466 static int memdbUnfetch(sqlite3_file *pFile, sqlite3_int64 iOfst, void *pPage){
467   MemStore *p = ((MemFile*)pFile)->pStore;
468   memdbEnter(p);
469   p->nMmap--;
470   memdbLeave(p);
471   return SQLITE_OK;
472 }
473 
474 /*
475 ** Open an mem file handle.
476 */
477 static int memdbOpen(
478   sqlite3_vfs *pVfs,
479   const char *zName,
480   sqlite3_file *pFd,
481   int flags,
482   int *pOutFlags
483 ){
484   MemFile *pFile = (MemFile*)pFd;
485   MemStore *p = 0;
486   int szName;
487   if( (flags & SQLITE_OPEN_MAIN_DB)==0 ){
488     return ORIGVFS(pVfs)->xOpen(ORIGVFS(pVfs), zName, pFd, flags, pOutFlags);
489   }
490   memset(pFile, 0, sizeof(*p));
491   szName = sqlite3Strlen30(zName);
492   if( szName>1 && zName[0]=='/' ){
493     int i;
494 #ifndef SQLITE_MUTEX_OMIT
495     sqlite3_mutex *pVfsMutex = sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_VFS1);
496 #endif
497     sqlite3_mutex_enter(pVfsMutex);
498     for(i=0; i<memdb_g.nMemStore; i++){
499       if( strcmp(memdb_g.apMemStore[i]->zFName,zName)==0 ){
500         p = memdb_g.apMemStore[i];
501         break;
502       }
503     }
504     if( p==0 ){
505       MemStore **apNew;
506       p = sqlite3Malloc( sizeof(*p) + szName + 3 );
507       if( p==0 ){
508         sqlite3_mutex_leave(pVfsMutex);
509         return SQLITE_NOMEM;
510       }
511       apNew = sqlite3Realloc(memdb_g.apMemStore,
512                              sizeof(apNew[0])*(memdb_g.nMemStore+1) );
513       if( apNew==0 ){
514         sqlite3_free(p);
515         sqlite3_mutex_leave(pVfsMutex);
516         return SQLITE_NOMEM;
517       }
518       apNew[memdb_g.nMemStore++] = p;
519       memdb_g.apMemStore = apNew;
520       memset(p, 0, sizeof(*p));
521       p->mFlags = SQLITE_DESERIALIZE_RESIZEABLE|SQLITE_DESERIALIZE_FREEONCLOSE;
522       p->szMax = sqlite3GlobalConfig.mxMemdbSize;
523       p->zFName = (char*)&p[1];
524       memcpy(p->zFName, zName, szName+1);
525       p->pMutex = sqlite3_mutex_alloc(SQLITE_MUTEX_FAST);
526       if( p->pMutex==0 ){
527         memdb_g.nMemStore--;
528         sqlite3_free(p);
529         sqlite3_mutex_leave(pVfsMutex);
530         return SQLITE_NOMEM;
531       }
532       p->nRef = 1;
533       memdbEnter(p);
534     }else{
535       memdbEnter(p);
536       p->nRef++;
537     }
538     sqlite3_mutex_leave(pVfsMutex);
539   }else{
540     p = sqlite3Malloc( sizeof(*p) );
541     if( p==0 ){
542       return SQLITE_NOMEM;
543     }
544     memset(p, 0, sizeof(*p));
545     p->mFlags = SQLITE_DESERIALIZE_RESIZEABLE | SQLITE_DESERIALIZE_FREEONCLOSE;
546     p->szMax = sqlite3GlobalConfig.mxMemdbSize;
547   }
548   pFile->pStore = p;
549   assert( pOutFlags!=0 );  /* True because flags==SQLITE_OPEN_MAIN_DB */
550   *pOutFlags = flags | SQLITE_OPEN_MEMORY;
551   pFd->pMethods = &memdb_io_methods;
552   memdbLeave(p);
553   return SQLITE_OK;
554 }
555 
556 #if 0 /* Only used to delete rollback journals, super-journals, and WAL
557       ** files, none of which exist in memdb.  So this routine is never used */
558 /*
559 ** Delete the file located at zPath. If the dirSync argument is true,
560 ** ensure the file-system modifications are synced to disk before
561 ** returning.
562 */
563 static int memdbDelete(sqlite3_vfs *pVfs, const char *zPath, int dirSync){
564   return SQLITE_IOERR_DELETE;
565 }
566 #endif
567 
568 /*
569 ** Test for access permissions. Return true if the requested permission
570 ** is available, or false otherwise.
571 **
572 ** With memdb, no files ever exist on disk.  So always return false.
573 */
574 static int memdbAccess(
575   sqlite3_vfs *pVfs,
576   const char *zPath,
577   int flags,
578   int *pResOut
579 ){
580   *pResOut = 0;
581   return SQLITE_OK;
582 }
583 
584 /*
585 ** Populate buffer zOut with the full canonical pathname corresponding
586 ** to the pathname in zPath. zOut is guaranteed to point to a buffer
587 ** of at least (INST_MAX_PATHNAME+1) bytes.
588 */
589 static int memdbFullPathname(
590   sqlite3_vfs *pVfs,
591   const char *zPath,
592   int nOut,
593   char *zOut
594 ){
595   sqlite3_snprintf(nOut, zOut, "%s", zPath);
596   return SQLITE_OK;
597 }
598 
599 /*
600 ** Open the dynamic library located at zPath and return a handle.
601 */
602 static void *memdbDlOpen(sqlite3_vfs *pVfs, const char *zPath){
603   return ORIGVFS(pVfs)->xDlOpen(ORIGVFS(pVfs), zPath);
604 }
605 
606 /*
607 ** Populate the buffer zErrMsg (size nByte bytes) with a human readable
608 ** utf-8 string describing the most recent error encountered associated
609 ** with dynamic libraries.
610 */
611 static void memdbDlError(sqlite3_vfs *pVfs, int nByte, char *zErrMsg){
612   ORIGVFS(pVfs)->xDlError(ORIGVFS(pVfs), nByte, zErrMsg);
613 }
614 
615 /*
616 ** Return a pointer to the symbol zSymbol in the dynamic library pHandle.
617 */
618 static void (*memdbDlSym(sqlite3_vfs *pVfs, void *p, const char *zSym))(void){
619   return ORIGVFS(pVfs)->xDlSym(ORIGVFS(pVfs), p, zSym);
620 }
621 
622 /*
623 ** Close the dynamic library handle pHandle.
624 */
625 static void memdbDlClose(sqlite3_vfs *pVfs, void *pHandle){
626   ORIGVFS(pVfs)->xDlClose(ORIGVFS(pVfs), pHandle);
627 }
628 
629 /*
630 ** Populate the buffer pointed to by zBufOut with nByte bytes of
631 ** random data.
632 */
633 static int memdbRandomness(sqlite3_vfs *pVfs, int nByte, char *zBufOut){
634   return ORIGVFS(pVfs)->xRandomness(ORIGVFS(pVfs), nByte, zBufOut);
635 }
636 
637 /*
638 ** Sleep for nMicro microseconds. Return the number of microseconds
639 ** actually slept.
640 */
641 static int memdbSleep(sqlite3_vfs *pVfs, int nMicro){
642   return ORIGVFS(pVfs)->xSleep(ORIGVFS(pVfs), nMicro);
643 }
644 
645 #if 0  /* Never used.  Modern cores only call xCurrentTimeInt64() */
646 /*
647 ** Return the current time as a Julian Day number in *pTimeOut.
648 */
649 static int memdbCurrentTime(sqlite3_vfs *pVfs, double *pTimeOut){
650   return ORIGVFS(pVfs)->xCurrentTime(ORIGVFS(pVfs), pTimeOut);
651 }
652 #endif
653 
654 static int memdbGetLastError(sqlite3_vfs *pVfs, int a, char *b){
655   return ORIGVFS(pVfs)->xGetLastError(ORIGVFS(pVfs), a, b);
656 }
657 static int memdbCurrentTimeInt64(sqlite3_vfs *pVfs, sqlite3_int64 *p){
658   return ORIGVFS(pVfs)->xCurrentTimeInt64(ORIGVFS(pVfs), p);
659 }
660 
661 /*
662 ** Translate a database connection pointer and schema name into a
663 ** MemFile pointer.
664 */
665 static MemFile *memdbFromDbSchema(sqlite3 *db, const char *zSchema){
666   MemFile *p = 0;
667   MemStore *pStore;
668   int rc = sqlite3_file_control(db, zSchema, SQLITE_FCNTL_FILE_POINTER, &p);
669   if( rc ) return 0;
670   if( p->base.pMethods!=&memdb_io_methods ) return 0;
671   pStore = p->pStore;
672   memdbEnter(pStore);
673   if( pStore->zFName!=0 ) p = 0;
674   memdbLeave(pStore);
675   return p;
676 }
677 
678 /*
679 ** Return the serialization of a database
680 */
681 unsigned char *sqlite3_serialize(
682   sqlite3 *db,              /* The database connection */
683   const char *zSchema,      /* Which database within the connection */
684   sqlite3_int64 *piSize,    /* Write size here, if not NULL */
685   unsigned int mFlags       /* Maybe SQLITE_SERIALIZE_NOCOPY */
686 ){
687   MemFile *p;
688   int iDb;
689   Btree *pBt;
690   sqlite3_int64 sz;
691   int szPage = 0;
692   sqlite3_stmt *pStmt = 0;
693   unsigned char *pOut;
694   char *zSql;
695   int rc;
696 
697 #ifdef SQLITE_ENABLE_API_ARMOR
698   if( !sqlite3SafetyCheckOk(db) ){
699     (void)SQLITE_MISUSE_BKPT;
700     return 0;
701   }
702 #endif
703 
704   if( zSchema==0 ) zSchema = db->aDb[0].zDbSName;
705   p = memdbFromDbSchema(db, zSchema);
706   iDb = sqlite3FindDbName(db, zSchema);
707   if( piSize ) *piSize = -1;
708   if( iDb<0 ) return 0;
709   if( p ){
710     MemStore *pStore = p->pStore;
711     assert( pStore->pMutex==0 );
712     if( piSize ) *piSize = pStore->sz;
713     if( mFlags & SQLITE_SERIALIZE_NOCOPY ){
714       pOut = pStore->aData;
715     }else{
716       pOut = sqlite3_malloc64( pStore->sz );
717       if( pOut ) memcpy(pOut, pStore->aData, pStore->sz);
718     }
719     return pOut;
720   }
721   pBt = db->aDb[iDb].pBt;
722   if( pBt==0 ) return 0;
723   szPage = sqlite3BtreeGetPageSize(pBt);
724   zSql = sqlite3_mprintf("PRAGMA \"%w\".page_count", zSchema);
725   rc = zSql ? sqlite3_prepare_v2(db, zSql, -1, &pStmt, 0) : SQLITE_NOMEM;
726   sqlite3_free(zSql);
727   if( rc ) return 0;
728   rc = sqlite3_step(pStmt);
729   if( rc!=SQLITE_ROW ){
730     pOut = 0;
731   }else{
732     sz = sqlite3_column_int64(pStmt, 0)*szPage;
733     if( piSize ) *piSize = sz;
734     if( mFlags & SQLITE_SERIALIZE_NOCOPY ){
735       pOut = 0;
736     }else{
737       pOut = sqlite3_malloc64( sz );
738       if( pOut ){
739         int nPage = sqlite3_column_int(pStmt, 0);
740         Pager *pPager = sqlite3BtreePager(pBt);
741         int pgno;
742         for(pgno=1; pgno<=nPage; pgno++){
743           DbPage *pPage = 0;
744           unsigned char *pTo = pOut + szPage*(sqlite3_int64)(pgno-1);
745           rc = sqlite3PagerGet(pPager, pgno, (DbPage**)&pPage, 0);
746           if( rc==SQLITE_OK ){
747             memcpy(pTo, sqlite3PagerGetData(pPage), szPage);
748           }else{
749             memset(pTo, 0, szPage);
750           }
751           sqlite3PagerUnref(pPage);
752         }
753       }
754     }
755   }
756   sqlite3_finalize(pStmt);
757   return pOut;
758 }
759 
760 /* Convert zSchema to a MemDB and initialize its content.
761 */
762 int sqlite3_deserialize(
763   sqlite3 *db,            /* The database connection */
764   const char *zSchema,    /* Which DB to reopen with the deserialization */
765   unsigned char *pData,   /* The serialized database content */
766   sqlite3_int64 szDb,     /* Number bytes in the deserialization */
767   sqlite3_int64 szBuf,    /* Total size of buffer pData[] */
768   unsigned mFlags         /* Zero or more SQLITE_DESERIALIZE_* flags */
769 ){
770   MemFile *p;
771   char *zSql;
772   sqlite3_stmt *pStmt = 0;
773   int rc;
774   int iDb;
775 
776 #ifdef SQLITE_ENABLE_API_ARMOR
777   if( !sqlite3SafetyCheckOk(db) ){
778     return SQLITE_MISUSE_BKPT;
779   }
780   if( szDb<0 ) return SQLITE_MISUSE_BKPT;
781   if( szBuf<0 ) return SQLITE_MISUSE_BKPT;
782 #endif
783 
784   sqlite3_mutex_enter(db->mutex);
785   if( zSchema==0 ) zSchema = db->aDb[0].zDbSName;
786   iDb = sqlite3FindDbName(db, zSchema);
787   if( iDb<0 ){
788     rc = SQLITE_ERROR;
789     goto end_deserialize;
790   }
791   zSql = sqlite3_mprintf("ATTACH x AS %Q", zSchema);
792   if( zSql==0 ){
793     rc = SQLITE_NOMEM;
794   }else{
795     rc = sqlite3_prepare_v2(db, zSql, -1, &pStmt, 0);
796     sqlite3_free(zSql);
797   }
798   if( rc ) goto end_deserialize;
799   db->init.iDb = (u8)iDb;
800   db->init.reopenMemdb = 1;
801   rc = sqlite3_step(pStmt);
802   db->init.reopenMemdb = 0;
803   if( rc!=SQLITE_DONE ){
804     rc = SQLITE_ERROR;
805     goto end_deserialize;
806   }
807   p = memdbFromDbSchema(db, zSchema);
808   if( p==0 ){
809     rc = SQLITE_ERROR;
810   }else{
811     MemStore *pStore = p->pStore;
812     pStore->aData = pData;
813     pData = 0;
814     pStore->sz = szDb;
815     pStore->szAlloc = szBuf;
816     pStore->szMax = szBuf;
817     if( pStore->szMax<sqlite3GlobalConfig.mxMemdbSize ){
818       pStore->szMax = sqlite3GlobalConfig.mxMemdbSize;
819     }
820     pStore->mFlags = mFlags;
821     rc = SQLITE_OK;
822   }
823 
824 end_deserialize:
825   sqlite3_finalize(pStmt);
826   if( pData && (mFlags & SQLITE_DESERIALIZE_FREEONCLOSE)!=0 ){
827     sqlite3_free(pData);
828   }
829   sqlite3_mutex_leave(db->mutex);
830   return rc;
831 }
832 
833 /*
834 ** This routine is called when the extension is loaded.
835 ** Register the new VFS.
836 */
837 int sqlite3MemdbInit(void){
838   sqlite3_vfs *pLower = sqlite3_vfs_find(0);
839   int sz = pLower->szOsFile;
840   memdb_vfs.pAppData = pLower;
841   /* The following conditional can only be true when compiled for
842   ** Windows x86 and SQLITE_MAX_MMAP_SIZE=0.  We always leave
843   ** it in, to be safe, but it is marked as NO_TEST since there
844   ** is no way to reach it under most builds. */
845   if( sz<sizeof(MemFile) ) sz = sizeof(MemFile); /*NO_TEST*/
846   memdb_vfs.szOsFile = sz;
847   return sqlite3_vfs_register(&memdb_vfs, 0);
848 }
849 #endif /* SQLITE_OMIT_DESERIALIZE */
850