os/persistentdata/persistentstorage/sql/SQLite/os.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/*
sl@0
     2
** 2005 November 29
sl@0
     3
**
sl@0
     4
** The author disclaims copyright to this source code.  In place of
sl@0
     5
** a legal notice, here is a blessing:
sl@0
     6
**
sl@0
     7
**    May you do good and not evil.
sl@0
     8
**    May you find forgiveness for yourself and forgive others.
sl@0
     9
**    May you share freely, never taking more than you give.
sl@0
    10
**
sl@0
    11
******************************************************************************
sl@0
    12
**
sl@0
    13
** This file contains OS interface code that is common to all
sl@0
    14
** architectures.
sl@0
    15
**
sl@0
    16
** $Id: os.c,v 1.120 2008/07/28 19:34:53 drh Exp $
sl@0
    17
*/
sl@0
    18
#define _SQLITE_OS_C_ 1
sl@0
    19
#include "sqliteInt.h"
sl@0
    20
#undef _SQLITE_OS_C_
sl@0
    21
sl@0
    22
/*
sl@0
    23
** The default SQLite sqlite3_vfs implementations do not allocate
sl@0
    24
** memory (actually, os_unix.c allocates a small amount of memory
sl@0
    25
** from within OsOpen()), but some third-party implementations may.
sl@0
    26
** So we test the effects of a malloc() failing and the sqlite3OsXXX()
sl@0
    27
** function returning SQLITE_IOERR_NOMEM using the DO_OS_MALLOC_TEST macro.
sl@0
    28
**
sl@0
    29
** The following functions are instrumented for malloc() failure 
sl@0
    30
** testing:
sl@0
    31
**
sl@0
    32
**     sqlite3OsOpen()
sl@0
    33
**     sqlite3OsRead()
sl@0
    34
**     sqlite3OsWrite()
sl@0
    35
**     sqlite3OsSync()
sl@0
    36
**     sqlite3OsLock()
sl@0
    37
**
sl@0
    38
*/
sl@0
    39
#if defined(SQLITE_TEST) && (SQLITE_OS_WIN==0) && 0
sl@0
    40
  #define DO_OS_MALLOC_TEST if (1) {            \
sl@0
    41
    void *pTstAlloc = sqlite3Malloc(10);       \
sl@0
    42
    if (!pTstAlloc) return SQLITE_IOERR_NOMEM;  \
sl@0
    43
    sqlite3_free(pTstAlloc);                    \
sl@0
    44
  }
sl@0
    45
#else
sl@0
    46
  #define DO_OS_MALLOC_TEST
sl@0
    47
#endif
sl@0
    48
sl@0
    49
/*
sl@0
    50
** The following routines are convenience wrappers around methods
sl@0
    51
** of the sqlite3_file object.  This is mostly just syntactic sugar. All
sl@0
    52
** of this would be completely automatic if SQLite were coded using
sl@0
    53
** C++ instead of plain old C.
sl@0
    54
*/
sl@0
    55
int sqlite3OsClose(sqlite3_file *pId){
sl@0
    56
  int rc = SQLITE_OK;
sl@0
    57
  if( pId->pMethods ){
sl@0
    58
    rc = pId->pMethods->xClose(pId);
sl@0
    59
    pId->pMethods = 0;
sl@0
    60
  }
sl@0
    61
  return rc;
sl@0
    62
}
sl@0
    63
int sqlite3OsRead(sqlite3_file *id, void *pBuf, int amt, i64 offset){
sl@0
    64
  DO_OS_MALLOC_TEST;
sl@0
    65
  return id->pMethods->xRead(id, pBuf, amt, offset);
sl@0
    66
}
sl@0
    67
int sqlite3OsWrite(sqlite3_file *id, const void *pBuf, int amt, i64 offset){
sl@0
    68
  DO_OS_MALLOC_TEST;
sl@0
    69
  return id->pMethods->xWrite(id, pBuf, amt, offset);
sl@0
    70
}
sl@0
    71
