os/persistentdata/persistentstorage/sql/SQLite364/os_os2.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
** 2006 Feb 14
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 code that is specific to OS/2.
sl@0
    14
**
sl@0
    15
** $Id: os_os2.c,v 1.57 2008/10/13 21:46:47 pweilbacher Exp $
sl@0
    16
*/
sl@0
    17
sl@0
    18
#include "sqliteInt.h"
sl@0
    19
sl@0
    20
#if SQLITE_OS_OS2
sl@0
    21
sl@0
    22
/*
sl@0
    23
** A Note About Memory Allocation:
sl@0
    24
**
sl@0
    25
** This driver uses malloc()/free() directly rather than going through
sl@0
    26
** the SQLite-wrappers sqlite3_malloc()/sqlite3_free().  Those wrappers
sl@0
    27
** are designed for use on embedded systems where memory is scarce and
sl@0
    28
** malloc failures happen frequently.  OS/2 does not typically run on
sl@0
    29
** embedded systems, and when it does the developers normally have bigger
sl@0
    30
** problems to worry about than running out of memory.  So there is not
sl@0
    31
** a compelling need to use the wrappers.
sl@0
    32
**
sl@0
    33
** But there is a good reason to not use the wrappers.  If we use the
sl@0
    34
** wrappers then we will get simulated malloc() failures within this
sl@0
    35
** driver.  And that causes all kinds of problems for our tests.  We
sl@0
    36
** could enhance SQLite to deal with simulated malloc failures within
sl@0
    37
** the OS driver, but the code to deal with those failure would not
sl@0
    38
** be exercised on Linux (which does not need to malloc() in the driver)
sl@0
    39
** and so we would have difficulty writing coverage tests for that
sl@0
    40
** code.  Better to leave the code out, we think.
sl@0
    41
**
sl@0
    42
** The point of this discussion is as follows:  When creating a new
sl@0
    43
** OS layer for an embedded system, if you use this file as an example,
sl@0
    44
** avoid the use of malloc()/free().  Those routines work ok on OS/2
sl@0
    45
** desktops but not so well in embedded systems.
sl@0
    46
*/
sl@0
    47
sl@0
    48
/*
sl@0
    49
** Macros used to determine whether or not to use threads.
sl@0
    50
*/
sl@0
    51
#if defined(SQLITE_THREADSAFE) && SQLITE_THREADSAFE
sl@0
    52
# define SQLITE_OS2_THREADS 1
sl@0
    53
#endif
sl@0
    54
sl@0
    55
/*
sl@0
    56
** Include code that is common to all os_*.c files
sl@0
    57
*/
sl@0
    58
#include "os_common.h"
sl@0
    59
sl@0
    60
/*
sl@0
    61
** The os2File structure is subclass of sqlite3_file specific for the OS/2
sl@0
    62
** protability layer.
sl@0
    63
*/
sl@0
    64
typedef struct os2File os2File;
sl@0
    65
struct os2File {
sl@0
    66
  const sqlite3_io_methods *pMethod;  /* Always the first entry */
sl@0
    67
  HFILE h;                  /* Handle for accessing the file */
sl@0
    68
  char* pathToDel;          /* Name of file to delete on close, NULL if not */
sl@0
    69
  unsigned char locktype;   /* Type of lock currently held on this file */
sl@0
    70
};
sl@0
    71
sl@0
    72
#define LOCK_TIMEOUT 10L /* the default locking timeout */
sl@0
    73
sl@0
    74
/*****************************************************************************
sl@0
    75
** The next group of routines implement the I/O methods specified
sl@0
    76
** by the sqlite3_io_methods object.
sl@0
    77
******************************************************************************/
sl@0
    78
sl@0
    79
/*
sl@0
    80
** Close a file.
sl@0
    81
*/
sl@0
    82
static int os2Close( sqlite3_file *id ){
sl@0
    83
  APIRET rc = NO_ERROR;
sl@0
    84
  os2File *pFile;
sl@0
    85
  if( id && (pFile = (os2File*)id) != 0 ){
sl@0
    86
    OSTRACE2( "CLOSE %d\n", pFile->h );
sl@0
    87
    rc = DosClose( pFile->h );
sl@0
    88
    pFile->locktype = NO_LOCK;
sl@0
    89
    if( pFile->pathToDel != NULL ){
sl@0
    90
      rc = DosForceDelete( (PSZ)pFile->pathToDel );
sl@0
    91
      free( pFile->pathToDel );
sl@0
    92
      pFile->pathToDel = NULL;
sl@0
    93
    }
sl@0
    94
    id = 0;
sl@0
    95
    OpenCounter( -1 );
sl@0
    96
  }
sl@0
    97
sl@0
    98
  return rc == NO_ERROR ? SQLITE_OK : SQLITE_IOERR;
sl@0
    99
}
sl@0
   100
sl@0
   101
/*
sl@0
   102
** Read data from a file into a buffer.  Return SQLITE_OK if all
sl@0
   103
** bytes were read successfully and SQLITE_IOERR if anything goes
sl@0
   104
** wrong.
sl@0
   105
*/
sl@0
   106
static int os2Read(
sl@0
   107
  sqlite3_file *id,               /* File to read from */
sl@0
   108
  void *pBuf,                     /* Write content into this buffer */
sl@0
   109
  int amt,                        /* Number of bytes to read */
sl@0
   110
  sqlite3_int64 offset            /* Begin reading at this offset */
sl@0
   111
){
sl@0
   112
  ULONG fileLocation = 0L;
sl@0
   113
  ULONG got;
sl@0
   114
  os2File *pFile = (os2File*)id;
sl@0
   115
  assert( id!=0 );
sl@0
   116
  SimulateIOError( return SQLITE_IOERR_READ );
sl@0
   117
  OSTRACE3( "READ %d lock=%d\n", pFile->h, pFile->locktype );
sl@0
   118
  if( DosSetFilePtr(pFile->h, offset, FILE_BEGIN, &fileLocation) != NO_ERROR ){
sl@0
   119
    return SQLITE_IOERR;
sl@0
   120
  }
sl@0
   121
  if( DosRead( pFile->h, pBuf, amt, &got ) != NO_ERROR ){
sl@0
   122
    return SQLITE_IOERR_READ;
sl@0
   123
  }
sl@0
   124
  if( got == (ULONG)amt )
sl@0
   125
    return SQLITE_OK;
sl@0
   126
  else {
sl@0
   127
    memset(&((char*)pBuf)[got], 0, amt-got);
sl@0
   128
    return SQLITE_IOERR_SHORT_READ;
sl@0
   129
  }
sl@0
   130
}
sl@0
   131
sl@0
   132
/*
sl@0
   133
** Write data from a buffer into a file.  Return SQLITE_OK on success
sl@0
   134
** or some other error code on failure.
sl@0
   135
*/
sl@0
   136
static int os2Write(
sl@0
   137
  sqlite3_file *id,               /* File to write into */
sl@0
   138
  const void *pBuf,               /* The bytes to be written */
sl@0
   139
  int amt,                        /* Number of bytes to write */
sl@0
   140
  sqlite3_int64 offset            /* Offset into the file to begin writing at */
sl@0
   141
){
sl@0
   142
  ULONG fileLocation = 0L;
sl@0
   143
  APIRET rc = NO_ERROR;
sl@0
   144
  ULONG wrote;
sl@0
   145
  os2File *pFile = (os2File*)id;
sl@0
   146
  assert( id!=0 );
sl@0
   147
  SimulateIOError( return SQLITE_IOERR_WRITE );
sl@0
   148
  SimulateDiskfullError( return SQLITE_FULL );
sl@0
   149
  OSTRACE3( "WRITE %d lock=%d\n", pFile->h, pFile->locktype );
sl@0
   150
  if( DosSetFilePtr(pFile->h, offset, FILE_BEGIN, &fileLocation) != NO_ERROR ){
sl@0
   151
    return SQLITE_IOERR;
sl@0
   152
  }
sl@0
   153
  assert( amt>0 );
sl@0
   154
  while( amt > 0 &&
sl@0
   155
         ( rc = DosWrite( pFile->h, (PVOID)pBuf, amt, &wrote ) ) == NO_ERROR &&
sl@0
   156
         wrote > 0
sl@0
   157
  ){
sl@0
   158
    amt -= wrote;
sl@0
   159
    pBuf = &((char*)pBuf)[wrote];
sl@0
   160
  }
sl@0
   161
sl@0
   162
  return ( rc != NO_ERROR || amt > (int)wrote ) ? SQLITE_FULL : SQLITE_OK;
sl@0
   163
}
sl@0
   164
