os/persistentdata/persistentstorage/sql/TEST/t_sqlmulti.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 2006-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
//
sl@0
    15
sl@0
    16
#include <e32test.h>
sl@0
    17
#include <bautils.h>
sl@0
    18
#include <sqldb.h>
sl@0
    19
sl@0
    20
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    21
sl@0
    22
RTest TheTest(_L("t_sqlmulti test"));
sl@0
    23
sl@0
    24
_LIT(KTestDir, "c:\\test\\");
sl@0
    25
_LIT(KTestDbName1, "c:\\test\\t_sqlmulti.db");
sl@0
    26
sl@0
    27
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    28
sl@0
    29
void DeleteTestFiles()
sl@0
    30
	{
sl@0
    31
	RSqlDatabase::Delete(KTestDbName1);
sl@0
    32
	}
sl@0
    33
sl@0
    34
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    35
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    36
//Test macros and functions
sl@0
    37
void Check1(TInt aValue, TInt aLine, TBool aPrintThreadName = EFalse)
sl@0
    38
	{
sl@0
    39
	if(!aValue)
sl@0
    40
		{
sl@0
    41
		DeleteTestFiles();
sl@0
    42
		if(aPrintThreadName)
sl@0
    43
			{
sl@0
    44
			RThread th;
sl@0
    45
			TName name = th.Name();
sl@0
    46
			RDebug::Print(_L("*** Thread %S, Line %d\r\n"), &name, aLine);
sl@0
    47
			}
sl@0
    48
		else
sl@0
    49
			{
sl@0
    50
			RDebug::Print(_L("*** Line %d\r\n"), aLine);
sl@0
    51
			}
sl@0
    52
		TheTest(EFalse, aLine);
sl@0
    53
		}
sl@0
    54
	}
sl@0
    55
void Check2(TInt aValue, TInt aExpected, TInt aLine, TBool aPrintThreadName = EFalse)
sl@0
    56
	{
sl@0
    57
	if(aValue != aExpected)
sl@0
    58
		{
sl@0
    59
		DeleteTestFiles();
sl@0
    60
		if(aPrintThreadName)
sl@0
    61
			{
sl@0
    62
			RThread th;
sl@0
    63
			TName name = th.Name();
sl@0
    64
			RDebug::Print(_L("*** Thread %S, Line %d Expected error: %d, got: %d\r\n"), &name, aLine, aExpected, aValue);
sl@0
    65
			}
sl@0
    66
		else
sl@0
    67
			{
sl@0
    68
			RDebug::Print(_L("*** Line %d, Expected error: %d, got: %d\r\n"), aLine, aExpected, aValue);
sl@0
    69
			}
sl@0
    70
		TheTest(EFalse, aLine);
sl@0
    71
		}
sl@0
    72
	}
sl@0
    73
#define TEST(arg) ::Check1((arg), __LINE__)
sl@0
    74
#define TEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__)
sl@0
    75
#define TTEST(arg) ::Check1((arg), __LINE__, ETrue)
sl@0
    76
#define TTEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__, ETrue)
sl@0
    77
sl@0
    78
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    79
sl@0
    80
void CreateTestDir()
sl@0
    81
    {
sl@0
    82
    RFs fs;
sl@0
    83
	TInt err = fs.Connect();
sl@0
    84
	TEST2(err, KErrNone);
sl@0
    85
sl@0
    86
	err = fs.MkDir(KTestDir);
sl@0
    87
	TEST(err == KErrNone || err == KErrAlreadyExists);
sl@0
    88
	
sl@0
    89
	fs.Close();
sl@0
    90
	}
sl@0
    91
sl@0
    92
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    93
sl@0
    94
/**
sl@0
    95
@SYMTestCaseID			SYSLIB-SQL-CT-1612
sl@0
    96
@SYMTestCaseDesc		Two connections to the same database in the same thread. Create a test database
sl@0
    97
						and insert some records from both connections. Verify that all records were inserted
sl@0
    98
						successfully.
sl@0
    99
@SYMTestPriority		High
sl@0
   100
@SYMTestActions			Testing SQL engine behaviour when having mutiple connections to the same database
sl@0
   101
						in the same thread.
sl@0
   102
@SYMTestExpectedResults Test must not fail
sl@0
   103
@SYMREQ					REQ5792
sl@0
   104
                        REQ5793
sl@0
   105
*/	
sl@0
   106
