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