sl@0
   165
/*
sl@0
   166
** Truncate an open file to a specified size
sl@0
   167
*/
sl@0
   168
static int os2Truncate( sqlite3_file *id, i64 nByte ){
sl@0
   169
  APIRET rc = NO_ERROR;
sl@0
   170
  os2File *pFile = (os2File*)id;
sl@0
   171
  OSTRACE3( "TRUNCATE %d %lld\n", pFile->h, nByte );
sl@0
   172
  SimulateIOError( return SQLITE_IOERR_TRUNCATE );
sl@0
   173
  rc = DosSetFileSize( pFile->h, nByte );
sl@0
   174
  return rc == NO_ERROR ? SQLITE_OK : SQLITE_IOERR_TRUNCATE;
sl@0
   175
}
sl@0
   176
sl@0
   177
#ifdef SQLITE_TEST
sl@0
   178
/*
sl@0
   179
** Count the number of fullsyncs and normal syncs.  This is used to test
sl@0
   180
** that syncs and fullsyncs are occuring at the right times.
sl@0
   181
*/
sl@0
   182
int sqlite3_sync_count = 0;
sl@0
   183
int sqlite3_fullsync_count = 0;
sl@0
   184
#endif
sl@0
   185
sl@0
   186
/*
sl@0
   187
** Make sure all writes to a particular file are committed to disk.
sl@0
   188
*/
sl@0
   189
static int os2Sync( sqlite3_file *id, int flags ){
sl@0
   190
  os2File *pFile = (os2File*)id;
sl@0
   191
  OSTRACE3( "SYNC %d lock=%d\n", pFile->h, pFile->locktype );
sl@0
   192
#ifdef SQLITE_TEST
sl@0
   193
  if( flags & SQLITE_SYNC_FULL){
sl@0
   194
    sqlite3_fullsync_count++;
sl@0
   195
  }
sl@0
   196
  sqlite3_sync_count++;
sl@0
   197
#endif
sl@0
   198
  return DosResetBuffer( pFile->h ) == NO_ERROR ? SQLITE_OK : SQLITE_IOERR;
sl@0
   199
}
sl@0
   200
sl@0
   201
/*
sl@0
   202
** Determine the current size of a file in bytes
sl@0
   203
*/
sl@0
   204
static int os2FileSize( sqlite3_file *id, sqlite3_int64 *pSize ){
sl@0
   205
  APIRET rc = NO_ERROR;
sl@0
   206
  FILESTATUS3 fsts3FileInfo;
sl@0
   207
  memset(&fsts3FileInfo, 0, sizeof(fsts3FileInfo));
sl@0
   208
  assert( id!=0 );
sl@0
   209
  SimulateIOError( return SQLITE_IOERR_FSTAT );
sl@0
   210
  rc = DosQueryFileInfo( ((os2File*)id)->h, FIL_STANDARD, &fsts3FileInfo, sizeof(FILESTATUS3) );
sl@0
   211
  if( rc == NO_ERROR ){
sl@0
   212
    *pSize = fsts3FileInfo.cbFile;
sl@0
   213
    return SQLITE_OK;
sl@0
   214
  }else{
sl@0
   215
    return SQLITE_IOERR_FSTAT;
sl@0
   216
  }
sl@0
   217
}
sl@0
   218
sl@0
   219
/*
sl@0
   220
** Acquire a reader lock.
sl@0
   221
*/
sl@0
   222
static int getReadLock( os2File *pFile ){
sl@0
   223
  FILELOCK  LockArea,
sl@0
   224
            UnlockArea;
sl@0
   225
  APIRET res;
sl@0
   226
  memset(&LockArea, 0, sizeof(LockArea));
sl@0
   227
  memset(&UnlockArea, 0, sizeof(UnlockArea));
sl@0
   228
  LockArea.lOffset = SHARED_FIRST;
sl@0
   229
  LockArea.lRange = SHARED_SIZE;
sl@0
   230
  UnlockArea.lOffset = 0L;
sl@0
   231
  UnlockArea.lRange = 0L;
sl@0
   232
  res = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 1L );
sl@0
   233
  OSTRACE3( "GETREADLOCK %d res=%d\n", pFile->h, res );
sl@0
   234
  return res;
sl@0
   235
}
sl@0
   236
sl@0
   237
/*
sl@0
   238
** Undo a readlock
sl@0
   239
*/
sl@0
   240
static int unlockReadLock( os2File *id ){
sl@0
   241
  FILELOCK  LockArea,
sl@0
   242
            UnlockArea;
sl@0
   243
  APIRET res;
sl@0
   244
  memset(&LockArea, 0, sizeof(LockArea));
sl@0
   245
  memset(&UnlockArea, 0, sizeof(UnlockArea));
sl@0
   246
  LockArea.lOffset = 0L;
sl@0
   247
  LockArea.lRange = 0L;
sl@0
   248
  UnlockArea.lOffset = SHARED_FIRST;
sl@0
   249
  UnlockArea.lRange = SHARED_SIZE;
sl@0
   250
  res = DosSetFileLocks( id->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 1L );
sl@0
   251
  OSTRACE3( "UNLOCK-READLOCK file handle=%d res=%d?\n", id->h, res );
sl@0
   252
  return res;
sl@0
   253
}
sl@0
   254
sl@0
   255
/*
sl@0
   256
** Lock the file with the lock specified by parameter locktype - one
sl@0
   257
** of the following:
sl@0
   258
**
sl@0
   259
**     (1) SHARED_LOCK
sl@0
   260
**     (2) RESERVED_LOCK
sl@0
   261
**     (3) PENDING_LOCK
sl@0
   262
**     (4) EXCLUSIVE_LOCK
sl@0
   263
**
sl@0
   264
** Sometimes when requesting one lock state, additional lock states
sl@0
   265
** are inserted in between.  The locking might fail on one of the later
sl@0
   266
** transitions leaving the lock state different from what it started but
sl@0
   267
** still short of its goal.  The following chart shows the allowed
sl@0
   268
** transitions and the inserted intermediate states:
sl@0
   269
**
sl@0
   270
**    UNLOCKED -> SHARED
sl@0
   271
**    SHARED -> RESERVED
sl@0
   272
**    SHARED -> (PENDING) -> EXCLUSIVE
sl@0
   273
**    RESERVED -> (PENDING) -> EXCLUSIVE
sl@0
   274
**    PENDING -> EXCLUSIVE
sl@0
   275
**
sl@0
   276
** This routine will only increase a lock.  The os2Unlock() routine
sl@0
   277
** erases all locks at once and returns us immediately to locking level 0.
sl@0
   278
** It is not possible to lower the locking level one step at a time.  You
sl@0
   279
** must go straight to locking level 0.
sl@0
   280
*/
sl@0
   281
