os/persistentdata/persistentstorage/sql/OsLayer/os_symbian.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
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