blob: 8ca0185d99ec95839371719ccdba69cd698bb629 [file] [log] [blame]
drh054889e2005-11-30 03:20:31 +00001/*
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*/
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/*
21** The following routines are convenience wrappers around methods
danielk197762079062007-08-15 17:08:46 +000022** of the sqlite3_file object. This is mostly just syntactic sugar. All
drh054889e2005-11-30 03:20:31 +000023** of this would be completely automatic if SQLite were coded using
24** C++ instead of plain old C.
25*/
danielk1977b4b47412007-08-17 15:53:36 +000026int sqlite3OsClose(sqlite3_file *pId){
danielk1977c7b60172007-08-22 11:22:03 +000027 int rc = SQLITE_OK;
28 if( pId->pMethods ){
29 rc = pId->pMethods->xClose(pId);
30 pId->pMethods = 0;
31 }
32 return rc;
drh054889e2005-11-30 03:20:31 +000033}
danielk197762079062007-08-15 17:08:46 +000034int sqlite3OsRead(sqlite3_file *id, void *pBuf, int amt, i64 offset){
35 return id->pMethods->xRead(id, pBuf, amt, offset);
drh054889e2005-11-30 03:20:31 +000036}
danielk197762079062007-08-15 17:08:46 +000037int sqlite3OsWrite(sqlite3_file *id, const void *pBuf, int amt, i64 offset){
38 return id->pMethods->xWrite(id, pBuf, amt, offset);
drh054889e2005-11-30 03:20:31 +000039}
danielk197762079062007-08-15 17:08:46 +000040int sqlite3OsTruncate(sqlite3_file *id, i64 size){
41 return id->pMethods->xTruncate(id, size);
drh054889e2005-11-30 03:20:31 +000042}
danielk197790949c22007-08-17 16:50:38 +000043int sqlite3OsSync(sqlite3_file *id, int flags){
44 return id->pMethods->xSync(id, flags);
drh054889e2005-11-30 03:20:31 +000045}
danielk197762079062007-08-15 17:08:46 +000046int sqlite3OsFileSize(sqlite3_file *id, i64 *pSize){
47 return id->pMethods->xFileSize(id, pSize);
drh054889e2005-11-30 03:20:31 +000048}
danielk197762079062007-08-15 17:08:46 +000049int sqlite3OsLock(sqlite3_file *id, int lockType){
50 return id->pMethods->xLock(id, lockType);
drh054889e2005-11-30 03:20:31 +000051}
danielk197762079062007-08-15 17:08:46 +000052int sqlite3OsUnlock(sqlite3_file *id, int lockType){
53 return id->pMethods->xUnlock(id, lockType);
drh054889e2005-11-30 03:20:31 +000054}
danielk197762079062007-08-15 17:08:46 +000055int sqlite3OsCheckReservedLock(sqlite3_file *id){
56 return id->pMethods->xCheckReservedLock(id);
drh054889e2005-11-30 03:20:31 +000057}
drhcc6bb3e2007-08-31 16:11:35 +000058int sqlite3OsFileControl(sqlite3_file *id, int op, void *pArg){
59 return id->pMethods->xFileControl(id,op,pArg);
60}
danielk19772ca0f862007-08-23 08:06:44 +000061
62#ifdef SQLITE_TEST
63 /* The following two variables are used to override the values returned
64 ** by the xSectorSize() and xDeviceCharacteristics() vfs methods for
65 ** testing purposes. They are usually set by a test command implemented
66 ** in test6.c.
67 */
68 int sqlite3_test_sector_size = 0;
69 int sqlite3_test_device_characteristics = 0;
70 int sqlite3OsDeviceCharacteristics(sqlite3_file *id){
71 int dc = id->pMethods->xDeviceCharacteristics(id);
72 return dc | sqlite3_test_device_characteristics;
73 }
74 int sqlite3OsSectorSize(sqlite3_file *id){
75 if( sqlite3_test_sector_size==0 ){
76 int (*xSectorSize)(sqlite3_file*) = id->pMethods->xSectorSize;
77 return (xSectorSize ? xSectorSize(id) : SQLITE_DEFAULT_SECTOR_SIZE);
78 }
79 return sqlite3_test_sector_size;
80 }
81#else
82 int sqlite3OsSectorSize(sqlite3_file *id){
83 int (*xSectorSize)(sqlite3_file*) = id->pMethods->xSectorSize;
84 return (xSectorSize ? xSectorSize(id) : SQLITE_DEFAULT_SECTOR_SIZE);
85 }
86 int sqlite3OsDeviceCharacteristics(sqlite3_file *id){
87 return id->pMethods->xDeviceCharacteristics(id);
88 }
89#endif
drh3f459022006-01-07 16:06:07 +000090
drh87cc3b32007-05-08 21:45:27 +000091#if defined(SQLITE_TEST) || defined(SQLITE_DEBUG)
92 /* These methods are currently only used for testing and debugging. */
danielk197762079062007-08-15 17:08:46 +000093 int sqlite3OsFileHandle(sqlite3_file *id){
94 /* return id->pMethods->xFileHandle(id); */
95 return 0;
drh87cc3b32007-05-08 21:45:27 +000096 }
danielk197762079062007-08-15 17:08:46 +000097 int sqlite3OsLockState(sqlite3_file *id){
danielk197790949c22007-08-17 16:50:38 +000098 return id->pMethods->xLockState(id);
drh87cc3b32007-05-08 21:45:27 +000099 }
100#endif
101
drhd677b3d2007-08-20 22:48:41 +0000102/*
103** The next group of routines are convenience wrappers around the
104** VFS methods.
105*/
danielk1977b4b47412007-08-17 15:53:36 +0000106int sqlite3OsOpen(
107 sqlite3_vfs *pVfs,
108 const char *zPath,
109 sqlite3_file *pFile,
110 int flags,
111 int *pFlagsOut
112){
drh153c62c2007-08-24 03:51:33 +0000113 return pVfs->xOpen(pVfs, zPath, pFile, flags, pFlagsOut);
drh3f459022006-01-07 16:06:07 +0000114}
danielk1977fee2d252007-08-18 10:59:19 +0000115int sqlite3OsDelete(sqlite3_vfs *pVfs, const char *zPath, int dirSync){
drh153c62c2007-08-24 03:51:33 +0000116 return pVfs->xDelete(pVfs, zPath, dirSync);
danielk1977b4b47412007-08-17 15:53:36 +0000117}
118int sqlite3OsAccess(sqlite3_vfs *pVfs, const char *zPath, int flags){
drh153c62c2007-08-24 03:51:33 +0000119 return pVfs->xAccess(pVfs, zPath, flags);
danielk1977b4b47412007-08-17 15:53:36 +0000120}
121int sqlite3OsGetTempName(sqlite3_vfs *pVfs, char *zBufOut){
drh153c62c2007-08-24 03:51:33 +0000122 return pVfs->xGetTempName(pVfs, zBufOut);
danielk1977b4b47412007-08-17 15:53:36 +0000123}
124int sqlite3OsFullPathname(sqlite3_vfs *pVfs, const char *zPath, char *zPathOut){
drh153c62c2007-08-24 03:51:33 +0000125 return pVfs->xFullPathname(pVfs, zPath, zPathOut);
danielk1977b4b47412007-08-17 15:53:36 +0000126}
127void *sqlite3OsDlOpen(sqlite3_vfs *pVfs, const char *zPath){
drh153c62c2007-08-24 03:51:33 +0000128 return pVfs->xDlOpen(pVfs, zPath);
danielk1977b4b47412007-08-17 15:53:36 +0000129}
130void sqlite3OsDlError(sqlite3_vfs *pVfs, int nByte, char *zBufOut){
drh153c62c2007-08-24 03:51:33 +0000131 pVfs->xDlError(pVfs, nByte, zBufOut);
danielk1977b4b47412007-08-17 15:53:36 +0000132}
133void *sqlite3OsDlSym(sqlite3_vfs *pVfs, void *pHandle, const char *zSymbol){
drh153c62c2007-08-24 03:51:33 +0000134 return pVfs->xDlSym(pVfs, pHandle, zSymbol);
danielk1977b4b47412007-08-17 15:53:36 +0000135}
136void sqlite3OsDlClose(sqlite3_vfs *pVfs, void *pHandle){
drh153c62c2007-08-24 03:51:33 +0000137 pVfs->xDlClose(pVfs, pHandle);
danielk1977b4b47412007-08-17 15:53:36 +0000138}
139int sqlite3OsRandomness(sqlite3_vfs *pVfs, int nByte, char *zBufOut){
drh153c62c2007-08-24 03:51:33 +0000140 return pVfs->xRandomness(pVfs, nByte, zBufOut);
danielk1977b4b47412007-08-17 15:53:36 +0000141}
142int sqlite3OsSleep(sqlite3_vfs *pVfs, int nMicro){
drh153c62c2007-08-24 03:51:33 +0000143 return pVfs->xSleep(pVfs, nMicro);
danielk1977b4b47412007-08-17 15:53:36 +0000144}
145int sqlite3OsCurrentTime(sqlite3_vfs *pVfs, double *pTimeOut){
drh153c62c2007-08-24 03:51:33 +0000146 return pVfs->xCurrentTime(pVfs, pTimeOut);
danielk1977b4b47412007-08-17 15:53:36 +0000147}
148
149int sqlite3OsOpenMalloc(
150 sqlite3_vfs *pVfs,
151 const char *zFile,
152 sqlite3_file **ppFile,
danielk1977967a4a12007-08-20 14:23:44 +0000153 int flags,
154 int *pOutFlags
danielk1977b4b47412007-08-17 15:53:36 +0000155){
156 int rc = SQLITE_NOMEM;
157 sqlite3_file *pFile;
158 pFile = (sqlite3_file *)sqlite3_malloc(pVfs->szOsFile);
159 if( pFile ){
danielk1977967a4a12007-08-20 14:23:44 +0000160 rc = sqlite3OsOpen(pVfs, zFile, pFile, flags, pOutFlags);
danielk1977b4b47412007-08-17 15:53:36 +0000161 if( rc!=SQLITE_OK ){
162 sqlite3_free(pFile);
163 }else{
164 *ppFile = pFile;
165 }
166 }
167 return rc;
168}
169int sqlite3OsCloseFree(sqlite3_file *pFile){
170 int rc = SQLITE_OK;
171 if( pFile ){
172 rc = sqlite3OsClose(pFile);
173 sqlite3_free(pFile);
174 }
175 return rc;
176}
177
drhd677b3d2007-08-20 22:48:41 +0000178/*
drh153c62c2007-08-24 03:51:33 +0000179** The list of all registered VFS implementations. This list is
180** initialized to the single VFS returned by sqlite3OsDefaultVfs()
181** upon the first call to sqlite3_vfs_find().
drhd677b3d2007-08-20 22:48:41 +0000182*/
drh153c62c2007-08-24 03:51:33 +0000183static sqlite3_vfs *vfsList = 0;
drhd677b3d2007-08-20 22:48:41 +0000184
185/*
186** Locate a VFS by name. If no name is given, simply return the
187** first VFS on the list.
188*/
189sqlite3_vfs *sqlite3_vfs_find(const char *zVfs){
190 sqlite3_mutex *mutex = sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_MASTER);
191 sqlite3_vfs *pVfs;
drh153c62c2007-08-24 03:51:33 +0000192 static int isInit = 0;
drhd677b3d2007-08-20 22:48:41 +0000193 sqlite3_mutex_enter(mutex);
drh153c62c2007-08-24 03:51:33 +0000194 if( !isInit ){
195 vfsList = sqlite3OsDefaultVfs();
196 isInit = 1;
197 }
drhd677b3d2007-08-20 22:48:41 +0000198 for(pVfs = vfsList; pVfs; pVfs=pVfs->pNext){
199 if( zVfs==0 ) break;
200 if( strcmp(zVfs, pVfs->zName)==0 ) break;
201 }
drhd677b3d2007-08-20 22:48:41 +0000202 sqlite3_mutex_leave(mutex);
203 return pVfs;
danielk1977b4b47412007-08-17 15:53:36 +0000204}
205
drhd677b3d2007-08-20 22:48:41 +0000206/*
drhd677b3d2007-08-20 22:48:41 +0000207** Unlink a VFS from the linked list
208*/
209static void vfsUnlink(sqlite3_vfs *pVfs){
210 assert( sqlite3_mutex_held(sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_MASTER)) );
211 if( vfsList==pVfs ){
212 vfsList = pVfs->pNext;
213 }else{
214 sqlite3_vfs *p = vfsList;
215 while( p->pNext && p->pNext!=pVfs ){
216 p = p->pNext;
217 }
218 if( p->pNext==pVfs ){
219 p->pNext = pVfs->pNext;
220 }
221 }
222}
223
224/*
225** Register a VFS with the system. It is harmless to register the same
226** VFS multiple times. The new VFS becomes the default if makeDflt is
227** true.
228*/
229int sqlite3_vfs_register(sqlite3_vfs *pVfs, int makeDflt){
230 sqlite3_mutex *mutex = sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_MASTER);
231 sqlite3_mutex_enter(mutex);
232 vfsUnlink(pVfs);
233 if( makeDflt || vfsList==0 ){
234 pVfs->pNext = vfsList;
235 vfsList = pVfs;
236 }else{
237 pVfs->pNext = vfsList->pNext;
238 pVfs->pNext = pVfs;
239 }
danielk1977f1da17a2007-08-21 13:07:46 +0000240 assert(vfsList);
drhd677b3d2007-08-20 22:48:41 +0000241 sqlite3_mutex_leave(mutex);
242 return SQLITE_OK;
243}
244
245/*
246** Unregister a VFS so that it is no longer accessible.
247*/
248int sqlite3_vfs_unregister(sqlite3_vfs *pVfs){
249 sqlite3_mutex *mutex = sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_MASTER);
250 sqlite3_mutex_enter(mutex);
251 vfsUnlink(pVfs);
danielk1977f1da17a2007-08-21 13:07:46 +0000252 assert(vfsList);
drhd677b3d2007-08-20 22:48:41 +0000253 sqlite3_mutex_leave(mutex);
254 return SQLITE_OK;
255}