static int os2Lock( sqlite3_file *id, int locktype ){
sl@0
   282
  int rc = SQLITE_OK;       /* Return code from subroutines */
sl@0
   283
  APIRET res = NO_ERROR;    /* Result of an OS/2 lock call */
sl@0
   284
  int newLocktype;       /* Set pFile->locktype to this value before exiting */
sl@0
   285
  int gotPendingLock = 0;/* True if we acquired a PENDING lock this time */
sl@0
   286
  FILELOCK  LockArea,
sl@0
   287
            UnlockArea;
sl@0
   288
  os2File *pFile = (os2File*)id;
sl@0
   289
  memset(&LockArea, 0, sizeof(LockArea));
sl@0
   290
  memset(&UnlockArea, 0, sizeof(UnlockArea));
sl@0
   291
  assert( pFile!=0 );
sl@0
   292
  OSTRACE4( "LOCK %d %d was %d\n", pFile->h, locktype, pFile->locktype );
sl@0
   293
sl@0
   294
  /* If there is already a lock of this type or more restrictive on the
sl@0
   295
  ** os2File, do nothing. Don't use the end_lock: exit path, as
sl@0
   296
  ** sqlite3_mutex_enter() hasn't been called yet.
sl@0
   297
  */
sl@0
   298
  if( pFile->locktype>=locktype ){
sl@0
   299
    OSTRACE3( "LOCK %d %d ok (already held)\n", pFile->h, locktype );
sl@0
   300
    return SQLITE_OK;
sl@0
   301
  }
sl@0
   302
sl@0
   303
  /* Make sure the locking sequence is correct
sl@0
   304
  */
sl@0
   305
  assert( pFile->locktype!=NO_LOCK || locktype==SHARED_LOCK );
sl@0
   306
  assert( locktype!=PENDING_LOCK );
sl@0
   307
  assert( locktype!=RESERVED_LOCK || pFile->locktype==SHARED_LOCK );
sl@0
   308
sl@0
   309
  /* Lock the PENDING_LOCK byte if we need to acquire a PENDING lock or
sl@0
   310
  ** a SHARED lock.  If we are acquiring a SHARED lock, the acquisition of
sl@0
   311
  ** the PENDING_LOCK byte is temporary.
sl@0
   312
  */
sl@0
   313
  newLocktype = pFile->locktype;
sl@0
   314
  if( pFile->locktype==NO_LOCK
sl@0
   315
      || (locktype==EXCLUSIVE_LOCK && pFile->locktype==RESERVED_LOCK)
sl@0
   316
  ){
sl@0
   317
    LockArea.lOffset = PENDING_BYTE;
sl@0
   318
    LockArea.lRange = 1L;
sl@0
   319
    UnlockArea.lOffset = 0L;
sl@0
   320
    UnlockArea.lRange = 0L;
sl@0
   321
sl@0
   322
    /* wait longer than LOCK_TIMEOUT here not to have to try multiple times */
sl@0
   323
    res = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, 100L, 0L );
sl@0
   324
    if( res == NO_ERROR ){
sl@0
   325
      gotPendingLock = 1;
sl@0
   326
      OSTRACE3( "LOCK %d pending lock boolean set.  res=%d\n", pFile->h, res );
sl@0
   327
    }
sl@0
   328
  }
sl@0
   329
sl@0
   330
  /* Acquire a shared lock
sl@0
   331
  */
sl@0
   332
  if( locktype==SHARED_LOCK && res == NO_ERROR ){
sl@0
   333
    assert( pFile->locktype==NO_LOCK );
sl@0
   334
    res = getReadLock(pFile);
sl@0
   335
    if( res == NO_ERROR ){
sl@0
   336
      newLocktype = SHARED_LOCK;
sl@0
   337
    }
sl@0
   338
    OSTRACE3( "LOCK %d acquire shared lock. res=%d\n", pFile->h, res );
sl@0
   339
  }
sl@0
   340
sl@0
   341
  /* Acquire a RESERVED lock
sl@0
   342
  */
sl@0
   343
  if( locktype==RESERVED_LOCK && res == NO_ERROR ){
sl@0
   344
    assert( pFile->locktype==SHARED_LOCK );
sl@0
   345
    LockArea.lOffset = RESERVED_BYTE;
sl@0
   346
    LockArea.lRange = 1L;
sl@0
   347
    UnlockArea.lOffset = 0L;
sl@0
   348
    UnlockArea.lRange = 0L;
sl@0
   349
    res = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 0L );
sl@0
   350
    if( res == NO_ERROR ){
sl@0
   351
      newLocktype = RESERVED_LOCK;
sl@0
   352
    }
sl@0
   353
    OSTRACE3( "LOCK %d acquire reserved lock. res=%d\n", pFile->h, res );
sl@0
   354
  }
sl@0
   355
sl@0
   356
  /* Acquire a PENDING lock
sl@0
   357
  */
sl@0
   358
  if( locktype==EXCLUSIVE_LOCK && res == NO_ERROR ){
sl@0
   359
    newLocktype = PENDING_LOCK;
sl@0
   360
    gotPendingLock = 0;
sl@0
   361
    OSTRACE2( "LOCK %d acquire pending lock. pending lock boolean unset.\n", pFile->h );
sl@0
   362
  }
sl@0
   363
sl@0
   364
  /* Acquire an EXCLUSIVE lock
sl@0
   365
  */
sl@0
   366
  if( locktype==EXCLUSIVE_LOCK && res == NO_ERROR ){
sl@0
   367
    assert( pFile->locktype>=SHARED_LOCK );
sl@0
   368
    res = unlockReadLock(pFile);
sl@0
   369
    OSTRACE2( "unreadlock = %d\n", res );
sl@0
   370
    LockArea.lOffset = SHARED_FIRST;
sl@0
   371
    LockArea.lRange = SHARED_SIZE;
sl@0
   372
    UnlockArea.lOffset = 0L;
sl@0
   373
    UnlockArea.lRange = 0L;
sl@0
   374
    res = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 0L );
sl@0
   375
    if( res == NO_ERROR ){
sl@0
   376
      newLocktype = EXCLUSIVE_LOCK;
sl@0
   377
    }else{
sl@0
   378
      OSTRACE2( "OS/2 error-code = %d\n", res );
sl@0
   379
      getReadLock(pFile);
sl@0
   380
    }
sl@0
   381
    OSTRACE3( "LOCK %d acquire exclusive lock.  res=%d\n", pFile->h, res );
sl@0
   382
  }
sl@0
   383
sl@0
   384
  /* If we are holding a PENDING lock that ought to be released, then
sl@0
   385
  ** release it now.
sl@0
   386
  */
sl@0
   387
  if( gotPendingLock && locktype==SHARED_LOCK ){
sl@0
   388
    int r;
sl@0
   389
    LockArea.lOffset = 0L;
sl@0
   390
    LockArea.lRange = 0L;
sl@0
   391
    UnlockArea.lOffset = PENDING_BYTE;
sl@0
   392
    UnlockArea.lRange = 1L;
sl@0
   393
    r = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 0L );
sl@0
   394
    OSTRACE3( "LOCK %d unlocking pending/is shared. r=%d\n", pFile->h, r );
sl@0
   395
  }
sl@0
   396
sl@0
   397
  /* Update the state of the lock has held in the file descriptor then
sl@0
   398
  ** return the appropriate result code.
sl@0
   399
  */
sl@0
   400
  if( res == NO_ERROR ){
sl@0
   401
    rc = SQLITE_OK;
sl@0
   402
  }else{
sl@0
   403
    OSTRACE4( "LOCK FAILED %d trying for %d but got %d\n", pFile->h,
sl@0
   404
              locktype, newLocktype );
sl@0
   405
    rc = SQLITE_BUSY;
sl@0
   406
  }
sl@0
   407
  pFile->locktype = newLocktype;
sl@0
   408
  OSTRACE3( "LOCK %d now %d\n", pFile->h, pFile->locktype );
sl@0
   409
  return rc;
sl@0
   410
}
sl@0
   411
sl@0
   412
/*
sl@0
   413
** This routine checks if there is a RESERVED lock held on the specified
sl@0
   414
** file by this or any other process. If such a lock is held, return
sl@0
   415
** non-zero, otherwise zero.
sl@0
   416
*/
sl@0
   417
