os/persistentdata/persistentstorage/sqlite3api/OsLayer/os_symbian_emul.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 2008-2010 Nokia Corporation and/or its subsidiary(-ies).
     2 // All rights reserved.
     3 // This component and the accompanying materials are made available
     4 // under the terms of "Eclipse Public License v1.0"
     5 // which accompanies this distribution, and is available
     6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // The Symbian OS porting layer - multi-threaded implementation. 
    15 // Platform dependend implementation of the static mutexes and the file session API.
    16 // 
    17 //
    18 
    19 /**
    20  @file
    21 */
    22 #include "os_symbian.h"
    23 #include <pls.h>
    24 #include <e32std.h>
    25 #include "SqliteUtil.h"
    26 #include "OstTraceDefinitions.h"
    27 #ifdef OST_TRACE_COMPILER_IN_USE
    28 #include "os_symbian_emulTraces.h"
    29 #endif
    30 #include "SqliteTraceDef.h"
    31 
    32 #ifdef SQLITE_OS_SYMBIAN
    33 
    34 const TUid KSqliteUid = {0x10285A79};//See UID3 in SQLite3.mmp file - it should be the same
    35 
    36 /**
    37 This class describes an object that serves as a container for SQLite global variables
    38 that are stored in an allocated WSD buffer.
    39 The global variables are organised in a map, where the pointer to the initial global variable value
    40 is used as a key.
    41 A hash table is used to speed-up the key search.
    42 A single TWsdMap object is created and stored in the process local storage - TPls.
    43 
    44 @see TPls
    45 
    46 @internalComponent
    47 */
    48 NONSHARABLE_CLASS(TWsdMap)
    49 	{
    50 	/**
    51 	Hash table entry.
    52 	*/
    53 	struct TPair
    54 		{
    55 		const TUint8* 	iKey;	//Global variable - key (a pointer the initial variable value)
    56 		TUint8* 		iData;	//Global variable - data
    57 		};
    58 		
    59 public:
    60 	enum {KBufferSize = 4096};		//WSD buffer size in bytes
    61 	enum {KMaxEntries = 37}; 		//Max No. of entries in the hash table - prime number
    62 	
    63 	TWsdMap() :
    64 		iNext(iBuffer),
    65 		iSize(0)
    66 		{
    67 		Mem::FillZ(iTable, sizeof(iTable));
    68 		}	
    69 	/**
    70 	Performs a search in the map for a global variable with aKey key.
    71 	If there is no such variable in the map, the variable will be added and the new variable will be initialized
    72 	with the data of aLength length, pointed by aKey.
    73 	@param aKey    Global variable key
    74 	@param aLength Global variable data length in bytes
    75 	@return Pointer to the global variable (located in the WSD buffer)
    76 	@panic SqliteMt  9 The global variables map is full
    77 	@panic SqliteMt 10 There is no space in the WSD buffer for the new variable
    78 	*/		
    79 	TUint8* Find(const TUint8* aKey, TInt aLength)
    80 		{
    81 		TInt idx = Hash((TUint)aKey);
    82 		TInt cnt = 0;
    83 		while(iTable[idx].iKey != aKey && iTable[idx].iKey != NULL && ++cnt < KMaxEntries)
    84 			{
    85 			idx = ++idx % KMaxEntries;
    86 			}
    87 		__ASSERT_ALWAYS(cnt < KMaxEntries, __SQLITEPANIC2(ESqliteOsPanicMaxKeysExceeded));
    88 		if(!iTable[idx].iKey)
    89 			{
    90 			Add(idx, aKey, aLength);
    91 			}
    92 		return iTable[idx].iData;
    93 		}
    94 		
    95 private:	
    96 	/**
    97 	Hash function. Calculates the index of the global variable key in the hash table.
    98 	@param aKey Global variable key (casted "const TUint8*" to "TUint")
    99 	@return Hash table index
   100 	*/	
   101 	TUint Hash(TUint aKey) const
   102 		{
   103 		return (aKey * (aKey + 3)) % KMaxEntries;
   104 		}
   105 	/**
   106 	Adds a new global variable to the WSD buffer and initializes a new entry in the hash table for the key.
   107 	@param aIdx    The entry index in the hash table
   108 	@param aKey    Global variable key
   109 	@param aLength Global variable data length in bytes
   110 	@panic SqliteMt 10 There is no space in the WSD buffer for the new variable
   111 	*/
   112 	void Add(TInt aIdx, const TUint8* aKey, TInt aLength)
   113 		{
   114 		__ASSERT_ALWAYS((iSize + aLength) <= KBufferSize, __SQLITEPANIC2(ESqliteOsPanicBufferSizeExceeded));
   115 		//Add new entry to the hash table and the intial value to the WSD buffer
   116 		iTable[aIdx].iKey = aKey;
   117 		iTable[aIdx].iData = iNext;
   118 		iNext = Mem::Copy(iNext, aKey, aLength);
   119 		iSize += aLength;
   120 		//////////////// DEBUG prints   /////////////////
   121 		//for(TInt i=0;i<KMaxEntries;++i)	
   122 		//	{
   123 		//	RDebug::Print(_L("%d %X %X %d\r\n"), i, (TUint)iTable[i].iKey, (TUint)iTable[i].iData, Hash((TUint)iTable[i].iKey));	
   124 		//	}
   125 		}
   126 		
   127 private:
   128 	TUint8	iBuffer[KBufferSize]; 	//WSD buffer
   129 	TUint8*	iNext; 					//Points to the next entry in the buffer	
   130 	TInt	iSize;  				//Amount of buffer currently allocated in bytes
   131 	TPair	iTable[KMaxEntries];	//Hash table matching the address of global varaibles to the address of the values in the WSD buffer	 
   132 		
   133 	};
   134 
   135 /**
   136 "Process local storage" structure, used for managing SQLite global variables.
   137 
   138 @see TStaticFs 
   139 @see TStaticMutex
   140 @see TWsdMap
   141 @see PlsInitialize
   142 
   143 @internalComponent
   144 */
   145 NONSHARABLE_STRUCT(TPls)
   146 	{
   147 	TStaticFs		iStaticFs;
   148 	TStaticMutex 	iStaticMutex[KStaticMutexCount];
   149 	TWsdMap			iWsdMap;
   150 	sqlite3_vfs		iVfsApi;
   151 	};
   152 
   153 /**
   154 This is a callback function used by the Pls() call to get the process local storage initialized.
   155 The function initializes the TPls data members.
   156 
   157 @param aPls A pointer to the process local storage.
   158 @return KErrNone The process local storage has been successfully initialized.
   159 
   160 @panic SqliteMt 1 Failed to connect the RFs
   161 @panic SqliteMt 2 Failed to create the static mutexes
   162 @panic SqliteMt 4 Null aPls parameter value
   163 
   164 @internalComponent
   165 */
   166 static TInt PlsInitialize(TPls* aPls)
   167 	{
   168 	SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, PLSINITIALIZE_ENTRY, "OS-Entry;0;PlsInitialize"));
   169 	__ASSERT_ALWAYS(aPls != NULL, __SQLITEPANIC2(ESqliteOsPanicNullPls1));
   170 	//Global RFs object
   171 	TInt err = aPls->iStaticFs.Connect();
   172 	SQLITE_TRACE_OS(OstTrace1(TRACE_INTERNALS, PLSINITIALIZE1, "OS;0;PlsInitialize;iStaticFs.Connect() err=%d", err));
   173 	__ASSERT_ALWAYS(err == KErrNone, __SQLITEPANIC2(ESqliteOsPanicFsCreationError));
   174 	//Static mutexes
   175 	TInt idx = 0;
   176 	for(;idx<(sizeof(aPls->iStaticMutex)/sizeof(aPls->iStaticMutex[0])) && err==KErrNone;++idx)
   177 		{
   178 		err = aPls->iStaticMutex[idx].Create();
   179 		}
   180 	SQLITE_TRACE_OS(OstTraceExt2(TRACE_INTERNALS, PLSINITIALIZE2, "OS;0;PlsInitialize;iStaticMutex[%d].Create() err=%d", idx, err));
   181 	__ASSERT_ALWAYS(err == KErrNone , __SQLITEPANIC2(ESqliteOsPanicMutexCreationError));
   182 	//WSD map
   183 	//...already initialized by its constructor
   184 	//sqlite3_vfs object	
   185 	aPls->iVfsApi.iVersion 		= 1;
   186 	aPls->iVfsApi.szOsFile 		= sizeof(TDbFile);
   187 	aPls->iVfsApi.mxPathname 	= KMaxFileName;
   188 	aPls->iVfsApi.pNext 		= NULL;
   189 	aPls->iVfsApi.zName 		= "SymbianSqliteMt";
   190 	aPls->iVfsApi.pAppData 		= NULL;
   191 	aPls->iVfsApi.xOpen 		= &TVfs::Open;
   192 	aPls->iVfsApi.xDelete 		= &TVfs::Delete;
   193 	aPls->iVfsApi.xAccess 		= &TVfs::Access;
   194 	aPls->iVfsApi.xFullPathname	= &TVfs::FullPathName;
   195 	aPls->iVfsApi.xDlOpen 		= NULL;
   196 	aPls->iVfsApi.xDlError 		= NULL;
   197 	aPls->iVfsApi.xDlSym 		= NULL;
   198 	aPls->iVfsApi.xDlClose 		= NULL;
   199 	aPls->iVfsApi.xRandomness 	= &TVfs::Randomness;
   200 	aPls->iVfsApi.xSleep 		= &TVfs::Sleep;
   201 	aPls->iVfsApi.xCurrentTime 	= &TVfs::CurrentTime;
   202 	aPls->iVfsApi.xGetLastError	= &TVfs::GetLastError;
   203 	
   204 	SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, PLSINITIALIZE_EXIT, "OS-Exit;0;PlsInitialize"));
   205 	return KErrNone;
   206 	}
   207 
   208 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   209 //////////////////////////  TStaticFs  /////////////////////////////////////////////////////////////////////////////////////////
   210 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   211 
   212 TStaticFs::TStaticFs()
   213 	{
   214 	}
   215 
   216 /**
   217 Returns a reference to the already created RFs object that is located in the process local storage.
   218 
   219 @return RFs reference
   220 
   221 @panic SqliteMt 3 Invalid RFs handle
   222 @panic SqliteMt 5 Process local storage initialization failure
   223 */
   224 RFs& TStaticFs::Fs()
   225 	{
   226 	TPls* pls = ::Pls(KSqliteUid, &PlsInitialize);
   227 	__ASSERT_ALWAYS(pls != 0, __SQLITEPANIC2(ESqliteOsPanicNullPls2));
   228 	__ASSERT_DEBUG(pls->iStaticFs.iFs.Handle() != KNullHandle, __SQLITEPANIC2(ESqliteOsPanicInvalidFs));
   229 	return pls->iStaticFs.iFs;
   230 	}
   231 
   232 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   233 //////////////////////////  TStaticMutex  //////////////////////////////////////////////////////////////////////////////////////
   234 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   235 
   236 TStaticMutex::TStaticMutex()
   237 	{
   238 	}
   239 
   240 sqlite3_mutex* StaticMutex(TInt aType)
   241 	{
   242 	TPls* pls = ::Pls(KSqliteUid, &PlsInitialize);
   243 	__ASSERT_ALWAYS(pls != 0, __SQLITEPANIC2(ESqliteOsPanicNullPls3));
   244 	__ASSERT_ALWAYS((TUint)aType < (sizeof(pls->iStaticMutex)/sizeof(pls->iStaticMutex[0])), __SQLITEPANIC2(ESqliteOsPanicInvalidMutexType));
   245 	return &pls->iStaticMutex[aType];
   246 	}
   247 
   248 
   249 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   250 ////////////////////////    sqlite3_wsd    /////////////////////////////////////////////////////////////////////////////////////
   251 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   252 
   253 /**
   254 The implementation of this function does not do anything special apart from checking that the requested WSD buffer size and
   255 global variables count (WSD entries) are not bigger than the max TWsdMap buffer size and max TWsdMap entry count.
   256 
   257 @param aWsdBufSize    SQLite expects the size of the WSD buffer to be "aWsdBufSize" bytes
   258 @param aWsdEntryCount SQLite can place in the WSD buffer up to "aWsdEntryCount" global variables.
   259 
   260 @panic SqliteMt 12 The requested WSD buffer is too big. Bigger than TWsdMap::KBufferSize.
   261 @panic SqliteMt 13 The requested global variables count is too big. Bigger than TWsdMap::KMaxEntries.
   262 
   263 @see TWsdMap
   264 */
   265 int sqlite3_wsd_init(int aWsdBufSize, int aWsdEntryCount)
   266 	{
   267 	__ASSERT_ALWAYS(aWsdBufSize <= TWsdMap::KBufferSize, __SQLITEPANIC2(ESqliteOsPanicWsdBufSize));
   268 	__ASSERT_ALWAYS(aWsdEntryCount <= TWsdMap::KMaxEntries, __SQLITEPANIC2(ESqliteOsPanicWsdEntryCount));
   269 	return SQLITE_OK;	
   270 	}
   271 
   272 /**
   273 Performs a search in the WSD map (in the process local storage) for a global variable identified by the aKey parameter.
   274 
   275 @param aKey    Global variable key
   276 @param aLength Global variable data length in bytes
   277 @return        Pointer to the global variable data
   278 
   279 @panic SqliteMt 11 Process local storage initialization failure
   280 
   281 @see TWsdMap
   282 @see TPls
   283 */
   284 void* sqlite3_wsd_find(void* aKey, int aLength)
   285 	{
   286 	__ASSERT_ALWAYS(aKey != NULL, __SQLITEPANIC2(ESqliteOsPanicNullKey));
   287 	return ::Pls(KSqliteUid, &PlsInitialize)->iWsdMap.Find(static_cast <const TUint8*> (aKey), aLength);
   288 	}
   289 
   290 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   291 //////////////////////////  sqlite3_vfs     ///////////////////////////////////////////////////////////////////////////////////
   292 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   293 
   294 sqlite3_vfs* VfsApi()
   295 	{
   296 	TPls* pls = ::Pls(KSqliteUid, &PlsInitialize);
   297 	__ASSERT_ALWAYS(pls != 0, __SQLITEPANIC2(ESqliteOsPanicNullPls4));
   298 	return &pls->iVfsApi;
   299 	}
   300 
   301 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   302 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   303 
   304 #endif SQLITE_OS_SYMBIAN