void TestMultiConnSameThread()
sl@0
   107
	{
sl@0
   108
	//Connection 1
sl@0
   109
	RSqlDatabase db1;
sl@0
   110
	TInt err = db1.Create(KTestDbName1);
sl@0
   111
	TEST2(err, KErrNone);
sl@0
   112
sl@0
   113
	//Create test database
sl@0
   114
	RDebug::Print(_L("###Create test database\r\n"));
sl@0
   115
	_LIT8(KCreateSql, "CREATE TABLE A(Id INTEGER PRIMARY KEY AUTOINCREMENT, Data INTEGER)");
sl@0
   116
	err = db1.Exec(KCreateSql);
sl@0
   117
	TEST(err >= 0);
sl@0
   118
	
sl@0
   119
	//Connection 2
sl@0
   120
	RSqlDatabase db2;
sl@0
   121
	err = db2.Open(KTestDbName1);
sl@0
   122
	TEST2(err, KErrNone);
sl@0
   123
	
sl@0
   124
	//Insert some records using both connections
sl@0
   125
	RDebug::Print(_L("###Insert some records\r\n"));
sl@0
   126
	const TInt KRecNum = 100;
sl@0
   127
	_LIT8(KInsertSql, "INSERT INTO A(Data) VALUES(");
sl@0
   128
	for(TInt i=0;i<KRecNum;++i)
sl@0
   129
		{
sl@0
   130
		TBuf8<100> sql(KInsertSql);
sl@0
   131
		sql.AppendNum((TInt64)i + 1);
sl@0
   132
		sql.Append(_L(");"));
sl@0
   133
		err = (i%2) ? db1.Exec(sql) : db2.Exec(sql);
sl@0
   134
		if(err < 0)
sl@0
   135
			{
sl@0
   136
			TPtrC msg = (i%2) ? db1.LastErrorMessage() : db2.LastErrorMessage();
sl@0
   137
			RDebug::Print(_L("##Db Error msg: \"%S\"\n\r"), &msg);
sl@0
   138
			}
sl@0
   139
		TEST2(err, 1);
sl@0
   140
		}
sl@0
   141
		
sl@0
   142
	//Check the database content
sl@0
   143
	RDebug::Print(_L("###Check the database content\r\n"));
sl@0
   144
	_LIT8(KSelectSql, "SELECT * FROM A");
sl@0
   145
	RSqlStatement stmt;
sl@0
   146
	err = stmt.Prepare(db1, KSelectSql);
sl@0
   147
	TEST2(err, KErrNone);
sl@0
   148
	
sl@0
   149
	for(TInt j=0;j<KRecNum;++j)
sl@0
   150
		{
sl@0
   151
		err = stmt.Next();
sl@0
   152
		TEST2(err, KSqlAtRow);
sl@0
   153
		TEST(stmt.AtRow());
sl@0
   154
		TInt id = stmt.ColumnInt(0);
sl@0
   155
		TInt data = stmt.ColumnInt(1);
sl@0
   156
		TEST(id == data);
sl@0
   157
		}
sl@0
   158
	
sl@0
   159
	stmt.Close();
sl@0
   160
sl@0
   161
	//Cleanup	
sl@0
   162
	db2.Close();
sl@0
   163
	db1.Close();
sl@0
   164
	RDebug::Print(_L("###Delete the test database\r\n"));
sl@0
   165
	(void)RSqlDatabase::Delete(KTestDbName1);
sl@0
   166
	}
sl@0
   167
sl@0
   168
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   169
sl@0
   170
struct TThreadData
sl@0
   171
	{
sl@0
   172
	TInt							iTransType;
sl@0
   173
	RSqlDatabase::TIsolationLevel	iIsolationLevel;
sl@0
   174
	TInt							iLowRecNo;
sl@0
   175
	TInt							iHighRecNo;	
sl@0
   176
	};
sl@0
   177
sl@0
   178