static int os2CheckReservedLock( sqlite3_file *id, int *pOut ){
sl@0
   418
  int r = 0;
sl@0
   419
  os2File *pFile = (os2File*)id;
sl@0
   420
  assert( pFile!=0 );
sl@0
   421
  if( pFile->locktype>=RESERVED_LOCK ){
sl@0
   422
    r = 1;
sl@0
   423
    OSTRACE3( "TEST WR-LOCK %d %d (local)\n", pFile->h, r );
sl@0
   424
  }else{
sl@0
   425
    FILELOCK  LockArea,
sl@0
   426
              UnlockArea;
sl@0
   427
    APIRET rc = NO_ERROR;
sl@0
   428
    memset(&LockArea, 0, sizeof(LockArea));
sl@0
   429
    memset(&UnlockArea, 0, sizeof(UnlockArea));
sl@0
   430
    LockArea.lOffset = RESERVED_BYTE;
sl@0
   431
    LockArea.lRange = 1L;
sl@0
   432
    UnlockArea.lOffset = 0L;
sl@0
   433
    UnlockArea.lRange = 0L;
sl@0
   434
    rc = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 0L );
sl@0
   435
    OSTRACE3( "TEST WR-LOCK %d lock reserved byte rc=%d\n", pFile->h, rc );
sl@0
   436
    if( rc == NO_ERROR ){
sl@0
   437
      APIRET rcu = NO_ERROR; /* return code for unlocking */
sl@0
   438
      LockArea.lOffset = 0L;
sl@0
   439
      LockArea.lRange = 0L;
sl@0
   440
      UnlockArea.lOffset = RESERVED_BYTE;
sl@0
   441
      UnlockArea.lRange = 1L;
sl@0
   442
      rcu = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 0L );
sl@0
   443
      OSTRACE3( "TEST WR-LOCK %d unlock reserved byte r=%d\n", pFile->h, rcu );
sl@0
   444
    }
sl@0
   445
    r = !(rc == NO_ERROR);
sl@0
   446
    OSTRACE3( "TEST WR-LOCK %d %d (remote)\n", pFile->h, r );
sl@0
   447
  }
sl@0
   448
  *pOut = r;
sl@0
   449
  return SQLITE_OK;
sl@0
   450
}
sl@0
   451
sl@0
   452
/*
sl@0
   453
** Lower the locking level on file descriptor id to locktype.  locktype
sl@0
   454
** must be either NO_LOCK or SHARED_LOCK.
sl@0
   455
**
sl@0
   456
** If the locking level of the file descriptor is already at or below
sl@0
   457
** the requested locking level, this routine is a no-op.
sl@0
   458
**
sl@0
   459
** It is not possible for this routine to fail if the second argument
sl@0
   460
** is NO_LOCK.  If the second argument is SHARED_LOCK then this routine
sl@0
   461
** might return SQLITE_IOERR;
sl@0
   462
*/
sl@0
   463
static int os2Unlock( sqlite3_file *id, int locktype ){
sl@0
   464
  int type;
sl@0
   465
  os2File *pFile = (os2File*)id;
sl@0
   466
  APIRET rc = SQLITE_OK;
sl@0
   467
  APIRET res = NO_ERROR;
sl@0
   468
  FILELOCK  LockArea,
sl@0
   469
            UnlockArea;
sl@0
   470
  memset(&LockArea, 0, sizeof(LockArea));
sl@0
   471
  memset(&UnlockArea, 0, sizeof(UnlockArea));
sl@0
   472
  assert( pFile!=0 );
sl@0
   473
  assert( locktype<=SHARED_LOCK );
sl@0
   474
  OSTRACE4( "UNLOCK %d to %d was %d\n", pFile->h, locktype, pFile->locktype );
sl@0
   475
  type = pFile->locktype;
sl@0
   476
  if( type>=EXCLUSIVE_LOCK ){
sl@0
   477
    LockArea.lOffset = 0L;
sl@0
   478
    LockArea.lRange = 0L;
sl@0
   479
    UnlockArea.lOffset = SHARED_FIRST;
sl@0
   480
    UnlockArea.lRange = SHARED_SIZE;
sl@0
   481
    res = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 0L );
sl@0
   482
    OSTRACE3( "UNLOCK %d exclusive lock res=%d\n", pFile->h, res );
sl@0
   483
    if( locktype==SHARED_LOCK && getReadLock(pFile) != NO_ERROR ){
sl@0
   484
      /* This should never happen.  We should always be able to
sl@0
   485
      ** reacquire the read lock */
sl@0
   486
      OSTRACE3( "UNLOCK %d to %d getReadLock() failed\n", pFile->h, locktype );
sl@0
   487
      rc = SQLITE_IOERR_UNLOCK;
sl@0
   488
    }
sl@0
   489
  }
sl@0
   490
  if( type>=RESERVED_LOCK ){
sl@0
   491
    LockArea.lOffset = 0L;
sl@0
   492
    LockArea.lRange = 0L;
sl@0
   493
    UnlockArea.lOffset = RESERVED_BYTE;
sl@0
   494
    UnlockArea.lRange = 1L;
sl@0
   495
    res = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 0L );
sl@0
   496
    OSTRACE3( "UNLOCK %d reserved res=%d\n", pFile->h, res );
sl@0
   497
  }
sl@0
   498
  if( locktype==NO_LOCK && type>=SHARED_LOCK ){
sl@0
   499
    res = unlockReadLock(pFile);
sl@0
   500
    OSTRACE5( "UNLOCK %d is %d want %d res=%d\n", pFile->h, type, locktype, res );
sl@0
   501
  }
sl@0
   502
  if( type>=PENDING_LOCK ){
sl@0
   503
    LockArea.lOffset = 0L;
sl@0
   504
    LockArea.lRange = 0L;
sl@0
   505
    UnlockArea.lOffset = PENDING_BYTE;
sl@0
   506
    UnlockArea.lRange = 1L;
sl@0
   507
    res = DosSetFileLocks( pFile->h, &UnlockArea, &LockArea, LOCK_TIMEOUT, 0L );
sl@0
   508
    OSTRACE3( "UNLOCK %d pending res=%d\n", pFile->h, res );
sl@0
   509
  }
sl@0
   510
  pFile->locktype = locktype;
sl@0
   511
  OSTRACE3( "UNLOCK %d now %d\n", pFile->h, pFile->locktype );
sl@0
   512
  return rc;
sl@0
   513
}
sl@0
   514
sl@0
   515
/*
sl@0
   516
** Control and query of the open file handle.
sl@0
   517
*/
sl@0
   518
static int os2FileControl(sqlite3_file *id, int op, void *pArg){
sl@0
   519
  switch( op ){
sl@0
   520
    case SQLITE_FCNTL_LOCKSTATE: {
sl@0
   521
      *(int*)pArg = ((os2File*)id)->locktype;
sl@0
   522
      OSTRACE3( "FCNTL_LOCKSTATE %d lock=%d\n", ((os2File*)id)->h, ((os2File*)id)->locktype );
sl@0
   523
      return SQLITE_OK;
sl@0
   524
    }
sl@0
   525
  }
sl@0
   526
  return SQLITE_ERROR;
sl@0
   527
}
sl@0
   528
sl@0
   529
/*
sl@0
   530
** Return the sector size in bytes of the underlying block device for
sl@0
   531
** the specified file. This is almost always 512 bytes, but may be
sl@0
   532
** larger for some devices.
sl@0
   533
**
sl@0
   534
** SQLite code assumes this function cannot fail. It also assumes that
sl@0
   535
** if two files are created in the same file-system directory (i.e.
sl@0
   536
** a database and its journal file) that the sector size will be the
sl@0
   537
** same for both.
sl@0
   538
*/
sl@0
   539
static int os2SectorSize(sqlite3_file *id){
sl@0
   540
  return SQLITE_DEFAULT_SECTOR_SIZE;
sl@0
   541
}
sl@0
   542
sl@0
   543
/*
sl@0
   544
** Return a vector of device characteristics.
sl@0
   545
*/
sl@0
   546
static int os2DeviceCharacteristics(sqlite3_file *id){
sl@0
   547
  return 0;
sl@0
   548
}
sl@0
   549
sl@0
   550
sl@0
   551
/*
sl@0
   552
** Character set conversion objects used by conversion routines.
sl@0
   553
*/
sl@0
   554
static UconvObject ucUtf8 = NULL; /* convert between UTF-8 and UCS-2 */
sl@0
   555
static UconvObject uclCp = NULL;  /* convert between local codepage and UCS-2 */
sl@0
   556
sl@0
   557
/*
sl@0
   558
** Helper function to initialize the conversion objects from and to UTF-8.
sl@0
   559
*/
sl@0
   560
static void initUconvObjects( void ){
sl@0
   561
  if( UniCreateUconvObject( UTF_8, &ucUtf8 ) != ULS_SUCCESS )
sl@0
   562
    ucUtf8 = NULL;
sl@0
   563
  if ( UniCreateUconvObject( (UniChar *)L"@path=yes", &uclCp ) != ULS_SUCCESS )
sl@0
   564
    uclCp = NULL;
sl@0
   565
}
sl@0
   566
