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 ** This file contains OS interface code that is common to all
16 ** $Id: os.c,v 1.123 2008/09/23 16:41:30 danielk1977 Exp $
18 #define _SQLITE_OS_C_ 1
19 #include "sqliteInt.h"
23 ** The default SQLite sqlite3_vfs implementations do not allocate
24 ** memory (actually, os_unix.c allocates a small amount of memory
25 ** from within OsOpen()), but some third-party implementations may.
26 ** So we test the effects of a malloc() failing and the sqlite3OsXXX()
27 ** function returning SQLITE_IOERR_NOMEM using the DO_OS_MALLOC_TEST macro.
29 ** The following functions are instrumented for malloc() failure
39 #if defined(SQLITE_TEST) && (SQLITE_OS_WIN==0)
40 #define DO_OS_MALLOC_TEST if (1) { \
41 void *pTstAlloc = sqlite3Malloc(10); \
42 if (!pTstAlloc) return SQLITE_IOERR_NOMEM; \
43 sqlite3_free(pTstAlloc); \
46 #define DO_OS_MALLOC_TEST
50 ** The following routines are convenience wrappers around methods
51 ** of the sqlite3_file object. This is mostly just syntactic sugar. All
52 ** of this would be completely automatic if SQLite were coded using
53 ** C++ instead of plain old C.
55 int sqlite3OsClose(sqlite3_file *pId){
58 rc = pId->pMethods->xClose(pId);
63 int sqlite3OsRead(sqlite3_file *id, void *pBuf, int amt, i64 offset){
65 return id->pMethods->xRead(id, pBuf, amt, offset);
67 int sqlite3OsWrite(sqlite3_file *id, const void *pBuf, int amt, i64 offset){
69 return id->pMethods->xWrite(id, pBuf, amt, offset);
71 int sqlite3OsTruncate(sqlite3_file *id, i64 size){
72 return id->pMethods->xTruncate(id, size);
74 int sqlite3OsSync(sqlite3_file *id, int flags){
76 return id->pMethods->xSync(id, flags);
78 int sqlite3OsFileSize(sqlite3_file *id, i64 *pSize){
80 return id->pMethods->xFileSize(id, pSize);
82 int sqlite3OsLock(sqlite3_file *id, int lockType){
84 return id->pMethods->xLock(id, lockType);
86 int sqlite3OsUnlock(sqlite3_file *id, int lockType){
87 return id->pMethods->xUnlock(id, lockType);
89 int sqlite3OsCheckReservedLock(sqlite3_file *id, int *pResOut){
91 return id->pMethods->xCheckReservedLock(id, pResOut);
93 int sqlite3OsFileControl(sqlite3_file *id, int op, void *pArg){
94 return id->pMethods->xFileControl(id, op, pArg);
96 int sqlite3OsSectorSize(sqlite3_file *id){
97 int (*xSectorSize)(sqlite3_file*) = id->pMethods->xSectorSize;
98 return (xSectorSize ? xSectorSize(id) : SQLITE_DEFAULT_SECTOR_SIZE);
100 int sqlite3OsDeviceCharacteristics(sqlite3_file *id){
101 return id->pMethods->xDeviceCharacteristics(id);
105 ** The next group of routines are convenience wrappers around the
116 return pVfs->xOpen(pVfs, zPath, pFile, flags, pFlagsOut);
118 int sqlite3OsDelete(sqlite3_vfs *pVfs, const char *zPath, int dirSync){
119 return pVfs->xDelete(pVfs, zPath, dirSync);
128 return pVfs->xAccess(pVfs, zPath, flags, pResOut);
130 int sqlite3OsFullPathname(
136 return pVfs->xFullPathname(pVfs, zPath, nPathOut, zPathOut);
138 #ifndef SQLITE_OMIT_LOAD_EXTENSION
139 void *sqlite3OsDlOpen(sqlite3_vfs *pVfs, const char *zPath){
140 return pVfs->xDlOpen(pVfs, zPath);
142 void sqlite3OsDlError(sqlite3_vfs *pVfs, int nByte, char *zBufOut){
143 pVfs->xDlError(pVfs, nByte, zBufOut);
145 void *sqlite3OsDlSym(sqlite3_vfs *pVfs, void *pHandle, const char *zSymbol){
146 return pVfs->xDlSym(pVfs, pHandle, zSymbol);
148 void sqlite3OsDlClose(sqlite3_vfs *pVfs, void *pHandle){
149 pVfs->xDlClose(pVfs, pHandle);
151 #endif /* SQLITE_OMIT_LOAD_EXTENSION */
152 int sqlite3OsRandomness(sqlite3_vfs *pVfs, int nByte, char *zBufOut){
153 return pVfs->xRandomness(pVfs, nByte, zBufOut);
155 int sqlite3OsSleep(sqlite3_vfs *pVfs, int nMicro){
156 return pVfs->xSleep(pVfs, nMicro);
158 int sqlite3OsCurrentTime(sqlite3_vfs *pVfs, double *pTimeOut){
159 return pVfs->xCurrentTime(pVfs, pTimeOut);
162 int sqlite3OsOpenMalloc(
165 sqlite3_file **ppFile,
169 int rc = SQLITE_NOMEM;
171 pFile = (sqlite3_file *)sqlite3Malloc(pVfs->szOsFile);
173 rc = sqlite3OsOpen(pVfs, zFile, pFile, flags, pOutFlags);
182 int sqlite3OsCloseFree(sqlite3_file *pFile){
185 rc = sqlite3OsClose(pFile);
191 ** The list of all registered VFS implementations.
193 static sqlite3_vfs * SQLITE_WSD vfsList = 0;
194 #define vfsList GLOBAL(sqlite3_vfs *, vfsList)
197 ** Locate a VFS by name. If no name is given, simply return the
198 ** first VFS on the list.
200 SQLITE_EXPORT sqlite3_vfs *sqlite3_vfs_find(const char *zVfs){
201 sqlite3_vfs *pVfs = 0;
202 #ifndef SQLITE_MUTEX_NOOP
203 sqlite3_mutex *mutex;
205 #ifndef SQLITE_OMIT_AUTOINIT
206 int rc = sqlite3_initialize();
209 #ifndef SQLITE_MUTEX_NOOP
210 mutex = sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_MASTER);
212 sqlite3_mutex_enter(mutex);
213 for(pVfs = vfsList; pVfs; pVfs=pVfs->pNext){
215 if( strcmp(zVfs, pVfs->zName)==0 ) break;
217 sqlite3_mutex_leave(mutex);
222 ** Unlink a VFS from the linked list
224 static void vfsUnlink(sqlite3_vfs *pVfs){
225 assert( sqlite3_mutex_held(sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_MASTER)) );
228 }else if( vfsList==pVfs ){
229 vfsList = pVfs->pNext;
231 sqlite3_vfs *p = vfsList;
232 while( p->pNext && p->pNext!=pVfs ){
235 if( p->pNext==pVfs ){
236 p->pNext = pVfs->pNext;
242 ** Register a VFS with the system. It is harmless to register the same
243 ** VFS multiple times. The new VFS becomes the default if makeDflt is
246 SQLITE_EXPORT int sqlite3_vfs_register(sqlite3_vfs *pVfs, int makeDflt){
247 sqlite3_mutex *mutex = 0;
248 #ifndef SQLITE_OMIT_AUTOINIT
249 int rc = sqlite3_initialize();
252 mutex = sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_MASTER);
253 sqlite3_mutex_enter(mutex);
255 if( makeDflt || vfsList==0 ){
256 pVfs->pNext = vfsList;
259 pVfs->pNext = vfsList->pNext;
260 vfsList->pNext = pVfs;
263 sqlite3_mutex_leave(mutex);
268 ** Unregister a VFS so that it is no longer accessible.
270 SQLITE_EXPORT int sqlite3_vfs_unregister(sqlite3_vfs *pVfs){
271 #ifndef SQLITE_MUTEX_NOOP
272 sqlite3_mutex *mutex = sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_MASTER);
274 sqlite3_mutex_enter(mutex);
276 sqlite3_mutex_leave(mutex);