sl@0: // Copyright (c) 2005-2010 Nokia Corporation and/or its subsidiary(-ies). sl@0: // All rights reserved. sl@0: // This component and the accompanying materials are made available sl@0: // under the terms of "Eclipse Public License v1.0" sl@0: // which accompanies this distribution, and is available sl@0: // at the URL "http://www.eclipse.org/legal/epl-v10.html". sl@0: // sl@0: // Initial Contributors: sl@0: // Nokia Corporation - initial contribution. sl@0: // sl@0: // Contributors: sl@0: // sl@0: // Description: sl@0: // The Symbian OS porting layer - single-threaded implementation. sl@0: // SQLite never accesses the file system and the OS services directly. sl@0: // SQLite uses for that sqlite3_vfs and sqlite3_file objects. sl@0: // sqlite3_vfs and sqlite3_file functionality is implemented in this file - sl@0: // TVfs and TFileIo classes. sl@0: // This file is also used for the COsLayerData implementation. A single COslayerData sl@0: // object is used by the OS porting layer for managing some global data. sl@0: // sl@0: // sl@0: sl@0: /** sl@0: @file sl@0: @see TVfs sl@0: @see TFileIo sl@0: */ sl@0: sl@0: #ifdef SQLITE_OS_SYMBIAN sl@0: sl@0: //#define _SQLPROFILER // Enable profiling //The same macro has to be enabled in SqlAssert.h file sl@0: sl@0: extern "C" sl@0: { sl@0: #include "sqliteInt.h" sl@0: #include "os.h" sl@0: } sl@0: #include "os_common.h" sl@0: #include "SqliteSymbian.h" sl@0: #include "FileBuf64.h" sl@0: #include sl@0: #ifdef _SQLPROFILER sl@0: #include sl@0: #include "../INC/SqlResourceProfiler.h" sl@0: #endif sl@0: #include "SqliteUtil.h" sl@0: #include "OstTraceDefinitions.h" sl@0: #ifdef OST_TRACE_COMPILER_IN_USE sl@0: #include "os_symbianTraces.h" sl@0: #endif sl@0: #include "SqliteTraceDef.h" sl@0: sl@0: ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: //The SQLite temp files willl be created in this subdir sl@0: _LIT(KTempFileDir, "temp"); sl@0: sl@0: //Bit-mask constant. If xOpen()'s "aFlag" parameter contains one of these bits set, then the the file top be sl@0: //opened or created is a journal file. sl@0: const TUint KJournalFileTypeBitMask = SQLITE_OPEN_MAIN_JOURNAL | SQLITE_OPEN_TEMP_JOURNAL | SQLITE_OPEN_SUBJOURNAL | SQLITE_OPEN_MASTER_JOURNAL; sl@0: sl@0: ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: #ifdef _SQLPROFILER sl@0: sl@0: //Profiling counters, defined in SqlSrvResourceprofiler.cpp sl@0: extern TInt TheSqlSrvProfilerFileRead; sl@0: extern TInt TheSqlSrvProfilerFileWrite; sl@0: extern TInt TheSqlSrvProfilerFileSync; sl@0: extern TInt TheSqlSrvProfilerFileSetSize; sl@0: sl@0: # define __COUNTER_INCR(counter) ++counter sl@0: sl@0: ////////////////////////////////////////////////////////////////////////////// sl@0: ////////////////////////// File I/O ////////////////////////////////////////// sl@0: ////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: This enum is used only in _SQLPROFILER mode for tracing the file system calls. sl@0: @internalComponent sl@0: @see FsCallBrkPt() sl@0: */ sl@0: enum TFsOpType sl@0: { sl@0: EFsOpFileCreate, sl@0: EFsOpFileOpen, sl@0: EFsOpFileClose, sl@0: EFsOpFileDelete, sl@0: EFsOpFileRead, sl@0: EFsOpFileWrite, sl@0: EFsOpFileSeek, sl@0: EFsOpFileSize, sl@0: EFsOpFileSetSize, sl@0: EFsOpFileSync, sl@0: EFsOpFileDrive, sl@0: EFsOpFileAdopt, sl@0: EFsOpFsClose, sl@0: EFsOpFsConnect, sl@0: EFsOpFsGetSystemDrive, sl@0: EFsOpFsCreatePrivatePath, sl@0: EFsOpFsPrivatePath, sl@0: EFsOpFsVolumeIoParam, sl@0: EFsOpFsEntry, sl@0: EFsOpFsAtt, sl@0: EFsOpFileCreateTemp, sl@0: EFsOpFileAttach, sl@0: // sl@0: EFsOpLast sl@0: }; sl@0: sl@0: TBool TheFileIoProfileEnabled = EFalse; sl@0: TUint32 TheFileOpCounter[EFsOpLast] = {0}; sl@0: TInt64 TheFileWriteAmount = 0; sl@0: TInt64 TheFileReadAmount = 0; sl@0: sl@0: /** sl@0: This function is used only in _SQLPROFILER mode as an appropriate place for: sl@0: - setting breakpoints for tracing the file system calls; sl@0: - collection information about the number of the file system calls; sl@0: sl@0: @param aFsOpType A TFsOpType enum item value, identifying the file system operation that will be executed; sl@0: @param a1 If the operation is "file read" or "file write" - the amount of data read/written; sl@0: sl@0: @internalComponent sl@0: sl@0: @see TFsOpType sl@0: */ sl@0: void FsCallBrkPt(TInt aFsOpType, TInt a1) sl@0: { sl@0: __ASSERT_DEBUG(aFsOpType >= 0 && aFsOpType < EFsOpLast, User::Invariant()); sl@0: if(!TheFileIoProfileEnabled) sl@0: { sl@0: return; sl@0: } sl@0: TFsOpType fsOpType = (TFsOpType)aFsOpType; sl@0: ++TheFileOpCounter[fsOpType]; sl@0: if(fsOpType == EFsOpFileWrite) sl@0: { sl@0: TheFileWriteAmount += a1; sl@0: } sl@0: else if(fsOpType == EFsOpFileRead) sl@0: { sl@0: TheFileReadAmount += a1; sl@0: } sl@0: } sl@0: sl@0: # define __FS_CALL(aFsOpType, a1) FsCallBrkPt(aFsOpType, a1) sl@0: sl@0: ////////////////////////////////////////////////////////////////////////////// sl@0: ////////////////////////// Heap Stats //////////////////////////////////////// sl@0: ////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: Memory operation type: alloc, realloc, free. Used during the profiling. sl@0: sl@0: @internalComponent sl@0: */ sl@0: enum TMemOpType sl@0: { sl@0: EMemOpAlloc, sl@0: EMemOpRealloc, sl@0: EMemOpFree, sl@0: // sl@0: EMemOpLast sl@0: }; sl@0: sl@0: TBool TheMemProfileEnabled = EFalse;//Mem operation call counters and time counters enabled/disabled sl@0: TBool TheMaxAllocProfileEnabled = EFalse;//Max mem allocation enabled/disabled sl@0: TUint32 TheMemOpCounter[EMemOpLast] = {0}; sl@0: TInt64 TheMemOpTicks[EMemOpLast] = {0}; sl@0: TInt64 TheAllocated = 0; sl@0: TInt64 TheFreed = 0; sl@0: TInt TheAllocMax = 0; sl@0: sl@0: /** sl@0: This class is used only in _SQLPROFILER mode as an appropriate place for: sl@0: - setting breakpoints for tracing the memory allocation/deallocation calls; sl@0: - collection information about the number of the memory allocation/deallocation calls and the time spent in the calls; sl@0: sl@0: The constructor's parameters are: sl@0: - aOpType A TMemOpType enum item value, identifying the operation that will be executed; sl@0: - aAmt1 The allocated/deallocated size; sl@0: - aAmt2 Used only if a block of memory is reallocated in which case a2 is the old size of the block; sl@0: sl@0: @internalComponent sl@0: sl@0: @see TMemOpType sl@0: */ sl@0: class TMemCallCounter sl@0: { sl@0: public: sl@0: TMemCallCounter(TMemOpType aOpType, TInt aAmt1, TInt aAmt2) : sl@0: iOpType(aOpType), sl@0: iStartTicks(0) sl@0: { sl@0: if(TheMaxAllocProfileEnabled && (iOpType == EMemOpAlloc || iOpType == EMemOpRealloc) && aAmt1 > TheAllocMax) sl@0: { sl@0: TheAllocMax = aAmt1; sl@0: } sl@0: if(TheMemProfileEnabled) sl@0: { sl@0: ++TheMemOpCounter[iOpType]; sl@0: switch(iOpType) sl@0: { sl@0: case EMemOpAlloc: sl@0: TheAllocated += aAmt1; sl@0: break; sl@0: case EMemOpRealloc: sl@0: TheAllocated += aAmt1; sl@0: TheFreed += aAmt2; sl@0: break; sl@0: case EMemOpFree: sl@0: TheFreed += aAmt1; sl@0: break; sl@0: default: sl@0: __ASSERT_DEBUG(0, User::Invariant()); sl@0: break; sl@0: } sl@0: iStartTicks = User::FastCounter(); sl@0: } sl@0: } sl@0: ~TMemCallCounter() sl@0: { sl@0: if(TheMemProfileEnabled) sl@0: { sl@0: TInt64 diffTicks = (TInt64)User::FastCounter() - (TInt64)iStartTicks; sl@0: if(diffTicks < 0) sl@0: { sl@0: diffTicks = KMaxTUint + diffTicks + 1; sl@0: } sl@0: TheMemOpTicks[iOpType] += diffTicks; sl@0: } sl@0: } sl@0: private: sl@0: TMemOpType iOpType; sl@0: TUint32 iStartTicks; sl@0: }; sl@0: sl@0: # define __MEM_CALL(aMemOpType, a1, a2) TMemCallCounter memCallCounter(aMemOpType, a1, a2) sl@0: sl@0: ////////////////////////////////////////////////////////////////////////////// sl@0: ////////////////////////// OS layer calls //////////////////////////////////// sl@0: ////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: //The OS porting layer call types sl@0: enum TOsOpType sl@0: { sl@0: EOsFileClose, sl@0: EOsFileRead, sl@0: EOsFileWrite, sl@0: EOsFileTruncate, sl@0: EOsFileSync, sl@0: EOsFileFileSize, sl@0: EOsFileLock, sl@0: EOsFileUnlock, sl@0: EOsFileCheckReservedLock, sl@0: EOsFileFileControl, sl@0: EOsFileSectorSize, sl@0: EOsFileDeviceCharacteristics, sl@0: // sl@0: EOsVfsOpen, sl@0: EOsVfsDelete, sl@0: EOsVfsAccess, sl@0: EOsVfsFullPathName, sl@0: EOsVfsRandomness, sl@0: EOsVfsSleep, sl@0: EOsVfsCurrentTime, sl@0: EOsVfsGetLastError, sl@0: // sl@0: EOsOpLast sl@0: }; sl@0: sl@0: TBool TheOsProfileEnabled = EFalse; sl@0: TUint32 TheOsOpCounter[EOsOpLast] = {0};//Each entry is a counter - how many times specific OS porting layer function has been called sl@0: sl@0: # define __OS_CALL(aOsOpType, a1, a2) \ sl@0: do \ sl@0: { \ sl@0: if(TheOsProfileEnabled) \ sl@0: { \ sl@0: ++TheOsOpCounter[aOsOpType];\ sl@0: } \ sl@0: } \ sl@0: while(0) sl@0: sl@0: ////////////////////////////////////////////////////////////////////////////// sl@0: ////////////////////////// OS layer timings ////////////////////////////////// sl@0: ////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: TBool TheOsCallTimeProfileEnabled = EFalse;//If true, the OS porting layer call timings are enabled. sl@0: TInt64 TheOsCallTicks[EOsOpLast];//Each entry represents the time in ticks spent in a specific OS porting layer function, sl@0: //disregarding the file type (main or journal) sl@0: sl@0: 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: TInt TheOpCounter = 0;//Operations counter. Each OS porting layer function call increments the counter. sl@0: sl@0: //Used for storing the OS call details: sl@0: // - iType - on which file the call has been made: main database file - 'M', or journal file - 'J'; sl@0: // - iIdentifier - two letters identifying the monitored OS porting layer function; sl@0: // - iCallCounter - how many times the monitored OS porting layer function has been called; sl@0: // - iTicksTotal - the total amount of time in ticks spent in the monitored OS porting layer function; sl@0: // - iBytesTotal - the total amount of bytes passed to the monitored OS porting layer function (if it is read or write); sl@0: struct TOsCallProfile sl@0: { sl@0: TOsCallProfile(char aType, const TDesC& aIdentifier) : sl@0: iType(aType), sl@0: iIdentifier(aIdentifier), sl@0: iCallCounter(0), sl@0: iTicksTotal(0), sl@0: iBytesTotal(0) sl@0: { sl@0: } sl@0: void Zero() sl@0: { sl@0: iCallCounter = 0; sl@0: iTicksTotal = 0; sl@0: iBytesTotal = 0; sl@0: } sl@0: char iType; sl@0: TBuf<32> iIdentifier; sl@0: TInt iCallCounter; sl@0: TInt64 iTicksTotal; sl@0: TInt64 iBytesTotal; sl@0: }; sl@0: sl@0: //An array of TOsCallProfile entries, each entry keeps the profile of a specifc OS porting layer function, when sl@0: //the function was used on the main database file sl@0: TOsCallProfile TheOsCallMProfile[EOsOpLast] = sl@0: { sl@0: TOsCallProfile('M', _L("Close")), TOsCallProfile('M', _L("Read")), TOsCallProfile('M', _L("Write")), TOsCallProfile('M', _L("Truncate")), sl@0: TOsCallProfile('M', _L("Sync")), TOsCallProfile('M', _L("Size")), TOsCallProfile('M', _L("Lock")), TOsCallProfile('M', _L("Unlock")), sl@0: TOsCallProfile('M', _L("CheckReservedLock")), TOsCallProfile('M', _L("FileControl")), TOsCallProfile('M', _L("SetSize")), TOsCallProfile('M', _L("DeviceCharacteristics")), sl@0: TOsCallProfile('M', _L("Open")), TOsCallProfile('M', _L("Delete")), TOsCallProfile('M', _L("Access")), TOsCallProfile('M', _L("FullPath")), sl@0: TOsCallProfile('M', _L("Randomness")), TOsCallProfile('M', _L("Sleep")), TOsCallProfile('M', _L("CurrentTime")), TOsCallProfile('M', _L("GetLastError")) sl@0: }; sl@0: sl@0: //An array of TOsCallProfile entries, each entry keeps the profile of a specifc OS porting layer function, when sl@0: //the function was used on the journal file sl@0: TOsCallProfile TheOsCallJProfile[EOsOpLast] = sl@0: { sl@0: TOsCallProfile('J', _L("Close")), TOsCallProfile('J', _L("Read")), TOsCallProfile('J', _L("Write")), TOsCallProfile('J', _L("Truncate")), sl@0: TOsCallProfile('J', _L("Sync")), TOsCallProfile('J', _L("Size")), TOsCallProfile('J', _L("Lock")), TOsCallProfile('J', _L("Unlock")), sl@0: TOsCallProfile('J', _L("CheckReservedLock")), TOsCallProfile('J', _L("FileControl")), TOsCallProfile('J', _L("SetSize")), TOsCallProfile('J', _L("DeviceCharacteristics")), sl@0: TOsCallProfile('J', _L("Open")), TOsCallProfile('J', _L("Delete")), TOsCallProfile('J', _L("Access")), TOsCallProfile('J', _L("FullPath")), sl@0: TOsCallProfile('J', _L("Randomness")), TOsCallProfile('J', _L("Sleep")), TOsCallProfile('J', _L("CurrentTime")), TOsCallProfile('J', _L("GetLastError")) sl@0: }; sl@0: sl@0: //The main class for the OS porting layer call profiles. sl@0: class TOsCallCounter sl@0: { sl@0: public: sl@0: //aOsCallTicksEntryRef - a reference to the related TheOsCallTicks[] entry; sl@0: //aProfileRef - a reference to the related TOsCallProfile object - TheOsCallMProfile[] or TheOsCallJProfile[] entry; sl@0: //aOffset - file offset in bytes; sl@0: //aBytes - amount of bytes to be read/written; sl@0: //aOptional - might be NULL. Used to print out the name of the file being processed. sl@0: TOsCallCounter(TInt64& aOsCallTicksEntryRef, TOsCallProfile& aOsCallProfileRef, TInt64 aOffset, TInt aBytes, sl@0: const sqlite3_file* aHandle, const char* aOptional) : sl@0: iOsCallTicksEntryRef(aOsCallTicksEntryRef), sl@0: iOsCallProfileRef(aOsCallProfileRef), sl@0: iOffset(aOffset), sl@0: iBytes(aBytes), sl@0: iHandle((TUint)aHandle), sl@0: iOptional((const TUint8*)aOptional) sl@0: { sl@0: if(TheOsCallTimeProfileEnabled) sl@0: { sl@0: iStartTicks = User::FastCounter(); sl@0: } sl@0: } sl@0: ~TOsCallCounter() sl@0: { sl@0: if(TheOsCallTimeProfileEnabled) sl@0: { sl@0: TInt64 diffTicks = (TInt64)User::FastCounter() - (TInt64)iStartTicks; sl@0: if(diffTicks < 0) sl@0: { sl@0: diffTicks = KMaxTUint + diffTicks + 1; sl@0: } sl@0: iOsCallTicksEntryRef += diffTicks; sl@0: if(TheOsCallTimeDetailedProfileEnabled) sl@0: { sl@0: ++TheOpCounter; sl@0: ++iOsCallProfileRef.iCallCounter; sl@0: iOsCallProfileRef.iTicksTotal += diffTicks; sl@0: iOsCallProfileRef.iBytesTotal += iBytes; sl@0: TFileName fname; sl@0: if(iOptional) sl@0: { sl@0: TPtrC8 fn8(iOptional); sl@0: fname.Copy(fn8); sl@0: } sl@0: // 0 1 2 3 4 5 6 7 8 9 10 sl@0: RDebug::Print(_L("[SQL-OS]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬%c¬%S¬%d¬%d¬%ld¬%d¬%ld¬%ld¬%ld¬%S\n"), sl@0: //[SQL-OS] sl@0: //Handle sl@0: //Time from start, microseconds sl@0: //Subtype sl@0: //IPC sequence counter sl@0: //IPC call name sl@0: iHandle, //sqlite3_file* sl@0: iOsCallProfileRef.iType, //1 - main or journal file sl@0: &iOsCallProfileRef.iIdentifier, //2 - operation type sl@0: TheOpCounter, //3 - Operation counter sl@0: iOsCallProfileRef.iCallCounter, //4 - This call type counter sl@0: iOffset, //5 - File offset sl@0: iBytes, //6 - Data, bytes sl@0: diffTicks, //7 - Ticks sl@0: iOsCallProfileRef.iBytesTotal, //8 - Data total, bytes sl@0: iOsCallProfileRef.iTicksTotal, //9 - Ticks total sl@0: &fname); //10- File name sl@0: } sl@0: } sl@0: } sl@0: private: sl@0: TInt64& iOsCallTicksEntryRef; sl@0: TOsCallProfile& iOsCallProfileRef; sl@0: TInt64 iOffset; sl@0: TInt iBytes; sl@0: TUint32 iStartTicks; sl@0: TUint iHandle; sl@0: const TUint8* iOptional; sl@0: }; sl@0: sl@0: inline TOsCallProfile& OsCallProfile(TBool aType, TInt aIndex) sl@0: { sl@0: return aType ? TheOsCallJProfile[aIndex] : TheOsCallMProfile[aIndex]; sl@0: } sl@0: sl@0: # define __OSTIME_COUNTER(aOsCallTicksRef, aOsCallProfileRef, aOffset, aBytes, aHandle, aOpt) TOsCallCounter osCallCounter(aOsCallTicksRef, aOsCallProfileRef, aOffset, aBytes, aHandle, aOpt) sl@0: sl@0: #else //_SQLPROFILER sl@0: sl@0: # define __COUNTER_INCR(counter) void(0) sl@0: sl@0: # define __FS_CALL(aFsOpType, a1) void(0) sl@0: sl@0: # define __MEM_CALL(aMemOpType, a1, a2) void(0) sl@0: sl@0: # define __OS_CALL(aOpType, a1, a2) void(0) sl@0: sl@0: # define __OSTIME_COUNTER(aOsCallTicksRef, aOsCallProfileRef, aOffset, aBytes, aHandle, aOpt) void(0) sl@0: sl@0: #endif//_SQLPROFILER sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ///////////////////// Profiling //////////////////////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: #ifdef _SQLPROFILER sl@0: sl@0: const TInt KMicroSecIn1Sec = 1000000; sl@0: sl@0: TInt FastCounterFrequency() sl@0: { sl@0: TInt ticksPerSec = 0; sl@0: TInt err = HAL::Get(HAL::EFastCounterFrequency, ticksPerSec); sl@0: __ASSERT_ALWAYS(err == KErrNone, __SQLITEPANIC2(ESqliteOsPanicFastCounterFreq)); sl@0: return ticksPerSec; sl@0: } sl@0: sl@0: TInt sqlite3SymbianProfilerStart(TInt aCounterType) sl@0: { sl@0: const TSqlResourceProfiler::TSqlCounter KCounterType = static_cast (aCounterType); sl@0: switch(KCounterType) sl@0: { sl@0: case TSqlResourceProfiler::ESqlCounterFileIO: sl@0: TheFileIoProfileEnabled = ETrue; sl@0: break; sl@0: case TSqlResourceProfiler::ESqlCounterOsCall: sl@0: TheOsProfileEnabled = ETrue; sl@0: break; sl@0: case TSqlResourceProfiler::ESqlCounterOsCallTime: sl@0: TheOsCallTimeProfileEnabled = ETrue; sl@0: break; sl@0: case TSqlResourceProfiler::ESqlCounterOsCallDetails: sl@0: TheOsCallTimeProfileEnabled = ETrue; sl@0: TheOsCallTimeDetailedProfileEnabled = ETrue; sl@0: break; sl@0: case TSqlResourceProfiler::ESqlCounterMemory: sl@0: TheMemProfileEnabled = ETrue; sl@0: break; sl@0: case TSqlResourceProfiler::ESqlCounterMaxAlloc: sl@0: TheMaxAllocProfileEnabled = ETrue; sl@0: break; sl@0: default: sl@0: return KErrNotSupported; sl@0: } sl@0: return KErrNone; sl@0: } sl@0: sl@0: TInt sqlite3SymbianProfilerStop(TInt aCounterType) sl@0: { sl@0: const TSqlResourceProfiler::TSqlCounter KCounterType = static_cast (aCounterType); sl@0: switch(KCounterType) sl@0: { sl@0: case TSqlResourceProfiler::ESqlCounterFileIO: sl@0: TheFileIoProfileEnabled = EFalse; sl@0: break; sl@0: case TSqlResourceProfiler::ESqlCounterOsCall: sl@0: TheOsProfileEnabled = EFalse; sl@0: break; sl@0: case TSqlResourceProfiler::ESqlCounterOsCallTime: sl@0: TheOsCallTimeProfileEnabled = EFalse; sl@0: break; sl@0: case TSqlResourceProfiler::ESqlCounterOsCallDetails: sl@0: TheOsCallTimeDetailedProfileEnabled = EFalse; sl@0: TheOsCallTimeProfileEnabled = EFalse; sl@0: break; sl@0: case TSqlResourceProfiler::ESqlCounterMemory: sl@0: TheMemProfileEnabled = EFalse; sl@0: break; sl@0: case TSqlResourceProfiler::ESqlCounterMaxAlloc: sl@0: TheMaxAllocProfileEnabled = EFalse; sl@0: break; sl@0: default: sl@0: return KErrNotSupported; sl@0: } sl@0: return KErrNone; sl@0: } sl@0: sl@0: TInt sqlite3SymbianProfilerReset(TInt aCounterType) sl@0: { sl@0: const TSqlResourceProfiler::TSqlCounter KCounterType = static_cast (aCounterType); sl@0: switch(KCounterType) sl@0: { sl@0: case TSqlResourceProfiler::ESqlCounterFileIO: sl@0: Mem::FillZ(TheFileOpCounter, sizeof(TheFileOpCounter)); sl@0: TheFileWriteAmount = TheFileReadAmount = 0; sl@0: break; sl@0: case TSqlResourceProfiler::ESqlCounterOsCall: sl@0: Mem::FillZ(TheOsOpCounter, sizeof(TheOsOpCounter)); sl@0: break; sl@0: case TSqlResourceProfiler::ESqlCounterOsCallTime: sl@0: case TSqlResourceProfiler::ESqlCounterOsCallDetails: sl@0: TheOpCounter = 0; sl@0: Mem::FillZ(TheOsCallTicks, sizeof(TheOsCallTicks)); sl@0: for(TInt i=0;i (aCounterType); sl@0: switch(KCounterType) sl@0: { sl@0: case TSqlResourceProfiler::ESqlCounterFileIO: sl@0: for(TInt i=0;i:\" + process's private data path. Initialized in sqlite3SymbianFsOpen(). sl@0: TInt64 iSeed; sl@0: sl@0: private: sl@0: static COsLayerData* iOsLayerData; sl@0: TInt iStoredOsErrorCode; //Contains the last OS error code. sl@0: const RMessage2* iMessage; //Fh data sl@0: TBool iReadOnly; //Fh data sl@0: }; sl@0: sl@0: /** sl@0: This functon returns a reference to the current thread allocator object. sl@0: The static RAllocator& variable will be initialized once at the moment when the function is called for sl@0: first time. sl@0: */ sl@0: static RAllocator& Allocator() sl@0: { sl@0: static RAllocator& allocator = User::Allocator(); sl@0: return allocator; sl@0: } sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: /////////////////////////////////// TDbFile struct declaration ///////////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: TDbFile derives from the sqlite3_file structure, adding data members needed for processing the SQLite requests to the OS layer. sl@0: When SQLite needs an access to a file, SQLite allocates memory for a new TDbFile instance and passes a pointer to that sl@0: instance to TVfs::Open(). TVfs::Open() creates/opens the file and initializes the TDbFile instance. sl@0: SQLite uses the initialized TDbFile instance (actually SQLite knows and uses the sqlite3_file, the base structure) sl@0: every time when needs to read or write from/to the file, using for that an appropriate TFileIo method. sl@0: sl@0: Note: currently RFileBuf64 object is used instead of RFile64. That improves the read/write file performance. sl@0: sl@0: No virtual methods here! sqlite3_file contains data members. If a virtual method is added, that will shift the offset of the sl@0: data members from the beginning of the sqlite3_file object by 4 bytes. This is not what SQLite (C code) expects. sl@0: sl@0: @internalComponent sl@0: sl@0: @see TVfs sl@0: @see TFileIo sl@0: @see TVfs::Open() sl@0: */ sl@0: NONSHARABLE_STRUCT(TDbFile) : public sqlite3_file sl@0: { sl@0: inline TDbFile(); sl@0: RFileBuf64 iFileBuf; sl@0: TInt iLockType; //File lock type sl@0: TBool iReadOnly; //True if the file is read-only sl@0: TInt iSectorSize; //Media sector-size sl@0: TInt iDeviceCharacteristics; sl@0: TSqlFreePageCallback iFreePageCallback; sl@0: TBool iIsFileCreated; //If the file to be created is a temp file, sl@0: //it will not be created until the data is to be written to. sl@0: #ifdef _SQLPROFILER sl@0: TBool iIsJournal; sl@0: #endif sl@0: }; sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: /////////////////////////////////// TFileIo class declaration ////////////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: TFileIo class offers static methods for performing operations on a file. sl@0: Every TFileIo method has a pointer to a sqlite3_file instance (so, a TDbFile instance) as its first argument. sl@0: sl@0: SQLite never accesses the file system directly, only through function pointers, data members of the sqlite3_io_methods structure. sl@0: The OS porting layer defines a single instance of sqlite3_io_methods structure, TheFileIoApi, and uses the TFileIo to initialize the sl@0: sqlite3_io_methods data members (function pointers). sl@0: 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: initialized sqlite3_io_methods instance (TheFileIoApi) that will be used later by SQLite for accessing the file. sl@0: sl@0: @internalComponent sl@0: sl@0: @see TVfs sl@0: @see TVfs::Open() sl@0: @see TheFileIoApi sl@0: @see TDbFile sl@0: */ sl@0: NONSHARABLE_CLASS(TFileIo) sl@0: { sl@0: public: sl@0: static int Close(sqlite3_file* aDbFile); sl@0: static int Read(sqlite3_file* aDbFile, void* aBuf, int aAmt, sqlite3_int64 aOffset); sl@0: static int Write(sqlite3_file* aDbFile, const void* aData, int aAmt, sqlite3_int64 aOffset); sl@0: static int Truncate(sqlite3_file* aDbFile, sqlite3_int64 aLength); sl@0: static int Sync(sqlite3_file* aDbFile, int aFlags); sl@0: static int FileSize(sqlite3_file* aDbFile, sqlite3_int64* aSize); sl@0: static int Lock(sqlite3_file* aDbFile, int aLockType); sl@0: static int Unlock(sqlite3_file* aDbFile, int aLockType); sl@0: static int CheckReservedLock(sqlite3_file* aDbFile, int *aResOut); sl@0: static int FileControl(sqlite3_file* aDbFile, int aOp, void* aArg); sl@0: static int SectorSize(sqlite3_file* aDbFile); sl@0: static int DeviceCharacteristics(sqlite3_file* aDbFile); sl@0: private: sl@0: static TInt DoCreateTempFile(TDbFile& aDbFile); sl@0: }; sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: /////////////////////////////////// TVfs class declaration ///////////////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: TVfs ("VFS" - virtual file system) class offers methods for creating/openning a file, deleting a file, sl@0: a "sleep" method, a "time" method, a "rand" method, etc. sl@0: SQLite never accesses the OS API directly, only through the API offered by TVfs and TFileIo classes. sl@0: sl@0: @internalComponent sl@0: sl@0: @see TFileIo sl@0: @see TheVfsApi sl@0: @see COsLayerData sl@0: */ sl@0: NONSHARABLE_CLASS(TVfs) sl@0: { sl@0: public: sl@0: static int Open(sqlite3_vfs* aVfs, const char* aFileName, sqlite3_file* aDbFile, int aFlags, int* aOutFlags); sl@0: static int Delete(sqlite3_vfs* aVfs, const char* aFileName, int aSyncDir); sl@0: static int Access(sqlite3_vfs* aVfs, const char* aFileName, int aFlags, int* aResOut); sl@0: static int FullPathName(sqlite3_vfs* aVfs, const char* aRelative, int aBufLen, char* aBuf); sl@0: static int Randomness(sqlite3_vfs* aVfs, int aBufLen, char* aBuf); sl@0: static int Sleep(sqlite3_vfs* aVfs, int aMicrosec); sl@0: static int CurrentTime(sqlite3_vfs* aVfs, double* aNow); sl@0: static int GetLastError(sqlite3_vfs *sVfs, int aBufLen, char* aBuf); sl@0: static TInt DoGetDeviceCharacteristicsAndSectorSize(TDbFile& aDbFile, TInt& aRecReadBufSize); sl@0: private: sl@0: static TInt DoOpenFromHandle(TDbFile& aDbFile, const RMessage2& aMsg, TBool aReadOnly); sl@0: static inline TInt DoGetVolumeIoParamInfo(RFs& aFs, TInt aDriveNo, TVolumeIOParamInfo& aVolumeInfo); sl@0: static TInt DoGetDeviceCharacteristics(const TDriveInfo& aDriveInfo, const TVolumeIOParamInfo& aVolumeInfo); sl@0: static TInt DoGetSectorSize(const TDriveInfo& aDriveInfo, const TVolumeIOParamInfo& aVolumeInfo); sl@0: static TInt DoFileSizeCorruptionCheck(TDbFile& aDbFile, const TDesC& aFname, TInt aFmode); sl@0: }; sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ///////////////////// Global variables, constants //////////////////////////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: // The following macro specifies the size of the RFileBuf64 file buffer in KB: sl@0: // __SQLITE_OS_SYMBIAN_FILEBUF_KBSIZE__ sl@0: // sl@0: // If not set, a default value of 8 is used sl@0: #if !defined(__SQLITE_OS_SYMBIAN_FILEBUF_KBSIZE__) sl@0: #define __SQLITE_OS_SYMBIAN_FILEBUF_KBSIZE__ 8 sl@0: #endif sl@0: const TInt KFileBufSize = __SQLITE_OS_SYMBIAN_FILEBUF_KBSIZE__ * 1024; sl@0: sl@0: /** sl@0: Pointer to the single COsLayerData instance. sl@0: sl@0: @see COsLayerData sl@0: sl@0: @internalComponent sl@0: */ sl@0: COsLayerData* COsLayerData::iOsLayerData = NULL; sl@0: sl@0: /** sl@0: Single sqlite3_io_methods instance, which data members (function pointers) are initialized with the addresses of sl@0: TFileIo members. sl@0: TheFileIoApi is used by SQLite for performing OS independent file I/O. sl@0: sl@0: @see TFileIo sl@0: @see TVfs sl@0: sl@0: @internalComponent sl@0: */ sl@0: static sqlite3_io_methods TheFileIoApi = sl@0: { sl@0: 1, //Version sl@0: &TFileIo::Close, sl@0: &TFileIo::Read, sl@0: &TFileIo::Write, sl@0: &TFileIo::Truncate, sl@0: &TFileIo::Sync, sl@0: &TFileIo::FileSize, sl@0: &TFileIo::Lock, sl@0: &TFileIo::Unlock, sl@0: &TFileIo::CheckReservedLock, sl@0: &TFileIo::FileControl, sl@0: &TFileIo::SectorSize, sl@0: &TFileIo::DeviceCharacteristics sl@0: }; sl@0: sl@0: /** sl@0: Single sqlite3_vfs instance, which data members (function pointers) are initialized with the addresses of sl@0: TVfs members. TheVfsApi also keeps information regarding some other OS dependend characteristics like sl@0: the TDbFile size and max file name length. sl@0: TheVfsApi is used by SQLite for accessing needed OS API. sl@0: sl@0: TheVfsApi can't be a constant definition. SQLite expects the "sqlite3_vfs" object to be a R/W one, because sl@0: SQLite may have and use a chain of sqlite3_vfs instances. sl@0: sl@0: @see TVfs sl@0: @see TTFileIo sl@0: @see TDbFile sl@0: sl@0: @internalComponent sl@0: */ sl@0: static sqlite3_vfs TheVfsApi = sl@0: { sl@0: 1, //iVersion sl@0: sizeof(TDbFile), //szOsFile sl@0: KMaxFileName, //mxPathname sl@0: 0, //pNext sl@0: "SymbianSql", //zName sl@0: 0, //pAppData sl@0: &TVfs::Open, sl@0: &TVfs::Delete, sl@0: &TVfs::Access, sl@0: &TVfs::FullPathName, sl@0: 0, sl@0: 0, sl@0: 0, sl@0: 0, sl@0: &TVfs::Randomness, sl@0: &TVfs::Sleep, sl@0: &TVfs::CurrentTime, sl@0: &TVfs::GetLastError sl@0: }; sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ///////////////////// COsLayerData class definition ////////////////////////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: Creates a single COsLayerData instance. sl@0: sl@0: @return KErrNone, The operation has completed succesfully; sl@0: KErrNoMemory, Out of memory condition has occured; sl@0: Note that other system-wide error codes may also be returned. sl@0: sl@0: @panic Sqlite 16 In _DEBUG mode - the COsLayerData instance has been created already. sl@0: */ sl@0: /* static */ TInt COsLayerData::Create() sl@0: { sl@0: SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, COSLAYERDATA_CREATE_ENTRY, "OS-Entry;0;COsLayerData::Create")); sl@0: __ASSERT_DEBUG(!COsLayerData::iOsLayerData, __SQLITEPANIC2(ESqliteOsPanicOsLayerDataExists)); sl@0: COsLayerData::iOsLayerData = new COsLayerData; sl@0: if(!COsLayerData::iOsLayerData) sl@0: { sl@0: SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, COSLAYERDATA_CREATE_EXIT1, "OS-Exit;0;COsLayerData::Create;err=KErrNoMemory")); sl@0: return KErrNoMemory; sl@0: } sl@0: TInt err = COsLayerData::iOsLayerData->DoCreate(); sl@0: if(err != KErrNone) sl@0: { sl@0: delete COsLayerData::iOsLayerData; sl@0: COsLayerData::iOsLayerData = NULL; sl@0: SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, COSLAYERDATA_CREATE_EXIT2, "OS-Exit;0;COsLayerData::Create;err=%d", err)); sl@0: return err; sl@0: } sl@0: SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, COSLAYERDATA_CREATE_EXIT3, "OS-Exit;0;COsLayerData::Create;err=KErrNone")); sl@0: return KErrNone; sl@0: } sl@0: sl@0: /** sl@0: Destroys the COsLayerData instance. sl@0: */ sl@0: /* static */ inline void COsLayerData::Destroy() sl@0: { sl@0: delete COsLayerData::iOsLayerData; sl@0: COsLayerData::iOsLayerData = NULL; sl@0: SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, COSLAYERDATA_DESTROY, "OS;0;COsLayerData::Destroy")); sl@0: } sl@0: sl@0: /** sl@0: Returns a reference to the single COsLayerData instance. sl@0: sl@0: @panic Sqlite 1 In _DEBUG mode if the COsLayerData instance is NULL. sl@0: */ sl@0: /* static */ inline COsLayerData& COsLayerData::Instance() sl@0: { sl@0: __ASSERT_DEBUG(COsLayerData::iOsLayerData != NULL, __SQLITEPANIC2(ESqliteOsPanicNullOsLayerDataPtr)); sl@0: return *COsLayerData::iOsLayerData; sl@0: } sl@0: sl@0: /** sl@0: Sets the last OS error code data member. The stored OS error code data member will be set only if it is sl@0: KErrNone. (If it is not KErrNone it means that its value has not been accessed yet) sl@0: An exception from the rule described above is KErrDiskFull error which, if happens, will be set always, because sl@0: this error has a special meaning for the database clients - special actions may have to be taken if the sl@0: disk is full. sl@0: sl@0: @param aError The OS error code sl@0: @return The OS error code sl@0: */ sl@0: inline TInt COsLayerData::SetOsErrorCode(TInt aError) sl@0: { sl@0: if(iStoredOsErrorCode == KErrNone || aError == KErrDiskFull) sl@0: { sl@0: iStoredOsErrorCode = aError; sl@0: } sl@0: return aError; sl@0: } sl@0: sl@0: /** sl@0: Returns the last stored OS error code, which was stored by SetOsErrorCode() call. sl@0: The function also resets the stored OS error code to KErrNone. sl@0: sl@0: @return The last stored OS error code sl@0: */ sl@0: inline TInt COsLayerData::StoredOsErrorCode() sl@0: { sl@0: TInt err = iStoredOsErrorCode; sl@0: iStoredOsErrorCode = KErrNone; sl@0: return err; sl@0: } sl@0: sl@0: /** sl@0: Stores the RMessage2 object address, file and file session handles and the read-only flag for later use when SQLite issues a sl@0: request for open the database file (private secure database). sl@0: sl@0: The aMsg argument of the function can be NULL, because this fucntion is also used to reset the stored "file handle" data. sl@0: sl@0: How this function is used: sl@0: 1) When the SQL server receives a request to establish a connection with private secure database, the SQL server sl@0: will add additional information to the private secure database file name, such as: sl@0: - the file handle (the private secure database is opened by the client side dll - sqldb.dll); sl@0: - a pointer to the RMessage2 object used in this request; sl@0: 2) The passed additional information will be used for adopting the file handle by calling RFile64::AdoptFromClient(). sl@0: 3) Before calling TVfs::Open() to establish a connection with the database, SQLite will call TVfs::FullPathName() sl@0: to retrieve the database file full path sl@0: 4) TVfs::FullPathName() will detect that the file name contains an additional information and will extraxt the information sl@0: calling COsLayerData::StoreFhData(). sl@0: 5) After TVfs::FullPathName() SQLite calls TVfs::Open() where the extracted information will be used for adopting the file handle sl@0: sl@0: @param aMsg A pointer to the current RMessage2 object sl@0: @param aReadOnly True if the private secure database is read-only sl@0: */ sl@0: inline void COsLayerData::StoreFhData(const RMessage2* aMsg, TBool aReadOnly) sl@0: { sl@0: iMessage = aMsg; sl@0: iReadOnly = aReadOnly; sl@0: } sl@0: sl@0: /** sl@0: Retrieves the RMessage2 object, file and file session handles. The stored data will be reset. sl@0: This function is used by TVfs::Open(), when a request for opening a secure private database is processed. sl@0: sl@0: @param aMsg Output parameter. A reference to a RMessage2 pointer, which will be initialized with the stored RMessage2 pointer. sl@0: @param aReadOnly Output parameter. The store read-only flag value will be set there. sl@0: sl@0: @panic Sqlite 13 In _DEBUG mode - aMsg is NULL. sl@0: */ sl@0: inline void COsLayerData::RetrieveAndResetFhData(const RMessage2*& aMsg, TBool& aReadOnly) sl@0: { sl@0: __ASSERT_DEBUG(iMessage != NULL, __SQLITEPANIC(ESqliteOsPanicInvalidFhData)); sl@0: aMsg = iMessage; sl@0: aReadOnly = iReadOnly; sl@0: iMessage = NULL; sl@0: } sl@0: sl@0: /** sl@0: Initializes the COsLayerData data members with their default values. sl@0: */ sl@0: inline COsLayerData::COsLayerData() : sl@0: iStoredOsErrorCode(KErrNone), sl@0: iMessage(0), sl@0: iReadOnly(EFalse) sl@0: { sl@0: TTime now; sl@0: now.UniversalTime(); sl@0: iSeed = now.Int64(); sl@0: } sl@0: sl@0: /** sl@0: Destroys the COsLayerData instance. sl@0: sl@0: Note: No SQLite functions should be called inside the destructor, because SQLite is already shutdown-ed! sl@0: */ sl@0: inline COsLayerData::~COsLayerData() sl@0: { sl@0: SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, COSLAYERDATA_COSLAYERDATA2, "OS;0;COsLayerData::~COsLayerData;iFs.Handle()=0x%X", (TUint)iFs.Handle())); sl@0: __FS_CALL(EFsOpFsClose, 0); sl@0: iFs.Close(); sl@0: } sl@0: sl@0: /** sl@0: Creates a file session instance. sl@0: sl@0: Creates the private path, where the temporary files will be stored (on the system drive). sl@0: sl@0: In a case of a failure COsLayerData::DoCreate() does not close the file session. sl@0: This will be made in the calling function - COsLayerData::Create(). sl@0: sl@0: Note: No SQLite functions should be called inside the DoCreate() implementation, because SQLite is not initialized yet! sl@0: sl@0: @return KErrNone, The operation has completed succesfully; sl@0: KErrGeneral The registration of TheVfsApi has failed; sl@0: KErrNoMemory, Out of memory condition has occured; sl@0: Note that other system-wide error codes may also be returned. sl@0: sl@0: @see TVfs sl@0: @see TheVfsApi sl@0: */ sl@0: TInt COsLayerData::DoCreate() sl@0: { sl@0: __FS_CALL(EFsOpFsConnect, 0); sl@0: TInt err = iFs.Connect(); sl@0: if(err != KErrNone) sl@0: { sl@0: return err; sl@0: } sl@0: //Get the system drive sl@0: __FS_CALL(EFsOpFsGetSystemDrive, 0); sl@0: TInt sysDrive = static_cast(RFs::GetSystemDrive()); sl@0: __FS_CALL(EFsOpFsCreatePrivatePath, 0); sl@0: if((err = iFs.CreatePrivatePath(sysDrive)) != KErrNone && err != KErrAlreadyExists) sl@0: { sl@0: return err; sl@0: } sl@0: TFileName privateDir; sl@0: __FS_CALL(EFsOpFsPrivatePath, 0); sl@0: if((err = iFs.PrivatePath(privateDir)) != KErrNone) sl@0: { sl@0: return err; sl@0: } sl@0: TDriveUnit drive(sysDrive); sl@0: TDriveName driveName = drive.Name(); sl@0: TParse parse; sl@0: (void)parse.Set(driveName, &privateDir, 0);//this call can't fail sl@0: iSysPrivDir.Copy(parse.DriveAndPath()); sl@0: //Create the temp files directory sl@0: (void)parse.AddDir(KTempFileDir);//this call can't fail sl@0: err = iFs.MkDir(parse.DriveAndPath()); sl@0: if(err != KErrNone && err != KErrAlreadyExists) sl@0: { sl@0: return err; sl@0: } sl@0: 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: return KErrNone; sl@0: } sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: /////////////////////// Symbian OS specific functions (called by the SQL server) /////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: Returns the last OS error which occured durring the operations with the database files. sl@0: The per-thread variable, where the last OS error is hold, will be set to KErrNone. sl@0: sl@0: This function is part of Symbian OS specific SQLITE API. sl@0: sl@0: @return The last OS error. sl@0: @internalComponent sl@0: */ sl@0: TInt sqlite3SymbianLastOsError(void) sl@0: { sl@0: return COsLayerData::Instance().StoredOsErrorCode(); sl@0: } sl@0: sl@0: /** sl@0: This function must be called once before any other SQLITE API call. sl@0: The function: sl@0: @code sl@0: - Initializes the OS poting layer; sl@0: - Initializes the SQLite library; sl@0: @endcode sl@0: sl@0: This function is part of the Symbian OS specific SQLITE API. sl@0: sl@0: @return Symbian OS specific error code, including KErrNoMemory. sl@0: sl@0: @internalComponent sl@0: */ sl@0: TInt sqlite3SymbianLibInit(void) sl@0: { sl@0: TInt osErr = COsLayerData::Create(); sl@0: if(osErr != KErrNone) sl@0: { sl@0: return osErr; sl@0: } sl@0: osErr = KErrNone; sl@0: TInt sqliteErr = sqlite3_initialize(); sl@0: if(sqliteErr != SQLITE_OK) sl@0: { sl@0: osErr = sqliteErr == SQLITE_NOMEM ? KErrNoMemory : KErrGeneral; sl@0: COsLayerData::Destroy(); sl@0: } sl@0: SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, SQLITE3SYMBIANLIBINIT, "OS;0;sqlite3SymbianLibInit;osErr=%d", osErr)); sl@0: return osErr; sl@0: } sl@0: sl@0: /** sl@0: This function must be called once after finishing working with sqlite. sl@0: The function: sl@0: @code sl@0: - Shuts down the SQLite library; sl@0: - Releases the allocated by the OS porting layer resources; sl@0: @endcode sl@0: sl@0: This function is part of the Symbian OS specific SQLITE API. sl@0: sl@0: @internalComponent sl@0: */ sl@0: void sqlite3SymbianLibFinalize(void) sl@0: { sl@0: (void)sqlite3_shutdown(); sl@0: COsLayerData::Destroy(); sl@0: SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, SQLITE3SYMBIANLIBFINALIZE, "OS;0;sqlite3SymbianLibFinalize")); sl@0: } sl@0: sl@0: /** sl@0: This function is part of Symbian OS specific SQLITE API. sl@0: sl@0: @return A reference to RFs instance used for sqlite file I/O operations. sl@0: @internalComponent sl@0: */ sl@0: RFs& sqlite3SymbianFs(void) sl@0: { sl@0: return COsLayerData::Instance().iFs; sl@0: } sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ////////////////// UTF16<-->UTF8, conversion functions //////////////////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: The function converts aFileName to UTF16 encoded file name, and stores the UTF16 encoded file name sl@0: to the place pointed by aFileNameDestBuf argument. sl@0: If the UTF16 conversion of the file name failed because the file name is too long or NULL, sl@0: the function returns EFalse. sl@0: sl@0: @param aFileName Expected to point to UTF8 encoded, zero terminated string. sl@0: Max allowed aFileName length is KMaxFileName (excluding terminating 0 character). sl@0: @param aFileNameDestBuf Output parameter. Will hold UTF16, non-zero-terminated string. sl@0: The max length must be at least KMaxFileName characters. sl@0: sl@0: @return True if the conversion has been completed successfully sl@0: */ sl@0: static TBool ConvertToUnicode(const char *aFileName, TDes& aFileNameDestBuf) sl@0: { sl@0: if(aFileName) sl@0: { sl@0: wchar_t* dest = reinterpret_cast (const_cast (aFileNameDestBuf.Ptr())); sl@0: TInt len = mbstowcs(dest, aFileName, aFileNameDestBuf.MaxLength()); sl@0: __ASSERT_DEBUG(len >= 0, __SQLITEPANIC2(ESqliteOsPanicInternalError)); sl@0: //If len == aFileNameDestBuf.MaxLength(), then the output buffer is too small. sl@0: if(len < aFileNameDestBuf.MaxLength()) sl@0: { sl@0: aFileNameDestBuf.SetLength(len); sl@0: return ETrue; sl@0: } sl@0: } sl@0: return EFalse; sl@0: } sl@0: sl@0: /** sl@0: The function converts aFileName to UTF8 encoded file name, and stores the UTF8 encoded file name sl@0: to the place pointed by aFileNameDestBuf argument. sl@0: If the UTF8 conversion of the file name failed because the file name is too long or NULL, sl@0: the function returns EFalse. sl@0: sl@0: @param aFileName Expected to point to UTF16 encoded, zero terminated string. sl@0: Max allowed aFileName length is KMaxFileName (excluding terminating 0 character). sl@0: @param aFileNameDestBuf Output parameter. Will hold UTF8, non-zero-terminated string. sl@0: The max length must be at least KMaxFileName characters. sl@0: sl@0: @return True if the conversion has been completed successfully sl@0: */ sl@0: static TBool ConvertFromUnicode(const TDesC& aFileName, TDes8& aFileNameDestBuf) sl@0: { sl@0: char* dest = reinterpret_cast (const_cast (aFileNameDestBuf.Ptr())); sl@0: const wchar_t* src = reinterpret_cast (aFileName.Ptr()); sl@0: TInt len = wcstombs(dest, src, aFileNameDestBuf.MaxLength()); sl@0: __ASSERT_DEBUG(len >= 0, __SQLITEPANIC2(ESqliteOsPanicInternalError)); sl@0: //If len == aFileNameDestBuf.MaxLength(), then the output buffer is too small. sl@0: if(len < aFileNameDestBuf.MaxLength()) sl@0: { sl@0: aFileNameDestBuf.SetLength(len); sl@0: return ETrue; sl@0: } sl@0: return EFalse; sl@0: } sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ////////////////// File name, containing handles, functions //////////////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: 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: const TInt KFhSessHandleIdx = 2;//The index of the file session handle in RMessage2 object sl@0: const TInt KFhFileHandleIdx = 3;//The index of the file handle in RMessage2 object sl@0: const TInt KFhMarkPos = 0; //if the symbol in this position is KFhSeparator, then the string contains file handles sl@0: const TInt KFhRoPos = 1; //read-only flag position in the string sl@0: const TInt KFhMsgAddrPos = 2; //RMessage2 address position in the string sl@0: const TInt KFhMsgAddrLen = 8; //RMessage2 address length sl@0: //const TInt KFhDrivePos = 1; //Drive position in the string (after removing the read-only flag and RMessage2 object's address) sl@0: sl@0: /** sl@0: File name string types: sl@0: @code sl@0: - ENotFhStr - The string does not contain file handles; sl@0: - EFhStr - The string contain file handles, but is not main db file; sl@0: - EFhMainDbStr - The string contain file handles and is the main db file; sl@0: @endcode sl@0: sl@0: Every file name passed to the OS porting layer's TVfs::Open() method falls into one of the following three categories: sl@0: ENotFhStr - the file name does not contain handles, EFhStr - the file name contains handles but is not a name of a private sl@0: secure database, EFhMainDbStr - the file name contains handles and is a name of a private secure database. sl@0: sl@0: @see TVfs::Open() sl@0: @see FhStringProps() sl@0: sl@0: @internalComponent sl@0: */ sl@0: enum TFhStrType sl@0: { sl@0: ENotFhStr, //The string does not contain file handles sl@0: EFhStr, //The string contain file handles, but is not main db file sl@0: EFhMainDbStr //The string contain file handles and is the main db file sl@0: }; sl@0: sl@0: /** sl@0: The TVfs::Open() implementation uses this function to determine the type of the file name, which can be sl@0: one of the TFhStrType enum item values. sl@0: sl@0: @param aFileName Zero-terminated, UTF8 encoded file name. sl@0: @return The file name type, one of the TFhStrType enum item values. sl@0: sl@0: @see TVfs::Open() sl@0: @see TFhStrType sl@0: sl@0: @internalComponent sl@0: */ sl@0: static TFhStrType FhStringProps(const char* aFileName) sl@0: { sl@0: char* first = strchr(aFileName, KFhSeparator); sl@0: if(!first) sl@0: { sl@0: return ENotFhStr; sl@0: } sl@0: char* last = strchr(first + 1, KFhSeparator); sl@0: if(!last) sl@0: { sl@0: return ENotFhStr; sl@0: } sl@0: return *(last + 1) == 0 ? EFhMainDbStr : EFhStr; sl@0: } sl@0: sl@0: /** sl@0: Removes all invalid characters in aFileName. sl@0: If the file name contains handles (so that's a private secure database related name), the additional sl@0: information (handles, flags, object addresses) has to be excluded from the name in order to make it usable sl@0: by the file system. sl@0: sl@0: The private file name format is (see FhExtractAndStore() comments): sl@0: sl@0: "||" sl@0: sl@0: Before opening or creating a file, SQLite will call TVfs::FullPathName() passing to the function the name of sl@0: the file and expecting the full file name (including path) as an output from the function. sl@0: After the TVfs::FullPathName() call, the full file name, returned to SQLite, will have the following format: sl@0: sl@0: "||" sl@0: sl@0: FhConvertToFileName() is called from TVfs::Open() and will remove the leading and trialing '|' characters from sl@0: the file name. sl@0: sl@0: @param aFileName Input/Output parameter. The cleaned file name will be copied there. sl@0: @param aPrivateDir The SQL server private data cage. sl@0: sl@0: @see TVfs::Open() sl@0: @see TFhStrType sl@0: @see FhStringProps() sl@0: sl@0: @internalComponent sl@0: */ sl@0: static void FhConvertToFileName(TDes& aFileName, const TDesC& aPrivateDir) sl@0: { sl@0: TInt firstPos = aFileName.Locate(TChar(KFhSeparator)); sl@0: if(firstPos >= 0) sl@0: { sl@0: aFileName.Delete(firstPos, 1); sl@0: TInt lastPos = aFileName.LocateReverse(TChar(KFhSeparator)); sl@0: if(lastPos >= 0) sl@0: { sl@0: aFileName.Delete(lastPos, 1); sl@0: TParse parse; sl@0: (void)parse.Set(aFileName, &aPrivateDir, 0);//the file name should be verified by the server sl@0: aFileName.Copy(parse.FullName()); sl@0: } sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Extracts the read-only flag and RMessage address from aDbFileName and stores them in the single COsLayerData instance. sl@0: sl@0: @param aDbFileName Input/output parameter. The file name. sl@0: It will be reformatted and won't contain the already extracted data. sl@0: The aDbFileName format is: sl@0: @code sl@0: || sl@0: @endcode sl@0: sl@0: @see TVfs::Open() sl@0: @see TFhStrType sl@0: @see FhStringProps() sl@0: sl@0: @internalComponent sl@0: sl@0: @panic Sqlite 12 In _DEBUG mode - invalid position of the "|" character in the file name. sl@0: @panic Sqlite 12 In _DEBUG mode - no RMessage2 pointer can be extracted from the file name. sl@0: @panic Sqlite 12 In _DEBUG mode - the extracted RMessage2 pointer is NULL. sl@0: */ sl@0: static void FhExtractAndStore(TDes& aDbFileName) sl@0: { sl@0: TInt fhStartPos = aDbFileName.Locate(TChar(KFhSeparator)); sl@0: __ASSERT_DEBUG(fhStartPos == KFhMarkPos, __SQLITEPANIC2(ESqliteOsPanicInvalidFhStr)); sl@0: //If this file name string contains file handles sl@0: if(fhStartPos == KFhMarkPos) sl@0: { sl@0: //Extract from aDbFileName string RMessage2 object's address sl@0: TLex lex(aDbFileName.Mid(fhStartPos + KFhMsgAddrPos, KFhMsgAddrLen)); sl@0: TUint32 addr; sl@0: TInt err = lex.Val(addr, EHex); sl@0: __ASSERT_DEBUG(err == KErrNone, __SQLITEPANIC2(ESqliteOsPanicInvalidFhStr)); sl@0: if(err == KErrNone) sl@0: { sl@0: //Cast the address to RMessage2 pointer. sl@0: const RMessage2* msg = reinterpret_cast (addr); sl@0: __ASSERT_DEBUG(msg != NULL, __SQLITEPANIC2(ESqliteOsPanicInvalidFhStr)); sl@0: if(msg) sl@0: { sl@0: //Store the data from aDbFileName in the single COsLayerData instance. sl@0: TBool readOnly = aDbFileName[fhStartPos + KFhRoPos] > '0'; sl@0: COsLayerData::Instance().StoreFhData(msg, readOnly); sl@0: //Remove: read-only flag and RMessage2 object's address sl@0: aDbFileName.Delete(fhStartPos + KFhRoPos, 1 + KFhMsgAddrLen); sl@0: } sl@0: } sl@0: } sl@0: } sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ///////////////////// TDbFile class definition /////////////////////////////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: Initializes TDbFile data members with their default values. sl@0: */ sl@0: inline TDbFile::TDbFile() : sl@0: iFileBuf(KFileBufSize), sl@0: iLockType(SQLITE_LOCK_NONE), sl@0: iReadOnly(EFalse), sl@0: iSectorSize(0), sl@0: iDeviceCharacteristics(-1), sl@0: iIsFileCreated(ETrue) sl@0: { sl@0: #ifdef _SQLPROFILER sl@0: iIsJournal = EFalse; sl@0: #endif sl@0: pMethods = 0; sl@0: SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, TDBFILE_TDBFILE, "OS;0x%X;TDbFile::TDbFile", (TUint)this)); sl@0: } sl@0: sl@0: /** sl@0: Casts the passed sqlite3_file pointer to a reference to the derived class - TDbFile&. sl@0: All sqlite3_file pointers passed to TFileIo methods are actually pointers to TDbFile instances. sl@0: So the cast is safe. sl@0: sl@0: @param aDbFile A pointer to a sqlite3_file instance sl@0: sl@0: @return A TDbFile reference. sl@0: @see TFileIo sl@0: @see TVfs sl@0: @see TDbFile sl@0: sl@0: @panic Sqlite 20 In _DEBUG mode if aDbFile is NULL. sl@0: sl@0: @internalComponent sl@0: */ sl@0: static inline TDbFile& DbFile(sqlite3_file* aDbFile) sl@0: { sl@0: __ASSERT_DEBUG(aDbFile != 0, __SQLITEPANIC2(ESqliteOsPanicNullDbFilePtr)); sl@0: return *(static_cast (aDbFile)); sl@0: } sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: ///////////////////// TFileIo class definition /////////////////////////////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Closes the file referred by aDbFile parameter. sl@0: sl@0: @param aDbFile A pointer to a TDbFile instance, than contains the file handle to be closed. sl@0: sl@0: @return SQLITE_OK sl@0: sl@0: @see TDbFile sl@0: */ sl@0: /* static */ int TFileIo::Close(sqlite3_file* aDbFile) sl@0: { sl@0: TDbFile& dbFile = ::DbFile(aDbFile); sl@0: SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, TFILEIO_CLOSE1, "OS;0x%X;TFileIo::Close", (TUint)&dbFile)); sl@0: __OS_CALL(EOsFileClose, 0, 0); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsFileClose], ::OsCallProfile(dbFile.iIsJournal, EOsFileClose), 0, 0, aDbFile, 0); sl@0: __FS_CALL(EFsOpFileClose, 0); sl@0: dbFile.iFileBuf.Close(); sl@0: return SQLITE_OK; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Reads from the file referred by the aDbFile parameter. sl@0: sl@0: @param aDbFile A pointer to a TDbFile instance, that contains the file handle to be read from. sl@0: @param aBuf Output parameter. The data read from the file will be copied there. sl@0: The buffer size must be at least aAmt bytes. sl@0: @param aAmt The amount of data to be read form the file. sl@0: @param aOffset The offset in the file where the read operation should start. sl@0: sl@0: @return SQLITE_IOERR_READ, The file read or seek operation has failed; sl@0: SQLITE_IOERR_SHORT_READ, The amount of the data read is less than aAmt; sl@0: SQLITE_IOERR_NOMEM, An out of memory condition has occured; sl@0: SQLITE_OK, The operation has completed successfully. sl@0: sl@0: If the function succeeds, COsLayerData::SetOsErrorCode() is called with KErrNone, otherwise COsLayerData::SetOsErrorCode() is called sl@0: with the reported by the OS API error. The stored error code will be used later by the SQLite API caller. sl@0: sl@0: @see COsLayerData::SetOsErrorCode() sl@0: @see TDbFile sl@0: */ sl@0: /* static */ int TFileIo::Read(sqlite3_file* aDbFile, void* aBuf, int aAmt, sqlite3_int64 aOffset) sl@0: { sl@0: TDbFile& dbFile = ::DbFile(aDbFile); sl@0: 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: __OS_CALL(EOsFileRead, 0, 0); sl@0: __COUNTER_INCR(TheSqlSrvProfilerFileRead); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsFileRead], ::OsCallProfile(dbFile.iIsJournal, EOsFileRead), aOffset, aAmt, aDbFile, 0); sl@0: TPtr8 ptr((TUint8*)aBuf, 0, aAmt); sl@0: TInt err = dbFile.iFileBuf.Read(aOffset, ptr); sl@0: TInt cnt = ptr.Length(); sl@0: TInt sqliteErr = ::Os2SqliteErr(err, SQLITE_IOERR_READ); sl@0: if(cnt != aAmt && (sqliteErr == SQLITE_OK || sqliteErr == SQLITE_IOERR_SHORT_READ)) sl@0: { sl@0: Mem::FillZ(static_cast (aBuf) + cnt, aAmt - cnt); sl@0: err = KErrEof; sl@0: sqliteErr = SQLITE_IOERR_SHORT_READ; sl@0: } sl@0: COsLayerData::Instance().SetOsErrorCode(err); sl@0: 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: return sqliteErr; sl@0: } sl@0: sl@0: //Creates a temporary file in "\temp" subdirectory of osLayerData.iSysPrivDir directory. sl@0: //If the function fails, the temp file will be closed and deleted, sl@0: //the related Symbian OS error will be returned to the caller. sl@0: /* static */TInt TFileIo::DoCreateTempFile(TDbFile& aDbFile) sl@0: { sl@0: COsLayerData& osLayerData = COsLayerData::Instance(); sl@0: //TParse2 is used in order to avoid the need of another TFileName stack based variable sl@0: class TParse2 : public TParse sl@0: { sl@0: public: sl@0: inline TFileName& FileName() sl@0: { sl@0: return static_cast (NameBuf()); sl@0: } sl@0: }; sl@0: TParse2 parse; sl@0: (void)parse.Set(osLayerData.iSysPrivDir, 0, 0);//this call can't fail sl@0: (void)parse.AddDir(KTempFileDir);//this call can't fail sl@0: __FS_CALL(EFsOpFileCreateTemp, 0); sl@0: TInt err = aDbFile.iFileBuf.Temp(osLayerData.iFs, parse.FullName(), parse.FileName(), EFileRead|EFileWrite|EDeleteOnClose); sl@0: if(err == KErrPathNotFound) sl@0: { sl@0: err = osLayerData.iFs.MkDirAll(parse.DriveAndPath()); sl@0: if(err == KErrNone) sl@0: { sl@0: err = aDbFile.iFileBuf.Temp(osLayerData.iFs, parse.FullName(), parse.FileName(), EFileRead|EFileWrite|EDeleteOnClose); sl@0: } sl@0: } sl@0: if(err == KErrNone) sl@0: { sl@0: TInt recReadBufSize = -1; sl@0: err = TVfs::DoGetDeviceCharacteristicsAndSectorSize(aDbFile, recReadBufSize); sl@0: if(err != KErrNone) sl@0: { sl@0: aDbFile.iFileBuf.Close();//With EDeleteOnClose flag set, the file will be deleted sl@0: } sl@0: else sl@0: { sl@0: (void)aDbFile.iFileBuf.SetReadAheadSize(aDbFile.iSectorSize, recReadBufSize); sl@0: aDbFile.iIsFileCreated = ETrue; sl@0: } sl@0: } sl@0: return err; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Writes to the file referred by the aDbFile parameter. sl@0: "Write beyond the end of the file" operations are allowed. sl@0: sl@0: If the write operation is in the 1st db file page and there is a registered "free pages" callback sl@0: (TDbFile::iFreePageCallback) and the free pages count is above the defined value, sl@0: then the callback will be called. sl@0: sl@0: If the file to be written to is a temp file, which is not created yet, then the file will be created. sl@0: sl@0: @param aDbFile A pointer to a TDbFile instance, that contains the file handle to be written to. sl@0: @param aData The data to be written to the file. The buffer size must be at least aAmt bytes. sl@0: @param aAmt The amount of data to be written to the file. sl@0: @param aOffset The offset in the file where the write operation should start. sl@0: sl@0: @return SQLITE_FULL, The file write or seek operation has failed. sl@0: The disk is full; sl@0: SQLITE_IOERR_NOMEM, An out of memory condition has occured; sl@0: SQLITE_OK, The operation has completed successfully. sl@0: sl@0: If the function succeeds, COsLayerData::SetOsErrorCode() is called with KErrNone, otherwise COsLayerData::SetOsErrorCode() is called sl@0: with the reported by the OS API error. The stored error code will be used later by the SQLite API caller. sl@0: sl@0: @see COsLayerData::SetOsErrorCode() sl@0: @see TDbFile sl@0: */ sl@0: /* static */ int TFileIo::Write(sqlite3_file* aDbFile, const void* aData, int aAmt, sqlite3_int64 aOffset) sl@0: { sl@0: TDbFile& dbFile = ::DbFile(aDbFile); sl@0: 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: __OS_CALL(EOsFileWrite, 0, 0); sl@0: __COUNTER_INCR(TheSqlSrvProfilerFileWrite); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsFileWrite], ::OsCallProfile(dbFile.iIsJournal, EOsFileWrite), aOffset, aAmt, aDbFile, 0); sl@0: TInt err = KErrNone; sl@0: if(!dbFile.iIsFileCreated) sl@0: {//Create a temp file if it has not been created. sl@0: err = TFileIo::DoCreateTempFile(dbFile); sl@0: } sl@0: if(err != KErrNone) sl@0: { sl@0: COsLayerData::Instance().SetOsErrorCode(err); sl@0: return ::Os2SqliteErr(err, SQLITE_FULL); sl@0: } sl@0: err = KErrAccessDenied; sl@0: if(!dbFile.iReadOnly) sl@0: { sl@0: TPtrC8 ptr((const TUint8*)aData, aAmt); sl@0: err = dbFile.iFileBuf.Write(aOffset, ptr); sl@0: } sl@0: COsLayerData::Instance().SetOsErrorCode(err); sl@0: sl@0: const TInt KFreePageCountOffset = 36;//hard-coded constant. SQLite does not offer anything - a constant or #define. sl@0: //The checks in the "if" bellow do: sl@0: // - "err == KErrNone" - check the free page count only after a successful "write"; sl@0: // - "aOffset == 0" - check the free page count only if the write operation affects the system page (at aOffset = 0); sl@0: // - "aAmt >= (KFreePageCountOffset + sizeof(int))" - check the free page count only if the amount of bytes to be written sl@0: // is more than the offset of the free page counter (othewrise the free page counter is not affected sl@0: // by this write operation); sl@0: // - "dbFile.iFreePageCallback.IsValid()" - check the free page count only if there is a valid callback; sl@0: if(err == KErrNone && aOffset == 0 && aAmt >= (KFreePageCountOffset + sizeof(int)) && dbFile.iFreePageCallback.IsValid()) sl@0: { sl@0: const TUint8* ptr = static_cast (aData) + KFreePageCountOffset; sl@0: TInt freePageCount = (ptr[0] << 24) | (ptr[1] << 16) | (ptr[2] << 8) | ptr[3]; sl@0: dbFile.iFreePageCallback.CheckAndCallback(freePageCount); sl@0: } sl@0: sl@0: TInt sqliteErr = ::Os2SqliteErr(err, SQLITE_FULL); sl@0: 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: return sqliteErr; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Truncates the file referred by the aDbFile parameter. sl@0: sl@0: @param aDbFile A pointer to a TDbFile instance, that contains the file handle. sl@0: @param aLength The new file size in bytes. sl@0: sl@0: @return SQLITE_FULL, The disk is full; sl@0: SQLITE_IOERR, This is a read-only file. sl@0: The file truncate operation has failed; sl@0: SQLITE_IOERR_NOMEM, An out of memory condition has occured; sl@0: SQLITE_OK, The operation has completed successfully. sl@0: sl@0: If the function succeeds, COsLayerData::SetOsErrorCode() is called with KErrNone, otherwise COsLayerData::SetOsErrorCode() is called sl@0: with the reported by the OS API error. The stored error code will be used later by the SQLite API caller. sl@0: sl@0: @see COsLayerData::SetOsErrorCode() sl@0: @see TDbFile sl@0: */ sl@0: /* static */ int TFileIo::Truncate(sqlite3_file* aDbFile, sqlite3_int64 aLength) sl@0: { sl@0: TDbFile& dbFile = ::DbFile(aDbFile); sl@0: SQLITE_TRACE_OS(OstTraceExt2(TRACE_INTERNALS, TFILEIO_TRUNCATE_ENTRY, "OS-Entry;0x%X;TFileIo::Truncate;aLength=%lld", (TUint)&dbFile, aLength)); sl@0: __OS_CALL(EOsFileTruncate, 0, 0); sl@0: __COUNTER_INCR(TheSqlSrvProfilerFileSetSize); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsFileTruncate], ::OsCallProfile(dbFile.iIsJournal, EOsFileTruncate), aLength, 0, aDbFile, 0); sl@0: if(dbFile.iReadOnly) sl@0: { sl@0: COsLayerData::Instance().SetOsErrorCode(KErrAccessDenied); sl@0: return SQLITE_IOERR; sl@0: } sl@0: __FS_CALL(EFsOpFileSetSize, 0); sl@0: TInt err = dbFile.iFileBuf.SetSize(aLength); sl@0: COsLayerData::Instance().SetOsErrorCode(err); sl@0: TInt sqliteErr = ::Os2SqliteErr(err, SQLITE_IOERR); sl@0: 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: return sqliteErr; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Flushes the file referred by the aDbFile parameter. sl@0: sl@0: @param aDbFile A pointer to a TDbFile instance, that contains the file handle. sl@0: sl@0: @return SQLITE_IOERR, This is a read-only file. sl@0: The file flush operation has failed; sl@0: SQLITE_IOERR_NOMEM, An out of memory condition has occured; sl@0: SQLITE_OK, The operation has completed successfully. sl@0: sl@0: If the function succeeds, COsLayerData::SetOsErrorCode() is called with KErrNone, otherwise COsLayerData::SetOsErrorCode() is called sl@0: with the reported by the OS API error. The stored error code will be used later by the SQLite API caller. sl@0: sl@0: @see COsLayerData::SetOsErrorCode() sl@0: @see TDbFile sl@0: */ sl@0: /* static */int TFileIo::Sync(sqlite3_file* aDbFile, int /* aFlags */) sl@0: { sl@0: TDbFile& dbFile = ::DbFile(aDbFile); sl@0: SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, TFILEIO_SYNC_ENTRY, "OS-Entry;0x%X;TFileIo::Sync", (TUint)&dbFile)); sl@0: __OS_CALL(EOsFileSync, 0, 0); sl@0: __COUNTER_INCR(TheSqlSrvProfilerFileSync); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsFileSync], ::OsCallProfile(dbFile.iIsJournal, EOsFileSync), 0, 0, aDbFile, 0); sl@0: if(dbFile.iReadOnly) sl@0: { sl@0: COsLayerData::Instance().SetOsErrorCode(KErrAccessDenied); sl@0: return SQLITE_IOERR; sl@0: } sl@0: __FS_CALL(EFsOpFileSync, 0); sl@0: TInt err = dbFile.iFileBuf.Flush(); sl@0: COsLayerData::Instance().SetOsErrorCode(err); sl@0: TInt sqliteErr = ::Os2SqliteErr(err, SQLITE_IOERR); sl@0: 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: return sqliteErr; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Returns the size of the file referred by the aDbFile parameter. sl@0: sl@0: @param aDbFile A pointer to a TDbFile instance, that contains the file handle. sl@0: @param aSize Output parameter. If the function completes successfully, the file size will be stored there. sl@0: sl@0: @return SQLITE_IOERR, The file size operation has failed; sl@0: SQLITE_IOERR_NOMEM, An out of memory condition has occured; sl@0: SQLITE_OK, The operation has completed successfully. sl@0: sl@0: If the function succeeds, COsLayerData::SetOsErrorCode() is called with KErrNone, otherwise COsLayerData::SetOsErrorCode() is called sl@0: with the reported by the OS API error. The stored error code will be used later by the SQLite API caller. sl@0: sl@0: @see COsLayerData::SetOsErrorCode() sl@0: @see TDbFile sl@0: */ sl@0: /* static */ int TFileIo::FileSize(sqlite3_file* aDbFile, sqlite3_int64* aSize) sl@0: { sl@0: TDbFile& dbFile = ::DbFile(aDbFile); sl@0: SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, TFILEIO_FILESIZE_ENTRY, "OS-Entry;0x%X;TFileIo::FileSize", (TUint)&dbFile)); sl@0: __OS_CALL(EOsFileFileSize, 0, 0); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsFileFileSize], ::OsCallProfile(dbFile.iIsJournal, EOsFileFileSize), 0, 0, aDbFile, 0); sl@0: __FS_CALL(EFsOpFileSize, 0); sl@0: TInt err = dbFile.iFileBuf.Size(*aSize); sl@0: COsLayerData::Instance().SetOsErrorCode(err); sl@0: TInt sqliteErr = ::Os2SqliteErr(err, SQLITE_IOERR); sl@0: 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: return sqliteErr; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Locks the file, referred by the aDbFile parameter, with the specified lock type. sl@0: Since this is a single-threaded OS porting layer implementation, the file is not actually locked - small sl@0: performance optimisation. The file lock type is stored for later use by the CheckReservedLock() call. sl@0: sl@0: @param aDbFile A pointer to a TDbFile instance, that contains the file handle. sl@0: @param aLockType Lock type: SQLITE_LOCK_NONE, SQLITE_LOCK_SHARED, SQLITE_LOCK_RESERVED, SQLITE_LOCK_PENDING or sl@0: SQLITE_LOCK_EXCLUSIVE. sl@0: sl@0: @return SQLITE_OK, The operation has completed successfully. sl@0: sl@0: @see TFileIo::CheckReservedLock() sl@0: @see TFileIo::Unlock() sl@0: sl@0: @see TDbFile sl@0: */ sl@0: /* static */ int TFileIo::Lock(sqlite3_file* aDbFile, int aLockType) sl@0: { sl@0: TDbFile& dbFile = ::DbFile(aDbFile); sl@0: __OS_CALL(EOsFileLock, 0, 0); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsFileLock], ::OsCallProfile(dbFile.iIsJournal, EOsFileLock), aLockType, 0, aDbFile, 0); sl@0: //If there is already a lock of this type or more restrictive on the database file, do nothing. sl@0: if(dbFile.iLockType >= aLockType) sl@0: { sl@0: return SQLITE_OK; sl@0: } sl@0: dbFile.iLockType = aLockType; sl@0: return SQLITE_OK; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Unlocks the file, referred by the aDbFile parameter. sl@0: Since this is a single-threaded OS porting layer implementation, the file never gets locked - small sl@0: performance optimisation. The Unlock() call only sets the stored file lock type with the aLockType value. sl@0: sl@0: @param aDbFile A pointer to a TDbFile instance, that contains the file handle. sl@0: @param aLockType Lock type: SQLITE_LOCK_NONE, SQLITE_LOCK_SHARED, SQLITE_LOCK_RESERVED, SQLITE_LOCK_PENDING or sl@0: SQLITE_LOCK_EXCLUSIVE. sl@0: sl@0: @return SQLITE_OK, The operation has completed successfully. sl@0: sl@0: @see TFileIo::CheckReservedLock() sl@0: @see TFileIo::Lock() sl@0: sl@0: @see TDbFile sl@0: */ sl@0: /* static */ int TFileIo::Unlock(sqlite3_file* aDbFile, int aLockType) sl@0: { sl@0: TDbFile& dbFile = ::DbFile(aDbFile); sl@0: __OS_CALL(EOsFileUnlock, 0, 0); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsFileUnlock], ::OsCallProfile(dbFile.iIsJournal, EOsFileUnlock), aLockType, 0, aDbFile, 0); sl@0: dbFile.iLockType = aLockType; sl@0: return SQLITE_OK; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Checks if the file lock type is SQLITE_LOCK_RESERVED or bigger. sl@0: Since this is a single-threaded OS porting layer implementation, the file never gets locked - small sl@0: performance optimisation. The CheckReservedLock() call only checks if the stored file lock type sl@0: is bigger or equal than SQLITE_LOCK_RESERVED. sl@0: sl@0: @param aDbFile A pointer to a TDbFile instance, that contains the file handle. sl@0: @param aResOut Output parameter. It should be set to 1 if the stored lock type is bigger or equal sl@0: than SQLITE_LOCK_RESERVED. sl@0: sl@0: @return SQLITE_OK. sl@0: sl@0: @see TFileIo::Lock() sl@0: @see TFileIo::Unlock() sl@0: sl@0: @see TDbFile sl@0: */ sl@0: /* static */ int TFileIo::CheckReservedLock(sqlite3_file* aDbFile, int *aResOut) sl@0: { sl@0: TDbFile& dbFile = ::DbFile(aDbFile); sl@0: __OS_CALL(EOsFileCheckReservedLock, 0, 0); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsFileCheckReservedLock], ::OsCallProfile(dbFile.iIsJournal, EOsFileCheckReservedLock), 0, 0, aDbFile, 0); sl@0: *aResOut = dbFile.iLockType >= SQLITE_LOCK_RESERVED ? 1 : 0; sl@0: return SQLITE_OK; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Performs an aOp operation on the file referred by the aDbFile parameter. sl@0: Since the only supported operation at the moment is SQLITE_FCNTL_LOCKSTATE, and the current lock type is stored as sl@0: a data memebr of TDbFile, the function implementation has been optimised - no file I/O calls. The stored file lock type sl@0: is retured if the operation is SQLITE_FCNTL_LOCKSTATE. sl@0: sl@0: Note: The range of supported operations includes KSqlFcntlRegisterFreePageCallback now. sl@0: When the function is called with aOp = KSqlFcntlRegisterFreePageCallback, then a callback will be registered sl@0: and called when the number of the free pages goes above certain threshold. sl@0: sl@0: @param aDbFile A pointer to a TDbFile instance, that contains the file handle. sl@0: @param aOp File operation type. Currently only SQLITE_FCNTL_LOCKSTATE is supported. sl@0: @param aArg An additional input/output parameter which purpose depends on the type of the current file operation. sl@0: If the file operation is SQLITE_FCNTL_LOCKSTATE, then aArg is used as an output parameter, where sl@0: the file lock type is stored. sl@0: If the operation type is KSqlFcntlRegisterFreePageCallback, then aArg points to a TSqlFreePageCallback object, sl@0: that contains the free page threshold and the callback. sl@0: sl@0: @return SQLITE_ERROR, Non-supported operation; sl@0: SQLITE_OK, The operation has completed successfully. sl@0: sl@0: @see TDbFile sl@0: */ sl@0: /* static */ int TFileIo::FileControl(sqlite3_file* aDbFile, int aOp, void* aArg) sl@0: { sl@0: TDbFile& dbFile = ::DbFile(aDbFile); sl@0: SQLITE_TRACE_OS(OstTraceExt2(TRACE_INTERNALS, TFILEIO_FILECONTROL_ENTRY, "OS-Entry;0x%X;TFileIo::FileControl;aOp=%d", (TUint)&dbFile, aOp)); sl@0: __OS_CALL(EOsFileFileControl, 0, 0); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsFileFileControl], ::OsCallProfile(dbFile.iIsJournal, EOsFileFileControl), aOp, 0, aDbFile, 0); sl@0: TInt err = KErrNone; sl@0: switch(aOp) sl@0: { sl@0: case SQLITE_FCNTL_LOCKSTATE: sl@0: *(int*)aArg = dbFile.iLockType; sl@0: break; sl@0: case KSqlFcntlRegisterFreePageCallback: sl@0: { sl@0: err = KErrArgument; sl@0: if(aArg) sl@0: { sl@0: TSqlFreePageCallback* rq = static_cast (aArg); sl@0: if(rq->IsValid()) sl@0: { sl@0: dbFile.iFreePageCallback = *rq; sl@0: err = KErrNone; sl@0: } sl@0: } sl@0: } sl@0: break; sl@0: default: sl@0: err = KErrArgument; sl@0: break; sl@0: } sl@0: COsLayerData::Instance().SetOsErrorCode(err); sl@0: TInt sqliteErr = ::Os2SqliteErr(err, SQLITE_ERROR); sl@0: 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: return sqliteErr; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Retrieves the sector size of the media of the file referred by the aDbFile parameter. sl@0: Since the sector size never changes till the file is open, the function has been optimised - no file I/O calls. sl@0: The sector size is retrieved during the TVfs::Open() call and stored in TDbFile::iSectorSize. The SectorSize() sl@0: call returns the value of TDbFile::iSectorSize. sl@0: sl@0: @param aDbFile A pointer to a TDbFile instance, that contains the file handle. sl@0: sl@0: @return The sector size. sl@0: sl@0: @panic Sqlite 19 In _DEBUG mode - TDbFile::iSectorSize is negative or 0 . sl@0: sl@0: @see TDbFile sl@0: @see TVfs::Open() sl@0: */ sl@0: /* static */ int TFileIo::SectorSize(sqlite3_file* aDbFile) sl@0: { sl@0: TDbFile& dbFile = ::DbFile(aDbFile); sl@0: __OS_CALL(EOsFileSectorSize, 0, 0); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsFileSectorSize], ::OsCallProfile(dbFile.iIsJournal, EOsFileSectorSize), 0, 0, aDbFile, 0); sl@0: __ASSERT_DEBUG(dbFile.iSectorSize > 0, __SQLITEPANIC2(ESqliteOsPanicInternalError)); sl@0: return dbFile.iSectorSize; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Retrieves the device characteristics of the device of the file referred by the aDbFile parameter. sl@0: Since the device characteristics never change till the file is open, the function has been optimised - no file I/O calls. sl@0: The device characteristics are retrieved during the TVfs::Open() call and stored in TDbFile::iDeviceCharacteristics. sl@0: The DeviceCharacteristics() call returns the value of TDbFile::iDeviceCharacteristics. sl@0: sl@0: @param aDbFile A pointer to a TDbFile instance, that contains the file handle. sl@0: sl@0: @return A bit set containing the device characteristics. sl@0: sl@0: @panic Sqlite 19 In _DEBUG mode - TDbFile::iDeviceCharacteristics is negative or 0 . sl@0: sl@0: @see TDbFile sl@0: @see TVfs::Open() sl@0: */ sl@0: /* static */ int TFileIo::DeviceCharacteristics(sqlite3_file* aDbFile) sl@0: { sl@0: TDbFile& dbFile = ::DbFile(aDbFile); sl@0: __OS_CALL(EOsFileDeviceCharacteristics, 0, 0); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsFileDeviceCharacteristics], ::OsCallProfile(dbFile.iIsJournal, EOsFileDeviceCharacteristics), 0, 0, aDbFile, 0); sl@0: __ASSERT_DEBUG(dbFile.iDeviceCharacteristics >= 0, __SQLITEPANIC2(ESqliteOsPanicInternalError)); sl@0: return dbFile.iDeviceCharacteristics; sl@0: } sl@0: sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: /////////////////////////////////// TVfs class definition /////////////////////////////////////////////////////////// sl@0: /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// sl@0: sl@0: sl@0: /** sl@0: Opens a private secure database. sl@0: Actually the database file has been created or opened by the client's process. sl@0: This function only adopts the passed in aMsg parameter file handle. sl@0: sl@0: @param aDbFile Output parameter, where the initialized file handle will be stored. sl@0: @param aMsg A reference to the current RMessage2 instance. Contains the file handle of the database created sl@0: or opened by the client's process. sl@0: @param aReadOnly True if the file is read-only. sl@0: sl@0: @return KErrNone, The operation has completed succesfully; sl@0: KErrNoMemory, Out of memory condition has occured; sl@0: Note that other system-wide error codes may also be returned. sl@0: sl@0: @see TDbFile sl@0: @see TVfs::Open() sl@0: */ sl@0: /* static */ TInt TVfs::DoOpenFromHandle(TDbFile& aDbFile, const RMessage2& aMsg, TBool aReadOnly) sl@0: { sl@0: __FS_CALL(EFsOpFileAdopt, 0); sl@0: TInt err = aDbFile.iFileBuf.AdoptFromClient(aMsg, KFhSessHandleIdx, KFhFileHandleIdx); sl@0: if(err == KErrNone) sl@0: { sl@0: aDbFile.iReadOnly = aReadOnly; sl@0: } sl@0: return err; sl@0: }; sl@0: sl@0: /** sl@0: Collects information about the drive referred by the aDriveNo parameter. sl@0: sl@0: @param aFs RFs instance. sl@0: @param aDriveNo The drive about which an information will be collected. sl@0: @param aVolumeInfo Output parameter. A reference to a TVolumeIOParamInfo object where the collected information will be stored. sl@0: sl@0: @return KErrNone, The operation has completed succesfully; sl@0: KErrNoMemory, Out of memory condition has occured; sl@0: Note that other system-wide error codes may also be returned. sl@0: sl@0: @see TVfs::Open() sl@0: */ sl@0: /* static */ inline TInt TVfs::DoGetVolumeIoParamInfo(RFs& aFs, TInt aDriveNo, TVolumeIOParamInfo& aVolumeInfo) sl@0: { sl@0: __FS_CALL(EFsOpFsVolumeIoParam, 0); sl@0: TInt err = aFs.VolumeIOParam(aDriveNo, aVolumeInfo); sl@0: SQLITE_TRACE_OS(OstTraceExt2(TRACE_INTERNALS, TVFS_DOGETVOLUMEIOPARAMINFO, "OS;0;TVfs::DoGetVolumeIoParamInfo;aDriveNo=%d;err=%d", aDriveNo, err)); sl@0: return err; sl@0: } sl@0: sl@0: //Maps disk sector sizes to SQLITE_IOCAP_ATOMIC constants sl@0: sl@0: struct TSqliteSectorSizeMap sl@0: { sl@0: TInt iSectorSize; sl@0: TInt iSqliteSectorSizeConstant; sl@0: }; sl@0: sl@0: //Used in TVfs::DoGetDeviceCharacteristics() to find which SQLITE_IOCAP_ATOMIC constant should be used sl@0: //for the specified sector size sl@0: const TSqliteSectorSizeMap KSqliteSectorSizeMap[] = sl@0: { sl@0: { 512, SQLITE_IOCAP_ATOMIC512}, sl@0: { 1024, SQLITE_IOCAP_ATOMIC1K}, sl@0: { 2048, SQLITE_IOCAP_ATOMIC2K}, sl@0: { 4096, SQLITE_IOCAP_ATOMIC4K}, sl@0: { 8192, SQLITE_IOCAP_ATOMIC8K}, sl@0: {16384, SQLITE_IOCAP_ATOMIC16K}, sl@0: {32768, SQLITE_IOCAP_ATOMIC32K}, sl@0: {65536, SQLITE_IOCAP_ATOMIC64K} sl@0: }; sl@0: sl@0: /** sl@0: Retrieves and returns in a bit set the device characteristics. sl@0: sl@0: @param aDriveInfo A TDriveInfo reference from which the device characteristics will be extracted. sl@0: @param aVolumeInfo A TVolumeIOParamInfo reference from which the device characteristics will be extracted. sl@0: sl@0: @return A bit set containing the device characteristics: sl@0: SQLITE_IOCAP_SAFE_APPEND, SQLITE_IOCAP_ATOMIC, the atomic block size. sl@0: sl@0: @see TVfs::DoGetVolumeIoParamInfo(); sl@0: @see TVfs::Open() sl@0: */ sl@0: /* static */ TInt TVfs::DoGetDeviceCharacteristics(const TDriveInfo& aDriveInfo, const TVolumeIOParamInfo& aVolumeInfo) sl@0: { sl@0: TInt deviceCharacteristics = 0; sl@0: if(aDriveInfo.iDriveAtt & (KDriveAttLocal | KDriveAttInternal)) sl@0: { sl@0: deviceCharacteristics |= SQLITE_IOCAP_SAFE_APPEND;//Data written first, file size updated second sl@0: } sl@0: if(aDriveInfo.iDriveAtt & KDriveAttTransaction) sl@0: { sl@0: deviceCharacteristics |= SQLITE_IOCAP_ATOMIC; sl@0: } sl@0: for(TInt i=0;i<(sizeof(KSqliteSectorSizeMap)/sizeof(KSqliteSectorSizeMap[0]));++i) sl@0: { sl@0: if(KSqliteSectorSizeMap[i].iSectorSize == aVolumeInfo.iBlockSize) sl@0: { sl@0: deviceCharacteristics |= KSqliteSectorSizeMap[i].iSqliteSectorSizeConstant; sl@0: break; sl@0: } sl@0: } sl@0: return deviceCharacteristics; sl@0: } sl@0: sl@0: /** sl@0: Retrieves and returns the sector size of the drive referred by the aDriveInfo parameter. sl@0: The sector size must be a power of two. sl@0: The sector size is extracted only if aDriveInfo refers to a removable device, otherwise the sl@0: SQLITE_DEFAULT_SECTOR_SIZE value (512 bytes) will be used as a sector size. sl@0: sl@0: @param aDriveInfo A TDriveInfo reference. sl@0: @param aVolumeInfo A TVolumeIOParamInfo reference. sl@0: sl@0: @return The sector size of the drive referred by the aDriveInfo parameter. sl@0: sl@0: @panic Sqlite 19 In _DEBUG mode - The sector size is negative, zero or is not a power of two. sl@0: sl@0: @see TVfs::Open() sl@0: */ sl@0: /* static */ TInt TVfs::DoGetSectorSize(const TDriveInfo& aDriveInfo, const TVolumeIOParamInfo& aVolumeInfo) sl@0: { sl@0: //Initialize the sectorSize variable only if: sl@0: // - aDriveInfo refers to a removable drive sl@0: // - aVolumeInfo.iBlockSize > SQLITE_DEFAULT_SECTOR_SIZE; sl@0: // - aVolumeInfo.iBlockSize is power of 2; sl@0: TInt sectorSize = SQLITE_DEFAULT_SECTOR_SIZE; sl@0: if(aDriveInfo.iDriveAtt & KDriveAttRemovable) sl@0: { sl@0: if(aVolumeInfo.iBlockSize > SQLITE_DEFAULT_SECTOR_SIZE && (aVolumeInfo.iBlockSize & (aVolumeInfo.iBlockSize - 1)) == 0) sl@0: { sl@0: sectorSize = aVolumeInfo.iBlockSize; sl@0: } sl@0: } sl@0: __ASSERT_DEBUG(sectorSize > 0 && (sectorSize & (sectorSize - 1)) == 0, __SQLITEPANIC2(ESqliteOsPanicInternalError)); sl@0: return sectorSize; sl@0: } sl@0: sl@0: /** sl@0: Retrieves in a bit set the device characteristics of the device of the file referred by the aDbFile parameter. sl@0: Retrieves the sector size of the drive of the file referred by the aDbFile parameter. sl@0: The sector size and the device characteristics will be stored in iSectorSize and iDeviceCharacteristics TDbFile data members. sl@0: The stored values will be used later by TFileIo::DeviceCharacteristics() and TFileIo::SectorSize(). sl@0: sl@0: @param aDbFile Input/Output parameter. A TDriveInfo reference. The collected information will be stored in TDbDrive sl@0: data members. sl@0: @param aRecReadBufSize Output parameter. The recommended buffer size for optimised reading performance. sl@0: sl@0: @return KErrNone, The operation has completed succesfully; sl@0: Note that other system-wide error codes may also be returned. sl@0: sl@0: @panic Sqlite 19 In _DEBUG mode - TDbFile::iSectorSize has been already initialized. sl@0: @panic Sqlite 19 In _DEBUG mode - TDbFile::iDeviceCharacteristics has been already initialized. sl@0: sl@0: @see TVfs::DoGetDeviceCharacteristics(); sl@0: @see TVfs::DoGetSectorSize(); sl@0: @see TVfs::Open() sl@0: @see TDbFile sl@0: @see TFileIo::DeviceCharacteristics() sl@0: @see TFileIo::SectorSize() sl@0: */ sl@0: /* static */ TInt TVfs::DoGetDeviceCharacteristicsAndSectorSize(TDbFile& aDbFile, TInt& aRecReadBufSize) sl@0: { sl@0: __ASSERT_DEBUG(aDbFile.iDeviceCharacteristics < 0, __SQLITEPANIC2(ESqliteOsPanicInternalError)); sl@0: __ASSERT_DEBUG(aDbFile.iSectorSize <= 0, __SQLITEPANIC2(ESqliteOsPanicInternalError)); sl@0: TInt driveNo; sl@0: TDriveInfo driveInfo; sl@0: __FS_CALL(EFsOpFileDrive, 0); sl@0: TInt err = aDbFile.iFileBuf.Drive(driveNo, driveInfo); sl@0: if(err != KErrNone) sl@0: { sl@0: return err; sl@0: } sl@0: TVolumeIOParamInfo volumeInfo; sl@0: err = TVfs::DoGetVolumeIoParamInfo(COsLayerData::Instance().iFs, driveNo, volumeInfo); sl@0: if(err != KErrNone) sl@0: { sl@0: return err; sl@0: } sl@0: aDbFile.iDeviceCharacteristics = TVfs::DoGetDeviceCharacteristics(driveInfo, volumeInfo); sl@0: aDbFile.iSectorSize = TVfs::DoGetSectorSize(driveInfo, volumeInfo); sl@0: aRecReadBufSize = volumeInfo.iRecReadBufSize; sl@0: 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: return KErrNone; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: The behaviour of the RFile/RFile64::SetSize operation is not atomic for non-rugged drives. sl@0: When RFile/RFile64::SetSize() is called 2 operations occurs:- sl@0: sl@0: 1)The cluster chain of the file is updated. sl@0: 2)The new file size is added to the file cache. sl@0: sl@0: If a power loss occurs after a SetSize there is a chance that the cluster chain was updated sl@0: but the new file size is not yet flushed to the file. This puts the file into an inconsistent state. sl@0: This is most likely to occur in the journal file where the time between a SetSize and Flush can sl@0: be long. sl@0: sl@0: For this reason this check is added when the file is opened to see if the end of the file can sl@0: be read straight away, if an error is returned then it is assumed that the SetSize has not be sl@0: completed previously. In this case the file is deleted and re-created. sl@0: sl@0: @param aDbFile A pointer to a TDbFile instance, that contains the file handle. sl@0: @param aFname A string of 16-bit wide characters containing name of the file to be checked. sl@0: @param aFmode The mode in which the file is opened. These mode are documented in TFileMode. sl@0: sl@0: @return KErrNone, The operation has completed succesfully; sl@0: Note that other system-wide error codes may also be returned. sl@0: @see TFileMode sl@0: @see TVfs::Open() sl@0: @see TDbFile sl@0: */ sl@0: /* static */ TInt TVfs::DoFileSizeCorruptionCheck(TDbFile& aDbFile, const TDesC& aFname, TInt aFmode) sl@0: { sl@0: const TInt KMinSize = 16; sl@0: TInt64 size; sl@0: TInt err = KErrNone ; sl@0: TBuf8 buf; sl@0: sl@0: err = aDbFile.iFileBuf.Size(size); sl@0: if (err != KErrNone) sl@0: { sl@0: return err; sl@0: } sl@0: TBool IsMinFileSize = (size >= KMinSize); sl@0: sl@0: if (IsMinFileSize) sl@0: { sl@0: err = aDbFile.iFileBuf.Read(size - KMinSize, buf); sl@0: } sl@0: sl@0: if (err == KErrCorrupt || err == KErrEof || !IsMinFileSize) sl@0: { sl@0: COsLayerData& osLayerData = COsLayerData::Instance(); sl@0: sl@0: aDbFile.iFileBuf.Close(); sl@0: __SQLITETRACE_OSEXPR(TInt err2 =) osLayerData.iFs.Delete(aFname); sl@0: 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: err = aDbFile.iFileBuf.Create(osLayerData.iFs, aFname, aFmode); sl@0: SQLITE_TRACE_OS(OstTraceExt2(TRACE_INTERNALS, TVFS_DOFILESIZECORRUPTIONCHECK2, "OS;0x%X;TVfs::DoFileSizeCorruptionCheck;createErr=%d", (TUint)&aDbFile, err)); sl@0: } sl@0: return err; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Opens or creates a file which name is in the aFileName parameter. sl@0: If the function succeeds, the file handle and other related information will be stored in the place pointed by the sl@0: aDbFile parameter, a memory block of sizeof(TDbFile) size for which is allocated by the caller. sl@0: The function will also retrieve the sector size and the device characteristics and store them in aDbFile, sl@0: which is actually a TDbFile pointer, for later use. sl@0: sl@0: @param aFileName Zero-terminated, UTF8 encoded file name. sl@0: If aFileName is NULL then a temporary file is created. sl@0: @param aDbFile Output parameter. The file handle and other related information will be stored there. sl@0: @param aFlags "Open/Create" input flags: sl@0: SQLITE_OPEN_DELETEONCLOSE, sl@0: SQLITE_OPEN_READWRITE, sl@0: SQLITE_OPEN_EXCLUSIVE, sl@0: SQLITE_OPEN_CREATE sl@0: @param aOutFlags "Open/Create" output flags: sl@0: SQLITE_OPEN_READWRITE, sl@0: SQLITE_OPEN_READONLY sl@0: sl@0: @return SQLITE_CANTOPEN, The aFileName parameter cannot be converted to UTF16. sl@0: Any other file I/O error will also be reported as SQLITE_CANTOPEN; sl@0: SQLITE_IOERR_NOMEM, An out of memory condition has occured; sl@0: SQLITE_OK, The operation has completed successfully. sl@0: sl@0: If the function succeeds, COsLayerData::SetOsErrorCode() is called with KErrNone, otherwise COsLayerData::SetOsErrorCode() is called sl@0: with the reported by the OS API error. The stored error code will be used later by the SQLite API caller. sl@0: sl@0: @see COsLayerData::SetOsErrorCode() sl@0: @see TDbFile sl@0: */ sl@0: /* static */ int TVfs::Open(sqlite3_vfs* aVfs, const char* aFileName, sqlite3_file* aDbFile, int aFlags, int* aOutFlags) sl@0: { sl@0: __OS_CALL(EOsVfsOpen, 0, 0); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsVfsOpen], ::OsCallProfile(EFalse, EOsVfsOpen), 0, 0, aDbFile, aFileName); sl@0: COsLayerData& osLayerData = COsLayerData::Instance(); sl@0: new (aDbFile) TDbFile; sl@0: TDbFile& dbFile = ::DbFile(aDbFile); sl@0: if(!aFileName) sl@0: { sl@0: //It is to create and open a temp file if aFileName is NULL. In this case, sl@0: //we will defer the file creation util it is needed. sl@0: dbFile.pMethods = &TheFileIoApi; sl@0: dbFile.iIsFileCreated = EFalse; sl@0: if(aOutFlags) sl@0: { sl@0: *aOutFlags = SQLITE_OPEN_READWRITE; sl@0: } sl@0: osLayerData.SetOsErrorCode(KErrNone); sl@0: return SQLITE_OK; sl@0: } sl@0: TFileName fname; sl@0: if(!::ConvertToUnicode(aFileName, fname)) sl@0: { sl@0: osLayerData.SetOsErrorCode(KErrBadName); sl@0: return SQLITE_CANTOPEN; sl@0: } sl@0: 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: TFhStrType fhStrType = aFileName ? ::FhStringProps(aFileName) : ENotFhStr; sl@0: TInt err = KErrNone; sl@0: if(fhStrType == EFhMainDbStr) sl@0: {//Main db file, open from handle sl@0: const RMessage2* msg; sl@0: TBool readOnly; sl@0: osLayerData.RetrieveAndResetFhData(msg, readOnly); sl@0: err = msg != NULL ? TVfs::DoOpenFromHandle(dbFile, *msg, readOnly) : KErrGeneral; sl@0: } sl@0: else sl@0: { sl@0: if(fhStrType == EFhStr) sl@0: {//Not the main db file. Remove invalid characters in the file name sl@0: ::FhConvertToFileName(fname, osLayerData.iSysPrivDir);//If fname does not have a path, iSysPrivDir will be used sl@0: } sl@0: TInt fmode = EFileRead; sl@0: if(aFlags & SQLITE_OPEN_READWRITE) sl@0: { sl@0: fmode |= EFileWrite; sl@0: } sl@0: if(aFlags & SQLITE_OPEN_EXCLUSIVE) sl@0: { sl@0: fmode |= EFileShareExclusive; sl@0: } sl@0: if(aFlags & SQLITE_OPEN_DELETEONCLOSE) sl@0: { sl@0: fmode |= EDeleteOnClose; sl@0: } sl@0: err = KErrAccessDenied; sl@0: TInt prevErr = KErrNone; sl@0: if(aFlags & SQLITE_OPEN_CREATE) sl@0: { sl@0: __FS_CALL(EFsOpFileCreate, 0); sl@0: prevErr = err = dbFile.iFileBuf.Create(osLayerData.iFs, fname, fmode); sl@0: } sl@0: if(err != KErrNone && err != KErrNoMemory && err != KErrDiskFull) sl@0: { sl@0: __FS_CALL(EFsOpFileOpen, 0); sl@0: err = dbFile.iFileBuf.Open(osLayerData.iFs, fname, fmode); sl@0: sl@0: if(err == KErrNone && (aFlags & KJournalFileTypeBitMask)) sl@0: { sl@0: err = TVfs::DoFileSizeCorruptionCheck(dbFile, fname, fmode); sl@0: } sl@0: } sl@0: if((err != KErrNone && err != KErrNoMemory && err != KErrDiskFull) && (aFlags & SQLITE_OPEN_READWRITE)) sl@0: { sl@0: aFlags &= ~SQLITE_OPEN_READWRITE; sl@0: aFlags |= SQLITE_OPEN_READONLY; sl@0: fmode &= ~EFileWrite; sl@0: __FS_CALL(EFsOpFileOpen, 0); sl@0: err = dbFile.iFileBuf.Open(osLayerData.iFs, fname, fmode); sl@0: } sl@0: if(err != KErrNone && prevErr == KErrAccessDenied) sl@0: { sl@0: err = KErrAccessDenied; sl@0: } sl@0: } sl@0: TInt recReadBufSize = -1; sl@0: if(err == KErrNone) sl@0: { sl@0: err = TVfs::DoGetDeviceCharacteristicsAndSectorSize(dbFile, recReadBufSize); sl@0: } sl@0: osLayerData.SetOsErrorCode(err); sl@0: if(err != KErrNone) sl@0: { sl@0: __FS_CALL(EFsOpFileClose, 0); sl@0: dbFile.iFileBuf.Close(); sl@0: } sl@0: else sl@0: { sl@0: dbFile.pMethods = &TheFileIoApi; sl@0: if(fhStrType != EFhMainDbStr) sl@0: { sl@0: dbFile.iReadOnly = !(aFlags & SQLITE_OPEN_READWRITE); sl@0: } sl@0: if(aOutFlags) sl@0: { sl@0: *aOutFlags = dbFile.iReadOnly ? SQLITE_OPEN_READONLY : SQLITE_OPEN_READWRITE; sl@0: } sl@0: (void)dbFile.iFileBuf.SetReadAheadSize(dbFile.iSectorSize, recReadBufSize); sl@0: } sl@0: #ifdef _SQLPROFILER sl@0: dbFile.iIsJournal = aFlags & KJournalFileTypeBitMask; sl@0: #endif sl@0: TInt sqliteErr = ::Os2SqliteErr(err, SQLITE_CANTOPEN); sl@0: 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: return sqliteErr; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Deletes a file which name is in the aFileName parameter. sl@0: sl@0: @param aFileName Zero-terminated, UTF8 encoded file name. sl@0: sl@0: @return SQLITE_ERROR, The aFileName parameter cannot be converted to UTF16. sl@0: The file name refers to a private secure database; sl@0: SQLITE_IOERR_NOMEM, An out of memory condition has occured; sl@0: SQLITE_IOERR_DELETE,The delete file operation has failed; sl@0: SQLITE_OK, The operation has completed successfully. sl@0: sl@0: If the function succeeds, COsLayerData::SetOsErrorCode() is called with KErrNone, otherwise COsLayerData::SetOsErrorCode() is called sl@0: with the reported by the OS API error. The stored error code will be used later by the SQLite API caller. sl@0: sl@0: @see COsLayerData::SetOsErrorCode() sl@0: */ sl@0: /* static */ int TVfs::Delete(sqlite3_vfs* aVfs, const char* aFileName, int /*aSyncDir*/) sl@0: { sl@0: __OS_CALL(EOsVfsDelete, 0, 0); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsVfsDelete], ::OsCallProfile(EFalse, EOsVfsDelete), 0, 0, 0, aFileName); sl@0: COsLayerData& osLayerData = COsLayerData::Instance(); sl@0: TBuf fname; sl@0: if(!::ConvertToUnicode(aFileName, fname)) sl@0: { sl@0: osLayerData.SetOsErrorCode(KErrBadName); sl@0: SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, TVFS_DELETE1, "OS;0;TVfs::Delete;err=KErrBadName")); sl@0: return SQLITE_ERROR; sl@0: } sl@0: SQLITE_TRACE_OS(OstTraceExt1(TRACE_INTERNALS, TVFS_DELETE_ENTRY, "OS-Entry;0;TVfs::Delete;fname=%S", __SQLITEPRNSTR(fname))); sl@0: TFhStrType fhStrType = FhStringProps(aFileName); sl@0: if(fhStrType == EFhMainDbStr) sl@0: {//Deleting files not in your own private data cage - not allowed! sl@0: osLayerData.SetOsErrorCode(KErrPermissionDenied); sl@0: SQLITE_TRACE_OS(OstTraceExt1(TRACE_INTERNALS, TVFS_DELETE2, "OS;0;TVfs::Delete;err=KErrPermissionDenied;fname=%S", __SQLITEPRNSTR(fname))); sl@0: return SQLITE_ERROR; sl@0: } sl@0: if(fhStrType == EFhStr) sl@0: { sl@0: ::FhConvertToFileName(fname, osLayerData.iSysPrivDir);//If fname does not have a path, iSysPrivDir will be used sl@0: } sl@0: __FS_CALL(EFsOpFileDelete, 0); sl@0: TInt err = osLayerData.iFs.Delete(fname); sl@0: osLayerData.SetOsErrorCode(err); sl@0: TInt sqliteErr = ::Os2SqliteErr(err, SQLITE_IOERR_DELETE); sl@0: SQLITE_TRACE_OS(OstTraceExt2(TRACE_INTERNALS, TVFS_DELETE_EXIT, "OS-Exit;0;TVfs::Delete;err=%d;sqliteErr=%d", err, sqliteErr)); sl@0: return sqliteErr; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Retrieves an information about a file which name is in the aFileName parameter. sl@0: The requested information type can be: does the file exist, is the file read-only or read/write. sl@0: sl@0: @param aFileName Zero-terminated, UTF8 encoded file name. sl@0: @param aFlags This parameter can be one of: SQLITE_ACCESS_READ, SQLITE_ACCESS_EXISTS or SQLITE_ACCESS_READWRITE. sl@0: @param aResOut Output parameter, set to 1 if the tested condition is true, 0 otherwise. sl@0: sl@0: @return SQLITE_OK, The call has completed successfully, sl@0: SQLITE_IOERR_NOMEM, An out of memory conditon has occured, sl@0: SQLITE_IOERR_ACCESS,File I/O error; sl@0: sl@0: If the function succeeds, COsLayerData::SetOsErrorCode() is called with KErrNone, otherwise COsLayerData::SetOsErrorCode() is called sl@0: with the reported by the OS API error. The stored error code will be used later by the SQLite API caller. sl@0: sl@0: @see COsLayerData::SetOsErrorCode() sl@0: */ sl@0: /* static */ int TVfs::Access(sqlite3_vfs* aVfs, const char* aFileName, int aFlags, int* aResOut) sl@0: { sl@0: __OS_CALL(EOsVfsAccess, 0, 0); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsVfsAccess], ::OsCallProfile(EFalse, EOsVfsAccess), aFlags, 0, 0, aFileName); sl@0: COsLayerData& osLayerData = COsLayerData::Instance(); sl@0: TBuf fname; sl@0: if(!::ConvertToUnicode(aFileName, fname)) sl@0: { sl@0: osLayerData.SetOsErrorCode(KErrGeneral); sl@0: SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, TVFS_ACCESS1, "OS;0;TVfs::Access;err=KErrGeneral")); sl@0: return SQLITE_IOERR_ACCESS; sl@0: } sl@0: 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: TFhStrType fhStrType = ::FhStringProps(aFileName); sl@0: if(fhStrType == EFhStr) sl@0: { sl@0: ::FhConvertToFileName(fname, osLayerData.iSysPrivDir);//If fname does not have a path, iSysPrivDir will be used sl@0: } sl@0: TEntry entry; sl@0: __FS_CALL(EFsOpFsEntry, 0); sl@0: TInt err = osLayerData.iFs.Entry(fname, entry); sl@0: if(aFlags == SQLITE_ACCESS_EXISTS && err == KErrNotFound) sl@0: { sl@0: osLayerData.SetOsErrorCode(KErrNone); sl@0: *aResOut = 0; sl@0: SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, TVFS_ACCESS_EXIT1, "OS-Exit;0;TVfs::Access;Exists-NoFound")); sl@0: return SQLITE_OK; sl@0: } sl@0: if(err != KErrNone) sl@0: { sl@0: osLayerData.SetOsErrorCode(err); sl@0: SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, TVFS_ACCESS_EXIT2, "OS-Exit;0;TVfs::Access;err=%d", err)); sl@0: return err == KErrNoMemory ? SQLITE_IOERR_NOMEM : SQLITE_IOERR_ACCESS; sl@0: } sl@0: *aResOut = 0; sl@0: switch(aFlags) sl@0: { sl@0: case SQLITE_ACCESS_READ: sl@0: *aResOut = entry.IsReadOnly(); sl@0: break; sl@0: case SQLITE_ACCESS_EXISTS: sl@0: *aResOut = 1; sl@0: break; sl@0: case SQLITE_ACCESS_READWRITE: sl@0: *aResOut = !entry.IsReadOnly(); sl@0: break; sl@0: default: sl@0: break; sl@0: } sl@0: osLayerData.SetOsErrorCode(KErrNone); sl@0: SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, TVFS_ACCESS_EXIT3, "OS-Exit;0;TVfs::Access;aResOut=%d", *aResOut)); sl@0: return SQLITE_OK; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Accepts UTF8 encoded, zero-terminated file as an input argument in the aRelative parameter sl@0: and constructs the full file path in the aBuf output parameter. sl@0: sl@0: If the format of aRelative argument is <[SID]FileName.[EXT]>, then the database file name will be sl@0: treated as a name of a secure database file which has to be created/opened in the server's private sl@0: directory on the system drive. sl@0: sl@0: If the format of aRelative argument is , then the database file name sl@0: will be treated as a name of a secure database file which has to be created/opened in the server's sl@0: private directory on drive. sl@0: sl@0: If the format of aRelative argument is , then the database file name sl@0: will be treated as a name of a non-secure database file in directory. sl@0: If aRelative contains file handles, then it will be treated as a name of a file belonging to server's sl@0: private data cage. sl@0: sl@0: @param aRelative The input file name, zero-terminated, UTF8 encoded. sl@0: @param aBufLen The output buffer length. sl@0: @param aBuf Output buffer for the constructed full file name path. The allocated buffer length must be at least aBufLen bytes. sl@0: sl@0: @return SQLITE_ERROR, The aRelative parameter is NULL or cannot be converted to UTF16; sl@0: SQLITE_OK The operation has completed successfully. sl@0: */ sl@0: /* static */ int TVfs::FullPathName(sqlite3_vfs* aVfs, const char* aRelative, int aBufLen, char* aBuf) sl@0: { sl@0: __OS_CALL(EOsVfsFullPathName, 0, 0); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsVfsFullPathName], ::OsCallProfile(EFalse, EOsVfsFullPathName), aBufLen, 0, 0, aRelative); sl@0: COsLayerData& osLayerData = COsLayerData::Instance(); sl@0: osLayerData.StoreFhData(NULL, EFalse); sl@0: //Convert the received file name to UTF16 sl@0: TBuf fname; sl@0: if(!::ConvertToUnicode(aRelative, fname)) sl@0: { sl@0: SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, TVFS_FULLPATHNAME1, "OS;0;TVfs::FullPathName;err=SQLITE_ERROR")); sl@0: return SQLITE_ERROR; sl@0: } sl@0: SQLITE_TRACE_OS(OstTraceExt2(TRACE_INTERNALS, TVFS_FULLPATHNAME_ENTRY, "OS-Entry;0;TVfs::FullPathName;fname=%S;aBufLen=%d", __SQLITEPRNSTR(fname), aBufLen)); sl@0: //Zero-terminate the converted file name sl@0: fname.Append(TChar(0)); sl@0: TParse parse; sl@0: TFhStrType strType = ::FhStringProps(aRelative);//Detect string type - it may not be a real file name sl@0: if(strType == EFhMainDbStr) sl@0: {//The additonal information has to be extracted and fname reformatted, because SQLITE will sl@0: //use the returned full file name when making a decission to share the cache. sl@0: ::FhExtractAndStore(fname); sl@0: (void)parse.Set(fname, 0, 0);//the file name has to be verified by the file server sl@0: } sl@0: else sl@0: { sl@0: (void)parse.Set(fname, &osLayerData.iSysPrivDir, 0);//If fname does not have a path, iSysPrivDir will be used sl@0: } sl@0: TPtr8 dest8(reinterpret_cast (aBuf), aBufLen); sl@0: if(!::ConvertFromUnicode(parse.FullName(), dest8)) sl@0: {//Zero the stored fh data, because it has been initialized by the FhExtractAndStore(fname) call (couple of lines above) sl@0: osLayerData.StoreFhData(NULL, EFalse); sl@0: SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, TVFS_FULLPATHNAME_EXIT1, "OS-Exit;0;TVfs::FullPathName;err=SQLITE_ERROR")); sl@0: return SQLITE_ERROR; sl@0: } sl@0: SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, TVFS_FULLPATHNAME_EXIT2, "OS-Exit;0;TVfs::FullPathName;err=SQLITE_OK")); sl@0: return SQLITE_OK; sl@0: } sl@0: sl@0: /** sl@0: SQLite OS porting layer API. sl@0: sl@0: Generates a set of random numbers and stores them in the aBuf output parameter. sl@0: sl@0: @param aBufLen The output buffer length. sl@0: @param aBuf Output buffer for the generated random numbers. The allocated buffer length must be at least aBufLen bytes. sl@0: sl@0: @return The length of the used part of the output buffer. sl@0: */ sl@0: /* static */ int TVfs::Randomness(sqlite3_vfs* aVfs, int aBufLen, char* aBuf) sl@0: { sl@0: __OS_CALL(EOsVfsRandomness, 0, 0); sl@0: __OSTIME_COUNTER(TheOsCallTicks[EOsVfsRandomness], ::OsCallProfile(EFalse, EOsVfsRandomness), aBufLen, 0, 0, 0); sl@0: COsLayerData& osLayerData = COsLayerData::Instance(); sl@0: const TInt KRandIterations = aBufLen / sizeof(int); sl@0: for(TInt i=0;i