sl@0
|
1 |
// Copyright (c) 2005-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 |
//
|
sl@0
|
15 |
|
sl@0
|
16 |
#include <hal.h>
|
sl@0
|
17 |
#include "SqlSrvMain.h" //CSqlServer
|
sl@0
|
18 |
#include "SqlSrvStartup.h" //SqlSrvVersion()
|
sl@0
|
19 |
#include "SqlSrvSession.h" //CSqlSrvSession
|
sl@0
|
20 |
#include "SqlSrvDbSysSettings.h"//TSqlDbSysSettings
|
sl@0
|
21 |
#include "SqlSrvStatementUtil.h"
|
sl@0
|
22 |
#include "SqlSrvStatement.h"
|
sl@0
|
23 |
#include "sqlite3.h" //sqlite3_enable_shared_cache()
|
sl@0
|
24 |
#include "SqliteSymbian.h" //sqlite3SymbianLibInit()
|
sl@0
|
25 |
#include "SqlCompact.h"
|
sl@0
|
26 |
#include "SqlCompactConn.h"
|
sl@0
|
27 |
#include "SqlSrvResourceProfiler.h"
|
sl@0
|
28 |
#ifdef _DEBUG
|
sl@0
|
29 |
#include <stdio.h>
|
sl@0
|
30 |
#endif
|
sl@0
|
31 |
#include "OstTraceDefinitions.h"
|
sl@0
|
32 |
#ifdef OST_TRACE_COMPILER_IN_USE
|
sl@0
|
33 |
#include "SqlSrvMainTraces.h"
|
sl@0
|
34 |
#endif
|
sl@0
|
35 |
#include "SqlTraceDef.h"
|
sl@0
|
36 |
|
sl@0
|
37 |
#ifndef SQLSRV_STARTUP_TEST
|
sl@0
|
38 |
static
|
sl@0
|
39 |
#endif
|
sl@0
|
40 |
CSqlServer* TheServer = NULL;//The single CSqlServer instance
|
sl@0
|
41 |
|
sl@0
|
42 |
#ifdef _DEBUG
|
sl@0
|
43 |
#define __SQLDEBUG_EXPR(expr) expr
|
sl@0
|
44 |
#else
|
sl@0
|
45 |
#define __SQLDEBUG_EXPR(expr)
|
sl@0
|
46 |
#endif
|
sl@0
|
47 |
|
sl@0
|
48 |
_LIT(KMatchAllDbFiles, "*");
|
sl@0
|
49 |
_LIT(KDefaultICollationDllName, "");
|
sl@0
|
50 |
|
sl@0
|
51 |
//Constants for enabling/disabling the shared cache
|
sl@0
|
52 |
enum TSharedCacheState
|
sl@0
|
53 |
{
|
sl@0
|
54 |
EDisableSharedCache = 0,
|
sl@0
|
55 |
EEnableSharedCache = 1
|
sl@0
|
56 |
};
|
sl@0
|
57 |
|
sl@0
|
58 |
#ifdef SYSLIBS_TEST
|
sl@0
|
59 |
//The "lookaside" optimisation is disabled if SYSLIBS_TEST macro is defined.
|
sl@0
|
60 |
//According to the SQLite authors recommendations, the OOM testing should be performed without this optimisation.
|
sl@0
|
61 |
const TInt KSqliteLookAsideCellSize = 0;
|
sl@0
|
62 |
const TInt KSqliteLookAsideCellCount = 0;
|
sl@0
|
63 |
#else
|
sl@0
|
64 |
//SQLite, "fixed heap cell size" constants
|
sl@0
|
65 |
//SQLite will preallocate KSqliteLookAsideCellSize * KSqliteLookAsideCellCount bytes from the heap and
|
sl@0
|
66 |
//use the allocated block for all allocation requests with size <= KSqliteLookAsideCellSize.
|
sl@0
|
67 |
//The malloc()/free() request time is constant, if the cell is retrieved/returned from/to the "fixed cell size" block.
|
sl@0
|
68 |
const TInt KSqliteLookAsideCellSize = 128;
|
sl@0
|
69 |
const TInt KSqliteLookAsideCellCount = 512;
|
sl@0
|
70 |
#endif
|
sl@0
|
71 |
|
sl@0
|
72 |
//Local function, used for comparing TSqlSecurityPair objects.
|
sl@0
|
73 |
//The keys are expected to be UTF8 encoded, zero-terminated.
|
sl@0
|
74 |
//
|
sl@0
|
75 |
//The function will panic with panic code 7 in _DEBUG mode if the key part of aLeft or
|
sl@0
|
76 |
//aRight argument is NULL.
|
sl@0
|
77 |
static TInt Compare(const TSqlSecurityPair& aLeft, const TSqlSecurityPair& aRight)
|
sl@0
|
78 |
{
|
sl@0
|
79 |
__ASSERT_DEBUG(aLeft.iKey != NULL && aRight.iKey != NULL, __SQLPANIC2(ESqlPanicInternalError));
|
sl@0
|
80 |
return ::CompareNoCase8(TPtrC8(aLeft.iKey), TPtrC8(aRight.iKey));
|
sl@0
|
81 |
}
|
sl@0
|
82 |
|
sl@0
|
83 |
/**
|
sl@0
|
84 |
Returns a reference to the sql server instance.
|
sl@0
|
85 |
|
sl@0
|
86 |
@return A reference to the sql server instance.
|
sl@0
|
87 |
|
sl@0
|
88 |
@panic SqlDb 2 If the sql server instance is NULL.
|
sl@0
|
89 |
|
sl@0
|
90 |
@internalComponent
|
sl@0
|
91 |
*/
|
sl@0
|
92 |
CSqlServer& SqlServer(void)
|
sl@0
|
93 |
{
|
sl@0
|
94 |
__ASSERT_ALWAYS(TheServer != NULL, __SQLPANIC2(ESqlPanicInvalidObj));
|
sl@0
|
95 |
return *TheServer;
|
sl@0
|
96 |
}
|
sl@0
|
97 |
|
sl@0
|
98 |
/**
|
sl@0
|
99 |
Creates new CSqlServer instance.
|
sl@0
|
100 |
The created instance will be pushed in the cleanup stack.
|
sl@0
|
101 |
|
sl@0
|
102 |
@return A pointer to the created CSqlServer instance.
|
sl@0
|
103 |
|
sl@0
|
104 |
@leave KErrNoMemory, an out of memory condition has occured;
|
sl@0
|
105 |
*/
|
sl@0
|
106 |
CSqlServer* CSqlServer::NewLC()
|
sl@0
|
107 |
{
|
sl@0
|
108 |
SQL_TRACE_INTERNALS(OstTrace0(TRACE_INTERNALS, CSQLSERVER_NEWLC_ENTRY, "Entry;0;CSqlServer::NewLC"));
|
sl@0
|
109 |
CSqlServer* self = new (ELeave) CSqlServer;
|
sl@0
|
110 |
CleanupStack::PushL(self);
|
sl@0
|
111 |
self->ConstructL();
|
sl@0
|
112 |
SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, CSQLSERVER_NEWLC_EXIT, "Exit;0x%X;CSqlServer::NewLC", (TUint)self));
|
sl@0
|
113 |
return self;
|
sl@0
|
114 |
}
|
sl@0
|
115 |
|
sl@0
|
116 |
/**
|
sl@0
|
117 |
Frees owned by CSqlServer memory and other resources.
|
sl@0
|
118 |
*/
|
sl@0
|
119 |
CSqlServer::~CSqlServer()
|
sl@0
|
120 |
{
|
sl@0
|
121 |
SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, CSQLSERVER_CSQLSERVER2_ENTRY, "Entry;0x%x;CSqlServer::~CSqlServer", (TUint)this));
|
sl@0
|
122 |
delete iCompactor;
|
sl@0
|
123 |
delete iBurEventMonitor;
|
sl@0
|
124 |
iDriveSpaceCol.ResetAndDestroy();
|
sl@0
|
125 |
sqlite3_soft_heap_limit(0);//Set to 0 the soft heap limit
|
sl@0
|
126 |
iSecurityMap.Close();
|
sl@0
|
127 |
(void)sqlite3_enable_shared_cache(static_cast <TInt> (EDisableSharedCache));
|
sl@0
|
128 |
iFlatBuf.Close();
|
sl@0
|
129 |
User::Free(iBuf);
|
sl@0
|
130 |
delete iDbConfigFiles;
|
sl@0
|
131 |
sqlite3SymbianLibFinalize();
|
sl@0
|
132 |
TheServer = NULL;
|
sl@0
|
133 |
SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, CSQLSERVER_CSQLSERVER2_EXIT, "Exit;0x%x;CSqlServer::~CSqlServer", (TUint)this));
|
sl@0
|
134 |
}
|
sl@0
|
135 |
|
sl@0
|
136 |
/**
|
sl@0
|
137 |
@param aMinLen Requested minimal byte size of the flat buffer
|
sl@0
|
138 |
|
sl@0
|
139 |
@return A reference to the server's general purpose flat bufer. The buffer cannot keep a state between calls.
|
sl@0
|
140 |
*/
|
sl@0
|
141 |
RSqlBufFlat& CSqlServer::GetFlatBufL(TInt aMinLen)
|
sl@0
|
142 |
{
|
sl@0
|
143 |
__ASSERT_DEBUG(aMinLen >= 0, __SQLPANIC(ESqlPanicBadArgument));
|
sl@0
|
144 |
__SQLLEAVE_IF_ERROR(iFlatBuf.ReAlloc(aMinLen));
|
sl@0
|
145 |
SQLPROFILER_REPORT_ALLOC(iFlatBuf.MaxSize());
|
sl@0
|
146 |
return iFlatBuf;
|
sl@0
|
147 |
}
|
sl@0
|
148 |
|
sl@0
|
149 |
/**
|
sl@0
|
150 |
Returns a 8-bit descriptor's reference to the server's general purpose buffer.
|
sl@0
|
151 |
Note that the function may reallocate the buffer if the buffer length is smaller than the requested minimal length.
|
sl@0
|
152 |
|
sl@0
|
153 |
@param aMinLen Requested minimal 8-bit character length of the buffer
|
sl@0
|
154 |
|
sl@0
|
155 |
@return TDes8 reference to the server's general purpose bufer. The buffer cannot keep a state between calls.
|
sl@0
|
156 |
*/
|
sl@0
|
157 |
TDes8& CSqlServer::GetBuf8L(TInt aMinLen)
|
sl@0
|
158 |
{
|
sl@0
|
159 |
__ASSERT_DEBUG(aMinLen >= 0, __SQLPANIC(ESqlPanicBadArgument));
|
sl@0
|
160 |
#ifdef _DEBUG
|
sl@0
|
161 |
TInt maxBufLen = iBufPtr8.MaxLength();
|
sl@0
|
162 |
maxBufLen = maxBufLen;
|
sl@0
|
163 |
#endif
|
sl@0
|
164 |
if(iBufPtr8.MaxLength() < aMinLen)
|
sl@0
|
165 |
{
|
sl@0
|
166 |
__SQLLEAVE_IF_ERROR(ReAllocBuf(aMinLen));
|
sl@0
|
167 |
}
|
sl@0
|
168 |
SQLPROFILER_REPORT_ALLOC(iBufPtr8.MaxLength());
|
sl@0
|
169 |
return iBufPtr8;
|
sl@0
|
170 |
}
|
sl@0
|
171 |
|
sl@0
|
172 |
/**
|
sl@0
|
173 |
Returns a 16-bit descriptor's reference to the server's general purpose buffer.
|
sl@0
|
174 |
Note that the function may reallocate the buffer if the buffer length is smaller than the requested minimal length.
|
sl@0
|
175 |
|
sl@0
|
176 |
@param aMinLen Requested minimal 16-bit character length of the buffer
|
sl@0
|
177 |
|
sl@0
|
178 |
@return TDes16 reference to the server's general purpose bufer. The buffer cannot keep a state between calls.
|
sl@0
|
179 |
*/
|
sl@0
|
180 |
TDes16& CSqlServer::GetBuf16L(TInt aMinLen)
|
sl@0
|
181 |
{
|
sl@0
|
182 |
__ASSERT_DEBUG(aMinLen >= 0, __SQLPANIC(ESqlPanicBadArgument));
|
sl@0
|
183 |
#ifdef _DEBUG
|
sl@0
|
184 |
TInt maxBufLen = iBufPtr16.MaxLength();
|
sl@0
|
185 |
maxBufLen = maxBufLen;
|
sl@0
|
186 |
#endif
|
sl@0
|
187 |
if(iBufPtr16.MaxLength() < aMinLen)
|
sl@0
|
188 |
{
|
sl@0
|
189 |
__SQLLEAVE_IF_ERROR(ReAllocBuf(aMinLen * sizeof(TUint16)));
|
sl@0
|
190 |
}
|
sl@0
|
191 |
SQLPROFILER_REPORT_ALLOC(iBufPtr16.MaxLength());
|
sl@0
|
192 |
return iBufPtr16;
|
sl@0
|
193 |
}
|
sl@0
|
194 |
|
sl@0
|
195 |
/**
|
sl@0
|
196 |
If iFlatBuf or iBuf allocated memory is more than KBufLimit bytes,
|
sl@0
|
197 |
then that buffer will be reallocated down to KBufLimit size.
|
sl@0
|
198 |
*/
|
sl@0
|
199 |
void CSqlServer::MinimizeBuffers()
|
sl@0
|
200 |
{
|
sl@0
|
201 |
iFlatBuf.ResetAndMinimize();
|
sl@0
|
202 |
#ifdef _DEBUG
|
sl@0
|
203 |
const TInt KBufLimit = 64;
|
sl@0
|
204 |
const TUint8* oldBuf = iBuf;
|
sl@0
|
205 |
#else
|
sl@0
|
206 |
const TInt KBufLimit = 8 * 1024;
|
sl@0
|
207 |
#endif
|
sl@0
|
208 |
if(iBufPtr8.MaxSize() > KBufLimit)
|
sl@0
|
209 |
{
|
sl@0
|
210 |
(void)ReAllocBuf(KBufLimit);
|
sl@0
|
211 |
__ASSERT_DEBUG(oldBuf == iBuf, __SQLPANIC(ESqlPanicInternalError));
|
sl@0
|
212 |
}
|
sl@0
|
213 |
}
|
sl@0
|
214 |
|
sl@0
|
215 |
/**
|
sl@0
|
216 |
Reallocates iBuf. iBuf content is not preserved.
|
sl@0
|
217 |
Sets iBufPtr8 and iBufPtr16 to point to iBuf.
|
sl@0
|
218 |
|
sl@0
|
219 |
@param aNewBufSize The new buffer size in bytes
|
sl@0
|
220 |
|
sl@0
|
221 |
@return KErrNoMemory, an out of memory condition has occurred;
|
sl@0
|
222 |
KErrNone, the operation has completed successfully;
|
sl@0
|
223 |
*/
|
sl@0
|
224 |
TInt CSqlServer::ReAllocBuf(TInt aNewBufSize)
|
sl@0
|
225 |
{
|
sl@0
|
226 |
__ASSERT_DEBUG(aNewBufSize >= 0, __SQLPANIC(ESqlPanicBadArgument));
|
sl@0
|
227 |
#ifdef _DEBUG
|
sl@0
|
228 |
const TInt KMinBufSize = 8;
|
sl@0
|
229 |
#else
|
sl@0
|
230 |
const TInt KMinBufSize = 2 * 1024;
|
sl@0
|
231 |
#endif
|
sl@0
|
232 |
const TInt KNewBufSize = Max(aNewBufSize, KMinBufSize);
|
sl@0
|
233 |
TUint8* newBuf = static_cast <TUint8*> (User::ReAlloc(iBuf, KNewBufSize));
|
sl@0
|
234 |
if(newBuf)
|
sl@0
|
235 |
{
|
sl@0
|
236 |
iBuf = newBuf;
|
sl@0
|
237 |
iBufPtr8.Set(iBuf, 0, KNewBufSize);
|
sl@0
|
238 |
iBufPtr16.Set(reinterpret_cast <TUint16*> (iBuf), 0, (TUint)KNewBufSize / sizeof(TUint16));
|
sl@0
|
239 |
return KErrNone;
|
sl@0
|
240 |
}
|
sl@0
|
241 |
else
|
sl@0
|
242 |
{//The reallocation has failed, iBuf - not changed
|
sl@0
|
243 |
iBufPtr8.Zero();
|
sl@0
|
244 |
iBufPtr16.Zero();
|
sl@0
|
245 |
return KErrNoMemory;
|
sl@0
|
246 |
}
|
sl@0
|
247 |
}
|
sl@0
|
248 |
|
sl@0
|
249 |
/**
|
sl@0
|
250 |
Creates new CSqlSrvSession instance.
|
sl@0
|
251 |
If SQLSRV_STARTUP_TEST macro is defined, then the function returns NULL.
|
sl@0
|
252 |
The "real" implementation of the function is not used in this case because the used unit test will require
|
sl@0
|
253 |
a lot of cpp files to be included into the test build (t_sqlstartup).
|
sl@0
|
254 |
|
sl@0
|
255 |
@return A pointer to the created CSqlSrvSession instance.
|
sl@0
|
256 |
|
sl@0
|
257 |
@leave KErrNoMemory, an out of memory condition has occured;
|
sl@0
|
258 |
KErrNotSupported, the client side library version differs from the server version.
|
sl@0
|
259 |
|
sl@0
|
260 |
@see CSqlSrvSession
|
sl@0
|
261 |
*/
|
sl@0
|
262 |
CSession2* CSqlServer::NewSessionL(const TVersion& aVersion, const RMessage2&) const
|
sl@0
|
263 |
{
|
sl@0
|
264 |
#ifdef SQLSRV_STARTUP_TEST
|
sl@0
|
265 |
aVersion.Name();//to prevent the compiler warning ("unused parameter").
|
sl@0
|
266 |
return NULL;
|
sl@0
|
267 |
#else
|
sl@0
|
268 |
if(!User::QueryVersionSupported(::SqlSrvVersion(), aVersion))
|
sl@0
|
269 |
{
|
sl@0
|
270 |
User::Leave(KErrNotSupported);
|
sl@0
|
271 |
}
|
sl@0
|
272 |
CSqlSrvSession* sess = CSqlSrvSession::NewL();
|
sl@0
|
273 |
return sess;
|
sl@0
|
274 |
#endif //SQLSRV_STARTUP_TEST
|
sl@0
|
275 |
}
|
sl@0
|
276 |
|
sl@0
|
277 |
/**
|
sl@0
|
278 |
CSqlServer's active object priority.
|
sl@0
|
279 |
|
sl@0
|
280 |
@internalComponent
|
sl@0
|
281 |
*/
|
sl@0
|
282 |
const TInt KSqlServerPriority = CActive::EPriorityStandard;
|
sl@0
|
283 |
|
sl@0
|
284 |
/**
|
sl@0
|
285 |
Initializes CSqlServer data members with default values.
|
sl@0
|
286 |
*/
|
sl@0
|
287 |
CSqlServer::CSqlServer() :
|
sl@0
|
288 |
CServer2(KSqlServerPriority, ESharableSessions),
|
sl@0
|
289 |
iSecurityMap(TSqlSecurityLinearOrder(&Compare), TSqlSecurityDestructor()),
|
sl@0
|
290 |
iBufPtr8(0, 0),
|
sl@0
|
291 |
iBufPtr16(0, 0)
|
sl@0
|
292 |
{
|
sl@0
|
293 |
}
|
sl@0
|
294 |
|
sl@0
|
295 |
/**
|
sl@0
|
296 |
Initializes CSqlServer instance:
|
sl@0
|
297 |
- starts the server;
|
sl@0
|
298 |
- opens sqlite library;
|
sl@0
|
299 |
- initializes the file session instance;
|
sl@0
|
300 |
- creates server's private directory on the system drive;
|
sl@0
|
301 |
- enables sqlite shared cache;
|
sl@0
|
302 |
|
sl@0
|
303 |
@leave KErrNoMemory, an out of memory condition has occured;
|
sl@0
|
304 |
Note that the function may also leave with some other database specific
|
sl@0
|
305 |
errors categorised as ESqlDbError.
|
sl@0
|
306 |
*/
|
sl@0
|
307 |
void CSqlServer::ConstructL()
|
sl@0
|
308 |
{
|
sl@0
|
309 |
#ifndef SQLSRV_STARTUP_TEST
|
sl@0
|
310 |
//Start the server only in "normal" builds, not in the case where t_sqlstartup unit test tests directly
|
sl@0
|
311 |
//the SQL server startup code.
|
sl@0
|
312 |
StartL(KSqlSrvName);
|
sl@0
|
313 |
#endif
|
sl@0
|
314 |
SQLPROFILER_SERVER_START();
|
sl@0
|
315 |
//Configure the SQLite library
|
sl@0
|
316 |
TInt sqliteErr = sqlite3_config(SQLITE_CONFIG_LOOKASIDE, KSqliteLookAsideCellSize, KSqliteLookAsideCellCount);
|
sl@0
|
317 |
__SQLLEAVE_IF_ERROR(::Sql2OsErrCode(sqliteErr, KErrArgument));
|
sl@0
|
318 |
//Open SQLITE library - this must be the first call after StartL() (os_symbian.cpp, "TheAllocator" initialization rellated).
|
sl@0
|
319 |
__SQLLEAVE_IF_ERROR(sqlite3SymbianLibInit());
|
sl@0
|
320 |
//Create buffers
|
sl@0
|
321 |
__SQLLEAVE_IF_ERROR(iFlatBuf.SetCount(0));
|
sl@0
|
322 |
//Get collation dll name
|
sl@0
|
323 |
GetCollationDllNameL();
|
sl@0
|
324 |
//Get the system drive.
|
sl@0
|
325 |
TInt sysDrive = static_cast<TInt>(RFs::GetSystemDrive());
|
sl@0
|
326 |
//Get the server private data path.
|
sl@0
|
327 |
RFs& fs = sqlite3SymbianFs();
|
sl@0
|
328 |
TFileName serverPrivatePath;
|
sl@0
|
329 |
__SQLLEAVE_IF_ERROR(fs.PrivatePath(serverPrivatePath));
|
sl@0
|
330 |
DeleteTempFilesL(sysDrive, serverPrivatePath);
|
sl@0
|
331 |
//Load config file parameter values (if config file exists) and initialize iFileData.
|
sl@0
|
332 |
TParse parse;
|
sl@0
|
333 |
__SQLLEAVE_IF_ERROR(parse.Set(KSqlSrvDefaultConfigFile, &serverPrivatePath, NULL));
|
sl@0
|
334 |
//Store the names of any existing database config files in memory
|
sl@0
|
335 |
CacheDbConfigFileNamesL(fs, serverPrivatePath);
|
sl@0
|
336 |
//Initialise the file data object
|
sl@0
|
337 |
iFileData.InitL(fs, TDriveUnit(sysDrive).Name(), serverPrivatePath, parse.FullName(), iDbConfigFiles);
|
sl@0
|
338 |
|
sl@0
|
339 |
//Set the soft heap limit (iFileData.ConfigParams() returns now a reference to the config file params, including the soft heap limit, if set)
|
sl@0
|
340 |
const TSqlSrvConfigParams& configParams = iFileData.ConfigParams();
|
sl@0
|
341 |
if(configParams.iSoftHeapLimitKb > 0)
|
sl@0
|
342 |
{
|
sl@0
|
343 |
__ASSERT_DEBUG(configParams.iSoftHeapLimitKb >= TSqlSrvConfigParams::KMinSoftHeapLimitKb &&
|
sl@0
|
344 |
configParams.iSoftHeapLimitKb <= TSqlSrvConfigParams::KMaxSoftHeapLimitKb, __SQLPANIC(ESqlPanicInternalError));
|
sl@0
|
345 |
sqlite3_soft_heap_limit(configParams.iSoftHeapLimitKb * 1024);
|
sl@0
|
346 |
}
|
sl@0
|
347 |
//Enable shared cache
|
sl@0
|
348 |
(void)sqlite3SymbianLastOsError();//clear last OS error
|
sl@0
|
349 |
TInt err = sqlite3_enable_shared_cache(static_cast <TInt> (EEnableSharedCache));
|
sl@0
|
350 |
__SQLLEAVE_IF_ERROR(::Sql2OsErrCode(err, sqlite3SymbianLastOsError()));
|
sl@0
|
351 |
//Create an empty "drive space" collection
|
sl@0
|
352 |
iDriveSpaceCol.Create(fs);
|
sl@0
|
353 |
// Create the BUR instance
|
sl@0
|
354 |
iBurEventMonitor = CSqlBurEventMonitor::NewL(*this);
|
sl@0
|
355 |
//Compactor
|
sl@0
|
356 |
iCompactor = CSqlCompactor::NewL(&SqlCreateCompactConnL, KSqlCompactStepIntervalMs);
|
sl@0
|
357 |
#ifdef _DEBUG
|
sl@0
|
358 |
//The following statements exist to prevent the failure of the OOM testing in debug mode.
|
sl@0
|
359 |
//The standard C library allocates some memory at the startup and stores a pointer to the allocated memory
|
sl@0
|
360 |
//in the TLS. During normal API OOM testing the SQL server is not restarted, it never goes down.
|
sl@0
|
361 |
//Then the TLS and the allocated memory are not released. In which case the OOM testing will fail
|
sl@0
|
362 |
//(because the standard C library performs a lazy initialization and the allocation and TLS usage will be made
|
sl@0
|
363 |
//at the point of first use of some C function. This is out of the control of the test code).
|
sl@0
|
364 |
//In order to avoid that, during the SQL server startup here, before the OOM test goes and checks what
|
sl@0
|
365 |
//is the allocated memory at the beginning, a fake sprintf() call is made in order to force the mentioned above
|
sl@0
|
366 |
//allocation in the standard C library.
|
sl@0
|
367 |
//All explanations above are true, except one case when the SQl server startup code is tested directly.
|
sl@0
|
368 |
#ifndef SQLSRV_STARTUP_TEST
|
sl@0
|
369 |
const TInt KAnyNumber = 0xAA55;
|
sl@0
|
370 |
char tmp[32];
|
sl@0
|
371 |
sprintf(tmp, "%04X", KAnyNumber);
|
sl@0
|
372 |
const TInt KGreatSize = 1024;
|
sl@0
|
373 |
__SQLLEAVE_IF_ERROR(ReAllocBuf(KGreatSize));
|
sl@0
|
374 |
#endif //SQLSRV_STARTUP_TEST
|
sl@0
|
375 |
#endif //_DEBUG
|
sl@0
|
376 |
}
|
sl@0
|
377 |
|
sl@0
|
378 |
/**
|
sl@0
|
379 |
Delete any temp files left the "temp" subdirectory in server's private directory.
|
sl@0
|
380 |
|
sl@0
|
381 |
The SQLite is configured to use shared page cache. When the shared page cache is enabled,
|
sl@0
|
382 |
those temp files created by SQLite are deleted only when the database gets closed. However,
|
sl@0
|
383 |
if during power down event the client application does not close the database,
|
sl@0
|
384 |
the temp files will never get deleted.
|
sl@0
|
385 |
This is why the SQL server should deletes all temp files during its start-up.
|
sl@0
|
386 |
|
sl@0
|
387 |
Note that all errors exept KErrNoMemory are ignored in the function body, becasuse
|
sl@0
|
388 |
the temp files deletion is not a critical operation to prevent the server start up.
|
sl@0
|
389 |
|
sl@0
|
390 |
@param aDriveNumber A drive number.
|
sl@0
|
391 |
@param aServerPath A server's private path.
|
sl@0
|
392 |
|
sl@0
|
393 |
*/
|
sl@0
|
394 |
void CSqlServer::DeleteTempFilesL(TInt aDriveNumber, const TDesC& aServerPath)const
|
sl@0
|
395 |
{
|
sl@0
|
396 |
_LIT(KTempFileDir, "temp");
|
sl@0
|
397 |
_LIT(KWildCard, "*.*");
|
sl@0
|
398 |
TDriveUnit drive(aDriveNumber);
|
sl@0
|
399 |
TDriveName driveName = drive.Name();
|
sl@0
|
400 |
TParse parse;
|
sl@0
|
401 |
(void)parse.Set(aServerPath, &driveName, 0);//this call can't fail
|
sl@0
|
402 |
(void)parse.AddDir(KTempFileDir);//this call can't fail
|
sl@0
|
403 |
TFileName tempfileDir(parse.FullName());
|
sl@0
|
404 |
(void)parse.Set(KWildCard, &tempfileDir, 0);//this call can't fail
|
sl@0
|
405 |
RFs& fs = sqlite3SymbianFs();
|
sl@0
|
406 |
CFileMan* fm = CFileMan::NewL(fs);
|
sl@0
|
407 |
(void)fm->Delete(parse.FullName());
|
sl@0
|
408 |
delete fm;
|
sl@0
|
409 |
}
|
sl@0
|
410 |
|
sl@0
|
411 |
/**
|
sl@0
|
412 |
Retrieves in iCollationDllName current(default) collation dll name.
|
sl@0
|
413 |
see TExtendedLocale
|
sl@0
|
414 |
*/
|
sl@0
|
415 |
void CSqlServer::GetCollationDllNameL()
|
sl@0
|
416 |
{
|
sl@0
|
417 |
TExtendedLocale extdlocale;
|
sl@0
|
418 |
extdlocale.LoadSystemSettings();
|
sl@0
|
419 |
TFileName fname;
|
sl@0
|
420 |
TParse fileName;
|
sl@0
|
421 |
TInt err = extdlocale.GetLocaleDllName(ELocaleCollateSetting, fname);
|
sl@0
|
422 |
if(err!= KErrNone)
|
sl@0
|
423 |
{
|
sl@0
|
424 |
iCollationDllName = KDefaultICollationDllName;
|
sl@0
|
425 |
}
|
sl@0
|
426 |
else
|
sl@0
|
427 |
{
|
sl@0
|
428 |
//only get the file name + extension
|
sl@0
|
429 |
fileName.Set(fname, NULL, NULL);
|
sl@0
|
430 |
iCollationDllName = fileName.NameAndExt();
|
sl@0
|
431 |
}
|
sl@0
|
432 |
SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, CSQLSERVER_GETCOLLATIONDLLNAMEL, "0x%x;CSqlServer::GetCollationDllNameL;iCollationDllName=%S;err=%d", (TUint)this, __SQLPRNSTR(iCollationDllName), err));
|
sl@0
|
433 |
}
|
sl@0
|
434 |
/**
|
sl@0
|
435 |
Finds and caches the name of each database configuration file
|
sl@0
|
436 |
that exists in the server's private data cage on the Z: drive
|
sl@0
|
437 |
*/
|
sl@0
|
438 |
void CSqlServer::CacheDbConfigFileNamesL(RFs& aFs, const TDesC& aServerPrivatePath)
|
sl@0
|
439 |
{
|
sl@0
|
440 |
//Create an in-memory array holding the names of the database config files, if any exist
|
sl@0
|
441 |
TParse parseDbConfig;
|
sl@0
|
442 |
__SQLLEAVE_IF_ERROR(parseDbConfig.Set(KSqlSrvDbConfigFileFormat, &aServerPrivatePath, NULL));
|
sl@0
|
443 |
TFileName configFilePath(parseDbConfig.FullName()); // get 'drive:\private path\cfg*' search string
|
sl@0
|
444 |
CDir* entryList = 0; // memory will be allocated for this in GetDir()
|
sl@0
|
445 |
TInt err = aFs.GetDir(configFilePath, KEntryAttNormal, ESortByName, entryList);
|
sl@0
|
446 |
if(err == KErrNone)
|
sl@0
|
447 |
{
|
sl@0
|
448 |
__ASSERT_DEBUG(entryList != NULL, __SQLPANIC(ESqlPanicInternalError));
|
sl@0
|
449 |
CleanupStack::PushL(entryList);
|
sl@0
|
450 |
if(entryList->Count() > 0)
|
sl@0
|
451 |
{
|
sl@0
|
452 |
iDbConfigFiles = CDbConfigFiles::NewL(*entryList);
|
sl@0
|
453 |
}
|
sl@0
|
454 |
CleanupStack::PopAndDestroy(entryList);
|
sl@0
|
455 |
}
|
sl@0
|
456 |
else
|
sl@0
|
457 |
{
|
sl@0
|
458 |
SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSERVER_CACHEDDBCONFIGFILENAMESL, "0x%X;CSqlServer::CacheDbConfigFileNamesL;GetDir() failed with error code %d", (TUint)this, err));
|
sl@0
|
459 |
__ASSERT_DEBUG(!entryList, __SQLPANIC(ESqlPanicInternalError));
|
sl@0
|
460 |
}
|
sl@0
|
461 |
}
|
sl@0
|
462 |
|
sl@0
|
463 |
//////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
464 |
//////////////////////////////////////// MSqlPolicyInspector implementation ///////////////////////////////
|
sl@0
|
465 |
//////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
466 |
|
sl@0
|
467 |
/**
|
sl@0
|
468 |
Implements MSqlPolicyInspector::Check() method.
|
sl@0
|
469 |
|
sl@0
|
470 |
@see MSqlPolicyInspector
|
sl@0
|
471 |
@see MSqlPolicyInspector::Check()
|
sl@0
|
472 |
*/
|
sl@0
|
473 |
TBool CSqlServer::Check(const TSecurityPolicy& aPolicy) const
|
sl@0
|
474 |
{
|
sl@0
|
475 |
#ifdef SQLSRV_STARTUP_TEST
|
sl@0
|
476 |
aPolicy.Package();//to prevent compiler warning
|
sl@0
|
477 |
return ETrue;
|
sl@0
|
478 |
#else
|
sl@0
|
479 |
return aPolicy.CheckPolicy(CServer2::Message());
|
sl@0
|
480 |
#endif
|
sl@0
|
481 |
}
|
sl@0
|
482 |
|
sl@0
|
483 |
//////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
484 |
//////////////////////////////////////// MSqlSrvBurInterface implementation //////////////////////////////
|
sl@0
|
485 |
//////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
486 |
|
sl@0
|
487 |
/**
|
sl@0
|
488 |
Implements MSqlSrvBurInterface::Fs().
|
sl@0
|
489 |
|
sl@0
|
490 |
@return A reference to the file session instance.
|
sl@0
|
491 |
*/
|
sl@0
|
492 |
RFs& CSqlServer::Fs()
|
sl@0
|
493 |
{
|
sl@0
|
494 |
return iFileData.Fs();
|
sl@0
|
495 |
}
|
sl@0
|
496 |
|
sl@0
|
497 |
/**
|
sl@0
|
498 |
Implements MSqlSrvBurInterface::GetBackupListL().
|
sl@0
|
499 |
Retrieves in aFileNameList parameter a list of secure database names (full database names, including path)
|
sl@0
|
500 |
which security UID matches aUid parameter.
|
sl@0
|
501 |
No databases will be included into the list, if the drive is read-only.
|
sl@0
|
502 |
|
sl@0
|
503 |
@param aUid Database security UID.
|
sl@0
|
504 |
@param aDrive The drive where the database search will be performed, in the SQL server private data cage.
|
sl@0
|
505 |
@param aFileNameList An output parameter.
|
sl@0
|
506 |
Each array entry represents the full name of a database in SQL server private data cage
|
sl@0
|
507 |
on the specified drive (aDrive), which uid matches the aUid parameter.
|
sl@0
|
508 |
|
sl@0
|
509 |
@leave KErrNoMemory, an out of memory condition has occured;
|
sl@0
|
510 |
Note that the function may leave also with some other database specific or OS specific
|
sl@0
|
511 |
error codes.
|
sl@0
|
512 |
*/
|
sl@0
|
513 |
void CSqlServer::GetBackUpListL(TSecureId aUid, TDriveNumber aDrive, RArray<HBufC*>& aFileNameList)
|
sl@0
|
514 |
{
|
sl@0
|
515 |
SQL_TRACE_INTERNALS(OstTraceExt3(TRACE_INTERNALS, CSQLSERVER_GETBACKUPLISTL_ENTRY, "Entry;0x%x;CSqlServer::GetBackUpListL;aDrive=%d;aUid=0x%X", (TUint)this, (TInt)aDrive, (TUint)aUid.iId));
|
sl@0
|
516 |
__ASSERT_DEBUG(aFileNameList.Count() == 0, __SQLPANIC(ESqlPanicBadArgument));
|
sl@0
|
517 |
RFs& fs = iFileData.Fs();
|
sl@0
|
518 |
//No files in the list if aDrive is a read-only drive
|
sl@0
|
519 |
TDriveInfo driveInfo;
|
sl@0
|
520 |
__SQLLEAVE_IF_ERROR(fs.Drive(driveInfo, aDrive));
|
sl@0
|
521 |
if(driveInfo.iDriveAtt & KDriveAttRom)
|
sl@0
|
522 |
{
|
sl@0
|
523 |
return;
|
sl@0
|
524 |
}
|
sl@0
|
525 |
//Compose the search path
|
sl@0
|
526 |
TDriveUnit driveUnit(aDrive);
|
sl@0
|
527 |
TDriveName driveName = driveUnit.Name();
|
sl@0
|
528 |
TFileName path;
|
sl@0
|
529 |
path.Copy(driveName);
|
sl@0
|
530 |
path.Append(iFileData.PrivatePath());
|
sl@0
|
531 |
//Include the aUid and the "*" mask
|
sl@0
|
532 |
TUidName uidName = (static_cast <TUid> (aUid)).Name();
|
sl@0
|
533 |
TBuf<KMaxUidName + sizeof(KMatchAllDbFiles)> fileNameMask(uidName);
|
sl@0
|
534 |
fileNameMask.Append(KMatchAllDbFiles);
|
sl@0
|
535 |
TParse parse;
|
sl@0
|
536 |
__SQLLEAVE_IF_ERROR(parse.Set(path, &fileNameMask, NULL));
|
sl@0
|
537 |
//Do the search
|
sl@0
|
538 |
TPtrC fullPath(parse.FullName());
|
sl@0
|
539 |
SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSERVER_GETBACKUPLISTL_FULLPATH, "Exit;0x%x;CSqlServer::GetBackUpListL;fullPath=%S", (TUint)this, __SQLPRNSTR(fullPath)));
|
sl@0
|
540 |
CDir* fileNameCol = NULL;
|
sl@0
|
541 |
TInt err = fs.GetDir(fullPath, KEntryAttNormal, ESortNone, fileNameCol);
|
sl@0
|
542 |
if(err == KErrNotFound)
|
sl@0
|
543 |
{
|
sl@0
|
544 |
__ASSERT_DEBUG(!fileNameCol, __SQLPANIC(ESqlPanicInternalError));
|
sl@0
|
545 |
SQL_TRACE_INTERNALS(OstTrace1(TRACE_INTERNALS, CSQLSERVER_GETBACKUPLISTL_EXIT1, "Exit;0x%x;CSqlServer::GetBackUpListL;no files found", (TUint)this));
|
sl@0
|
546 |
return;
|
sl@0
|
547 |
}
|
sl@0
|
548 |
__SQLLEAVE_IF_ERROR(err);
|
sl@0
|
549 |
__ASSERT_DEBUG(fileNameCol != NULL, __SQLPANIC(ESqlPanicInternalError));
|
sl@0
|
550 |
CleanupStack::PushL(fileNameCol);
|
sl@0
|
551 |
TInt fileCount = fileNameCol->Count();
|
sl@0
|
552 |
__SQLLEAVE_IF_ERROR(aFileNameList.Reserve(fileCount));
|
sl@0
|
553 |
//Append the full database file paths to the file names list.
|
sl@0
|
554 |
for(TInt i=0;i<fileCount;++i)
|
sl@0
|
555 |
{
|
sl@0
|
556 |
const ::TEntry& entry = (*fileNameCol)[i];
|
sl@0
|
557 |
__ASSERT_DEBUG(!entry.IsDir(), __SQLPANIC(ESqlPanicInternalError));//RFs::GetDir() search attributes exclude directories (see the GetDir() call above).
|
sl@0
|
558 |
__SQLLEAVE_IF_ERROR(parse.Set(path, &entry.iName, NULL));
|
sl@0
|
559 |
TPtrC fname(parse.FullName());
|
sl@0
|
560 |
SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSERVER_GETBACKUPLISTL, "0x%x;CSqlServer::GetBackUpListL;fname=%S", (TUint)this, __SQLPRNSTR(fname)));
|
sl@0
|
561 |
HBufC* fnameBuf = fname.AllocL();
|
sl@0
|
562 |
__SQLDEBUG_EXPR(err = )aFileNameList.Append(fnameBuf);
|
sl@0
|
563 |
__ASSERT_DEBUG(err == KErrNone, __SQLPANIC(ESqlPanicInternalError));
|
sl@0
|
564 |
}
|
sl@0
|
565 |
CleanupStack::PopAndDestroy(fileNameCol);
|
sl@0
|
566 |
SQL_TRACE_INTERNALS(OstTraceExt2(TRACE_INTERNALS, CSQLSERVER_GETBACKUPLISTL_EXIT2, "Exit;0x%x;CSqlServer::GetBackUpListL;file count=%d", (TUint)this, fileCount));
|
sl@0
|
567 |
}
|
sl@0
|
568 |
|
sl@0
|
569 |
//////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
570 |
//////////////////////////////////////// SQL server startup //////////////////////////////////////////////
|
sl@0
|
571 |
//////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
572 |
|
sl@0
|
573 |
#ifndef SQLSRV_STARTUP_TEST
|
sl@0
|
574 |
|
sl@0
|
575 |
//Run the SQL server
|
sl@0
|
576 |
static void RunServerL()
|
sl@0
|
577 |
{
|
sl@0
|
578 |
// naming the server thread after the server helps to debug panics
|
sl@0
|
579 |
User::LeaveIfError(User::RenameThread(KSqlSrvName));
|
sl@0
|
580 |
|
sl@0
|
581 |
// create and install the active scheduler we need
|
sl@0
|
582 |
CActiveScheduler* scheduler = new (ELeave) CActiveScheduler;
|
sl@0
|
583 |
CleanupStack::PushL(scheduler);
|
sl@0
|
584 |
CActiveScheduler::Install(scheduler);
|
sl@0
|
585 |
TheServer = CSqlServer::NewLC();
|
sl@0
|
586 |
RProcess::Rendezvous(KErrNone);
|
sl@0
|
587 |
CActiveScheduler::Start();
|
sl@0
|
588 |
|
sl@0
|
589 |
CleanupStack::PopAndDestroy(2, scheduler);//CSqlServer, scheduler
|
sl@0
|
590 |
}
|
sl@0
|
591 |
|
sl@0
|
592 |
// SQL server process entry point
|
sl@0
|
593 |
TInt E32Main()
|
sl@0
|
594 |
{
|
sl@0
|
595 |
__UHEAP_MARK;
|
sl@0
|
596 |
|
sl@0
|
597 |
CTrapCleanup* cleanup = CTrapCleanup::New();
|
sl@0
|
598 |
TInt err = KErrNoMemory;
|
sl@0
|
599 |
if(cleanup)
|
sl@0
|
600 |
{
|
sl@0
|
601 |
TRAP(err, ::RunServerL());
|
sl@0
|
602 |
delete cleanup;
|
sl@0
|
603 |
}
|
sl@0
|
604 |
|
sl@0
|
605 |
__UHEAP_MARKEND;
|
sl@0
|
606 |
|
sl@0
|
607 |
return err;
|
sl@0
|
608 |
}
|
sl@0
|
609 |
|
sl@0
|
610 |
#endif //SQLSRV_STARTUP_TEST
|