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 */
|