blob: ebac1ba999f9675d29d9ebbdd48631769fdf8a80 [file] [log] [blame]
danielk1977a7a8e142008-02-13 18:25:27 +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){
drh19db9352008-03-27 22:42:51 +0000118 int rc;
119#ifdef SQLITE_TEST
120 void *pTstAlloc = sqlite3_malloc(10);
121 if (!pTstAlloc) return -1;
122 sqlite3_free(pTstAlloc);
123#endif
124 rc = pVfs->xAccess(pVfs, zPath, flags);
drh7e2e6dc2008-03-26 17:18:48 +0000125 assert( rc==0 || rc==1 );
126 return rc;
danielk1977b4b47412007-08-17 15:53:36 +0000127}
danielk1977adfb9b02007-09-17 07:02:56 +0000128int sqlite3OsGetTempname(sqlite3_vfs *pVfs, int nBufOut, char *zBufOut){
129 return pVfs->xGetTempname(pVfs, nBufOut, zBufOut);
danielk1977b4b47412007-08-17 15:53:36 +0000130}
danielk1977adfb9b02007-09-17 07:02:56 +0000131int sqlite3OsFullPathname(
132 sqlite3_vfs *pVfs,
133 const char *zPath,
134 int nPathOut,
135 char *zPathOut
136){
137 return pVfs->xFullPathname(pVfs, zPath, nPathOut, zPathOut);
danielk1977b4b47412007-08-17 15:53:36 +0000138}
139void *sqlite3OsDlOpen(sqlite3_vfs *pVfs, const char *zPath){
drh153c62c2007-08-24 03:51:33 +0000140 return pVfs->xDlOpen(pVfs, zPath);
danielk1977b4b47412007-08-17 15:53:36 +0000141}
142void sqlite3OsDlError(sqlite3_vfs *pVfs, int nByte, char *zBufOut){
drh153c62c2007-08-24 03:51:33 +0000143 pVfs->xDlError(pVfs, nByte, zBufOut);
danielk1977b4b47412007-08-17 15:53:36 +0000144}
145void *sqlite3OsDlSym(sqlite3_vfs *pVfs, void *pHandle, const char *zSymbol){
drh153c62c2007-08-24 03:51:33 +0000146 return pVfs->xDlSym(pVfs, pHandle, zSymbol);
danielk1977b4b47412007-08-17 15:53:36 +0000147}
148void sqlite3OsDlClose(sqlite3_vfs *pVfs, void *pHandle){
drh153c62c2007-08-24 03:51:33 +0000149 pVfs->xDlClose(pVfs, pHandle);
danielk1977b4b47412007-08-17 15:53:36 +0000150}
151int sqlite3OsRandomness(sqlite3_vfs *pVfs, int nByte, char *zBufOut){
drh153c62c2007-08-24 03:51:33 +0000152 return pVfs->xRandomness(pVfs, nByte, zBufOut);
danielk1977b4b47412007-08-17 15:53:36 +0000153}
154int sqlite3OsSleep(sqlite3_vfs *pVfs, int nMicro){
drh153c62c2007-08-24 03:51:33 +0000155 return pVfs->xSleep(pVfs, nMicro);
danielk1977b4b47412007-08-17 15:53:36 +0000156}
157int sqlite3OsCurrentTime(sqlite3_vfs *pVfs, double *pTimeOut){
drh153c62c2007-08-24 03:51:33 +0000158 return pVfs->xCurrentTime(pVfs, pTimeOut);
danielk1977b4b47412007-08-17 15:53:36 +0000159}
160
161int sqlite3OsOpenMalloc(
162 sqlite3_vfs *pVfs,
163 const char *zFile,
164 sqlite3_file **ppFile,
danielk1977967a4a12007-08-20 14:23:44 +0000165 int flags,
166 int *pOutFlags
danielk1977b4b47412007-08-17 15:53:36 +0000167){
168 int rc = SQLITE_NOMEM;
169 sqlite3_file *pFile;
170 pFile = (sqlite3_file *)sqlite3_malloc(pVfs->szOsFile);
171 if( pFile ){
danielk1977967a4a12007-08-20 14:23:44 +0000172 rc = sqlite3OsOpen(pVfs, zFile, pFile, flags, pOutFlags);
danielk1977b4b47412007-08-17 15:53:36 +0000173 if( rc!=SQLITE_OK ){
174 sqlite3_free(pFile);
175 }else{
176 *ppFile = pFile;
177 }
178 }
179 return rc;
180}
181int sqlite3OsCloseFree(sqlite3_file *pFile){
182 int rc = SQLITE_OK;
mlcreech40032982008-04-08 03:09:22 +0000183 assert( pFile );
184 rc = sqlite3OsClose(pFile);
185 sqlite3_free(pFile);
danielk1977b4b47412007-08-17 15:53:36 +0000186 return rc;
187}
188
drhd677b3d2007-08-20 22:48:41 +0000189/*
drh153c62c2007-08-24 03:51:33 +0000190** The list of all registered VFS implementations. This list is
191** initialized to the single VFS returned by sqlite3OsDefaultVfs()
192** upon the first call to sqlite3_vfs_find().
drhd677b3d2007-08-20 22:48:41 +0000193*/
drh153c62c2007-08-24 03:51:33 +0000194static sqlite3_vfs *vfsList = 0;
drhd677b3d2007-08-20 22:48:41 +0000195
196/*
197** Locate a VFS by name. If no name is given, simply return the
198** first VFS on the list.
199*/
200sqlite3_vfs *sqlite3_vfs_find(const char *zVfs){
drh7e8b8482008-01-23 03:03:05 +0000201#ifndef SQLITE_MUTEX_NOOP
drhd677b3d2007-08-20 22:48:41 +0000202 sqlite3_mutex *mutex = sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_MASTER);
drh7e8b8482008-01-23 03:03:05 +0000203#endif
drh9bc54492007-10-23 14:49:59 +0000204 sqlite3_vfs *pVfs = 0;
drh153c62c2007-08-24 03:51:33 +0000205 static int isInit = 0;
drhd677b3d2007-08-20 22:48:41 +0000206 sqlite3_mutex_enter(mutex);
drh153c62c2007-08-24 03:51:33 +0000207 if( !isInit ){
208 vfsList = sqlite3OsDefaultVfs();
209 isInit = 1;
210 }
drhd677b3d2007-08-20 22:48:41 +0000211 for(pVfs = vfsList; pVfs; pVfs=pVfs->pNext){
212 if( zVfs==0 ) break;
213 if( strcmp(zVfs, pVfs->zName)==0 ) break;
214 }
drhd677b3d2007-08-20 22:48:41 +0000215 sqlite3_mutex_leave(mutex);
216 return pVfs;
danielk1977b4b47412007-08-17 15:53:36 +0000217}
218
drhd677b3d2007-08-20 22:48:41 +0000219/*
drhd677b3d2007-08-20 22:48:41 +0000220** Unlink a VFS from the linked list
221*/
222static void vfsUnlink(sqlite3_vfs *pVfs){
223 assert( sqlite3_mutex_held(sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_MASTER)) );
drh9bc54492007-10-23 14:49:59 +0000224 if( pVfs==0 ){
225 /* No-op */
226 }else if( vfsList==pVfs ){
drhd677b3d2007-08-20 22:48:41 +0000227 vfsList = pVfs->pNext;
drh9bc54492007-10-23 14:49:59 +0000228 }else if( vfsList ){
drhd677b3d2007-08-20 22:48:41 +0000229 sqlite3_vfs *p = vfsList;
230 while( p->pNext && p->pNext!=pVfs ){
231 p = p->pNext;
232 }
233 if( p->pNext==pVfs ){
234 p->pNext = pVfs->pNext;
235 }
236 }
237}
238
239/*
240** Register a VFS with the system. It is harmless to register the same
241** VFS multiple times. The new VFS becomes the default if makeDflt is
242** true.
243*/
244int sqlite3_vfs_register(sqlite3_vfs *pVfs, int makeDflt){
drh7e8b8482008-01-23 03:03:05 +0000245#ifndef SQLITE_MUTEX_NOOP
drhd677b3d2007-08-20 22:48:41 +0000246 sqlite3_mutex *mutex = sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_MASTER);
drh7e8b8482008-01-23 03:03:05 +0000247#endif
drhe9bb50a2007-09-02 17:52:04 +0000248 sqlite3_vfs_find(0); /* Make sure we are initialized */
drhd677b3d2007-08-20 22:48:41 +0000249 sqlite3_mutex_enter(mutex);
250 vfsUnlink(pVfs);
251 if( makeDflt || vfsList==0 ){
252 pVfs->pNext = vfsList;
253 vfsList = pVfs;
254 }else{
255 pVfs->pNext = vfsList->pNext;
danielk197795c8a542007-09-01 06:51:27 +0000256 vfsList->pNext = pVfs;
drhd677b3d2007-08-20 22:48:41 +0000257 }
danielk1977f1da17a2007-08-21 13:07:46 +0000258 assert(vfsList);
drhd677b3d2007-08-20 22:48:41 +0000259 sqlite3_mutex_leave(mutex);
260 return SQLITE_OK;
261}
262
263/*
264** Unregister a VFS so that it is no longer accessible.
265*/
266int sqlite3_vfs_unregister(sqlite3_vfs *pVfs){
drh7e8b8482008-01-23 03:03:05 +0000267#ifndef SQLITE_MUTEX_NOOP
drhd677b3d2007-08-20 22:48:41 +0000268 sqlite3_mutex *mutex = sqlite3_mutex_alloc(SQLITE_MUTEX_STATIC_MASTER);
drh7e8b8482008-01-23 03:03:05 +0000269#endif
drhd677b3d2007-08-20 22:48:41 +0000270 sqlite3_mutex_enter(mutex);
271 vfsUnlink(pVfs);
272 sqlite3_mutex_leave(mutex);
273 return SQLITE_OK;
274}
drh7e2e6dc2008-03-26 17:18:48 +0000275
276/*
277** Provide a default sqlite3OsDefaultVfs() implementation in the
278** cases where none of the standard backends are used.
279*/
280#if !OS_UNIX && !OS_WIN && !OS_OS2
281sqlite3_vfs *sqlite3OsDefaultVfs(void){ return 0; }
282#endif