TInt ThreadFunc(void* aData)
sl@0
   179
	{
sl@0
   180
	__UHEAP_MARK;
sl@0
   181
	
sl@0
   182
	CTrapCleanup* tc = CTrapCleanup::New();
sl@0
   183
	TTEST(tc != NULL);
sl@0
   184
sl@0
   185
	TThreadData* data = static_cast<TThreadData*> (aData);
sl@0
   186
	TTEST(data != NULL);
sl@0
   187
sl@0
   188
	RSqlDatabase db;
sl@0
   189
	TInt err = db.Open(KTestDbName1);
sl@0
   190
	TTEST2(err, KErrNone);
sl@0
   191
	
sl@0
   192
	err = db.SetIsolationLevel(data->iIsolationLevel);
sl@0
   193
	TTEST2(err, KErrNone);
sl@0
   194
		
sl@0
   195
	if(data->iTransType == 1)
sl@0
   196
		{
sl@0
   197
		_LIT8(KBeginTrans, "BEGIN");
sl@0
   198
		err = db.Exec(KBeginTrans);
sl@0
   199
		TTEST(err >= 0);
sl@0
   200
		}
sl@0
   201
sl@0
   202
	_LIT8(KInsertSql, "INSERT INTO A(Id) VALUES(");	
sl@0
   203
	for(TInt id=data->iLowRecNo;id<=data->iHighRecNo;++id)
sl@0
   204
		{
sl@0
   205
		TBuf8<128> sql(KInsertSql);
sl@0
   206
		sql.AppendNum((TInt64)id);
sl@0
   207
		sql.Append(_L(")"));
sl@0
   208
		err = KSqlErrBusy;
sl@0
   209
		const TInt KAttempts = 20;
sl@0
   210
		for(TInt i=0;i<KAttempts&&err==KSqlErrBusy;++i)
sl@0
   211
			{
sl@0
   212
			err = db.Exec(sql);
sl@0
   213
			if(err == KSqlErrBusy)
sl@0
   214
				{
sl@0
   215
				RThread th;
sl@0
   216
				TName name = th.Name();
sl@0
   217
				RDebug::Print(_L("!!!Database locked, Thread: %S, Attempt %d, column value %d\r\n"), &name, i + 1, id);
sl@0
   218
				User::After(1000000);
sl@0
   219
				}
sl@0
   220
			}
sl@0
   221
		TTEST2(err, 1);
sl@0
   222
		}
sl@0
   223
sl@0
   224
	if(data->iTransType == 1)
sl@0
   225
		{
sl@0
   226
		_LIT8(KCommitTrans, "COMMIT");
sl@0
   227
		err = db.Exec(KCommitTrans);
sl@0
   228
		TTEST(err >= 0);
sl@0
   229
		}
sl@0
   230
sl@0
   231
	db.Close();		
sl@0
   232
	delete tc;
sl@0
   233
	
sl@0
   234
	__UHEAP_MARKEND;
sl@0
   235
	
sl@0
   236
	return KErrNone;		
sl@0
   237
	}
sl@0
   238
sl@0
   239
/**
sl@0
   240
@SYMTestCaseID			SYSLIB-SQL-CT-1613
sl@0
   241
@SYMTestCaseDesc		Multiple connections to the same database from different threads. 
sl@0
   242
						Each thread inserts set of record to the same table. Verify that all expected records
sl@0
   243
						and their column values meet the expectations.
sl@0
   244
@SYMTestPriority		High
sl@0
   245
@SYMTestActions			Testing SQL engine behaviour when having mutiple connections to the same database
sl@0
   246
						from different threads.
sl@0
   247
@SYMTestExpectedResults Test must not fail
sl@0
   248
@SYMREQ					REQ5792
sl@0
   249
                        REQ5793
sl@0
   250
*/	
sl@0
   251
