1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/persistentdata/persistentstorage/sqlite3api/SQLite/os_os2.c Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,1113 @@
1.4 +/*
1.5 +** 2006 Feb 14
1.6 +**
1.7 +** The author disclaims copyright to this source code. In place of
1.8 +** a legal notice, here is a blessing:
1.9 +**
1.10 +** May you do good and not evil.
1.11 +** May you find forgiveness for yourself and forgive others.
1.12 +** May you share freely, never taking more than you give.
1.13 +**
1.14 +******************************************************************************
1.15 +**
1.16 +** This file contains code that is specific to OS/2.
1.17 +**
1.18 +** $Id: os_os2.c,v 1.56 2008/08/22 13:47:57 pweilbacher Exp $
1.19 +*/
1.20 +
1.21 +#include "sqliteInt.h"
1.22 +
1.23 +#if SQLITE_OS_OS2
1.24 +
1.25 +/*
1.26 +** A Note About Memory Allocation:
1.27 +**
1.28 +** This driver uses malloc()/free() directly rather than going through
1.29 +** the SQLite-wrappers sqlite3_malloc()/sqlite3_free(). Those wrappers
1.30 +** are designed for use on embedded systems where memory is scarce and
1.31 +** malloc failures happen frequently. OS/2 does not typically run on
1.32 +** embedded systems, and when it does the developers normally have bigger
1.33 +** problems to worry about than running out of memory. So there is not
1.34 +** a compelling need to use the wrappers.
1.35 +**
1.36 +** But there is a good reason to not use the wrappers. If we use the
1.37 +** wrappers then we will get simulated malloc() failures within this
1.38 +** driver. And that causes all kinds of problems for our tests. We
1.39 +** could enhance SQLite to deal with simulated malloc failures within
1.40 +** the OS driver, but the code to deal with those failure would not
1.41 +** be exercised on Linux (which does not need to malloc() in the driver)
1.42 +** and so we would have difficulty writing coverage tests for that
1.43 +** code. Better to leave the code out, we think.
1.44 +**
1.45 +** The point of this discussion is as follows: When creating a new
1.46 +** OS layer for an embedded system, if you use this file as an example,
1.47 +** avoid the use of malloc()/free(). Those routines work ok on OS/2
1.48 +** desktops but not so well in embedded systems.
1.49 +*/
1.50 +
1.51 +/*
1.52 +** Macros used to determine whether or not to use threads.
1.53 +*/
1.54 +#if defined(SQLITE_THREADSAFE) && SQLITE_THREADSAFE
1.55 +# define SQLITE_OS2_THREADS 1
1.56 +#endif
1.57 +
1.58 +/*
1.59 +** Include code that is common to all os_*.c files
1.60 +*/
1.61 +#include "os_common.h"
1.62 +
1.63 +/*
1.64 +** The os2File structure is subclass of sqlite3_file specific for the OS/2
1.65 +** protability layer.
1.66 +*/
1.67 +typedef struct os2File os2File;
1.68 +struct os2File {
1.69 + const sqlite3_io_methods *pMethod; /* Always the first entry */
1.70 + HFILE h; /* Handle for accessing the file */
1.71 + char* pathToDel; /* Name of file to delete on close, NULL if not */
1.72 + unsigned char locktype; /* Type of lock currently held on this file */
1.73 +};
1.74 +
1.75 +#define LOCK_TIMEOUT 10L /* the default locking timeout */
1.76 +
1.77 +/*****************************************************************************
1.78 +** The next group of routines implement the I/O methods specified
1.79 +** by the sqlite3_io_methods object.
1.80 +******************************************************************************/
1.81 +
1.82 +/*
1.83 +** Close a file.
1.84 +*/
1.85 +static int os2Close( sqlite3_file *id ){
1.86 + APIRET rc = NO_ERROR;
1.87 + os2File *pFile;
1.88 + if( id && (pFile = (os2File*)id) != 0 ){
1.89 + OSTRACE2( "CLOSE %d\n", pFile->h );
1.90 + rc = DosClose( pFile->h );
1.91 + pFile->locktype = NO_LOCK;
1.92 + if( pFile->pathToDel != NULL ){
1.93 + rc = DosForceDelete( (PSZ)pFile->pathToDel );
1.94 + free( pFile->pathToDel );
1.95 + pFile->pathToDel = NULL;
1.96 + }
1.97 + id = 0;
1.98 + OpenCounter( -1 );
1.99 + }
1.100 +
1.101 + return rc == NO_ERROR ? SQLITE_OK : SQLITE_IOERR;
1.102 +}
1.103 +
1.104 +/*
1.105 +** Read data from a file into a buffer. Return SQLITE_OK if all
1.106 +** bytes were read successfully and SQLITE_IOERR if anything goes
1.107 +** wrong.
1.108 +*/
1.109 +static int os2Read(
1.110 + sqlite3_file *id, /* File to read from */
1.111 + void *pBuf, /* Write content into this buffer */
1.112 + int amt, /* Number of bytes to read */
1.113 + sqlite3_int64 offset /* Begin reading at this offset */
1.114 +){
1.115 + ULONG fileLocation = 0L;
1.116 + ULONG got;
1.117 + os2File *pFile = (os2File*)id;
1.118 + assert( id!=0 );
1.119 + SimulateIOError( return SQLITE_IOERR_READ );
1.120 + OSTRACE3( "READ %d lock=%d\n", pFile->h, pFile->locktype );
1.121 + if( DosSetFilePtr(pFile->h, offset, FILE_BEGIN, &fileLocation) != NO_ERROR ){
1.122 + return SQLITE_IOERR;
1.123 + }
1.124 + if( DosRead( pFile->h, pBuf, amt, &got ) != NO_ERROR ){
1.125 + return SQLITE_IOERR_READ;
1.126 + }
1.127 + if( got == (ULONG)amt )
1.128 + return SQLITE_OK;
1.129 + else {
1.130 + memset(&((char*)pBuf)[got], 0, amt-got);
1.131 + return SQLITE_IOERR_SHORT_READ;
1.132 + }
1.133 +}
1.134 +
1.135 +/*
1.136 +** Write data from a buffer into a file. Return SQLITE_OK on success
1.137 +** or some other error code on failure.
1.138 +*/
1.139 +static int os2Write(
1.140 + sqlite3_file *id, /* File to write into */
1.141 + const void *pBuf, /* The bytes to be written */
1.142 + int amt, /* Number of bytes to write */
1.143 + sqlite3_int64 offset /* Offset into the file to begin writing at */
1.144 +){
1.145 + ULONG fileLocation = 0L;
1.146 + APIRET rc = NO_ERROR;
1.147 + ULONG wrote;
1.148 + os2File *pFile = (os2File*)id;
1.149 + assert( id!=0 );
1.150 + SimulateIOError( return SQLITE_IOERR_WRITE );
1.151 + SimulateDiskfullError( return SQLITE_FULL );
1.152 + OSTRACE3( "WRITE %d lock=%d\n", pFile->h, pFile->locktype );
1.153 + if( DosSetFilePtr(pFile->h, offset, FILE_BEGIN, &fileLocation) != NO_ERROR ){
1.154 + return SQLITE_IOERR;
1.155 + }
1.156 + assert( amt>0 );
1.157 + while( amt > 0 &&
1.158 + ( rc = DosWrite( pFile->h, (PVOID)pBuf, amt, &wrote ) ) == NO_ERROR &&
1.159 + wrote > 0
1.160 + ){
1.161 + amt -= wrote;
1.162 + pBuf = &((char*)pBuf)[wrote];
1.163 + }
1.164 +
1.165 + return ( rc != NO_ERROR || amt > (int)wrote ) ? SQLITE_FULL : SQLITE_OK;
1.166 +}
1.167 +
1.168 +/*
1.169 +** Truncate an open file to a specified size
1.170 +*/
1.171 +static int os2Truncate( sqlite3_file *id, i64 nByte ){
1.172 + APIRET rc = NO_ERROR;
1.173 + os2File *pFile = (os2File*)id;
1.174 + OSTRACE3( "TRUNCATE %d %lld\n", pFile->h, nByte );
1.175 + SimulateIOError( return SQLITE_IOERR_TRUNCATE );
1.176 + rc = DosSetFileSize( pFile->h, nByte );
1.177 + return rc == NO_ERROR ? SQLITE_OK : SQLITE_IOERR;
1.178 +}
1.179 +
1.180 +#ifdef SQLITE_TEST
1.181 +/*
1.182 +** Count the number of fullsyncs and normal syncs. This is used to test
1.183 +** that syncs and fullsyncs are occuring at the right times.
1.184 +*/
1.185 +int sqlite3_sync_count = 0;
1.186 +int sqlite3_fullsync_count = 0;
1.187 +#endif
1.188 +
1.189 +/*
1.190 +** Make sure all writes to a particular file are committed to disk.
1.191 +*/
1.192 +static int os2Sync( sqlite3_file *id, int flags ){
1.193 + os2File *pFile = (os2File*)id;
1.194 + OSTRACE3( "SYNC %d lock=%d\n", pFile->h, pFile->locktype );
1.195 +#ifdef SQLITE_TEST
1.196 + if( flags & SQLITE_SYNC_FULL){
1.197 + sqlite3_fullsync_count++;
1.198 + }
1.199 + sqlite3_sync_count++;
1.200 +#endif
1.201 + return DosResetBuffer( pFile->h ) == NO_ERROR ? SQLITE_OK : SQLITE_IOERR;
1.202 +}
1.203 +
1.204 +/*
1.205 +** Determine the current size of a file in bytes
1.206 +*/
1.207 +static int os2FileSize( sqlite3_file *id, sqlite3_int64 *pSize ){
1.208 + APIRET rc = NO_ERROR;
1.209 + FILESTATUS3 fsts3FileInfo;
1.210 + memset(&fsts3FileInfo, 0, sizeof(fsts3FileInfo));
1.211 + assert( id!=0 );
1.212 + SimulateIOError( return SQLITE_IOERR );
1.213 + rc = DosQueryFileInfo( ((os2File*)id)->h, FIL_STANDARD, &fsts3FileInfo, sizeof(FILESTATUS3) );
1.214 + if( rc == NO_ERROR ){
1.215 + *pSize = fsts3FileInfo.cbFile;
1.216 + return SQLITE_OK;
1.217 + }else{
1.218 + return SQLITE_IOERR;
1.219 + }
1.220 +}
1.221 +
1.222 +/*
1.223 +** Acquire a reader lock.
1.224 +*/
1.225 +static int getReadLock( os2File *pFile ){
1.226 + FILELOCK LockArea,
1.227 + UnlockArea;
1.228 + APIRET res;
1.229 + memset(&LockArea, 0, sizeof(LockArea));
1.230 + memset(&UnlockArea, 0, sizeof(UnlockArea));
1.231 + LockArea.lOffset = SHARED_FIRST;
1.232 + LockArea.lRange = SHARED_SIZE;
1.233 + UnlockArea.lOffset = 0L;
1.234 + UnlockArea.lRange = 0L;
1.235 + res = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 1L );
1.236 + OSTRACE3( "GETREADLOCK %d res=%d\n", pFile->h, res );
1.237 + return res;
1.238 +}
1.239 +
1.240 +/*
1.241 +** Undo a readlock
1.242 +*/
1.243 +static int unlockReadLock( os2File *id ){
1.244 + FILELOCK LockArea,
1.245 + UnlockArea;
1.246 + APIRET res;
1.247 + memset(&LockArea, 0, sizeof(LockArea));
1.248 + memset(&UnlockArea, 0, sizeof(UnlockArea));
1.249 + LockArea.lOffset = 0L;
1.250 + LockArea.lRange = 0L;
1.251 + UnlockArea.lOffset = SHARED_FIRST;
1.252 + UnlockArea.lRange = SHARED_SIZE;
1.253 + res = DosSetFileLocks( id->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 1L );
1.254 + OSTRACE3( "UNLOCK-READLOCK file handle=%d res=%d?\n", id->h, res );
1.255 + return res;
1.256 +}
1.257 +
1.258 +/*
1.259 +** Lock the file with the lock specified by parameter locktype - one
1.260 +** of the following:
1.261 +**
1.262 +** (1) SHARED_LOCK
1.263 +** (2) RESERVED_LOCK
1.264 +** (3) PENDING_LOCK
1.265 +** (4) EXCLUSIVE_LOCK
1.266 +**
1.267 +** Sometimes when requesting one lock state, additional lock states
1.268 +** are inserted in between. The locking might fail on one of the later
1.269 +** transitions leaving the lock state different from what it started but
1.270 +** still short of its goal. The following chart shows the allowed
1.271 +** transitions and the inserted intermediate states:
1.272 +**
1.273 +** UNLOCKED -> SHARED
1.274 +** SHARED -> RESERVED
1.275 +** SHARED -> (PENDING) -> EXCLUSIVE
1.276 +** RESERVED -> (PENDING) -> EXCLUSIVE
1.277 +** PENDING -> EXCLUSIVE
1.278 +**
1.279 +** This routine will only increase a lock. The os2Unlock() routine
1.280 +** erases all locks at once and returns us immediately to locking level 0.
1.281 +** It is not possible to lower the locking level one step at a time. You
1.282 +** must go straight to locking level 0.
1.283 +*/
1.284 +static int os2Lock( sqlite3_file *id, int locktype ){
1.285 + int rc = SQLITE_OK; /* Return code from subroutines */
1.286 + APIRET res = NO_ERROR; /* Result of an OS/2 lock call */
1.287 + int newLocktype; /* Set pFile->locktype to this value before exiting */
1.288 + int gotPendingLock = 0;/* True if we acquired a PENDING lock this time */
1.289 + FILELOCK LockArea,
1.290 + UnlockArea;
1.291 + os2File *pFile = (os2File*)id;
1.292 + memset(&LockArea, 0, sizeof(LockArea));
1.293 + memset(&UnlockArea, 0, sizeof(UnlockArea));
1.294 + assert( pFile!=0 );
1.295 + OSTRACE4( "LOCK %d %d was %d\n", pFile->h, locktype, pFile->locktype );
1.296 +
1.297 + /* If there is already a lock of this type or more restrictive on the
1.298 + ** os2File, do nothing. Don't use the end_lock: exit path, as
1.299 + ** sqlite3_mutex_enter() hasn't been called yet.
1.300 + */
1.301 + if( pFile->locktype>=locktype ){
1.302 + OSTRACE3( "LOCK %d %d ok (already held)\n", pFile->h, locktype );
1.303 + return SQLITE_OK;
1.304 + }
1.305 +
1.306 + /* Make sure the locking sequence is correct
1.307 + */
1.308 + assert( pFile->locktype!=NO_LOCK || locktype==SHARED_LOCK );
1.309 + assert( locktype!=PENDING_LOCK );
1.310 + assert( locktype!=RESERVED_LOCK || pFile->locktype==SHARED_LOCK );
1.311 +
1.312 + /* Lock the PENDING_LOCK byte if we need to acquire a PENDING lock or
1.313 + ** a SHARED lock. If we are acquiring a SHARED lock, the acquisition of
1.314 + ** the PENDING_LOCK byte is temporary.
1.315 + */
1.316 + newLocktype = pFile->locktype;
1.317 + if( pFile->locktype==NO_LOCK
1.318 + || (locktype==EXCLUSIVE_LOCK && pFile->locktype==RESERVED_LOCK)
1.319 + ){
1.320 + LockArea.lOffset = PENDING_BYTE;
1.321 + LockArea.lRange = 1L;
1.322 + UnlockArea.lOffset = 0L;
1.323 + UnlockArea.lRange = 0L;
1.324 +
1.325 + /* wait longer than LOCK_TIMEOUT here not to have to try multiple times */
1.326 + res = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, 100L, 0L );
1.327 + if( res == NO_ERROR ){
1.328 + gotPendingLock = 1;
1.329 + OSTRACE3( "LOCK %d pending lock boolean set. res=%d\n", pFile->h, res );
1.330 + }
1.331 + }
1.332 +
1.333 + /* Acquire a shared lock
1.334 + */
1.335 + if( locktype==SHARED_LOCK && res == NO_ERROR ){
1.336 + assert( pFile->locktype==NO_LOCK );
1.337 + res = getReadLock(pFile);
1.338 + if( res == NO_ERROR ){
1.339 + newLocktype = SHARED_LOCK;
1.340 + }
1.341 + OSTRACE3( "LOCK %d acquire shared lock. res=%d\n", pFile->h, res );
1.342 + }
1.343 +
1.344 + /* Acquire a RESERVED lock
1.345 + */
1.346 + if( locktype==RESERVED_LOCK && res == NO_ERROR ){
1.347 + assert( pFile->locktype==SHARED_LOCK );
1.348 + LockArea.lOffset = RESERVED_BYTE;
1.349 + LockArea.lRange = 1L;
1.350 + UnlockArea.lOffset = 0L;
1.351 + UnlockArea.lRange = 0L;
1.352 + res = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 0L );
1.353 + if( res == NO_ERROR ){
1.354 + newLocktype = RESERVED_LOCK;
1.355 + }
1.356 + OSTRACE3( "LOCK %d acquire reserved lock. res=%d\n", pFile->h, res );
1.357 + }
1.358 +
1.359 + /* Acquire a PENDING lock
1.360 + */
1.361 + if( locktype==EXCLUSIVE_LOCK && res == NO_ERROR ){
1.362 + newLocktype = PENDING_LOCK;
1.363 + gotPendingLock = 0;
1.364 + OSTRACE2( "LOCK %d acquire pending lock. pending lock boolean unset.\n", pFile->h );
1.365 + }
1.366 +
1.367 + /* Acquire an EXCLUSIVE lock
1.368 + */
1.369 + if( locktype==EXCLUSIVE_LOCK && res == NO_ERROR ){
1.370 + assert( pFile->locktype>=SHARED_LOCK );
1.371 + res = unlockReadLock(pFile);
1.372 + OSTRACE2( "unreadlock = %d\n", res );
1.373 + LockArea.lOffset = SHARED_FIRST;
1.374 + LockArea.lRange = SHARED_SIZE;
1.375 + UnlockArea.lOffset = 0L;
1.376 + UnlockArea.lRange = 0L;
1.377 + res = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 0L );
1.378 + if( res == NO_ERROR ){
1.379 + newLocktype = EXCLUSIVE_LOCK;
1.380 + }else{
1.381 + OSTRACE2( "OS/2 error-code = %d\n", res );
1.382 + getReadLock(pFile);
1.383 + }
1.384 + OSTRACE3( "LOCK %d acquire exclusive lock. res=%d\n", pFile->h, res );
1.385 + }
1.386 +
1.387 + /* If we are holding a PENDING lock that ought to be released, then
1.388 + ** release it now.
1.389 + */
1.390 + if( gotPendingLock && locktype==SHARED_LOCK ){
1.391 + int r;
1.392 + LockArea.lOffset = 0L;
1.393 + LockArea.lRange = 0L;
1.394 + UnlockArea.lOffset = PENDING_BYTE;
1.395 + UnlockArea.lRange = 1L;
1.396 + r = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 0L );
1.397 + OSTRACE3( "LOCK %d unlocking pending/is shared. r=%d\n", pFile->h, r );
1.398 + }
1.399 +
1.400 + /* Update the state of the lock has held in the file descriptor then
1.401 + ** return the appropriate result code.
1.402 + */
1.403 + if( res == NO_ERROR ){
1.404 + rc = SQLITE_OK;
1.405 + }else{
1.406 + OSTRACE4( "LOCK FAILED %d trying for %d but got %d\n", pFile->h,
1.407 + locktype, newLocktype );
1.408 + rc = SQLITE_BUSY;
1.409 + }
1.410 + pFile->locktype = newLocktype;
1.411 + OSTRACE3( "LOCK %d now %d\n", pFile->h, pFile->locktype );
1.412 + return rc;
1.413 +}
1.414 +
1.415 +/*
1.416 +** This routine checks if there is a RESERVED lock held on the specified
1.417 +** file by this or any other process. If such a lock is held, return
1.418 +** non-zero, otherwise zero.
1.419 +*/
1.420 +static int os2CheckReservedLock( sqlite3_file *id, int *pOut ){
1.421 + int r = 0;
1.422 + os2File *pFile = (os2File*)id;
1.423 + assert( pFile!=0 );
1.424 + if( pFile->locktype>=RESERVED_LOCK ){
1.425 + r = 1;
1.426 + OSTRACE3( "TEST WR-LOCK %d %d (local)\n", pFile->h, r );
1.427 + }else{
1.428 + FILELOCK LockArea,
1.429 + UnlockArea;
1.430 + APIRET rc = NO_ERROR;
1.431 + memset(&LockArea, 0, sizeof(LockArea));
1.432 + memset(&UnlockArea, 0, sizeof(UnlockArea));
1.433 + LockArea.lOffset = RESERVED_BYTE;
1.434 + LockArea.lRange = 1L;
1.435 + UnlockArea.lOffset = 0L;
1.436 + UnlockArea.lRange = 0L;
1.437 + rc = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 0L );
1.438 + OSTRACE3( "TEST WR-LOCK %d lock reserved byte rc=%d\n", pFile->h, rc );
1.439 + if( rc == NO_ERROR ){
1.440 + APIRET rcu = NO_ERROR; /* return code for unlocking */
1.441 + LockArea.lOffset = 0L;
1.442 + LockArea.lRange = 0L;
1.443 + UnlockArea.lOffset = RESERVED_BYTE;
1.444 + UnlockArea.lRange = 1L;
1.445 + rcu = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 0L );
1.446 + OSTRACE3( "TEST WR-LOCK %d unlock reserved byte r=%d\n", pFile->h, rcu );
1.447 + }
1.448 + r = !(rc == NO_ERROR);
1.449 + OSTRACE3( "TEST WR-LOCK %d %d (remote)\n", pFile->h, r );
1.450 + }
1.451 + *pOut = r;
1.452 + return SQLITE_OK;
1.453 +}
1.454 +
1.455 +/*
1.456 +** Lower the locking level on file descriptor id to locktype. locktype
1.457 +** must be either NO_LOCK or SHARED_LOCK.
1.458 +**
1.459 +** If the locking level of the file descriptor is already at or below
1.460 +** the requested locking level, this routine is a no-op.
1.461 +**
1.462 +** It is not possible for this routine to fail if the second argument
1.463 +** is NO_LOCK. If the second argument is SHARED_LOCK then this routine
1.464 +** might return SQLITE_IOERR;
1.465 +*/
1.466 +static int os2Unlock( sqlite3_file *id, int locktype ){
1.467 + int type;
1.468 + os2File *pFile = (os2File*)id;
1.469 + APIRET rc = SQLITE_OK;
1.470 + APIRET res = NO_ERROR;
1.471 + FILELOCK LockArea,
1.472 + UnlockArea;
1.473 + memset(&LockArea, 0, sizeof(LockArea));
1.474 + memset(&UnlockArea, 0, sizeof(UnlockArea));
1.475 + assert( pFile!=0 );
1.476 + assert( locktype<=SHARED_LOCK );
1.477 + OSTRACE4( "UNLOCK %d to %d was %d\n", pFile->h, locktype, pFile->locktype );
1.478 + type = pFile->locktype;
1.479 + if( type>=EXCLUSIVE_LOCK ){
1.480 + LockArea.lOffset = 0L;
1.481 + LockArea.lRange = 0L;
1.482 + UnlockArea.lOffset = SHARED_FIRST;
1.483 + UnlockArea.lRange = SHARED_SIZE;
1.484 + res = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 0L );
1.485 + OSTRACE3( "UNLOCK %d exclusive lock res=%d\n", pFile->h, res );
1.486 + if( locktype==SHARED_LOCK && getReadLock(pFile) != NO_ERROR ){
1.487 + /* This should never happen. We should always be able to
1.488 + ** reacquire the read lock */
1.489 + OSTRACE3( "UNLOCK %d to %d getReadLock() failed\n", pFile->h, locktype );
1.490 + rc = SQLITE_IOERR_UNLOCK;
1.491 + }
1.492 + }
1.493 + if( type>=RESERVED_LOCK ){
1.494 + LockArea.lOffset = 0L;
1.495 + LockArea.lRange = 0L;
1.496 + UnlockArea.lOffset = RESERVED_BYTE;
1.497 + UnlockArea.lRange = 1L;
1.498 + res = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 0L );
1.499 + OSTRACE3( "UNLOCK %d reserved res=%d\n", pFile->h, res );
1.500 + }
1.501 + if( locktype==NO_LOCK && type>=SHARED_LOCK ){
1.502 + res = unlockReadLock(pFile);
1.503 + OSTRACE5( "UNLOCK %d is %d want %d res=%d\n", pFile->h, type, locktype, res );
1.504 + }
1.505 + if( type>=PENDING_LOCK ){
1.506 + LockArea.lOffset = 0L;
1.507 + LockArea.lRange = 0L;
1.508 + UnlockArea.lOffset = PENDING_BYTE;
1.509 + UnlockArea.lRange = 1L;
1.510 + res = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 0L );
1.511 + OSTRACE3( "UNLOCK %d pending res=%d\n", pFile->h, res );
1.512 + }
1.513 + pFile->locktype = locktype;
1.514 + OSTRACE3( "UNLOCK %d now %d\n", pFile->h, pFile->locktype );
1.515 + return rc;
1.516 +}
1.517 +
1.518 +/*
1.519 +** Control and query of the open file handle.
1.520 +*/
1.521 +static int os2FileControl(sqlite3_file *id, int op, void *pArg){
1.522 + switch( op ){
1.523 + case SQLITE_FCNTL_LOCKSTATE: {
1.524 + *(int*)pArg = ((os2File*)id)->locktype;
1.525 + OSTRACE3( "FCNTL_LOCKSTATE %d lock=%d\n", ((os2File*)id)->h, ((os2File*)id)->locktype );
1.526 + return SQLITE_OK;
1.527 + }
1.528 + }
1.529 + return SQLITE_ERROR;
1.530 +}
1.531 +
1.532 +/*
1.533 +** Return the sector size in bytes of the underlying block device for
1.534 +** the specified file. This is almost always 512 bytes, but may be
1.535 +** larger for some devices.
1.536 +**
1.537 +** SQLite code assumes this function cannot fail. It also assumes that
1.538 +** if two files are created in the same file-system directory (i.e.
1.539 +** a database and its journal file) that the sector size will be the
1.540 +** same for both.
1.541 +*/
1.542 +static int os2SectorSize(sqlite3_file *id){
1.543 + return SQLITE_DEFAULT_SECTOR_SIZE;
1.544 +}
1.545 +
1.546 +/*
1.547 +** Return a vector of device characteristics.
1.548 +*/
1.549 +static int os2DeviceCharacteristics(sqlite3_file *id){
1.550 + return 0;
1.551 +}
1.552 +
1.553 +
1.554 +/*
1.555 +** Character set conversion objects used by conversion routines.
1.556 +*/
1.557 +static UconvObject ucUtf8 = NULL; /* convert between UTF-8 and UCS-2 */
1.558 +static UconvObject uclCp = NULL; /* convert between local codepage and UCS-2 */
1.559 +
1.560 +/*
1.561 +** Helper function to initialize the conversion objects from and to UTF-8.
1.562 +*/
1.563 +static void initUconvObjects( void ){
1.564 + if( UniCreateUconvObject( UTF_8, &ucUtf8 ) != ULS_SUCCESS )
1.565 + ucUtf8 = NULL;
1.566 + if ( UniCreateUconvObject( (UniChar *)L"@path=yes", &uclCp ) != ULS_SUCCESS )
1.567 + uclCp = NULL;
1.568 +}
1.569 +
1.570 +/*
1.571 +** Helper function to free the conversion objects from and to UTF-8.
1.572 +*/
1.573 +static void freeUconvObjects( void ){
1.574 + if ( ucUtf8 )
1.575 + UniFreeUconvObject( ucUtf8 );
1.576 + if ( uclCp )
1.577 + UniFreeUconvObject( uclCp );
1.578 + ucUtf8 = NULL;
1.579 + uclCp = NULL;
1.580 +}
1.581 +
1.582 +/*
1.583 +** Helper function to convert UTF-8 filenames to local OS/2 codepage.
1.584 +** The two-step process: first convert the incoming UTF-8 string
1.585 +** into UCS-2 and then from UCS-2 to the current codepage.
1.586 +** The returned char pointer has to be freed.
1.587 +*/
1.588 +static char *convertUtf8PathToCp( const char *in ){
1.589 + UniChar tempPath[CCHMAXPATH];
1.590 + char *out = (char *)calloc( CCHMAXPATH, 1 );
1.591 +
1.592 + if( !out )
1.593 + return NULL;
1.594 +
1.595 + if( !ucUtf8 || !uclCp )
1.596 + initUconvObjects();
1.597 +
1.598 + /* determine string for the conversion of UTF-8 which is CP1208 */
1.599 + if( UniStrToUcs( ucUtf8, tempPath, (char *)in, CCHMAXPATH ) != ULS_SUCCESS )
1.600 + return out; /* if conversion fails, return the empty string */
1.601 +
1.602 + /* conversion for current codepage which can be used for paths */
1.603 + UniStrFromUcs( uclCp, out, tempPath, CCHMAXPATH );
1.604 +
1.605 + return out;
1.606 +}
1.607 +
1.608 +/*
1.609 +** Helper function to convert filenames from local codepage to UTF-8.
1.610 +** The two-step process: first convert the incoming codepage-specific
1.611 +** string into UCS-2 and then from UCS-2 to the codepage of UTF-8.
1.612 +** The returned char pointer has to be freed.
1.613 +**
1.614 +** This function is non-static to be able to use this in shell.c and
1.615 +** similar applications that take command line arguments.
1.616 +*/
1.617 +char *convertCpPathToUtf8( const char *in ){
1.618 + UniChar tempPath[CCHMAXPATH];
1.619 + char *out = (char *)calloc( CCHMAXPATH, 1 );
1.620 +
1.621 + if( !out )
1.622 + return NULL;
1.623 +
1.624 + if( !ucUtf8 || !uclCp )
1.625 + initUconvObjects();
1.626 +
1.627 + /* conversion for current codepage which can be used for paths */
1.628 + if( UniStrToUcs( uclCp, tempPath, (char *)in, CCHMAXPATH ) != ULS_SUCCESS )
1.629 + return out; /* if conversion fails, return the empty string */
1.630 +
1.631 + /* determine string for the conversion of UTF-8 which is CP1208 */
1.632 + UniStrFromUcs( ucUtf8, out, tempPath, CCHMAXPATH );
1.633 +
1.634 + return out;
1.635 +}
1.636 +
1.637 +/*
1.638 +** This vector defines all the methods that can operate on an
1.639 +** sqlite3_file for os2.
1.640 +*/
1.641 +static const sqlite3_io_methods os2IoMethod = {
1.642 + 1, /* iVersion */
1.643 + os2Close,
1.644 + os2Read,
1.645 + os2Write,
1.646 + os2Truncate,
1.647 + os2Sync,
1.648 + os2FileSize,
1.649 + os2Lock,
1.650 + os2Unlock,
1.651 + os2CheckReservedLock,
1.652 + os2FileControl,
1.653 + os2SectorSize,
1.654 + os2DeviceCharacteristics
1.655 +};
1.656 +
1.657 +/***************************************************************************
1.658 +** Here ends the I/O methods that form the sqlite3_io_methods object.
1.659 +**
1.660 +** The next block of code implements the VFS methods.
1.661 +****************************************************************************/
1.662 +
1.663 +/*
1.664 +** Create a temporary file name in zBuf. zBuf must be big enough to
1.665 +** hold at pVfs->mxPathname characters.
1.666 +*/
1.667 +static int getTempname(int nBuf, char *zBuf ){
1.668 + static const unsigned char zChars[] =
1.669 + "abcdefghijklmnopqrstuvwxyz"
1.670 + "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
1.671 + "0123456789";
1.672 + int i, j;
1.673 + char zTempPathBuf[3];
1.674 + PSZ zTempPath = (PSZ)&zTempPathBuf;
1.675 + if( sqlite3_temp_directory ){
1.676 + zTempPath = sqlite3_temp_directory;
1.677 + }else{
1.678 + if( DosScanEnv( (PSZ)"TEMP", &zTempPath ) ){
1.679 + if( DosScanEnv( (PSZ)"TMP", &zTempPath ) ){
1.680 + if( DosScanEnv( (PSZ)"TMPDIR", &zTempPath ) ){
1.681 + ULONG ulDriveNum = 0, ulDriveMap = 0;
1.682 + DosQueryCurrentDisk( &ulDriveNum, &ulDriveMap );
1.683 + sprintf( (char*)zTempPath, "%c:", (char)( 'A' + ulDriveNum - 1 ) );
1.684 + }
1.685 + }
1.686 + }
1.687 + }
1.688 + /* Strip off a trailing slashes or backslashes, otherwise we would get *
1.689 + * multiple (back)slashes which causes DosOpen() to fail. *
1.690 + * Trailing spaces are not allowed, either. */
1.691 + j = strlen(zTempPath);
1.692 + while( j > 0 && ( zTempPath[j-1] == '\\' || zTempPath[j-1] == '/'
1.693 + || zTempPath[j-1] == ' ' ) ){
1.694 + j--;
1.695 + }
1.696 + zTempPath[j] = '\0';
1.697 + if( !sqlite3_temp_directory ){
1.698 + char *zTempPathUTF = convertCpPathToUtf8( zTempPath );
1.699 + sqlite3_snprintf( nBuf-30, zBuf,
1.700 + "%s\\"SQLITE_TEMP_FILE_PREFIX, zTempPathUTF );
1.701 + free( zTempPathUTF );
1.702 + }else{
1.703 + sqlite3_snprintf( nBuf-30, zBuf,
1.704 + "%s\\"SQLITE_TEMP_FILE_PREFIX, zTempPath );
1.705 + }
1.706 + j = strlen( zBuf );
1.707 + sqlite3_randomness( 20, &zBuf[j] );
1.708 + for( i = 0; i < 20; i++, j++ ){
1.709 + zBuf[j] = (char)zChars[ ((unsigned char)zBuf[j])%(sizeof(zChars)-1) ];
1.710 + }
1.711 + zBuf[j] = 0;
1.712 + OSTRACE2( "TEMP FILENAME: %s\n", zBuf );
1.713 + return SQLITE_OK;
1.714 +}
1.715 +
1.716 +
1.717 +/*
1.718 +** Turn a relative pathname into a full pathname. Write the full
1.719 +** pathname into zFull[]. zFull[] will be at least pVfs->mxPathname
1.720 +** bytes in size.
1.721 +*/
1.722 +static int os2FullPathname(
1.723 + sqlite3_vfs *pVfs, /* Pointer to vfs object */
1.724 + const char *zRelative, /* Possibly relative input path */
1.725 + int nFull, /* Size of output buffer in bytes */
1.726 + char *zFull /* Output buffer */
1.727 +){
1.728 + char *zRelativeCp = convertUtf8PathToCp( zRelative );
1.729 + char zFullCp[CCHMAXPATH] = "\0";
1.730 + char *zFullUTF;
1.731 + APIRET rc = DosQueryPathInfo( zRelativeCp, FIL_QUERYFULLNAME, zFullCp,
1.732 + CCHMAXPATH );
1.733 + free( zRelativeCp );
1.734 + zFullUTF = convertCpPathToUtf8( zFullCp );
1.735 + sqlite3_snprintf( nFull, zFull, zFullUTF );
1.736 + free( zFullUTF );
1.737 + return rc == NO_ERROR ? SQLITE_OK : SQLITE_IOERR;
1.738 +}
1.739 +
1.740 +
1.741 +/*
1.742 +** Open a file.
1.743 +*/
1.744 +static int os2Open(
1.745 + sqlite3_vfs *pVfs, /* Not used */
1.746 + const char *zName, /* Name of the file */
1.747 + sqlite3_file *id, /* Write the SQLite file handle here */
1.748 + int flags, /* Open mode flags */
1.749 + int *pOutFlags /* Status return flags */
1.750 +){
1.751 + HFILE h;
1.752 + ULONG ulFileAttribute = FILE_NORMAL;
1.753 + ULONG ulOpenFlags = 0;
1.754 + ULONG ulOpenMode = 0;
1.755 + os2File *pFile = (os2File*)id;
1.756 + APIRET rc = NO_ERROR;
1.757 + ULONG ulAction;
1.758 + char *zNameCp;
1.759 + char zTmpname[CCHMAXPATH+1]; /* Buffer to hold name of temp file */
1.760 +
1.761 + /* If the second argument to this function is NULL, generate a
1.762 + ** temporary file name to use
1.763 + */
1.764 + if( !zName ){
1.765 + int rc = getTempname(CCHMAXPATH+1, zTmpname);
1.766 + if( rc!=SQLITE_OK ){
1.767 + return rc;
1.768 + }
1.769 + zName = zTmpname;
1.770 + }
1.771 +
1.772 +
1.773 + memset( pFile, 0, sizeof(*pFile) );
1.774 +
1.775 + OSTRACE2( "OPEN want %d\n", flags );
1.776 +
1.777 + if( flags & SQLITE_OPEN_READWRITE ){
1.778 + ulOpenMode |= OPEN_ACCESS_READWRITE;
1.779 + OSTRACE1( "OPEN read/write\n" );
1.780 + }else{
1.781 + ulOpenMode |= OPEN_ACCESS_READONLY;
1.782 + OSTRACE1( "OPEN read only\n" );
1.783 + }
1.784 +
1.785 + if( flags & SQLITE_OPEN_CREATE ){
1.786 + ulOpenFlags |= OPEN_ACTION_OPEN_IF_EXISTS | OPEN_ACTION_CREATE_IF_NEW;
1.787 + OSTRACE1( "OPEN open new/create\n" );
1.788 + }else{
1.789 + ulOpenFlags |= OPEN_ACTION_OPEN_IF_EXISTS | OPEN_ACTION_FAIL_IF_NEW;
1.790 + OSTRACE1( "OPEN open existing\n" );
1.791 + }
1.792 +
1.793 + if( flags & SQLITE_OPEN_MAIN_DB ){
1.794 + ulOpenMode |= OPEN_SHARE_DENYNONE;
1.795 + OSTRACE1( "OPEN share read/write\n" );
1.796 + }else{
1.797 + ulOpenMode |= OPEN_SHARE_DENYWRITE;
1.798 + OSTRACE1( "OPEN share read only\n" );
1.799 + }
1.800 +
1.801 + if( flags & SQLITE_OPEN_DELETEONCLOSE ){
1.802 + char pathUtf8[CCHMAXPATH];
1.803 +#ifdef NDEBUG /* when debugging we want to make sure it is deleted */
1.804 + ulFileAttribute = FILE_HIDDEN;
1.805 +#endif
1.806 + os2FullPathname( pVfs, zName, CCHMAXPATH, pathUtf8 );
1.807 + pFile->pathToDel = convertUtf8PathToCp( pathUtf8 );
1.808 + OSTRACE1( "OPEN hidden/delete on close file attributes\n" );
1.809 + }else{
1.810 + pFile->pathToDel = NULL;
1.811 + OSTRACE1( "OPEN normal file attribute\n" );
1.812 + }
1.813 +
1.814 + /* always open in random access mode for possibly better speed */
1.815 + ulOpenMode |= OPEN_FLAGS_RANDOM;
1.816 + ulOpenMode |= OPEN_FLAGS_FAIL_ON_ERROR;
1.817 + ulOpenMode |= OPEN_FLAGS_NOINHERIT;
1.818 +
1.819 + zNameCp = convertUtf8PathToCp( zName );
1.820 + rc = DosOpen( (PSZ)zNameCp,
1.821 + &h,
1.822 + &ulAction,
1.823 + 0L,
1.824 + ulFileAttribute,
1.825 + ulOpenFlags,
1.826 + ulOpenMode,
1.827 + (PEAOP2)NULL );
1.828 + free( zNameCp );
1.829 + if( rc != NO_ERROR ){
1.830 + OSTRACE7( "OPEN Invalid handle rc=%d: zName=%s, ulAction=%#lx, ulAttr=%#lx, ulFlags=%#lx, ulMode=%#lx\n",
1.831 + rc, zName, ulAction, ulFileAttribute, ulOpenFlags, ulOpenMode );
1.832 + if( pFile->pathToDel )
1.833 + free( pFile->pathToDel );
1.834 + pFile->pathToDel = NULL;
1.835 + if( flags & SQLITE_OPEN_READWRITE ){
1.836 + OSTRACE2( "OPEN %d Invalid handle\n", ((flags | SQLITE_OPEN_READONLY) & ~SQLITE_OPEN_READWRITE) );
1.837 + return os2Open( pVfs, zName, id,
1.838 + ((flags | SQLITE_OPEN_READONLY) & ~SQLITE_OPEN_READWRITE),
1.839 + pOutFlags );
1.840 + }else{
1.841 + return SQLITE_CANTOPEN;
1.842 + }
1.843 + }
1.844 +
1.845 + if( pOutFlags ){
1.846 + *pOutFlags = flags & SQLITE_OPEN_READWRITE ? SQLITE_OPEN_READWRITE : SQLITE_OPEN_READONLY;
1.847 + }
1.848 +
1.849 + pFile->pMethod = &os2IoMethod;
1.850 + pFile->h = h;
1.851 + OpenCounter(+1);
1.852 + OSTRACE3( "OPEN %d pOutFlags=%d\n", pFile->h, pOutFlags );
1.853 + return SQLITE_OK;
1.854 +}
1.855 +
1.856 +/*
1.857 +** Delete the named file.
1.858 +*/
1.859 +static int os2Delete(
1.860 + sqlite3_vfs *pVfs, /* Not used on os2 */
1.861 + const char *zFilename, /* Name of file to delete */
1.862 + int syncDir /* Not used on os2 */
1.863 +){
1.864 + APIRET rc = NO_ERROR;
1.865 + char *zFilenameCp = convertUtf8PathToCp( zFilename );
1.866 + SimulateIOError( return SQLITE_IOERR_DELETE );
1.867 + rc = DosDelete( (PSZ)zFilenameCp );
1.868 + free( zFilenameCp );
1.869 + OSTRACE2( "DELETE \"%s\"\n", zFilename );
1.870 + return rc == NO_ERROR ? SQLITE_OK : SQLITE_IOERR;
1.871 +}
1.872 +
1.873 +/*
1.874 +** Check the existance and status of a file.
1.875 +*/
1.876 +static int os2Access(
1.877 + sqlite3_vfs *pVfs, /* Not used on os2 */
1.878 + const char *zFilename, /* Name of file to check */
1.879 + int flags, /* Type of test to make on this file */
1.880 + int *pOut /* Write results here */
1.881 +){
1.882 + FILESTATUS3 fsts3ConfigInfo;
1.883 + APIRET rc = NO_ERROR;
1.884 + char *zFilenameCp = convertUtf8PathToCp( zFilename );
1.885 +
1.886 + memset( &fsts3ConfigInfo, 0, sizeof(fsts3ConfigInfo) );
1.887 + rc = DosQueryPathInfo( (PSZ)zFilenameCp, FIL_STANDARD,
1.888 + &fsts3ConfigInfo, sizeof(FILESTATUS3) );
1.889 + free( zFilenameCp );
1.890 + OSTRACE4( "ACCESS fsts3ConfigInfo.attrFile=%d flags=%d rc=%d\n",
1.891 + fsts3ConfigInfo.attrFile, flags, rc );
1.892 + switch( flags ){
1.893 + case SQLITE_ACCESS_READ:
1.894 + case SQLITE_ACCESS_EXISTS:
1.895 + rc = (rc == NO_ERROR);
1.896 + OSTRACE3( "ACCESS %s access of read and exists rc=%d\n", zFilename, rc );
1.897 + break;
1.898 + case SQLITE_ACCESS_READWRITE:
1.899 + rc = (rc == NO_ERROR) && ( (fsts3ConfigInfo.attrFile & FILE_READONLY) == 0 );
1.900 + OSTRACE3( "ACCESS %s access of read/write rc=%d\n", zFilename, rc );
1.901 + break;
1.902 + default:
1.903 + assert( !"Invalid flags argument" );
1.904 + }
1.905 + *pOut = rc;
1.906 + return SQLITE_OK;
1.907 +}
1.908 +
1.909 +
1.910 +#ifndef SQLITE_OMIT_LOAD_EXTENSION
1.911 +/*
1.912 +** Interfaces for opening a shared library, finding entry points
1.913 +** within the shared library, and closing the shared library.
1.914 +*/
1.915 +/*
1.916 +** Interfaces for opening a shared library, finding entry points
1.917 +** within the shared library, and closing the shared library.
1.918 +*/
1.919 +static void *os2DlOpen(sqlite3_vfs *pVfs, const char *zFilename){
1.920 + UCHAR loadErr[256];
1.921 + HMODULE hmod;
1.922 + APIRET rc;
1.923 + char *zFilenameCp = convertUtf8PathToCp(zFilename);
1.924 + rc = DosLoadModule((PSZ)loadErr, sizeof(loadErr), zFilenameCp, &hmod);
1.925 + free(zFilenameCp);
1.926 + return rc != NO_ERROR ? 0 : (void*)hmod;
1.927 +}
1.928 +/*
1.929 +** A no-op since the error code is returned on the DosLoadModule call.
1.930 +** os2Dlopen returns zero if DosLoadModule is not successful.
1.931 +*/
1.932 +static void os2DlError(sqlite3_vfs *pVfs, int nBuf, char *zBufOut){
1.933 +/* no-op */
1.934 +}
1.935 +static void *os2DlSym(sqlite3_vfs *pVfs, void *pHandle, const char *zSymbol){
1.936 + PFN pfn;
1.937 + APIRET rc;
1.938 + rc = DosQueryProcAddr((HMODULE)pHandle, 0L, zSymbol, &pfn);
1.939 + if( rc != NO_ERROR ){
1.940 + /* if the symbol itself was not found, search again for the same
1.941 + * symbol with an extra underscore, that might be needed depending
1.942 + * on the calling convention */
1.943 + char _zSymbol[256] = "_";
1.944 + strncat(_zSymbol, zSymbol, 255);
1.945 + rc = DosQueryProcAddr((HMODULE)pHandle, 0L, _zSymbol, &pfn);
1.946 + }
1.947 + return rc != NO_ERROR ? 0 : (void*)pfn;
1.948 +}
1.949 +static void os2DlClose(sqlite3_vfs *pVfs, void *pHandle){
1.950 + DosFreeModule((HMODULE)pHandle);
1.951 +}
1.952 +#else /* if SQLITE_OMIT_LOAD_EXTENSION is defined: */
1.953 + #define os2DlOpen 0
1.954 + #define os2DlError 0
1.955 + #define os2DlSym 0
1.956 + #define os2DlClose 0
1.957 +#endif
1.958 +
1.959 +
1.960 +/*
1.961 +** Write up to nBuf bytes of randomness into zBuf.
1.962 +*/
1.963 +static int os2Randomness(sqlite3_vfs *pVfs, int nBuf, char *zBuf ){
1.964 + ULONG sizeofULong = sizeof(ULONG);
1.965 + int n = 0;
1.966 + if( sizeof(DATETIME) <= nBuf - n ){
1.967 + DATETIME x;
1.968 + DosGetDateTime(&x);
1.969 + memcpy(&zBuf[n], &x, sizeof(x));
1.970 + n += sizeof(x);
1.971 + }
1.972 +
1.973 + if( sizeofULong <= nBuf - n ){
1.974 + PPIB ppib;
1.975 + DosGetInfoBlocks(NULL, &ppib);
1.976 + memcpy(&zBuf[n], &ppib->pib_ulpid, sizeofULong);
1.977 + n += sizeofULong;
1.978 + }
1.979 +
1.980 + if( sizeofULong <= nBuf - n ){
1.981 + PTIB ptib;
1.982 + DosGetInfoBlocks(&ptib, NULL);
1.983 + memcpy(&zBuf[n], &ptib->tib_ptib2->tib2_ultid, sizeofULong);
1.984 + n += sizeofULong;
1.985 + }
1.986 +
1.987 + /* if we still haven't filled the buffer yet the following will */
1.988 + /* grab everything once instead of making several calls for a single item */
1.989 + if( sizeofULong <= nBuf - n ){
1.990 + ULONG ulSysInfo[QSV_MAX];
1.991 + DosQuerySysInfo(1L, QSV_MAX, ulSysInfo, sizeofULong * QSV_MAX);
1.992 +
1.993 + memcpy(&zBuf[n], &ulSysInfo[QSV_MS_COUNT - 1], sizeofULong);
1.994 + n += sizeofULong;
1.995 +
1.996 + if( sizeofULong <= nBuf - n ){
1.997 + memcpy(&zBuf[n], &ulSysInfo[QSV_TIMER_INTERVAL - 1], sizeofULong);
1.998 + n += sizeofULong;
1.999 + }
1.1000 + if( sizeofULong <= nBuf - n ){
1.1001 + memcpy(&zBuf[n], &ulSysInfo[QSV_TIME_LOW - 1], sizeofULong);
1.1002 + n += sizeofULong;
1.1003 + }
1.1004 + if( sizeofULong <= nBuf - n ){
1.1005 + memcpy(&zBuf[n], &ulSysInfo[QSV_TIME_HIGH - 1], sizeofULong);
1.1006 + n += sizeofULong;
1.1007 + }
1.1008 + if( sizeofULong <= nBuf - n ){
1.1009 + memcpy(&zBuf[n], &ulSysInfo[QSV_TOTAVAILMEM - 1], sizeofULong);
1.1010 + n += sizeofULong;
1.1011 + }
1.1012 + }
1.1013 +
1.1014 + return n;
1.1015 +}
1.1016 +
1.1017 +/*
1.1018 +** Sleep for a little while. Return the amount of time slept.
1.1019 +** The argument is the number of microseconds we want to sleep.
1.1020 +** The return value is the number of microseconds of sleep actually
1.1021 +** requested from the underlying operating system, a number which
1.1022 +** might be greater than or equal to the argument, but not less
1.1023 +** than the argument.
1.1024 +*/
1.1025 +static int os2Sleep( sqlite3_vfs *pVfs, int microsec ){
1.1026 + DosSleep( (microsec/1000) );
1.1027 + return microsec;
1.1028 +}
1.1029 +
1.1030 +/*
1.1031 +** The following variable, if set to a non-zero value, becomes the result
1.1032 +** returned from sqlite3OsCurrentTime(). This is used for testing.
1.1033 +*/
1.1034 +#ifdef SQLITE_TEST
1.1035 +int sqlite3_current_time = 0;
1.1036 +#endif
1.1037 +
1.1038 +/*
1.1039 +** Find the current time (in Universal Coordinated Time). Write the
1.1040 +** current time and date as a Julian Day number into *prNow and
1.1041 +** return 0. Return 1 if the time and date cannot be found.
1.1042 +*/
1.1043 +int os2CurrentTime( sqlite3_vfs *pVfs, double *prNow ){
1.1044 + double now;
1.1045 + SHORT minute; /* needs to be able to cope with negative timezone offset */
1.1046 + USHORT second, hour,
1.1047 + day, month, year;
1.1048 + DATETIME dt;
1.1049 + DosGetDateTime( &dt );
1.1050 + second = (USHORT)dt.seconds;
1.1051 + minute = (SHORT)dt.minutes + dt.timezone;
1.1052 + hour = (USHORT)dt.hours;
1.1053 + day = (USHORT)dt.day;
1.1054 + month = (USHORT)dt.month;
1.1055 + year = (USHORT)dt.year;
1.1056 +
1.1057 + /* Calculations from http://www.astro.keele.ac.uk/~rno/Astronomy/hjd.html
1.1058 + http://www.astro.keele.ac.uk/~rno/Astronomy/hjd-0.1.c */
1.1059 + /* Calculate the Julian days */
1.1060 + now = day - 32076 +
1.1061 + 1461*(year + 4800 + (month - 14)/12)/4 +
1.1062 + 367*(month - 2 - (month - 14)/12*12)/12 -
1.1063 + 3*((year + 4900 + (month - 14)/12)/100)/4;
1.1064 +
1.1065 + /* Add the fractional hours, mins and seconds */
1.1066 + now += (hour + 12.0)/24.0;
1.1067 + now += minute/1440.0;
1.1068 + now += second/86400.0;
1.1069 + *prNow = now;
1.1070 +#ifdef SQLITE_TEST
1.1071 + if( sqlite3_current_time ){
1.1072 + *prNow = sqlite3_current_time/86400.0 + 2440587.5;
1.1073 + }
1.1074 +#endif
1.1075 + return 0;
1.1076 +}
1.1077 +
1.1078 +static int os2GetLastError(sqlite3_vfs *pVfs, int nBuf, char *zBuf){
1.1079 + return 0;
1.1080 +}
1.1081 +
1.1082 +/*
1.1083 +** Initialize and deinitialize the operating system interface.
1.1084 +*/
1.1085 +int sqlite3_os_init(void){
1.1086 + static sqlite3_vfs os2Vfs = {
1.1087 + 1, /* iVersion */
1.1088 + sizeof(os2File), /* szOsFile */
1.1089 + CCHMAXPATH, /* mxPathname */
1.1090 + 0, /* pNext */
1.1091 + "os2", /* zName */
1.1092 + 0, /* pAppData */
1.1093 +
1.1094 + os2Open, /* xOpen */
1.1095 + os2Delete, /* xDelete */
1.1096 + os2Access, /* xAccess */
1.1097 + os2FullPathname, /* xFullPathname */
1.1098 + os2DlOpen, /* xDlOpen */
1.1099 + os2DlError, /* xDlError */
1.1100 + os2DlSym, /* xDlSym */
1.1101 + os2DlClose, /* xDlClose */
1.1102 + os2Randomness, /* xRandomness */
1.1103 + os2Sleep, /* xSleep */
1.1104 + os2CurrentTime, /* xCurrentTime */
1.1105 + os2GetLastError /* xGetLastError */
1.1106 + };
1.1107 + sqlite3_vfs_register(&os2Vfs, 1);
1.1108 + initUconvObjects();
1.1109 + return SQLITE_OK;
1.1110 +}
1.1111 +int sqlite3_os_end(void){
1.1112 + freeUconvObjects();
1.1113 + return SQLITE_OK;
1.1114 +}
1.1115 +
1.1116 +#endif /* SQLITE_OS_OS2 */