sl@0
|
1 |
// Copyright (c) 2005-2010 Nokia Corporation and/or its subsidiary(-ies).
|
sl@0
|
2 |
// All rights reserved.
|
sl@0
|
3 |
// This component and the accompanying materials are made available
|
sl@0
|
4 |
// under the terms of "Eclipse Public License v1.0"
|
sl@0
|
5 |
// which accompanies this distribution, and is available
|
sl@0
|
6 |
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
|
sl@0
|
7 |
//
|
sl@0
|
8 |
// Initial Contributors:
|
sl@0
|
9 |
// Nokia Corporation - initial contribution.
|
sl@0
|
10 |
//
|
sl@0
|
11 |
// Contributors:
|
sl@0
|
12 |
//
|
sl@0
|
13 |
// Description:
|
sl@0
|
14 |
// The Symbian OS porting layer - single-threaded implementation.
|
sl@0
|
15 |
// SQLite never accesses the file system and the OS services directly.
|
sl@0
|
16 |
// SQLite uses for that sqlite3_vfs and sqlite3_file objects.
|
sl@0
|
17 |
// sqlite3_vfs and sqlite3_file functionality is implemented in this file -
|
sl@0
|
18 |
// TVfs and TFileIo classes.
|
sl@0
|
19 |
// This file is also used for the COsLayerData implementation. A single COslayerData
|
sl@0
|
20 |
// object is used by the OS porting layer for managing some global data.
|
sl@0
|
21 |
//
|
sl@0
|
22 |
//
|
sl@0
|
23 |
|
sl@0
|
24 |
/**
|
sl@0
|
25 |
@file
|
sl@0
|
26 |
@see TVfs
|
sl@0
|
27 |
@see TFileIo
|
sl@0
|
28 |
*/
|
sl@0
|
29 |
|
sl@0
|
30 |
#ifdef SQLITE_OS_SYMBIAN
|
sl@0
|
31 |
|
sl@0
|
32 |
//#define _SQLPROFILER // Enable profiling //The same macro has to be enabled in SqlAssert.h file
|
sl@0
|
33 |
|
sl@0
|
34 |
extern "C"
|
sl@0
|
35 |
{
|
sl@0
|
36 |
#include "sqliteInt.h"
|
sl@0
|
37 |
#include "os.h"
|
sl@0
|
38 |
}
|
sl@0
|
39 |
#include "os_common.h"
|
sl@0
|
40 |
#include "SqliteSymbian.h"
|
sl@0
|
41 |
#include "FileBuf64.h"
|
sl@0
|
42 |
#include <e32math.h>
|
sl@0
|
43 |
#ifdef _SQLPROFILER
|
sl@0
|
44 |
#include <hal.h>
|
sl@0
|
45 |
#include "../INC/SqlResourceProfiler.h"
|
sl@0
|
46 |
#endif
|
sl@0
|
47 |
#include "SqliteUtil.h"
|
sl@0
|
48 |
#include "OstTraceDefinitions.h"
|
sl@0
|
49 |
#ifdef OST_TRACE_COMPILER_IN_USE
|
sl@0
|
50 |
#include "os_symbianTraces.h"
|
sl@0
|
51 |
#endif
|
sl@0
|
52 |
#include "SqliteTraceDef.h"
|
sl@0
|
53 |
|
sl@0
|
54 |
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
55 |
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
56 |
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
57 |
|
sl@0
|
58 |
//The SQLite temp files willl be created in this subdir
|
sl@0
|
59 |
_LIT(KTempFileDir, "temp");
|
sl@0
|
60 |
|
sl@0
|
61 |
//Bit-mask constant. If xOpen()'s "aFlag" parameter contains one of these bits set, then the the file top be
|
sl@0
|
62 |
//opened or created is a journal file.
|
sl@0
|
63 |
const TUint KJournalFileTypeBitMask = SQLITE_OPEN_MAIN_JOURNAL | SQLITE_OPEN_TEMP_JOURNAL | SQLITE_OPEN_SUBJOURNAL | SQLITE_OPEN_MASTER_JOURNAL;
|
sl@0
|
64 |
|
sl@0
|
65 |
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
66 |
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
67 |
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
68 |
#ifdef _SQLPROFILER
|
sl@0
|
69 |
|
sl@0
|
70 |
//Profiling counters, defined in SqlSrvResourceprofiler.cpp
|
sl@0
|
71 |
extern TInt TheSqlSrvProfilerFileRead;
|
sl@0
|
72 |
extern TInt TheSqlSrvProfilerFileWrite;
|
sl@0
|
73 |
extern TInt TheSqlSrvProfilerFileSync;
|
sl@0
|
74 |
extern TInt TheSqlSrvProfilerFileSetSize;
|
sl@0
|
75 |
|
sl@0
|
76 |
# define __COUNTER_INCR(counter) ++counter
|
sl@0
|
77 |
|
sl@0
|
78 |
//////////////////////////////////////////////////////////////////////////////
|
sl@0
|
79 |
////////////////////////// File I/O //////////////////////////////////////////
|
sl@0
|
80 |
//////////////////////////////////////////////////////////////////////////////
|
sl@0
|
81 |
|
sl@0
|
82 |
/**
|
sl@0
|
83 |
This enum is used only in _SQLPROFILER mode for tracing the file system calls.
|
sl@0
|
84 |
@internalComponent
|
sl@0
|
85 |
@see FsCallBrkPt()
|
sl@0
|
86 |
*/
|
sl@0
|
87 |
enum TFsOpType
|
sl@0
|
88 |
{
|
sl@0
|
89 |
EFsOpFileCreate,
|
sl@0
|
90 |
EFsOpFileOpen,
|
sl@0
|
91 |
EFsOpFileClose,
|
sl@0
|
92 |
EFsOpFileDelete,
|
sl@0
|
93 |
EFsOpFileRead,
|
sl@0
|
94 |
EFsOpFileWrite,
|
sl@0
|
95 |
EFsOpFileSeek,
|
sl@0
|
96 |
EFsOpFileSize,
|
sl@0
|
97 |
EFsOpFileSetSize,
|
sl@0
|
98 |
EFsOpFileSync,
|
sl@0
|
99 |
EFsOpFileDrive,
|
sl@0
|
100 |
EFsOpFileAdopt,
|
sl@0
|
101 |
EFsOpFsClose,
|
sl@0
|
102 |
EFsOpFsConnect,
|
sl@0
|
103 |
EFsOpFsGetSystemDrive,
|
sl@0
|
104 |
EFsOpFsCreatePrivatePath,
|
sl@0
|
105 |
EFsOpFsPrivatePath,
|
sl@0
|
106 |
EFsOpFsVolumeIoParam,
|
sl@0
|
107 |
EFsOpFsEntry,
|
sl@0
|
108 |
EFsOpFsAtt,
|
sl@0
|
109 |
EFsOpFileCreateTemp,
|
sl@0
|
110 |
EFsOpFileAttach,
|
sl@0
|
111 |
//
|
sl@0
|
112 |
EFsOpLast
|
sl@0
|
113 |
};
|
sl@0
|
114 |
|
sl@0
|
115 |
TBool TheFileIoProfileEnabled = EFalse;
|
sl@0
|
116 |
TUint32 TheFileOpCounter[EFsOpLast] = {0};
|
sl@0
|
117 |
TInt64 TheFileWriteAmount = 0;
|
sl@0
|
118 |
TInt64 TheFileReadAmount = 0;
|
sl@0
|
119 |
|
sl@0
|
120 |
/**
|
sl@0
|
121 |
This function is used only in _SQLPROFILER mode as an appropriate place for:
|
sl@0
|
122 |
- setting breakpoints for tracing the file system calls;
|
sl@0
|
123 |
- collection information about the number of the file system calls;
|
sl@0
|
124 |
|
sl@0
|
125 |
@param aFsOpType A TFsOpType enum item value, identifying the file system operation that will be executed;
|
sl@0
|
126 |
@param a1 If the operation is "file read" or "file write" - the amount of data read/written;
|
sl@0
|
127 |
|
sl@0
|
128 |
@internalComponent
|
sl@0
|
129 |
|
sl@0
|
130 |
@see TFsOpType
|
sl@0
|
131 |
*/
|
sl@0
|
132 |
void FsCallBrkPt(TInt aFsOpType, TInt a1)
|
sl@0
|
133 |
{
|
sl@0
|
134 |
__ASSERT_DEBUG(aFsOpType >= 0 && aFsOpType < EFsOpLast, User::Invariant());
|
sl@0
|
135 |
if(!TheFileIoProfileEnabled)
|
sl@0
|
136 |
{
|
sl@0
|
137 |
return;
|
sl@0
|
138 |
}
|
sl@0
|
139 |
TFsOpType fsOpType = (TFsOpType)aFsOpType;
|
sl@0
|
140 |
++TheFileOpCounter[fsOpType];
|
sl@0
|
141 |
if(fsOpType == EFsOpFileWrite)
|
sl@0
|
142 |
{
|
sl@0
|
143 |
TheFileWriteAmount += a1;
|
sl@0
|
144 |
}
|
sl@0
|
145 |
else if(fsOpType == EFsOpFileRead)
|
sl@0
|
146 |
{
|
sl@0
|
147 |
TheFileReadAmount += a1;
|
sl@0
|
148 |
}
|
sl@0
|
149 |
}
|
sl@0
|
150 |
|
sl@0
|
151 |
# define __FS_CALL(aFsOpType, a1) FsCallBrkPt(aFsOpType, a1)
|
sl@0
|
152 |
|
sl@0
|
153 |
//////////////////////////////////////////////////////////////////////////////
|
sl@0
|
154 |
////////////////////////// Heap Stats ////////////////////////////////////////
|
sl@0
|
155 |
//////////////////////////////////////////////////////////////////////////////
|
sl@0
|
156 |
|
sl@0
|
157 |
/**
|
sl@0
|
158 |
Memory operation type: alloc, realloc, free. Used during the profiling.
|
sl@0
|
159 |
|
sl@0
|
160 |
@internalComponent
|
sl@0
|
161 |
*/
|
sl@0
|
162 |
enum TMemOpType
|
sl@0
|
163 |
{
|
sl@0
|
164 |
EMemOpAlloc,
|
sl@0
|
165 |
EMemOpRealloc,
|
sl@0
|
166 |
EMemOpFree,
|
sl@0
|
167 |
//
|
sl@0
|
168 |
EMemOpLast
|
sl@0
|
169 |
};
|
sl@0
|
170 |
|
sl@0
|
171 |
TBool TheMemProfileEnabled = EFalse;//Mem operation call counters and time counters enabled/disabled
|
sl@0
|
172 |
TBool TheMaxAllocProfileEnabled = EFalse;//Max mem allocation enabled/disabled
|
sl@0
|
173 |
TUint32 TheMemOpCounter[EMemOpLast] = {0};
|
sl@0
|
174 |
TInt64 TheMemOpTicks[EMemOpLast] = {0};
|
sl@0
|
175 |
TInt64 TheAllocated = 0;
|
sl@0
|
176 |
TInt64 TheFreed = 0;
|
sl@0
|
177 |
TInt TheAllocMax = 0;
|
sl@0
|
178 |
|
sl@0
|
179 |
/**
|
sl@0
|
180 |
This class is used only in _SQLPROFILER mode as an appropriate place for:
|
sl@0
|
181 |
- setting breakpoints for tracing the memory allocation/deallocation calls;
|
sl@0
|
182 |
- collection information about the number of the memory allocation/deallocation calls and the time spent in the calls;
|
sl@0
|
183 |
|
sl@0
|
184 |
The constructor's parameters are:
|
sl@0
|
185 |
- aOpType A TMemOpType enum item value, identifying the operation that will be executed;
|
sl@0
|
186 |
- aAmt1 The allocated/deallocated size;
|
sl@0
|
187 |
- aAmt2 Used only if a block of memory is reallocated in which case a2 is the old size of the block;
|
sl@0
|
188 |
|
sl@0
|
189 |
@internalComponent
|
sl@0
|
190 |
|
sl@0
|
191 |
@see TMemOpType
|
sl@0
|
192 |
*/
|
sl@0
|
193 |
class TMemCallCounter
|
sl@0
|
194 |
{
|
sl@0
|
195 |
public:
|
sl@0
|
196 |
TMemCallCounter(TMemOpType aOpType, TInt aAmt1, TInt aAmt2) :
|
sl@0
|
197 |
iOpType(aOpType),
|
sl@0
|
198 |
iStartTicks(0)
|
sl@0
|
199 |
{
|
sl@0
|
200 |
if(TheMaxAllocProfileEnabled && (iOpType == EMemOpAlloc || iOpType == EMemOpRealloc) && aAmt1 > TheAllocMax)
|
sl@0
|
201 |
{
|
sl@0
|
202 |
TheAllocMax = aAmt1;
|
sl@0
|
203 |
}
|
sl@0
|
204 |
if(TheMemProfileEnabled)
|
sl@0
|
205 |
{
|
sl@0
|
206 |
++TheMemOpCounter[iOpType];
|
sl@0
|
207 |
switch(iOpType)
|
sl@0
|
208 |
{
|
sl@0
|
209 |
case EMemOpAlloc:
|
sl@0
|
210 |
TheAllocated += aAmt1;
|
sl@0
|
211 |
break;
|
sl@0
|
212 |
case EMemOpRealloc:
|
sl@0
|
213 |
TheAllocated += aAmt1;
|
sl@0
|
214 |
TheFreed += aAmt2;
|
sl@0
|
215 |
break;
|
sl@0
|
216 |
case EMemOpFree:
|
sl@0
|
217 |
TheFreed += aAmt1;
|
sl@0
|
218 |
break;
|
sl@0
|
219 |
default:
|
sl@0
|
220 |
__ASSERT_DEBUG(0, User::Invariant());
|
sl@0
|
221 |
break;
|
sl@0
|
222 |
}
|
sl@0
|
223 |
iStartTicks = User::FastCounter();
|
sl@0
|
224 |
}
|
sl@0
|
225 |
}
|
sl@0
|
226 |
~TMemCallCounter()
|
sl@0
|
227 |
{
|
sl@0
|
228 |
if(TheMemProfileEnabled)
|
sl@0
|
229 |
{
|
sl@0
|
230 |
TInt64 diffTicks = (TInt64)User::FastCounter() - (TInt64)iStartTicks;
|
sl@0
|
231 |
if(diffTicks < 0)
|
sl@0
|
232 |
{
|
sl@0
|
233 |
diffTicks = KMaxTUint + diffTicks + 1;
|
sl@0
|
234 |
}
|
sl@0
|
235 |
TheMemOpTicks[iOpType] += diffTicks;
|
sl@0
|
236 |
}
|
sl@0
|
237 |
}
|
sl@0
|
238 |
private:
|
sl@0
|
239 |
TMemOpType iOpType;
|
sl@0
|
240 |
TUint32 iStartTicks;
|
sl@0
|
241 |
};
|
sl@0
|
242 |
|
sl@0
|
243 |
# define __MEM_CALL(aMemOpType, a1, a2) TMemCallCounter memCallCounter(aMemOpType, a1, a2)
|
sl@0
|
244 |
|
sl@0
|
245 |
//////////////////////////////////////////////////////////////////////////////
|
sl@0
|
246 |
////////////////////////// OS layer calls ////////////////////////////////////
|
sl@0
|
247 |
//////////////////////////////////////////////////////////////////////////////
|
sl@0
|
248 |
|
sl@0
|
249 |
//The OS porting layer call types
|
sl@0
|
250 |
enum TOsOpType
|
sl@0
|
251 |
{
|
sl@0
|
252 |
EOsFileClose,
|
sl@0
|
253 |
EOsFileRead,
|
sl@0
|
254 |
EOsFileWrite,
|
sl@0
|
255 |
EOsFileTruncate,
|
sl@0
|
256 |
EOsFileSync,
|
sl@0
|
257 |
EOsFileFileSize,
|
sl@0
|
258 |
EOsFileLock,
|
sl@0
|
259 |
EOsFileUnlock,
|
sl@0
|
260 |
EOsFileCheckReservedLock,
|
sl@0
|
261 |
EOsFileFileControl,
|
sl@0
|
262 |
EOsFileSectorSize,
|
sl@0
|
263 |
EOsFileDeviceCharacteristics,
|
sl@0
|
264 |
//
|
sl@0
|
265 |
EOsVfsOpen,
|
sl@0
|
266 |
EOsVfsDelete,
|
sl@0
|
267 |
EOsVfsAccess,
|
sl@0
|
268 |
EOsVfsFullPathName,
|
sl@0
|
269 |
EOsVfsRandomness,
|
sl@0
|
270 |
EOsVfsSleep,
|
sl@0
|
271 |
EOsVfsCurrentTime,
|
sl@0
|
272 |
EOsVfsGetLastError,
|
sl@0
|
273 |
//
|
sl@0
|
274 |
EOsOpLast
|
sl@0
|
275 |
};
|
sl@0
|
276 |
|
sl@0
|
277 |
TBool TheOsProfileEnabled = EFalse;
|
sl@0
|
278 |
TUint32 TheOsOpCounter[EOsOpLast] = {0};//Each entry is a counter - how many times specific OS porting layer function has been called
|
sl@0
|
279 |
|
sl@0
|
280 |
# define __OS_CALL(aOsOpType, a1, a2) \
|
sl@0
|
281 |
do \
|
sl@0
|
282 |
{ \
|
sl@0
|
283 |
if(TheOsProfileEnabled) \
|
sl@0
|
284 |
{ \
|
sl@0
|
285 |
++TheOsOpCounter[aOsOpType];\
|
sl@0
|
286 |
} \
|
sl@0
|
287 |
} \
|
sl@0
|
288 |
while(0)
|
sl@0
|
289 |
|
sl@0
|
290 |
//////////////////////////////////////////////////////////////////////////////
|
sl@0
|
291 |
////////////////////////// OS layer timings //////////////////////////////////
|
sl@0
|
292 |
//////////////////////////////////////////////////////////////////////////////
|
sl@0
|
293 |
|
sl@0
|
294 |
TBool TheOsCallTimeProfileEnabled = EFalse;//If true, the OS porting layer call timings are enabled.
|
sl@0
|
295 |
TInt64 TheOsCallTicks[EOsOpLast];//Each entry represents the time in ticks spent in a specific OS porting layer function,
|
sl@0
|
296 |
//disregarding the file type (main or journal)
|
sl@0
|
297 |
|
sl@0
|
298 |
TBool TheOsCallTimeDetailedProfileEnabled = EFalse;//If true, the OS porting layer call details are enabled and for each call an entry will be added to the log file (epocwind.out).
|
sl@0
|
299 |
TInt TheOpCounter = 0;//Operations counter. Each OS porting layer function call increments the counter.
|
sl@0
|
300 |
|
sl@0
|
301 |
//Used for storing the OS call details:
|
sl@0
|
302 |
// - iType - on which file the call has been made: main database file - 'M', or journal file - 'J';
|
sl@0
|
303 |
// - iIdentifier - two letters identifying the monitored OS porting layer function;
|
sl@0
|
304 |
// - iCallCounter - how many times the monitored OS porting layer function has been called;
|
sl@0
|
305 |
// - iTicksTotal - the total amount of time in ticks spent in the monitored OS porting layer function;
|
sl@0
|
306 |
// - iBytesTotal - the total amount of bytes passed to the monitored OS porting layer function (if it is read or write);
|
sl@0
|
307 |
struct TOsCallProfile
|
sl@0
|
308 |
{
|
sl@0
|
309 |
TOsCallProfile(char aType, const TDesC& aIdentifier) :
|
sl@0
|
310 |
iType(aType),
|
sl@0
|
311 |
iIdentifier(aIdentifier),
|
sl@0
|
312 |
iCallCounter(0),
|
sl@0
|
313 |
iTicksTotal(0),
|
sl@0
|
314 |
iBytesTotal(0)
|
sl@0
|
315 |
{
|
sl@0
|
316 |
}
|
sl@0
|
317 |
void Zero()
|
sl@0
|
318 |
{
|
sl@0
|
319 |
iCallCounter = 0;
|
sl@0
|
320 |
iTicksTotal = 0;
|
sl@0
|
321 |
iBytesTotal = 0;
|
sl@0
|
322 |
}
|
sl@0
|
323 |
char iType;
|
sl@0
|
324 |
TBuf<32> iIdentifier;
|
sl@0
|
325 |
TInt iCallCounter;
|
sl@0
|
326 |
TInt64 iTicksTotal;
|
sl@0
|
327 |
TInt64 iBytesTotal;
|
sl@0
|
328 |
};
|
sl@0
|
329 |
|
sl@0
|
330 |
//An array of TOsCallProfile entries, each entry keeps the profile of a specifc OS porting layer function, when
|
sl@0
|
331 |
//the function was used on the main database file
|
sl@0
|
332 |
TOsCallProfile TheOsCallMProfile[EOsOpLast] =
|
sl@0
|
333 |
{
|
sl@0
|
334 |
TOsCallProfile('M', _L("Close")), TOsCallProfile('M', _L("Read")), TOsCallProfile('M', _L("Write")), TOsCallProfile('M', _L("Truncate")),
|
sl@0
|
335 |
TOsCallProfile('M', _L("Sync")), TOsCallProfile('M', _L("Size")), TOsCallProfile('M', _L("Lock")), TOsCallProfile('M', _L("Unlock")),
|
sl@0
|
336 |
TOsCallProfile('M', _L("CheckReservedLock")), TOsCallProfile('M', _L("FileControl")), TOsCallProfile('M', _L("SetSize")), TOsCallProfile('M', _L("DeviceCharacteristics")),
|
sl@0
|
337 |
TOsCallProfile('M', _L("Open")), TOsCallProfile('M', _L("Delete")), TOsCallProfile('M', _L("Access")), TOsCallProfile('M', _L("FullPath")),
|
sl@0
|
338 |
TOsCallProfile('M', _L("Randomness")), TOsCallProfile('M', _L("Sleep")), TOsCallProfile('M', _L("CurrentTime")), TOsCallProfile('M', _L("GetLastError"))
|
sl@0
|
339 |
};
|
sl@0
|
340 |
|
sl@0
|
341 |
//An array of TOsCallProfile entries, each entry keeps the profile of a specifc OS porting layer function, when
|
sl@0
|
342 |
//the function was used on the journal file
|
sl@0
|
343 |
TOsCallProfile TheOsCallJProfile[EOsOpLast] =
|
sl@0
|
344 |
{
|
sl@0
|
345 |
TOsCallProfile('J', _L("Close")), TOsCallProfile('J', _L("Read")), TOsCallProfile('J', _L("Write")), TOsCallProfile('J', _L("Truncate")),
|
sl@0
|
346 |
TOsCallProfile('J', _L("Sync")), TOsCallProfile('J', _L("Size")), TOsCallProfile('J', _L("Lock")), TOsCallProfile('J', _L("Unlock")),
|
sl@0
|
347 |
TOsCallProfile('J', _L("CheckReservedLock")), TOsCallProfile('J', _L("FileControl")), TOsCallProfile('J', _L("SetSize")), TOsCallProfile('J', _L("DeviceCharacteristics")),
|
sl@0
|
348 |
TOsCallProfile('J', _L("Open")), TOsCallProfile('J', _L("Delete")), TOsCallProfile('J', _L("Access")), TOsCallProfile('J', _L("FullPath")),
|
sl@0
|
349 |
TOsCallProfile('J', _L("Randomness")), TOsCallProfile('J', _L("Sleep")), TOsCallProfile('J', _L("CurrentTime")), TOsCallProfile('J', _L("GetLastError"))
|
sl@0
|
350 |
};
|
sl@0
|
351 |
|
sl@0
|
352 |
//The main class for the OS porting layer call profiles.
|
sl@0
|
353 |
class TOsCallCounter
|
sl@0
|
354 |
{
|
sl@0
|
355 |
public:
|
sl@0
|
356 |
//aOsCallTicksEntryRef - a reference to the related TheOsCallTicks[] entry;
|
sl@0
|
357 |
//aProfileRef - a reference to the related TOsCallProfile object - TheOsCallMProfile[] or TheOsCallJProfile[] entry;
|
sl@0
|
358 |
//aOffset - file offset in bytes;
|
sl@0
|
359 |
//aBytes - amount of bytes to be read/written;
|
sl@0
|
360 |
//aOptional - might be NULL. Used to print out the name of the file being processed.
|
sl@0
|
361 |
TOsCallCounter(TInt64& aOsCallTicksEntryRef, TOsCallProfile& aOsCallProfileRef, TInt64 aOffset, TInt aBytes,
|
sl@0
|
362 |
const sqlite3_file* aHandle, const char* aOptional) :
|
sl@0
|
363 |
iOsCallTicksEntryRef(aOsCallTicksEntryRef),
|
sl@0
|
364 |
iOsCallProfileRef(aOsCallProfileRef),
|
sl@0
|
365 |
iOffset(aOffset),
|
sl@0
|
366 |
iBytes(aBytes),
|
sl@0
|
367 |
iHandle((TUint)aHandle),
|
sl@0
|
368 |
iOptional((const TUint8*)aOptional)
|
sl@0
|
369 |
{
|
sl@0
|
370 |
if(TheOsCallTimeProfileEnabled)
|
sl@0
|
371 |
{
|
sl@0
|
372 |
iStartTicks = User::FastCounter();
|
sl@0
|
373 |
}
|
sl@0
|
374 |
}
|
sl@0
|
375 |
~TOsCallCounter()
|
sl@0
|
376 |
{
|
sl@0
|
377 |
if(TheOsCallTimeProfileEnabled)
|
sl@0
|
378 |
{
|
sl@0
|
379 |
TInt64 diffTicks = (TInt64)User::FastCounter() - (TInt64)iStartTicks;
|
sl@0
|
380 |
if(diffTicks < 0)
|
sl@0
|
381 |
{
|
sl@0
|
382 |
diffTicks = KMaxTUint + diffTicks + 1;
|
sl@0
|
383 |
}
|
sl@0
|
384 |
iOsCallTicksEntryRef += diffTicks;
|
sl@0
|
385 |
if(TheOsCallTimeDetailedProfileEnabled)
|
sl@0
|
386 |
{
|
sl@0
|
387 |
++TheOpCounter;
|
sl@0
|
388 |
++iOsCallProfileRef.iCallCounter;
|
sl@0
|
389 |
iOsCallProfileRef.iTicksTotal += diffTicks;
|
sl@0
|
390 |
iOsCallProfileRef.iBytesTotal += iBytes;
|
sl@0
|
391 |
TFileName fname;
|
sl@0
|
392 |
if(iOptional)
|
sl@0
|
393 |
{
|
sl@0
|
394 |
TPtrC8 fn8(iOptional);
|
sl@0
|
395 |
fname.Copy(fn8);
|
sl@0
|
396 |
}
|
sl@0
|
397 |
// 0 1 2 3 4 5 6 7 8 9 10
|
sl@0
|
398 |
RDebug::Print(_L("[SQL-OS]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬%c¬%S¬%d¬%d¬%ld¬%d¬%ld¬%ld¬%ld¬%S\n"),
|
sl@0
|
399 |
//[SQL-OS]
|
sl@0
|
400 |
//Handle
|
sl@0
|
401 |
//Time from start, microseconds
|
sl@0
|
402 |
//Subtype
|
sl@0
|
403 |
//IPC sequence counter
|
sl@0
|
404 |
//IPC call name
|
sl@0
|
405 |
iHandle, //sqlite3_file*
|
sl@0
|
406 |
iOsCallProfileRef.iType, //1 - main or journal file
|
sl@0
|
407 |
&iOsCallProfileRef.iIdentifier, //2 - operation type
|
sl@0
|
408 |
TheOpCounter, //3 - Operation counter
|
sl@0
|
409 |
iOsCallProfileRef.iCallCounter, //4 - This call type counter
|
sl@0
|
410 |
iOffset, //5 - File offset
|
sl@0
|
411 |
iBytes, //6 - Data, bytes
|
sl@0
|
412 |
diffTicks, //7 - Ticks
|
sl@0
|
413 |
iOsCallProfileRef.iBytesTotal, //8 - Data total, bytes
|
sl@0
|
414 |
iOsCallProfileRef.iTicksTotal, //9 - Ticks total
|
sl@0
|
415 |
&fname); //10- File name
|
sl@0
|
416 |
}
|
sl@0
|
417 |
}
|
sl@0
|
418 |
}
|
sl@0
|
419 |
private:
|
sl@0
|
420 |
TInt64& iOsCallTicksEntryRef;
|
sl@0
|
421 |
TOsCallProfile& iOsCallProfileRef;
|
sl@0
|
422 |
TInt64 iOffset;
|
sl@0
|
423 |
TInt iBytes;
|
sl@0
|
424 |
TUint32 iStartTicks;
|
sl@0
|
425 |
TUint iHandle;
|
sl@0
|
426 |
const TUint8* iOptional;
|
sl@0
|
427 |
};
|
sl@0
|
428 |
|
sl@0
|
429 |
inline TOsCallProfile& OsCallProfile(TBool aType, TInt aIndex)
|
sl@0
|
430 |
{
|
sl@0
|
431 |
return aType ? TheOsCallJProfile[aIndex] : TheOsCallMProfile[aIndex];
|
sl@0
|
432 |
}
|
sl@0
|
433 |
|
sl@0
|
434 |
# define __OSTIME_COUNTER(aOsCallTicksRef, aOsCallProfileRef, aOffset, aBytes, aHandle, aOpt) TOsCallCounter osCallCounter(aOsCallTicksRef, aOsCallProfileRef, aOffset, aBytes, aHandle, aOpt)
|
sl@0
|
435 |
|
sl@0
|
436 |
#else //_SQLPROFILER
|
sl@0
|
437 |
|
sl@0
|
438 |
# define __COUNTER_INCR(counter) void(0)
|
sl@0
|
439 |
|
sl@0
|
440 |
# define __FS_CALL(aFsOpType, a1) void(0)
|
sl@0
|
441 |
|
sl@0
|
442 |
# define __MEM_CALL(aMemOpType, a1, a2) void(0)
|
sl@0
|
443 |
|
sl@0
|
444 |
# define __OS_CALL(aOpType, a1, a2) void(0)
|
sl@0
|
445 |
|
sl@0
|
446 |
# define __OSTIME_COUNTER(aOsCallTicksRef, aOsCallProfileRef, aOffset, aBytes, aHandle, aOpt) void(0)
|
sl@0
|
447 |
|
sl@0
|
448 |
#endif//_SQLPROFILER
|
sl@0
|
449 |
|
sl@0
|
450 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
451 |
///////////////////// Profiling ////////////////////////////////////////////////////////////////
|
sl@0
|
452 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
453 |
|
sl@0
|
454 |
#ifdef _SQLPROFILER
|
sl@0
|
455 |
|
sl@0
|
456 |
const TInt KMicroSecIn1Sec = 1000000;
|
sl@0
|
457 |
|
sl@0
|
458 |
TInt FastCounterFrequency()
|
sl@0
|
459 |
{
|
sl@0
|
460 |
TInt ticksPerSec = 0;
|
sl@0
|
461 |
TInt err = HAL::Get(HAL::EFastCounterFrequency, ticksPerSec);
|
sl@0
|
462 |
__ASSERT_ALWAYS(err == KErrNone, __SQLITEPANIC2(ESqliteOsPanicFastCounterFreq));
|
sl@0
|
463 |
return ticksPerSec;
|
sl@0
|
464 |
}
|
sl@0
|
465 |
|
sl@0
|
466 |
TInt sqlite3SymbianProfilerStart(TInt aCounterType)
|
sl@0
|
467 |
{
|
sl@0
|
468 |
const TSqlResourceProfiler::TSqlCounter KCounterType = static_cast <TSqlResourceProfiler::TSqlCounter> (aCounterType);
|
sl@0
|
469 |
switch(KCounterType)
|
sl@0
|
470 |
{
|
sl@0
|
471 |
case TSqlResourceProfiler::ESqlCounterFileIO:
|
sl@0
|
472 |
TheFileIoProfileEnabled = ETrue;
|
sl@0
|
473 |
break;
|
sl@0
|
474 |
case TSqlResourceProfiler::ESqlCounterOsCall:
|
sl@0
|
475 |
TheOsProfileEnabled = ETrue;
|
sl@0
|
476 |
break;
|
sl@0
|
477 |
case TSqlResourceProfiler::ESqlCounterOsCallTime:
|
sl@0
|
478 |
TheOsCallTimeProfileEnabled = ETrue;
|
sl@0
|
479 |
break;
|
sl@0
|
480 |
case TSqlResourceProfiler::ESqlCounterOsCallDetails:
|
sl@0
|
481 |
TheOsCallTimeProfileEnabled = ETrue;
|
sl@0
|
482 |
TheOsCallTimeDetailedProfileEnabled = ETrue;
|
sl@0
|
483 |
break;
|
sl@0
|
484 |
case TSqlResourceProfiler::ESqlCounterMemory:
|
sl@0
|
485 |
TheMemProfileEnabled = ETrue;
|
sl@0
|
486 |
break;
|
sl@0
|
487 |
case TSqlResourceProfiler::ESqlCounterMaxAlloc:
|
sl@0
|
488 |
TheMaxAllocProfileEnabled = ETrue;
|
sl@0
|
489 |
break;
|
sl@0
|
490 |
default:
|
sl@0
|
491 |
return KErrNotSupported;
|
sl@0
|
492 |
}
|
sl@0
|
493 |
return KErrNone;
|
sl@0
|
494 |
}
|
sl@0
|
495 |
|
sl@0
|
496 |
TInt sqlite3SymbianProfilerStop(TInt aCounterType)
|
sl@0
|
497 |
{
|
sl@0
|
498 |
const TSqlResourceProfiler::TSqlCounter KCounterType = static_cast <TSqlResourceProfiler::TSqlCounter> (aCounterType);
|
sl@0
|
499 |
switch(KCounterType)
|
sl@0
|
500 |
{
|
sl@0
|
501 |
case TSqlResourceProfiler::ESqlCounterFileIO:
|
sl@0
|
502 |
TheFileIoProfileEnabled = EFalse;
|
sl@0
|
503 |
break;
|
sl@0
|
504 |
case TSqlResourceProfiler::ESqlCounterOsCall:
|
sl@0
|
505 |
TheOsProfileEnabled = EFalse;
|
sl@0
|
506 |
break;
|
sl@0
|
507 |
case TSqlResourceProfiler::ESqlCounterOsCallTime:
|
sl@0
|
508 |
TheOsCallTimeProfileEnabled = EFalse;
|
sl@0
|
509 |
break;
|
sl@0
|
510 |
case TSqlResourceProfiler::ESqlCounterOsCallDetails:
|
sl@0
|
511 |
TheOsCallTimeDetailedProfileEnabled = EFalse;
|
sl@0
|
512 |
TheOsCallTimeProfileEnabled = EFalse;
|
sl@0
|
513 |
break;
|
sl@0
|
514 |
case TSqlResourceProfiler::ESqlCounterMemory:
|
sl@0
|
515 |
TheMemProfileEnabled = EFalse;
|
sl@0
|
516 |
break;
|
sl@0
|
517 |
case TSqlResourceProfiler::ESqlCounterMaxAlloc:
|
sl@0
|
518 |
TheMaxAllocProfileEnabled = EFalse;
|
sl@0
|
519 |
break;
|
sl@0
|
520 |
default:
|
sl@0
|
521 |
return KErrNotSupported;
|
sl@0
|
522 |
}
|
sl@0
|
523 |
return KErrNone;
|
sl@0
|
524 |
}
|
sl@0
|
525 |
|
sl@0
|
526 |
TInt sqlite3SymbianProfilerReset(TInt aCounterType)
|
sl@0
|
527 |
{
|
sl@0
|
528 |
const TSqlResourceProfiler::TSqlCounter KCounterType = static_cast <TSqlResourceProfiler::TSqlCounter> (aCounterType);
|
sl@0
|
529 |
switch(KCounterType)
|
sl@0
|
530 |
{
|
sl@0
|
531 |
case TSqlResourceProfiler::ESqlCounterFileIO:
|
sl@0
|
532 |
Mem::FillZ(TheFileOpCounter, sizeof(TheFileOpCounter));
|
sl@0
|
533 |
TheFileWriteAmount = TheFileReadAmount = 0;
|
sl@0
|
534 |
break;
|
sl@0
|
535 |
case TSqlResourceProfiler::ESqlCounterOsCall:
|
sl@0
|
536 |
Mem::FillZ(TheOsOpCounter, sizeof(TheOsOpCounter));
|
sl@0
|
537 |
break;
|
sl@0
|
538 |
case TSqlResourceProfiler::ESqlCounterOsCallTime:
|
sl@0
|
539 |
case TSqlResourceProfiler::ESqlCounterOsCallDetails:
|
sl@0
|
540 |
TheOpCounter = 0;
|
sl@0
|
541 |
Mem::FillZ(TheOsCallTicks, sizeof(TheOsCallTicks));
|
sl@0
|
542 |
for(TInt i=0;i<EOsOpLast;++i)
|
sl@0
|
543 |
{
|
sl@0
|
544 |
TheOsCallMProfile[i].Zero();
|
sl@0
|
545 |
TheOsCallJProfile[i].Zero();
|
sl@0
|
546 |
}
|
sl@0
|
547 |
break;
|
sl@0
|
548 |
case TSqlResourceProfiler::ESqlCounterMemory:
|
sl@0
|
549 |
Mem::FillZ(TheMemOpCounter, sizeof(TheMemOpCounter));
|
sl@0
|
550 |
Mem::FillZ(TheMemOpTicks, sizeof(TheMemOpTicks));
|
sl@0
|
551 |
TheAllocated = TheFreed = 0;
|
sl@0
|
552 |
break;
|
sl@0
|
553 |
case TSqlResourceProfiler::ESqlCounterMaxAlloc:
|
sl@0
|
554 |
TheAllocMax = 0;
|
sl@0
|
555 |
break;
|
sl@0
|
556 |
default:
|
sl@0
|
557 |
return KErrNotSupported;
|
sl@0
|
558 |
}
|
sl@0
|
559 |
return KErrNone;
|
sl@0
|
560 |
}
|
sl@0
|
561 |
|
sl@0
|
562 |
TInt sqlite3SymbianProfilerQuery(TInt aCounterType, TDes8& aResult)
|
sl@0
|
563 |
{
|
sl@0
|
564 |
const TSqlResourceProfiler::TSqlCounter KCounterType = static_cast <TSqlResourceProfiler::TSqlCounter> (aCounterType);
|
sl@0
|
565 |
switch(KCounterType)
|
sl@0
|
566 |
{
|
sl@0
|
567 |
case TSqlResourceProfiler::ESqlCounterFileIO:
|
sl@0
|
568 |
for(TInt i=0;i<EFsOpLast;++i)
|
sl@0
|
569 |
{
|
sl@0
|
570 |
aResult.AppendNum(TheFileOpCounter[i]);
|
sl@0
|
571 |
aResult.Append(TChar(';'));
|
sl@0
|
572 |
}
|
sl@0
|
573 |
aResult.AppendNum(TheFileWriteAmount);
|
sl@0
|
574 |
aResult.Append(TChar(';'));
|
sl@0
|
575 |
aResult.AppendNum(TheFileReadAmount);
|
sl@0
|
576 |
aResult.Append(TChar(';'));
|
sl@0
|
577 |
break;
|
sl@0
|
578 |
case TSqlResourceProfiler::ESqlCounterOsCall:
|
sl@0
|
579 |
for(TInt i=0;i<EOsOpLast;++i)
|
sl@0
|
580 |
{
|
sl@0
|
581 |
aResult.AppendNum(TheOsOpCounter[i]);
|
sl@0
|
582 |
aResult.Append(TChar(';'));
|
sl@0
|
583 |
}
|
sl@0
|
584 |
break;
|
sl@0
|
585 |
case TSqlResourceProfiler::ESqlCounterOsCallTime:
|
sl@0
|
586 |
case TSqlResourceProfiler::ESqlCounterOsCallDetails:
|
sl@0
|
587 |
{
|
sl@0
|
588 |
TInt ticksPerSec = FastCounterFrequency();
|
sl@0
|
589 |
for(TInt i=0;i<EOsOpLast;++i)
|
sl@0
|
590 |
{
|
sl@0
|
591 |
TInt64 osCallTimeUs = (TheOsCallTicks[i] * KMicroSecIn1Sec) / ticksPerSec;
|
sl@0
|
592 |
aResult.AppendNum(osCallTimeUs);
|
sl@0
|
593 |
aResult.Append(TChar(';'));
|
sl@0
|
594 |
}
|
sl@0
|
595 |
}
|
sl@0
|
596 |
break;
|
sl@0
|
597 |
case TSqlResourceProfiler::ESqlCounterMemory:
|
sl@0
|
598 |
aResult.AppendNum(TheMemOpCounter[EMemOpAlloc]);
|
sl@0
|
599 |
aResult.Append(TChar(';'));
|
sl@0
|
600 |
aResult.AppendNum(TheMemOpCounter[EMemOpRealloc]);
|
sl@0
|
601 |
aResult.Append(TChar(';'));
|
sl@0
|
602 |
aResult.AppendNum(TheMemOpCounter[EMemOpFree]);
|
sl@0
|
603 |
aResult.Append(TChar(';'));
|
sl@0
|
604 |
aResult.AppendNum(TheAllocated);
|
sl@0
|
605 |
aResult.Append(TChar(';'));
|
sl@0
|
606 |
aResult.AppendNum(TheFreed);
|
sl@0
|
607 |
aResult.Append(TChar(';'));
|
sl@0
|
608 |
{
|
sl@0
|
609 |
TInt ticksPerSec = FastCounterFrequency();
|
sl@0
|
610 |
TInt64 memOpCallTimeUs = (TheMemOpTicks[EMemOpAlloc] * KMicroSecIn1Sec) / ticksPerSec;
|
sl@0
|
611 |
aResult.AppendNum(memOpCallTimeUs);
|
sl@0
|
612 |
aResult.Append(TChar(';'));
|
sl@0
|
613 |
memOpCallTimeUs = (TheMemOpTicks[EMemOpRealloc] * KMicroSecIn1Sec) / ticksPerSec;
|
sl@0
|
614 |
aResult.AppendNum(memOpCallTimeUs);
|
sl@0
|
615 |
aResult.Append(TChar(';'));
|
sl@0
|
616 |
memOpCallTimeUs = (TheMemOpTicks[EMemOpFree] * KMicroSecIn1Sec) / ticksPerSec;
|
sl@0
|
617 |
aResult.AppendNum(memOpCallTimeUs);
|
sl@0
|
618 |
aResult.Append(TChar(';'));
|
sl@0
|
619 |
}
|
sl@0
|
620 |
break;
|
sl@0
|
621 |
case TSqlResourceProfiler::ESqlCounterMaxAlloc:
|
sl@0
|
622 |
aResult.AppendNum(TheAllocMax);
|
sl@0
|
623 |
aResult.Append(TChar(';'));
|
sl@0
|
624 |
break;
|
sl@0
|
625 |
default:
|
sl@0
|
626 |
return KErrNotSupported;
|
sl@0
|
627 |
}
|
sl@0
|
628 |
return KErrNone;
|
sl@0
|
629 |
}
|
sl@0
|
630 |
|
sl@0
|
631 |
#else//_SQLPROFILER
|
sl@0
|
632 |
|
sl@0
|
633 |
TInt sqlite3SymbianProfilerStart(TInt)
|
sl@0
|
634 |
{
|
sl@0
|
635 |
return KErrNotSupported;
|
sl@0
|
636 |
}
|
sl@0
|
637 |
|
sl@0
|
638 |
TInt sqlite3SymbianProfilerStop(TInt)
|
sl@0
|
639 |
{
|
sl@0
|
640 |
return KErrNotSupported;
|
sl@0
|
641 |
}
|
sl@0
|
642 |
|
sl@0
|
643 |
TInt sqlite3SymbianProfilerReset(TInt)
|
sl@0
|
644 |
{
|
sl@0
|
645 |
return KErrNotSupported;
|
sl@0
|
646 |
}
|
sl@0
|
647 |
|
sl@0
|
648 |
TInt sqlite3SymbianProfilerQuery(TInt, TDes8&)
|
sl@0
|
649 |
{
|
sl@0
|
650 |
return KErrNotSupported;
|
sl@0
|
651 |
}
|
sl@0
|
652 |
|
sl@0
|
653 |
#endif//_SQLPROFILER
|
sl@0
|
654 |
|
sl@0
|
655 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
656 |
|
sl@0
|
657 |
/**
|
sl@0
|
658 |
Os2SqliteErr() is called at the end of many of the interface functions of the OS porting layer (wherever it is appropriate -
|
sl@0
|
659 |
TFileIo and TVfs interfaces). The purpose of this function is to identify the "out of memory" and "disk is full" errors
|
sl@0
|
660 |
reported by the used Symbian OS APIs (aOsErr parameter) and report them to SQLite as SQLITE_FULL and SQLITE_NOMEM errors.
|
sl@0
|
661 |
The KErrEof error (TFileIo::Read() can return KErrEof) is reported to SQLite as SQLITE_IOERR_SHORT_READ. The rest of failures
|
sl@0
|
662 |
are reported as the error specified in aDefaultErr parameter.
|
sl@0
|
663 |
|
sl@0
|
664 |
@param aOsErr Symbian OS error
|
sl@0
|
665 |
@param aDefaultErr The default SQLite error that should be used if the aOsErr parameter is not one of:
|
sl@0
|
666 |
KErrNone, KErrEof, KErrNoMemory, KErrDiskFull
|
sl@0
|
667 |
@return SQLITE_OK, The OS porting layer function call has completed successfully,
|
sl@0
|
668 |
SQLITE_IOERR_SHORT_READ, The amount of the data read is less than the requested amount,
|
sl@0
|
669 |
SQLITE_IOERR_NOMEM, Out of memory,
|
sl@0
|
670 |
SQLITE_FULL, The disk is full,
|
sl@0
|
671 |
aDefaultErr, The rest of failures will be reported as aDefaultErr.
|
sl@0
|
672 |
*/
|
sl@0
|
673 |
static TInt Os2SqliteErr(TInt aOsErr, TInt aDefaultErr)
|
sl@0
|
674 |
{
|
sl@0
|
675 |
switch(aOsErr)
|
sl@0
|
676 |
{
|
sl@0
|
677 |
case KErrNone:
|
sl@0
|
678 |
return SQLITE_OK;
|
sl@0
|
679 |
case KErrEof:
|
sl@0
|
680 |
return SQLITE_IOERR_SHORT_READ;
|
sl@0
|
681 |
case KErrNoMemory:
|
sl@0
|
682 |
return SQLITE_IOERR_NOMEM;
|
sl@0
|
683 |
case KErrDiskFull:
|
sl@0
|
684 |
return SQLITE_FULL;
|
sl@0
|
685 |
default:
|
sl@0
|
686 |
#ifdef _DEBUG
|
sl@0
|
687 |
RDebug::Print(_L("SQLite OS porting layer, Os2SqliteErr(), err=%d\n"), aOsErr);
|
sl@0
|
688 |
SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, OS2SQLITEERR, "OS;0;Os2SqliteErr;aOsErr=%d", aOsErr));
|
sl@0
|
689 |
#endif
|
sl@0
|
690 |
break;
|
sl@0
|
691 |
}
|
sl@0
|
692 |
return aDefaultErr;
|
sl@0
|
693 |
}
|
sl@0
|
694 |
|
sl@0
|
695 |
|
sl@0
|
696 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
697 |
///////////////////// COsLayerData class declaration //////////////////////////////////////////////////////////////////
|
sl@0
|
698 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
699 |
|
sl@0
|
700 |
/**
|
sl@0
|
701 |
COsLayerData singleton class manages a single SQLite OS layer copy of the following data:
|
sl@0
|
702 |
@code
|
sl@0
|
703 |
- file session instance;
|
sl@0
|
704 |
- process's private data path, where the temporary file will be stored (on the system drive);
|
sl@0
|
705 |
- the last OS error code, every Symbian OS API call (file system calls, etc.) will set it;
|
sl@0
|
706 |
- the stored OS error code, initialized with the last OS error code only if stored OS error code is KErrNone.
|
sl@0
|
707 |
Each StoredOsErrorCode() call will reset it to KErrNone;
|
sl@0
|
708 |
The reason of having two data members for storing the OS error codes is that if there is just one variable
|
sl@0
|
709 |
and it gets initialized with the error value reported by a failed OS API call, the next successful OS API
|
sl@0
|
710 |
call will reset it and the COsLayerData client will miss the last "real" OS API error.
|
sl@0
|
711 |
- A pointer to the current RMessage2 object, if the current operation is "open a private secure database";
|
sl@0
|
712 |
- A boolean flag - iReadOnly - used only for private secure databases, indicating whether the database is read-only or not;
|
sl@0
|
713 |
@endcode
|
sl@0
|
714 |
|
sl@0
|
715 |
@internalComponent
|
sl@0
|
716 |
*/
|
sl@0
|
717 |
NONSHARABLE_CLASS(COsLayerData)
|
sl@0
|
718 |
{
|
sl@0
|
719 |
public:
|
sl@0
|
720 |
static TInt Create();
|
sl@0
|
721 |
static inline void Destroy();
|
sl@0
|
722 |
static inline COsLayerData& Instance();
|
sl@0
|
723 |
|
sl@0
|
724 |
inline TInt SetOsErrorCode(TInt aError);
|
sl@0
|
725 |
inline TInt StoredOsErrorCode();
|
sl@0
|
726 |
|
sl@0
|
727 |
inline void StoreFhData(const RMessage2* aMsg, TBool aReadOnly);
|
sl@0
|
728 |
inline void RetrieveAndResetFhData(const RMessage2*& aMsg, TBool& aReadOnly);
|
sl@0
|
729 |
|
sl@0
|
730 |
private:
|
sl@0
|
731 |
inline COsLayerData();
|
sl@0
|
732 |
inline ~COsLayerData();
|
sl@0
|
733 |
TInt DoCreate();
|
sl@0
|
734 |
|
sl@0
|
735 |
public:
|
sl@0
|
736 |
RFs iFs; //File session instance.
|
sl@0
|
737 |
TFileName iSysPrivDir;//"<system drive>:\" + process's private data path. Initialized in sqlite3SymbianFsOpen().
|
sl@0
|
738 |
TInt64 iSeed;
|
sl@0
|
739 |
|
sl@0
|
740 |
private:
|
sl@0
|
741 |
static COsLayerData* iOsLayerData;
|
sl@0
|
742 |
TInt iStoredOsErrorCode; //Contains the last OS error code.
|
sl@0
|
743 |
const RMessage2* iMessage; //Fh data
|
sl@0
|
744 |
TBool iReadOnly; //Fh data
|
sl@0
|
745 |
};
|
sl@0
|
746 |
|
sl@0
|
747 |
/**
|
sl@0
|
748 |
This functon returns a reference to the current thread allocator object.
|
sl@0
|
749 |
The static RAllocator& variable will be initialized once at the moment when the function is called for
|
sl@0
|
750 |
first time.
|
sl@0
|
751 |
*/
|
sl@0
|
752 |
static RAllocator& Allocator()
|
sl@0
|
753 |
{
|
sl@0
|
754 |
static RAllocator& allocator = User::Allocator();
|
sl@0
|
755 |
return allocator;
|
sl@0
|
756 |
}
|
sl@0
|
757 |
|
sl@0
|
758 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
759 |
/////////////////////////////////// TDbFile struct declaration /////////////////////////////////////////////////////
|
sl@0
|
760 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
761 |
|
sl@0
|
762 |
/**
|
sl@0
|
763 |
TDbFile derives from the sqlite3_file structure, adding data members needed for processing the SQLite requests to the OS layer.
|
sl@0
|
764 |
When SQLite needs an access to a file, SQLite allocates memory for a new TDbFile instance and passes a pointer to that
|
sl@0
|
765 |
instance to TVfs::Open(). TVfs::Open() creates/opens the file and initializes the TDbFile instance.
|
sl@0
|
766 |
SQLite uses the initialized TDbFile instance (actually SQLite knows and uses the sqlite3_file, the base structure)
|
sl@0
|
767 |
every time when needs to read or write from/to the file, using for that an appropriate TFileIo method.
|
sl@0
|
768 |
|
sl@0
|
769 |
Note: currently RFileBuf64 object is used instead of RFile64. That improves the read/write file performance.
|
sl@0
|
770 |
|
sl@0
|
771 |
No virtual methods here! sqlite3_file contains data members. If a virtual method is added, that will shift the offset of the
|
sl@0
|
772 |
data members from the beginning of the sqlite3_file object by 4 bytes. This is not what SQLite (C code) expects.
|
sl@0
|
773 |
|
sl@0
|
774 |
@internalComponent
|
sl@0
|
775 |
|
sl@0
|
776 |
@see TVfs
|
sl@0
|
777 |
@see TFileIo
|
sl@0
|
778 |
@see TVfs::Open()
|
sl@0
|
779 |
*/
|
sl@0
|
780 |
NONSHARABLE_STRUCT(TDbFile) : public sqlite3_file
|
sl@0
|
781 |
{
|
sl@0
|
782 |
inline TDbFile();
|
sl@0
|
783 |
RFileBuf64 iFileBuf;
|
sl@0
|
784 |
TInt iLockType; //File lock type
|
sl@0
|
785 |
TBool iReadOnly; //True if the file is read-only
|
sl@0
|
786 |
TInt iSectorSize; //Media sector-size
|
sl@0
|
787 |
TInt iDeviceCharacteristics;
|
sl@0
|
788 |
TSqlFreePageCallback iFreePageCallback;
|
sl@0
|
789 |
TBool iIsFileCreated; //If the file to be created is a temp file,
|
sl@0
|
790 |
//it will not be created until the data is to be written to.
|
sl@0
|
791 |
#ifdef _SQLPROFILER
|
sl@0
|
792 |
TBool iIsJournal;
|
sl@0
|
793 |
#endif
|
sl@0
|
794 |
};
|
sl@0
|
795 |
|
sl@0
|
796 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
797 |
/////////////////////////////////// TFileIo class declaration //////////////////////////////////////////////////////
|
sl@0
|
798 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
799 |
|
sl@0
|
800 |
/**
|
sl@0
|
801 |
TFileIo class offers static methods for performing operations on a file.
|
sl@0
|
802 |
Every TFileIo method has a pointer to a sqlite3_file instance (so, a TDbFile instance) as its first argument.
|
sl@0
|
803 |
|
sl@0
|
804 |
SQLite never accesses the file system directly, only through function pointers, data members of the sqlite3_io_methods structure.
|
sl@0
|
805 |
The OS porting layer defines a single instance of sqlite3_io_methods structure, TheFileIoApi, and uses the TFileIo to initialize the
|
sl@0
|
806 |
sqlite3_io_methods data members (function pointers).
|
sl@0
|
807 |
Every time when SQLite creates/opens a file using TVfs::Open(), TVfs::Open() will pass back to SQLite a pointer to the single
|
sl@0
|
808 |
initialized sqlite3_io_methods instance (TheFileIoApi) that will be used later by SQLite for accessing the file.
|
sl@0
|
809 |
|
sl@0
|
810 |
@internalComponent
|
sl@0
|
811 |
|
sl@0
|
812 |
@see TVfs
|
sl@0
|
813 |
@see TVfs::Open()
|
sl@0
|
814 |
@see TheFileIoApi
|
sl@0
|
815 |
@see TDbFile
|
sl@0
|
816 |
*/
|
sl@0
|
817 |
NONSHARABLE_CLASS(TFileIo)
|
sl@0
|
818 |
{
|
sl@0
|
819 |
public:
|
sl@0
|
820 |
static int Close(sqlite3_file* aDbFile);
|
sl@0
|
821 |
static int Read(sqlite3_file* aDbFile, void* aBuf, int aAmt, sqlite3_int64 aOffset);
|
sl@0
|
822 |
static int Write(sqlite3_file* aDbFile, const void* aData, int aAmt, sqlite3_int64 aOffset);
|
sl@0
|
823 |
static int Truncate(sqlite3_file* aDbFile, sqlite3_int64 aLength);
|
sl@0
|
824 |
static int Sync(sqlite3_file* aDbFile, int aFlags);
|
sl@0
|
825 |
static int FileSize(sqlite3_file* aDbFile, sqlite3_int64* aSize);
|
sl@0
|
826 |
static int Lock(sqlite3_file* aDbFile, int aLockType);
|
sl@0
|
827 |
static int Unlock(sqlite3_file* aDbFile, int aLockType);
|
sl@0
|
828 |
static int CheckReservedLock(sqlite3_file* aDbFile, int *aResOut);
|
sl@0
|
829 |
static int FileControl(sqlite3_file* aDbFile, int aOp, void* aArg);
|
sl@0
|
830 |
static int SectorSize(sqlite3_file* aDbFile);
|
sl@0
|
831 |
static int DeviceCharacteristics(sqlite3_file* aDbFile);
|
sl@0
|
832 |
private:
|
sl@0
|
833 |
static TInt DoCreateTempFile(TDbFile& aDbFile);
|
sl@0
|
834 |
};
|
sl@0
|
835 |
|
sl@0
|
836 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
837 |
/////////////////////////////////// TVfs class declaration /////////////////////////////////////////////////////////
|
sl@0
|
838 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
839 |
|
sl@0
|
840 |
/**
|
sl@0
|
841 |
TVfs ("VFS" - virtual file system) class offers methods for creating/openning a file, deleting a file,
|
sl@0
|
842 |
a "sleep" method, a "time" method, a "rand" method, etc.
|
sl@0
|
843 |
SQLite never accesses the OS API directly, only through the API offered by TVfs and TFileIo classes.
|
sl@0
|
844 |
|
sl@0
|
845 |
@internalComponent
|
sl@0
|
846 |
|
sl@0
|
847 |
@see TFileIo
|
sl@0
|
848 |
@see TheVfsApi
|
sl@0
|
849 |
@see COsLayerData
|
sl@0
|
850 |
*/
|
sl@0
|
851 |
NONSHARABLE_CLASS(TVfs)
|
sl@0
|
852 |
{
|
sl@0
|
853 |
public:
|
sl@0
|
854 |
static int Open(sqlite3_vfs* aVfs, const char* aFileName, sqlite3_file* aDbFile, int aFlags, int* aOutFlags);
|
sl@0
|
855 |
static int Delete(sqlite3_vfs* aVfs, const char* aFileName, int aSyncDir);
|
sl@0
|
856 |
static int Access(sqlite3_vfs* aVfs, const char* aFileName, int aFlags, int* aResOut);
|
sl@0
|
857 |
static int FullPathName(sqlite3_vfs* aVfs, const char* aRelative, int aBufLen, char* aBuf);
|
sl@0
|
858 |
static int Randomness(sqlite3_vfs* aVfs, int aBufLen, char* aBuf);
|
sl@0
|
859 |
static int Sleep(sqlite3_vfs* aVfs, int aMicrosec);
|
sl@0
|
860 |
static int CurrentTime(sqlite3_vfs* aVfs, double* aNow);
|
sl@0
|
861 |
static int GetLastError(sqlite3_vfs *sVfs, int aBufLen, char* aBuf);
|
sl@0
|
862 |
static TInt DoGetDeviceCharacteristicsAndSectorSize(TDbFile& aDbFile, TInt& aRecReadBufSize);
|
sl@0
|
863 |
private:
|
sl@0
|
864 |
static TInt DoOpenFromHandle(TDbFile& aDbFile, const RMessage2& aMsg, TBool aReadOnly);
|
sl@0
|
865 |
static inline TInt DoGetVolumeIoParamInfo(RFs& aFs, TInt aDriveNo, TVolumeIOParamInfo& aVolumeInfo);
|
sl@0
|
866 |
static TInt DoGetDeviceCharacteristics(const TDriveInfo& aDriveInfo, const TVolumeIOParamInfo& aVolumeInfo);
|
sl@0
|
867 |
static TInt DoGetSectorSize(const TDriveInfo& aDriveInfo, const TVolumeIOParamInfo& aVolumeInfo);
|
sl@0
|
868 |
static TInt DoFileSizeCorruptionCheck(TDbFile& aDbFile, const TDesC& aFname, TInt aFmode);
|
sl@0
|
869 |
};
|
sl@0
|
870 |
|
sl@0
|
871 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
872 |
///////////////////// Global variables, constants ////////////////////////////////////////////////////////////////////
|
sl@0
|
873 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
874 |
|
sl@0
|
875 |
// The following macro specifies the size of the RFileBuf64 file buffer in KB:
|
sl@0
|
876 |
// __SQLITE_OS_SYMBIAN_FILEBUF_KBSIZE__
|
sl@0
|
877 |
//
|
sl@0
|
878 |
// If not set, a default value of 8 is used
|
sl@0
|
879 |
#if !defined(__SQLITE_OS_SYMBIAN_FILEBUF_KBSIZE__)
|
sl@0
|
880 |
#define __SQLITE_OS_SYMBIAN_FILEBUF_KBSIZE__ 8
|
sl@0
|
881 |
#endif
|
sl@0
|
882 |
const TInt KFileBufSize = __SQLITE_OS_SYMBIAN_FILEBUF_KBSIZE__ * 1024;
|
sl@0
|
883 |
|
sl@0
|
884 |
/**
|
sl@0
|
885 |
Pointer to the single COsLayerData instance.
|
sl@0
|
886 |
|
sl@0
|
887 |
@see COsLayerData
|
sl@0
|
888 |
|
sl@0
|
889 |
@internalComponent
|
sl@0
|
890 |
*/
|
sl@0
|
891 |
COsLayerData* COsLayerData::iOsLayerData = NULL;
|
sl@0
|
892 |
|
sl@0
|
893 |
/**
|
sl@0
|
894 |
Single sqlite3_io_methods instance, which data members (function pointers) are initialized with the addresses of
|
sl@0
|
895 |
TFileIo members.
|
sl@0
|
896 |
TheFileIoApi is used by SQLite for performing OS independent file I/O.
|
sl@0
|
897 |
|
sl@0
|
898 |
@see TFileIo
|
sl@0
|
899 |
@see TVfs
|
sl@0
|
900 |
|
sl@0
|
901 |
@internalComponent
|
sl@0
|
902 |
*/
|
sl@0
|
903 |
static sqlite3_io_methods TheFileIoApi =
|
sl@0
|
904 |
{
|
sl@0
|
905 |
1, //Version
|
sl@0
|
906 |
&TFileIo::Close,
|
sl@0
|
907 |
&TFileIo::Read,
|
sl@0
|
908 |
&TFileIo::Write,
|
sl@0
|
909 |
&TFileIo::Truncate,
|
sl@0
|
910 |
&TFileIo::Sync,
|
sl@0
|
911 |
&TFileIo::FileSize,
|
sl@0
|
912 |
&TFileIo::Lock,
|
sl@0
|
913 |
&TFileIo::Unlock,
|
sl@0
|
914 |
&TFileIo::CheckReservedLock,
|
sl@0
|
915 |
&TFileIo::FileControl,
|
sl@0
|
916 |
&TFileIo::SectorSize,
|
sl@0
|
917 |
&TFileIo::DeviceCharacteristics
|
sl@0
|
918 |
};
|
sl@0
|
919 |
|
sl@0
|
920 |
/**
|
sl@0
|
921 |
Single sqlite3_vfs instance, which data members (function pointers) are initialized with the addresses of
|
sl@0
|
922 |
TVfs members. TheVfsApi also keeps information regarding some other OS dependend characteristics like
|
sl@0
|
923 |
the TDbFile size and max file name length.
|
sl@0
|
924 |
TheVfsApi is used by SQLite for accessing needed OS API.
|
sl@0
|
925 |
|
sl@0
|
926 |
TheVfsApi can't be a constant definition. SQLite expects the "sqlite3_vfs" object to be a R/W one, because
|
sl@0
|
927 |
SQLite may have and use a chain of sqlite3_vfs instances.
|
sl@0
|
928 |
|
sl@0
|
929 |
@see TVfs
|
sl@0
|
930 |
@see TTFileIo
|
sl@0
|
931 |
@see TDbFile
|
sl@0
|
932 |
|
sl@0
|
933 |
@internalComponent
|
sl@0
|
934 |
*/
|
sl@0
|
935 |
static sqlite3_vfs TheVfsApi =
|
sl@0
|
936 |
{
|
sl@0
|
937 |
1, //iVersion
|
sl@0
|
938 |
sizeof(TDbFile), //szOsFile
|
sl@0
|
939 |
KMaxFileName, //mxPathname
|
sl@0
|
940 |
0, //pNext
|
sl@0
|
941 |
"SymbianSql", //zName
|
sl@0
|
942 |
0, //pAppData
|
sl@0
|
943 |
&TVfs::Open,
|
sl@0
|
944 |
&TVfs::Delete,
|
sl@0
|
945 |
&TVfs::Access,
|
sl@0
|
946 |
&TVfs::FullPathName,
|
sl@0
|
947 |
0,
|
sl@0
|
948 |
0,
|
sl@0
|
949 |
0,
|
sl@0
|
950 |
0,
|
sl@0
|
951 |
&TVfs::Randomness,
|
sl@0
|
952 |
&TVfs::Sleep,
|
sl@0
|
953 |
&TVfs::CurrentTime,
|
sl@0
|
954 |
&TVfs::GetLastError
|
sl@0
|
955 |
};
|
sl@0
|
956 |
|
sl@0
|
957 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
958 |
///////////////////// COsLayerData class definition //////////////////////////////////////////////////////////////////
|
sl@0
|
959 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
960 |
|
sl@0
|
961 |
/**
|
sl@0
|
962 |
Creates a single COsLayerData instance.
|
sl@0
|
963 |
|
sl@0
|
964 |
@return KErrNone, The operation has completed succesfully;
|
sl@0
|
965 |
KErrNoMemory, Out of memory condition has occured;
|
sl@0
|
966 |
Note that other system-wide error codes may also be returned.
|
sl@0
|
967 |
|
sl@0
|
968 |
@panic Sqlite 16 In _DEBUG mode - the COsLayerData instance has been created already.
|
sl@0
|
969 |
*/
|
sl@0
|
970 |
/* static */ TInt COsLayerData::Create()
|
sl@0
|
971 |
{
|
sl@0
|
972 |
SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, COSLAYERDATA_CREATE_ENTRY, "OS-Entry;0;COsLayerData::Create"));
|
sl@0
|
973 |
__ASSERT_DEBUG(!COsLayerData::iOsLayerData, __SQLITEPANIC2(ESqliteOsPanicOsLayerDataExists));
|
sl@0
|
974 |
COsLayerData::iOsLayerData = new COsLayerData;
|
sl@0
|
975 |
if(!COsLayerData::iOsLayerData)
|
sl@0
|
976 |
{
|
sl@0
|
977 |
SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, COSLAYERDATA_CREATE_EXIT1, "OS-Exit;0;COsLayerData::Create;err=KErrNoMemory"));
|
sl@0
|
978 |
return KErrNoMemory;
|
sl@0
|
979 |
}
|
sl@0
|
980 |
TInt err = COsLayerData::iOsLayerData->DoCreate();
|
sl@0
|
981 |
if(err != KErrNone)
|
sl@0
|
982 |
{
|
sl@0
|
983 |
delete COsLayerData::iOsLayerData;
|
sl@0
|
984 |
COsLayerData::iOsLayerData = NULL;
|
sl@0
|
985 |
SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, COSLAYERDATA_CREATE_EXIT2, "OS-Exit;0;COsLayerData::Create;err=%d", err));
|
sl@0
|
986 |
return err;
|
sl@0
|
987 |
}
|
sl@0
|
988 |
SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, COSLAYERDATA_CREATE_EXIT3, "OS-Exit;0;COsLayerData::Create;err=KErrNone"));
|
sl@0
|
989 |
return KErrNone;
|
sl@0
|
990 |
}
|
sl@0
|
991 |
|
sl@0
|
992 |
/**
|
sl@0
|
993 |
Destroys the COsLayerData instance.
|
sl@0
|
994 |
*/
|
sl@0
|
995 |
/* static */ inline void COsLayerData::Destroy()
|
sl@0
|
996 |
{
|
sl@0
|
997 |
delete COsLayerData::iOsLayerData;
|
sl@0
|
998 |
COsLayerData::iOsLayerData = NULL;
|
sl@0
|
999 |
SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, COSLAYERDATA_DESTROY, "OS;0;COsLayerData::Destroy"));
|
sl@0
|
1000 |
}
|
sl@0
|
1001 |
|
sl@0
|
1002 |
/**
|
sl@0
|
1003 |
Returns a reference to the single COsLayerData instance.
|
sl@0
|
1004 |
|
sl@0
|
1005 |
@panic Sqlite 1 In _DEBUG mode if the COsLayerData instance is NULL.
|
sl@0
|
1006 |
*/
|
sl@0
|
1007 |
/* static */ inline COsLayerData& COsLayerData::Instance()
|
sl@0
|
1008 |
{
|
sl@0
|
1009 |
__ASSERT_DEBUG(COsLayerData::iOsLayerData != NULL, __SQLITEPANIC2(ESqliteOsPanicNullOsLayerDataPtr));
|
sl@0
|
1010 |
return *COsLayerData::iOsLayerData;
|
sl@0
|
1011 |
}
|
sl@0
|
1012 |
|
sl@0
|
1013 |
/**
|
sl@0
|
1014 |
Sets the last OS error code data member. The stored OS error code data member will be set only if it is
|
sl@0
|
1015 |
KErrNone. (If it is not KErrNone it means that its value has not been accessed yet)
|
sl@0
|
1016 |
An exception from the rule described above is KErrDiskFull error which, if happens, will be set always, because
|
sl@0
|
1017 |
this error has a special meaning for the database clients - special actions may have to be taken if the
|
sl@0
|
1018 |
disk is full.
|
sl@0
|
1019 |
|
sl@0
|
1020 |
@param aError The OS error code
|
sl@0
|
1021 |
@return The OS error code
|
sl@0
|
1022 |
*/
|
sl@0
|
1023 |
inline TInt COsLayerData::SetOsErrorCode(TInt aError)
|
sl@0
|
1024 |
{
|
sl@0
|
1025 |
if(iStoredOsErrorCode == KErrNone || aError == KErrDiskFull)
|
sl@0
|
1026 |
{
|
sl@0
|
1027 |
iStoredOsErrorCode = aError;
|
sl@0
|
1028 |
}
|
sl@0
|
1029 |
return aError;
|
sl@0
|
1030 |
}
|
sl@0
|
1031 |
|
sl@0
|
1032 |
/**
|
sl@0
|
1033 |
Returns the last stored OS error code, which was stored by SetOsErrorCode() call.
|
sl@0
|
1034 |
The function also resets the stored OS error code to KErrNone.
|
sl@0
|
1035 |
|
sl@0
|
1036 |
@return The last stored OS error code
|
sl@0
|
1037 |
*/
|
sl@0
|
1038 |
inline TInt COsLayerData::StoredOsErrorCode()
|
sl@0
|
1039 |
{
|
sl@0
|
1040 |
TInt err = iStoredOsErrorCode;
|
sl@0
|
1041 |
iStoredOsErrorCode = KErrNone;
|
sl@0
|
1042 |
return err;
|
sl@0
|
1043 |
}
|
sl@0
|
1044 |
|
sl@0
|
1045 |
/**
|
sl@0
|
1046 |
Stores the RMessage2 object address, file and file session handles and the read-only flag for later use when SQLite issues a
|
sl@0
|
1047 |
request for open the database file (private secure database).
|
sl@0
|
1048 |
|
sl@0
|
1049 |
The aMsg argument of the function can be NULL, because this fucntion is also used to reset the stored "file handle" data.
|
sl@0
|
1050 |
|
sl@0
|
1051 |
How this function is used:
|
sl@0
|
1052 |
1) When the SQL server receives a request to establish a connection with private secure database, the SQL server
|
sl@0
|
1053 |
will add additional information to the private secure database file name, such as:
|
sl@0
|
1054 |
- the file handle (the private secure database is opened by the client side dll - sqldb.dll);
|
sl@0
|
1055 |
- a pointer to the RMessage2 object used in this request;
|
sl@0
|
1056 |
2) The passed additional information will be used for adopting the file handle by calling RFile64::AdoptFromClient().
|
sl@0
|
1057 |
3) Before calling TVfs::Open() to establish a connection with the database, SQLite will call TVfs::FullPathName()
|
sl@0
|
1058 |
to retrieve the database file full path
|
sl@0
|
1059 |
4) TVfs::FullPathName() will detect that the file name contains an additional information and will extraxt the information
|
sl@0
|
1060 |
calling COsLayerData::StoreFhData().
|
sl@0
|
1061 |
5) After TVfs::FullPathName() SQLite calls TVfs::Open() where the extracted information will be used for adopting the file handle
|
sl@0
|
1062 |
|
sl@0
|
1063 |
@param aMsg A pointer to the current RMessage2 object
|
sl@0
|
1064 |
@param aReadOnly True if the private secure database is read-only
|
sl@0
|
1065 |
*/
|
sl@0
|
1066 |
inline void COsLayerData::StoreFhData(const RMessage2* aMsg, TBool aReadOnly)
|
sl@0
|
1067 |
{
|
sl@0
|
1068 |
iMessage = aMsg;
|
sl@0
|
1069 |
iReadOnly = aReadOnly;
|
sl@0
|
1070 |
}
|
sl@0
|
1071 |
|
sl@0
|
1072 |
/**
|
sl@0
|
1073 |
Retrieves the RMessage2 object, file and file session handles. The stored data will be reset.
|
sl@0
|
1074 |
This function is used by TVfs::Open(), when a request for opening a secure private database is processed.
|
sl@0
|
1075 |
|
sl@0
|
1076 |
@param aMsg Output parameter. A reference to a RMessage2 pointer, which will be initialized with the stored RMessage2 pointer.
|
sl@0
|
1077 |
@param aReadOnly Output parameter. The store read-only flag value will be set there.
|
sl@0
|
1078 |
|
sl@0
|
1079 |
@panic Sqlite 13 In _DEBUG mode - aMsg is NULL.
|
sl@0
|
1080 |
*/
|
sl@0
|
1081 |
inline void COsLayerData::RetrieveAndResetFhData(const RMessage2*& aMsg, TBool& aReadOnly)
|
sl@0
|
1082 |
{
|
sl@0
|
1083 |
__ASSERT_DEBUG(iMessage != NULL, __SQLITEPANIC(ESqliteOsPanicInvalidFhData));
|
sl@0
|
1084 |
aMsg = iMessage;
|
sl@0
|
1085 |
aReadOnly = iReadOnly;
|
sl@0
|
1086 |
iMessage = NULL;
|
sl@0
|
1087 |
}
|
sl@0
|
1088 |
|
sl@0
|
1089 |
/**
|
sl@0
|
1090 |
Initializes the COsLayerData data members with their default values.
|
sl@0
|
1091 |
*/
|
sl@0
|
1092 |
inline COsLayerData::COsLayerData() :
|
sl@0
|
1093 |
iStoredOsErrorCode(KErrNone),
|
sl@0
|
1094 |
iMessage(0),
|
sl@0
|
1095 |
iReadOnly(EFalse)
|
sl@0
|
1096 |
{
|
sl@0
|
1097 |
TTime now;
|
sl@0
|
1098 |
now.UniversalTime();
|
sl@0
|
1099 |
iSeed = now.Int64();
|
sl@0
|
1100 |
}
|
sl@0
|
1101 |
|
sl@0
|
1102 |
/**
|
sl@0
|
1103 |
Destroys the COsLayerData instance.
|
sl@0
|
1104 |
|
sl@0
|
1105 |
Note: No SQLite functions should be called inside the destructor, because SQLite is already shutdown-ed!
|
sl@0
|
1106 |
*/
|
sl@0
|
1107 |
inline COsLayerData::~COsLayerData()
|
sl@0
|
1108 |
{
|
sl@0
|
1109 |
SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, COSLAYERDATA_COSLAYERDATA2, "OS;0;COsLayerData::~COsLayerData;iFs.Handle()=0x%X", (TUint)iFs.Handle()));
|
sl@0
|
1110 |
__FS_CALL(EFsOpFsClose, 0);
|
sl@0
|
1111 |
iFs.Close();
|
sl@0
|
1112 |
}
|
sl@0
|
1113 |
|
sl@0
|
1114 |
/**
|
sl@0
|
1115 |
Creates a file session instance.
|
sl@0
|
1116 |
|
sl@0
|
1117 |
Creates the private path, where the temporary files will be stored (on the system drive).
|
sl@0
|
1118 |
|
sl@0
|
1119 |
In a case of a failure COsLayerData::DoCreate() does not close the file session.
|
sl@0
|
1120 |
This will be made in the calling function - COsLayerData::Create().
|
sl@0
|
1121 |
|
sl@0
|
1122 |
Note: No SQLite functions should be called inside the DoCreate() implementation, because SQLite is not initialized yet!
|
sl@0
|
1123 |
|
sl@0
|
1124 |
@return KErrNone, The operation has completed succesfully;
|
sl@0
|
1125 |
KErrGeneral The registration of TheVfsApi has failed;
|
sl@0
|
1126 |
KErrNoMemory, Out of memory condition has occured;
|
sl@0
|
1127 |
Note that other system-wide error codes may also be returned.
|
sl@0
|
1128 |
|
sl@0
|
1129 |
@see TVfs
|
sl@0
|
1130 |
@see TheVfsApi
|
sl@0
|
1131 |
*/
|
sl@0
|
1132 |
TInt COsLayerData::DoCreate()
|
sl@0
|
1133 |
{
|
sl@0
|
1134 |
__FS_CALL(EFsOpFsConnect, 0);
|
sl@0
|
1135 |
TInt err = iFs.Connect();
|
sl@0
|
1136 |
if(err != KErrNone)
|
sl@0
|
1137 |
{
|
sl@0
|
1138 |
return err;
|
sl@0
|
1139 |
}
|
sl@0
|
1140 |
//Get the system drive
|
sl@0
|
1141 |
__FS_CALL(EFsOpFsGetSystemDrive, 0);
|
sl@0
|
1142 |
TInt sysDrive = static_cast<TInt>(RFs::GetSystemDrive());
|
sl@0
|
1143 |
__FS_CALL(EFsOpFsCreatePrivatePath, 0);
|
sl@0
|
1144 |
if((err = iFs.CreatePrivatePath(sysDrive)) != KErrNone && err != KErrAlreadyExists)
|
sl@0
|
1145 |
{
|
sl@0
|
1146 |
return err;
|
sl@0
|
1147 |
}
|
sl@0
|
1148 |
TFileName privateDir;
|
sl@0
|
1149 |
__FS_CALL(EFsOpFsPrivatePath, 0);
|
sl@0
|
1150 |
if((err = iFs.PrivatePath(privateDir)) != KErrNone)
|
sl@0
|
1151 |
{
|
sl@0
|
1152 |
return err;
|
sl@0
|
1153 |
}
|
sl@0
|
1154 |
TDriveUnit drive(sysDrive);
|
sl@0
|
1155 |
TDriveName driveName = drive.Name();
|
sl@0
|
1156 |
TParse parse;
|
sl@0
|
1157 |
(void)parse.Set(driveName, &privateDir, 0);//this call can't fail
|
sl@0
|
1158 |
iSysPrivDir.Copy(parse.DriveAndPath());
|
sl@0
|
1159 |
//Create the temp files directory
|
sl@0
|
1160 |
(void)parse.AddDir(KTempFileDir);//this call can't fail
|
sl@0
|
1161 |
err = iFs.MkDir(parse.DriveAndPath());
|
sl@0
|
1162 |
if(err != KErrNone && err != KErrAlreadyExists)
|
sl@0
|
1163 |
{
|
sl@0
|
1164 |
return err;
|
sl@0
|
1165 |
}
|
sl@0
|
1166 |
SQLITE_TRACE_OS(OstTraceExt3(TRACE_INTERNALS, COSLAYERDATA_DOCREATE, "OS;0;COsLayerData::DoCreate;iFs.Handle()=0x%X;sysDrive=%d;iSysPrivDir=%S", (TUint)iFs.Handle(), sysDrive, __SQLITEPRNSTR(iSysPrivDir)));
|
sl@0
|
1167 |
return KErrNone;
|
sl@0
|
1168 |
}
|
sl@0
|
1169 |
|
sl@0
|
1170 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
1171 |
/////////////////////// Symbian OS specific functions (called by the SQL server) ///////////////////////////////////
|
sl@0
|
1172 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
1173 |
|
sl@0
|
1174 |
/**
|
sl@0
|
1175 |
Returns the last OS error which occured durring the operations with the database files.
|
sl@0
|
1176 |
The per-thread variable, where the last OS error is hold, will be set to KErrNone.
|
sl@0
|
1177 |
|
sl@0
|
1178 |
This function is part of Symbian OS specific SQLITE API.
|
sl@0
|
1179 |
|
sl@0
|
1180 |
@return The last OS error.
|
sl@0
|
1181 |
@internalComponent
|
sl@0
|
1182 |
*/
|
sl@0
|
1183 |
TInt sqlite3SymbianLastOsError(void)
|
sl@0
|
1184 |
{
|
sl@0
|
1185 |
return COsLayerData::Instance().StoredOsErrorCode();
|
sl@0
|
1186 |
}
|
sl@0
|
1187 |
|
sl@0
|
1188 |
/**
|
sl@0
|
1189 |
This function must be called once before any other SQLITE API call.
|
sl@0
|
1190 |
The function:
|
sl@0
|
1191 |
@code
|
sl@0
|
1192 |
- Initializes the OS poting layer;
|
sl@0
|
1193 |
- Initializes the SQLite library;
|
sl@0
|
1194 |
@endcode
|
sl@0
|
1195 |
|
sl@0
|
1196 |
This function is part of the Symbian OS specific SQLITE API.
|
sl@0
|
1197 |
|
sl@0
|
1198 |
@return Symbian OS specific error code, including KErrNoMemory.
|
sl@0
|
1199 |
|
sl@0
|
1200 |
@internalComponent
|
sl@0
|
1201 |
*/
|
sl@0
|
1202 |
TInt sqlite3SymbianLibInit(void)
|
sl@0
|
1203 |
{
|
sl@0
|
1204 |
TInt osErr = COsLayerData::Create();
|
sl@0
|
1205 |
if(osErr != KErrNone)
|
sl@0
|
1206 |
{
|
sl@0
|
1207 |
return osErr;
|
sl@0
|
1208 |
}
|
sl@0
|
1209 |
osErr = KErrNone;
|
sl@0
|
1210 |
TInt sqliteErr = sqlite3_initialize();
|
sl@0
|
1211 |
if(sqliteErr != SQLITE_OK)
|
sl@0
|
1212 |
{
|
sl@0
|
1213 |
osErr = sqliteErr == SQLITE_NOMEM ? KErrNoMemory : KErrGeneral;
|
sl@0
|
1214 |
COsLayerData::Destroy();
|
sl@0
|
1215 |
}
|
sl@0
|
1216 |
SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, SQLITE3SYMBIANLIBINIT, "OS;0;sqlite3SymbianLibInit;osErr=%d", osErr));
|
sl@0
|
1217 |
return osErr;
|
sl@0
|
1218 |
}
|
sl@0
|
1219 |
|
sl@0
|
1220 |
/**
|
sl@0
|
1221 |
This function must be called once after finishing working with sqlite.
|
sl@0
|
1222 |
The function:
|
sl@0
|
1223 |
@code
|
sl@0
|
1224 |
- Shuts down the SQLite library;
|
sl@0
|
1225 |
- Releases the allocated by the OS porting layer resources;
|
sl@0
|
1226 |
@endcode
|
sl@0
|
1227 |
|
sl@0
|
1228 |
This function is part of the Symbian OS specific SQLITE API.
|
sl@0
|
1229 |
|
sl@0
|
1230 |
@internalComponent
|
sl@0
|
1231 |
*/
|
sl@0
|
1232 |
void sqlite3SymbianLibFinalize(void)
|
sl@0
|
1233 |
{
|
sl@0
|
1234 |
(void)sqlite3_shutdown();
|
sl@0
|
1235 |
COsLayerData::Destroy();
|
sl@0
|
1236 |
SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, SQLITE3SYMBIANLIBFINALIZE, "OS;0;sqlite3SymbianLibFinalize"));
|
sl@0
|
1237 |
}
|
sl@0
|
1238 |
|
sl@0
|
1239 |
/**
|
sl@0
|
1240 |
This function is part of Symbian OS specific SQLITE API.
|
sl@0
|
1241 |
|
sl@0
|
1242 |
@return A reference to RFs instance used for sqlite file I/O operations.
|
sl@0
|
1243 |
@internalComponent
|
sl@0
|
1244 |
*/
|
sl@0
|
1245 |
RFs& sqlite3SymbianFs(void)
|
sl@0
|
1246 |
{
|
sl@0
|
1247 |
return COsLayerData::Instance().iFs;
|
sl@0
|
1248 |
}
|
sl@0
|
1249 |
|
sl@0
|
1250 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
1251 |
////////////////// UTF16<-->UTF8, conversion functions ////////////////////////////////////////////////////////////
|
sl@0
|
1252 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
1253 |
|
sl@0
|
1254 |
/**
|
sl@0
|
1255 |
The function converts aFileName to UTF16 encoded file name, and stores the UTF16 encoded file name
|
sl@0
|
1256 |
to the place pointed by aFileNameDestBuf argument.
|
sl@0
|
1257 |
If the UTF16 conversion of the file name failed because the file name is too long or NULL,
|
sl@0
|
1258 |
the function returns EFalse.
|
sl@0
|
1259 |
|
sl@0
|
1260 |
@param aFileName Expected to point to UTF8 encoded, zero terminated string.
|
sl@0
|
1261 |
Max allowed aFileName length is KMaxFileName (excluding terminating 0 character).
|
sl@0
|
1262 |
@param aFileNameDestBuf Output parameter. Will hold UTF16, non-zero-terminated string.
|
sl@0
|
1263 |
The max length must be at least KMaxFileName characters.
|
sl@0
|
1264 |
|
sl@0
|
1265 |
@return True if the conversion has been completed successfully
|
sl@0
|
1266 |
*/
|
sl@0
|
1267 |
static TBool ConvertToUnicode(const char *aFileName, TDes& aFileNameDestBuf)
|
sl@0
|
1268 |
{
|
sl@0
|
1269 |
if(aFileName)
|
sl@0
|
1270 |
{
|
sl@0
|
1271 |
wchar_t* dest = reinterpret_cast <wchar_t*> (const_cast <TUint16*> (aFileNameDestBuf.Ptr()));
|
sl@0
|
1272 |
TInt len = mbstowcs(dest, aFileName, aFileNameDestBuf.MaxLength());
|
sl@0
|
1273 |
__ASSERT_DEBUG(len >= 0, __SQLITEPANIC2(ESqliteOsPanicInternalError));
|
sl@0
|
1274 |
//If len == aFileNameDestBuf.MaxLength(), then the output buffer is too small.
|
sl@0
|
1275 |
if(len < aFileNameDestBuf.MaxLength())
|
sl@0
|
1276 |
{
|
sl@0
|
1277 |
aFileNameDestBuf.SetLength(len);
|
sl@0
|
1278 |
return ETrue;
|
sl@0
|
1279 |
}
|
sl@0
|
1280 |
}
|
sl@0
|
1281 |
return EFalse;
|
sl@0
|
1282 |
}
|
sl@0
|
1283 |
|
sl@0
|
1284 |
/**
|
sl@0
|
1285 |
The function converts aFileName to UTF8 encoded file name, and stores the UTF8 encoded file name
|
sl@0
|
1286 |
to the place pointed by aFileNameDestBuf argument.
|
sl@0
|
1287 |
If the UTF8 conversion of the file name failed because the file name is too long or NULL,
|
sl@0
|
1288 |
the function returns EFalse.
|
sl@0
|
1289 |
|
sl@0
|
1290 |
@param aFileName Expected to point to UTF16 encoded, zero terminated string.
|
sl@0
|
1291 |
Max allowed aFileName length is KMaxFileName (excluding terminating 0 character).
|
sl@0
|
1292 |
@param aFileNameDestBuf Output parameter. Will hold UTF8, non-zero-terminated string.
|
sl@0
|
1293 |
The max length must be at least KMaxFileName characters.
|
sl@0
|
1294 |
|
sl@0
|
1295 |
@return True if the conversion has been completed successfully
|
sl@0
|
1296 |
*/
|
sl@0
|
1297 |
static TBool ConvertFromUnicode(const TDesC& aFileName, TDes8& aFileNameDestBuf)
|
sl@0
|
1298 |
{
|
sl@0
|
1299 |
char* dest = reinterpret_cast <char*> (const_cast <TUint8*> (aFileNameDestBuf.Ptr()));
|
sl@0
|
1300 |
const wchar_t* src = reinterpret_cast <const wchar_t*> (aFileName.Ptr());
|
sl@0
|
1301 |
TInt len = wcstombs(dest, src, aFileNameDestBuf.MaxLength());
|
sl@0
|
1302 |
__ASSERT_DEBUG(len >= 0, __SQLITEPANIC2(ESqliteOsPanicInternalError));
|
sl@0
|
1303 |
//If len == aFileNameDestBuf.MaxLength(), then the output buffer is too small.
|
sl@0
|
1304 |
if(len < aFileNameDestBuf.MaxLength())
|
sl@0
|
1305 |
{
|
sl@0
|
1306 |
aFileNameDestBuf.SetLength(len);
|
sl@0
|
1307 |
return ETrue;
|
sl@0
|
1308 |
}
|
sl@0
|
1309 |
return EFalse;
|
sl@0
|
1310 |
}
|
sl@0
|
1311 |
|
sl@0
|
1312 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
1313 |
////////////////// File name, containing handles, functions ////////////////////////////////////////////////////////
|
sl@0
|
1314 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
1315 |
|
sl@0
|
1316 |
const char KFhSeparator = '|'; //The symbol, which when used in the file name means that the string does not contain a real file name but file handles
|
sl@0
|
1317 |
const TInt KFhSessHandleIdx = 2;//The index of the file session handle in RMessage2 object
|
sl@0
|
1318 |
const TInt KFhFileHandleIdx = 3;//The index of the file handle in RMessage2 object
|
sl@0
|
1319 |
const TInt KFhMarkPos = 0; //if the symbol in this position is KFhSeparator, then the string contains file handles
|
sl@0
|
1320 |
const TInt KFhRoPos = 1; //read-only flag position in the string
|
sl@0
|
1321 |
const TInt KFhMsgAddrPos = 2; //RMessage2 address position in the string
|
sl@0
|
1322 |
const TInt KFhMsgAddrLen = 8; //RMessage2 address length
|
sl@0
|
1323 |
//const TInt KFhDrivePos = 1; //Drive position in the string (after removing the read-only flag and RMessage2 object's address)
|
sl@0
|
1324 |
|
sl@0
|
1325 |
/**
|
sl@0
|
1326 |
File name string types:
|
sl@0
|
1327 |
@code
|
sl@0
|
1328 |
- ENotFhStr - The string does not contain file handles;
|
sl@0
|
1329 |
- EFhStr - The string contain file handles, but is not main db file;
|
sl@0
|
1330 |
- EFhMainDbStr - The string contain file handles and is the main db file;
|
sl@0
|
1331 |
@endcode
|
sl@0
|
1332 |
|
sl@0
|
1333 |
Every file name passed to the OS porting layer's TVfs::Open() method falls into one of the following three categories:
|
sl@0
|
1334 |
ENotFhStr - the file name does not contain handles, EFhStr - the file name contains handles but is not a name of a private
|
sl@0
|
1335 |
secure database, EFhMainDbStr - the file name contains handles and is a name of a private secure database.
|
sl@0
|
1336 |
|
sl@0
|
1337 |
@see TVfs::Open()
|
sl@0
|
1338 |
@see FhStringProps()
|
sl@0
|
1339 |
|
sl@0
|
1340 |
@internalComponent
|
sl@0
|
1341 |
*/
|
sl@0
|
1342 |
enum TFhStrType
|
sl@0
|
1343 |
{
|
sl@0
|
1344 |
ENotFhStr, //The string does not contain file handles
|
sl@0
|
1345 |
EFhStr, //The string contain file handles, but is not main db file
|
sl@0
|
1346 |
EFhMainDbStr //The string contain file handles and is the main db file
|
sl@0
|
1347 |
};
|
sl@0
|
1348 |
|
sl@0
|
1349 |
/**
|
sl@0
|
1350 |
The TVfs::Open() implementation uses this function to determine the type of the file name, which can be
|
sl@0
|
1351 |
one of the TFhStrType enum item values.
|
sl@0
|
1352 |
|
sl@0
|
1353 |
@param aFileName Zero-terminated, UTF8 encoded file name.
|
sl@0
|
1354 |
@return The file name type, one of the TFhStrType enum item values.
|
sl@0
|
1355 |
|
sl@0
|
1356 |
@see TVfs::Open()
|
sl@0
|
1357 |
@see TFhStrType
|
sl@0
|
1358 |
|
sl@0
|
1359 |
@internalComponent
|
sl@0
|
1360 |
*/
|
sl@0
|
1361 |
static TFhStrType FhStringProps(const char* aFileName)
|
sl@0
|
1362 |
{
|
sl@0
|
1363 |
char* first = strchr(aFileName, KFhSeparator);
|
sl@0
|
1364 |
if(!first)
|
sl@0
|
1365 |
{
|
sl@0
|
1366 |
return ENotFhStr;
|
sl@0
|
1367 |
}
|
sl@0
|
1368 |
char* last = strchr(first + 1, KFhSeparator);
|
sl@0
|
1369 |
if(!last)
|
sl@0
|
1370 |
{
|
sl@0
|
1371 |
return ENotFhStr;
|
sl@0
|
1372 |
}
|
sl@0
|
1373 |
return *(last + 1) == 0 ? EFhMainDbStr : EFhStr;
|
sl@0
|
1374 |
}
|
sl@0
|
1375 |
|
sl@0
|
1376 |
/**
|
sl@0
|
1377 |
Removes all invalid characters in aFileName.
|
sl@0
|
1378 |
If the file name contains handles (so that's a private secure database related name), the additional
|
sl@0
|
1379 |
information (handles, flags, object addresses) has to be excluded from the name in order to make it usable
|
sl@0
|
1380 |
by the file system.
|
sl@0
|
1381 |
|
sl@0
|
1382 |
The private file name format is (see FhExtractAndStore() comments):
|
sl@0
|
1383 |
|
sl@0
|
1384 |
"|<R/O flag><RMessage2 pointer><drive><app SID><file_name><file_ext>|"
|
sl@0
|
1385 |
|
sl@0
|
1386 |
Before opening or creating a file, SQLite will call TVfs::FullPathName() passing to the function the name of
|
sl@0
|
1387 |
the file and expecting the full file name (including path) as an output from the function.
|
sl@0
|
1388 |
After the TVfs::FullPathName() call, the full file name, returned to SQLite, will have the following format:
|
sl@0
|
1389 |
|
sl@0
|
1390 |
"|<drive><path><file_name><file_ext>|"
|
sl@0
|
1391 |
|
sl@0
|
1392 |
FhConvertToFileName() is called from TVfs::Open() and will remove the leading and trialing '|' characters from
|
sl@0
|
1393 |
the file name.
|
sl@0
|
1394 |
|
sl@0
|
1395 |
@param aFileName Input/Output parameter. The cleaned file name will be copied there.
|
sl@0
|
1396 |
@param aPrivateDir The SQL server private data cage.
|
sl@0
|
1397 |
|
sl@0
|
1398 |
@see TVfs::Open()
|
sl@0
|
1399 |
@see TFhStrType
|
sl@0
|
1400 |
@see FhStringProps()
|
sl@0
|
1401 |
|
sl@0
|
1402 |
@internalComponent
|
sl@0
|
1403 |
*/
|
sl@0
|
1404 |
static void FhConvertToFileName(TDes& aFileName, const TDesC& aPrivateDir)
|
sl@0
|
1405 |
{
|
sl@0
|
1406 |
TInt firstPos = aFileName.Locate(TChar(KFhSeparator));
|
sl@0
|
1407 |
if(firstPos >= 0)
|
sl@0
|
1408 |
{
|
sl@0
|
1409 |
aFileName.Delete(firstPos, 1);
|
sl@0
|
1410 |
TInt lastPos = aFileName.LocateReverse(TChar(KFhSeparator));
|
sl@0
|
1411 |
if(lastPos >= 0)
|
sl@0
|
1412 |
{
|
sl@0
|
1413 |
aFileName.Delete(lastPos, 1);
|
sl@0
|
1414 |
TParse parse;
|
sl@0
|
1415 |
(void)parse.Set(aFileName, &aPrivateDir, 0);//the file name should be verified by the server
|
sl@0
|
1416 |
aFileName.Copy(parse.FullName());
|
sl@0
|
1417 |
}
|
sl@0
|
1418 |
}
|
sl@0
|
1419 |
}
|
sl@0
|
1420 |
|
sl@0
|
1421 |
/**
|
sl@0
|
1422 |
Extracts the read-only flag and RMessage address from aDbFileName and stores them in the single COsLayerData instance.
|
sl@0
|
1423 |
|
sl@0
|
1424 |
@param aDbFileName Input/output parameter. The file name.
|
sl@0
|
1425 |
It will be reformatted and won't contain the already extracted data.
|
sl@0
|
1426 |
The aDbFileName format is:
|
sl@0
|
1427 |
@code
|
sl@0
|
1428 |
|<R/O flag><RMessage2 pointer><drive><app SID><file_name><file_ext>|
|
sl@0
|
1429 |
@endcode
|
sl@0
|
1430 |
|
sl@0
|
1431 |
@see TVfs::Open()
|
sl@0
|
1432 |
@see TFhStrType
|
sl@0
|
1433 |
@see FhStringProps()
|
sl@0
|
1434 |
|
sl@0
|
1435 |
@internalComponent
|
sl@0
|
1436 |
|
sl@0
|
1437 |
@panic Sqlite 12 In _DEBUG mode - invalid position of the "|" character in the file name.
|
sl@0
|
1438 |
@panic Sqlite 12 In _DEBUG mode - no RMessage2 pointer can be extracted from the file name.
|
sl@0
|
1439 |
@panic Sqlite 12 In _DEBUG mode - the extracted RMessage2 pointer is NULL.
|
sl@0
|
1440 |
*/
|
sl@0
|
1441 |
static void FhExtractAndStore(TDes& aDbFileName)
|
sl@0
|
1442 |
{
|
sl@0
|
1443 |
TInt fhStartPos = aDbFileName.Locate(TChar(KFhSeparator));
|
sl@0
|
1444 |
__ASSERT_DEBUG(fhStartPos == KFhMarkPos, __SQLITEPANIC2(ESqliteOsPanicInvalidFhStr));
|
sl@0
|
1445 |
//If this file name string contains file handles
|
sl@0
|
1446 |
if(fhStartPos == KFhMarkPos)
|
sl@0
|
1447 |
{
|
sl@0
|
1448 |
//Extract from aDbFileName string RMessage2 object's address
|
sl@0
|
1449 |
TLex lex(aDbFileName.Mid(fhStartPos + KFhMsgAddrPos, KFhMsgAddrLen));
|
sl@0
|
1450 |
TUint32 addr;
|
sl@0
|
1451 |
TInt err = lex.Val(addr, EHex);
|
sl@0
|
1452 |
__ASSERT_DEBUG(err == KErrNone, __SQLITEPANIC2(ESqliteOsPanicInvalidFhStr));
|
sl@0
|
1453 |
if(err == KErrNone)
|
sl@0
|
1454 |
{
|
sl@0
|
1455 |
//Cast the address to RMessage2 pointer.
|
sl@0
|
1456 |
const RMessage2* msg = reinterpret_cast <const RMessage2*> (addr);
|
sl@0
|
1457 |
__ASSERT_DEBUG(msg != NULL, __SQLITEPANIC2(ESqliteOsPanicInvalidFhStr));
|
sl@0
|
1458 |
if(msg)
|
sl@0
|
1459 |
{
|
sl@0
|
1460 |
//Store the data from aDbFileName in the single COsLayerData instance.
|
sl@0
|
1461 |
TBool readOnly = aDbFileName[fhStartPos + KFhRoPos] > '0';
|
sl@0
|
1462 |
COsLayerData::Instance().StoreFhData(msg, readOnly);
|
sl@0
|
1463 |
//Remove: read-only flag and RMessage2 object's address
|
sl@0
|
1464 |
aDbFileName.Delete(fhStartPos + KFhRoPos, 1 + KFhMsgAddrLen);
|
sl@0
|
1465 |
}
|
sl@0
|
1466 |
}
|
sl@0
|
1467 |
}
|
sl@0
|
1468 |
}
|
sl@0
|
1469 |
|
sl@0
|
1470 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
1471 |
///////////////////// TDbFile class definition ///////////////////////////////////////////////////////////////////////
|
sl@0
|
1472 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
1473 |
|
sl@0
|
1474 |
/**
|
sl@0
|
1475 |
Initializes TDbFile data members with their default values.
|
sl@0
|
1476 |
*/
|
sl@0
|
1477 |
inline TDbFile::TDbFile() :
|
sl@0
|
1478 |
iFileBuf(KFileBufSize),
|
sl@0
|
1479 |
iLockType(SQLITE_LOCK_NONE),
|
sl@0
|
1480 |
iReadOnly(EFalse),
|
sl@0
|
1481 |
iSectorSize(0),
|
sl@0
|
1482 |
iDeviceCharacteristics(-1),
|
sl@0
|
1483 |
iIsFileCreated(ETrue)
|
sl@0
|
1484 |
{
|
sl@0
|
1485 |
#ifdef _SQLPROFILER
|
sl@0
|
1486 |
iIsJournal = EFalse;
|
sl@0
|
1487 |
#endif
|
sl@0
|
1488 |
pMethods = 0;
|
sl@0
|
1489 |
SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, TDBFILE_TDBFILE, "OS;0x%X;TDbFile::TDbFile", (TUint)this));
|
sl@0
|
1490 |
}
|
sl@0
|
1491 |
|
sl@0
|
1492 |
/**
|
sl@0
|
1493 |
Casts the passed sqlite3_file pointer to a reference to the derived class - TDbFile&.
|
sl@0
|
1494 |
All sqlite3_file pointers passed to TFileIo methods are actually pointers to TDbFile instances.
|
sl@0
|
1495 |
So the cast is safe.
|
sl@0
|
1496 |
|
sl@0
|
1497 |
@param aDbFile A pointer to a sqlite3_file instance
|
sl@0
|
1498 |
|
sl@0
|
1499 |
@return A TDbFile reference.
|
sl@0
|
1500 |
@see TFileIo
|
sl@0
|
1501 |
@see TVfs
|
sl@0
|
1502 |
@see TDbFile
|
sl@0
|
1503 |
|
sl@0
|
1504 |
@panic Sqlite 20 In _DEBUG mode if aDbFile is NULL.
|
sl@0
|
1505 |
|
sl@0
|
1506 |
@internalComponent
|
sl@0
|
1507 |
*/
|
sl@0
|
1508 |
static inline TDbFile& DbFile(sqlite3_file* aDbFile)
|
sl@0
|
1509 |
{
|
sl@0
|
1510 |
__ASSERT_DEBUG(aDbFile != 0, __SQLITEPANIC2(ESqliteOsPanicNullDbFilePtr));
|
sl@0
|
1511 |
return *(static_cast <TDbFile*> (aDbFile));
|
sl@0
|
1512 |
}
|
sl@0
|
1513 |
|
sl@0
|
1514 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
1515 |
///////////////////// TFileIo class definition ///////////////////////////////////////////////////////////////////////
|
sl@0
|
1516 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
1517 |
|
sl@0
|
1518 |
/**
|
sl@0
|
1519 |
SQLite OS porting layer API.
|
sl@0
|
1520 |
|
sl@0
|
1521 |
Closes the file referred by aDbFile parameter.
|
sl@0
|
1522 |
|
sl@0
|
1523 |
@param aDbFile A pointer to a TDbFile instance, than contains the file handle to be closed.
|
sl@0
|
1524 |
|
sl@0
|
1525 |
@return SQLITE_OK
|
sl@0
|
1526 |
|
sl@0
|
1527 |
@see TDbFile
|
sl@0
|
1528 |
*/
|
sl@0
|
1529 |
/* static */ int TFileIo::Close(sqlite3_file* aDbFile)
|
sl@0
|
1530 |
{
|
sl@0
|
1531 |
TDbFile& dbFile = ::DbFile(aDbFile);
|
sl@0
|
1532 |
SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, TFILEIO_CLOSE1, "OS;0x%X;TFileIo::Close", (TUint)&dbFile));
|
sl@0
|
1533 |
__OS_CALL(EOsFileClose, 0, 0);
|
sl@0
|
1534 |
__OSTIME_COUNTER(TheOsCallTicks[EOsFileClose], ::OsCallProfile(dbFile.iIsJournal, EOsFileClose), 0, 0, aDbFile, 0);
|
sl@0
|
1535 |
__FS_CALL(EFsOpFileClose, 0);
|
sl@0
|
1536 |
dbFile.iFileBuf.Close();
|
sl@0
|
1537 |
return SQLITE_OK;
|
sl@0
|
1538 |
}
|
sl@0
|
1539 |
|
sl@0
|
1540 |
/**
|
sl@0
|
1541 |
SQLite OS porting layer API.
|
sl@0
|
1542 |
|
sl@0
|
1543 |
Reads from the file referred by the aDbFile parameter.
|
sl@0
|
1544 |
|
sl@0
|
1545 |
@param aDbFile A pointer to a TDbFile instance, that contains the file handle to be read from.
|
sl@0
|
1546 |
@param aBuf Output parameter. The data read from the file will be copied there.
|
sl@0
|
1547 |
The buffer size must be at least aAmt bytes.
|
sl@0
|
1548 |
@param aAmt The amount of data to be read form the file.
|
sl@0
|
1549 |
@param aOffset The offset in the file where the read operation should start.
|
sl@0
|
1550 |
|
sl@0
|
1551 |
@return SQLITE_IOERR_READ, The file read or seek operation has failed;
|
sl@0
|
1552 |
SQLITE_IOERR_SHORT_READ, The amount of the data read is less than aAmt;
|
sl@0
|
1553 |
SQLITE_IOERR_NOMEM, An out of memory condition has occured;
|
sl@0
|
1554 |
SQLITE_OK, The operation has completed successfully.
|
sl@0
|
1555 |
|
sl@0
|
1556 |
If the function succeeds, COsLayerData::SetOsErrorCode() is called with KErrNone, otherwise COsLayerData::SetOsErrorCode() is called
|
sl@0
|
1557 |
with the reported by the OS API error. The stored error code will be used later by the SQLite API caller.
|
sl@0
|
1558 |
|
sl@0
|
1559 |
@see COsLayerData::SetOsErrorCode()
|
sl@0
|
1560 |
@see TDbFile
|
sl@0
|
1561 |
*/
|
sl@0
|
1562 |
/* static */ int TFileIo::Read(sqlite3_file* aDbFile, void* aBuf, int aAmt, sqlite3_int64 aOffset)
|
sl@0
|
1563 |
{
|
sl@0
|
1564 |
TDbFile& dbFile = ::DbFile(aDbFile);
|
sl@0
|
1565 |
SQLITE_TRACE_OS(OstTraceExt3(TRACE_INTERNALS, TFILEIO_READ_ENTRY, "OS-Entry;0x%X;TFileIo::Read;aAmt=%d;aOffset=%lld", (TUint)&dbFile, aAmt, aOffset));
|
sl@0
|
1566 |
__OS_CALL(EOsFileRead, 0, 0);
|
sl@0
|
1567 |
__COUNTER_INCR(TheSqlSrvProfilerFileRead);
|
sl@0
|
1568 |
__OSTIME_COUNTER(TheOsCallTicks[EOsFileRead], ::OsCallProfile(dbFile.iIsJournal, EOsFileRead), aOffset, aAmt, aDbFile, 0);
|
sl@0
|
1569 |
TPtr8 ptr((TUint8*)aBuf, 0, aAmt);
|
sl@0
|
1570 |
TInt err = dbFile.iFileBuf.Read(aOffset, ptr);
|
sl@0
|
1571 |
TInt cnt = ptr.Length();
|
sl@0
|
1572 |
TInt sqliteErr = ::Os2SqliteErr(err, SQLITE_IOERR_READ);
|
sl@0
|
1573 |
if(cnt != aAmt && (sqliteErr == SQLITE_OK || sqliteErr == SQLITE_IOERR_SHORT_READ))
|
sl@0
|
1574 |
{
|
sl@0
|
1575 |
Mem::FillZ(static_cast <TUint8*> (aBuf) + cnt, aAmt - cnt);
|
sl@0
|
1576 |
err = KErrEof;
|
sl@0
|
1577 |
sqliteErr = SQLITE_IOERR_SHORT_READ;
|
sl@0
|
1578 |
}
|
sl@0
|
1579 |
COsLayerData::Instance().SetOsErrorCode(err);
|
sl@0
|
1580 |
SQLITE_TRACE_OS(OstTraceExt4(TRACE_INTERNALS, TFILEIO_READ_EXIT, "OS-Exit;0x%X;TFileIo::Read;cnt=%d;err=%d;sqliteErr=%d", (TUint)&dbFile, cnt, err, sqliteErr));
|
sl@0
|
1581 |
return sqliteErr;
|
sl@0
|
1582 |
}
|
sl@0
|
1583 |
|
sl@0
|
1584 |
//Creates a temporary file in "\temp" subdirectory of osLayerData.iSysPrivDir directory.
|
sl@0
|
1585 |
//If the function fails, the temp file will be closed and deleted,
|
sl@0
|
1586 |
//the related Symbian OS error will be returned to the caller.
|
sl@0
|
1587 |
/* static */TInt TFileIo::DoCreateTempFile(TDbFile& aDbFile)
|
sl@0
|
1588 |
{
|
sl@0
|
1589 |
COsLayerData& osLayerData = COsLayerData::Instance();
|
sl@0
|
1590 |
//TParse2 is used in order to avoid the need of another TFileName stack based variable
|
sl@0
|
1591 |
class TParse2 : public TParse
|
sl@0
|
1592 |
{
|
sl@0
|
1593 |
public:
|
sl@0
|
1594 |
inline TFileName& FileName()
|
sl@0
|
1595 |
{
|
sl@0
|
1596 |
return static_cast <TFileName&> (NameBuf());
|
sl@0
|
1597 |
}
|
sl@0
|
1598 |
};
|
sl@0
|
1599 |
TParse2 parse;
|
sl@0
|
1600 |
(void)parse.Set(osLayerData.iSysPrivDir, 0, 0);//this call can't fail
|
sl@0
|
1601 |
(void)parse.AddDir(KTempFileDir);//this call can't fail
|
sl@0
|
1602 |
__FS_CALL(EFsOpFileCreateTemp, 0);
|
sl@0
|
1603 |
TInt err = aDbFile.iFileBuf.Temp(osLayerData.iFs, parse.FullName(), parse.FileName(), EFileRead|EFileWrite|EDeleteOnClose);
|
sl@0
|
1604 |
if(err == KErrPathNotFound)
|
sl@0
|
1605 |
{
|
sl@0
|
1606 |
err = osLayerData.iFs.MkDirAll(parse.DriveAndPath());
|
sl@0
|
1607 |
if(err == KErrNone)
|
sl@0
|
1608 |
{
|
sl@0
|
1609 |
err = aDbFile.iFileBuf.Temp(osLayerData.iFs, parse.FullName(), parse.FileName(), EFileRead|EFileWrite|EDeleteOnClose);
|
sl@0
|
1610 |
}
|
sl@0
|
1611 |
}
|
sl@0
|
1612 |
if(err == KErrNone)
|
sl@0
|
1613 |
{
|
sl@0
|
1614 |
TInt recReadBufSize = -1;
|
sl@0
|
1615 |
err = TVfs::DoGetDeviceCharacteristicsAndSectorSize(aDbFile, recReadBufSize);
|
sl@0
|
1616 |
if(err != KErrNone)
|
sl@0
|
1617 |
{
|
sl@0
|
1618 |
aDbFile.iFileBuf.Close();//With EDeleteOnClose flag set, the file will be deleted
|
sl@0
|
1619 |
}
|
sl@0
|
1620 |
else
|
sl@0
|
1621 |
{
|
sl@0
|
1622 |
(void)aDbFile.iFileBuf.SetReadAheadSize(aDbFile.iSectorSize, recReadBufSize);
|
sl@0
|
1623 |
aDbFile.iIsFileCreated = ETrue;
|
sl@0
|
1624 |
}
|
sl@0
|
1625 |
}
|
sl@0
|
1626 |
return err;
|
sl@0
|
1627 |
}
|
sl@0
|
1628 |
|
sl@0
|
1629 |
/**
|
sl@0
|
1630 |
SQLite OS porting layer API.
|
sl@0
|
1631 |
|
sl@0
|
1632 |
Writes to the file referred by the aDbFile parameter.
|
sl@0
|
1633 |
"Write beyond the end of the file" operations are allowed.
|
sl@0
|
1634 |
|
sl@0
|
1635 |
If the write operation is in the 1st db file page and there is a registered "free pages" callback
|
sl@0
|
1636 |
(TDbFile::iFreePageCallback) and the free pages count is above the defined value,
|
sl@0
|
1637 |
then the callback will be called.
|
sl@0
|
1638 |
|
sl@0
|
1639 |
If the file to be written to is a temp file, which is not created yet, then the file will be created.
|
sl@0
|
1640 |
|
sl@0
|
1641 |
@param aDbFile A pointer to a TDbFile instance, that contains the file handle to be written to.
|
sl@0
|
1642 |
@param aData The data to be written to the file. The buffer size must be at least aAmt bytes.
|
sl@0
|
1643 |
@param aAmt The amount of data to be written to the file.
|
sl@0
|
1644 |
@param aOffset The offset in the file where the write operation should start.
|
sl@0
|
1645 |
|
sl@0
|
1646 |
@return SQLITE_FULL, The file write or seek operation has failed.
|
sl@0
|
1647 |
The disk is full;
|
sl@0
|
1648 |
SQLITE_IOERR_NOMEM, An out of memory condition has occured;
|
sl@0
|
1649 |
SQLITE_OK, The operation has completed successfully.
|
sl@0
|
1650 |
|
sl@0
|
1651 |
If the function succeeds, COsLayerData::SetOsErrorCode() is called with KErrNone, otherwise COsLayerData::SetOsErrorCode() is called
|
sl@0
|
1652 |
with the reported by the OS API error. The stored error code will be used later by the SQLite API caller.
|
sl@0
|
1653 |
|
sl@0
|
1654 |
@see COsLayerData::SetOsErrorCode()
|
sl@0
|
1655 |
@see TDbFile
|
sl@0
|
1656 |
*/
|
sl@0
|
1657 |
/* static */ int TFileIo::Write(sqlite3_file* aDbFile, const void* aData, int aAmt, sqlite3_int64 aOffset)
|
sl@0
|
1658 |
{
|
sl@0
|
1659 |
TDbFile& dbFile = ::DbFile(aDbFile);
|
sl@0
|
1660 |
SQLITE_TRACE_OS(OstTraceExt3(TRACE_INTERNALS, TFILEIO_WRITE_ENTRY, "OS-Entry;0x%X;TFileIo::Write;aAmt=%d;aOffset=%lld", (TUint)&dbFile, aAmt, aOffset));
|
sl@0
|
1661 |
__OS_CALL(EOsFileWrite, 0, 0);
|
sl@0
|
1662 |
__COUNTER_INCR(TheSqlSrvProfilerFileWrite);
|
sl@0
|
1663 |
__OSTIME_COUNTER(TheOsCallTicks[EOsFileWrite], ::OsCallProfile(dbFile.iIsJournal, EOsFileWrite), aOffset, aAmt, aDbFile, 0);
|
sl@0
|
1664 |
TInt err = KErrNone;
|
sl@0
|
1665 |
if(!dbFile.iIsFileCreated)
|
sl@0
|
1666 |
{//Create a temp file if it has not been created.
|
sl@0
|
1667 |
err = TFileIo::DoCreateTempFile(dbFile);
|
sl@0
|
1668 |
}
|
sl@0
|
1669 |
if(err != KErrNone)
|
sl@0
|
1670 |
{
|
sl@0
|
1671 |
COsLayerData::Instance().SetOsErrorCode(err);
|
sl@0
|
1672 |
return ::Os2SqliteErr(err, SQLITE_FULL);
|
sl@0
|
1673 |
}
|
sl@0
|
1674 |
err = KErrAccessDenied;
|
sl@0
|
1675 |
if(!dbFile.iReadOnly)
|
sl@0
|
1676 |
{
|
sl@0
|
1677 |
TPtrC8 ptr((const TUint8*)aData, aAmt);
|
sl@0
|
1678 |
err = dbFile.iFileBuf.Write(aOffset, ptr);
|
sl@0
|
1679 |
}
|
sl@0
|
1680 |
COsLayerData::Instance().SetOsErrorCode(err);
|
sl@0
|
1681 |
|
sl@0
|
1682 |
const TInt KFreePageCountOffset = 36;//hard-coded constant. SQLite does not offer anything - a constant or #define.
|
sl@0
|
1683 |
//The checks in the "if" bellow do:
|
sl@0
|
1684 |
// - "err == KErrNone" - check the free page count only after a successful "write";
|
sl@0
|
1685 |
// - "aOffset == 0" - check the free page count only if the write operation affects the system page (at aOffset = 0);
|
sl@0
|
1686 |
// - "aAmt >= (KFreePageCountOffset + sizeof(int))" - check the free page count only if the amount of bytes to be written
|
sl@0
|
1687 |
// is more than the offset of the free page counter (othewrise the free page counter is not affected
|
sl@0
|
1688 |
// by this write operation);
|
sl@0
|
1689 |
// - "dbFile.iFreePageCallback.IsValid()" - check the free page count only if there is a valid callback;
|
sl@0
|
1690 |
if(err == KErrNone && aOffset == 0 && aAmt >= (KFreePageCountOffset + sizeof(int)) && dbFile.iFreePageCallback.IsValid())
|
sl@0
|
1691 |
{
|
sl@0
|
1692 |
const TUint8* ptr = static_cast <const TUint8*> (aData) + KFreePageCountOffset;
|
sl@0
|
1693 |
TInt freePageCount = (ptr[0] << 24) | (ptr[1] << 16) | (ptr[2] << 8) | ptr[3];
|
sl@0
|
1694 |
dbFile.iFreePageCallback.CheckAndCallback(freePageCount);
|
sl@0
|
1695 |
}
|
sl@0
|
1696 |
|
sl@0
|
1697 |
TInt sqliteErr = ::Os2SqliteErr(err, SQLITE_FULL);
|
sl@0
|
1698 |
SQLITE_TRACE_OS(OstTraceExt3(TRACE_INTERNALS, TFILEIO_WRITE_EXIT, "OS-Exit;0x%X;TFileIo::Write;err=%d;sqliteErr=%d", (TUint)&dbFile, err, sqliteErr));
|
sl@0
|
1699 |
return sqliteErr;
|
sl@0
|
1700 |
}
|
sl@0
|
1701 |
|
sl@0
|
1702 |
/**
|
sl@0
|
1703 |
SQLite OS porting layer API.
|
sl@0
|
1704 |
|
sl@0
|
1705 |
Truncates the file referred by the aDbFile parameter.
|
sl@0
|
1706 |
|
sl@0
|
1707 |
@param aDbFile A pointer to a TDbFile instance, that contains the file handle.
|
sl@0
|
1708 |
@param aLength The new file size in bytes.
|
sl@0
|
1709 |
|
sl@0
|
1710 |
@return SQLITE_FULL, The disk is full;
|
sl@0
|
1711 |
SQLITE_IOERR, This is a read-only file.
|
sl@0
|
1712 |
The file truncate operation has failed;
|
sl@0
|
1713 |
SQLITE_IOERR_NOMEM, An out of memory condition has occured;
|
sl@0
|
1714 |
SQLITE_OK, The operation has completed successfully.
|
sl@0
|
1715 |
|
sl@0
|
1716 |
If the function succeeds, COsLayerData::SetOsErrorCode() is called with KErrNone, otherwise COsLayerData::SetOsErrorCode() is called
|
sl@0
|
1717 |
with the reported by the OS API error. The stored error code will be used later by the SQLite API caller.
|
sl@0
|
1718 |
|
sl@0
|
1719 |
@see COsLayerData::SetOsErrorCode()
|
sl@0
|
1720 |
@see TDbFile
|
sl@0
|
1721 |
*/
|
sl@0
|
1722 |
/* static */ int TFileIo::Truncate(sqlite3_file* aDbFile, sqlite3_int64 aLength)
|
sl@0
|
1723 |
{
|
sl@0
|
1724 |
TDbFile& dbFile = ::DbFile(aDbFile);
|
sl@0
|
1725 |
SQLITE_TRACE_OS(OstTraceExt2(TRACE_INTERNALS, TFILEIO_TRUNCATE_ENTRY, "OS-Entry;0x%X;TFileIo::Truncate;aLength=%lld", (TUint)&dbFile, aLength));
|
sl@0
|
1726 |
__OS_CALL(EOsFileTruncate, 0, 0);
|
sl@0
|
1727 |
__COUNTER_INCR(TheSqlSrvProfilerFileSetSize);
|
sl@0
|
1728 |
__OSTIME_COUNTER(TheOsCallTicks[EOsFileTruncate], ::OsCallProfile(dbFile.iIsJournal, EOsFileTruncate), aLength, 0, aDbFile, 0);
|
sl@0
|
1729 |
if(dbFile.iReadOnly)
|
sl@0
|
1730 |
{
|
sl@0
|
1731 |
COsLayerData::Instance().SetOsErrorCode(KErrAccessDenied);
|
sl@0
|
1732 |
return SQLITE_IOERR;
|
sl@0
|
1733 |
}
|
sl@0
|
1734 |
__FS_CALL(EFsOpFileSetSize, 0);
|
sl@0
|
1735 |
TInt err = dbFile.iFileBuf.SetSize(aLength);
|
sl@0
|
1736 |
COsLayerData::Instance().SetOsErrorCode(err);
|
sl@0
|
1737 |
TInt sqliteErr = ::Os2SqliteErr(err, SQLITE_IOERR);
|
sl@0
|
1738 |
SQLITE_TRACE_OS(OstTraceExt3(TRACE_INTERNALS, TFILEIO_TRUNCATE_EXIT, "OS-Exit;0x%X;TFileIo::Truncate;err=%d;sqliteErr=%d", (TUint)&dbFile, err, sqliteErr));
|
sl@0
|
1739 |
return sqliteErr;
|
sl@0
|
1740 |
}
|
sl@0
|
1741 |
|
sl@0
|
1742 |
/**
|
sl@0
|
1743 |
SQLite OS porting layer API.
|
sl@0
|
1744 |
|
sl@0
|
1745 |
Flushes the file referred by the aDbFile parameter.
|
sl@0
|
1746 |
|
sl@0
|
1747 |
@param aDbFile A pointer to a TDbFile instance, that contains the file handle.
|
sl@0
|
1748 |
|
sl@0
|
1749 |
@return SQLITE_IOERR, This is a read-only file.
|
sl@0
|
1750 |
The file flush operation has failed;
|
sl@0
|
1751 |
SQLITE_IOERR_NOMEM, An out of memory condition has occured;
|
sl@0
|
1752 |
SQLITE_OK, The operation has completed successfully.
|
sl@0
|
1753 |
|
sl@0
|
1754 |
If the function succeeds, COsLayerData::SetOsErrorCode() is called with KErrNone, otherwise COsLayerData::SetOsErrorCode() is called
|
sl@0
|
1755 |
with the reported by the OS API error. The stored error code will be used later by the SQLite API caller.
|
sl@0
|
1756 |
|
sl@0
|
1757 |
@see COsLayerData::SetOsErrorCode()
|
sl@0
|
1758 |
@see TDbFile
|
sl@0
|
1759 |
*/
|
sl@0
|
1760 |
/* static */int TFileIo::Sync(sqlite3_file* aDbFile, int /* aFlags */)
|
sl@0
|
1761 |
{
|
sl@0
|
1762 |
TDbFile& dbFile = ::DbFile(aDbFile);
|
sl@0
|
1763 |
SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, TFILEIO_SYNC_ENTRY, "OS-Entry;0x%X;TFileIo::Sync", (TUint)&dbFile));
|
sl@0
|
1764 |
__OS_CALL(EOsFileSync, 0, 0);
|
sl@0
|
1765 |
__COUNTER_INCR(TheSqlSrvProfilerFileSync);
|
sl@0
|
1766 |
__OSTIME_COUNTER(TheOsCallTicks[EOsFileSync], ::OsCallProfile(dbFile.iIsJournal, EOsFileSync), 0, 0, aDbFile, 0);
|
sl@0
|
1767 |
if(dbFile.iReadOnly)
|
sl@0
|
1768 |
{
|
sl@0
|
1769 |
COsLayerData::Instance().SetOsErrorCode(KErrAccessDenied);
|
sl@0
|
1770 |
return SQLITE_IOERR;
|
sl@0
|
1771 |
}
|
sl@0
|
1772 |
__FS_CALL(EFsOpFileSync, 0);
|
sl@0
|
1773 |
TInt err = dbFile.iFileBuf.Flush();
|
sl@0
|
1774 |
COsLayerData::Instance().SetOsErrorCode(err);
|
sl@0
|
1775 |
TInt sqliteErr = ::Os2SqliteErr(err, SQLITE_IOERR);
|
sl@0
|
1776 |
SQLITE_TRACE_OS(OstTraceExt3(TRACE_INTERNALS, TFILEIO_SYNC_EXIT, "OS-Exit;0x%X;TFileIo::Sync;err=%d;sqliteErr=%d", (TUint)&dbFile, err, sqliteErr));
|
sl@0
|
1777 |
return sqliteErr;
|
sl@0
|
1778 |
}
|
sl@0
|
1779 |
|
sl@0
|
1780 |
/**
|
sl@0
|
1781 |
SQLite OS porting layer API.
|
sl@0
|
1782 |
|
sl@0
|
1783 |
Returns the size of the file referred by the aDbFile parameter.
|
sl@0
|
1784 |
|
sl@0
|
1785 |
@param aDbFile A pointer to a TDbFile instance, that contains the file handle.
|
sl@0
|
1786 |
@param aSize Output parameter. If the function completes successfully, the file size will be stored there.
|
sl@0
|
1787 |
|
sl@0
|
1788 |
@return SQLITE_IOERR, The file size operation has failed;
|
sl@0
|
1789 |
SQLITE_IOERR_NOMEM, An out of memory condition has occured;
|
sl@0
|
1790 |
SQLITE_OK, The operation has completed successfully.
|
sl@0
|
1791 |
|
sl@0
|
1792 |
If the function succeeds, COsLayerData::SetOsErrorCode() is called with KErrNone, otherwise COsLayerData::SetOsErrorCode() is called
|
sl@0
|
1793 |
with the reported by the OS API error. The stored error code will be used later by the SQLite API caller.
|
sl@0
|
1794 |
|
sl@0
|
1795 |
@see COsLayerData::SetOsErrorCode()
|
sl@0
|
1796 |
@see TDbFile
|
sl@0
|
1797 |
*/
|
sl@0
|
1798 |
/* static */ int TFileIo::FileSize(sqlite3_file* aDbFile, sqlite3_int64* aSize)
|
sl@0
|
1799 |
{
|
sl@0
|
1800 |
TDbFile& dbFile = ::DbFile(aDbFile);
|
sl@0
|
1801 |
SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, TFILEIO_FILESIZE_ENTRY, "OS-Entry;0x%X;TFileIo::FileSize", (TUint)&dbFile));
|
sl@0
|
1802 |
__OS_CALL(EOsFileFileSize, 0, 0);
|
sl@0
|
1803 |
__OSTIME_COUNTER(TheOsCallTicks[EOsFileFileSize], ::OsCallProfile(dbFile.iIsJournal, EOsFileFileSize), 0, 0, aDbFile, 0);
|
sl@0
|
1804 |
__FS_CALL(EFsOpFileSize, 0);
|
sl@0
|
1805 |
TInt err = dbFile.iFileBuf.Size(*aSize);
|
sl@0
|
1806 |
COsLayerData::Instance().SetOsErrorCode(err);
|
sl@0
|
1807 |
TInt sqliteErr = ::Os2SqliteErr(err, SQLITE_IOERR);
|
sl@0
|
1808 |
SQLITE_TRACE_OS(OstTraceExt3(TRACE_INTERNALS, TFILEIO_FILESIZE_EXIT, "OS-Exit;0x%X;TFileIo::FileSize;err=%d;sqliteErr=%d", (TUint)&dbFile, err, sqliteErr));
|
sl@0
|
1809 |
return sqliteErr;
|
sl@0
|
1810 |
}
|
sl@0
|
1811 |
|
sl@0
|
1812 |
/**
|
sl@0
|
1813 |
SQLite OS porting layer API.
|
sl@0
|
1814 |
|
sl@0
|
1815 |
Locks the file, referred by the aDbFile parameter, with the specified lock type.
|
sl@0
|
1816 |
Since this is a single-threaded OS porting layer implementation, the file is not actually locked - small
|
sl@0
|
1817 |
performance optimisation. The file lock type is stored for later use by the CheckReservedLock() call.
|
sl@0
|
1818 |
|
sl@0
|
1819 |
@param aDbFile A pointer to a TDbFile instance, that contains the file handle.
|
sl@0
|
1820 |
@param aLockType Lock type: SQLITE_LOCK_NONE, SQLITE_LOCK_SHARED, SQLITE_LOCK_RESERVED, SQLITE_LOCK_PENDING or
|
sl@0
|
1821 |
SQLITE_LOCK_EXCLUSIVE.
|
sl@0
|
1822 |
|
sl@0
|
1823 |
@return SQLITE_OK, The operation has completed successfully.
|
sl@0
|
1824 |
|
sl@0
|
1825 |
@see TFileIo::CheckReservedLock()
|
sl@0
|
1826 |
@see TFileIo::Unlock()
|
sl@0
|
1827 |
|
sl@0
|
1828 |
@see TDbFile
|
sl@0
|
1829 |
*/
|
sl@0
|
1830 |
/* static */ int TFileIo::Lock(sqlite3_file* aDbFile, int aLockType)
|
sl@0
|
1831 |
{
|
sl@0
|
1832 |
TDbFile& dbFile = ::DbFile(aDbFile);
|
sl@0
|
1833 |
__OS_CALL(EOsFileLock, 0, 0);
|
sl@0
|
1834 |
__OSTIME_COUNTER(TheOsCallTicks[EOsFileLock], ::OsCallProfile(dbFile.iIsJournal, EOsFileLock), aLockType, 0, aDbFile, 0);
|
sl@0
|
1835 |
//If there is already a lock of this type or more restrictive on the database file, do nothing.
|
sl@0
|
1836 |
if(dbFile.iLockType >= aLockType)
|
sl@0
|
1837 |
{
|
sl@0
|
1838 |
return SQLITE_OK;
|
sl@0
|
1839 |
}
|
sl@0
|
1840 |
dbFile.iLockType = aLockType;
|
sl@0
|
1841 |
return SQLITE_OK;
|
sl@0
|
1842 |
}
|
sl@0
|
1843 |
|
sl@0
|
1844 |
/**
|
sl@0
|
1845 |
SQLite OS porting layer API.
|
sl@0
|
1846 |
|
sl@0
|
1847 |
Unlocks the file, referred by the aDbFile parameter.
|
sl@0
|
1848 |
Since this is a single-threaded OS porting layer implementation, the file never gets locked - small
|
sl@0
|
1849 |
performance optimisation. The Unlock() call only sets the stored file lock type with the aLockType value.
|
sl@0
|
1850 |
|
sl@0
|
1851 |
@param aDbFile A pointer to a TDbFile instance, that contains the file handle.
|
sl@0
|
1852 |
@param aLockType Lock type: SQLITE_LOCK_NONE, SQLITE_LOCK_SHARED, SQLITE_LOCK_RESERVED, SQLITE_LOCK_PENDING or
|
sl@0
|
1853 |
SQLITE_LOCK_EXCLUSIVE.
|
sl@0
|
1854 |
|
sl@0
|
1855 |
@return SQLITE_OK, The operation has completed successfully.
|
sl@0
|
1856 |
|
sl@0
|
1857 |
@see TFileIo::CheckReservedLock()
|
sl@0
|
1858 |
@see TFileIo::Lock()
|
sl@0
|
1859 |
|
sl@0
|
1860 |
@see TDbFile
|
sl@0
|
1861 |
*/
|
sl@0
|
1862 |
/* static */ int TFileIo::Unlock(sqlite3_file* aDbFile, int aLockType)
|
sl@0
|
1863 |
{
|
sl@0
|
1864 |
TDbFile& dbFile = ::DbFile(aDbFile);
|
sl@0
|
1865 |
__OS_CALL(EOsFileUnlock, 0, 0);
|
sl@0
|
1866 |
__OSTIME_COUNTER(TheOsCallTicks[EOsFileUnlock], ::OsCallProfile(dbFile.iIsJournal, EOsFileUnlock), aLockType, 0, aDbFile, 0);
|
sl@0
|
1867 |
dbFile.iLockType = aLockType;
|
sl@0
|
1868 |
return SQLITE_OK;
|
sl@0
|
1869 |
}
|
sl@0
|
1870 |
|
sl@0
|
1871 |
/**
|
sl@0
|
1872 |
SQLite OS porting layer API.
|
sl@0
|
1873 |
|
sl@0
|
1874 |
Checks if the file lock type is SQLITE_LOCK_RESERVED or bigger.
|
sl@0
|
1875 |
Since this is a single-threaded OS porting layer implementation, the file never gets locked - small
|
sl@0
|
1876 |
performance optimisation. The CheckReservedLock() call only checks if the stored file lock type
|
sl@0
|
1877 |
is bigger or equal than SQLITE_LOCK_RESERVED.
|
sl@0
|
1878 |
|
sl@0
|
1879 |
@param aDbFile A pointer to a TDbFile instance, that contains the file handle.
|
sl@0
|
1880 |
@param aResOut Output parameter. It should be set to 1 if the stored lock type is bigger or equal
|
sl@0
|
1881 |
than SQLITE_LOCK_RESERVED.
|
sl@0
|
1882 |
|
sl@0
|
1883 |
@return SQLITE_OK.
|
sl@0
|
1884 |
|
sl@0
|
1885 |
@see TFileIo::Lock()
|
sl@0
|
1886 |
@see TFileIo::Unlock()
|
sl@0
|
1887 |
|
sl@0
|
1888 |
@see TDbFile
|
sl@0
|
1889 |
*/
|
sl@0
|
1890 |
/* static */ int TFileIo::CheckReservedLock(sqlite3_file* aDbFile, int *aResOut)
|
sl@0
|
1891 |
{
|
sl@0
|
1892 |
TDbFile& dbFile = ::DbFile(aDbFile);
|
sl@0
|
1893 |
__OS_CALL(EOsFileCheckReservedLock, 0, 0);
|
sl@0
|
1894 |
__OSTIME_COUNTER(TheOsCallTicks[EOsFileCheckReservedLock], ::OsCallProfile(dbFile.iIsJournal, EOsFileCheckReservedLock), 0, 0, aDbFile, 0);
|
sl@0
|
1895 |
*aResOut = dbFile.iLockType >= SQLITE_LOCK_RESERVED ? 1 : 0;
|
sl@0
|
1896 |
return SQLITE_OK;
|
sl@0
|
1897 |
}
|
sl@0
|
1898 |
|
sl@0
|
1899 |
/**
|
sl@0
|
1900 |
SQLite OS porting layer API.
|
sl@0
|
1901 |
|
sl@0
|
1902 |
Performs an aOp operation on the file referred by the aDbFile parameter.
|
sl@0
|
1903 |
Since the only supported operation at the moment is SQLITE_FCNTL_LOCKSTATE, and the current lock type is stored as
|
sl@0
|
1904 |
a data memebr of TDbFile, the function implementation has been optimised - no file I/O calls. The stored file lock type
|
sl@0
|
1905 |
is retured if the operation is SQLITE_FCNTL_LOCKSTATE.
|
sl@0
|
1906 |
|
sl@0
|
1907 |
Note: The range of supported operations includes KSqlFcntlRegisterFreePageCallback now.
|
sl@0
|
1908 |
When the function is called with aOp = KSqlFcntlRegisterFreePageCallback, then a callback will be registered
|
sl@0
|
1909 |
and called when the number of the free pages goes above certain threshold.
|
sl@0
|
1910 |
|
sl@0
|
1911 |
@param aDbFile A pointer to a TDbFile instance, that contains the file handle.
|
sl@0
|
1912 |
@param aOp File operation type. Currently only SQLITE_FCNTL_LOCKSTATE is supported.
|
sl@0
|
1913 |
@param aArg An additional input/output parameter which purpose depends on the type of the current file operation.
|
sl@0
|
1914 |
If the file operation is SQLITE_FCNTL_LOCKSTATE, then aArg is used as an output parameter, where
|
sl@0
|
1915 |
the file lock type is stored.
|
sl@0
|
1916 |
If the operation type is KSqlFcntlRegisterFreePageCallback, then aArg points to a TSqlFreePageCallback object,
|
sl@0
|
1917 |
that contains the free page threshold and the callback.
|
sl@0
|
1918 |
|
sl@0
|
1919 |
@return SQLITE_ERROR, Non-supported operation;
|
sl@0
|
1920 |
SQLITE_OK, The operation has completed successfully.
|
sl@0
|
1921 |
|
sl@0
|
1922 |
@see TDbFile
|
sl@0
|
1923 |
*/
|
sl@0
|
1924 |
/* static */ int TFileIo::FileControl(sqlite3_file* aDbFile, int aOp, void* aArg)
|
sl@0
|
1925 |
{
|
sl@0
|
1926 |
TDbFile& dbFile = ::DbFile(aDbFile);
|
sl@0
|
1927 |
SQLITE_TRACE_OS(OstTraceExt2(TRACE_INTERNALS, TFILEIO_FILECONTROL_ENTRY, "OS-Entry;0x%X;TFileIo::FileControl;aOp=%d", (TUint)&dbFile, aOp));
|
sl@0
|
1928 |
__OS_CALL(EOsFileFileControl, 0, 0);
|
sl@0
|
1929 |
__OSTIME_COUNTER(TheOsCallTicks[EOsFileFileControl], ::OsCallProfile(dbFile.iIsJournal, EOsFileFileControl), aOp, 0, aDbFile, 0);
|
sl@0
|
1930 |
TInt err = KErrNone;
|
sl@0
|
1931 |
switch(aOp)
|
sl@0
|
1932 |
{
|
sl@0
|
1933 |
case SQLITE_FCNTL_LOCKSTATE:
|
sl@0
|
1934 |
*(int*)aArg = dbFile.iLockType;
|
sl@0
|
1935 |
break;
|
sl@0
|
1936 |
case KSqlFcntlRegisterFreePageCallback:
|
sl@0
|
1937 |
{
|
sl@0
|
1938 |
err = KErrArgument;
|
sl@0
|
1939 |
if(aArg)
|
sl@0
|
1940 |
{
|
sl@0
|
1941 |
TSqlFreePageCallback* rq = static_cast <TSqlFreePageCallback*> (aArg);
|
sl@0
|
1942 |
if(rq->IsValid())
|
sl@0
|
1943 |
{
|
sl@0
|
1944 |
dbFile.iFreePageCallback = *rq;
|
sl@0
|
1945 |
err = KErrNone;
|
sl@0
|
1946 |
}
|
sl@0
|
1947 |
}
|
sl@0
|
1948 |
}
|
sl@0
|
1949 |
break;
|
sl@0
|
1950 |
default:
|
sl@0
|
1951 |
err = KErrArgument;
|
sl@0
|
1952 |
break;
|
sl@0
|
1953 |
}
|
sl@0
|
1954 |
COsLayerData::Instance().SetOsErrorCode(err);
|
sl@0
|
1955 |
TInt sqliteErr = ::Os2SqliteErr(err, SQLITE_ERROR);
|
sl@0
|
1956 |
SQLITE_TRACE_OS(OstTraceExt3(TRACE_INTERNALS, TFILEIO_FILECONTROL_EXIT, "OS-Exit;0x%X;TFileIo::FileControl;err=%d;sqliteErr=%d", (TUint)&dbFile, err, sqliteErr));
|
sl@0
|
1957 |
return sqliteErr;
|
sl@0
|
1958 |
}
|
sl@0
|
1959 |
|
sl@0
|
1960 |
/**
|
sl@0
|
1961 |
SQLite OS porting layer API.
|
sl@0
|
1962 |
|
sl@0
|
1963 |
Retrieves the sector size of the media of the file referred by the aDbFile parameter.
|
sl@0
|
1964 |
Since the sector size never changes till the file is open, the function has been optimised - no file I/O calls.
|
sl@0
|
1965 |
The sector size is retrieved during the TVfs::Open() call and stored in TDbFile::iSectorSize. The SectorSize()
|
sl@0
|
1966 |
call returns the value of TDbFile::iSectorSize.
|
sl@0
|
1967 |
|
sl@0
|
1968 |
@param aDbFile A pointer to a TDbFile instance, that contains the file handle.
|
sl@0
|
1969 |
|
sl@0
|
1970 |
@return The sector size.
|
sl@0
|
1971 |
|
sl@0
|
1972 |
@panic Sqlite 19 In _DEBUG mode - TDbFile::iSectorSize is negative or 0 .
|
sl@0
|
1973 |
|
sl@0
|
1974 |
@see TDbFile
|
sl@0
|
1975 |
@see TVfs::Open()
|
sl@0
|
1976 |
*/
|
sl@0
|
1977 |
/* static */ int TFileIo::SectorSize(sqlite3_file* aDbFile)
|
sl@0
|
1978 |
{
|
sl@0
|
1979 |
TDbFile& dbFile = ::DbFile(aDbFile);
|
sl@0
|
1980 |
__OS_CALL(EOsFileSectorSize, 0, 0);
|
sl@0
|
1981 |
__OSTIME_COUNTER(TheOsCallTicks[EOsFileSectorSize], ::OsCallProfile(dbFile.iIsJournal, EOsFileSectorSize), 0, 0, aDbFile, 0);
|
sl@0
|
1982 |
__ASSERT_DEBUG(dbFile.iSectorSize > 0, __SQLITEPANIC2(ESqliteOsPanicInternalError));
|
sl@0
|
1983 |
return dbFile.iSectorSize;
|
sl@0
|
1984 |
}
|
sl@0
|
1985 |
|
sl@0
|
1986 |
/**
|
sl@0
|
1987 |
SQLite OS porting layer API.
|
sl@0
|
1988 |
|
sl@0
|
1989 |
Retrieves the device characteristics of the device of the file referred by the aDbFile parameter.
|
sl@0
|
1990 |
Since the device characteristics never change till the file is open, the function has been optimised - no file I/O calls.
|
sl@0
|
1991 |
The device characteristics are retrieved during the TVfs::Open() call and stored in TDbFile::iDeviceCharacteristics.
|
sl@0
|
1992 |
The DeviceCharacteristics() call returns the value of TDbFile::iDeviceCharacteristics.
|
sl@0
|
1993 |
|
sl@0
|
1994 |
@param aDbFile A pointer to a TDbFile instance, that contains the file handle.
|
sl@0
|
1995 |
|
sl@0
|
1996 |
@return A bit set containing the device characteristics.
|
sl@0
|
1997 |
|
sl@0
|
1998 |
@panic Sqlite 19 In _DEBUG mode - TDbFile::iDeviceCharacteristics is negative or 0 .
|
sl@0
|
1999 |
|
sl@0
|
2000 |
@see TDbFile
|
sl@0
|
2001 |
@see TVfs::Open()
|
sl@0
|
2002 |
*/
|
sl@0
|
2003 |
/* static */ int TFileIo::DeviceCharacteristics(sqlite3_file* aDbFile)
|
sl@0
|
2004 |
{
|
sl@0
|
2005 |
TDbFile& dbFile = ::DbFile(aDbFile);
|
sl@0
|
2006 |
__OS_CALL(EOsFileDeviceCharacteristics, 0, 0);
|
sl@0
|
2007 |
__OSTIME_COUNTER(TheOsCallTicks[EOsFileDeviceCharacteristics], ::OsCallProfile(dbFile.iIsJournal, EOsFileDeviceCharacteristics), 0, 0, aDbFile, 0);
|
sl@0
|
2008 |
__ASSERT_DEBUG(dbFile.iDeviceCharacteristics >= 0, __SQLITEPANIC2(ESqliteOsPanicInternalError));
|
sl@0
|
2009 |
return dbFile.iDeviceCharacteristics;
|
sl@0
|
2010 |
}
|
sl@0
|
2011 |
|
sl@0
|
2012 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
2013 |
/////////////////////////////////// TVfs class definition ///////////////////////////////////////////////////////////
|
sl@0
|
2014 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
2015 |
|
sl@0
|
2016 |
|
sl@0
|
2017 |
/**
|
sl@0
|
2018 |
Opens a private secure database.
|
sl@0
|
2019 |
Actually the database file has been created or opened by the client's process.
|
sl@0
|
2020 |
This function only adopts the passed in aMsg parameter file handle.
|
sl@0
|
2021 |
|
sl@0
|
2022 |
@param aDbFile Output parameter, where the initialized file handle will be stored.
|
sl@0
|
2023 |
@param aMsg A reference to the current RMessage2 instance. Contains the file handle of the database created
|
sl@0
|
2024 |
or opened by the client's process.
|
sl@0
|
2025 |
@param aReadOnly True if the file is read-only.
|
sl@0
|
2026 |
|
sl@0
|
2027 |
@return KErrNone, The operation has completed succesfully;
|
sl@0
|
2028 |
KErrNoMemory, Out of memory condition has occured;
|
sl@0
|
2029 |
Note that other system-wide error codes may also be returned.
|
sl@0
|
2030 |
|
sl@0
|
2031 |
@see TDbFile
|
sl@0
|
2032 |
@see TVfs::Open()
|
sl@0
|
2033 |
*/
|
sl@0
|
2034 |
/* static */ TInt TVfs::DoOpenFromHandle(TDbFile& aDbFile, const RMessage2& aMsg, TBool aReadOnly)
|
sl@0
|
2035 |
{
|
sl@0
|
2036 |
__FS_CALL(EFsOpFileAdopt, 0);
|
sl@0
|
2037 |
TInt err = aDbFile.iFileBuf.AdoptFromClient(aMsg, KFhSessHandleIdx, KFhFileHandleIdx);
|
sl@0
|
2038 |
if(err == KErrNone)
|
sl@0
|
2039 |
{
|
sl@0
|
2040 |
aDbFile.iReadOnly = aReadOnly;
|
sl@0
|
2041 |
}
|
sl@0
|
2042 |
return err;
|
sl@0
|
2043 |
};
|
sl@0
|
2044 |
|
sl@0
|
2045 |
/**
|
sl@0
|
2046 |
Collects information about the drive referred by the aDriveNo parameter.
|
sl@0
|
2047 |
|
sl@0
|
2048 |
@param aFs RFs instance.
|
sl@0
|
2049 |
@param aDriveNo The drive about which an information will be collected.
|
sl@0
|
2050 |
@param aVolumeInfo Output parameter. A reference to a TVolumeIOParamInfo object where the collected information will be stored.
|
sl@0
|
2051 |
|
sl@0
|
2052 |
@return KErrNone, The operation has completed succesfully;
|
sl@0
|
2053 |
KErrNoMemory, Out of memory condition has occured;
|
sl@0
|
2054 |
Note that other system-wide error codes may also be returned.
|
sl@0
|
2055 |
|
sl@0
|
2056 |
@see TVfs::Open()
|
sl@0
|
2057 |
*/
|
sl@0
|
2058 |
/* static */ inline TInt TVfs::DoGetVolumeIoParamInfo(RFs& aFs, TInt aDriveNo, TVolumeIOParamInfo& aVolumeInfo)
|
sl@0
|
2059 |
{
|
sl@0
|
2060 |
__FS_CALL(EFsOpFsVolumeIoParam, 0);
|
sl@0
|
2061 |
TInt err = aFs.VolumeIOParam(aDriveNo, aVolumeInfo);
|
sl@0
|
2062 |
SQLITE_TRACE_OS(OstTraceExt2(TRACE_INTERNALS, TVFS_DOGETVOLUMEIOPARAMINFO, "OS;0;TVfs::DoGetVolumeIoParamInfo;aDriveNo=%d;err=%d", aDriveNo, err));
|
sl@0
|
2063 |
return err;
|
sl@0
|
2064 |
}
|
sl@0
|
2065 |
|
sl@0
|
2066 |
//Maps disk sector sizes to SQLITE_IOCAP_ATOMIC<n> constants
|
sl@0
|
2067 |
|
sl@0
|
2068 |
struct TSqliteSectorSizeMap
|
sl@0
|
2069 |
{
|
sl@0
|
2070 |
TInt iSectorSize;
|
sl@0
|
2071 |
TInt iSqliteSectorSizeConstant;
|
sl@0
|
2072 |
};
|
sl@0
|
2073 |
|
sl@0
|
2074 |
//Used in TVfs::DoGetDeviceCharacteristics() to find which SQLITE_IOCAP_ATOMIC<n> constant should be used
|
sl@0
|
2075 |
//for the specified sector size
|
sl@0
|
2076 |
const TSqliteSectorSizeMap KSqliteSectorSizeMap[] =
|
sl@0
|
2077 |
{
|
sl@0
|
2078 |
{ 512, SQLITE_IOCAP_ATOMIC512},
|
sl@0
|
2079 |
{ 1024, SQLITE_IOCAP_ATOMIC1K},
|
sl@0
|
2080 |
{ 2048, SQLITE_IOCAP_ATOMIC2K},
|
sl@0
|
2081 |
{ 4096, SQLITE_IOCAP_ATOMIC4K},
|
sl@0
|
2082 |
{ 8192, SQLITE_IOCAP_ATOMIC8K},
|
sl@0
|
2083 |
{16384, SQLITE_IOCAP_ATOMIC16K},
|
sl@0
|
2084 |
{32768, SQLITE_IOCAP_ATOMIC32K},
|
sl@0
|
2085 |
{65536, SQLITE_IOCAP_ATOMIC64K}
|
sl@0
|
2086 |
};
|
sl@0
|
2087 |
|
sl@0
|
2088 |
/**
|
sl@0
|
2089 |
Retrieves and returns in a bit set the device characteristics.
|
sl@0
|
2090 |
|
sl@0
|
2091 |
@param aDriveInfo A TDriveInfo reference from which the device characteristics will be extracted.
|
sl@0
|
2092 |
@param aVolumeInfo A TVolumeIOParamInfo reference from which the device characteristics will be extracted.
|
sl@0
|
2093 |
|
sl@0
|
2094 |
@return A bit set containing the device characteristics:
|
sl@0
|
2095 |
SQLITE_IOCAP_SAFE_APPEND, SQLITE_IOCAP_ATOMIC, the atomic block size.
|
sl@0
|
2096 |
|
sl@0
|
2097 |
@see TVfs::DoGetVolumeIoParamInfo();
|
sl@0
|
2098 |
@see TVfs::Open()
|
sl@0
|
2099 |
*/
|
sl@0
|
2100 |
/* static */ TInt TVfs::DoGetDeviceCharacteristics(const TDriveInfo& aDriveInfo, const TVolumeIOParamInfo& aVolumeInfo)
|
sl@0
|
2101 |
{
|
sl@0
|
2102 |
TInt deviceCharacteristics = 0;
|
sl@0
|
2103 |
if(aDriveInfo.iDriveAtt & (KDriveAttLocal | KDriveAttInternal))
|
sl@0
|
2104 |
{
|
sl@0
|
2105 |
deviceCharacteristics |= SQLITE_IOCAP_SAFE_APPEND;//Data written first, file size updated second
|
sl@0
|
2106 |
}
|
sl@0
|
2107 |
if(aDriveInfo.iDriveAtt & KDriveAttTransaction)
|
sl@0
|
2108 |
{
|
sl@0
|
2109 |
deviceCharacteristics |= SQLITE_IOCAP_ATOMIC;
|
sl@0
|
2110 |
}
|
sl@0
|
2111 |
for(TInt i=0;i<(sizeof(KSqliteSectorSizeMap)/sizeof(KSqliteSectorSizeMap[0]));++i)
|
sl@0
|
2112 |
{
|
sl@0
|
2113 |
if(KSqliteSectorSizeMap[i].iSectorSize == aVolumeInfo.iBlockSize)
|
sl@0
|
2114 |
{
|
sl@0
|
2115 |
deviceCharacteristics |= KSqliteSectorSizeMap[i].iSqliteSectorSizeConstant;
|
sl@0
|
2116 |
break;
|
sl@0
|
2117 |
}
|
sl@0
|
2118 |
}
|
sl@0
|
2119 |
return deviceCharacteristics;
|
sl@0
|
2120 |
}
|
sl@0
|
2121 |
|
sl@0
|
2122 |
/**
|
sl@0
|
2123 |
Retrieves and returns the sector size of the drive referred by the aDriveInfo parameter.
|
sl@0
|
2124 |
The sector size must be a power of two.
|
sl@0
|
2125 |
The sector size is extracted only if aDriveInfo refers to a removable device, otherwise the
|
sl@0
|
2126 |
SQLITE_DEFAULT_SECTOR_SIZE value (512 bytes) will be used as a sector size.
|
sl@0
|
2127 |
|
sl@0
|
2128 |
@param aDriveInfo A TDriveInfo reference.
|
sl@0
|
2129 |
@param aVolumeInfo A TVolumeIOParamInfo reference.
|
sl@0
|
2130 |
|
sl@0
|
2131 |
@return The sector size of the drive referred by the aDriveInfo parameter.
|
sl@0
|
2132 |
|
sl@0
|
2133 |
@panic Sqlite 19 In _DEBUG mode - The sector size is negative, zero or is not a power of two.
|
sl@0
|
2134 |
|
sl@0
|
2135 |
@see TVfs::Open()
|
sl@0
|
2136 |
*/
|
sl@0
|
2137 |
/* static */ TInt TVfs::DoGetSectorSize(const TDriveInfo& aDriveInfo, const TVolumeIOParamInfo& aVolumeInfo)
|
sl@0
|
2138 |
{
|
sl@0
|
2139 |
//Initialize the sectorSize variable only if:
|
sl@0
|
2140 |
// - aDriveInfo refers to a removable drive
|
sl@0
|
2141 |
// - aVolumeInfo.iBlockSize > SQLITE_DEFAULT_SECTOR_SIZE;
|
sl@0
|
2142 |
// - aVolumeInfo.iBlockSize is power of 2;
|
sl@0
|
2143 |
TInt sectorSize = SQLITE_DEFAULT_SECTOR_SIZE;
|
sl@0
|
2144 |
if(aDriveInfo.iDriveAtt & KDriveAttRemovable)
|
sl@0
|
2145 |
{
|
sl@0
|
2146 |
if(aVolumeInfo.iBlockSize > SQLITE_DEFAULT_SECTOR_SIZE && (aVolumeInfo.iBlockSize & (aVolumeInfo.iBlockSize - 1)) == 0)
|
sl@0
|
2147 |
{
|
sl@0
|
2148 |
sectorSize = aVolumeInfo.iBlockSize;
|
sl@0
|
2149 |
}
|
sl@0
|
2150 |
}
|
sl@0
|
2151 |
__ASSERT_DEBUG(sectorSize > 0 && (sectorSize & (sectorSize - 1)) == 0, __SQLITEPANIC2(ESqliteOsPanicInternalError));
|
sl@0
|
2152 |
return sectorSize;
|
sl@0
|
2153 |
}
|
sl@0
|
2154 |
|
sl@0
|
2155 |
/**
|
sl@0
|
2156 |
Retrieves in a bit set the device characteristics of the device of the file referred by the aDbFile parameter.
|
sl@0
|
2157 |
Retrieves the sector size of the drive of the file referred by the aDbFile parameter.
|
sl@0
|
2158 |
The sector size and the device characteristics will be stored in iSectorSize and iDeviceCharacteristics TDbFile data members.
|
sl@0
|
2159 |
The stored values will be used later by TFileIo::DeviceCharacteristics() and TFileIo::SectorSize().
|
sl@0
|
2160 |
|
sl@0
|
2161 |
@param aDbFile Input/Output parameter. A TDriveInfo reference. The collected information will be stored in TDbDrive
|
sl@0
|
2162 |
data members.
|
sl@0
|
2163 |
@param aRecReadBufSize Output parameter. The recommended buffer size for optimised reading performance.
|
sl@0
|
2164 |
|
sl@0
|
2165 |
@return KErrNone, The operation has completed succesfully;
|
sl@0
|
2166 |
Note that other system-wide error codes may also be returned.
|
sl@0
|
2167 |
|
sl@0
|
2168 |
@panic Sqlite 19 In _DEBUG mode - TDbFile::iSectorSize has been already initialized.
|
sl@0
|
2169 |
@panic Sqlite 19 In _DEBUG mode - TDbFile::iDeviceCharacteristics has been already initialized.
|
sl@0
|
2170 |
|
sl@0
|
2171 |
@see TVfs::DoGetDeviceCharacteristics();
|
sl@0
|
2172 |
@see TVfs::DoGetSectorSize();
|
sl@0
|
2173 |
@see TVfs::Open()
|
sl@0
|
2174 |
@see TDbFile
|
sl@0
|
2175 |
@see TFileIo::DeviceCharacteristics()
|
sl@0
|
2176 |
@see TFileIo::SectorSize()
|
sl@0
|
2177 |
*/
|
sl@0
|
2178 |
/* static */ TInt TVfs::DoGetDeviceCharacteristicsAndSectorSize(TDbFile& aDbFile, TInt& aRecReadBufSize)
|
sl@0
|
2179 |
{
|
sl@0
|
2180 |
__ASSERT_DEBUG(aDbFile.iDeviceCharacteristics < 0, __SQLITEPANIC2(ESqliteOsPanicInternalError));
|
sl@0
|
2181 |
__ASSERT_DEBUG(aDbFile.iSectorSize <= 0, __SQLITEPANIC2(ESqliteOsPanicInternalError));
|
sl@0
|
2182 |
TInt driveNo;
|
sl@0
|
2183 |
TDriveInfo driveInfo;
|
sl@0
|
2184 |
__FS_CALL(EFsOpFileDrive, 0);
|
sl@0
|
2185 |
TInt err = aDbFile.iFileBuf.Drive(driveNo, driveInfo);
|
sl@0
|
2186 |
if(err != KErrNone)
|
sl@0
|
2187 |
{
|
sl@0
|
2188 |
return err;
|
sl@0
|
2189 |
}
|
sl@0
|
2190 |
TVolumeIOParamInfo volumeInfo;
|
sl@0
|
2191 |
err = TVfs::DoGetVolumeIoParamInfo(COsLayerData::Instance().iFs, driveNo, volumeInfo);
|
sl@0
|
2192 |
if(err != KErrNone)
|
sl@0
|
2193 |
{
|
sl@0
|
2194 |
return err;
|
sl@0
|
2195 |
}
|
sl@0
|
2196 |
aDbFile.iDeviceCharacteristics = TVfs::DoGetDeviceCharacteristics(driveInfo, volumeInfo);
|
sl@0
|
2197 |
aDbFile.iSectorSize = TVfs::DoGetSectorSize(driveInfo, volumeInfo);
|
sl@0
|
2198 |
aRecReadBufSize = volumeInfo.iRecReadBufSize;
|
sl@0
|
2199 |
SQLITE_TRACE_OS(OstTraceExt5(TRACE_INTERNALS, TVFS_DOGETGETDEVICECHARACTERISTICSANDSECTORSIZE, "OS;0x%X;TVfs::DoGetDeviceCharacteristicsAndSectorSize;driveNo=%d;sectorSize=%d;devCharact=0x%X;readBufSize=%d", (TUint)&aDbFile, driveNo, aDbFile.iSectorSize, (TUint)aDbFile.iDeviceCharacteristics, volumeInfo.iRecReadBufSize));
|
sl@0
|
2200 |
return KErrNone;
|
sl@0
|
2201 |
}
|
sl@0
|
2202 |
|
sl@0
|
2203 |
/**
|
sl@0
|
2204 |
SQLite OS porting layer API.
|
sl@0
|
2205 |
|
sl@0
|
2206 |
The behaviour of the RFile/RFile64::SetSize operation is not atomic for non-rugged drives.
|
sl@0
|
2207 |
When RFile/RFile64::SetSize() is called 2 operations occurs:-
|
sl@0
|
2208 |
|
sl@0
|
2209 |
1)The cluster chain of the file is updated.
|
sl@0
|
2210 |
2)The new file size is added to the file cache.
|
sl@0
|
2211 |
|
sl@0
|
2212 |
If a power loss occurs after a SetSize there is a chance that the cluster chain was updated
|
sl@0
|
2213 |
but the new file size is not yet flushed to the file. This puts the file into an inconsistent state.
|
sl@0
|
2214 |
This is most likely to occur in the journal file where the time between a SetSize and Flush can
|
sl@0
|
2215 |
be long.
|
sl@0
|
2216 |
|
sl@0
|
2217 |
For this reason this check is added when the file is opened to see if the end of the file can
|
sl@0
|
2218 |
be read straight away, if an error is returned then it is assumed that the SetSize has not be
|
sl@0
|
2219 |
completed previously. In this case the file is deleted and re-created.
|
sl@0
|
2220 |
|
sl@0
|
2221 |
@param aDbFile A pointer to a TDbFile instance, that contains the file handle.
|
sl@0
|
2222 |
@param aFname A string of 16-bit wide characters containing name of the file to be checked.
|
sl@0
|
2223 |
@param aFmode The mode in which the file is opened. These mode are documented in TFileMode.
|
sl@0
|
2224 |
|
sl@0
|
2225 |
@return KErrNone, The operation has completed succesfully;
|
sl@0
|
2226 |
Note that other system-wide error codes may also be returned.
|
sl@0
|
2227 |
@see TFileMode
|
sl@0
|
2228 |
@see TVfs::Open()
|
sl@0
|
2229 |
@see TDbFile
|
sl@0
|
2230 |
*/
|
sl@0
|
2231 |
/* static */ TInt TVfs::DoFileSizeCorruptionCheck(TDbFile& aDbFile, const TDesC& aFname, TInt aFmode)
|
sl@0
|
2232 |
{
|
sl@0
|
2233 |
const TInt KMinSize = 16;
|
sl@0
|
2234 |
TInt64 size;
|
sl@0
|
2235 |
TInt err = KErrNone ;
|
sl@0
|
2236 |
TBuf8<KMinSize> buf;
|
sl@0
|
2237 |
|
sl@0
|
2238 |
err = aDbFile.iFileBuf.Size(size);
|
sl@0
|
2239 |
if (err != KErrNone)
|
sl@0
|
2240 |
{
|
sl@0
|
2241 |
return err;
|
sl@0
|
2242 |
}
|
sl@0
|
2243 |
TBool IsMinFileSize = (size >= KMinSize);
|
sl@0
|
2244 |
|
sl@0
|
2245 |
if (IsMinFileSize)
|
sl@0
|
2246 |
{
|
sl@0
|
2247 |
err = aDbFile.iFileBuf.Read(size - KMinSize, buf);
|
sl@0
|
2248 |
}
|
sl@0
|
2249 |
|
sl@0
|
2250 |
if (err == KErrCorrupt || err == KErrEof || !IsMinFileSize)
|
sl@0
|
2251 |
{
|
sl@0
|
2252 |
COsLayerData& osLayerData = COsLayerData::Instance();
|
sl@0
|
2253 |
|
sl@0
|
2254 |
aDbFile.iFileBuf.Close();
|
sl@0
|
2255 |
__SQLITETRACE_OSEXPR(TInt err2 =) osLayerData.iFs.Delete(aFname);
|
sl@0
|
2256 |
SQLITE_TRACE_OS(OstTraceExt4(TRACE_INTERNALS, TVFS_DOFILESIZECORRUPTIONCHECK1, "OS;0x%X;TVfs::DoFileSizeCorruptionCheck;size=%lld;err=%d;deleteErr=%d", (TUint)&aDbFile, size, err, err2));
|
sl@0
|
2257 |
err = aDbFile.iFileBuf.Create(osLayerData.iFs, aFname, aFmode);
|
sl@0
|
2258 |
SQLITE_TRACE_OS(OstTraceExt2(TRACE_INTERNALS, TVFS_DOFILESIZECORRUPTIONCHECK2, "OS;0x%X;TVfs::DoFileSizeCorruptionCheck;createErr=%d", (TUint)&aDbFile, err));
|
sl@0
|
2259 |
}
|
sl@0
|
2260 |
return err;
|
sl@0
|
2261 |
}
|
sl@0
|
2262 |
|
sl@0
|
2263 |
/**
|
sl@0
|
2264 |
SQLite OS porting layer API.
|
sl@0
|
2265 |
|
sl@0
|
2266 |
Opens or creates a file which name is in the aFileName parameter.
|
sl@0
|
2267 |
If the function succeeds, the file handle and other related information will be stored in the place pointed by the
|
sl@0
|
2268 |
aDbFile parameter, a memory block of sizeof(TDbFile) size for which is allocated by the caller.
|
sl@0
|
2269 |
The function will also retrieve the sector size and the device characteristics and store them in aDbFile,
|
sl@0
|
2270 |
which is actually a TDbFile pointer, for later use.
|
sl@0
|
2271 |
|
sl@0
|
2272 |
@param aFileName Zero-terminated, UTF8 encoded file name.
|
sl@0
|
2273 |
If aFileName is NULL then a temporary file is created.
|
sl@0
|
2274 |
@param aDbFile Output parameter. The file handle and other related information will be stored there.
|
sl@0
|
2275 |
@param aFlags "Open/Create" input flags:
|
sl@0
|
2276 |
SQLITE_OPEN_DELETEONCLOSE,
|
sl@0
|
2277 |
SQLITE_OPEN_READWRITE,
|
sl@0
|
2278 |
SQLITE_OPEN_EXCLUSIVE,
|
sl@0
|
2279 |
SQLITE_OPEN_CREATE
|
sl@0
|
2280 |
@param aOutFlags "Open/Create" output flags:
|
sl@0
|
2281 |
SQLITE_OPEN_READWRITE,
|
sl@0
|
2282 |
SQLITE_OPEN_READONLY
|
sl@0
|
2283 |
|
sl@0
|
2284 |
@return SQLITE_CANTOPEN, The aFileName parameter cannot be converted to UTF16.
|
sl@0
|
2285 |
Any other file I/O error will also be reported as SQLITE_CANTOPEN;
|
sl@0
|
2286 |
SQLITE_IOERR_NOMEM, An out of memory condition has occured;
|
sl@0
|
2287 |
SQLITE_OK, The operation has completed successfully.
|
sl@0
|
2288 |
|
sl@0
|
2289 |
If the function succeeds, COsLayerData::SetOsErrorCode() is called with KErrNone, otherwise COsLayerData::SetOsErrorCode() is called
|
sl@0
|
2290 |
with the reported by the OS API error. The stored error code will be used later by the SQLite API caller.
|
sl@0
|
2291 |
|
sl@0
|
2292 |
@see COsLayerData::SetOsErrorCode()
|
sl@0
|
2293 |
@see TDbFile
|
sl@0
|
2294 |
*/
|
sl@0
|
2295 |
/* static */ int TVfs::Open(sqlite3_vfs* aVfs, const char* aFileName, sqlite3_file* aDbFile, int aFlags, int* aOutFlags)
|
sl@0
|
2296 |
{
|
sl@0
|
2297 |
__OS_CALL(EOsVfsOpen, 0, 0);
|
sl@0
|
2298 |
__OSTIME_COUNTER(TheOsCallTicks[EOsVfsOpen], ::OsCallProfile(EFalse, EOsVfsOpen), 0, 0, aDbFile, aFileName);
|
sl@0
|
2299 |
COsLayerData& osLayerData = COsLayerData::Instance();
|
sl@0
|
2300 |
new (aDbFile) TDbFile;
|
sl@0
|
2301 |
TDbFile& dbFile = ::DbFile(aDbFile);
|
sl@0
|
2302 |
if(!aFileName)
|
sl@0
|
2303 |
{
|
sl@0
|
2304 |
//It is to create and open a temp file if aFileName is NULL. In this case,
|
sl@0
|
2305 |
//we will defer the file creation util it is needed.
|
sl@0
|
2306 |
dbFile.pMethods = &TheFileIoApi;
|
sl@0
|
2307 |
dbFile.iIsFileCreated = EFalse;
|
sl@0
|
2308 |
if(aOutFlags)
|
sl@0
|
2309 |
{
|
sl@0
|
2310 |
*aOutFlags = SQLITE_OPEN_READWRITE;
|
sl@0
|
2311 |
}
|
sl@0
|
2312 |
osLayerData.SetOsErrorCode(KErrNone);
|
sl@0
|
2313 |
return SQLITE_OK;
|
sl@0
|
2314 |
}
|
sl@0
|
2315 |
TFileName fname;
|
sl@0
|
2316 |
if(!::ConvertToUnicode(aFileName, fname))
|
sl@0
|
2317 |
{
|
sl@0
|
2318 |
osLayerData.SetOsErrorCode(KErrBadName);
|
sl@0
|
2319 |
return SQLITE_CANTOPEN;
|
sl@0
|
2320 |
}
|
sl@0
|
2321 |
SQLITE_TRACE_OS(OstTraceExt3(TRACE_INTERNALS, TVFS_OPEN_ENTRY, "OS-Entry;0x%X;TVfs::Open;fname=%S;aFlags=0x%X", (TUint)&aDbFile, __SQLITEPRNSTR(fname), (TUint)aFlags));
|
sl@0
|
2322 |
TFhStrType fhStrType = aFileName ? ::FhStringProps(aFileName) : ENotFhStr;
|
sl@0
|
2323 |
TInt err = KErrNone;
|
sl@0
|
2324 |
if(fhStrType == EFhMainDbStr)
|
sl@0
|
2325 |
{//Main db file, open from handle
|
sl@0
|
2326 |
const RMessage2* msg;
|
sl@0
|
2327 |
TBool readOnly;
|
sl@0
|
2328 |
osLayerData.RetrieveAndResetFhData(msg, readOnly);
|
sl@0
|
2329 |
err = msg != NULL ? TVfs::DoOpenFromHandle(dbFile, *msg, readOnly) : KErrGeneral;
|
sl@0
|
2330 |
}
|
sl@0
|
2331 |
else
|
sl@0
|
2332 |
{
|
sl@0
|
2333 |
if(fhStrType == EFhStr)
|
sl@0
|
2334 |
{//Not the main db file. Remove invalid characters in the file name
|
sl@0
|
2335 |
::FhConvertToFileName(fname, osLayerData.iSysPrivDir);//If fname does not have a path, iSysPrivDir will be used
|
sl@0
|
2336 |
}
|
sl@0
|
2337 |
TInt fmode = EFileRead;
|
sl@0
|
2338 |
if(aFlags & SQLITE_OPEN_READWRITE)
|
sl@0
|
2339 |
{
|
sl@0
|
2340 |
fmode |= EFileWrite;
|
sl@0
|
2341 |
}
|
sl@0
|
2342 |
if(aFlags & SQLITE_OPEN_EXCLUSIVE)
|
sl@0
|
2343 |
{
|
sl@0
|
2344 |
fmode |= EFileShareExclusive;
|
sl@0
|
2345 |
}
|
sl@0
|
2346 |
if(aFlags & SQLITE_OPEN_DELETEONCLOSE)
|
sl@0
|
2347 |
{
|
sl@0
|
2348 |
fmode |= EDeleteOnClose;
|
sl@0
|
2349 |
}
|
sl@0
|
2350 |
err = KErrAccessDenied;
|
sl@0
|
2351 |
TInt prevErr = KErrNone;
|
sl@0
|
2352 |
if(aFlags & SQLITE_OPEN_CREATE)
|
sl@0
|
2353 |
{
|
sl@0
|
2354 |
__FS_CALL(EFsOpFileCreate, 0);
|
sl@0
|
2355 |
prevErr = err = dbFile.iFileBuf.Create(osLayerData.iFs, fname, fmode);
|
sl@0
|
2356 |
}
|
sl@0
|
2357 |
if(err != KErrNone && err != KErrNoMemory && err != KErrDiskFull)
|
sl@0
|
2358 |
{
|
sl@0
|
2359 |
__FS_CALL(EFsOpFileOpen, 0);
|
sl@0
|
2360 |
err = dbFile.iFileBuf.Open(osLayerData.iFs, fname, fmode);
|
sl@0
|
2361 |
|
sl@0
|
2362 |
if(err == KErrNone && (aFlags & KJournalFileTypeBitMask))
|
sl@0
|
2363 |
{
|
sl@0
|
2364 |
err = TVfs::DoFileSizeCorruptionCheck(dbFile, fname, fmode);
|
sl@0
|
2365 |
}
|
sl@0
|
2366 |
}
|
sl@0
|
2367 |
if((err != KErrNone && err != KErrNoMemory && err != KErrDiskFull) && (aFlags & SQLITE_OPEN_READWRITE))
|
sl@0
|
2368 |
{
|
sl@0
|
2369 |
aFlags &= ~SQLITE_OPEN_READWRITE;
|
sl@0
|
2370 |
aFlags |= SQLITE_OPEN_READONLY;
|
sl@0
|
2371 |
fmode &= ~EFileWrite;
|
sl@0
|
2372 |
__FS_CALL(EFsOpFileOpen, 0);
|
sl@0
|
2373 |
err = dbFile.iFileBuf.Open(osLayerData.iFs, fname, fmode);
|
sl@0
|
2374 |
}
|
sl@0
|
2375 |
if(err != KErrNone && prevErr == KErrAccessDenied)
|
sl@0
|
2376 |
{
|
sl@0
|
2377 |
err = KErrAccessDenied;
|
sl@0
|
2378 |
}
|
sl@0
|
2379 |
}
|
sl@0
|
2380 |
TInt recReadBufSize = -1;
|
sl@0
|
2381 |
if(err == KErrNone)
|
sl@0
|
2382 |
{
|
sl@0
|
2383 |
err = TVfs::DoGetDeviceCharacteristicsAndSectorSize(dbFile, recReadBufSize);
|
sl@0
|
2384 |
}
|
sl@0
|
2385 |
osLayerData.SetOsErrorCode(err);
|
sl@0
|
2386 |
if(err != KErrNone)
|
sl@0
|
2387 |
{
|
sl@0
|
2388 |
__FS_CALL(EFsOpFileClose, 0);
|
sl@0
|
2389 |
dbFile.iFileBuf.Close();
|
sl@0
|
2390 |
}
|
sl@0
|
2391 |
else
|
sl@0
|
2392 |
{
|
sl@0
|
2393 |
dbFile.pMethods = &TheFileIoApi;
|
sl@0
|
2394 |
if(fhStrType != EFhMainDbStr)
|
sl@0
|
2395 |
{
|
sl@0
|
2396 |
dbFile.iReadOnly = !(aFlags & SQLITE_OPEN_READWRITE);
|
sl@0
|
2397 |
}
|
sl@0
|
2398 |
if(aOutFlags)
|
sl@0
|
2399 |
{
|
sl@0
|
2400 |
*aOutFlags = dbFile.iReadOnly ? SQLITE_OPEN_READONLY : SQLITE_OPEN_READWRITE;
|
sl@0
|
2401 |
}
|
sl@0
|
2402 |
(void)dbFile.iFileBuf.SetReadAheadSize(dbFile.iSectorSize, recReadBufSize);
|
sl@0
|
2403 |
}
|
sl@0
|
2404 |
#ifdef _SQLPROFILER
|
sl@0
|
2405 |
dbFile.iIsJournal = aFlags & KJournalFileTypeBitMask;
|
sl@0
|
2406 |
#endif
|
sl@0
|
2407 |
TInt sqliteErr = ::Os2SqliteErr(err, SQLITE_CANTOPEN);
|
sl@0
|
2408 |
SQLITE_TRACE_OS(OstTraceExt4(TRACE_INTERNALS, TVFS_OPEN_EXIT, "OS-Exit;0x%X;TVfs::Open;outFlags=0x%X;err=%d;sqliteErr=%d", (TUint)&aDbFile, aOutFlags ? (TUint)*aOutFlags : 0, err, sqliteErr));
|
sl@0
|
2409 |
return sqliteErr;
|
sl@0
|
2410 |
}
|
sl@0
|
2411 |
|
sl@0
|
2412 |
/**
|
sl@0
|
2413 |
SQLite OS porting layer API.
|
sl@0
|
2414 |
|
sl@0
|
2415 |
Deletes a file which name is in the aFileName parameter.
|
sl@0
|
2416 |
|
sl@0
|
2417 |
@param aFileName Zero-terminated, UTF8 encoded file name.
|
sl@0
|
2418 |
|
sl@0
|
2419 |
@return SQLITE_ERROR, The aFileName parameter cannot be converted to UTF16.
|
sl@0
|
2420 |
The file name refers to a private secure database;
|
sl@0
|
2421 |
SQLITE_IOERR_NOMEM, An out of memory condition has occured;
|
sl@0
|
2422 |
SQLITE_IOERR_DELETE,The delete file operation has failed;
|
sl@0
|
2423 |
SQLITE_OK, The operation has completed successfully.
|
sl@0
|
2424 |
|
sl@0
|
2425 |
If the function succeeds, COsLayerData::SetOsErrorCode() is called with KErrNone, otherwise COsLayerData::SetOsErrorCode() is called
|
sl@0
|
2426 |
with the reported by the OS API error. The stored error code will be used later by the SQLite API caller.
|
sl@0
|
2427 |
|
sl@0
|
2428 |
@see COsLayerData::SetOsErrorCode()
|
sl@0
|
2429 |
*/
|
sl@0
|
2430 |
/* static */ int TVfs::Delete(sqlite3_vfs* aVfs, const char* aFileName, int /*aSyncDir*/)
|
sl@0
|
2431 |
{
|
sl@0
|
2432 |
__OS_CALL(EOsVfsDelete, 0, 0);
|
sl@0
|
2433 |
__OSTIME_COUNTER(TheOsCallTicks[EOsVfsDelete], ::OsCallProfile(EFalse, EOsVfsDelete), 0, 0, 0, aFileName);
|
sl@0
|
2434 |
COsLayerData& osLayerData = COsLayerData::Instance();
|
sl@0
|
2435 |
TBuf<KMaxFileName + 1> fname;
|
sl@0
|
2436 |
if(!::ConvertToUnicode(aFileName, fname))
|
sl@0
|
2437 |
{
|
sl@0
|
2438 |
osLayerData.SetOsErrorCode(KErrBadName);
|
sl@0
|
2439 |
SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, TVFS_DELETE1, "OS;0;TVfs::Delete;err=KErrBadName"));
|
sl@0
|
2440 |
return SQLITE_ERROR;
|
sl@0
|
2441 |
}
|
sl@0
|
2442 |
SQLITE_TRACE_OS(OstTraceExt1(TRACE_INTERNALS, TVFS_DELETE_ENTRY, "OS-Entry;0;TVfs::Delete;fname=%S", __SQLITEPRNSTR(fname)));
|
sl@0
|
2443 |
TFhStrType fhStrType = FhStringProps(aFileName);
|
sl@0
|
2444 |
if(fhStrType == EFhMainDbStr)
|
sl@0
|
2445 |
{//Deleting files not in your own private data cage - not allowed!
|
sl@0
|
2446 |
osLayerData.SetOsErrorCode(KErrPermissionDenied);
|
sl@0
|
2447 |
SQLITE_TRACE_OS(OstTraceExt1(TRACE_INTERNALS, TVFS_DELETE2, "OS;0;TVfs::Delete;err=KErrPermissionDenied;fname=%S", __SQLITEPRNSTR(fname)));
|
sl@0
|
2448 |
return SQLITE_ERROR;
|
sl@0
|
2449 |
}
|
sl@0
|
2450 |
if(fhStrType == EFhStr)
|
sl@0
|
2451 |
{
|
sl@0
|
2452 |
::FhConvertToFileName(fname, osLayerData.iSysPrivDir);//If fname does not have a path, iSysPrivDir will be used
|
sl@0
|
2453 |
}
|
sl@0
|
2454 |
__FS_CALL(EFsOpFileDelete, 0);
|
sl@0
|
2455 |
TInt err = osLayerData.iFs.Delete(fname);
|
sl@0
|
2456 |
osLayerData.SetOsErrorCode(err);
|
sl@0
|
2457 |
TInt sqliteErr = ::Os2SqliteErr(err, SQLITE_IOERR_DELETE);
|
sl@0
|
2458 |
SQLITE_TRACE_OS(OstTraceExt2(TRACE_INTERNALS, TVFS_DELETE_EXIT, "OS-Exit;0;TVfs::Delete;err=%d;sqliteErr=%d", err, sqliteErr));
|
sl@0
|
2459 |
return sqliteErr;
|
sl@0
|
2460 |
}
|
sl@0
|
2461 |
|
sl@0
|
2462 |
/**
|
sl@0
|
2463 |
SQLite OS porting layer API.
|
sl@0
|
2464 |
|
sl@0
|
2465 |
Retrieves an information about a file which name is in the aFileName parameter.
|
sl@0
|
2466 |
The requested information type can be: does the file exist, is the file read-only or read/write.
|
sl@0
|
2467 |
|
sl@0
|
2468 |
@param aFileName Zero-terminated, UTF8 encoded file name.
|
sl@0
|
2469 |
@param aFlags This parameter can be one of: SQLITE_ACCESS_READ, SQLITE_ACCESS_EXISTS or SQLITE_ACCESS_READWRITE.
|
sl@0
|
2470 |
@param aResOut Output parameter, set to 1 if the tested condition is true, 0 otherwise.
|
sl@0
|
2471 |
|
sl@0
|
2472 |
@return SQLITE_OK, The call has completed successfully,
|
sl@0
|
2473 |
SQLITE_IOERR_NOMEM, An out of memory conditon has occured,
|
sl@0
|
2474 |
SQLITE_IOERR_ACCESS,File I/O error;
|
sl@0
|
2475 |
|
sl@0
|
2476 |
If the function succeeds, COsLayerData::SetOsErrorCode() is called with KErrNone, otherwise COsLayerData::SetOsErrorCode() is called
|
sl@0
|
2477 |
with the reported by the OS API error. The stored error code will be used later by the SQLite API caller.
|
sl@0
|
2478 |
|
sl@0
|
2479 |
@see COsLayerData::SetOsErrorCode()
|
sl@0
|
2480 |
*/
|
sl@0
|
2481 |
/* static */ int TVfs::Access(sqlite3_vfs* aVfs, const char* aFileName, int aFlags, int* aResOut)
|
sl@0
|
2482 |
{
|
sl@0
|
2483 |
__OS_CALL(EOsVfsAccess, 0, 0);
|
sl@0
|
2484 |
__OSTIME_COUNTER(TheOsCallTicks[EOsVfsAccess], ::OsCallProfile(EFalse, EOsVfsAccess), aFlags, 0, 0, aFileName);
|
sl@0
|
2485 |
COsLayerData& osLayerData = COsLayerData::Instance();
|
sl@0
|
2486 |
TBuf<KMaxFileName + 1> fname;
|
sl@0
|
2487 |
if(!::ConvertToUnicode(aFileName, fname))
|
sl@0
|
2488 |
{
|
sl@0
|
2489 |
osLayerData.SetOsErrorCode(KErrGeneral);
|
sl@0
|
2490 |
SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, TVFS_ACCESS1, "OS;0;TVfs::Access;err=KErrGeneral"));
|
sl@0
|
2491 |
return SQLITE_IOERR_ACCESS;
|
sl@0
|
2492 |
}
|
sl@0
|
2493 |
SQLITE_TRACE_OS(OstTraceExt2(TRACE_INTERNALS, TVFS_ACCESS_ENTRY, "OS-Entry;0;TVfs::Access;fname=%S;aFlags=0x%X", __SQLITEPRNSTR(fname), (TUint)aFlags));
|
sl@0
|
2494 |
TFhStrType fhStrType = ::FhStringProps(aFileName);
|
sl@0
|
2495 |
if(fhStrType == EFhStr)
|
sl@0
|
2496 |
{
|
sl@0
|
2497 |
::FhConvertToFileName(fname, osLayerData.iSysPrivDir);//If fname does not have a path, iSysPrivDir will be used
|
sl@0
|
2498 |
}
|
sl@0
|
2499 |
TEntry entry;
|
sl@0
|
2500 |
__FS_CALL(EFsOpFsEntry, 0);
|
sl@0
|
2501 |
TInt err = osLayerData.iFs.Entry(fname, entry);
|
sl@0
|
2502 |
if(aFlags == SQLITE_ACCESS_EXISTS && err == KErrNotFound)
|
sl@0
|
2503 |
{
|
sl@0
|
2504 |
osLayerData.SetOsErrorCode(KErrNone);
|
sl@0
|
2505 |
*aResOut = 0;
|
sl@0
|
2506 |
SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, TVFS_ACCESS_EXIT1, "OS-Exit;0;TVfs::Access;Exists-NoFound"));
|
sl@0
|
2507 |
return SQLITE_OK;
|
sl@0
|
2508 |
}
|
sl@0
|
2509 |
if(err != KErrNone)
|
sl@0
|
2510 |
{
|
sl@0
|
2511 |
osLayerData.SetOsErrorCode(err);
|
sl@0
|
2512 |
SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, TVFS_ACCESS_EXIT2, "OS-Exit;0;TVfs::Access;err=%d", err));
|
sl@0
|
2513 |
return err == KErrNoMemory ? SQLITE_IOERR_NOMEM : SQLITE_IOERR_ACCESS;
|
sl@0
|
2514 |
}
|
sl@0
|
2515 |
*aResOut = 0;
|
sl@0
|
2516 |
switch(aFlags)
|
sl@0
|
2517 |
{
|
sl@0
|
2518 |
case SQLITE_ACCESS_READ:
|
sl@0
|
2519 |
*aResOut = entry.IsReadOnly();
|
sl@0
|
2520 |
break;
|
sl@0
|
2521 |
case SQLITE_ACCESS_EXISTS:
|
sl@0
|
2522 |
*aResOut = 1;
|
sl@0
|
2523 |
break;
|
sl@0
|
2524 |
case SQLITE_ACCESS_READWRITE:
|
sl@0
|
2525 |
*aResOut = !entry.IsReadOnly();
|
sl@0
|
2526 |
break;
|
sl@0
|
2527 |
default:
|
sl@0
|
2528 |
break;
|
sl@0
|
2529 |
}
|
sl@0
|
2530 |
osLayerData.SetOsErrorCode(KErrNone);
|
sl@0
|
2531 |
SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, TVFS_ACCESS_EXIT3, "OS-Exit;0;TVfs::Access;aResOut=%d", *aResOut));
|
sl@0
|
2532 |
return SQLITE_OK;
|
sl@0
|
2533 |
}
|
sl@0
|
2534 |
|
sl@0
|
2535 |
/**
|
sl@0
|
2536 |
SQLite OS porting layer API.
|
sl@0
|
2537 |
|
sl@0
|
2538 |
Accepts UTF8 encoded, zero-terminated file as an input argument in the aRelative parameter
|
sl@0
|
2539 |
and constructs the full file path in the aBuf output parameter.
|
sl@0
|
2540 |
|
sl@0
|
2541 |
If the format of aRelative argument is <[SID]FileName.[EXT]>, then the database file name will be
|
sl@0
|
2542 |
treated as a name of a secure database file which has to be created/opened in the server's private
|
sl@0
|
2543 |
directory on the system drive.
|
sl@0
|
2544 |
|
sl@0
|
2545 |
If the format of aRelative argument is <Drive:[SID]FileName.[EXT]>, then the database file name
|
sl@0
|
2546 |
will be treated as a name of a secure database file which has to be created/opened in the server's
|
sl@0
|
2547 |
private directory on <Drive:> drive.
|
sl@0
|
2548 |
|
sl@0
|
2549 |
If the format of aRelative argument is <Drive:\Path\FileName.[EXT]>, then the database file name
|
sl@0
|
2550 |
will be treated as a name of a non-secure database file in <Drive:\Path\> directory.
|
sl@0
|
2551 |
If aRelative contains file handles, then it will be treated as a name of a file belonging to server's
|
sl@0
|
2552 |
private data cage.
|
sl@0
|
2553 |
|
sl@0
|
2554 |
@param aRelative The input file name, zero-terminated, UTF8 encoded.
|
sl@0
|
2555 |
@param aBufLen The output buffer length.
|
sl@0
|
2556 |
@param aBuf Output buffer for the constructed full file name path. The allocated buffer length must be at least aBufLen bytes.
|
sl@0
|
2557 |
|
sl@0
|
2558 |
@return SQLITE_ERROR, The aRelative parameter is NULL or cannot be converted to UTF16;
|
sl@0
|
2559 |
SQLITE_OK The operation has completed successfully.
|
sl@0
|
2560 |
*/
|
sl@0
|
2561 |
/* static */ int TVfs::FullPathName(sqlite3_vfs* aVfs, const char* aRelative, int aBufLen, char* aBuf)
|
sl@0
|
2562 |
{
|
sl@0
|
2563 |
__OS_CALL(EOsVfsFullPathName, 0, 0);
|
sl@0
|
2564 |
__OSTIME_COUNTER(TheOsCallTicks[EOsVfsFullPathName], ::OsCallProfile(EFalse, EOsVfsFullPathName), aBufLen, 0, 0, aRelative);
|
sl@0
|
2565 |
COsLayerData& osLayerData = COsLayerData::Instance();
|
sl@0
|
2566 |
osLayerData.StoreFhData(NULL, EFalse);
|
sl@0
|
2567 |
//Convert the received file name to UTF16
|
sl@0
|
2568 |
TBuf<KMaxFileName + 1> fname;
|
sl@0
|
2569 |
if(!::ConvertToUnicode(aRelative, fname))
|
sl@0
|
2570 |
{
|
sl@0
|
2571 |
SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, TVFS_FULLPATHNAME1, "OS;0;TVfs::FullPathName;err=SQLITE_ERROR"));
|
sl@0
|
2572 |
return SQLITE_ERROR;
|
sl@0
|
2573 |
}
|
sl@0
|
2574 |
SQLITE_TRACE_OS(OstTraceExt2(TRACE_INTERNALS, TVFS_FULLPATHNAME_ENTRY, "OS-Entry;0;TVfs::FullPathName;fname=%S;aBufLen=%d", __SQLITEPRNSTR(fname), aBufLen));
|
sl@0
|
2575 |
//Zero-terminate the converted file name
|
sl@0
|
2576 |
fname.Append(TChar(0));
|
sl@0
|
2577 |
TParse parse;
|
sl@0
|
2578 |
TFhStrType strType = ::FhStringProps(aRelative);//Detect string type - it may not be a real file name
|
sl@0
|
2579 |
if(strType == EFhMainDbStr)
|
sl@0
|
2580 |
{//The additonal information has to be extracted and fname reformatted, because SQLITE will
|
sl@0
|
2581 |
//use the returned full file name when making a decission to share the cache.
|
sl@0
|
2582 |
::FhExtractAndStore(fname);
|
sl@0
|
2583 |
(void)parse.Set(fname, 0, 0);//the file name has to be verified by the file server
|
sl@0
|
2584 |
}
|
sl@0
|
2585 |
else
|
sl@0
|
2586 |
{
|
sl@0
|
2587 |
(void)parse.Set(fname, &osLayerData.iSysPrivDir, 0);//If fname does not have a path, iSysPrivDir will be used
|
sl@0
|
2588 |
}
|
sl@0
|
2589 |
TPtr8 dest8(reinterpret_cast <TUint8*> (aBuf), aBufLen);
|
sl@0
|
2590 |
if(!::ConvertFromUnicode(parse.FullName(), dest8))
|
sl@0
|
2591 |
{//Zero the stored fh data, because it has been initialized by the FhExtractAndStore(fname) call (couple of lines above)
|
sl@0
|
2592 |
osLayerData.StoreFhData(NULL, EFalse);
|
sl@0
|
2593 |
SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, TVFS_FULLPATHNAME_EXIT1, "OS-Exit;0;TVfs::FullPathName;err=SQLITE_ERROR"));
|
sl@0
|
2594 |
return SQLITE_ERROR;
|
sl@0
|
2595 |
}
|
sl@0
|
2596 |
SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, TVFS_FULLPATHNAME_EXIT2, "OS-Exit;0;TVfs::FullPathName;err=SQLITE_OK"));
|
sl@0
|
2597 |
return SQLITE_OK;
|
sl@0
|
2598 |
}
|
sl@0
|
2599 |
|
sl@0
|
2600 |
/**
|
sl@0
|
2601 |
SQLite OS porting layer API.
|
sl@0
|
2602 |
|
sl@0
|
2603 |
Generates a set of random numbers and stores them in the aBuf output parameter.
|
sl@0
|
2604 |
|
sl@0
|
2605 |
@param aBufLen The output buffer length.
|
sl@0
|
2606 |
@param aBuf Output buffer for the generated random numbers. The allocated buffer length must be at least aBufLen bytes.
|
sl@0
|
2607 |
|
sl@0
|
2608 |
@return The length of the used part of the output buffer.
|
sl@0
|
2609 |
*/
|
sl@0
|
2610 |
/* static */ int TVfs::Randomness(sqlite3_vfs* aVfs, int aBufLen, char* aBuf)
|
sl@0
|
2611 |
{
|
sl@0
|
2612 |
__OS_CALL(EOsVfsRandomness, 0, 0);
|
sl@0
|
2613 |
__OSTIME_COUNTER(TheOsCallTicks[EOsVfsRandomness], ::OsCallProfile(EFalse, EOsVfsRandomness), aBufLen, 0, 0, 0);
|
sl@0
|
2614 |
COsLayerData& osLayerData = COsLayerData::Instance();
|
sl@0
|
2615 |
const TInt KRandIterations = aBufLen / sizeof(int);
|
sl@0
|
2616 |
for(TInt i=0;i<KRandIterations;++i)
|
sl@0
|
2617 |
{
|
sl@0
|
2618 |
TInt val = Math::Rand(osLayerData.iSeed);
|
sl@0
|
2619 |
Mem::Copy(&aBuf[i * sizeof(int)], &val, sizeof(val));
|
sl@0
|
2620 |
}
|
sl@0
|
2621 |
return KRandIterations * sizeof(int);
|
sl@0
|
2622 |
}
|
sl@0
|
2623 |
|
sl@0
|
2624 |
/**
|
sl@0
|
2625 |
SQLite OS porting layer API.
|
sl@0
|
2626 |
|
sl@0
|
2627 |
Sleeps for aMicrosec microseconds.
|
sl@0
|
2628 |
|
sl@0
|
2629 |
@param aMicrosec The sleep interval in microseconds.
|
sl@0
|
2630 |
|
sl@0
|
2631 |
@return The aMicrosec value.
|
sl@0
|
2632 |
*/
|
sl@0
|
2633 |
/* static */ int TVfs::Sleep(sqlite3_vfs* aVfs, int aMicrosec)
|
sl@0
|
2634 |
{
|
sl@0
|
2635 |
__OS_CALL(EOsVfsSleep, 0, 0);
|
sl@0
|
2636 |
__OSTIME_COUNTER(TheOsCallTicks[EOsVfsSleep], ::OsCallProfile(EFalse, EOsVfsSleep), aMicrosec, 0, 0, 0);
|
sl@0
|
2637 |
User::AfterHighRes(TTimeIntervalMicroSeconds32(aMicrosec));
|
sl@0
|
2638 |
return aMicrosec;
|
sl@0
|
2639 |
}
|
sl@0
|
2640 |
|
sl@0
|
2641 |
/**
|
sl@0
|
2642 |
SQLite OS porting layer API.
|
sl@0
|
2643 |
|
sl@0
|
2644 |
Retrieves the current date and time.
|
sl@0
|
2645 |
|
sl@0
|
2646 |
@param aNow Output parameter, where the data and time value will be stored.
|
sl@0
|
2647 |
SQLite processes all times and dates as Julian Day numbers and
|
sl@0
|
2648 |
aNow parameter will contain the julian date and time.
|
sl@0
|
2649 |
|
sl@0
|
2650 |
@return 0.
|
sl@0
|
2651 |
*/
|
sl@0
|
2652 |
/* static */ int TVfs::CurrentTime(sqlite3_vfs* aVfs, double* aNow)
|
sl@0
|
2653 |
{
|
sl@0
|
2654 |
__OS_CALL(EOsVfsCurrentTime, 0, 0);
|
sl@0
|
2655 |
__OSTIME_COUNTER(TheOsCallTicks[EOsVfsCurrentTime], ::OsCallProfile(EFalse, EOsVfsCurrentTime), 0, 0, 0, 0);
|
sl@0
|
2656 |
TTime now;
|
sl@0
|
2657 |
now.UniversalTime();
|
sl@0
|
2658 |
TDateTime date = now.DateTime();
|
sl@0
|
2659 |
TInt year = date.Year();
|
sl@0
|
2660 |
TInt month = date.Month() + 1;
|
sl@0
|
2661 |
TInt day = date.Day() + 1;
|
sl@0
|
2662 |
|
sl@0
|
2663 |
//Calculate the Julian days
|
sl@0
|
2664 |
TInt jd = day - 32076 +
|
sl@0
|
2665 |
1461*(year + 4800 + (month - 14)/12)/4 +
|
sl@0
|
2666 |
367*(month - 2 - (month - 14)/12*12)/12 -
|
sl@0
|
2667 |
3*((year + 4900 + (month - 14)/12)/100)/4;
|
sl@0
|
2668 |
|
sl@0
|
2669 |
*aNow = jd;
|
sl@0
|
2670 |
|
sl@0
|
2671 |
// Add the fractional hours, mins and seconds
|
sl@0
|
2672 |
*aNow += (date.Hour() + 12.0) / 24.0;
|
sl@0
|
2673 |
*aNow += date.Minute() / 1440.0;
|
sl@0
|
2674 |
*aNow += date.Second() / 86400.0;
|
sl@0
|
2675 |
|
sl@0
|
2676 |
return 0;
|
sl@0
|
2677 |
}
|
sl@0
|
2678 |
|
sl@0
|
2679 |
/**
|
sl@0
|
2680 |
SQLite OS porting layer API.
|
sl@0
|
2681 |
|
sl@0
|
2682 |
Retrieves a text description of the last OS error.
|
sl@0
|
2683 |
Note: the method has a default "no-op" implementation at the moment.
|
sl@0
|
2684 |
|
sl@0
|
2685 |
@return 0.
|
sl@0
|
2686 |
*/
|
sl@0
|
2687 |
/* static */int TVfs::GetLastError(sqlite3_vfs* aVfs, int /*aBufLen*/, char* /*aBuf*/)
|
sl@0
|
2688 |
{
|
sl@0
|
2689 |
__OS_CALL(EOsVfsGetLastError, 0, 0);
|
sl@0
|
2690 |
__OSTIME_COUNTER(TheOsCallTicks[EOsVfsGetLastError], ::OsCallProfile(EFalse, EOsVfsGetLastError), 0, 0, 0, 0);
|
sl@0
|
2691 |
return 0;
|
sl@0
|
2692 |
}
|
sl@0
|
2693 |
|
sl@0
|
2694 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
2695 |
/////////////////////////////////// Memory allocation functions /////////////////////////////////////////////////////
|
sl@0
|
2696 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
2697 |
|
sl@0
|
2698 |
/**
|
sl@0
|
2699 |
SQLite OS porting layer API.
|
sl@0
|
2700 |
|
sl@0
|
2701 |
Memory allocation routine.
|
sl@0
|
2702 |
|
sl@0
|
2703 |
EXPORT_C required to match IMPORT_C in stdlib.h and avoid compiler warning
|
sl@0
|
2704 |
|
sl@0
|
2705 |
@internalComponent
|
sl@0
|
2706 |
*/
|
sl@0
|
2707 |
extern "C" EXPORT_C void* sqlite3SymbianMalloc(size_t aSize)
|
sl@0
|
2708 |
{
|
sl@0
|
2709 |
__MEM_CALL(EMemOpAlloc, aSize, 0);
|
sl@0
|
2710 |
return Allocator().Alloc(aSize);
|
sl@0
|
2711 |
}
|
sl@0
|
2712 |
|
sl@0
|
2713 |
/**
|
sl@0
|
2714 |
SQLite OS porting layer API.
|
sl@0
|
2715 |
|
sl@0
|
2716 |
Memory reallocation routine.
|
sl@0
|
2717 |
|
sl@0
|
2718 |
EXPORT_C required to match IMPORT_C in stdlib.h and avoid compiler warning
|
sl@0
|
2719 |
|
sl@0
|
2720 |
@internalComponent
|
sl@0
|
2721 |
*/
|
sl@0
|
2722 |
extern "C" EXPORT_C void* sqlite3SymbianRealloc(void* aPtr, size_t aSize)
|
sl@0
|
2723 |
{
|
sl@0
|
2724 |
#ifdef _SQLPROFILER
|
sl@0
|
2725 |
TInt size = Allocator().AllocLen(aPtr);
|
sl@0
|
2726 |
__MEM_CALL(EMemOpRealloc, aSize, size);
|
sl@0
|
2727 |
#endif
|
sl@0
|
2728 |
return Allocator().ReAlloc(aPtr, aSize);
|
sl@0
|
2729 |
}
|
sl@0
|
2730 |
|
sl@0
|
2731 |
/**
|
sl@0
|
2732 |
SQLite OS porting layer API.
|
sl@0
|
2733 |
|
sl@0
|
2734 |
Memory free routine.
|
sl@0
|
2735 |
|
sl@0
|
2736 |
EXPORT_C required to match IMPORT_C in stdlib.h and avoid compiler warning
|
sl@0
|
2737 |
|
sl@0
|
2738 |
@internalComponent
|
sl@0
|
2739 |
*/
|
sl@0
|
2740 |
extern "C" EXPORT_C void sqlite3SymbianFree(void* aPtr)
|
sl@0
|
2741 |
{
|
sl@0
|
2742 |
#ifdef _SQLPROFILER
|
sl@0
|
2743 |
TInt size = Allocator().AllocLen(aPtr);
|
sl@0
|
2744 |
__MEM_CALL(EMemOpFree, size, 0);
|
sl@0
|
2745 |
#endif
|
sl@0
|
2746 |
Allocator().Free(aPtr);
|
sl@0
|
2747 |
}
|
sl@0
|
2748 |
|
sl@0
|
2749 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
2750 |
/////////////////////////////////// SQLite init/release functions ///////////////////////////////////////////////////
|
sl@0
|
2751 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
2752 |
|
sl@0
|
2753 |
/**
|
sl@0
|
2754 |
Registers the single sqlite3_vfs instance ("TheVfsApi" global variable) by calling sqlite3_vfs_register()
|
sl@0
|
2755 |
*/
|
sl@0
|
2756 |
extern "C" int sqlite3_os_init(void)
|
sl@0
|
2757 |
{
|
sl@0
|
2758 |
return sqlite3_vfs_register(&TheVfsApi, 1);//"1" means - make TheVfsApi to be the default VFS object
|
sl@0
|
2759 |
}
|
sl@0
|
2760 |
|
sl@0
|
2761 |
/**
|
sl@0
|
2762 |
Unregisters the single sqlite3_vfs instance ("TheVfsApi" global variable) by calling sqlite3_vfs_unregister()
|
sl@0
|
2763 |
*/
|
sl@0
|
2764 |
extern "C" int sqlite3_os_end(void)
|
sl@0
|
2765 |
{
|
sl@0
|
2766 |
return sqlite3_vfs_unregister(&TheVfsApi);
|
sl@0
|
2767 |
}
|
sl@0
|
2768 |
|
sl@0
|
2769 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
2770 |
|
sl@0
|
2771 |
#endif//SQLITE_OS_SYMBIAN
|
sl@0
|
2772 |
|