os/persistentdata/persistentstorage/sql/TEST/t_sqlperformance.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 2006-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 //
    15 
    16 #include <e32test.h>
    17 #include <e32math.h>
    18 #include <bautils.h>
    19 #include <hal.h>
    20 #include <stdlib.h>
    21 #include <sqldb.h>
    22 #include "t_sqlcmdlineutil.h"
    23 #include "SqlSrvStrings.h"
    24 #include "sqlite3.h"
    25 #include "SqliteSymbian.h"
    26 
    27 ///////////////////////////////////////////////////////////////////////////////////////
    28 
    29 RTest TheTest(_L("t_sqlperformance test"));
    30 RFs   TheFs;
    31 TBuf<200> TheTestTitle;
    32 TCmdLineParams TheCmdLineParams;
    33 TBuf8<200> TheSqlConfigString;
    34 
    35 _LIT(KUtf8,  "UTF8 ");
    36 _LIT(KUtf16, "UTF16");
    37 
    38 TFileName TheSecureDbName;
    39 TFileName TheNonSecureDbName;
    40 TFileName TheNonSecureDbName2;
    41 TFileName TheNonSecureTmpDbName;
    42 TFileName TheSglRecDbFileName;
    43 
    44 _LIT(KSqlServerPrivateDir, "\\private\\10281e17\\");
    45 
    46 _LIT(KCreateDbScript, "z:\\test\\contacts_schema_to_vendors.sql");
    47 _LIT(KFillDbScript, "z:\\test\\add_simple_contacts.sql");
    48 
    49 _LIT8(KCommitStr8, "COMMIT;");
    50 _LIT16(KCommitStr16, "COMMIT;");
    51 
    52 _LIT8(KUpdateSql8, "UPDATE IDENTITYTABLE SET CM_FIRSTNAME=:V1, CM_LASTNAME=:V2,CM_COMPANYNAME=:V3 WHERE PARENT_CMID=:ID");
    53 _LIT16(KUpdateSql16, "UPDATE IDENTITYTABLE SET CM_FIRSTNAME=:V1, CM_LASTNAME=:V2,CM_COMPANYNAME=:V3 WHERE PARENT_CMID=:ID");
    54 
    55 _LIT8(KUpdateSql2_8, "UPDATE IDENTITYTABLE SET CM_FIRSTNAME='%S%d',CM_LASTNAME='%S%d',CM_COMPANYNAME='%S%d' WHERE PARENT_CMID=%d");
    56 _LIT16(KUpdateSql2_16, "UPDATE IDENTITYTABLE SET CM_FIRSTNAME='%S%d',CM_LASTNAME='%S%d',CM_COMPANYNAME='%S%d' WHERE PARENT_CMID=%d");
    57 
    58 _LIT8(KSelectSql8, "SELECT CM_FIRSTNAME, CM_LASTNAME, CM_COMPANYNAME FROM IDENTITYTABLE WHERE PARENT_CMID > 50");
    59 _LIT16(KSelectSql16, "SELECT CM_FIRSTNAME, CM_LASTNAME, CM_COMPANYNAME FROM IDENTITYTABLE WHERE PARENT_CMID > 50");
    60 
    61 _LIT8(KDeleteSql8, "DELETE FROM IDENTITYTABLE WHERE PARENT_CMID > 50");
    62 _LIT16(KDeleteSql16, "DELETE FROM IDENTITYTABLE WHERE PARENT_CMID > 50");
    63 
    64 extern TPtrC GetFirstSqlStmt(TPtr& aString);
    65 
    66 _LIT(KFirstName, "FirstName-");
    67 _LIT(KLastName, "LastName-");
    68 _LIT(KCompanyName, "CompanyName-");
    69 
    70 _LIT(KFirstName2, "12345678-");
    71 _LIT(KLastName2, "ABCDEFGHIJK-");
    72 _LIT(KCompanyName2, "KKKKKKKKKK10-");
    73 
    74 const char* KRawPrmName1 = ":V1";
    75 const char* KRawPrmName2 = ":V2";
    76 const char* KRawPrmName3 = ":V3";
    77 const char* KRawPrmName4 = ":ID";
    78 
    79 _LIT(KPrmName1, ":V1");	
    80 _LIT(KPrmName2, ":V2");	
    81 _LIT(KPrmName3, ":V3");	
    82 _LIT(KPrmName4, ":ID");	
    83 
    84 const TInt KTestTecordCount = 1000;
    85 	
    86 ///////////////////////////////////////////////////////////////////////////////////////
    87 
    88 void TestEnvDestroy()
    89 	{
    90 	(void)RSqlDatabase::Delete(TheNonSecureTmpDbName);
    91 	(void)RSqlDatabase::Delete(TheNonSecureDbName2);
    92 	(void)RSqlDatabase::Delete(TheNonSecureDbName);
    93 	(void)RSqlDatabase::Delete(TheSecureDbName);
    94 	TheFs.Close();
    95 	}
    96 
    97 ///////////////////////////////////////////////////////////////////////////////////////
    98 ///////////////////////////////////////////////////////////////////////////////////////
    99 //Test macros and functions
   100 void Check1(TInt aValue, TInt aLine)
   101 	{
   102 	if(!aValue)
   103 		{
   104 		TestEnvDestroy();
   105 		TheTest.Printf(_L("*** Line %d\r\n"), aLine);
   106 		TheTest(EFalse, aLine);
   107 		}
   108 	}
   109 void Check2(TInt aValue, TInt aExpected, TInt aLine)
   110 	{
   111 	if(aValue != aExpected)
   112 		{
   113 		TestEnvDestroy();
   114 		TheTest.Printf(_L("*** Line %d, Expected error: %d, got: %d\r\n"), aLine, aExpected, aValue);
   115 		TheTest(EFalse, aLine);
   116 		}
   117 	}
   118 #define TEST(arg) ::Check1((arg), __LINE__)
   119 #define TEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__)
   120 
   121 ///////////////////////////////////////////////////////////////////////////////////////
   122 
   123 void TestEnvInit()
   124     {
   125 	TInt err = TheFs.Connect();
   126 	TEST2(err, KErrNone);
   127 
   128 	err = TheFs.MkDir(TheNonSecureDbName);
   129 	TEST(err == KErrNone || err == KErrAlreadyExists);
   130 	}
   131 
   132 //Reads a SQL file and returns the file content as HBUFC string.
   133 //The caller is responsible for destroying the returned HBUFC object.
   134 template <class HBUFC> HBUFC* ReadSqlScript(const TDesC& aSqlFileName)
   135 	{
   136 	RFile file;
   137 	TEST2(file.Open(TheFs, aSqlFileName, EFileRead), KErrNone);
   138 	
   139 	TInt size = 0;
   140 	TEST2(file.Size(size), KErrNone);
   141 	
   142 	HBufC8* sql = HBufC8::New(size);
   143 	TEST(sql != NULL);
   144 	
   145 	TPtr8 ptr = sql->Des();
   146 	TEST2(file.Read(ptr, size), KErrNone);
   147 
   148 	file.Close();
   149 	
   150 	HBUFC* sql2 = HBUFC::New(size + 1);
   151 	TEST(sql2 != NULL);
   152 	sql2->Des().Copy(sql->Des());
   153 	sql2->Des().Append(TChar(0));
   154 	delete sql;
   155 	
   156 	return sql2;
   157 	}
   158 //Explicit ReadSqlScript() template instantiations.
   159 template HBufC8* ReadSqlScript<HBufC8>(const TDesC&);
   160 template HBufC16* ReadSqlScript<HBufC16>(const TDesC&);
   161 
   162 //Searches for the next aCommitStr appearance in aSqlScript string and returns a PTRC object holding
   163 //the SQL strings from the beginning of aSqlScript till the aCommitStr (including it).
   164 template <class PTRC, class DESC> PTRC GetNextTrans(PTRC& aSqlScript, const DESC& aCommitStr)
   165 	{
   166 	PTRC res(NULL, 0);
   167 	TInt pos = aSqlScript.FindF(aCommitStr);
   168 	if(pos >= 0)
   169 		{
   170 		pos += aCommitStr.Length();
   171 		res.Set(aSqlScript.Left(pos));
   172 		aSqlScript.Set(aSqlScript.Mid(pos));
   173 		}
   174 	return res;
   175 	}
   176 //Explicit GetNextTrans() template instantiations.
   177 template TPtrC8 GetNextTrans<TPtrC8, TDesC8>(TPtrC8&, const TDesC8&);
   178 template TPtrC16 GetNextTrans<TPtrC16, TDesC16>(TPtrC16&, const TDesC16&);
   179 
   180 //Prints aTicks parameter (converted to ms)
   181 void PrintStats(TUint32 aStartTicks, TUint32 aEndTicks)
   182 	{
   183 	static TInt freq = 0;
   184 	if(freq == 0)
   185 		{
   186 		TEST2(HAL::Get(HAL::EFastCounterFrequency, freq), KErrNone);
   187 		}
   188 	TInt64 diffTicks = (TInt64)aEndTicks - (TInt64)aStartTicks;
   189 	if(diffTicks < 0)
   190 		{
   191 		diffTicks = KMaxTUint32 + diffTicks + 1;
   192 		}
   193 	const TInt KMicroSecIn1Sec = 1000000;
   194 	TInt32 us = (diffTicks * KMicroSecIn1Sec) / freq;
   195 	TheTest.Printf(_L("####Execution time: %d ms\r\n"), us / 1000);
   196 	}
   197 	
   198 void PrintFileSize(const TDesC& aFileName)
   199 	{
   200 	TParse parse;
   201 	parse.Set(aFileName, &KSqlServerPrivateDir(), 0);
   202 	RFile file;
   203 	TInt err = file.Open(TheFs, parse.FullName(), EFileRead);
   204 	TEST2(err, KErrNone);
   205 	TInt size = 0;
   206 	err = file.Size(size);
   207 	TEST2(err, KErrNone);
   208 	TheTest.Printf(_L("####FileSize: %d\r\n"), size);
   209 	file.Close();
   210 	}
   211 
   212 //////////////////////////////////////////////////////////////////////////////////////////////////////////////
   213 ///////     SQL SERVER performance tests
   214 //////////////////////////////////////////////////////////////////////////////////////////////////////////////
   215 	
   216 enum TDbType {ENonSecureDb, ESecureDb};
   217 
   218 //Template class offering Create() and Open() methods for creating/opening a RSqlDatabase object
   219 template <TDbType TYPE> class TDbHelper
   220 	{
   221 public:	
   222 	static void Create(const TDesC& aDbName);
   223 	static RSqlDatabase Open(const TDesC& aDbName);
   224 	};
   225 
   226 //Creates aDb database schema.
   227 void CreateDbSchema(RSqlDatabase& aDb)
   228 	{
   229 	HBufC8* createDbScript = ReadSqlScript<HBufC8>(KCreateDbScript());
   230 	TInt err = aDb.Exec(createDbScript->Des());
   231 	TEST(err >= 0);	
   232 	delete createDbScript;
   233 	}
   234 
   235 //Explicit TDbHelper class specialization for creating/opening a nonsecure RSqlDatabase object
   236 template <> class TDbHelper<ENonSecureDb>
   237 	{
   238 public:	
   239 	static void Create(const TDesC& aDbName)
   240 		{
   241 		RSqlDatabase::Delete(aDbName);
   242 		RSqlDatabase db;
   243 		TInt err = db.Create(aDbName, &TheSqlConfigString);
   244 		TEST2(err, KErrNone);
   245 		CreateDbSchema(db);
   246 		db.Close();
   247 		}
   248 	static RSqlDatabase Open(const TDesC& aDbName)
   249 		{
   250 		RSqlDatabase db;
   251 		TInt err = db.Open(aDbName);
   252 		TEST2(err, KErrNone);
   253 		return db;
   254 		}
   255 	};
   256 
   257 //Explicit TDbHelper class specialization for creating/opening a secure RSqlDatabase object
   258 template <> class TDbHelper<ESecureDb>
   259 	{
   260 public:	
   261 	static void Create(const TDesC& aDbName)
   262 		{
   263 		RSqlDatabase::Delete(aDbName);
   264 		RSqlSecurityPolicy securityPolicy;
   265 		TInt err = securityPolicy.Create(TSecurityPolicy(TSecurityPolicy::EAlwaysPass));
   266 		TEST2(err, KErrNone);	
   267 		securityPolicy.SetDbPolicy(RSqlSecurityPolicy::ESchemaPolicy, TSecurityPolicy(ECapabilityReadUserData, ECapabilityWriteUserData));
   268 		securityPolicy.SetDbPolicy(RSqlSecurityPolicy::EWritePolicy, TSecurityPolicy(ECapabilityWriteUserData));
   269 		securityPolicy.SetDbPolicy(RSqlSecurityPolicy::EReadPolicy, TSecurityPolicy(ECapabilityReadUserData));
   270 		RSqlDatabase db;
   271 		err = db.Create(aDbName, securityPolicy, &TheSqlConfigString);
   272 		TEST2(err, KErrNone);	
   273 		securityPolicy.Close();
   274 		CreateDbSchema(db);
   275 		db.Close();
   276 		}
   277 	static RSqlDatabase Open(const TDesC& aDbName)
   278 		{
   279 		RSqlDatabase db;
   280 		TInt err = db.Open(aDbName);
   281 		TEST2(err, KErrNone);
   282 		return db;
   283 		}
   284 	};
   285 
   286 //Executes SQL script
   287 template <class HBUFC, class PTRC, class DESC> void ExecuteSqlScript(RSqlDatabase& aDb, const TDesC& aScriptFileName, const DESC& aCommitStr)
   288 	{
   289 	HBUFC* fillDbScript = ReadSqlScript<HBUFC>(aScriptFileName);
   290 	TUint32 start = User::FastCounter();
   291 	PTRC ptr(fillDbScript->Des());
   292 	PTRC sql(GetNextTrans<PTRC, DESC>(ptr, aCommitStr));
   293 	while(sql.Length() > 0)
   294 		{
   295 		TInt err = aDb.Exec(sql);
   296 		if(err == KErrNoMemory)
   297 			{
   298 			TheTest.Printf(_L("###ERROR 'Out of memory'! The test cannot be completed!\r\n"));
   299 			return;	
   300 			}
   301 		TEST(err > 0);	
   302 		sql.Set(GetNextTrans<PTRC, DESC>(ptr, aCommitStr));
   303 		}
   304 	TUint32 end = User::FastCounter();
   305 	PrintStats(start, end);
   306 	delete fillDbScript;
   307 	}
   308 //Explicit ExecuteSqlScript() template instantiations.
   309 template void ExecuteSqlScript<HBufC8, TPtrC8, TDesC8>(RSqlDatabase&, const TDesC&, const TDesC8&);
   310 template void ExecuteSqlScript<HBufC16, TPtrC16, TDesC16>(RSqlDatabase&, const TDesC&, const TDesC16&);
   311 
   312 ///////////////////////////////////////////////////////////////////////////////////////
   313 
   314 //"INSERT" test function
   315 template <class HBUFC, class PTRC, class DESC, TDbType TYPE> void InsertTest(const TDesC& aDbFileName, const DESC& aCommitStr)
   316 	{
   317 	TheTest.Printf(_L("\"Insert\" test\r\n"));
   318 	RSqlDatabase db = TDbHelper<TYPE>::Open(aDbFileName);
   319 	ExecuteSqlScript<HBUFC, PTRC, DESC>(db, KFillDbScript, aCommitStr);
   320 	db.Close();	
   321 	}
   322 //Explicit InsertTest() template instantiations.
   323 template void InsertTest<HBufC8, TPtrC8, TDesC8, ENonSecureDb>(const TDesC&, const TDesC8&);
   324 template void InsertTest<HBufC8, TPtrC8, TDesC8, ESecureDb>(const TDesC&, const TDesC8&);
   325 template void InsertTest<HBufC16, TPtrC16, TDesC16, ENonSecureDb>(const TDesC&, const TDesC16&);
   326 template void InsertTest<HBufC16, TPtrC16, TDesC16, ESecureDb>(const TDesC&, const TDesC16&);
   327 
   328 //"UPDATE" test function (parametrized update)
   329 template <class DESC, TDbType TYPE> void UpdateTest(const TDesC& aDbFileName, const DESC& aUpdateSql)
   330 	{
   331 	TheTest.Printf(_L("\"Update (parametrized)\" test\r\n"));
   332 	RSqlDatabase db = TDbHelper<TYPE>::Open(aDbFileName);
   333 	RSqlStatement stmt;
   334 	TInt err = stmt.Prepare(db, aUpdateSql);
   335 	TEST2(err, KErrNone);
   336 	
   337 	TInt firstNamePrmIdx = stmt.ParameterIndex(KPrmName1());
   338 	TEST(firstNamePrmIdx >= 0);
   339 	TInt lastNamePrmIdx = stmt.ParameterIndex(KPrmName2());
   340 	TEST(lastNamePrmIdx >= 0);
   341 	TInt companyNamePrmIdx = stmt.ParameterIndex(KPrmName3());
   342 	TEST(companyNamePrmIdx >= 0);
   343 	TInt idIdx = stmt.ParameterIndex(KPrmName4());
   344 	TEST(idIdx >= 0);
   345 	
   346 	TUint32 start = User::FastCounter();
   347 	for(TInt id=1;id<=KTestTecordCount;++id)
   348 		{
   349 		TBuf<20> buf;
   350 		buf.Copy(KFirstName);
   351 		buf.AppendNum(id);
   352 		err = stmt.BindText(firstNamePrmIdx, buf);
   353 		TEST2(err, KErrNone);
   354 		buf.Copy(KLastName);
   355 		buf.AppendNum(id);
   356 		err = stmt.BindText(lastNamePrmIdx, buf);
   357 		TEST2(err, KErrNone);
   358 		buf.Copy(KCompanyName);
   359 		buf.AppendNum(id);
   360 		err = stmt.BindText(companyNamePrmIdx, buf);
   361 		TEST2(err, KErrNone);
   362 		err = stmt.BindInt(idIdx, id);
   363 		TEST2(err, KErrNone);
   364 		err = stmt.Exec();
   365 		TEST(err > 0);	
   366 		err = stmt.Reset();
   367 		TEST2(err, KErrNone);
   368 		}
   369 	TUint32 end = User::FastCounter();
   370 	PrintStats(start, end);
   371 	stmt.Close();
   372 	db.Close();
   373 	}
   374 //Explicit UpdateTest() template instantiations.
   375 template void UpdateTest<TDesC8, ENonSecureDb>(const TDesC&, const TDesC8&);
   376 template void UpdateTest<TDesC8, ESecureDb>(const TDesC&, const TDesC8&);
   377 template void UpdateTest<TDesC16, ENonSecureDb>(const TDesC&, const TDesC16&);
   378 template void UpdateTest<TDesC16, ESecureDb>(const TDesC&, const TDesC16&);
   379 
   380 //"UPDATE" test function (without parameters) - SQL server
   381 template <class BUF, class DESC, TDbType TYPE> void UpdateWPTest(const TDesC& aDbFileName, const DESC& aUpdateSql)
   382 	{
   383 	TheTest.Printf(_L("\"Update (without parameters)\" test\r\n"));
   384 	RSqlDatabase db = TDbHelper<TYPE>::Open(aDbFileName);
   385 	TBuf<200> fmtstr;
   386 	fmtstr.Copy(aUpdateSql);
   387 	
   388 	TUint32 start = User::FastCounter();
   389 	for(TInt id=1;id<=KTestTecordCount;++id)
   390 		{
   391 		TBuf<200> buf;
   392 		buf.Format(fmtstr, &KFirstName2, id, &KLastName2, id, &KCompanyName2, id, id);
   393 		BUF sql;
   394 		sql.Copy(buf);
   395 		TInt err = db.Exec(sql);
   396 		TEST(err > 0);	
   397 		}
   398 	TUint32 end = User::FastCounter();
   399 	PrintStats(start, end);
   400 	db.Close();
   401 	PrintFileSize(aDbFileName);
   402 	}
   403 //Explicit UpdateWPTest() template instantiations.
   404 template void UpdateWPTest<TBuf8<200>, TDesC8, ENonSecureDb>(const TDesC&, const TDesC8&);
   405 template void UpdateWPTest<TBuf8<200>, TDesC8, ESecureDb>(const TDesC&, const TDesC8&);
   406 template void UpdateWPTest<TBuf16<200>, TDesC16, ENonSecureDb>(const TDesC&, const TDesC16&);
   407 template void UpdateWPTest<TBuf16<200>, TDesC16, ESecureDb>(const TDesC&, const TDesC16&);
   408 
   409 //"SELECT" test function
   410 template <class DESC, TDbType TYPE> void SelectTest(const TDesC& aDbFileName, const DESC& aSelectSql)
   411 	{
   412 	TheTest.Printf(_L("\"Select\" test\r\n"));
   413 	RSqlDatabase db = TDbHelper<TYPE>::Open(aDbFileName);
   414 	RSqlStatement stmt;
   415 	TInt err = stmt.Prepare(db, aSelectSql);
   416 	TEST2(err, KErrNone);
   417 	TUint32 start = User::FastCounter();
   418 	while((err = stmt.Next()) == KSqlAtRow)
   419 		{
   420 		TBuf<20> buf;
   421 		err = stmt.ColumnText(0, buf);			
   422 		TEST2(err, KErrNone);
   423 		TEST(buf.Length() > 0);
   424 		err = stmt.ColumnText(1, buf);			
   425 		TEST2(err, KErrNone);
   426 		TEST(buf.Length() > 0);
   427 		err = stmt.ColumnText(2, buf);			
   428 		TEST2(err, KErrNone);
   429 		TEST(buf.Length() > 0);
   430 		}
   431 	TEST2(err, KSqlAtEnd);
   432 	TUint32 end = User::FastCounter();
   433 	PrintStats(start, end);
   434 	stmt.Close();
   435 	db.Close();	
   436 	}
   437 //Explicit SelectTest() template instantiations.
   438 template void SelectTest<TDesC8, ENonSecureDb>(const TDesC&, const TDesC8&);
   439 template void SelectTest<TDesC8, ESecureDb>(const TDesC&, const TDesC8&);
   440 template void SelectTest<TDesC16, ENonSecureDb>(const TDesC&, const TDesC16&);
   441 template void SelectTest<TDesC16, ESecureDb>(const TDesC&, const TDesC16&);
   442 
   443 //"DELETE" test function
   444 template <class DESC, TDbType TYPE> void DeleteTest(const TDesC& aDbFileName, const DESC& aDeleteSql)
   445 	{
   446 	TheTest.Printf(_L("\"Delete\" test\r\n"));
   447 	RSqlDatabase db = TDbHelper<TYPE>::Open(aDbFileName);
   448 	TUint32 start = User::FastCounter();
   449 	TInt err = db.Exec(aDeleteSql);
   450 	TEST(err > 0);	
   451 	TUint32 end = User::FastCounter();
   452 	PrintStats(start, end);
   453 	db.Close();	
   454 	}
   455 //Explicit SelectTest() template instantiations.
   456 template void DeleteTest<TDesC8, ENonSecureDb>(const TDesC&, const TDesC8&);
   457 template void DeleteTest<TDesC8, ESecureDb>(const TDesC&, const TDesC8&);
   458 template void DeleteTest<TDesC16, ENonSecureDb>(const TDesC&, const TDesC16&);
   459 template void DeleteTest<TDesC16, ESecureDb>(const TDesC&, const TDesC16&);
   460 	
   461 //Performance test function: INSERT, UPDATE, SELECT
   462 template <class HBUFC, class PTRC, class DESC, TDbType TYPE> void PerformanceTest(const TDesC& aDbFileName,
   463 																				  const DESC& aCommitStr,
   464 																				  const DESC& aUpdateSql,
   465 																				  const DESC& aSelectSql,
   466 																				  const DESC& aDeleteSql)
   467 	{
   468 	CFileMan* fm = NULL;
   469 	TRAPD(err, fm = CFileMan::NewL(TheFs));
   470 	TEST2(err, KErrNone);
   471 	
   472 	TDbHelper<TYPE>::Create(aDbFileName);
   473 	InsertTest<HBUFC, PTRC, DESC, TYPE>(aDbFileName, aCommitStr);
   474 	PrintFileSize(aDbFileName);
   475 	(void)fm->Copy(aDbFileName, TheNonSecureTmpDbName);
   476 	
   477 	UpdateTest<DESC, TYPE>(aDbFileName, aUpdateSql);
   478 	PrintFileSize(aDbFileName);
   479 	SelectTest<DESC, TYPE>(aDbFileName, aSelectSql);
   480 	
   481 	DeleteTest<DESC, TYPE>(aDbFileName, aDeleteSql);
   482 	PrintFileSize(aDbFileName);
   483 	
   484 	(void)fm->Copy(TheNonSecureTmpDbName, aDbFileName);
   485 	(void)fm->Delete(TheNonSecureTmpDbName);
   486 	delete fm;
   487 	}
   488 //Explicit PerformanceTest() template instantiations.
   489 template void PerformanceTest<HBufC8, TPtrC8, TDesC8, ENonSecureDb>(const TDesC&, const TDesC8&, const TDesC8&, const TDesC8&, const TDesC8&);
   490 template void PerformanceTest<HBufC8, TPtrC8, TDesC8, ESecureDb>(const TDesC&, const TDesC8&, const TDesC8&, const TDesC8&, const TDesC8&);
   491 template void PerformanceTest<HBufC16, TPtrC16, TDesC16, ENonSecureDb>(const TDesC&, const TDesC16&, const TDesC16&, const TDesC16&, const TDesC16&);
   492 template void PerformanceTest<HBufC16, TPtrC16, TDesC16, ESecureDb>(const TDesC&, const TDesC16&, const TDesC16&, const TDesC16&, const TDesC16&);
   493 
   494 //////////////////////////////////////////////////////////////////////////////////////////////////////////////
   495 ///////     SQLITE API used directly  
   496 //////////////////////////////////////////////////////////////////////////////////////////////////////////////
   497 
   498 //Template class offering Create() and Open() methods for creating/opening a sqlite3 handle.
   499 template <TCmdLineParams::TDbEncoding TYPE> class TDbHelper2
   500 	{
   501 public:	
   502 	static void Create(const TDesC& aDbName);
   503 	static sqlite3* Open(const TDesC& aDbName);
   504 	};
   505 
   506 //If error - prints error message and panics the application
   507 void PanicIfError(sqlite3* aDbHandle, TInt aErr)
   508 	{
   509   	if(aErr != SQLITE_OK && aErr != SQLITE_DONE && aErr != SQLITE_ROW)
   510   		{
   511   		if(aDbHandle)
   512   			{
   513 	  		TPtrC p((const TUint16*)sqlite3_errmsg16(aDbHandle));
   514 	  		TheTest.Printf(_L("Database err %d, msg: %S\n\n"), aErr, &p);
   515 			(void)sqlite3_close(aDbHandle);
   516   			}
   517     	TEST(0);
   518   		}
   519 	}
   520 
   521 //If error - prints error message and returns zero
   522 TInt ReportIfError(sqlite3* aDbHandle, TInt aErr)
   523 	{
   524   	if(aErr != SQLITE_OK && aErr != SQLITE_DONE && aErr != SQLITE_ROW)
   525   		{
   526   		if(aDbHandle)
   527   			{
   528 	  		TPtrC p((const TUint16*)sqlite3_errmsg16(aDbHandle));
   529 	  		TheTest.Printf(_L("Database err %d, msg: %S\n\n"), aErr, &p);
   530   			}
   531   		return ETrue;
   532   		}
   533   	return EFalse;
   534 	}
   535 
   536 //Explicit TDbHelper2 class specialization for creating/opening a database with UTF8 default encoding
   537 template <> class TDbHelper2<TCmdLineParams::EDbUtf8>
   538 	{
   539 public:	
   540 	static void Create(const TDesC& aDbFileName)
   541 		{
   542 		(void)TheFs.Delete(aDbFileName);
   543 		HBufC8* dbName = HBufC8::New(aDbFileName.Length() + 1);
   544 		TEST(dbName != NULL);
   545 		dbName->Des().Copy(aDbFileName);
   546 		sqlite3* dbHandle = NULL;
   547 		TInt err = sqlite3_open((const char*)(dbName->Des().PtrZ()), &dbHandle);
   548 		delete dbName;
   549 		PanicIfError(dbHandle, err);
   550 		HBufC8* createDbScript = ReadSqlScript<HBufC8>(KCreateDbScript());
   551 		HBufC8* createDbScript2 = HBufC8::New(createDbScript->Des().Length() + 1);
   552 		TEST(createDbScript2 != NULL);
   553 		createDbScript2->Des().Copy(createDbScript->Des());
   554 		delete createDbScript;
   555 		createDbScript2->Des().Append(TChar(0));
   556 		err = sqlite3_exec(dbHandle, (const char*)createDbScript2->Des().Ptr(), NULL, 0, NULL);
   557 		PanicIfError(dbHandle, err);
   558 		delete createDbScript2;
   559 		(void)sqlite3_close(dbHandle);
   560 		}
   561 	static sqlite3* Open(const TDesC& aDbFileName)
   562 		{
   563 		HBufC8* dbName = HBufC8::New(aDbFileName.Length() + 1);
   564 		TEST(dbName != NULL);
   565 		dbName->Des().Copy(aDbFileName);
   566 		sqlite3* dbHandle = NULL;
   567 		TInt err = sqlite3_open((const char*)(dbName->Des().PtrZ()), &dbHandle);
   568 		delete dbName;
   569 		PanicIfError(dbHandle, err);
   570 		return dbHandle;
   571 		}
   572 	};
   573 
   574 //Explicit TDbHelper2 class specialization for creating/opening a database with UTF16 default encoding
   575 template <> class TDbHelper2<TCmdLineParams::EDbUtf16>
   576 	{
   577 public:	
   578 	static void Create(const TDesC& aDbFileName)
   579 		{
   580 		(void)TheFs.Delete(aDbFileName);
   581 		HBufC16* dbName = HBufC16::New(aDbFileName.Length() + 1);
   582 		TEST(dbName != NULL);
   583 		dbName->Des().Copy(aDbFileName);
   584 		sqlite3* dbHandle = NULL;
   585 		TInt err = sqlite3_open16((const void*)(dbName->Des().PtrZ()), &dbHandle);
   586 		delete dbName;
   587 		PanicIfError(dbHandle, err);
   588 		HBufC8* createDbScript = ReadSqlScript<HBufC8>(KCreateDbScript());
   589 		HBufC8* createDbScript2 = HBufC8::New(createDbScript->Des().Length() + 1);
   590 		TEST(createDbScript2 != NULL);
   591 		createDbScript2->Des().Copy(createDbScript->Des());
   592 		delete createDbScript;
   593 		createDbScript2->Des().Append(TChar(0));
   594 		err = sqlite3_exec(dbHandle, (const char*)createDbScript2->Des().Ptr(), NULL, 0, NULL);
   595 		PanicIfError(dbHandle, err);
   596 		delete createDbScript2;
   597 		(void)sqlite3_close(dbHandle);
   598 		}
   599 	static sqlite3* Open(const TDesC& aDbFileName)
   600 		{
   601 		HBufC16* dbName = HBufC16::New(aDbFileName.Length() + 1);
   602 		TEST(dbName != NULL);
   603 		dbName->Des().Copy(aDbFileName);
   604 		sqlite3* dbHandle = NULL;
   605 		TInt err = sqlite3_open16((const void*)(dbName->Des().PtrZ()), &dbHandle);
   606 		delete dbName;
   607 		PanicIfError(dbHandle, err);
   608 		return dbHandle;
   609 		}
   610 	};
   611 
   612 template <class HBUFC, class PTRC, class DESC> void InsertTest2(sqlite3* aDbHandle, const TDesC& aScriptFileName, const DESC& aCommitStr);
   613 
   614 //Explicit InsertTest2() template specialization for UTF8 encoded SQL strings
   615 template <> void InsertTest2<HBufC8, TPtrC8, TDesC8>(sqlite3* aDbHandle, const TDesC& aScriptFileName, const TDesC8& aCommitStr)
   616 	{
   617 	TheTest.Printf(_L("\"Insert\" test\r\n"));
   618 	HBufC8* fillDbScript = ReadSqlScript<HBufC8>(aScriptFileName);
   619 	TUint32 start = User::FastCounter();
   620 	TPtrC8 ptr(fillDbScript->Des());
   621 	TPtrC8 sql(GetNextTrans<TPtrC8, TDesC8>(ptr, aCommitStr));
   622 	while(sql.Length() > 0)
   623 		{
   624 		TUint8* p = (TUint8*)sql.Ptr();
   625 		p[sql.Length() - 1] = 0;
   626 		TInt err = sqlite3_exec(aDbHandle, (const char*)sql.Ptr(), NULL, 0, NULL);
   627 		PanicIfError(aDbHandle, err);
   628 		sql.Set(GetNextTrans<TPtrC8, TDesC8>(ptr, aCommitStr));
   629 		}
   630 	TUint32 end = User::FastCounter();
   631 	PrintStats(start, end);
   632 	delete fillDbScript;
   633 	}
   634 
   635 //Explicit InsertTest2() template specialization for UTF16 encoded SQL strings
   636 template <> void InsertTest2<HBufC16, TPtrC16, TDesC16>(sqlite3* aDbHandle, const TDesC& aScriptFileName, const TDesC16& aCommitStr)
   637 	{
   638 	TheTest.Printf(_L("\"Insert\" test\r\n"));
   639 	HBufC16* fillDbScript = ReadSqlScript<HBufC16>(aScriptFileName);
   640 	TUint32 start = User::FastCounter();
   641 	TPtrC16 ptr(fillDbScript->Des());
   642 	TPtrC16 sql(GetNextTrans<TPtrC16, TDesC16>(ptr, aCommitStr));
   643 	while(sql != KNullDesC16)
   644 		{
   645 		TPtr16 p((TUint16*)sql.Ptr(), sql.Length(), sql.Length());
   646 		TPtrC16 current(KNullDesC16);
   647 		while(p.Length() > 1) //"> 1" because it is a zero terminated string
   648 			{
   649 			current.Set(GetFirstSqlStmt(p));
   650 			sqlite3_stmt* stmtHandle = NULL;
   651 			const void* stmtTail = NULL;
   652 			TInt err = sqlite3_prepare16_v2(aDbHandle, current.Ptr(), -1, &stmtHandle, &stmtTail);
   653 			if(stmtHandle)	//stmtHandle can be NULL for statements like this: ";".
   654 				{
   655 				if(err == SQLITE_OK)
   656 					{
   657 					while((err = sqlite3_step(stmtHandle)) == SQLITE_ROW)
   658 						{
   659 						}
   660 					}
   661 				TInt err2 = sqlite3_finalize(stmtHandle);
   662 				TEST2(err2, SQLITE_OK);
   663 				}
   664 			if(ReportIfError(aDbHandle, err))
   665 				{
   666 				delete fillDbScript;
   667 				return;
   668 				}
   669   			}//while(p.Length() > 1)
   670 		sql.Set(GetNextTrans<TPtrC16, TDesC16>(ptr, aCommitStr));
   671 		}//while(sql != KNullDesC16)
   672 	TUint32 end = User::FastCounter();
   673 	PrintStats(start, end);
   674 	delete fillDbScript;
   675 	}
   676 
   677 template <class HBUFC, class DESC> void UpdateTest2(sqlite3* aDbHandle, const DESC& aUpdateSql);
   678 
   679 //Explicit UpdateTest2() template specialization for UTF8 encoded SQL strings
   680 template <> void UpdateTest2<HBufC8, TDesC8>(sqlite3* aDbHandle, const TDesC8& aUpdateSql)
   681 	{
   682 	TheTest.Printf(_L("\"Update\" test\r\n"));
   683 	HBufC8* sql = HBufC8::New(aUpdateSql.Length() + 1);
   684 	TEST(sql != NULL);
   685 	sql->Des().Copy(aUpdateSql);
   686 	sql->Des().Append(TChar(0));
   687 
   688 	sqlite3_stmt* stmtHandle = NULL;
   689 	const char* stmtTail = NULL;
   690 	TInt err = sqlite3_prepare_v2(aDbHandle, (const char*)sql->Des().Ptr(), -1, &stmtHandle, &stmtTail);
   691 	delete sql;
   692 	PanicIfError(aDbHandle, err);
   693 	TInt firstNamePrmIdx = sqlite3_bind_parameter_index(stmtHandle, KRawPrmName1);
   694 	TEST(firstNamePrmIdx >= 0);
   695 	TInt lastNamePrmIdx = sqlite3_bind_parameter_index(stmtHandle, KRawPrmName2);
   696 	TEST(lastNamePrmIdx >= 0);
   697 	TInt companyNamePrmIdx = sqlite3_bind_parameter_index(stmtHandle, KRawPrmName3);
   698 	TEST(companyNamePrmIdx >= 0);
   699 	TInt idIdx = sqlite3_bind_parameter_index(stmtHandle, KRawPrmName4);
   700 	TEST(idIdx >= 0);
   701 	
   702 	TUint32 start = User::FastCounter();
   703 	for(TInt id=1;id<=KTestTecordCount;++id)
   704 		{
   705 		TBuf8<20> buf1;
   706 		buf1.Copy(KFirstName);
   707 		buf1.AppendNum(id);
   708 		buf1.Append(TChar(0));
   709 		err = sqlite3_bind_text(stmtHandle, firstNamePrmIdx, (const char*)buf1.Ptr(), -1, SQLITE_STATIC);
   710 		TEST2(err, SQLITE_OK);
   711 		TBuf8<20> buf2;
   712 		buf2.Copy(KLastName);
   713 		buf2.AppendNum(id);
   714 		buf2.Append(TChar(0));
   715 		err = sqlite3_bind_text(stmtHandle, lastNamePrmIdx, (const char*)buf2.Ptr(), -1, SQLITE_STATIC);
   716 		TEST2(err, SQLITE_OK);
   717 		TBuf8<20> buf3;
   718 		buf3.Copy(KCompanyName);
   719 		buf3.AppendNum(id);
   720 		buf3.Append(TChar(0));
   721 		err = sqlite3_bind_text(stmtHandle, companyNamePrmIdx, (const char*)buf3.Ptr(), -1, SQLITE_STATIC);
   722 		TEST2(err, SQLITE_OK);
   723 		err = sqlite3_bind_int(stmtHandle, idIdx, id);
   724 		TEST2(err, SQLITE_OK);
   725 		while((err = sqlite3_step(stmtHandle)) == SQLITE_ROW)
   726 			{
   727 			}
   728 		PanicIfError(aDbHandle, err);
   729 		err = sqlite3_reset(stmtHandle);
   730 		PanicIfError(aDbHandle, err);
   731 		}
   732 	TUint32 end = User::FastCounter();
   733 	PrintStats(start, end);
   734 	err = sqlite3_finalize(stmtHandle);
   735 	TEST2(err, SQLITE_OK);
   736 	}
   737 
   738 //Explicit UpdateTest2() template specialization for UTF16 encoded SQL strings
   739 template <> void UpdateTest2<HBufC16, TDesC16>(sqlite3* aDbHandle, const TDesC16& aUpdateSql)
   740 	{
   741 	TheTest.Printf(_L("\"Update\" test\r\n"));
   742 	HBufC16* sql = HBufC16::New(aUpdateSql.Length() + 1);
   743 	TEST(sql != NULL);
   744 	sql->Des().Copy(aUpdateSql);
   745 	sql->Des().Append(TChar(0));
   746 
   747 	sqlite3_stmt* stmtHandle = NULL;
   748 	const void* stmtTail = NULL;
   749 	TInt err = sqlite3_prepare16_v2(aDbHandle, (const void*)sql->Des().Ptr(), -1, &stmtHandle, &stmtTail);
   750 	delete sql;
   751 	PanicIfError(aDbHandle, err);
   752 	TInt firstNamePrmIdx = sqlite3_bind_parameter_index(stmtHandle, KRawPrmName1);
   753 	TEST(firstNamePrmIdx >= 0);
   754 	TInt lastNamePrmIdx = sqlite3_bind_parameter_index(stmtHandle, KRawPrmName2);
   755 	TEST(lastNamePrmIdx >= 0);
   756 	TInt companyNamePrmIdx = sqlite3_bind_parameter_index(stmtHandle, KRawPrmName3);
   757 	TEST(companyNamePrmIdx >= 0);
   758 	TInt idIdx = sqlite3_bind_parameter_index(stmtHandle, KRawPrmName4);
   759 	TEST(idIdx >= 0);
   760 	
   761 	TUint32 start = User::FastCounter();
   762 	for(TInt id=1;id<=KTestTecordCount;++id)
   763 		{
   764 		TBuf16<20> buf1;
   765 		buf1.Copy(KFirstName);
   766 		buf1.AppendNum(id);
   767 		buf1.Append(TChar(0));
   768 		err = sqlite3_bind_text16(stmtHandle, firstNamePrmIdx, (const void*)buf1.Ptr(), -1, SQLITE_STATIC);
   769 		TEST2(err, SQLITE_OK);
   770 		TBuf16<20> buf2;
   771 		buf2.Copy(KLastName);
   772 		buf2.AppendNum(id);
   773 		buf2.Append(TChar(0));
   774 		err = sqlite3_bind_text16(stmtHandle, lastNamePrmIdx, (const void*)buf2.Ptr(), -1, SQLITE_STATIC);
   775 		TEST2(err, SQLITE_OK);
   776 		TBuf16<20> buf3;
   777 		buf3.Copy(KCompanyName);
   778 		buf3.AppendNum(id);
   779 		buf3.Append(TChar(0));
   780 		err = sqlite3_bind_text16(stmtHandle, companyNamePrmIdx, (const void*)buf3.Ptr(), -1, SQLITE_STATIC);
   781 		TEST2(err, SQLITE_OK);
   782 		err = sqlite3_bind_int(stmtHandle, idIdx, id);
   783 		TEST2(err, SQLITE_OK);
   784 		while((err = sqlite3_step(stmtHandle)) == SQLITE_ROW)
   785 			{
   786 			}
   787 		PanicIfError(aDbHandle, err);
   788 		err = sqlite3_reset(stmtHandle);
   789 		PanicIfError(aDbHandle, err);
   790 		}
   791 	TUint32 end = User::FastCounter();
   792 	PrintStats(start, end);
   793 	err = sqlite3_finalize(stmtHandle);
   794 	TEST2(err, SQLITE_OK);
   795 	}
   796 
   797 //"UPDATE" test function (without parameters) - SQLITE
   798 template <class DESC> void UpdateWPTest2(const TDesC& aDbName, const DESC& aUpdateSql);
   799 
   800 //Explicit UpdateWPTest2() template specialization for UTF8 encoded SQL strings
   801 template <> void UpdateWPTest2<TDesC8>(const TDesC& aDbName, const TDesC8& aUpdateSql)
   802 	{
   803 	TheTest.Printf(_L("\"Update (without parameters)\" test\r\n"));
   804 	TBuf<200> fmtstr;
   805 	fmtstr.Copy(aUpdateSql);
   806 
   807 	sqlite3SymbianLibInit();
   808 	sqlite3* dbHandle = TDbHelper2<TCmdLineParams::EDbUtf8>::Open(aDbName);
   809 	
   810 	TUint32 start = User::FastCounter();
   811 	for(TInt id=1;id<=KTestTecordCount;++id)
   812 		{
   813 		TBuf<200> buf;
   814 		buf.Format(fmtstr, &KFirstName2, id, &KLastName2, id, &KCompanyName2, id, id);
   815 		TBuf8<200> sql;
   816 		sql.Copy(buf);
   817 		sql.Append(0);
   818 		TInt err = sqlite3_exec(dbHandle, (const char*)sql.Ptr(), 0, 0, 0);
   819 		TEST2(err, SQLITE_OK);
   820 		}
   821 	TUint32 end = User::FastCounter();
   822 	PrintStats(start, end);
   823 	
   824 	TInt err2 = sqlite3_close(dbHandle);
   825 	TEST2(err2, SQLITE_OK);
   826 	sqlite3SymbianLibFinalize();
   827 	CloseSTDLIB();
   828 	}
   829 
   830 //Explicit UpdateWPTest2() template specialization for UTF16 encoded SQL strings
   831 template <> void UpdateWPTest2<TDesC16>(const TDesC& aDbName, const TDesC16& aUpdateSql)
   832 	{
   833 	TheTest.Printf(_L("\"Update (without parameters)\" test\r\n"));
   834 
   835 	sqlite3SymbianLibInit();
   836 	sqlite3* dbHandle = TDbHelper2<TCmdLineParams::EDbUtf16>::Open(aDbName);
   837 	
   838 	TUint32 start = User::FastCounter();
   839 	for(TInt id=1;id<=KTestTecordCount;++id)
   840 		{
   841 		TBuf<200> sql;
   842 		sql.Format(aUpdateSql, &KFirstName2, id, &KLastName2, id, &KCompanyName2, id, id);
   843 		sql.Append(0);
   844 		sqlite3_stmt* stmtHandle = NULL;
   845 		const void* stmtTail = NULL;
   846 		TInt err = sqlite3_prepare16_v2(dbHandle, sql.Ptr(), -1, &stmtHandle, &stmtTail);
   847 		TEST2(err, SQLITE_OK);
   848 		if(stmtHandle)	//stmtHandle can be NULL for statements like this: ";".
   849 			{
   850 			if(err == SQLITE_OK)
   851 				{
   852 				while((err = sqlite3_step(stmtHandle)) == SQLITE_ROW)
   853 					{
   854 					}
   855 				}
   856 			TInt err2 = sqlite3_finalize(stmtHandle);
   857 			TEST2(err2, SQLITE_OK);
   858 			TEST2(err, SQLITE_DONE);
   859 			}
   860 		}
   861 	TUint32 end = User::FastCounter();
   862 	PrintStats(start, end);
   863 	
   864 	TInt err2 = sqlite3_close(dbHandle);
   865 	TEST2(err2, SQLITE_OK);
   866 	sqlite3SymbianLibFinalize();
   867 	CloseSTDLIB();
   868 	}
   869 
   870 template <class HBUFC, class DESC> void SelectTest2(sqlite3* aDbHandle, const DESC& aSelectSql);
   871 
   872 //Explicit SelectTest2() template specialization for UTF8 encoded SQL strings
   873 template <> void SelectTest2<HBufC8, TDesC8>(sqlite3* aDbHandle, const TDesC8& aSelectSql)
   874 	{
   875 	TheTest.Printf(_L("\"Select\" test\r\n"));
   876 	HBufC8* sql = HBufC8::New(aSelectSql.Length() + 1);
   877 	TEST(sql != NULL);
   878 	sql->Des().Copy(aSelectSql);
   879 	sql->Des().Append(TChar(0));
   880 
   881 	sqlite3_stmt* stmtHandle = NULL;
   882 	const char* stmtTail = NULL;
   883 	TInt err = sqlite3_prepare_v2(aDbHandle, (const char*)sql->Des().Ptr(), -1, &stmtHandle, &stmtTail);
   884 	delete sql;
   885 	PanicIfError(aDbHandle, err);
   886 	
   887 	TUint32 start = User::FastCounter();
   888 	while((err = sqlite3_step(stmtHandle)) == SQLITE_ROW)
   889 		{
   890 		TBuf8<20> buf;
   891 		const unsigned char* p = sqlite3_column_text(stmtHandle, 0);
   892 		buf.Copy(p, User::StringLength(p));
   893 		TEST(buf.Length() > 0);
   894 		p = sqlite3_column_text(stmtHandle, 1);
   895 		buf.Copy(p, User::StringLength(p));
   896 		TEST(buf.Length() > 0);
   897 		p = sqlite3_column_text(stmtHandle, 2);
   898 		buf.Copy(p, User::StringLength(p));
   899 		TEST(buf.Length() > 0);
   900 		}
   901 	TEST2(err, SQLITE_DONE);
   902 	TUint32 end = User::FastCounter();
   903 	PrintStats(start, end);
   904 	err = sqlite3_finalize(stmtHandle);
   905 	TEST2(err, SQLITE_OK);
   906 	}
   907 
   908 //Explicit SelectTest2() template specialization for UTF16 encoded SQL strings
   909 template <> void SelectTest2<HBufC16, TDesC16>(sqlite3* aDbHandle, const TDesC16& aSelectSql)
   910 	{
   911 	TheTest.Printf(_L("\"Select\" test\r\n"));
   912 	HBufC16* sql = HBufC16::New(aSelectSql.Length() + 1);
   913 	TEST(sql != NULL);
   914 	sql->Des().Copy(aSelectSql);
   915 	sql->Des().Append(TChar(0));
   916 
   917 	sqlite3_stmt* stmtHandle = NULL;
   918 	const void* stmtTail = NULL;
   919 	TInt err = sqlite3_prepare16_v2(aDbHandle, (const void*)sql->Des().Ptr(), -1, &stmtHandle, &stmtTail);
   920 	delete sql;
   921 	PanicIfError(aDbHandle, err);
   922 	
   923 	TUint32 start = User::FastCounter();
   924 	while((err = sqlite3_step(stmtHandle)) == SQLITE_ROW)
   925 		{
   926 		TBuf16<20> buf;
   927 		const void* p = sqlite3_column_text16(stmtHandle, 0);
   928 		buf.Copy((const TUint16*)p, User::StringLength((const TUint16*)p));
   929 		TEST(buf.Length() > 0);
   930 		p = sqlite3_column_text16(stmtHandle, 1);
   931 		buf.Copy((const TUint16*)p, User::StringLength((const TUint16*)p));
   932 		TEST(buf.Length() > 0);
   933 		p = sqlite3_column_text16(stmtHandle, 2);
   934 		buf.Copy((const TUint16*)p, User::StringLength((const TUint16*)p));
   935 		TEST(buf.Length() > 0);
   936 		}
   937 	TEST2(err, SQLITE_DONE);
   938 	TUint32 end = User::FastCounter();
   939 	PrintStats(start, end);
   940 	err = sqlite3_finalize(stmtHandle);
   941 	TEST2(err, SQLITE_OK);
   942 	}
   943 
   944 template <class HBUFC, class DESC> void DeleteTest2(sqlite3* aDbHandle, const DESC& aDeleteSql);
   945 
   946 //Explicit DeleteTest2() template specialization for UTF8 encoded SQL strings
   947 template <> void DeleteTest2<HBufC8, TDesC8>(sqlite3* aDbHandle, const TDesC8& aDeleteSql)
   948 	{
   949 	TheTest.Printf(_L("\"Delete\" test\r\n"));
   950 	HBufC8* sql = HBufC8::New(aDeleteSql.Length() + 1);
   951 	TEST(sql != NULL);
   952 	sql->Des().Copy(aDeleteSql);
   953 	sql->Des().Append(TChar(0));
   954 	TUint32 start = User::FastCounter();
   955 	TInt err = sqlite3_exec(aDbHandle, (const char*)sql->Des().Ptr(), 0, 0, 0);
   956 	TEST2(err, SQLITE_OK);
   957 	TUint32 end = User::FastCounter();
   958 	PrintStats(start, end);
   959 	delete sql;
   960 	}
   961 
   962 //Explicit DeleteTest2() template specialization for UTF16 encoded SQL strings
   963 template <> void DeleteTest2<HBufC16, TDesC16>(sqlite3* aDbHandle, const TDesC16& aDeleteSql)
   964 	{
   965 	TheTest.Printf(_L("\"Delete\" test\r\n"));
   966 	HBufC16* sql = HBufC16::New(aDeleteSql.Length() + 1);
   967 	TEST(sql != NULL);
   968 	sql->Des().Copy(aDeleteSql);
   969 	sql->Des().Append(TChar(0));
   970 	TUint32 start = User::FastCounter();
   971 	sqlite3_stmt* stmtHandle = NULL;
   972 	const void* stmtTail = NULL;
   973 	TInt err = sqlite3_prepare16_v2(aDbHandle, sql->Des().Ptr(), -1, &stmtHandle, &stmtTail);
   974 	TEST2(err, SQLITE_OK);
   975 	if(stmtHandle)	//stmtHandle can be NULL for statements like this: ";".
   976 		{
   977 		if(err == SQLITE_OK)
   978 			{
   979 			while((err = sqlite3_step(stmtHandle)) == SQLITE_ROW)
   980 				{
   981 				}
   982 			}
   983 		TInt err2 = sqlite3_finalize(stmtHandle);
   984 		TEST2(err2, SQLITE_OK);
   985 		TEST2(err, SQLITE_DONE);
   986 		}
   987 	TUint32 end = User::FastCounter();
   988 	PrintStats(start, end);
   989 	delete sql;
   990 	}
   991 
   992 template <TCmdLineParams::TDbEncoding TYPE, class HBUFC, class PTRC, class DESC> void PerformanceTest2(
   993 																					   const TDesC& aDbFileName, 
   994 																					   const DESC& aCommitStr,
   995 																					   const DESC& aUpdateSql,
   996 																					   const DESC& aSelectSql,
   997 																					   const DESC& aDeleteSql)
   998 	{
   999 	CFileMan* fm = NULL;
  1000 	TRAPD(err, fm = CFileMan::NewL(TheFs));
  1001 	TEST2(err, KErrNone);
  1002 	
  1003 	sqlite3SymbianLibInit();
  1004 	TDbHelper2<TYPE>::Create(aDbFileName);
  1005 	
  1006 	sqlite3* dbHandle = TDbHelper2<TYPE>::Open(aDbFileName);
  1007 	InsertTest2<HBUFC, PTRC, DESC>(dbHandle, KFillDbScript(), aCommitStr);
  1008 	(void)sqlite3_close(dbHandle); dbHandle = NULL;
  1009 	PrintFileSize(aDbFileName);
  1010 
  1011 	dbHandle = TDbHelper2<TYPE>::Open(aDbFileName);
  1012 	UpdateTest2<HBUFC, DESC>(dbHandle, aUpdateSql);
  1013 	(void)sqlite3_close(dbHandle); dbHandle = NULL;
  1014 
  1015 	dbHandle = TDbHelper2<TYPE>::Open(aDbFileName);
  1016 	SelectTest2<HBUFC, DESC>(dbHandle, aSelectSql);
  1017 	(void)sqlite3_close(dbHandle); dbHandle = NULL;
  1018 
  1019 	(void)fm->Copy(aDbFileName, TheNonSecureTmpDbName);
  1020 	
  1021 	dbHandle = TDbHelper2<TYPE>::Open(aDbFileName);
  1022 	DeleteTest2<HBUFC, DESC>(dbHandle, aDeleteSql);
  1023 	(void)sqlite3_close(dbHandle); dbHandle = NULL;
  1024 	PrintFileSize(aDbFileName);
  1025 	
  1026 	sqlite3SymbianLibFinalize();
  1027 	CloseSTDLIB();
  1028 
  1029 	(void)fm->Copy(TheNonSecureTmpDbName, aDbFileName);
  1030 	(void)fm->Delete(TheNonSecureTmpDbName);
  1031 	delete fm;
  1032 	}
  1033 //Explicit PerformanceTest2() template instantiations.
  1034 template void PerformanceTest2<TCmdLineParams::EDbUtf8, HBufC8, TPtrC8, TDesC8>(const TDesC&, const TDesC8&, const TDesC8&, const TDesC8&, const TDesC8&);
  1035 template void PerformanceTest2<TCmdLineParams::EDbUtf16, HBufC8, TPtrC8, TDesC8>(const TDesC&, const TDesC8&, const TDesC8&, const TDesC8&, const TDesC8&);
  1036 template void PerformanceTest2<TCmdLineParams::EDbUtf8, HBufC16, TPtrC16, TDesC16>(const TDesC&, const TDesC16&, const TDesC16&, const TDesC16&, const TDesC16&);
  1037 template void PerformanceTest2<TCmdLineParams::EDbUtf16, HBufC16, TPtrC16, TDesC16>(const TDesC&, const TDesC16&, const TDesC16&, const TDesC16&, const TDesC16&);
  1038 
  1039 void ColumnValueAccessTest()
  1040 	{
  1041 	_LIT(KColName1, "Column1");
  1042 	_LIT(KColName2, "Column2");
  1043 	_LIT(KColName3, "Column3");
  1044 	_LIT(KColName4, "Column4");
  1045 	const TInt KColCount = 4;
  1046 	TPtrC colNames[KColCount] = {KColName1(), KColName2(), KColName3(), KColName4()};
  1047 	TBuf<100> sql;
  1048 	
  1049 	//Create a test database
  1050 	RSqlDatabase db;
  1051 	TInt err = db.Create(TheNonSecureDbName2, &TheSqlConfigString);
  1052 	TEST2(err, KErrNone);
  1053 	_LIT(KCreateSql, "CREATE TABLE A(%S INTEGER, %S INTEGER, %S INTEGER, %S INTEGER)");
  1054 	sql.Format(KCreateSql, &colNames[0], &colNames[1], &colNames[2], &colNames[3]);
  1055 	err = db.Exec(sql);
  1056 	TEST(err >= 0);	
  1057 	//Insert a record in the test database
  1058 	_LIT(KInsertSql, "INSERT INTO A(%S, %S, %S, %S) VALUES(0, 1, 2, 3)");
  1059 	sql.Format(KInsertSql, &colNames[0], &colNames[1], &colNames[2], &colNames[3]);
  1060 	err = db.Exec(sql);
  1061 	TEST2(err, 1);
  1062 	//Retrieve the record
  1063 	RSqlStatement stmt;
  1064 	err = stmt.Prepare(db, _L8("SELECT * FROM A"));
  1065 	TEST2(err, KErrNone);
  1066 	err = stmt.Next();
  1067 	TEST2(err, KSqlAtRow);
  1068 	
  1069 	//Init the random numbers generator
  1070 	TInt64 seed = (TInt64)&ColumnValueAccessTest;
  1071 	const TInt KTestCount = 20000;
  1072 	TInt val;
  1073 	
  1074 	//Test 1: retrieve column values using the column index
  1075 	TUint32 start = User::FastCounter();
  1076 	for(TInt i1=0;i1<KTestCount;++i1)	
  1077 		{
  1078 		TInt colIdx = Math::Rand(seed) % KColCount;
  1079 		val = stmt.ColumnInt(colIdx);
  1080 		TEST(val == colIdx);
  1081 		}
  1082 	TUint32 end = User::FastCounter();
  1083 	PrintStats(start, end);
  1084 	
  1085 	//Test 2: retrieve column values using the column name
  1086 	start = User::FastCounter();
  1087 	for(TInt i2=0;i2<KTestCount;++i2)	
  1088 		{
  1089 		TInt colNameIdx = Math::Rand(seed) % KColCount;
  1090 		TInt colIdx = stmt.ColumnIndex(colNames[colNameIdx]);
  1091 		val = stmt.ColumnInt(colIdx);
  1092 		TEST(val == colIdx);
  1093 		}
  1094 	end = User::FastCounter();
  1095 	PrintStats(start, end);
  1096 	
  1097 	//Cleanup
  1098 	stmt.Close();
  1099 	db.Close();
  1100 	(void)RSqlDatabase::Delete(TheNonSecureDbName2);
  1101 	}
  1102 
  1103 ///////////////////////////////////////////////////////////////////////////////////
  1104 ////////////      "Single record" INSERT/UPDATE/DELETE tests       ////////////////
  1105 ///////////////////////////////////////////////////////////////////////////////////
  1106 
  1107 _LIT(KZDbName, "z:\\test\\default_avacon.dbSQL");
  1108 
  1109 RSqlDatabase TheDb;
  1110 
  1111 void OpenTestDb()
  1112 	{
  1113 	(void)RSqlDatabase::Delete(TheSglRecDbFileName);
  1114 	TInt err = RSqlDatabase::Copy(KZDbName, TheSglRecDbFileName);
  1115 	TEST2(err, KErrNone);
  1116 	err = TheDb.Open(TheSglRecDbFileName);
  1117 	TEST2(err, KErrNone);
  1118 	}
  1119 	
  1120 void CloseTestDb()
  1121 	{
  1122 	TheDb.Close();
  1123 	TInt err = RSqlDatabase::Delete(TheSglRecDbFileName);
  1124 	TEST2(err, KErrNone);
  1125 	}
  1126 
  1127 void SingleUpdateTest()
  1128 	{
  1129 	OpenTestDb();
  1130 	
  1131 	_LIT(KUpdateSql, "UPDATE Table1 SET i2=41 WHERE i1 = 1");
  1132 	
  1133 	TUint32 start = User::FastCounter();
  1134 	TInt err = TheDb.Exec(KUpdateSql);
  1135 	TUint32 end = User::FastCounter();
  1136 	TEST2(err, 1);
  1137 
  1138 	PrintStats(start, end);
  1139 	
  1140 	CloseTestDb();
  1141 	}
  1142 
  1143 void SingleInsertTest()
  1144 	{
  1145 	OpenTestDb();
  1146 
  1147 	_LIT(KInsertSql, "INSERT INTO Table1 (i1, i2, i3, i4, r5, t6, t7, t8, lt9, lb10) values(1001, 1, 2002, 1001, 1001.001001, '', 'attic attic baker', 'attic attic baker attic attic baker attic attic baker attic attic baker ', 'ATTIC ATTIC BAKER ATTIC ATTIC BAKER ATTIC ATTIC BAKER ATTIC ATTIC BAKER ', NULL)");
  1148 	
  1149 	TUint32 start = User::FastCounter();
  1150 	TInt err = TheDb.Exec(KInsertSql);
  1151 	TUint32 end = User::FastCounter();
  1152 	TEST2(err, 1);
  1153 
  1154 	PrintStats(start, end);
  1155 	
  1156 	CloseTestDb();
  1157 	}
  1158 
  1159 void SingleDeleteTest()
  1160 	{
  1161 	OpenTestDb();
  1162 
  1163 	_LIT(KDeleteSql, "Delete from Table1 where i1 = 1");
  1164 	
  1165 	TUint32 start = User::FastCounter();
  1166 	TInt err = TheDb.Exec(KDeleteSql);
  1167 	TUint32 end = User::FastCounter();
  1168 	TEST2(err, 1);
  1169 
  1170 	PrintStats(start, end);
  1171 	
  1172 	CloseTestDb();
  1173 	}
  1174 	
  1175 void SelectLikeQueryPerfTest()
  1176 {
  1177 	_LIT8(KServerConfigString1, "encoding =  \"UTF-8\"");
  1178 	//create a sql db file
  1179 	(void)RSqlDatabase::Delete(TheNonSecureDbName);
  1180 	TInt err = TheDb.Create(TheNonSecureDbName, &KServerConfigString1);
  1181 	TEST2(err, KErrNone);
  1182 	
  1183 	//Create a test table and insert some records
  1184 	err = TheDb.Exec(_L("CREATE TABLE A(Id INTEGER PRIMARY KEY, Name TEXT)"));
  1185 	TEST(err >= 0);
  1186 	
  1187 	//Add 100 records to the table
  1188 	for(TInt k=0;k<100;++k)
  1189 		{
  1190 		TBuf<100> sql;
  1191 		sql.Format(_L("INSERT INTO A VALUES(%d, 'Dvorák')"), k + 1);
  1192 		err = TheDb.Exec(sql);
  1193 		TEST2(err, 1);
  1194 		}
  1195 	
  1196 	//Update one record  
  1197 	err = TheDb.Exec(_L("UPDATE A SET Name='dvorak1' WHERE ID = 50"));
  1198 	TEST2(err, 1);
  1199 
  1200 	TUint32 start = User::FastCounter();
  1201 	
  1202 	//Create a statement object and issue a SELECT SQL statement + LIKE clause
  1203 	//Test case 1 = full name search with LIKE
  1204 	RSqlStatement stmt;
  1205 	err = stmt.Prepare(TheDb, _L("SELECT COUNT(*) FROM A WHERE Name LIKE 'DVORAK1'"));
  1206 	TEST2(err, KErrNone);
  1207 	err = stmt.Next();
  1208 	TEST2(err, KSqlAtRow);
  1209 	TInt cnt = stmt.ColumnInt(0);
  1210 	TEST2(cnt, 1);
  1211 	
  1212 	stmt.Close();
  1213 	
  1214 	TUint32 end = User::FastCounter();
  1215 	TEST2(err, 1);
  1216 	PrintStats(start, end);
  1217 	
  1218 	TheDb.Close();
  1219 	(void)RSqlDatabase::Delete(TheNonSecureDbName);
  1220 }
  1221 
  1222 ///////////////////////////////////////////////////////////////////////////////////
  1223 ///////////////////////////////////////////////////////////////////////////////////
  1224 
  1225 /**
  1226 @SYMTestCaseID			SYSLIB-SQL-CT-1648
  1227 @SYMTestCaseDesc		SQL database performance tests.
  1228 						Three test types used: "insert records", "update records" and "select records".
  1229 						The tests are executed on: secure database, non-secure database, client side database.
  1230 						Each test executed twice using UTF8 and UTF16 encoded SQL statements.
  1231 @SYMTestPriority		High
  1232 @SYMTestActions			SQL database performance tests.
  1233 @SYMTestExpectedResults Test must not fail
  1234 @SYMREQ					REQ5792
  1235                         REQ5793
  1236                         REQ5912
  1237                         REQ10273
  1238 */	
  1239 void DoTests()
  1240 	{
  1241 	TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1648 SQL performance tests"));
  1242 	
  1243 	TheTest.Printf(_L("Single \"update\" test\r\n"));
  1244 	SingleUpdateTest();
  1245 	
  1246 	TheTest.Printf(_L("Single \"insert\" test\r\n"));
  1247 	SingleInsertTest();
  1248 	
  1249 	TheTest.Printf(_L("Single \"delete\" test\r\n"));
  1250 	SingleDeleteTest();
  1251 	
  1252 	TheTestTitle.Format(_L("SERVER, UTF8 SQL strings, non-secure, encoding: \"%S\", page size: %d\r\n"), 
  1253 			TheCmdLineParams.iDbEncoding == TCmdLineParams::EDbUtf16 ? &KUtf16 : &KUtf8, TheCmdLineParams.iPageSize);
  1254 	TheTest.Printf(TheTestTitle);
  1255 	PerformanceTest<HBufC8, TPtrC8, TDesC8, ENonSecureDb>(TheNonSecureDbName, KCommitStr8(), KUpdateSql8(), KSelectSql8(), KDeleteSql8());
  1256 
  1257 	TheTestTitle.Format(_L("SERVER, UTF8 SQL strings, non-secure, update test (without parameters), encoding: \"%S\", page size: %d\r\n"), 
  1258 			TheCmdLineParams.iDbEncoding == TCmdLineParams::EDbUtf16 ? &KUtf16 : &KUtf8, TheCmdLineParams.iPageSize);
  1259 	TheTest.Printf(TheTestTitle);
  1260 	UpdateWPTest<TBuf8<200>, TDesC8, ENonSecureDb>(TheNonSecureDbName, KUpdateSql2_8());
  1261 
  1262 	TheTestTitle.Format(_L("SERVER, UTF8 SQL strings, secure, encoding: \"%S\", page size: %d\r\n"), 
  1263 			TheCmdLineParams.iDbEncoding == TCmdLineParams::EDbUtf16 ? &KUtf16 : &KUtf8, TheCmdLineParams.iPageSize);
  1264 	TheTest.Printf(TheTestTitle);
  1265 	PerformanceTest<HBufC8, TPtrC8, TDesC8, ESecureDb>(TheSecureDbName, KCommitStr8(), KUpdateSql8(), KSelectSql8(), KDeleteSql8());
  1266 
  1267 	TheTest.Printf(_L("SQLITE, UTF8 encoded database, UTF8 SQL strings\r\n"));
  1268 	PerformanceTest2<TCmdLineParams::EDbUtf8, HBufC8, TPtrC8, TDesC8>(TheNonSecureDbName, KCommitStr8(), KUpdateSql8(), KSelectSql8(), KDeleteSql8());
  1269 
  1270 	TheTest.Printf(_L("SQLITE, UTF8 encoded database, UTF8 SQL strings, update test (without parameters)\r\n"));
  1271 	UpdateWPTest2<TDesC8>(TheNonSecureDbName, KUpdateSql2_8());
  1272 
  1273 	TheTest.Printf(_L("SQLITE, UTF16 encoded database, UTF8 SQL strings\r\n"));
  1274 	PerformanceTest2<TCmdLineParams::EDbUtf16, HBufC8, TPtrC8, TDesC8>(TheNonSecureDbName, KCommitStr8(), KUpdateSql8(), KSelectSql8(), KDeleteSql8());
  1275 
  1276 	TheTestTitle.Format(_L("SERVER, UTF16 SQL strings, non-secure, encoding: \"%S\", page size: %d\r\n"), 
  1277 			TheCmdLineParams.iDbEncoding == TCmdLineParams::EDbUtf16 ? &KUtf16 : &KUtf8, TheCmdLineParams.iPageSize);
  1278 	TheTest.Printf(TheTestTitle);
  1279 	PerformanceTest<HBufC16, TPtrC16, TDesC16, ENonSecureDb>(TheNonSecureDbName, KCommitStr16(), KUpdateSql16(), KSelectSql16(), KDeleteSql16());
  1280 	
  1281 	TheTestTitle.Format(_L("SERVER, UTF16 SQL strings, non-secure, update test (without parameters), encoding: \"%S\", page size: %d\r\n"), 
  1282 			TheCmdLineParams.iDbEncoding == TCmdLineParams::EDbUtf16 ? &KUtf16 : &KUtf8, TheCmdLineParams.iPageSize);
  1283 	TheTest.Printf(TheTestTitle);
  1284 	UpdateWPTest<TBuf16<200>, TDesC16, ENonSecureDb>(TheNonSecureDbName, KUpdateSql2_16());
  1285 
  1286 	TheTestTitle.Format(_L("SERVER, UTF16 SQL strings, secure, encoding: \"%S\", page size: %d\r\n"), 
  1287 			TheCmdLineParams.iDbEncoding == TCmdLineParams::EDbUtf16 ? &KUtf16 : &KUtf8, TheCmdLineParams.iPageSize);
  1288 	TheTest.Printf(TheTestTitle);
  1289 	PerformanceTest<HBufC16, TPtrC16, TDesC16, ESecureDb>(TheSecureDbName, KCommitStr16(), KUpdateSql16(), KSelectSql16(), KDeleteSql16());
  1290 
  1291 	TheTest.Printf(_L("SQLITE, UTF8 encoded database, UTF16 SQL strings\r\n"));
  1292 	PerformanceTest2<TCmdLineParams::EDbUtf8, HBufC16, TPtrC16, TDesC16>(TheNonSecureDbName, KCommitStr16(), KUpdateSql16(), KSelectSql16(), KDeleteSql16());
  1293 
  1294 	TheTest.Printf(_L("SQLITE, UTF16 encoded database, UTF16 SQL strings\r\n"));
  1295 	PerformanceTest2<TCmdLineParams::EDbUtf16, HBufC16, TPtrC16, TDesC16>(TheNonSecureDbName, KCommitStr16(), KUpdateSql16(), KSelectSql16(), KDeleteSql16());
  1296 
  1297 	TheTest.Printf(_L("SQLITE, UTF16 encoded database, UTF16 SQL strings, update test (without parameters)\r\n"));
  1298 	UpdateWPTest2<TDesC16>(TheNonSecureDbName, KUpdateSql2_16());
  1299 
  1300 	TheTest.Printf(_L("Accessing column value by index or by name\r\n"));
  1301 	ColumnValueAccessTest();
  1302 	
  1303 	TheTest.Printf(_L("Retrieving data from UTF8 Database using SELECT LIKE statements\r\n"));
  1304 	SelectLikeQueryPerfTest();
  1305 
  1306 	}
  1307 
  1308 TInt E32Main()
  1309 	{
  1310 	TheTest.Title();
  1311 
  1312 	CTrapCleanup* tc = CTrapCleanup::New();
  1313 	TheTest(tc != NULL);
  1314 
  1315 	GetCmdLineParamsAndSqlConfigString(TheTest, _L("t_sqlperformance"), TheCmdLineParams, TheSqlConfigString);
  1316 	_LIT(KSecureDbName, "c:[2121212A]t_perfdb.db");
  1317 	PrepareDbName(KSecureDbName, TheCmdLineParams.iDriveName, TheSecureDbName);
  1318 	_LIT(KNonSecureDbName, "c:\\test\\t_perfdb.db");
  1319 	PrepareDbName(KNonSecureDbName, TheCmdLineParams.iDriveName, TheNonSecureDbName);
  1320 	_LIT(KNonSecureDbName2, "c:\\test\\t_perfdb2.db");
  1321 	PrepareDbName(KNonSecureDbName2, TheCmdLineParams.iDriveName, TheNonSecureDbName2);
  1322 	_LIT(KNonSecureTmpDbName, "c:\\test\\tmp.db");
  1323 	PrepareDbName(KNonSecureTmpDbName, TheCmdLineParams.iDriveName, TheNonSecureTmpDbName);
  1324 	_LIT(KSglRecDbName, "c:\\test\\default_avacon.dbSQL");
  1325 	PrepareDbName(KSglRecDbName, TheCmdLineParams.iDriveName, TheSglRecDbFileName);
  1326 
  1327 	TheTest.Printf(_L("==Databases: %S, %S, %S, %S, %S\r\n"), &TheSecureDbName, &TheNonSecureDbName, 
  1328 															  &TheNonSecureDbName2, &TheNonSecureTmpDbName, 
  1329 															  &TheSglRecDbFileName);
  1330 	
  1331 	__UHEAP_MARK;
  1332 	
  1333 	TestEnvInit();
  1334 	DoTests();
  1335 	TestEnvDestroy();
  1336 
  1337 	__UHEAP_MARKEND;
  1338 	
  1339 	TheTest.End();
  1340 	TheTest.Close();
  1341 	
  1342 	delete tc;
  1343 
  1344 	User::Heap().Check();
  1345 	return KErrNone;
  1346 	}