xref: /sqlite-3.40.0/src/os.c (revision a3f06598)
1 /*
2 ** 2005 November 29
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 contains OS interface code that is common to all
14 ** architectures.
15 **
16 ** $Id: os.c,v 1.126 2009/03/25 14:24:42 drh Exp $
17 */
18 #define _SQLITE_OS_C_ 1
19 #include "sqliteInt.h"
20 #undef _SQLITE_OS_C_
21 
22 /*
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.
28 **
29 ** The following functions are instrumented for malloc() failure
30 ** testing:
31 **
32 **     sqlite3OsOpen()
33 **     sqlite3OsRead()
34 **     sqlite3OsWrite()
35 **     sqlite3OsSync()
36 **     sqlite3OsLock()
37 **
38 */
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);                    \
44   }
45 #else
46   #define DO_OS_MALLOC_TEST
47 #endif
48 
49 /*
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.
54 */
55 int sqlite3OsClose(sqlite3_file *pId){
56   int rc = SQLITE_OK;
57   if( pId->pMethods ){
58     rc = pId->pMethods->xClose(pId);
59     pId->pMethods = 0;
60   }
61   return rc;
62 }
63 int sqlite3OsRead(sqlite3_file *id, void *pBuf, int amt, i64 offset){
64   DO_OS_MALLOC_TEST;
65   return id->pMethods->xRead(id, pBuf, amt, offset);
66 }
67 int sqlite3OsWrite(sqlite3_file *id, const void *pBuf, int amt, i64 offset){
68   DO_OS_MALLOC_TEST;
69   return id->pMethods->xWrite(id, pBuf, amt, offset);
70 }
71 int sqlite3OsTruncate(sqlite3_file *id, i64 size){
72   return id->pMethods->xTruncate(id, size);
73 }
74 int sqlite3OsSync(sqlite3_file *id, int flags){
75   DO_OS_MALLOC_TEST;
76   return id->pMethods->xSync(id, flags);
77 }
78 int sqlite3OsFileSize(sqlite3_file *id, i64 *pSize){
79   DO_OS_MALLOC_TEST;
80   return id->pMethods->xFileSize(id, pSize);
81 }
82 int sqlite3OsLock(sqlite3_file *id, int lockType){
83   DO_OS_MALLOC_TEST;
84   return id->pMethods->xLock(id, lockType);
85 }
86 int sqlite3OsUnlock(sqlite3_file *id, int lockType){
87   return id->pMethods->xUnlock(id, lockType);
88 }
89 int sqlite3OsCheckReservedLock(sqlite3_file *id, int *pResOut){
90   DO_OS_MALLOC_TEST;
91   return id->pMethods->xCheckReservedLock(id, pResOut);
92 }
93 int sqlite3OsFileControl(sqlite3_file *id, int op, void *pArg){
94   return id->pMethods->xFileControl(id, op, pArg);
95 }
96 int sqlite3OsSectorSize(sqlite3_file *id){
97   int (*xSectorSize)(sqlite3_file*) = id->pMethods->xSectorSize;
98   return (xSectorSize ? xSectorSize(id) : SQLITE_DEFAULT_SECTOR_SIZE);
99 }
100 int sqlite3OsDeviceCharacteristics(sqlite3_file *id){
101   return id->pMethods->xDeviceCharacteristics(id);
102 }
103 
104 /*
105 ** The next group of routines are convenience wrappers around the
106 ** VFS methods.
107 */
108 int sqlite3OsOpen(
109   sqlite3_vfs *pVfs,
110   const char *zPath,
111   sqlite3_file *pFile,
112   int flags,
113   int *pFlagsOut
114 ){
115   int rc;
116   DO_OS_MALLOC_TEST;
117   rc = pVfs->xOpen(pVfs, zPath, pFile, flags, pFlagsOut);
118   assert( rc==SQLITE_OK || pFile->pMethods==0 );
119   return rc;
120 }
121 int sqlite3OsDelete(sqlite3_vfs *pVfs, const char *zPath, int dirSync){
122   return pVfs->xDelete(pVfs, zPath, dirSync);
123 }
124 int sqlite3OsAccess(
125   sqlite3_vfs *pVfs,
126   const char *zPath,
127   int flags,
128   int *pResOut
129 ){
130   DO_OS_MALLOC_TEST;
131   return pVfs->xAccess(pVfs, zPath, flags, pResOut);
132 }
133 int sqlite3OsFullPathname(
134   sqlite3_vfs *pVfs,
135   const char *zPath,
136   int nPathOut,
137   char *zPathOut
138 ){
139   return pVfs->xFullPathname(pVfs, zPath, nPathOut, zPathOut);
140 }
141 #ifndef SQLITE_OMIT_LOAD_EXTENSION
142 void *sqlite3OsDlOpen(sqlite3_vfs *pVfs, const char *zPath){
143   return pVfs->xDlOpen(pVfs, zPath);
144 }
145 void sqlite3OsDlError(sqlite3_vfs *pVfs, int nByte, char *zBufOut){
146   pVfs->xDlError(pVfs, nByte, zBufOut);
147 }
148 void (*sqlite3OsDlSym(sqlite3_vfs *pVfs, void *pHdle, const char *zSym))(void){
149   return pVfs->xDlSym(pVfs, pHdle, zSym);
150 }
151 void sqlite3OsDlClose(sqlite3_vfs *pVfs, void *pHandle){
152   pVfs->xDlClose(pVfs, pHandle);
153 }
154 #endif /* SQLITE_OMIT_LOAD_EXTENSION */
155 int sqlite3OsRandomness(sqlite3_vfs *pVfs, int nByte, char *zBufOut){
156   return pVfs->xRandomness(pVfs, nByte, zBufOut);
157 }
158 int sqlite3OsSleep(sqlite3_vfs *pVfs, int nMicro){
159   return pVfs->xSleep(pVfs, nMicro);
160 }
161 int sqlite3OsCurrentTime(sqlite3_vfs *pVfs, double *pTimeOut){
162   return pVfs->xCurrentTime(pVfs, pTimeOut);
163 }
164 
165 int sqlite3OsOpenMalloc(
166   sqlite3_vfs *pVfs,
167   const char *zFile,
168   sqlite3_file **ppFile,
169   int flags,
170   int *pOutFlags
171 ){
172   int rc = SQLITE_NOMEM;
173   sqlite3_file *pFile;
174   pFile = (sqlite3_file *)sqlite3Malloc(pVfs->szOsFile);
175   if( pFile ){
176     rc = sqlite3OsOpen(pVfs, zFile, pFile, flags, pOutFlags);
177     if( rc!=SQLITE_OK ){
178       sqlite3_free(pFile);
179     }else{
180       *ppFile = pFile;
181     }
182   }
183   return rc;
184 }
185 int sqlite3OsCloseFree(sqlite3_file *pFile){
186   int rc = SQLITE_OK;
187   assert( pFile );
188   rc = sqlite3OsClose(pFile);
189   sqlite3_free(pFile);
190   return rc;
191 }
192 
193 /*
194 ** The list of all registered VFS implementations.
195 */
196 static sqlite3_vfs * SQLITE_WSD vfsList = 0;
197 #define vfsList GLOBAL(sqlite3_vfs *, vfsList)
198 
199 /*
200 ** Locate a VFS by name.  If no name is given, simply return the
201 ** first VFS on the list.
202 */
203 sqlite3_vfs *sqlite3_vfs_find(const char *zVfs){
204   sqlite3_vfs *pVfs = 0;
205 #if SQLITE_THREADSAFE
206   sqlite3_mutex *mutex;
207 #endif
208 #ifndef SQLITE_OMIT_AUTOINIT
209   int rc = sqlite3_initialize();
210   if( rc ) return 0;
211 #endif
212 #if SQLITE_THREADSAFE
213   mutex = sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_MASTER);
214 #endif
215   sqlite3_mutex_enter(mutex);
216   for(pVfs = vfsList; pVfs; pVfs=pVfs->pNext){
217     if( zVfs==0 ) break;
218     if( strcmp(zVfs, pVfs->zName)==0 ) break;
219   }
220   sqlite3_mutex_leave(mutex);
221   return pVfs;
222 }
223 
224 /*
225 ** Unlink a VFS from the linked list
226 */
227 static void vfsUnlink(sqlite3_vfs *pVfs){
228   assert( sqlite3_mutex_held(sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_MASTER)) );
229   if( pVfs==0 ){
230     /* No-op */
231   }else if( vfsList==pVfs ){
232     vfsList = pVfs->pNext;
233   }else if( vfsList ){
234     sqlite3_vfs *p = vfsList;
235     while( p->pNext && p->pNext!=pVfs ){
236       p = p->pNext;
237     }
238     if( p->pNext==pVfs ){
239       p->pNext = pVfs->pNext;
240     }
241   }
242 }
243 
244 /*
245 ** Register a VFS with the system.  It is harmless to register the same
246 ** VFS multiple times.  The new VFS becomes the default if makeDflt is
247 ** true.
248 */
249 int sqlite3_vfs_register(sqlite3_vfs *pVfs, int makeDflt){
250   sqlite3_mutex *mutex = 0;
251 #ifndef SQLITE_OMIT_AUTOINIT
252   int rc = sqlite3_initialize();
253   if( rc ) return rc;
254 #endif
255   mutex = sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_MASTER);
256   sqlite3_mutex_enter(mutex);
257   vfsUnlink(pVfs);
258   if( makeDflt || vfsList==0 ){
259     pVfs->pNext = vfsList;
260     vfsList = pVfs;
261   }else{
262     pVfs->pNext = vfsList->pNext;
263     vfsList->pNext = pVfs;
264   }
265   assert(vfsList);
266   sqlite3_mutex_leave(mutex);
267   return SQLITE_OK;
268 }
269 
270 /*
271 ** Unregister a VFS so that it is no longer accessible.
272 */
273 int sqlite3_vfs_unregister(sqlite3_vfs *pVfs){
274 #if SQLITE_THREADSAFE
275   sqlite3_mutex *mutex = sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_MASTER);
276 #endif
277   sqlite3_mutex_enter(mutex);
278   vfsUnlink(pVfs);
279   sqlite3_mutex_leave(mutex);
280   return SQLITE_OK;
281 }
282