void TestMultiConnDiffThread()
sl@0
   252
	{
sl@0
   253
	//Create a test database
sl@0
   254
	RDebug::Print(_L("+++:MainThread: Create test database\r\n"));
sl@0
   255
	RSqlDatabase db;
sl@0
   256
	TInt err = db.Create(KTestDbName1);
sl@0
   257
	TEST2(err, KErrNone);
sl@0
   258
	
sl@0
   259
	//Create a test table
sl@0
   260
	RDebug::Print(_L("+++:MainThread: Create a table in the test database\r\n"));
sl@0
   261
	_LIT8(KCreateSql, "CREATE TABLE A(Id INTEGER PRIMARY KEY)");
sl@0
   262
	err = db.Exec(KCreateSql);
sl@0
   263
	TEST(err >= 0);
sl@0
   264
	
sl@0
   265
	const TInt KThreadCnt = 4;
sl@0
   266
	const TInt KRange = 100;
sl@0
   267
	
sl@0
   268
	const TInt KIsolationLevelCnt = 2;
sl@0
   269
	TPtrC KIsolationLevelName[KIsolationLevelCnt] = {_L("Read Uncommitted"), _L("Serializable")};
sl@0
   270
	const RSqlDatabase::TIsolationLevel KIsolationLevels[KIsolationLevelCnt] = {
sl@0
   271
											RSqlDatabase::EReadUncommitted, RSqlDatabase::ESerializable};
sl@0
   272
											
sl@0
   273
	const TInt KTransTypeCnt = 2;											
sl@0
   274
sl@0
   275
	//Do the tests:
sl@0
   276
	// - doing each per thread database operation in a single transaction;
sl@0
   277
	// - doing all per thread database operations in a single transaction;
sl@0
   278
	for(TInt transType=0;transType<KTransTypeCnt;++transType)	
sl@0
   279
		{
sl@0
   280
		//For both supported isolation levels: read uncommitted and serializable
sl@0
   281
		for(TInt isolLevel=0;isolLevel<KIsolationLevelCnt;++isolLevel)
sl@0
   282
			{
sl@0
   283
			TInt low = 1;
sl@0
   284
			TInt high = KRange;
sl@0
   285
			
sl@0
   286
			RDebug::Print(_L("+++:MainThread: Test: thread count %d, records %d, trans type %d, isolation level: %S\r\n"), 
sl@0
   287
									KThreadCnt, KRange, transType, &KIsolationLevelName[isolLevel]);
sl@0
   288
									
sl@0
   289
			RThread thread[KThreadCnt];
sl@0
   290
			TRequestStatus status[KThreadCnt];
sl@0
   291
			TThreadData	data[KThreadCnt];
sl@0
   292
sl@0
   293
			//Create the test threads and run them. Each thread establishes a connection with the test database
sl@0
   294
			//and attempts to write set of records in the test table.
sl@0
   295
			TInt j;
sl@0
   296
			for(j=0;j<KThreadCnt;++j,low=high+1,high+=KRange)
sl@0
   297
				{
sl@0
   298
				data[j].iTransType = transType;
sl@0
   299
				data[j].iIsolationLevel = KIsolationLevels[isolLevel];
sl@0
   300
				data[j].iLowRecNo = low;
sl@0
   301
				data[j].iHighRecNo = high;
sl@0
   302
				
sl@0
   303
				_LIT(KThreadName,"Thr-");
sl@0
   304
				TBuf<32> threadName(KThreadName);
sl@0
   305
				threadName.AppendNum((TInt64)j + 1);
sl@0
   306
				
sl@0
   307
				TEST2(thread[j].Create(threadName, &ThreadFunc, 0x2000, 0x1000, 0x10000, (void*)&data[j], EOwnerThread), KErrNone);
sl@0
   308
				thread[j].Logon(status[j]);
sl@0
   309
				TEST2(status[j].Int(), KRequestPending);
sl@0
   310
				thread[j].Resume();
sl@0
   311
				}
sl@0
   312
			
sl@0
   313
			User::After(2000000);
sl@0
   314
			//Wait until threads finish the database operations and close them.				
sl@0
   315
			for(j=0;j<KThreadCnt;++j)
sl@0
   316
				{
sl@0
   317
				User::WaitForRequest(status[j]);
sl@0
   318
				TEST(thread[j].ExitType() != EExitPanic);
sl@0
   319
				thread[j].Close();
sl@0
   320
				}
sl@0
   321
sl@0
   322
			//Check that all records which are esupposed to be in the database, are there.
sl@0
   323
			RDebug::Print(_L("+++:MainThread: Check that all records have been written\r\n"));
sl@0
   324
			_LIT8(KSelectSql1, "SELECT COUNT(*) FROM A;");
sl@0
   325
			RSqlStatement stmt;
sl@0
   326
			err = stmt.Prepare(db, KSelectSql1);
sl@0
   327
			TEST2(err, KErrNone);
sl@0
   328
			err = stmt.Next();
sl@0
   329
			TEST2(err, KSqlAtRow);
sl@0
   330
			TInt cnt = stmt.ColumnInt(0);
sl@0
   331
			TEST2(cnt, KThreadCnt * KRange);
sl@0
   332
			stmt.Close();
sl@0
   333
			
sl@0
   334
			//Check that all records have expected column values.
sl@0
   335
			RDebug::Print(_L("+++:MainThread: Check that all records have expected column values\r\n"));
sl@0
   336
			_LIT8(KSelectSql2, "SELECT * FROM A;");
sl@0
   337
			err = stmt.Prepare(db, KSelectSql2);
sl@0
   338
			TEST2(err, KErrNone);
sl@0
   339
			for(TInt k=0;k<(KThreadCnt*KRange);++k)
sl@0
   340
				{
sl@0
   341
				err = stmt.Next();
sl@0
   342
				TEST2(err, KSqlAtRow);
sl@0
   343
				TInt val = stmt.ColumnInt(0);
sl@0
   344
				TEST(val > 0 && val <= (KThreadCnt * KRange));
sl@0
   345
				}
sl@0
   346
			stmt.Close();
sl@0
   347
sl@0
   348
			//Prepare for the next test run - delete all records.
sl@0
   349
			RDebug::Print(_L("+++:MainThread: Delete all records\r\n"));
sl@0
   350
			_LIT8(KDeleteSql, "DELETE FROM A");
sl@0
   351
			err = db.Exec(KDeleteSql);
sl@0
   352
			TEST(err >= 0);
sl@0
   353
			}//end of "for(TInt isolLevel=0;isolLevel<KIsolationLevelCnt;++isolLevel)"
sl@0
   354
		}//end of "for(TInt transType=0;transType<KTransTypeCnt;++transType)"
sl@0
   355
		
sl@0
   356
	db.Close();
sl@0
   357
	RDebug::Print(_L("+++:MainThread: Delete the test database\r\n"));
sl@0
   358
	(void)RSqlDatabase::Delete(KTestDbName1);
sl@0
   359
	}