int sqlite3OsTruncate(sqlite3_file *id, i64 size){
sl@0
    72
  return id->pMethods->xTruncate(id, size);
sl@0
    73
}
sl@0
    74
int sqlite3OsSync(sqlite3_file *id, int flags){
sl@0
    75
  DO_OS_MALLOC_TEST;
sl@0
    76
  return id->pMethods->xSync(id, flags);
sl@0
    77
}
sl@0
    78
int sqlite3OsFileSize(sqlite3_file *id, i64 *pSize){
sl@0
    79
  DO_OS_MALLOC_TEST;
sl@0
    80
  return id->pMethods->xFileSize(id, pSize);
sl@0
    81
}
sl@0
    82
int sqlite3OsLock(sqlite3_file *id, int lockType){
sl@0
    83
  DO_OS_MALLOC_TEST;
sl@0
    84
  return id->pMethods->xLock(id, lockType);
sl@0
    85
}
sl@0
    86
int sqlite3OsUnlock(sqlite3_file *id, int lockType){
sl@0
    87
  return id->pMethods->xUnlock(id, lockType);
sl@0
    88
}
sl@0
    89
int sqlite3OsCheckReservedLock(sqlite3_file *id, int *pResOut){
sl@0
    90
  DO_OS_MALLOC_TEST;
sl@0
    91
  return id->pMethods->xCheckReservedLock(id, pResOut);
sl@0
    92
}
sl@0
    93
int sqlite3OsFileControl(sqlite3_file *id, int op, void *pArg){
sl@0
    94
  return id->pMethods->xFileControl(id, op, pArg);
sl@0
    95
}
sl@0
    96
int sqlite3OsSectorSize(sqlite3_file *id){
sl@0
    97
  int (*xSectorSize)(sqlite3_file*) = id->pMethods->xSectorSize;
sl@0
    98
  return (xSectorSize ? xSectorSize(id) : SQLITE_DEFAULT_SECTOR_SIZE);
sl@0
    99
}
sl@0
   100
int sqlite3OsDeviceCharacteristics(sqlite3_file *id){
sl@0
   101
  return id->pMethods->xDeviceCharacteristics(id);
sl@0
   102
}
sl@0
   103
sl@0
   104
/*
sl@0
   105
** The next group of routines are convenience wrappers around the
sl@0
   106
** VFS methods.
sl@0
   107
*/
sl@0
   108
int sqlite3OsOpen(
sl@0
   109
  sqlite3_vfs *pVfs, 
sl@0
   110
  const char *zPath, 
sl@0
   111
  sqlite3_file *pFile, 
sl@0
   112
  int flags, 
sl@0
   113
  int *pFlagsOut
sl@0
   114
){
sl@0
   115
  DO_OS_MALLOC_TEST;
sl@0
   116
  return pVfs->xOpen(pVfs, zPath, pFile, flags, pFlagsOut);
sl@0
   117
}
sl@0
   118
int sqlite3OsDelete(sqlite3_vfs *pVfs, const char *zPath, int dirSync){
sl@0
   119
  return pVfs->xDelete(pVfs, zPath, dirSync);
sl@0
   120
}
sl@0
   121
int sqlite3OsAccess(
sl@0
   122
  sqlite3_vfs *pVfs, 
sl@0
   123
  const char *zPath, 
sl@0
   124
  int flags, 
sl@0
   125
  int *pResOut
sl@0
   126
){
sl@0
   127
  DO_OS_MALLOC_TEST;
sl@0
   128
  return pVfs->xAccess(pVfs, zPath, flags, pResOut);
sl@0
   129
}
sl@0
   130
int sqlite3OsFullPathname(
sl@0
   131
  sqlite3_vfs *pVfs, 
sl@0
   132
  const char *zPath, 
sl@0
   133
  int nPathOut, 
sl@0
   134
  char *zPathOut
sl@0
   135
){
sl@0
   136
  return pVfs->xFullPathname(pVfs, zPath, nPathOut, zPathOut);
sl@0
   137
}
sl@0
   138
