blob: 57e2d0fdd0bfbe6fdd5a0aa61ed91652a083ccda [file] [log] [blame]
drhe9bb50a2007-09-02 17:52:04 +00001 /*
drh054889e2005-11-30 03:20:31 +00002** 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*/
drh3f459022006-01-07 16:06:07 +000016#define _SQLITE_OS_C_ 1
drh054889e2005-11-30 03:20:31 +000017#include "sqliteInt.h"
drhbd08af42007-04-05 21:58:33 +000018#undef _SQLITE_OS_C_
drh054889e2005-11-30 03:20:31 +000019
20/*
danielk1977ae72d982007-10-03 08:46:44 +000021** The default SQLite sqlite3_vfs implementations do not allocate
22** memory (actually, os_unix.c allocates a small amount of memory
23** from within OsOpen()), but some third-party implementations may.
24** So we test the effects of a malloc() failing and the sqlite3OsXXX()
25** function returning SQLITE_IOERR_NOMEM using the DO_OS_MALLOC_TEST macro.
26**
27** The following functions are instrumented for malloc() failure
28** testing:
29**
30** sqlite3OsOpen()
31** sqlite3OsRead()
32** sqlite3OsWrite()
33** sqlite3OsSync()
34** sqlite3OsLock()
35**
36*/
37#ifdef SQLITE_TEST
38 #define DO_OS_MALLOC_TEST if (1) { \
39 void *pTstAlloc = sqlite3_malloc(10); \
40 if (!pTstAlloc) return SQLITE_IOERR_NOMEM; \
41 sqlite3_free(pTstAlloc); \
42 }
43#else
44 #define DO_OS_MALLOC_TEST
45#endif
46
47/*
drh054889e2005-11-30 03:20:31 +000048** The following routines are convenience wrappers around methods
danielk197762079062007-08-15 17:08:46 +000049** of the sqlite3_file object. This is mostly just syntactic sugar. All
drh054889e2005-11-30 03:20:31 +000050** of this would be completely automatic if SQLite were coded using
51** C++ instead of plain old C.
52*/
danielk1977b4b47412007-08-17 15:53:36 +000053int sqlite3OsClose(sqlite3_file *pId){
danielk1977c7b60172007-08-22 11:22:03 +000054 int rc = SQLITE_OK;
55 if( pId->pMethods ){
56 rc = pId->pMethods->xClose(pId);
57 pId->pMethods = 0;
58 }
59 return rc;
drh054889e2005-11-30 03:20:31 +000060}
danielk197762079062007-08-15 17:08:46 +000061int sqlite3OsRead(sqlite3_file *id, void *pBuf, int amt, i64 offset){
danielk1977ae72d982007-10-03 08:46:44 +000062 DO_OS_MALLOC_TEST;
danielk197762079062007-08-15 17:08:46 +000063 return id->pMethods->xRead(id, pBuf, amt, offset);
drh054889e2005-11-30 03:20:31 +000064}
danielk197762079062007-08-15 17:08:46 +000065int sqlite3OsWrite(sqlite3_file *id, const void *pBuf, int amt, i64 offset){
danielk1977ae72d982007-10-03 08:46:44 +000066 DO_OS_MALLOC_TEST;
danielk197762079062007-08-15 17:08:46 +000067 return id->pMethods->xWrite(id, pBuf, amt, offset);
drh054889e2005-11-30 03:20:31 +000068}
danielk197762079062007-08-15 17:08:46 +000069int sqlite3OsTruncate(sqlite3_file *id, i64 size){
70 return id->pMethods->xTruncate(id, size);
drh054889e2005-11-30 03:20:31 +000071}
danielk197790949c22007-08-17 16:50:38 +000072int sqlite3OsSync(sqlite3_file *id, int flags){
danielk1977ae72d982007-10-03 08:46:44 +000073 DO_OS_MALLOC_TEST;
danielk197790949c22007-08-17 16:50:38 +000074 return id->pMethods->xSync(id, flags);
drh054889e2005-11-30 03:20:31 +000075}
danielk197762079062007-08-15 17:08:46 +000076int sqlite3OsFileSize(sqlite3_file *id, i64 *pSize){
77 return id->pMethods->xFileSize(id, pSize);
drh054889e2005-11-30 03:20:31 +000078}
danielk197762079062007-08-15 17:08:46 +000079int sqlite3OsLock(sqlite3_file *id, int lockType){
danielk1977ae72d982007-10-03 08:46:44 +000080 DO_OS_MALLOC_TEST;
danielk197762079062007-08-15 17:08:46 +000081 return id->pMethods->xLock(id, lockType);
drh054889e2005-11-30 03:20:31 +000082}
danielk197762079062007-08-15 17:08:46 +000083int sqlite3OsUnlock(sqlite3_file *id, int lockType){
84 return id->pMethods->xUnlock(id, lockType);
drh054889e2005-11-30 03:20:31 +000085}
danielk197762079062007-08-15 17:08:46 +000086int sqlite3OsCheckReservedLock(sqlite3_file *id){
87 return id->pMethods->xCheckReservedLock(id);
drh054889e2005-11-30 03:20:31 +000088}
drhcc6bb3e2007-08-31 16:11:35 +000089int sqlite3OsFileControl(sqlite3_file *id, int op, void *pArg){
90 return id->pMethods->xFileControl(id,op,pArg);
91}
danielk1977bf260972008-01-22 11:50:13 +000092int sqlite3OsSectorSize(sqlite3_file *id){
93 int (*xSectorSize)(sqlite3_file*) = id->pMethods->xSectorSize;
94 return (xSectorSize ? xSectorSize(id) : SQLITE_DEFAULT_SECTOR_SIZE);
95}
96int sqlite3OsDeviceCharacteristics(sqlite3_file *id){
97 return id->pMethods->xDeviceCharacteristics(id);
98}
drh3f459022006-01-07 16:06:07 +000099
drhd677b3d2007-08-20 22:48:41 +0000100/*
101** The next group of routines are convenience wrappers around the
102** VFS methods.
103*/
danielk1977b4b47412007-08-17 15:53:36 +0000104int sqlite3OsOpen(
105 sqlite3_vfs *pVfs,
106 const char *zPath,
107 sqlite3_file *pFile,
108 int flags,
109 int *pFlagsOut
110){
danielk1977ae72d982007-10-03 08:46:44 +0000111 DO_OS_MALLOC_TEST;
drh153c62c2007-08-24 03:51:33 +0000112 return pVfs->xOpen(pVfs, zPath, pFile, flags, pFlagsOut);
drh3f459022006-01-07 16:06:07 +0000113}
danielk1977fee2d252007-08-18 10:59:19 +0000114int sqlite3OsDelete(sqlite3_vfs *pVfs, const char *zPath, int dirSync){
drh153c62c2007-08-24 03:51:33 +0000115 return pVfs->xDelete(pVfs, zPath, dirSync);
danielk1977b4b47412007-08-17 15:53:36 +0000116}
117int sqlite3OsAccess(sqlite3_vfs *pVfs, const char *zPath, int flags){
drh153c62c2007-08-24 03:51:33 +0000118 return pVfs->xAccess(pVfs, zPath, flags);
danielk1977b4b47412007-08-17 15:53:36 +0000119}
danielk1977adfb9b02007-09-17 07:02:56 +0000120int sqlite3OsGetTempname(sqlite3_vfs *pVfs, int nBufOut, char *zBufOut){
121 return pVfs->xGetTempname(pVfs, nBufOut, zBufOut);
danielk1977b4b47412007-08-17 15:53:36 +0000122}
danielk1977adfb9b02007-09-17 07:02:56 +0000123int sqlite3OsFullPathname(
124 sqlite3_vfs *pVfs,
125 const char *zPath,
126 int nPathOut,
127 char *zPathOut
128){
129 return pVfs->xFullPathname(pVfs, zPath, nPathOut, zPathOut);
danielk1977b4b47412007-08-17 15:53:36 +0000130}
131void *sqlite3OsDlOpen(sqlite3_vfs *pVfs, const char *zPath){
drh153c62c2007-08-24 03:51:33 +0000132 return pVfs->xDlOpen(pVfs, zPath);
danielk1977b4b47412007-08-17 15:53:36 +0000133}
134void sqlite3OsDlError(sqlite3_vfs *pVfs, int nByte, char *zBufOut){
drh153c62c2007-08-24 03:51:33 +0000135 pVfs->xDlError(pVfs, nByte, zBufOut);
danielk1977b4b47412007-08-17 15:53:36 +0000136}
137void *sqlite3OsDlSym(sqlite3_vfs *pVfs, void *pHandle, const char *zSymbol){
drh153c62c2007-08-24 03:51:33 +0000138 return pVfs->xDlSym(pVfs, pHandle, zSymbol);
danielk1977b4b47412007-08-17 15:53:36 +0000139}
140void sqlite3OsDlClose(sqlite3_vfs *pVfs, void *pHandle){
drh153c62c2007-08-24 03:51:33 +0000141 pVfs->xDlClose(pVfs, pHandle);
danielk1977b4b47412007-08-17 15:53:36 +0000142}
143int sqlite3OsRandomness(sqlite3_vfs *pVfs, int nByte, char *zBufOut){
drh153c62c2007-08-24 03:51:33 +0000144 return pVfs->xRandomness(pVfs, nByte, zBufOut);
danielk1977b4b47412007-08-17 15:53:36 +0000145}
146int sqlite3OsSleep(sqlite3_vfs *pVfs, int nMicro){
drh153c62c2007-08-24 03:51:33 +0000147 return pVfs->xSleep(pVfs, nMicro);
danielk1977b4b47412007-08-17 15:53:36 +0000148}
149int sqlite3OsCurrentTime(sqlite3_vfs *pVfs, double *pTimeOut){
drh153c62c2007-08-24 03:51:33 +0000150 return pVfs->xCurrentTime(pVfs, pTimeOut);
danielk1977b4b47412007-08-17 15:53:36 +0000151}
152
153int sqlite3OsOpenMalloc(
154 sqlite3_vfs *pVfs,
155 const char *zFile,
156 sqlite3_file **ppFile,
danielk1977967a4a12007-08-20 14:23:44 +0000157 int flags,
158 int *pOutFlags
danielk1977b4b47412007-08-17 15:53:36 +0000159){
160 int rc = SQLITE_NOMEM;
161 sqlite3_file *pFile;
162 pFile = (sqlite3_file *)sqlite3_malloc(pVfs->szOsFile);
163 if( pFile ){
danielk1977967a4a12007-08-20 14:23:44 +0000164 rc = sqlite3OsOpen(pVfs, zFile, pFile, flags, pOutFlags);
danielk1977b4b47412007-08-17 15:53:36 +0000165 if( rc!=SQLITE_OK ){
166 sqlite3_free(pFile);
167 }else{
168 *ppFile = pFile;
169 }
170 }
171 return rc;
172}
173int sqlite3OsCloseFree(sqlite3_file *pFile){
174 int rc = SQLITE_OK;
175 if( pFile ){
176 rc = sqlite3OsClose(pFile);
177 sqlite3_free(pFile);
178 }
179 return rc;
180}
181
drhd677b3d2007-08-20 22:48:41 +0000182/*
drh153c62c2007-08-24 03:51:33 +0000183** The list of all registered VFS implementations. This list is
184** initialized to the single VFS returned by sqlite3OsDefaultVfs()
185** upon the first call to sqlite3_vfs_find().
drhd677b3d2007-08-20 22:48:41 +0000186*/
drh153c62c2007-08-24 03:51:33 +0000187static sqlite3_vfs *vfsList = 0;
drhd677b3d2007-08-20 22:48:41 +0000188
189/*
190** Locate a VFS by name. If no name is given, simply return the
191** first VFS on the list.
192*/
193sqlite3_vfs *sqlite3_vfs_find(const char *zVfs){
drh7e8b8482008-01-23 03:03:05 +0000194#ifndef SQLITE_MUTEX_NOOP
drhd677b3d2007-08-20 22:48:41 +0000195 sqlite3_mutex *mutex = sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_MASTER);
drh7e8b8482008-01-23 03:03:05 +0000196#endif
drh9bc54492007-10-23 14:49:59 +0000197 sqlite3_vfs *pVfs = 0;
drh153c62c2007-08-24 03:51:33 +0000198 static int isInit = 0;
drhd677b3d2007-08-20 22:48:41 +0000199 sqlite3_mutex_enter(mutex);
drh153c62c2007-08-24 03:51:33 +0000200 if( !isInit ){
201 vfsList = sqlite3OsDefaultVfs();
202 isInit = 1;
203 }
drhd677b3d2007-08-20 22:48:41 +0000204 for(pVfs = vfsList; pVfs; pVfs=pVfs->pNext){
205 if( zVfs==0 ) break;
206 if( strcmp(zVfs, pVfs->zName)==0 ) break;
207 }
drhd677b3d2007-08-20 22:48:41 +0000208 sqlite3_mutex_leave(mutex);
209 return pVfs;
danielk1977b4b47412007-08-17 15:53:36 +0000210}
211
drhd677b3d2007-08-20 22:48:41 +0000212/*
drhd677b3d2007-08-20 22:48:41 +0000213** Unlink a VFS from the linked list
214*/
215static void vfsUnlink(sqlite3_vfs *pVfs){
216 assert( sqlite3_mutex_held(sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_MASTER)) );
drh9bc54492007-10-23 14:49:59 +0000217 if( pVfs==0 ){
218 /* No-op */
219 }else if( vfsList==pVfs ){
drhd677b3d2007-08-20 22:48:41 +0000220 vfsList = pVfs->pNext;
drh9bc54492007-10-23 14:49:59 +0000221 }else if( vfsList ){
drhd677b3d2007-08-20 22:48:41 +0000222 sqlite3_vfs *p = vfsList;
223 while( p->pNext && p->pNext!=pVfs ){
224 p = p->pNext;
225 }
226 if( p->pNext==pVfs ){
227 p->pNext = pVfs->pNext;
228 }
229 }
230}
231
232/*
233** Register a VFS with the system. It is harmless to register the same
234** VFS multiple times. The new VFS becomes the default if makeDflt is
235** true.
236*/
237int sqlite3_vfs_register(sqlite3_vfs *pVfs, int makeDflt){
drh7e8b8482008-01-23 03:03:05 +0000238#ifndef SQLITE_MUTEX_NOOP
drhd677b3d2007-08-20 22:48:41 +0000239 sqlite3_mutex *mutex = sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_MASTER);
drh7e8b8482008-01-23 03:03:05 +0000240#endif
drhe9bb50a2007-09-02 17:52:04 +0000241 sqlite3_vfs_find(0); /* Make sure we are initialized */
drhd677b3d2007-08-20 22:48:41 +0000242 sqlite3_mutex_enter(mutex);
243 vfsUnlink(pVfs);
244 if( makeDflt || vfsList==0 ){
245 pVfs->pNext = vfsList;
246 vfsList = pVfs;
247 }else{
248 pVfs->pNext = vfsList->pNext;
danielk197795c8a542007-09-01 06:51:27 +0000249 vfsList->pNext = pVfs;
drhd677b3d2007-08-20 22:48:41 +0000250 }
danielk1977f1da17a2007-08-21 13:07:46 +0000251 assert(vfsList);
drhd677b3d2007-08-20 22:48:41 +0000252 sqlite3_mutex_leave(mutex);
253 return SQLITE_OK;
254}
255
256/*
257** Unregister a VFS so that it is no longer accessible.
258*/
259int sqlite3_vfs_unregister(sqlite3_vfs *pVfs){
drh7e8b8482008-01-23 03:03:05 +0000260#ifndef SQLITE_MUTEX_NOOP
drhd677b3d2007-08-20 22:48:41 +0000261 sqlite3_mutex *mutex = sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_MASTER);
drh7e8b8482008-01-23 03:03:05 +0000262#endif
drhd677b3d2007-08-20 22:48:41 +0000263 sqlite3_mutex_enter(mutex);
264 vfsUnlink(pVfs);
265 sqlite3_mutex_leave(mutex);
266 return SQLITE_OK;
267}