os/persistentdata/persistentstorage/sql/TEST/t_sqlload.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 <sqldb.h>
    20 #include "SqlResourceTester.h"
    21 
    22 ///////////////////////////////////////////////////////////////////////////////////////
    23 
    24 #define UNUSED_VAR(a) (a) = (a)
    25 
    26 RTest TheTest(_L("t_sqlload test"));
    27 
    28 TDriveNumber KTestDrive = EDriveC;
    29 
    30 _LIT(KTestDir, "c:\\test\\");
    31 _LIT(KTestDbName1, "c:\\test\\t_sqlload_1.db");
    32 _LIT(KTestDbName2, "c:\\test\\t_sqlload_2.db");
    33 _LIT(KTestDbName3, "c:\\test\\t_sqlload_3.db");
    34 _LIT(KTestDbName4, "c:\\test\\t_sqlload_4.db");
    35 _LIT(KTestDbName5, "c:\\test\\t_sqlload_5.db");
    36 
    37 //Test thread count
    38 const TInt KTestThreadCnt = 4;
    39 
    40 //Test database names
    41 const TPtrC KTestDbNames[] =
    42 	{
    43 	KTestDbName1(),
    44 	KTestDbName2(),
    45 	KTestDbName3()
    46 	};
    47 
    48 //Test database count
    49 const TInt KTestDbCnt = sizeof(KTestDbNames) / sizeof(KTestDbNames[0]);
    50 
    51 //Test duration
    52 const TInt KTestDuration = 120;//seconds
    53 
    54 //Test record count
    55 const TInt KRecordCnt = 100;
    56 //Record count which will be used in the test SQL queries
    57 const TInt KQueriedRecordCnt = 40;
    58 //Every SQL query will be processed (stepped) in KTestStepCnt steps.
    59 const TInt KTestStepCnt = 4;
    60 //RSqlStatement object count which will be used in the tests
    61 const TInt KStatementCnt = 10;
    62 //Max allowed alive RSqlStatement objects per thread
    63 const TInt KMaxStatementPerThread = 30;
    64 //Binary data length
    65 const TInt KBinDataLen = 2003;
    66 
    67 //StatementMaxNumberTest() time limit in seconds.
    68 const TInt KTestTimeLimit = 60;//seconds
    69 
    70 ///////////////////////////////////////////////////////////////////////////////////////
    71 
    72 void DeleteTestFiles()
    73 	{
    74 	RSqlDatabase::Delete(KTestDbName5);
    75 	RSqlDatabase::Delete(KTestDbName4);
    76 	RSqlDatabase::Delete(KTestDbName3);
    77 	RSqlDatabase::Delete(KTestDbName2);
    78 	RSqlDatabase::Delete(KTestDbName1);
    79 	}
    80 
    81 void GetHomeTimeAsString(TDes& aStr)
    82 	{
    83 	TTime time;
    84 	time.HomeTime();
    85 	TDateTime dt = time.DateTime();
    86 	aStr.Format(_L("%02d:%02d:%02d.%06d"), dt.Hour(), dt.Minute(), dt.Second(), dt.MicroSecond());
    87 	}
    88 
    89 ///////////////////////////////////////////////////////////////////////////////////////
    90 ///////////////////////////////////////////////////////////////////////////////////////
    91 //Test macros and functions
    92 void Check1(TInt aValue, TInt aLine, TBool aPrintThreadName = EFalse)
    93 	{
    94 	if(!aValue)
    95 		{
    96 		DeleteTestFiles();
    97 		if(aPrintThreadName)
    98 			{
    99 			RThread th;
   100 			TName name = th.Name();
   101 			RDebug::Print(_L("*** Thread %S, Line %d\r\n"), &name, aLine);
   102 			}
   103 		else
   104 			{
   105 			RDebug::Print(_L("*** Line %d\r\n"), aLine);
   106 			}
   107 		TheTest(EFalse, aLine);
   108 		}
   109 	}
   110 void Check2(TInt aValue, TInt aExpected, TInt aLine, TBool aPrintThreadName = EFalse)
   111 	{
   112 	if(aValue != aExpected)
   113 		{
   114 		DeleteTestFiles();
   115 		if(aPrintThreadName)
   116 			{
   117 			RThread th;
   118 			TName name = th.Name();
   119 			RDebug::Print(_L("*** Thread %S, Line %d Expected error: %d, got: %d\r\n"), &name, aLine, aExpected, aValue);
   120 			}
   121 		else
   122 			{
   123 			RDebug::Print(_L("*** Line %d, Expected error: %d, got: %d\r\n"), aLine, aExpected, aValue);
   124 			}
   125 		TheTest(EFalse, aLine);
   126 		}
   127 	}
   128 #define TEST(arg) ::Check1((arg), __LINE__)
   129 #define TEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__)
   130 #define TTEST(arg) ::Check1((arg), __LINE__, ETrue)
   131 #define TTEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__, ETrue)
   132 
   133 ///////////////////////////////////////////////////////////////////////////////////////
   134 
   135 //StatementMaxNumberTest() timeouts in WDP builds.
   136 //This function is used to return the seconds passed from the start of the test case.
   137 TTimeIntervalSeconds ExecutionTimeSeconds(TTime& aStartTime)
   138 	{
   139 	TTime currTime;
   140 	currTime.HomeTime();
   141 	
   142 	TTimeIntervalSeconds s;
   143 	TInt err = currTime.SecondsFrom(aStartTime, s);
   144 	TEST2(err, KErrNone);
   145 	return s;
   146 	}
   147 
   148 void CreateTestDir()
   149     {
   150     RFs fs;
   151 	TInt err = fs.Connect();
   152 	TEST2(err, KErrNone);
   153 
   154 	err = fs.MkDir(KTestDir);
   155 	TEST(err == KErrNone || err == KErrAlreadyExists);
   156 
   157 	err = fs.CreatePrivatePath(KTestDrive);
   158 	TEST(err == KErrNone || err == KErrAlreadyExists);
   159 	
   160 	fs.Close();
   161 	}
   162 
   163 ///////////////////////////////////////////////////////////////////////////////////////
   164 
   165 void CreateTestDatabases()
   166 	{
   167 	HBufC8* recData = HBufC8::New(KBinDataLen * 2 + 50);//"* 2" - hex values for the INSERT SQL statement
   168 	TEST(recData != NULL);
   169 	TPtr8 sql = recData->Des();
   170 
   171 	for(TInt dbIdx=0;dbIdx<KTestDbCnt;++dbIdx)
   172 		{
   173 		//Create test database
   174 		RSqlDatabase db;
   175 		TInt err = db.Create(KTestDbNames[dbIdx]);
   176 		TEST2(err, KErrNone);
   177 
   178 		//Create test table
   179 		_LIT8(KCreateSql, "CREATE TABLE A(F1 INTEGER, F2 BLOB)");
   180 		err = db.Exec(KCreateSql);
   181 		TEST(err >= 0);
   182 
   183 		//Insert records in the test table
   184 		for(TInt recIdx=1;recIdx<=KRecordCnt;++recIdx)
   185 			{
   186 			_LIT8(KInsertSql, "INSERT INTO A(F1, F2) VALUES(");
   187 			sql.Copy(KInsertSql);
   188 			sql.AppendNum((TInt64)recIdx);
   189 			sql.Append(_L(", X'"));
   190 			for(TInt k=0;k<KBinDataLen;++k)
   191 				{
   192 				sql.AppendFormat(_L8("%02X"), recIdx);
   193 				}
   194 			sql.Append(_L("')"));
   195 			err = db.Exec(sql);
   196 			TEST2(err, 1);
   197 			}
   198 
   199 		db.Close();
   200 		}
   201 
   202 	delete recData;
   203 	}
   204 
   205 //Structure used by the test thread function for orginizing its set of test data.
   206 struct TSqlStatement
   207 	{
   208 	RSqlStatement	iObj;
   209 	TBool 			iAlive;			//Non-zero if iObj is alive
   210 	TInt			iCurIndex;		//The number of the current record in the set controlled by iObj statement
   211 	TInt			iEndIndex;		//The last record number in the set controlled by iObj statement
   212 	TInt			iCount;			//Records count in the set controlled by iObj statement
   213 	};
   214 	
   215 typedef RArray<TSqlStatement> RSqlStatementArray;	
   216 
   217 //Inits the random numbers generator.
   218 //Opens one of the test databases.
   219 void PreTest(RSqlDatabase& aDb, TInt64& aSeed, TName& aThreadName)
   220 	{
   221 	RThread currThread;
   222 	
   223 	//Init the random numbers generator
   224 	TTime now;
   225 	now.UniversalTime();
   226 	aSeed = now.Int64() + currThread.Id();
   227 
   228 	//Open one of the test databases
   229 	const TInt KDbIndex = Math::Rand(aSeed) % KTestDbCnt;
   230 
   231 	aThreadName = currThread.Name();
   232 	RDebug::Print(_L("=== Thread %S, database %S\r\n"), &aThreadName, &KTestDbNames[KDbIndex]);
   233 
   234 	TInt err = aDb.Open(KTestDbNames[KDbIndex]);
   235 	TTEST2(err, KErrNone);
   236 	}
   237 
   238 //Creates N statements, where 0 < N < KStatementCnt
   239 TInt CreateStatements(RSqlDatabase& aDb, TInt64& aSeed, RSqlStatementArray& aStmtArray)
   240 	{
   241 	TInt stmtCount = Math::Rand(aSeed) % KStatementCnt;
   242 	if(stmtCount == 0)
   243 		{
   244 		stmtCount = 1;
   245 		}
   246 	for(TInt i=0;i<stmtCount;++i)
   247 		{
   248 		TSqlStatement stmt;
   249 		stmt.iAlive = EFalse;
   250 		stmt.iCount = KQueriedRecordCnt;
   251 		stmt.iCurIndex = Math::Rand(aSeed) % KRecordCnt;
   252 		if(stmt.iCurIndex == 0)
   253 			{
   254 			stmt.iCurIndex = 1;
   255 			}
   256 		if(stmt.iCurIndex > (KRecordCnt - KQueriedRecordCnt))
   257 			{
   258 			stmt.iCurIndex = KRecordCnt - KQueriedRecordCnt;
   259 			}
   260 		stmt.iEndIndex = stmt.iCurIndex + KQueriedRecordCnt;
   261 		TBuf8<100> sql;
   262 		sql.Copy(_L8("SELECT * FROM A WHERE F1 >= "));
   263 		sql.AppendNum(stmt.iCurIndex);
   264 		sql.Append(_L8(" AND F1 < "));
   265 		sql.AppendNum(stmt.iEndIndex);
   266 		TInt err = stmt.iObj.Prepare(aDb, sql);
   267 		TTEST2(err, KErrNone);
   268 		stmt.iAlive = ETrue;
   269 		err = aStmtArray.Append(stmt);
   270 		TTEST2(err, KErrNone);
   271 		}
   272 	return stmtCount;		
   273 	}
   274 	
   275 //For each alive statement object - do (TSqlStatement::iCount / KTestStepCnt)
   276 //RSqlStatement::Next() calls. If the Next() call reaches the end - close the statement object.
   277 TInt ProcessStatements(RSqlStatementArray& aStmtArray)
   278 	{
   279 	const TInt KTotalStmtCount = aStmtArray.Count();
   280 	TInt alive = 0;
   281 	TInt completed = 0;
   282 	for(TInt k=0;k<KTotalStmtCount;++k)
   283 		{
   284 		TSqlStatement& stmt = aStmtArray[k];
   285 		if(stmt.iAlive)
   286 			{
   287 			++alive;
   288 			TInt endIndex = stmt.iCurIndex + stmt.iCount / KTestStepCnt;
   289 			if(endIndex <= stmt.iEndIndex)
   290 				{
   291 				while(stmt.iCurIndex < endIndex)
   292 					{
   293 					TInt err = stmt.iObj.Next();
   294 					TTEST2(err, KSqlAtRow);
   295 					//test column values
   296 					TInt val1 = stmt.iObj.ColumnInt(0);
   297 					TTEST(val1 == stmt.iCurIndex);
   298 					RSqlColumnReadStream strm;
   299 					err = strm.ColumnBinary(stmt.iObj, 1);
   300 					TTEST2(err, KErrNone);
   301 					for(TInt ii=0;ii<KBinDataLen;++ii)
   302 						{
   303 						TUint8 byte = 0;
   304 						TRAP(err, byte = strm.ReadUint8L());
   305 						TTEST2(err, KErrNone);
   306 						TTEST(byte == (TUint8)val1);
   307 						}
   308 					strm.Close();
   309 					++stmt.iCurIndex;
   310 					}
   311 				}
   312 			if(stmt.iCurIndex >= stmt.iEndIndex)
   313 				{
   314 				stmt.iObj.Close();
   315 				stmt.iAlive = EFalse;
   316 				++completed;
   317 				}
   318 			}
   319 		}
   320 	return completed;
   321 	}
   322 
   323 //Close up to N statements, where 0 < N < KStatementCnt
   324 TInt CloseStatements(RSqlStatementArray& aStmtArray, TInt64& aSeed)
   325 	{
   326 	TInt stmtCount = Math::Rand(aSeed) % KStatementCnt;
   327 	if(stmtCount == 0)
   328 		{
   329 		stmtCount = 1;
   330 		}
   331 	const TInt KTotalStmtCount = aStmtArray.Count();
   332 	TInt closed = 0;
   333 	for(TInt j=0;j<stmtCount;++j)
   334 		{
   335 		const TInt KIdx = Math::Rand(aSeed) % KTotalStmtCount;
   336 		TInt idx = KIdx;
   337 		while((idx = (++idx % KTotalStmtCount)) != KIdx)
   338 			{
   339 			if(aStmtArray[idx].iAlive)
   340 				{
   341 				aStmtArray[idx].iObj.Close();
   342 				aStmtArray[idx].iAlive = EFalse;
   343 				++closed;
   344 				break;
   345 				}
   346 			}
   347 		}
   348 	return closed;
   349 	}
   350 
   351 //Counts the alive statements
   352 TInt AliveStatementsCount(RSqlStatementArray& aStmtArray)
   353 	{
   354 	TInt aliveCnt = 0;
   355 	const TInt KTotalStmtCount = aStmtArray.Count();
   356 	for(TInt l=0;l<KTotalStmtCount;++l)
   357 		{
   358 		if(aStmtArray[l].iAlive)
   359 			{
   360 			++aliveCnt;
   361 			}
   362 		}
   363 	return aliveCnt;
   364 	}
   365 	
   366 //Close all alive statements
   367 void CloseAllStatements(RSqlStatementArray& aStmtArray)
   368 	{
   369 	const TInt KTotalStmtCount = aStmtArray.Count();
   370 	for(TInt i=0;i<KTotalStmtCount;++i)
   371 		{
   372 		if(aStmtArray[i].iAlive)
   373 			{
   374 			aStmtArray[i].iObj.Close();
   375 			}
   376 		}
   377 	TTEST2(TSqlResourceTester::Count(), 0);
   378 	}
   379 
   380 //Removes the already closed statements and compresses the array
   381 void RemoveDeadStatements(RSqlStatementArray& aStmtArray)
   382 	{
   383 	for(TInt i=aStmtArray.Count()-1;i>=0;--i)
   384 		{
   385 		if(!aStmtArray[i].iAlive)
   386 			{
   387 			aStmtArray.Remove(i);
   388 			}
   389 		}
   390 	aStmtArray.Compress();		
   391 	}
   392 
   393 //Close statement objects, statements array and the database object
   394 TInt PostTest(RSqlDatabase& aDb, RSqlStatementArray& aStmtArray)
   395 	{
   396 	TInt statementsAlive = AliveStatementsCount(aStmtArray);
   397 	CloseAllStatements(aStmtArray);
   398 	aStmtArray.Close();
   399 	aDb.Close();
   400 	return statementsAlive;
   401 	}
   402 
   403 //Test thread function
   404 //The thread function works with a set of TSqlStatement objects
   405 //The test consists of 4 steps:
   406 //Step 1: the test thread creates m TSqlStatement objects, 0 < m < KStatementCnt.
   407 //		  With each of the created TSqlStatement objects the test thread prepares SELECT SQL query
   408 //        "SELECT * FROM A WHERE F1 >= K1 AND F1 < K2", where K1 is random generated number, such that:
   409 //        0 < K1 < (KRecordCnt - KQueriedRecordCnt)
   410 //        K2 = K1 + KQueriedRecordCnt
   411 //		  All just created TSqlStatement objects are marked as alive.
   412 //Step 2: For each alive TSqlStatement object the test thread calls iObj.Next() method KTestStepCnt times,
   413 //        KTestStepCnt < KQueriedRecordCnt.
   414 //        The column values are retrieved and checked.
   415 //Step 3: the test thread closes n TSqlStatement objects, 0 < n < KStatementCnt.
   416 //Step 4: the test thread counts how many alive TSqlStatement objects are there.
   417 //        If this count > KMaxStatementPerThread then the test thread closes all alive TSqlStatement objects
   418 //		  to avoid OOM errors during the test.
   419 //
   420 //		  Each test thread does steps 1..4 for a period of KTestDuration seconds.
   421 //		  At the end all TSqlStatement objects are closed.
   422 //
   423 //		  The idea of the test is to load the SQL server creating several amount of statement and stream objects
   424 //		  and see that it is working stable and without problems.
   425 TInt ThreadFunc(void*)
   426 	{
   427 	__UHEAP_MARK;
   428 
   429 	CTrapCleanup* tc = CTrapCleanup::New();
   430 	TTEST(tc != NULL);
   431 
   432 	TInt64 seed = 0;
   433 	RSqlDatabase db;
   434 	TName threadName;
   435 	RSqlStatementArray statements;
   436 	
   437 	//Init the random numbers generator, opens the database
   438 	PreTest(db, seed, threadName);
   439 
   440 	//Main test loop
   441 	TInt iteration = 0;
   442 	TTime currTime;
   443 	currTime.UniversalTime();
   444 	TTime endTime = currTime + TTimeIntervalSeconds(KTestDuration);
   445 	while(currTime < endTime)
   446 		{
   447 		++iteration;
   448 		///////////////////////////////////////////////////////////////////////
   449 		TInt statementsAliveBegin = statements.Count();
   450 		//Step 1: Create N statements, where 0 < N < KStatementCnt
   451 		TInt statementsCreated = CreateStatements(db, seed, statements);
   452 		///////////////////////////////////////////////////////////////////////
   453 		//Step 2: For each alive statement object - do (TSqlStatement::iCount / KTestStepCnt)
   454 		//        RSqlStatement::Next() calls. If the Next() call reaches the end - close the statement object.
   455 		TInt statementsCompleted = ProcessStatements(statements);
   456 		///////////////////////////////////////////////////////////////////////
   457 		//Step 3: Close up to N statements, where 0 < N < KStatementCnt
   458 		TInt statementsClosed = CloseStatements(statements, seed);
   459 		///////////////////////////////////////////////////////////////////////
   460 		//Step 4: If the alive statement count is more than KMaxStatementPerThread, then close them all
   461 		TInt statementsAliveEnd = AliveStatementsCount(statements);
   462 		if(statementsAliveEnd > KMaxStatementPerThread)
   463 			{
   464 			RDebug::Print(_L("!!! Thread %S, iteration %d, alive %d, close all\r\n"), &threadName, iteration, statementsAliveEnd);
   465 			CloseAllStatements(statements);
   466 			statementsAliveEnd = 0;
   467 			}
   468 		///////////////////////////////////////////////////////////////////////
   469 		RemoveDeadStatements(statements);
   470 		RDebug::Print(_L("=== Thread %S, iteration % 4d, begin: % 3d, created % 2d, closed % 2d, completed % 2d, end % 3d, \r\n"),
   471 							&threadName, iteration, statementsAliveBegin, 
   472 													statementsCreated, statementsClosed, statementsCompleted, 
   473 													statementsAliveEnd);
   474 		currTime.UniversalTime();
   475 		}
   476 
   477 	//Close statement objects and the database object
   478 	TInt statementsAlive = PostTest(db, statements);
   479 
   480 	delete tc;
   481 
   482 	__UHEAP_MARKEND;
   483 
   484 	RDebug::Print(_L("=== Thread %S exit, still alive %d\r\n"), &threadName, statementsAlive);
   485 
   486 	return KErrNone;
   487 	}
   488 
   489 void CreateTestThreads(RThread aThreads[], TRequestStatus aStatuses[], TInt aMaxCount)
   490 	{
   491 	_LIT(KThreadName, "TstThr");
   492 	for(TInt i=0;i<aMaxCount;++i)
   493 		{
   494 		TBuf<20> threadName(KThreadName);
   495 		threadName.AppendNum((TInt64)(i + 1));
   496 		TEST2(aThreads[i].Create(threadName, &ThreadFunc, 0x2000, 0x1000, 0x10000, NULL, EOwnerProcess), KErrNone);
   497 		aThreads[i].Logon(aStatuses[i]);
   498 		TEST2(aStatuses[i].Int(), KRequestPending);
   499 		}
   500 	}
   501 
   502 void ResumeTestThreads(RThread aThreads[], TInt aMaxCount)
   503 	{
   504 	for(TInt i=0;i<aMaxCount;++i)
   505 		{
   506 		aThreads[i].Resume();
   507 		}
   508 	}
   509 
   510 
   511 void CloseTestThreads(RThread aThreads[], TRequestStatus aStatuses[], TInt aMaxCount)
   512 	{
   513 	for(TInt i=0;i<aMaxCount;++i)
   514 		{
   515 		User::WaitForRequest(aStatuses[i]);
   516 		TEST(aThreads[i].ExitType() != EExitPanic);
   517 		aThreads[i].Close();
   518 		}
   519 	}
   520 
   521 /**
   522 @SYMTestCaseID			SYSLIB-SQL-CT-1627-0001
   523 @SYMTestCaseDesc		SQL server load test. The test creates KTestThreadCnt threads, KTestDbCnt test databases and
   524 						inserts in each of them KRecordCnt test records.
   525 						Pre-test step: each test thread randomly chooses and opens one of the test databases.
   526 						Then, each of the test threads is doing the following 4 test steps:
   527 						Step 1: the test thread creates m TSqlStatement objects, 0 < m < KStatementCnt.
   528 						With each of the created TSqlStatement objects the test thread prepares SELECT SQL query
   529 						"SELECT * FROM A WHERE F1 >= K1 AND F1 < K2", where K1 is random generated number, such that:
   530 						0 < K1 < (KRecordCnt - KQueriedRecordCnt)
   531 						K2 = K1 + KQueriedRecordCnt
   532 						All just created TSqlStatement objects are marked as alive.
   533 						Step 2: For each alive TSqlStatement object the test thread calls iObj.Next() method KTestStepCnt times,
   534 						KTestStepCnt < KQueriedRecordCnt.
   535 						The column values are retrieved and checked.
   536 						Step 3: the test thread closes n TSqlStatement objects, 0 < n < KStatementCnt.
   537 						Step 4: the test thread counts how many alive TSqlStatement objects are there.
   538 						If this count > KMaxStatementPerThread then the test thread closes all alive TSqlStatement objects
   539 						to avoid OOM errors during the test.
   540 
   541 						Each test thread does steps 1..4 for a period of KTestDuration seconds.
   542 						At the end all TSqlStatement objects are closed.
   543 
   544 						The idea of the test is to load the SQL server creating several amount of statement and stream objects
   545 						and see that it is working stable and without problems.
   546 @SYMTestPriority		High
   547 @SYMTestActions			SQL server load test
   548 @SYMTestExpectedResults Test must not fail
   549 @SYMREQ					REQ5792
   550                         REQ5793
   551 */
   552 void SqlLoadTest()
   553 	{
   554 	CreateTestDatabases();
   555 
   556 	RThread threads[KTestThreadCnt];
   557 	TRequestStatus statuses[KTestThreadCnt];
   558 
   559 	CreateTestThreads(threads, statuses, KTestThreadCnt);
   560 
   561 	ResumeTestThreads(threads, KTestThreadCnt);
   562 
   563 	User::After(2000000);
   564 
   565 	CloseTestThreads(threads, statuses, KTestThreadCnt);
   566 	}
   567 
   568 /**
   569 @SYMTestCaseID          PDS-SQL-CT-4201
   570 @SYMTestCaseDesc        Max number of SQL statements test.
   571 @SYMTestPriority        High
   572 @SYMTestActions         The test creates a table with couple of records and then
   573 						creates as many as possible SQL statements. The expected result is
   574 						that either the statement creation process will fail with KErrNoMemory or
   575 						the max number of statements to be created is reached (100000).
   576 						Then the test deletes 1/2 of the created statements objects and
   577 						after that attempts to execute Next() on the rest of them.
   578 						Note that the test has a time limit of 120 seconds. Otherwise on some platforms
   579 						with WDP feature switched on the test may timeout.
   580 @SYMTestExpectedResults Test must not fail
   581 @SYMDEF                 DEF145236
   582 */  
   583 void StatementMaxNumberTest()
   584 	{
   585 	TBuf<30> time;
   586 	GetHomeTimeAsString(time);
   587 	TheTest.Printf(_L("=== %S: Create database\r\n"), &time);
   588 	
   589 	(void)RSqlDatabase::Delete(KTestDbName1);
   590 	RSqlDatabase db;
   591 	TInt err = db.Create(KTestDbName1);
   592 	TEST2(err, KErrNone);
   593 	err = db.Exec(_L("CREATE TABLE A(I INTEGER); INSERT INTO A(I) VALUES(1); INSERT INTO A(I) VALUES(2);"));
   594 	TEST(err >= 0);
   595 
   596 	GetHomeTimeAsString(time);
   597 	TheTest.Printf(_L("=== %S: Create statements array\r\n"), &time);
   598 	
   599 	//Reserve memory for the statement objects
   600 	const TInt KMaxStmtCount = 100000;
   601 	RSqlStatement* stmt = new RSqlStatement[KMaxStmtCount];
   602 	TEST(stmt != NULL);
   603 
   604 	TTime startTime;
   605 	startTime.HomeTime();
   606 	
   607 	//Create as many statement objects as possible
   608 	TInt idx = 0;
   609 	err = KErrNone;
   610 	for(;idx<KMaxStmtCount;++idx)
   611 		{
   612 		err = stmt[idx].Prepare(db, _L("SELECT * FROM A WHERE I>=0 AND I<10"));
   613 		if(err != KErrNone)
   614 			{
   615 			break;
   616 			}
   617 		TTimeIntervalSeconds s = ExecutionTimeSeconds(startTime);
   618 		if((idx % 100) == 0)
   619 			{
   620 			GetHomeTimeAsString(time);
   621 			TheTest.Printf(_L("=== %S: Create % 5d statements. %d seconds.\r\n"), &time, idx + 1, s.Int());
   622 			}
   623 		if(s.Int() > KTestTimeLimit)
   624 			{
   625 			GetHomeTimeAsString(time);
   626 			TheTest.Printf(_L("=== %S: The time limit reached.\r\n"), &time);
   627 			++idx;//The idx-th statement is valid, the statement count is idx + 1.
   628 			break;
   629 			}
   630 		}
   631 	
   632 	TInt stmtCnt = idx;
   633 	TheTest.Printf(_L("%d created statement objects. Last error: %d.\r\n"), stmtCnt, err);
   634 	TEST(err == KErrNone || err == KErrNoMemory);
   635 
   636 	//Close 1/2 of the statements to free some memory
   637 	idx = 0;
   638 	for(;idx<(stmtCnt/2);++idx)
   639 		{
   640 		stmt[idx].Close();
   641 		if((idx % 100) == 0)
   642 			{
   643 			GetHomeTimeAsString(time);
   644 			TheTest.Printf(_L("=== %S: % 5d statements closed\r\n"), &time, idx + 1);
   645 			}
   646 		}
   647 	
   648 	//Now, there should be enough memory to be able to execute Next() on the rest of the statements
   649 	for(TInt j=0;idx<stmtCnt;++idx,++j)
   650 		{
   651 		err = stmt[idx].Next();
   652 		TEST2(err, KSqlAtRow);
   653 		err = stmt[idx].Next();
   654 		TEST2(err, KSqlAtRow);
   655 		err = stmt[idx].Next();
   656 		TEST2(err, KSqlAtEnd);
   657 		GetHomeTimeAsString(time);
   658 		TTimeIntervalSeconds s = ExecutionTimeSeconds(startTime);
   659 		if((j % 100) == 0)
   660 			{
   661 			TheTest.Printf(_L("=== %S: % 5d statements processed. %d seconds.\r\n"), &time, j + 1, s.Int());
   662 			}
   663 		if(s.Int() > KTestTimeLimit)
   664 			{
   665 			TheTest.Printf(_L("=== %S: The time limit reached.\r\n"), &time);
   666 			break;
   667 			}
   668 		}
   669 
   670 	//Cleanup
   671 	for(idx=0;idx<stmtCnt;++idx)
   672 		{
   673 		stmt[idx].Close();
   674 		if((idx % 100) == 0)
   675 			{
   676 			GetHomeTimeAsString(time);
   677 			TheTest.Printf(_L("=== %S: % 5d statements closed\r\n"), &time, idx + 1);
   678 			}
   679 		}
   680 	delete [] stmt;
   681 	db.Close();
   682 	(void)RSqlDatabase::Delete(KTestDbName1);
   683 	GetHomeTimeAsString(time);
   684 	TheTest.Printf(_L("=== %S: Test case end\r\n"), &time);
   685 	}
   686 
   687 TInt CreateFileSessions(TInt& aIdx, RFs aFs[], TInt aMaxFsSessCount)
   688 	{
   689 	TBuf<30> time;
   690 	TTime startTime;
   691 	startTime.HomeTime();
   692 	//Create as many file session objects as possible
   693 	TInt err = KErrNone;
   694 	for(;aIdx<aMaxFsSessCount;++aIdx)
   695 		{
   696 		err = aFs[aIdx].Connect();
   697 		if(err != KErrNone)
   698 			{
   699 			break;
   700 			}
   701 		TTimeIntervalSeconds s = ExecutionTimeSeconds(startTime);
   702 		if((aIdx % 500) == 0)
   703 			{
   704 			GetHomeTimeAsString(time);
   705 			TheTest.Printf(_L("=== %S: Create % 5d file sessions. %d seconds.\r\n"), &time, aIdx + 1, s.Int());
   706 			}
   707 		if(s.Int() > KTestTimeLimit)
   708 			{
   709 			GetHomeTimeAsString(time);
   710 			TheTest.Printf(_L("=== %S: The time limit reached.\r\n"), &time);
   711 			++aIdx;//The idx-th file session object is valid, the file session count is idx + 1.
   712 			break;
   713 			}
   714 		}
   715 	return err;
   716 	}
   717 
   718 /**
   719 @SYMTestCaseID          PDS-SQL-CT-4237
   720 @SYMTestCaseDesc        Max file session number test.
   721 @SYMTestPriority        High
   722 @SYMTestActions         The test creates as many as possible file session objects. The expected result is
   723 						that either the file session creation process will fail with KErrNoMemory or
   724 						the max number of file sessions to be created is reached (100000).
   725 						Then the test attempts to create a database. If there is no memory, the test
   726 						closes some of the file session objects. The test also attempts to copy
   727 						the created database and to delete it after that, both operations performed
   728 						with all file session objects still open. The expectation is that the test
   729 						will not crash the SQL server or the client side SQL dll.
   730 						Note that the test has a time limit of 120 seconds. Otherwise on some platforms
   731 						with WDP feature switched on the test may timeout.
   732 @SYMTestExpectedResults Test must not fail
   733 */  
   734 void FileSessionMaxNumberTest()
   735 	{
   736 	TBuf<30> time;
   737 	GetHomeTimeAsString(time);
   738 	TheTest.Printf(_L("=== %S: Create file sessions\r\n"), &time);
   739 
   740 	const TInt KMaxFsCount = 100000;
   741 	RFs* fs = new RFs[KMaxFsCount];
   742 	TEST(fs != NULL);
   743 
   744 	//Create as many file session objects as possible
   745 	TInt idx = 0;
   746 	TInt err = CreateFileSessions(idx, fs, KMaxFsCount);
   747 	TheTest.Printf(_L("%d created file session objects. Last error: %d.\r\n"), idx, err);
   748 	TEST(err == KErrNone || err == KErrNoMemory);
   749 	
   750 	TBool dbCreated = EFalse;
   751 	RSqlDatabase db;
   752 	
   753 	//An attempt to create a database
   754 	while(idx > 0 && err == KErrNoMemory)
   755 		{
   756 		(void)RSqlDatabase::Delete(KTestDbName1);
   757 		err = db.Create(KTestDbName1);
   758 		if(err == KErrNone)
   759 			{
   760 			err = db.Exec(_L("CREATE TABLE A(I INTEGER); INSERT INTO A(I) VALUES(1); INSERT INTO A(I) VALUES(2);"));
   761 			}
   762 		TheTest.Printf(_L("Database creation. Last error: %d.\r\n"), err);
   763 		TEST(err == KErrNoMemory || err >= 0);
   764 		if(err == KErrNoMemory)
   765 			{
   766 			fs[--idx].Close();
   767 			}
   768 		else
   769 			{
   770 			dbCreated = ETrue;
   771 			}
   772 		db.Close();
   773 		}
   774 
   775 	if(dbCreated)
   776 		{
   777 		//Create again file session objects - as many as possible
   778 		err = CreateFileSessions(idx, fs, KMaxFsCount);
   779 		TEST(err == KErrNone || err == KErrNoMemory);
   780 		//Try to copy the database
   781 		err = RSqlDatabase::Copy(KTestDbName1, KTestDbName4);
   782 		TheTest.Printf(_L("Copy database. Last error: %d.\r\n"), err);
   783 		TEST(err == KErrNone || err == KErrNoMemory);
   784 		//Try to delete the databases
   785 		if(err == KErrNone)
   786 			{
   787 			err = RSqlDatabase::Delete(KTestDbName4);
   788 			TheTest.Printf(_L("Delete database copy. Last error: %d.\r\n"), err);
   789 			TEST(err == KErrNone || err == KErrNoMemory);
   790 			}
   791 		err = RSqlDatabase::Delete(KTestDbName1);
   792 		TheTest.Printf(_L("Delete database. Last error: %d.\r\n"), err);
   793 		TEST(err == KErrNone || err == KErrNoMemory);
   794 		}
   795 	
   796 	//Cleanup
   797 	for(TInt i=0;i<idx;++i)
   798 		{
   799 		fs[i].Close();
   800 		if((i % 500) == 0)
   801 			{
   802 			GetHomeTimeAsString(time);
   803 			TheTest.Printf(_L("=== %S: % 5d file sessions closed\r\n"), &time, i + 1);
   804 			}
   805 		}
   806 	delete [] fs;
   807 	err = RSqlDatabase::Delete(KTestDbName4);
   808 	TEST(err == KErrNone || err == KErrNotFound);
   809 	err = RSqlDatabase::Delete(KTestDbName1);
   810 	TEST(err == KErrNone || err == KErrNotFound);
   811 	}
   812 
   813 TInt CreateSqlConnections(TInt& aIdx, RSqlDatabase aDb[], TInt aMaxSqlConnCount)
   814 	{
   815 	TBuf<30> time;
   816 	TTime startTime;
   817 	startTime.HomeTime();
   818 	//Create as many file session objects as possible
   819 	TInt err = KErrNone;
   820 	for(;aIdx<aMaxSqlConnCount;++aIdx)
   821 		{
   822 		err = aDb[aIdx].Open(KTestDbName1);
   823 		if(err != KErrNone)
   824 			{
   825 			break;
   826 			}
   827 		TTimeIntervalSeconds s = ExecutionTimeSeconds(startTime);
   828 		if((aIdx % 100) == 0)
   829 			{
   830 			GetHomeTimeAsString(time);
   831 			TheTest.Printf(_L("=== %S: Create % 5d sql connections. %d seconds.\r\n"), &time, aIdx + 1, s.Int());
   832 			}
   833 		if(s.Int() > KTestTimeLimit)
   834 			{
   835 			GetHomeTimeAsString(time);
   836 			TheTest.Printf(_L("=== %S: The time limit reached.\r\n"), &time);
   837 			++aIdx;//The idx-th sql connection is valid, the sql connection count is idx + 1.
   838 			break;
   839 			}
   840 		}
   841 	return err;
   842 	}
   843 
   844 /**
   845 @SYMTestCaseID          PDS-SQL-CT-4238
   846 @SYMTestCaseDesc        Max sql connection number test.
   847 @SYMTestPriority        High
   848 @SYMTestActions         The test creates as many as possible sql connection objects. The expected result is
   849 						that either the sql connection creation process will fail with KErrNoMemory or
   850 						the max number of sql connection to be created is reached (100000).
   851 						Then the test attempts to create a database. If there is no memory, the test
   852 						closes some of the sql connection objects. The test also attempts to copy
   853 						the created database and to delete it after that, both operations performed
   854 						with all sql connection objects still open. The expectation is that the test
   855 						will not crash the SQL server or the client side SQL dll.
   856 						Note that the test has a time limit of 120 seconds. Otherwise on some platforms
   857 						with WDP feature switched on the test may timeout.
   858 @SYMTestExpectedResults Test must not fail
   859 */  
   860 void SqlConnectionMaxNumberTest()
   861 	{
   862 	TBuf<30> time;
   863 	GetHomeTimeAsString(time);
   864 	TheTest.Printf(_L("=== %S: Create sql connections\r\n"), &time);
   865 
   866 	(void)RSqlDatabase::Delete(KTestDbName1);
   867 	RSqlDatabase db1;
   868 	TInt err = db1.Create(KTestDbName1);//CreateSqlConnections() opens the already existing KTestDbName1 database
   869 	TEST2(err, KErrNone);
   870 	
   871 	const TInt KMaxConnCount = 100000;
   872 	RSqlDatabase* db = new RSqlDatabase[KMaxConnCount];
   873 	TEST(db != NULL);
   874 
   875 	//Create as many sql connection objects as possible
   876 	TInt idx = 0;
   877 	err = CreateSqlConnections(idx, db, KMaxConnCount);
   878 	TheTest.Printf(_L("%d created sql connection objects. Last error: %d.\r\n"), idx, err);
   879 	TEST(err == KErrNone || err == KErrNoMemory);
   880 	
   881 	TBool dbCreated = EFalse;
   882 	RSqlDatabase db2;
   883 	
   884 	//An attempt to create a database
   885 	while(idx > 0 && err == KErrNoMemory)
   886 		{
   887 		(void)RSqlDatabase::Delete(KTestDbName4);
   888 		err = db2.Create(KTestDbName4);
   889 		if(err == KErrNone)
   890 			{
   891 			err = db2.Exec(_L("CREATE TABLE A(I INTEGER); INSERT INTO A(I) VALUES(1); INSERT INTO A(I) VALUES(2);"));
   892 			}
   893 		TheTest.Printf(_L("Database creation. Last error: %d.\r\n"), err);
   894 		TEST(err == KErrNoMemory || err >= 0);
   895 		if(err == KErrNoMemory)
   896 			{
   897 			db[--idx].Close();
   898 			}
   899 		else
   900 			{
   901 			dbCreated = ETrue;
   902 			}
   903 		db2.Close();
   904 		}
   905 	
   906 	if(dbCreated)
   907 		{
   908 		//Create again sql connection objects - as many as possible
   909 		err = CreateSqlConnections(idx, db, KMaxConnCount);
   910 		TEST(err == KErrNone || err == KErrNoMemory);
   911 		//Try to copy the database
   912 		err = RSqlDatabase::Copy(KTestDbName4, KTestDbName5);
   913 		TheTest.Printf(_L("Copy database. Last error: %d.\r\n"), err);
   914 		TEST(err == KErrNone || err == KErrNoMemory);
   915 		//Try to delete the databases
   916 		if(err == KErrNone)
   917 			{
   918 			err = RSqlDatabase::Delete(KTestDbName5);
   919 			TheTest.Printf(_L("Delete database copy. Last error: %d.\r\n"), err);
   920 			TEST(err == KErrNone || err == KErrNoMemory);
   921 			}
   922 		err = RSqlDatabase::Delete(KTestDbName4);
   923 		TheTest.Printf(_L("Delete database. Last error: %d.\r\n"), err);
   924 		TEST(err == KErrNone || err == KErrNoMemory);
   925 		}
   926 	
   927 	//Cleanup
   928 	for(TInt i=0;i<idx;++i)
   929 		{
   930 		db[i].Close();
   931 		if((i % 100) == 0)
   932 			{
   933 			GetHomeTimeAsString(time);
   934 			TheTest.Printf(_L("=== %S: % 5d sql connections closed\r\n"), &time, i + 1);
   935 			}
   936 		}
   937 	delete [] db;
   938 	db1.Close();
   939 	err = RSqlDatabase::Delete(KTestDbName5);
   940 	TEST(err == KErrNone || err == KErrNotFound);
   941 	err = RSqlDatabase::Delete(KTestDbName4);
   942 	TEST(err == KErrNone || err == KErrNotFound);
   943 	err = RSqlDatabase::Delete(KTestDbName1);
   944 	TEST(err == KErrNone || err == KErrNotFound);
   945 	}
   946 
   947 void DoTests()
   948 	{
   949 	TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1627-0001 SQL server load test "));
   950 	SqlLoadTest();
   951 	TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-CT-4201 Statement max number test"));
   952 	StatementMaxNumberTest();
   953 #if defined __WINS__ ||	defined __WINSCW__
   954 	//The next two tests are likely to timeout on hardware because they create a lot of file sessions and sql connections.
   955 	//The SQL server heap is 32Mb on hardware but only 6Mb on the Emulator. 
   956 	TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-CT-4237 File session max number test"));
   957 	FileSessionMaxNumberTest();
   958 	TheTest.Next(_L(" @SYMTestCaseID:PDS-SQL-CT-4238 Sql connection max number test"));
   959 	SqlConnectionMaxNumberTest();
   960 #endif	
   961 	}
   962 
   963 TInt E32Main()
   964 	{
   965 	TheTest.Title();
   966 
   967 	CTrapCleanup* tc = CTrapCleanup::New();
   968 
   969 	__UHEAP_MARK;
   970 
   971 	CreateTestDir();
   972 	DeleteTestFiles();
   973 	DoTests();
   974 	DeleteTestFiles();
   975 
   976 	__UHEAP_MARKEND;
   977 
   978 	TheTest.End();
   979 	TheTest.Close();
   980 
   981 	delete tc;
   982 
   983 	User::Heap().Check();
   984 	return KErrNone;
   985 	}