sl@0
   360
sl@0
   361
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   362
sl@0
   363
static RCriticalSection UpdateThreadCrS;
sl@0
   364
static RCriticalSection MainThreadCrS;
sl@0
   365
const TInt KInitialValue = 10;
sl@0
   366
const TInt KUpdatedValue = 20;
sl@0
   367
sl@0
   368
TInt UpdateThreadFunc(void*)
sl@0
   369
	{
sl@0
   370
	__UHEAP_MARK;
sl@0
   371
	
sl@0
   372
	CTrapCleanup* tc = CTrapCleanup::New();
sl@0
   373
	TTEST(tc != NULL);
sl@0
   374
sl@0
   375
	RSqlDatabase db;
sl@0
   376
	TInt err = db.Open(KTestDbName1);
sl@0
   377
	TTEST2(err, KErrNone);
sl@0
   378
sl@0
   379
	RDebug::Print(_L("---:UpdThread: Set the isolation level to \"Read uncommitted\"\r\n"));
sl@0
   380
	err = db.SetIsolationLevel(RSqlDatabase::EReadUncommitted);
sl@0
   381
	TTEST2(err, KErrNone);
sl@0
   382
sl@0
   383
	RDebug::Print(_L("---:UpdThread: Begin a write transaction\r\n"));
sl@0
   384
	_LIT8(KBeginTransSql, "BEGIN IMMEDIATE TRANSACTION");
sl@0
   385
	err = db.Exec(KBeginTransSql);
sl@0
   386
	TTEST(err >= 0);
sl@0
   387
sl@0
   388
	RDebug::Print(_L("---:UpdThread: Update the record\r\n"));
sl@0
   389
	_LIT8(KUpdateSql, "UPDATE A SET Id = ");
sl@0
   390
	TBuf8<64> sql(KUpdateSql);
sl@0
   391
	sql.AppendNum((TInt64)KUpdatedValue);
sl@0
   392
	err = db.Exec(sql);
sl@0
   393
	TTEST(err >= 0);
sl@0
   394
sl@0
   395
	RDebug::Print(_L("---:UpdThread: Notify the main thread about the update\r\n"));
sl@0
   396
	MainThreadCrS.Signal();
sl@0
   397
	
sl@0
   398
	RDebug::Print(_L("---:UpdThread: Wait for permisson to continue...\r\n"));
sl@0
   399
	UpdateThreadCrS.Wait();
sl@0
   400
sl@0
   401
	RDebug::Print(_L("---:UpdThread: Rollback the update\r\n"));
sl@0
   402
	_LIT8(KRollBackTransSql, "ROLLBACK TRANSACTION");
sl@0
   403
	err = db.Exec(KRollBackTransSql);
sl@0
   404
	TTEST(err >= 0);
sl@0
   405
sl@0
   406
	RDebug::Print(_L("---:UpdThread: Notify the main thread about the rollback\r\n"));
sl@0
   407
	MainThreadCrS.Signal();
sl@0
   408
	
sl@0
   409
	db.Close();
sl@0
   410
	delete tc;
sl@0
   411
	
sl@0
   412
	__UHEAP_MARKEND;
sl@0
   413
	
sl@0
   414
	return KErrNone;		
sl@0
   415
	}
