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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // The Symbian OS porting layer - multi-threaded implementation.
15 // Platform dependend implementation of the static mutexes and the file session API.
22 #include "os_symbian.h"
25 #include "SqliteUtil.h"
26 #include "OstTraceDefinitions.h"
27 #ifdef OST_TRACE_COMPILER_IN_USE
28 #include "os_symbian_emulTraces.h"
30 #include "SqliteTraceDef.h"
32 #ifdef SQLITE_OS_SYMBIAN
34 const TUid KSqliteUid = {0x10285A79};//See UID3 in SQLite3.mmp file - it should be the same
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
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.
48 NONSHARABLE_CLASS(TWsdMap)
55 const TUint8* iKey; //Global variable - key (a pointer the initial variable value)
56 TUint8* iData; //Global variable - data
60 enum {KBufferSize = 4096}; //WSD buffer size in bytes
61 enum {KMaxEntries = 37}; //Max No. of entries in the hash table - prime number
67 Mem::FillZ(iTable, sizeof(iTable));
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
79 TUint8* Find(const TUint8* aKey, TInt aLength)
81 TInt idx = Hash((TUint)aKey);
83 while(iTable[idx].iKey != aKey && iTable[idx].iKey != NULL && ++cnt < KMaxEntries)
85 idx = ++idx % KMaxEntries;
87 __ASSERT_ALWAYS(cnt < KMaxEntries, __SQLITEPANIC2(ESqliteOsPanicMaxKeysExceeded));
90 Add(idx, aKey, aLength);
92 return iTable[idx].iData;
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
101 TUint Hash(TUint aKey) const
103 return (aKey * (aKey + 3)) % KMaxEntries;
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
112 void Add(TInt aIdx, const TUint8* aKey, TInt aLength)
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);
120 //////////////// DEBUG prints /////////////////
121 //for(TInt i=0;i<KMaxEntries;++i)
123 // RDebug::Print(_L("%d %X %X %d\r\n"), i, (TUint)iTable[i].iKey, (TUint)iTable[i].iData, Hash((TUint)iTable[i].iKey));
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
136 "Process local storage" structure, used for managing SQLite global variables.
145 NONSHARABLE_STRUCT(TPls)
148 TStaticMutex iStaticMutex[KStaticMutexCount];
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.
157 @param aPls A pointer to the process local storage.
158 @return KErrNone The process local storage has been successfully initialized.
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
166 static TInt PlsInitialize(TPls* aPls)
168 SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, PLSINITIALIZE_ENTRY, "OS-Entry;0;PlsInitialize"));
169 __ASSERT_ALWAYS(aPls != NULL, __SQLITEPANIC2(ESqliteOsPanicNullPls1));
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));
176 for(;idx<(sizeof(aPls->iStaticMutex)/sizeof(aPls->iStaticMutex[0])) && err==KErrNone;++idx)
178 err = aPls->iStaticMutex[idx].Create();
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));
183 //...already initialized by its constructor
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;
204 SQLITE_TRACE_OS(OstTrace0(TRACE_INTERNALS, PLSINITIALIZE_EXIT, "OS-Exit;0;PlsInitialize"));
208 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
209 ////////////////////////// TStaticFs /////////////////////////////////////////////////////////////////////////////////////////
210 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
212 TStaticFs::TStaticFs()
217 Returns a reference to the already created RFs object that is located in the process local storage.
219 @return RFs reference
221 @panic SqliteMt 3 Invalid RFs handle
222 @panic SqliteMt 5 Process local storage initialization failure
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;
232 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
233 ////////////////////////// TStaticMutex //////////////////////////////////////////////////////////////////////////////////////
234 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
236 TStaticMutex::TStaticMutex()
240 sqlite3_mutex* StaticMutex(TInt aType)
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];
249 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
250 //////////////////////// sqlite3_wsd /////////////////////////////////////////////////////////////////////////////////////
251 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
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.
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.
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.
265 int sqlite3_wsd_init(int aWsdBufSize, int aWsdEntryCount)
267 __ASSERT_ALWAYS(aWsdBufSize <= TWsdMap::KBufferSize, __SQLITEPANIC2(ESqliteOsPanicWsdBufSize));
268 __ASSERT_ALWAYS(aWsdEntryCount <= TWsdMap::KMaxEntries, __SQLITEPANIC2(ESqliteOsPanicWsdEntryCount));
273 Performs a search in the WSD map (in the process local storage) for a global variable identified by the aKey parameter.
275 @param aKey Global variable key
276 @param aLength Global variable data length in bytes
277 @return Pointer to the global variable data
279 @panic SqliteMt 11 Process local storage initialization failure
284 void* sqlite3_wsd_find(void* aKey, int aLength)
286 __ASSERT_ALWAYS(aKey != NULL, __SQLITEPANIC2(ESqliteOsPanicNullKey));
287 return ::Pls(KSqliteUid, &PlsInitialize)->iWsdMap.Find(static_cast <const TUint8*> (aKey), aLength);
290 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
291 ////////////////////////// sqlite3_vfs ///////////////////////////////////////////////////////////////////////////////////
292 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
294 sqlite3_vfs* VfsApi()
296 TPls* pls = ::Pls(KSqliteUid, &PlsInitialize);
297 __ASSERT_ALWAYS(pls != 0, __SQLITEPANIC2(ESqliteOsPanicNullPls4));
298 return &pls->iVfsApi;
301 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
302 ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
304 #endif SQLITE_OS_SYMBIAN