sl@0
   567
/*
sl@0
   568
** Helper function to free the conversion objects from and to UTF-8.
sl@0
   569
*/
sl@0
   570
static void freeUconvObjects( void ){
sl@0
   571
  if ( ucUtf8 )
sl@0
   572
    UniFreeUconvObject( ucUtf8 );
sl@0
   573
  if ( uclCp )
sl@0
   574
    UniFreeUconvObject( uclCp );
sl@0
   575
  ucUtf8 = NULL;
sl@0
   576
  uclCp = NULL;
sl@0
   577
}
sl@0
   578
sl@0
   579
/*
sl@0
   580
** Helper function to convert UTF-8 filenames to local OS/2 codepage.
sl@0
   581
** The two-step process: first convert the incoming UTF-8 string
sl@0
   582
** into UCS-2 and then from UCS-2 to the current codepage.
sl@0
   583
** The returned char pointer has to be freed.
sl@0
   584
*/
sl@0
   585
static char *convertUtf8PathToCp( const char *in ){
sl@0
   586
  UniChar tempPath[CCHMAXPATH];
sl@0
   587
  char *out = (char *)calloc( CCHMAXPATH, 1 );
sl@0
   588
sl@0
   589
  if( !out )
sl@0
   590
    return NULL;
sl@0
   591
sl@0
   592
  if( !ucUtf8 || !uclCp )
sl@0
   593
    initUconvObjects();
sl@0
   594
sl@0
   595
  /* determine string for the conversion of UTF-8 which is CP1208 */
sl@0
   596
  if( UniStrToUcs( ucUtf8, tempPath, (char *)in, CCHMAXPATH ) != ULS_SUCCESS )
sl@0
   597
    return out; /* if conversion fails, return the empty string */
sl@0
   598
sl@0
   599
  /* conversion for current codepage which can be used for paths */
sl@0
   600
  UniStrFromUcs( uclCp, out, tempPath, CCHMAXPATH );
sl@0
   601
sl@0
   602
  return out;
sl@0
   603
}
sl@0
   604
sl@0
   605
/*
sl@0
   606
** Helper function to convert filenames from local codepage to UTF-8.
sl@0
   607
** The two-step process: first convert the incoming codepage-specific
sl@0
   608
** string into UCS-2 and then from UCS-2 to the codepage of UTF-8.
sl@0
   609
** The returned char pointer has to be freed.
sl@0
   610
**
sl@0
   611
** This function is non-static to be able to use this in shell.c and
sl@0
   612
** similar applications that take command line arguments.
sl@0
   613
*/
sl@0
   614
char *convertCpPathToUtf8( const char *in ){
sl@0
   615
  UniChar tempPath[CCHMAXPATH];
sl@0
   616
  char *out = (char *)calloc( CCHMAXPATH, 1 );
sl@0
   617
sl@0
   618
  if( !out )
sl@0
   619
    return NULL;
sl@0
   620
sl@0
   621
  if( !ucUtf8 || !uclCp )
sl@0
   622
    initUconvObjects();
sl@0
   623
sl@0
   624
  /* conversion for current codepage which can be used for paths */
sl@0
   625
  if( UniStrToUcs( uclCp, tempPath, (char *)in, CCHMAXPATH ) != ULS_SUCCESS )
sl@0
   626
    return out; /* if conversion fails, return the empty string */
sl@0
   627
sl@0
   628
  /* determine string for the conversion of UTF-8 which is CP1208 */
sl@0
   629
  UniStrFromUcs( ucUtf8, out, tempPath, CCHMAXPATH );
sl@0
   630
sl@0
   631
  return out;
sl@0
   632
}
sl@0
   633
sl@0
   634
/*
sl@0
   635
** This vector defines all the methods that can operate on an
sl@0
   636
** sqlite3_file for os2.
sl@0
   637
*/
sl@0
   638
static const sqlite3_io_methods os2IoMethod = {
sl@0
   639
  1,                        /* iVersion */
sl@0
   640
  os2Close,
sl@0
   641
  os2Read,
sl@0
   642
  os2Write,
sl@0
   643
  os2Truncate,
sl@0
   644
  os2Sync,
sl@0
   645
  os2FileSize,
sl@0
   646
  os2Lock,
sl@0
   647
  os2Unlock,
sl@0
   648
  os2CheckReservedLock,
sl@0
   649
  os2FileControl,
sl@0
   650
  os2SectorSize,
sl@0
   651
  os2DeviceCharacteristics
sl@0
   652
};
sl@0
   653
sl@0
   654
/***************************************************************************
sl@0
   655
** Here ends the I/O methods that form the sqlite3_io_methods object.
sl@0
   656
**
sl@0
   657
** The next block of code implements the VFS methods.
sl@0
   658
****************************************************************************/
sl@0
   659
sl@0
   660
/*
sl@0
   661
** Create a temporary file name in zBuf.  zBuf must be big enough to
sl@0
   662
** hold at pVfs->mxPathname characters.
sl@0
   663
*/
sl@0
   664
static int getTempname(int nBuf, char *zBuf ){
sl@0
   665
  static const unsigned char zChars[] =
sl@0
   666
    "abcdefghijklmnopqrstuvwxyz"
sl@0
   667
    "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
sl@0
   668
    "0123456789";
sl@0
   669
  int i, j;
sl@0
   670
  char zTempPathBuf[3];
sl@0
   671
  PSZ zTempPath = (PSZ)&zTempPathBuf;
sl@0
   672
  if( sqlite3_temp_directory ){
sl@0
   673
    zTempPath = sqlite3_temp_directory;
sl@0
   674
  }else{
sl@0
   675
    if( DosScanEnv( (PSZ)"TEMP", &zTempPath ) ){
sl@0
   676
      if( DosScanEnv( (PSZ)"TMP", &zTempPath ) ){
sl@0
   677
        if( DosScanEnv( (PSZ)"TMPDIR", &zTempPath ) ){
sl@0
   678
           ULONG ulDriveNum = 0, ulDriveMap = 0;
sl@0
   679
           DosQueryCurrentDisk( &ulDriveNum, &ulDriveMap );
sl@0
   680
           sprintf( (char*)zTempPath, "%c:", (char)( 'A' + ulDriveNum - 1 ) );
sl@0
   681
        }
sl@0
   682
      }
sl@0
   683
    }
sl@0
   684
  }
sl@0
   685
  /* Strip off a trailing slashes or backslashes, otherwise we would get *
sl@0
   686
   * multiple (back)slashes which causes DosOpen() to fail.              *
sl@0
   687
   * Trailing spaces are not allowed, either.                            */
sl@0
   688
  j = strlen(zTempPath);
sl@0
   689
  while( j > 0 && ( zTempPath[j-1] == '\\' || zTempPath[j-1] == '/'
sl@0
   690
                    || zTempPath[j-1] == ' ' ) ){
sl@0
   691
    j--;
sl@0
   692
  }
sl@0
   693
  zTempPath[j] = '\0';
sl@0
   694
  if( !sqlite3_temp_directory ){
sl@0
   695
    char *zTempPathUTF = convertCpPathToUtf8( zTempPath );
sl@0
   696
    sqlite3_snprintf( nBuf-30, zBuf,
sl@0
   697
                      "%s\\"SQLITE_TEMP_FILE_PREFIX, zTempPathUTF );
sl@0
   698
    free( zTempPathUTF );
sl@0
   699
  }else{
sl@0
   700
    sqlite3_snprintf( nBuf-30, zBuf,
sl@0
   701
                      "%s\\"SQLITE_TEMP_FILE_PREFIX, zTempPath );
sl@0
   702
  }
sl@0
   703
  j = strlen( zBuf );
sl@0
   704
  sqlite3_randomness( 20, &zBuf[j] );
sl@0
   705
  for( i = 0; i < 20; i++, j++ ){
sl@0
   706
    zBuf[j] = (char)zChars[ ((unsigned char)zBuf[j])%(sizeof(zChars)-1) ];
sl@0
   707
  }
sl@0
   708
  zBuf[j] = 0;
sl@0
   709
  OSTRACE2( "TEMP FILENAME: %s\n", zBuf );
sl@0
   710
  return SQLITE_OK;
sl@0
   711
}
sl@0
   712