sl@0
   416
sl@0
   417
/**
sl@0
   418
@SYMTestCaseID			SYSLIB-SQL-CT-1614
sl@0
   419
@SYMTestCaseDesc		Verifying that when having 2 database connections in different threads, both set
sl@0
   420
						the isolation level to "Read Uncommitted", the reading thread can make "dirty read"
sl@0
   421
						operations (can read the updated but not committed yet record values made by the
sl@0
   422
						writing thread).
sl@0
   423
@SYMTestPriority		High
sl@0
   424
@SYMTestActions			Testing "Read Uncommitted" database isolation level.
sl@0
   425
@SYMTestExpectedResults Test must not fail
sl@0
   426
@SYMREQ					REQ5792
sl@0
   427
                        REQ5793
sl@0
   428
*/	
sl@0
   429
void TestIsolationLevel()
sl@0
   430
	{
sl@0
   431
	RDebug::Print(_L("+++:MainThread: Create critical sections\r\n"));
sl@0
   432
	TEST2(UpdateThreadCrS.CreateLocal(), KErrNone);
sl@0
   433
	UpdateThreadCrS.Wait();
sl@0
   434
	TEST2(MainThreadCrS.CreateLocal(), KErrNone);
sl@0
   435
	MainThreadCrS.Wait();
sl@0
   436
	
sl@0
   437
	RDebug::Print(_L("+++:MainThread: Create test database\r\n"));
sl@0
   438
	RSqlDatabase db;
sl@0
   439
	TInt err = db.Create(KTestDbName1);
sl@0
   440
	TEST2(err, KErrNone);
sl@0
   441
sl@0
   442
	RDebug::Print(_L("+++:MainThread: Set the isolation level to \"Read uncommitted\"\r\n"));
sl@0
   443
	err = db.SetIsolationLevel(RSqlDatabase::EReadUncommitted);
sl@0
   444
	TEST2(err, KErrNone);
sl@0
   445
	
sl@0
   446
	RDebug::Print(_L("+++:MainThread: Create a table in the test database\r\n"));
sl@0
   447
	_LIT8(KCreateSql, "CREATE TABLE A(Id INTEGER)");
sl@0
   448
	err = db.Exec(KCreateSql);
sl@0
   449
	TEST(err >= 0);
sl@0
   450
sl@0
   451
	RDebug::Print(_L("+++:MainThread: Insert one record in the table\r\n"));
sl@0
   452
	_LIT8(KInsertSql, "INSERT INTO A(Id) VALUES(");
sl@0
   453
	TBuf8<64> sql(KInsertSql);
sl@0
   454
	sql.AppendNum((TInt64)KInitialValue);
sl@0
   455
	sql.Append(_L(")"));
sl@0
   456
	err = db.Exec(sql);
sl@0
   457
	TEST2(err, 1);
sl@0
   458
sl@0
   459
	RDebug::Print(_L("+++:MainThread: Create the \"update\" thread\r\n"));
sl@0
   460
	_LIT(KThreadName, "UpdTh");
sl@0
   461
	RThread thread;
sl@0
   462
	TEST2(thread.Create(KThreadName, &UpdateThreadFunc, 0x2000, 0x1000, 0x10000, NULL, EOwnerThread), KErrNone);
sl@0
   463
	TRequestStatus status;
sl@0
   464
	thread.Logon(status);
sl@0
   465
	TEST2(status.Int(), KRequestPending);
sl@0
   466
	thread.Resume();
sl@0
   467
sl@0
   468
	RDebug::Print(_L("+++:MainThread: Wait for record update completion...\r\n"));
sl@0
   469
	MainThreadCrS.Wait();
sl@0
   470
sl@0
   471
	RDebug::Print(_L("+++:MainThread: Read the record and check the data...\r\n"));
sl@0
   472
	_LIT8(KSelectSql, "SELECT * FROM A");
sl@0
   473
	RSqlStatement stmt;
sl@0
   474
	err = stmt.Prepare(db, KSelectSql);
sl@0
   475
	TEST2(err, KErrNone);
sl@0
   476
	err = stmt.Next();
sl@0
   477
	TEST2(err, KSqlAtRow);
sl@0
   478
	TInt val = stmt.ColumnInt(0);
sl@0
   479
	TEST(val == KUpdatedValue);
sl@0
   480
	stmt.Close();
sl@0
   481
sl@0
   482
	RDebug::Print(_L("+++:MainThread: Notify the update thread that it can rollback\r\n"));
sl@0
   483
	UpdateThreadCrS.Signal();
sl@0
   484
sl@0
   485
	RDebug::Print(_L("+++:MainThread: Wait for  rollback  completion...\r\n"));
sl@0
   486
	MainThreadCrS.Wait();
sl@0
   487
sl@0
   488
	RDebug::Print(_L("+++:MainThread: Read the record and check the data...\r\n"));
sl@0
   489
	err = stmt.Prepare(db, KSelectSql);
sl@0
   490
	TEST2(err, KErrNone);
sl@0
   491
	err = stmt.Next();
sl@0
   492
	TEST2(err, KSqlAtRow);
sl@0
   493
	val = stmt.ColumnInt(0);
sl@0
   494
	TEST2(val, KInitialValue);
sl@0
   495
	stmt.Close();
sl@0
   496
sl@0
   497
	User::WaitForRequest(status);
sl@0
   498
	thread.Close();
sl@0
   499
sl@0
   500
	db.Close();
sl@0
   501
	RDebug::Print(_L("+++:MainThread: Delete the test database\r\n"));
sl@0
   502
	(void)RSqlDatabase::Delete(KTestDbName1);
sl@0
   503
sl@0
   504
	RDebug::Print(_L("+++:MainThread: Close critical sections\r\n"));
sl@0
   505
	MainThreadCrS.Close();
sl@0
   506
	UpdateThreadCrS.Close();
sl@0
   507
	}