#ifndef SQLITE_OMIT_LOAD_EXTENSION
sl@0
   139
void *sqlite3OsDlOpen(sqlite3_vfs *pVfs, const char *zPath){
sl@0
   140
  return pVfs->xDlOpen(pVfs, zPath);
sl@0
   141
}
sl@0
   142
void sqlite3OsDlError(sqlite3_vfs *pVfs, int nByte, char *zBufOut){
sl@0
   143
  pVfs->xDlError(pVfs, nByte, zBufOut);
sl@0
   144
}
sl@0
   145
void *sqlite3OsDlSym(sqlite3_vfs *pVfs, void *pHandle, const char *zSymbol){
sl@0
   146
  return pVfs->xDlSym(pVfs, pHandle, zSymbol);
sl@0
   147
}
sl@0
   148
void sqlite3OsDlClose(sqlite3_vfs *pVfs, void *pHandle){
sl@0
   149
  pVfs->xDlClose(pVfs, pHandle);
sl@0
   150
}
sl@0
   151
#endif /* SQLITE_OMIT_LOAD_EXTENSION */
sl@0
   152
int sqlite3OsRandomness(sqlite3_vfs *pVfs, int nByte, char *zBufOut){
sl@0
   153
  return pVfs->xRandomness(pVfs, nByte, zBufOut);
sl@0
   154
}
sl@0
   155
int sqlite3OsSleep(sqlite3_vfs *pVfs, int nMicro){
sl@0
   156
  return pVfs->xSleep(pVfs, nMicro);
sl@0
   157
}
sl@0
   158
int sqlite3OsCurrentTime(sqlite3_vfs *pVfs, double *pTimeOut){
sl@0
   159
  return pVfs->xCurrentTime(pVfs, pTimeOut);
sl@0
   160
}
sl@0
   161
sl@0
   162
int sqlite3OsOpenMalloc(
sl@0
   163
  sqlite3_vfs *pVfs, 
sl@0
   164
  const char *zFile, 
sl@0
   165
  sqlite3_file **ppFile, 
sl@0
   166
  int flags,
sl@0
   167
  int *pOutFlags
sl@0
   168
){
sl@0
   169
  int rc = SQLITE_NOMEM;
sl@0
   170
  sqlite3_file *pFile;
sl@0
   171
  pFile = (sqlite3_file *)sqlite3Malloc(pVfs->szOsFile);
sl@0
   172
  if( pFile ){
sl@0
   173
    rc = sqlite3OsOpen(pVfs, zFile, pFile, flags, pOutFlags);
sl@0
   174
    if( rc!=SQLITE_OK ){
sl@0
   175
      sqlite3_free(pFile);
sl@0
   176
    }else{
sl@0
   177
      *ppFile = pFile;
sl@0
   178
    }
sl@0
   179
  }
sl@0
   180
  return rc;
sl@0
   181
}
sl@0
   182
int sqlite3OsCloseFree(sqlite3_file *pFile){
sl@0
   183
  int rc = SQLITE_OK;
sl@0
   184
  assert( pFile );
sl@0
   185
  rc = sqlite3OsClose(pFile);
sl@0
   186
  sqlite3_free(pFile);
sl@0
   187
  return rc;
sl@0
   188
}
sl@0
   189
sl@0
   190
/*
sl@0
   191
** The list of all registered VFS implementations.
sl@0
   192
*/
sl@0
   193
static sqlite3_vfs *vfsList = 0;
sl@0
   194
sl@0
   195
/*
sl@0
   196
** Locate a VFS by name.  If no name is given, simply return the
sl@0
   197
** first VFS on the list.
sl@0
   198
*/
sl@0
   199
sqlite3_vfs *sqlite3_vfs_find(const char *zVfs){
sl@0
   200
  sqlite3_vfs *pVfs = 0;
sl@0
   201
#ifndef SQLITE_MUTEX_NOOP
sl@0
   202
  sqlite3_mutex *mutex;
sl@0
   203
#endif
sl@0
   204
#ifndef SQLITE_OMIT_AUTOINIT
sl@0
   205
  int rc = sqlite3_initialize();
sl@0
   206
  if( rc ) return 0;
sl@0
   207
#endif
sl@0
   208
#ifndef SQLITE_MUTEX_NOOP
sl@0
   209
  mutex = sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_MASTER);