sl@0
   713
sl@0
   714
/*
sl@0
   715
** Turn a relative pathname into a full pathname.  Write the full
sl@0
   716
** pathname into zFull[].  zFull[] will be at least pVfs->mxPathname
sl@0
   717
** bytes in size.
sl@0
   718
*/
sl@0
   719
static int os2FullPathname(
sl@0
   720
  sqlite3_vfs *pVfs,          /* Pointer to vfs object */
sl@0
   721
  const char *zRelative,      /* Possibly relative input path */
sl@0
   722
  int nFull,                  /* Size of output buffer in bytes */
sl@0
   723
  char *zFull                 /* Output buffer */
sl@0
   724
){
sl@0
   725
  char *zRelativeCp = convertUtf8PathToCp( zRelative );
sl@0
   726
  char zFullCp[CCHMAXPATH] = "\0";
sl@0
   727
  char *zFullUTF;
sl@0
   728
  APIRET rc = DosQueryPathInfo( zRelativeCp, FIL_QUERYFULLNAME, zFullCp,
sl@0
   729
                                CCHMAXPATH );
sl@0
   730
  free( zRelativeCp );
sl@0
   731
  zFullUTF = convertCpPathToUtf8( zFullCp );
sl@0
   732
  sqlite3_snprintf( nFull, zFull, zFullUTF );
sl@0
   733
  free( zFullUTF );
sl@0
   734
  return rc == NO_ERROR ? SQLITE_OK : SQLITE_IOERR;
sl@0
   735
}
sl@0
   736
sl@0
   737
sl@0
   738
/*
sl@0
   739
** Open a file.
sl@0
   740
*/
sl@0
   741
static int os2Open(
sl@0
   742
  sqlite3_vfs *pVfs,            /* Not used */
sl@0
   743
  const char *zName,            /* Name of the file */
sl@0
   744
  sqlite3_file *id,             /* Write the SQLite file handle here */
sl@0
   745
  int flags,                    /* Open mode flags */
sl@0
   746
  int *pOutFlags                /* Status return flags */
sl@0
   747
){
sl@0
   748
  HFILE h;
sl@0
   749
  ULONG ulFileAttribute = FILE_NORMAL;
sl@0
   750
  ULONG ulOpenFlags = 0;
sl@0
   751
  ULONG ulOpenMode = 0;
sl@0
   752
  os2File *pFile = (os2File*)id;
sl@0
   753
  APIRET rc = NO_ERROR;
sl@0
   754
  ULONG ulAction;
sl@0
   755
  char *zNameCp;
sl@0
   756
  char zTmpname[CCHMAXPATH+1];    /* Buffer to hold name of temp file */
sl@0
   757
sl@0
   758
  /* If the second argument to this function is NULL, generate a 
sl@0
   759
  ** temporary file name to use 
sl@0
   760
  */
sl@0
   761
  if( !zName ){
sl@0
   762
    int rc = getTempname(CCHMAXPATH+1, zTmpname);
sl@0
   763
    if( rc!=SQLITE_OK ){
sl@0
   764
      return rc;
sl@0
   765
    }
sl@0
   766
    zName = zTmpname;
sl@0
   767
  }
sl@0
   768
sl@0
   769
sl@0
   770
  memset( pFile, 0, sizeof(*pFile) );
sl@0
   771
sl@0
   772
  OSTRACE2( "OPEN want %d\n", flags );
sl@0
   773
sl@0
   774
  if( flags & SQLITE_OPEN_READWRITE ){
sl@0
   775
    ulOpenMode |= OPEN_ACCESS_READWRITE;
sl@0
   776
    OSTRACE1( "OPEN read/write\n" );
sl@0
   777
  }else{
sl@0
   778
    ulOpenMode |= OPEN_ACCESS_READONLY;
sl@0
   779
    OSTRACE1( "OPEN read only\n" );
sl@0
   780
  }
sl@0
   781
sl@0
   782
  if( flags & SQLITE_OPEN_CREATE ){
sl@0
   783
    ulOpenFlags |= OPEN_ACTION_OPEN_IF_EXISTS | OPEN_ACTION_CREATE_IF_NEW;
sl@0
   784
    OSTRACE1( "OPEN open new/create\n" );
sl@0
   785
  }else{
sl@0
   786
    ulOpenFlags |= OPEN_ACTION_OPEN_IF_EXISTS | OPEN_ACTION_FAIL_IF_NEW;
sl@0
   787
    OSTRACE1( "OPEN open existing\n" );
sl@0
   788
  }
sl@0
   789
sl@0
   790
  if( flags & SQLITE_OPEN_MAIN_DB ){
sl@0
   791
    ulOpenMode |= OPEN_SHARE_DENYNONE;
sl@0
   792
    OSTRACE1( "OPEN share read/write\n" );
sl@0
   793
  }else{
sl@0
   794
    ulOpenMode |= OPEN_SHARE_DENYWRITE;
sl@0
   795
    OSTRACE1( "OPEN share read only\n" );
sl@0
   796
  }
sl@0
   797
sl@0
   798
  if( flags & SQLITE_OPEN_DELETEONCLOSE ){
sl@0
   799
    char pathUtf8[CCHMAXPATH];
sl@0
   800
#ifdef NDEBUG /* when debugging we want to make sure it is deleted */
sl@0
   801
    ulFileAttribute = FILE_HIDDEN;
sl@0
   802
#endif
sl@0
   803
    os2FullPathname( pVfs, zName, CCHMAXPATH, pathUtf8 );
sl@0
   804
    pFile->pathToDel = convertUtf8PathToCp( pathUtf8 );
sl@0
   805
    OSTRACE1( "OPEN hidden/delete on close file attributes\n" );
sl@0
   806
  }else{
sl@0
   807
    pFile->pathToDel = NULL;
sl@0
   808
    OSTRACE1( "OPEN normal file attribute\n" );
sl@0
   809
  }
sl@0
   810
sl@0
   811
  /* always open in random access mode for possibly better speed */
sl@0
   812
  ulOpenMode |= OPEN_FLAGS_RANDOM;
sl@0
   813
  ulOpenMode |= OPEN_FLAGS_FAIL_ON_ERROR;
sl@0
   814
  ulOpenMode |= OPEN_FLAGS_NOINHERIT;
sl@0
   815
sl@0
   816
  zNameCp = convertUtf8PathToCp( zName );
sl@0
   817
  rc = DosOpen( (PSZ)zNameCp,
sl@0
   818
                &h,
sl@0
   819
                &ulAction,
sl@0
   820
                0L,
sl@0
   821
                ulFileAttribute,
sl@0
   822
                ulOpenFlags,
sl@0
   823
                ulOpenMode,
sl@0
   824
                (PEAOP2)NULL );
sl@0
   825
  free( zNameCp );
sl@0
   826
  if( rc != NO_ERROR ){
sl@0
   827
    OSTRACE7( "OPEN Invalid handle rc=%d: zName=%s, ulAction=%#lx, ulAttr=%#lx, ulFlags=%#lx, ulMode=%#lx\n",
sl@0
   828
              rc, zName, ulAction, ulFileAttribute, ulOpenFlags, ulOpenMode );
sl@0
   829
    if( pFile->pathToDel )
sl@0
   830
      free( pFile->pathToDel );
sl@0
   831
    pFile->pathToDel = NULL;
sl@0
   832
    if( flags & SQLITE_OPEN_READWRITE ){
sl@0
   833
      OSTRACE2( "OPEN %d Invalid handle\n", ((flags | SQLITE_OPEN_READONLY) & ~SQLITE_OPEN_READWRITE) );
sl@0
   834
      return os2Open( pVfs, zName, id,
sl@0
   835
                      ((flags | SQLITE_OPEN_READONLY) & ~SQLITE_OPEN_READWRITE),
sl@0
   836
                      pOutFlags );
sl@0
   837
    }else{
sl@0
   838
      return SQLITE_CANTOPEN;
sl@0
   839
    }
sl@0
   840
  }
sl@0
   841
sl@0
   842
  if( pOutFlags ){
sl@0
   843
    *pOutFlags = flags & SQLITE_OPEN_READWRITE ? SQLITE_OPEN_READWRITE : SQLITE_OPEN_READONLY;
sl@0
   844
  }
sl@0
   845
sl@0
   846
  pFile->pMethod = &os2IoMethod;
sl@0
   847
  pFile->h = h;
sl@0
   848
  OpenCounter(+1);
sl@0
   849
  OSTRACE3( "OPEN %d pOutFlags=%d\n", pFile->h, pOutFlags );
sl@0
   850
  return SQLITE_OK;
sl@0
   851
}
sl@0
   852