sl@0
   508
sl@0
   509
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   510
sl@0
   511
void DoTestsL()
sl@0
   512
	{
sl@0
   513
	TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1612 Multiple connections, the same thread "));
sl@0
   514
	TestMultiConnSameThread();
sl@0
   515
sl@0
   516
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1613 Multiple connections, different threads "));
sl@0
   517
	TestMultiConnDiffThread();
sl@0
   518
sl@0
   519
	TheTest.Next(_L(" @SYMTestCaseID:SYSLIB-SQL-CT-1614 Isolation level "));
sl@0
   520
	TestIsolationLevel();
sl@0
   521
	}
sl@0
   522
sl@0
   523
TInt E32Main()
sl@0
   524
	{
sl@0
   525
	TheTest.Title();
sl@0
   526
	
sl@0
   527
	CTrapCleanup* tc = CTrapCleanup::New();
sl@0
   528
	
sl@0
   529
	__UHEAP_MARK;
sl@0
   530
	
sl@0
   531
	CreateTestDir();
sl@0
   532
	DeleteTestFiles();
sl@0
   533
	TRAPD(err, DoTestsL());
sl@0
   534
	DeleteTestFiles();
sl@0
   535
	TEST2(err, KErrNone);
sl@0
   536
sl@0
   537
	__UHEAP_MARKEND;
sl@0
   538
	
sl@0
   539
	TheTest.End();
sl@0
   540
	TheTest.Close();
sl@0
   541
	
sl@0
   542
	delete tc;
sl@0
   543
	
sl@0
   544
	User::Heap().Check();
sl@0
   545
	return KErrNone;
sl@0
   546
	}