sl@0
   210
#endif
sl@0
   211
  sqlite3_mutex_enter(mutex);
sl@0
   212
  for(pVfs = vfsList; pVfs; pVfs=pVfs->pNext){
sl@0
   213
    if( zVfs==0 ) break;
sl@0
   214
    if( strcmp(zVfs, pVfs->zName)==0 ) break;
sl@0
   215
  }
sl@0
   216
  sqlite3_mutex_leave(mutex);
sl@0
   217
  return pVfs;
sl@0
   218
}
sl@0
   219
sl@0
   220
/*
sl@0
   221
** Unlink a VFS from the linked list
sl@0
   222
*/
sl@0
   223
static void vfsUnlink(sqlite3_vfs *pVfs){
sl@0
   224
  assert( sqlite3_mutex_held(sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_MASTER)) );
sl@0
   225
  if( pVfs==0 ){
sl@0
   226
    /* No-op */
sl@0
   227
  }else if( vfsList==pVfs ){
sl@0
   228
    vfsList = pVfs->pNext;
sl@0
   229
  }else if( vfsList ){
sl@0
   230
    sqlite3_vfs *p = vfsList;
sl@0
   231
    while( p->pNext && p->pNext!=pVfs ){
sl@0
   232
      p = p->pNext;
sl@0
   233
    }
sl@0
   234
    if( p->pNext==pVfs ){
sl@0
   235
      p->pNext = pVfs->pNext;
sl@0
   236
    }
sl@0
   237
  }
sl@0
   238
}
sl@0
   239
sl@0
   240
/*
sl@0
   241
** Register a VFS with the system.  It is harmless to register the same
sl@0
   242
** VFS multiple times.  The new VFS becomes the default if makeDflt is
sl@0
   243
** true.
sl@0
   244
*/
sl@0
   245
int sqlite3_vfs_register(sqlite3_vfs *pVfs, int makeDflt){
sl@0
   246
  sqlite3_mutex *mutex = 0;
sl@0
   247
#ifndef SQLITE_OMIT_AUTOINIT
sl@0
   248
  int rc = sqlite3_initialize();
sl@0
   249
  if( rc ) return rc;
sl@0
   250
#endif
sl@0
   251
  mutex = sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_MASTER);
sl@0
   252
  sqlite3_mutex_enter(mutex);
sl@0
   253
  vfsUnlink(pVfs);
sl@0
   254
  if( makeDflt || vfsList==0 ){
sl@0
   255
    pVfs->pNext = vfsList;
sl@0
   256
    vfsList = pVfs;
sl@0
   257
  }else{
sl@0
   258
    pVfs->pNext = vfsList->pNext;
sl@0
   259
    vfsList->pNext = pVfs;
sl@0
   260
  }
sl@0
   261
  assert(vfsList);
sl@0
   262
  sqlite3_mutex_leave(mutex);
sl@0
   263
  return SQLITE_OK;
sl@0
   264
}
sl@0
   265
sl@0
   266
/*
sl@0
   267
** Unregister a VFS so that it is no longer accessible.
sl@0
   268
*/
sl@0
   269
int sqlite3_vfs_unregister(sqlite3_vfs *pVfs){
sl@0
   270
#ifndef SQLITE_MUTEX_NOOP
sl@0
   271
  sqlite3_mutex *mutex = sqlite3MutexAlloc(SQLITE_MUTEX_STATIC_MASTER);
sl@0
   272
#endif
sl@0
   273
  sqlite3_mutex_enter(mutex);
sl@0
   274
  vfsUnlink(pVfs);
sl@0
   275
  sqlite3_mutex_leave(mutex);
sl@0
   276
  return SQLITE_OK;
sl@0
   277
}