sl@0
   853
/*
sl@0
   854
** Delete the named file.
sl@0
   855
*/
sl@0
   856
static int os2Delete(
sl@0
   857
  sqlite3_vfs *pVfs,                     /* Not used on os2 */
sl@0
   858
  const char *zFilename,                 /* Name of file to delete */
sl@0
   859
  int syncDir                            /* Not used on os2 */
sl@0
   860
){
sl@0
   861
  APIRET rc = NO_ERROR;
sl@0
   862
  char *zFilenameCp = convertUtf8PathToCp( zFilename );
sl@0
   863
  SimulateIOError( return SQLITE_IOERR_DELETE );
sl@0
   864
  rc = DosDelete( (PSZ)zFilenameCp );
sl@0
   865
  free( zFilenameCp );
sl@0
   866
  OSTRACE2( "DELETE \"%s\"\n", zFilename );
sl@0
   867
  return rc == NO_ERROR ? SQLITE_OK : SQLITE_IOERR_DELETE;
sl@0
   868
}
sl@0
   869
sl@0
   870
/*
sl@0
   871
** Check the existance and status of a file.
sl@0
   872
*/
sl@0
   873
static int os2Access(
sl@0
   874
  sqlite3_vfs *pVfs,        /* Not used on os2 */
sl@0
   875
  const char *zFilename,    /* Name of file to check */
sl@0
   876
  int flags,                /* Type of test to make on this file */
sl@0
   877
  int *pOut                 /* Write results here */
sl@0
   878
){
sl@0
   879
  FILESTATUS3 fsts3ConfigInfo;
sl@0
   880
  APIRET rc = NO_ERROR;
sl@0
   881
  char *zFilenameCp = convertUtf8PathToCp( zFilename );
sl@0
   882
sl@0
   883
  memset( &fsts3ConfigInfo, 0, sizeof(fsts3ConfigInfo) );
sl@0
   884
  rc = DosQueryPathInfo( (PSZ)zFilenameCp, FIL_STANDARD,
sl@0
   885
                         &fsts3ConfigInfo, sizeof(FILESTATUS3) );
sl@0
   886
  free( zFilenameCp );
sl@0
   887
  OSTRACE4( "ACCESS fsts3ConfigInfo.attrFile=%d flags=%d rc=%d\n",
sl@0
   888
            fsts3ConfigInfo.attrFile, flags, rc );
sl@0
   889
  switch( flags ){
sl@0
   890
    case SQLITE_ACCESS_READ:
sl@0
   891
    case SQLITE_ACCESS_EXISTS:
sl@0
   892
      rc = (rc == NO_ERROR);
sl@0
   893
      OSTRACE3( "ACCESS %s access of read and exists  rc=%d\n", zFilename, rc );
sl@0
   894
      break;
sl@0
   895
    case SQLITE_ACCESS_READWRITE:
sl@0
   896
      rc = (rc == NO_ERROR) && ( (fsts3ConfigInfo.attrFile & FILE_READONLY) == 0 );
sl@0
   897
      OSTRACE3( "ACCESS %s access of read/write  rc=%d\n", zFilename, rc );
sl@0
   898
      break;
sl@0
   899
    default:
sl@0
   900
      assert( !"Invalid flags argument" );
sl@0
   901
  }
sl@0
   902
  *pOut = rc;
sl@0
   903
  return SQLITE_OK;
sl@0
   904
}
sl@0
   905
sl@0
   906
sl@0
   907
#ifndef SQLITE_OMIT_LOAD_EXTENSION
sl@0
   908
/*
sl@0
   909
** Interfaces for opening a shared library, finding entry points
sl@0
   910
** within the shared library, and closing the shared library.
sl@0
   911
*/
sl@0
   912
/*
sl@0
   913
** Interfaces for opening a shared library, finding entry points
sl@0
   914
** within the shared library, and closing the shared library.
sl@0
   915
*/
sl@0
   916
static void *os2DlOpen(sqlite3_vfs *pVfs, const char *zFilename){
sl@0
   917
  UCHAR loadErr[256];
sl@0
   918
  HMODULE hmod;
sl@0
   919
  APIRET rc;
sl@0
   920
  char *zFilenameCp = convertUtf8PathToCp(zFilename);
sl@0
   921
  rc = DosLoadModule((PSZ)loadErr, sizeof(loadErr), zFilenameCp, &hmod);
sl@0
   922
  free(zFilenameCp);
sl@0
   923
  return rc != NO_ERROR ? 0 : (void*)hmod;
sl@0
   924
}
sl@0
   925
/*
sl@0
   926
** A no-op since the error code is returned on the DosLoadModule call.
sl@0
   927
** os2Dlopen returns zero if DosLoadModule is not successful.
sl@0
   928
*/
sl@0
   929
static void os2DlError(sqlite3_vfs *pVfs, int nBuf, char *zBufOut){
sl@0
   930
/* no-op */
sl@0
   931
}
sl@0
   932
static void *os2DlSym(sqlite3_vfs *pVfs, void *pHandle, const char *zSymbol){
sl@0
   933
  PFN pfn;
sl@0
   934
  APIRET rc;
sl@0
   935
  rc = DosQueryProcAddr((HMODULE)pHandle, 0L, zSymbol, &pfn);
sl@0
   936
  if( rc != NO_ERROR ){
sl@0
   937
    /* if the symbol itself was not found, search again for the same
sl@0
   938
     * symbol with an extra underscore, that might be needed depending
sl@0
   939
     * on the calling convention */
sl@0
   940
    char _zSymbol[256] = "_";
sl@0
   941
    strncat(_zSymbol, zSymbol, 255);
sl@0
   942
    rc = DosQueryProcAddr((HMODULE)pHandle, 0L, _zSymbol, &pfn);
sl@0
   943
  }
sl@0
   944
  return rc != NO_ERROR ? 0 : (void*)pfn;
sl@0
   945
}
sl@0
   946
static void os2DlClose(sqlite3_vfs *pVfs, void *pHandle){
sl@0
   947
  DosFreeModule((HMODULE)pHandle);
sl@0
   948
}
sl@0
   949
#else /* if SQLITE_OMIT_LOAD_EXTENSION is defined: */
sl@0
   950
  #define os2DlOpen 0
sl@0
   951
  #define os2DlError 0
sl@0
   952
  #define os2DlSym 0
sl@0
   953
  #define os2DlClose 0
sl@0
   954
#endif
sl@0
   955
sl@0
   956
sl@0
   957
/*
sl@0
   958
** Write up to nBuf bytes of randomness into zBuf.
sl@0
   959
*/
sl@0
   960
