1 /* 2 ** 2008 Jan 22 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 code that modified the OS layer in order to simulate 14 ** different device types (by overriding the return values of the 15 ** xDeviceCharacteristics() and xSectorSize() methods). 16 ** 17 ** $Id: test_devsym.c,v 1.9 2008/12/09 01:32:03 drh Exp $ 18 */ 19 #if SQLITE_TEST /* This file is used for testing only */ 20 21 #include "sqlite3.h" 22 #include "sqliteInt.h" 23 24 /* 25 ** Maximum pathname length supported by the devsym backend. 26 */ 27 #define DEVSYM_MAX_PATHNAME 512 28 29 /* 30 ** Name used to identify this VFS. 31 */ 32 #define DEVSYM_VFS_NAME "devsym" 33 34 typedef struct devsym_file devsym_file; 35 struct devsym_file { 36 sqlite3_file base; 37 sqlite3_file *pReal; 38 }; 39 40 /* 41 ** Method declarations for devsym_file. 42 */ 43 static int devsymClose(sqlite3_file*); 44 static int devsymRead(sqlite3_file*, void*, int iAmt, sqlite3_int64 iOfst); 45 static int devsymWrite(sqlite3_file*,const void*,int iAmt, sqlite3_int64 iOfst); 46 static int devsymTruncate(sqlite3_file*, sqlite3_int64 size); 47 static int devsymSync(sqlite3_file*, int flags); 48 static int devsymFileSize(sqlite3_file*, sqlite3_int64 *pSize); 49 static int devsymLock(sqlite3_file*, int); 50 static int devsymUnlock(sqlite3_file*, int); 51 static int devsymCheckReservedLock(sqlite3_file*, int *); 52 static int devsymFileControl(sqlite3_file*, int op, void *pArg); 53 static int devsymSectorSize(sqlite3_file*); 54 static int devsymDeviceCharacteristics(sqlite3_file*); 55 56 /* 57 ** Method declarations for devsym_vfs. 58 */ 59 static int devsymOpen(sqlite3_vfs*, const char *, sqlite3_file*, int , int *); 60 static int devsymDelete(sqlite3_vfs*, const char *zName, int syncDir); 61 static int devsymAccess(sqlite3_vfs*, const char *zName, int flags, int *); 62 static int devsymFullPathname(sqlite3_vfs*, const char *zName, int, char *zOut); 63 #ifndef SQLITE_OMIT_LOAD_EXTENSION 64 static void *devsymDlOpen(sqlite3_vfs*, const char *zFilename); 65 static void devsymDlError(sqlite3_vfs*, int nByte, char *zErrMsg); 66 static void (*devsymDlSym(sqlite3_vfs*,void*, const char *zSymbol))(void); 67 static void devsymDlClose(sqlite3_vfs*, void*); 68 #endif /* SQLITE_OMIT_LOAD_EXTENSION */ 69 static int devsymRandomness(sqlite3_vfs*, int nByte, char *zOut); 70 static int devsymSleep(sqlite3_vfs*, int microseconds); 71 static int devsymCurrentTime(sqlite3_vfs*, double*); 72 73 static sqlite3_vfs devsym_vfs = { 74 1, /* iVersion */ 75 sizeof(devsym_file), /* szOsFile */ 76 DEVSYM_MAX_PATHNAME, /* mxPathname */ 77 0, /* pNext */ 78 DEVSYM_VFS_NAME, /* zName */ 79 0, /* pAppData */ 80 devsymOpen, /* xOpen */ 81 devsymDelete, /* xDelete */ 82 devsymAccess, /* xAccess */ 83 devsymFullPathname, /* xFullPathname */ 84 #ifndef SQLITE_OMIT_LOAD_EXTENSION 85 devsymDlOpen, /* xDlOpen */ 86 devsymDlError, /* xDlError */ 87 devsymDlSym, /* xDlSym */ 88 devsymDlClose, /* xDlClose */ 89 #else 90 0, /* xDlOpen */ 91 0, /* xDlError */ 92 0, /* xDlSym */ 93 0, /* xDlClose */ 94 #endif /* SQLITE_OMIT_LOAD_EXTENSION */ 95 devsymRandomness, /* xRandomness */ 96 devsymSleep, /* xSleep */ 97 devsymCurrentTime /* xCurrentTime */ 98 }; 99 100 static sqlite3_io_methods devsym_io_methods = { 101 1, /* iVersion */ 102 devsymClose, /* xClose */ 103 devsymRead, /* xRead */ 104 devsymWrite, /* xWrite */ 105 devsymTruncate, /* xTruncate */ 106 devsymSync, /* xSync */ 107 devsymFileSize, /* xFileSize */ 108 devsymLock, /* xLock */ 109 devsymUnlock, /* xUnlock */ 110 devsymCheckReservedLock, /* xCheckReservedLock */ 111 devsymFileControl, /* xFileControl */ 112 devsymSectorSize, /* xSectorSize */ 113 devsymDeviceCharacteristics /* xDeviceCharacteristics */ 114 }; 115 116 struct DevsymGlobal { 117 sqlite3_vfs *pVfs; 118 int iDeviceChar; 119 int iSectorSize; 120 }; 121 struct DevsymGlobal g = {0, 0, 512}; 122 123 /* 124 ** Close an devsym-file. 125 */ 126 static int devsymClose(sqlite3_file *pFile){ 127 devsym_file *p = (devsym_file *)pFile; 128 return sqlite3OsClose(p->pReal); 129 } 130 131 /* 132 ** Read data from an devsym-file. 133 */ 134 static int devsymRead( 135 sqlite3_file *pFile, 136 void *zBuf, 137 int iAmt, 138 sqlite_int64 iOfst 139 ){ 140 devsym_file *p = (devsym_file *)pFile; 141 return sqlite3OsRead(p->pReal, zBuf, iAmt, iOfst); 142 } 143 144 /* 145 ** Write data to an devsym-file. 146 */ 147 static int devsymWrite( 148 sqlite3_file *pFile, 149 const void *zBuf, 150 int iAmt, 151 sqlite_int64 iOfst 152 ){ 153 devsym_file *p = (devsym_file *)pFile; 154 return sqlite3OsWrite(p->pReal, zBuf, iAmt, iOfst); 155 } 156 157 /* 158 ** Truncate an devsym-file. 159 */ 160 static int devsymTruncate(sqlite3_file *pFile, sqlite_int64 size){ 161 devsym_file *p = (devsym_file *)pFile; 162 return sqlite3OsTruncate(p->pReal, size); 163 } 164 165 /* 166 ** Sync an devsym-file. 167 */ 168 static int devsymSync(sqlite3_file *pFile, int flags){ 169 devsym_file *p = (devsym_file *)pFile; 170 return sqlite3OsSync(p->pReal, flags); 171 } 172 173 /* 174 ** Return the current file-size of an devsym-file. 175 */ 176 static int devsymFileSize(sqlite3_file *pFile, sqlite_int64 *pSize){ 177 devsym_file *p = (devsym_file *)pFile; 178 return sqlite3OsFileSize(p->pReal, pSize); 179 } 180 181 /* 182 ** Lock an devsym-file. 183 */ 184 static int devsymLock(sqlite3_file *pFile, int eLock){ 185 devsym_file *p = (devsym_file *)pFile; 186 return sqlite3OsLock(p->pReal, eLock); 187 } 188 189 /* 190 ** Unlock an devsym-file. 191 */ 192 static int devsymUnlock(sqlite3_file *pFile, int eLock){ 193 devsym_file *p = (devsym_file *)pFile; 194 return sqlite3OsUnlock(p->pReal, eLock); 195 } 196 197 /* 198 ** Check if another file-handle holds a RESERVED lock on an devsym-file. 199 */ 200 static int devsymCheckReservedLock(sqlite3_file *pFile, int *pResOut){ 201 devsym_file *p = (devsym_file *)pFile; 202 return sqlite3OsCheckReservedLock(p->pReal, pResOut); 203 } 204 205 /* 206 ** File control method. For custom operations on an devsym-file. 207 */ 208 static int devsymFileControl(sqlite3_file *pFile, int op, void *pArg){ 209 devsym_file *p = (devsym_file *)pFile; 210 return sqlite3OsFileControl(p->pReal, op, pArg); 211 } 212 213 /* 214 ** Return the sector-size in bytes for an devsym-file. 215 */ 216 static int devsymSectorSize(sqlite3_file *pFile){ 217 return g.iSectorSize; 218 } 219 220 /* 221 ** Return the device characteristic flags supported by an devsym-file. 222 */ 223 static int devsymDeviceCharacteristics(sqlite3_file *pFile){ 224 return g.iDeviceChar; 225 } 226 227 /* 228 ** Open an devsym file handle. 229 */ 230 static int devsymOpen( 231 sqlite3_vfs *pVfs, 232 const char *zName, 233 sqlite3_file *pFile, 234 int flags, 235 int *pOutFlags 236 ){ 237 int rc; 238 devsym_file *p = (devsym_file *)pFile; 239 p->pReal = (sqlite3_file *)&p[1]; 240 rc = sqlite3OsOpen(g.pVfs, zName, p->pReal, flags, pOutFlags); 241 if( p->pReal->pMethods ){ 242 pFile->pMethods = &devsym_io_methods; 243 } 244 return rc; 245 } 246 247 /* 248 ** Delete the file located at zPath. If the dirSync argument is true, 249 ** ensure the file-system modifications are synced to disk before 250 ** returning. 251 */ 252 static int devsymDelete(sqlite3_vfs *pVfs, const char *zPath, int dirSync){ 253 return sqlite3OsDelete(g.pVfs, zPath, dirSync); 254 } 255 256 /* 257 ** Test for access permissions. Return true if the requested permission 258 ** is available, or false otherwise. 259 */ 260 static int devsymAccess( 261 sqlite3_vfs *pVfs, 262 const char *zPath, 263 int flags, 264 int *pResOut 265 ){ 266 return sqlite3OsAccess(g.pVfs, zPath, flags, pResOut); 267 } 268 269 /* 270 ** Populate buffer zOut with the full canonical pathname corresponding 271 ** to the pathname in zPath. zOut is guaranteed to point to a buffer 272 ** of at least (DEVSYM_MAX_PATHNAME+1) bytes. 273 */ 274 static int devsymFullPathname( 275 sqlite3_vfs *pVfs, 276 const char *zPath, 277 int nOut, 278 char *zOut 279 ){ 280 return sqlite3OsFullPathname(g.pVfs, zPath, nOut, zOut); 281 } 282 283 #ifndef SQLITE_OMIT_LOAD_EXTENSION 284 /* 285 ** Open the dynamic library located at zPath and return a handle. 286 */ 287 static void *devsymDlOpen(sqlite3_vfs *pVfs, const char *zPath){ 288 return sqlite3OsDlOpen(g.pVfs, zPath); 289 } 290 291 /* 292 ** Populate the buffer zErrMsg (size nByte bytes) with a human readable 293 ** utf-8 string describing the most recent error encountered associated 294 ** with dynamic libraries. 295 */ 296 static void devsymDlError(sqlite3_vfs *pVfs, int nByte, char *zErrMsg){ 297 sqlite3OsDlError(g.pVfs, nByte, zErrMsg); 298 } 299 300 /* 301 ** Return a pointer to the symbol zSymbol in the dynamic library pHandle. 302 */ 303 static void (*devsymDlSym(sqlite3_vfs *pVfs, void *p, const char *zSym))(void){ 304 return sqlite3OsDlSym(g.pVfs, p, zSym); 305 } 306 307 /* 308 ** Close the dynamic library handle pHandle. 309 */ 310 static void devsymDlClose(sqlite3_vfs *pVfs, void *pHandle){ 311 sqlite3OsDlClose(g.pVfs, pHandle); 312 } 313 #endif /* SQLITE_OMIT_LOAD_EXTENSION */ 314 315 /* 316 ** Populate the buffer pointed to by zBufOut with nByte bytes of 317 ** random data. 318 */ 319 static int devsymRandomness(sqlite3_vfs *pVfs, int nByte, char *zBufOut){ 320 return sqlite3OsRandomness(g.pVfs, nByte, zBufOut); 321 } 322 323 /* 324 ** Sleep for nMicro microseconds. Return the number of microseconds 325 ** actually slept. 326 */ 327 static int devsymSleep(sqlite3_vfs *pVfs, int nMicro){ 328 return sqlite3OsSleep(g.pVfs, nMicro); 329 } 330 331 /* 332 ** Return the current time as a Julian Day number in *pTimeOut. 333 */ 334 static int devsymCurrentTime(sqlite3_vfs *pVfs, double *pTimeOut){ 335 return sqlite3OsCurrentTime(g.pVfs, pTimeOut); 336 } 337 338 /* 339 ** This procedure registers the devsym vfs with SQLite. If the argument is 340 ** true, the devsym vfs becomes the new default vfs. It is the only publicly 341 ** available function in this file. 342 */ 343 void devsym_register(int iDeviceChar, int iSectorSize){ 344 if( g.pVfs==0 ){ 345 g.pVfs = sqlite3_vfs_find(0); 346 devsym_vfs.szOsFile += g.pVfs->szOsFile; 347 sqlite3_vfs_register(&devsym_vfs, 0); 348 } 349 if( iDeviceChar>=0 ){ 350 g.iDeviceChar = iDeviceChar; 351 } 352 if( iSectorSize>=0 ){ 353 g.iSectorSize = iSectorSize; 354 } 355 } 356 357 #endif 358