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
|