static int os2Randomness(sqlite3_vfs *pVfs, int nBuf, char *zBuf ){
sl@0
   961
  ULONG sizeofULong = sizeof(ULONG);
sl@0
   962
  int n = 0;
sl@0
   963
  if( sizeof(DATETIME) <= nBuf - n ){
sl@0
   964
    DATETIME x;
sl@0
   965
    DosGetDateTime(&x);
sl@0
   966
    memcpy(&zBuf[n], &x, sizeof(x));
sl@0
   967
    n += sizeof(x);
sl@0
   968
  }
sl@0
   969
sl@0
   970
  if( sizeofULong <= nBuf - n ){
sl@0
   971
    PPIB ppib;
sl@0
   972
    DosGetInfoBlocks(NULL, &ppib);
sl@0
   973
    memcpy(&zBuf[n], &ppib->pib_ulpid, sizeofULong);
sl@0
   974
    n += sizeofULong;
sl@0
   975
  }
sl@0
   976
sl@0
   977
  if( sizeofULong <= nBuf - n ){
sl@0
   978
    PTIB ptib;
sl@0
   979
    DosGetInfoBlocks(&ptib, NULL);
sl@0
   980
    memcpy(&zBuf[n], &ptib->tib_ptib2->tib2_ultid, sizeofULong);
sl@0
   981
    n += sizeofULong;
sl@0
   982
  }
sl@0
   983
sl@0
   984
  /* if we still haven't filled the buffer yet the following will */
sl@0
   985
  /* grab everything once instead of making several calls for a single item */
sl@0
   986
  if( sizeofULong <= nBuf - n ){
sl@0
   987
    ULONG ulSysInfo[QSV_MAX];
sl@0
   988
    DosQuerySysInfo(1L, QSV_MAX, ulSysInfo, sizeofULong * QSV_MAX);
sl@0
   989
sl@0
   990
    memcpy(&zBuf[n], &ulSysInfo[QSV_MS_COUNT - 1], sizeofULong);
sl@0
   991
    n += sizeofULong;
sl@0
   992
sl@0
   993
    if( sizeofULong <= nBuf - n ){
sl@0
   994
      memcpy(&zBuf[n], &ulSysInfo[QSV_TIMER_INTERVAL - 1], sizeofULong);
sl@0
   995
      n += sizeofULong;
sl@0
   996
    }
sl@0
   997
    if( sizeofULong <= nBuf - n ){
sl@0
   998
      memcpy(&zBuf[n], &ulSysInfo[QSV_TIME_LOW - 1], sizeofULong);
sl@0
   999
      n += sizeofULong;
sl@0
  1000
    }
sl@0
  1001
    if( sizeofULong <= nBuf - n ){
sl@0
  1002
      memcpy(&zBuf[n], &ulSysInfo[QSV_TIME_HIGH - 1], sizeofULong);
sl@0
  1003
      n += sizeofULong;
sl@0
  1004
    }
sl@0
  1005
    if( sizeofULong <= nBuf - n ){
sl@0
  1006
      memcpy(&zBuf[n], &ulSysInfo[QSV_TOTAVAILMEM - 1], sizeofULong);
sl@0
  1007
      n += sizeofULong;
sl@0
  1008
    }
sl@0
  1009
  }
sl@0
  1010
sl@0
  1011
  return n;
sl@0
  1012
}
sl@0
  1013
sl@0
  1014
/*
sl@0
  1015
** Sleep for a little while.  Return the amount of time slept.
sl@0
  1016
** The argument is the number of microseconds we want to sleep.
sl@0
  1017
** The return value is the number of microseconds of sleep actually
sl@0
  1018
** requested from the underlying operating system, a number which
sl@0
  1019
** might be greater than or equal to the argument, but not less
sl@0
  1020
** than the argument.
sl@0
  1021
*/
sl@0
  1022
static int os2Sleep( sqlite3_vfs *pVfs, int microsec ){
sl@0
  1023
  DosSleep( (microsec/1000) );
sl@0
  1024
  return microsec;
sl@0
  1025
}
sl@0
  1026
sl@0
  1027
/*
sl@0
  1028
** The following variable, if set to a non-zero value, becomes the result
sl@0
  1029
** returned from sqlite3OsCurrentTime().  This is used for testing.
sl@0
  1030
*/
sl@0
  1031
#ifdef SQLITE_TEST
sl@0
  1032
int sqlite3_current_time = 0;
sl@0
  1033
#endif
sl@0
  1034
sl@0
  1035
/*
sl@0
  1036
** Find the current time (in Universal Coordinated Time).  Write the
sl@0
  1037
** current time and date as a Julian Day number into *prNow and
sl@0
  1038
** return 0.  Return 1 if the time and date cannot be found.
sl@0
  1039
*/
sl@0
  1040
int os2CurrentTime( sqlite3_vfs *pVfs, double *prNow ){
sl@0
  1041
  double now;
sl@0
  1042
  SHORT minute; /* needs to be able to cope with negative timezone offset */
sl@0
  1043
  USHORT second, hour,
sl@0
  1044
         day, month, year;
sl@0
  1045
  DATETIME dt;
sl@0
  1046
  DosGetDateTime( &dt );
sl@0
  1047
  second = (USHORT)dt.seconds;
sl@0
  1048
  minute = (SHORT)dt.minutes + dt.timezone;
sl@0
  1049
  hour = (USHORT)dt.hours;
sl@0
  1050
  day = (USHORT)dt.day;
sl@0
  1051
  month = (USHORT)dt.month;
sl@0
  1052
  year = (USHORT)dt.year;
sl@0
  1053
sl@0
  1054
  /* Calculations from http://www.astro.keele.ac.uk/~rno/Astronomy/hjd.html
sl@0
  1055
     http://www.astro.keele.ac.uk/~rno/Astronomy/hjd-0.1.c */
sl@0
  1056
  /* Calculate the Julian days */
sl@0
  1057
  now = day - 32076 +
sl@0
  1058
    1461*(year + 4800 + (month - 14)/12)/4 +
sl@0
  1059
    367*(month - 2 - (month - 14)/12*12)/12 -
sl@0
  1060
    3*((year + 4900 + (month - 14)/12)/100)/4;
sl@0
  1061
sl@0
  1062
  /* Add the fractional hours, mins and seconds */
sl@0
  1063
  now += (hour + 12.0)/24.0;
sl@0
  1064
  now += minute/1440.0;
sl@0
  1065
  now += second/86400.0;
sl@0
  1066
  *prNow = now;
sl@0
  1067
#ifdef SQLITE_TEST
sl@0
  1068
  if( sqlite3_current_time ){
sl@0
  1069
    *prNow = sqlite3_current_time/86400.0 + 2440587.5;
sl@0
  1070
  }
sl@0
  1071
#endif
sl@0
  1072
  return 0;
sl@0
  1073
}
sl@0
  1074
sl@0
  1075
static int os2GetLastError(sqlite3_vfs *pVfs, int nBuf, char *zBuf){
sl@0
  1076
  return 0;
sl@0
  1077
}
sl@0
  1078
sl@0
  1079
/*
sl@0
  1080
** Initialize and deinitialize the operating system interface.
sl@0
  1081
*/
sl@0
  1082
int sqlite3_os_init(void){
sl@0
  1083
  static sqlite3_vfs os2Vfs = {
sl@0
  1084
    1,                 /* iVersion */
sl@0
  1085
    sizeof(os2File),   /* szOsFile */
sl@0
  1086
    CCHMAXPATH,        /* mxPathname */
sl@0
  1087
    0,                 /* pNext */
sl@0
  1088
    "os2",             /* zName */
sl@0
  1089
    0,                 /* pAppData */
sl@0
  1090
sl@0
  1091
    os2Open,           /* xOpen */
sl@0
  1092
    os2Delete,         /* xDelete */
sl@0
  1093
    os2Access,         /* xAccess */
sl@0
  1094
    os2FullPathname,   /* xFullPathname */
sl@0
  1095
    os2DlOpen,         /* xDlOpen */
sl@0
  1096
    os2DlError,        /* xDlError */
sl@0
  1097
    os2DlSym,          /* xDlSym */
sl@0
  1098
    os2DlClose,        /* xDlClose */
sl@0
  1099
    os2Randomness,     /* xRandomness */
sl@0
  1100
    os2Sleep,          /* xSleep */
sl@0
  1101
    os2CurrentTime,    /* xCurrentTime */
sl@0
  1102
    os2GetLastError    /* xGetLastError */
sl@0
  1103
  };
sl@0
  1104
  sqlite3_vfs_register(&os2Vfs, 1);
sl@0
  1105
  initUconvObjects();
sl@0
  1106
  return SQLITE_OK;
sl@0
  1107
}
sl@0
  1108
int sqlite3_os_end(void){
sl@0
  1109
  freeUconvObjects();
sl@0
  1110
  return SQLITE_OK;
sl@0
  1111
}
sl@0
  1112
sl@0
  1113
#endif /* SQLITE_OS_OS2 */