os/persistentdata/persistentstorage/dbms/tdbms/t_dbstress.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
// Copyright (c) 1998-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 "t_dbstress.h"
sl@0
    17
sl@0
    18
//#define __DUMP_STATE
sl@0
    19
#if defined(__WINS__)
sl@0
    20
//#define _INSTALL_FILE_SYSTEM
sl@0
    21
#endif
sl@0
    22
sl@0
    23
GLDEF_D RTest TheTest(_L("t_dbstress: Stress testing DBMS"));
sl@0
    24
sl@0
    25
GLDEF_D TPtrC KTestDir=_S("\\DBMS-TST\\");
sl@0
    26
GLDEF_D TPtrC KLogFile=_L("T_STRESS.LOG");
sl@0
    27
GLDEF_D TPtrC KTestDatabase=_S("T_STRESS.DB");
sl@0
    28
GLDEF_D TInt NewCount,OldCount;
sl@0
    29
GLDEF_D TInt TransId;
sl@0
    30
GLDEF_D Timer RunTimer;
sl@0
    31
sl@0
    32
sl@0
    33
LOCAL_D RFs TheFs;
sl@0
    34
LOCAL_D RThread TheThread;
sl@0
    35
LOCAL_D TRequestStatus TheStatus;
sl@0
    36
LOCAL_D RDbStoreDatabase TheDatabase;
sl@0
    37
LOCAL_D RDbView Accs;
sl@0
    38
LOCAL_D RDbView Trans;
sl@0
    39
LOCAL_D RDbTable TheTable;
sl@0
    40
LOCAL_D TInt Shot,ShotDuringCommit;
sl@0
    41
LOCAL_D TInt64 RunningTime(1);
sl@0
    42
LOCAL_D Timer Stopwatch;
sl@0
    43
sl@0
    44
#ifdef __DUMP_STATE
sl@0
    45
const TPtrC KDumpFile=_S("T_STRESS.DMP");
sl@0
    46
#endif
sl@0
    47
const TInt KTestCleanupStack=0x20;
sl@0
    48
sl@0
    49
void Timer::Start()
sl@0
    50
	{
sl@0
    51
	iTime.UniversalTime();
sl@0
    52
	}
sl@0
    53
sl@0
    54
TInt64 Timer::Stop()
sl@0
    55
	{
sl@0
    56
	TTime t;
sl@0
    57
	t.UniversalTime();
sl@0
    58
	return ((t.MicroSecondsFrom(iTime).Int64()) + 500)/1000;
sl@0
    59
	}
sl@0
    60
sl@0
    61
void Timer::Print()
sl@0
    62
	{
sl@0
    63
	TInt64 milli=Stop();
sl@0
    64
	TheTest.Printf(_L("  %u milliseconds\n"), I64LOW(milli) );
sl@0
    65
	}
sl@0
    66
sl@0
    67
class Set
sl@0
    68
	{
sl@0
    69
public:
sl@0
    70
	struct SColDef
sl@0
    71
		{
sl@0
    72
		const TDesC* iName;
sl@0
    73
		TDbColType iType;
sl@0
    74
		TInt iAttributes;
sl@0
    75
		};
sl@0
    76
public:
sl@0
    77
	static CDbColSet* CreateL(const SColDef* aDef);
sl@0
    78
	};
sl@0
    79
CDbColSet* Set::CreateL(const SColDef* aDef)
sl@0
    80
	{
sl@0
    81
	CDbColSet *set=CDbColSet::NewLC();
sl@0
    82
	for (;aDef->iName!=NULL;++aDef)
sl@0
    83
		{
sl@0
    84
		TDbCol col(*aDef->iName,aDef->iType);
sl@0
    85
		col.iAttributes=aDef->iAttributes;
sl@0
    86
		set->AddL(col);
sl@0
    87
		}
sl@0
    88
	CleanupStack::Pop();
sl@0
    89
	return set;
sl@0
    90
	}
sl@0
    91
sl@0
    92
// Accounts table
sl@0
    93
const TPtrC KAccounts=_S("ACCOUNTS");
sl@0
    94
const TPtrC KAccountsID=_S("ID");
sl@0
    95
const TPtrC KAccountsBalance=_S("BALANCE");
sl@0
    96
const TPtrC KAccountsStatement=_S("STATEMENT_BALANCE");
sl@0
    97
Set::SColDef const AccountsDef[]=
sl@0
    98
	{
sl@0
    99
	{&KAccountsID,EDbColInt32,TDbCol::ENotNull},
sl@0
   100
	{&KAccountsBalance,EDbColInt32,TDbCol::ENotNull},
sl@0
   101
	{&KAccountsStatement,EDbColInt32,TDbCol::ENotNull},
sl@0
   102
	{0}
sl@0
   103
	};
sl@0
   104
const TInt KInitialCash=100000;
sl@0
   105
const TInt KInitialBalance=1000;
sl@0
   106
sl@0
   107
// Transaction table
sl@0
   108
const TPtrC KTransactions=_S("TRANSACTIONS");
sl@0
   109
const TPtrC KTransactionDate=_S("T_DATE");
sl@0
   110
const TPtrC KTransactionFrom=_S("FROM_ID");
sl@0
   111
const TPtrC KTransactionTo=_S("TO_ID");
sl@0
   112
const TPtrC KTransactionAmount=_S("AMOUNT");
sl@0
   113
Set::SColDef const TransactionsDef[]=
sl@0
   114
	{
sl@0
   115
//	{&KTransactionDate,EDbColDateTime,TDbCol::ENotNull},
sl@0
   116
	{&KTransactionDate,EDbColInt32,TDbCol::ENotNull},
sl@0
   117
	{&KTransactionFrom,EDbColInt32,TDbCol::ENotNull},
sl@0
   118
	{&KTransactionTo,EDbColInt32,TDbCol::ENotNull},
sl@0
   119
	{&KTransactionAmount,EDbColInt32,TDbCol::ENotNull},
sl@0
   120
	{0}
sl@0
   121
	};
sl@0
   122
sl@0
   123
LOCAL_D TInt32 TotalMonies;
sl@0
   124
LOCAL_D TBuf<100> Buf;
sl@0
   125
sl@0
   126
GLDEF_C TInt Random(TInt aRange)
sl@0
   127
	{
sl@0
   128
	return (Math::Random()>>11)%aRange;
sl@0
   129
	}
sl@0
   130
sl@0
   131
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   132
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   133
sl@0
   134
TPtrC FileName(const TText* aFile)
sl@0
   135
    {
sl@0
   136
    TPtrC p(aFile);
sl@0
   137
    TInt ix=p.LocateReverse('\\');
sl@0
   138
    if (ix<0)
sl@0
   139
        ix=p.LocateReverse('/');
sl@0
   140
    if (ix>=0)
sl@0
   141
        p.Set(p.Mid(1+ix));
sl@0
   142
    return p;
sl@0
   143
    }
sl@0
   144
sl@0
   145
//Test macros and functions
sl@0
   146
void Check1(TInt aValue, const TText* aFile, TInt aLine)
sl@0
   147
    {
sl@0
   148
    if(!aValue)
sl@0
   149
        {
sl@0
   150
        TPtrC fname(FileName(aFile));
sl@0
   151
        TheTest.Printf(_L("*** Expression evaluated to false. %S-%d\r\n"), &fname, aLine);
sl@0
   152
        TheTest(EFalse, aLine);
sl@0
   153
        }
sl@0
   154
    }
sl@0
   155
void Check2(TInt aValue, TInt aExpected, const TText* aFile, TInt aLine)
sl@0
   156
    {
sl@0
   157
    if(aValue != aExpected)
sl@0
   158
        {
sl@0
   159
        TPtrC fname(FileName(aFile));
sl@0
   160
        TheTest.Printf(_L("*** Expected %d, got %d. %S-%d\r\n"), aExpected, aValue, &fname, aLine);
sl@0
   161
        TheTest(EFalse, aLine);
sl@0
   162
        }
sl@0
   163
    }
sl@0
   164
sl@0
   165
///////////////////////////////////////////////////////////////////////////////////////
sl@0
   166
sl@0
   167
LOCAL_C void CreateIndexL(RDbDatabase& aDatabase,const TDesC& aTable,const TDesC& aColumn,TBool aUnique)
sl@0
   168
	{
sl@0
   169
	CDbKey* key=CDbKey::NewLC();
sl@0
   170
	key->AddL(aColumn);
sl@0
   171
	if (aUnique)
sl@0
   172
		key->MakeUnique();
sl@0
   173
	TEST2(aDatabase.CreateIndex(aColumn,aTable,*key),KErrNone);
sl@0
   174
	CleanupStack::PopAndDestroy();
sl@0
   175
	}
sl@0
   176
sl@0
   177
//
sl@0
   178
// Create the database
sl@0
   179
//
sl@0
   180
LOCAL_C void CreateDatabaseL()
sl@0
   181
	{
sl@0
   182
	CFileStore* store=CPermanentFileStore::ReplaceLC(TheFs,KTestDatabase,EFileRead|EFileWrite);
sl@0
   183
	store->SetTypeL(KPermanentFileStoreLayoutUid);
sl@0
   184
	store->SetRootL(TheDatabase.CreateL(store));
sl@0
   185
//	create the tables
sl@0
   186
	TheDatabase.Begin();
sl@0
   187
	CDbColSet* set=Set::CreateL(AccountsDef);
sl@0
   188
	TEST2(TheDatabase.CreateTable(KAccounts,*set),KErrNone);
sl@0
   189
	delete set;
sl@0
   190
	CreateIndexL(TheDatabase,KAccounts,KAccountsID,ETrue);
sl@0
   191
	CreateIndexL(TheDatabase,KAccounts,KAccountsBalance,EFalse);
sl@0
   192
	set=Set::CreateL(TransactionsDef);
sl@0
   193
	TEST2(TheDatabase.CreateTable(KTransactions,*set),KErrNone);
sl@0
   194
	delete set;
sl@0
   195
	CreateIndexL(TheDatabase,KTransactions,KTransactionDate,EFalse);
sl@0
   196
	TEST2(TheDatabase.Commit(),KErrNone);
sl@0
   197
	OldCount=NewCount=0;
sl@0
   198
// prepare Accs table
sl@0
   199
	TheDatabase.Begin();
sl@0
   200
	TEST2(Accs.Prepare(TheDatabase,_L("select * from accounts"),Accs.EInsertOnly),KErrNone);
sl@0
   201
	Accs.InsertL();
sl@0
   202
	Accs.SetColL(1,TInt32(ECash));
sl@0
   203
	Accs.SetColL(2,KInitialCash);
sl@0
   204
	Accs.SetColL(3,KInitialCash);
sl@0
   205
	Accs.PutL();
sl@0
   206
	TotalMonies=KInitialCash;
sl@0
   207
	for (TInt ii=EJohn;ii<=EPenny;++ii)
sl@0
   208
		{
sl@0
   209
		Accs.InsertL();
sl@0
   210
		Accs.SetColL(1,ii);
sl@0
   211
		Accs.SetColL(2,KInitialBalance);
sl@0
   212
		Accs.SetColL(3,KInitialBalance);
sl@0
   213
		Accs.PutL();
sl@0
   214
		TotalMonies+=KInitialBalance;
sl@0
   215
		}
sl@0
   216
	TEST2(TheDatabase.Commit(),KErrNone);
sl@0
   217
	Accs.Close();
sl@0
   218
	TheDatabase.Close();
sl@0
   219
	CleanupStack::PopAndDestroy();	// store
sl@0
   220
	}
sl@0
   221
sl@0
   222
sl@0
   223
#ifdef __DUMP_STATE
sl@0
   224
LOCAL_C void DumpStateL()
sl@0
   225
	{
sl@0
   226
	RFile file;
sl@0
   227
	CleanupClosePushL(file);
sl@0
   228
	User::LeaveIfError(file.Replace(TheFs,KLogFile,EFileWrite|EFileStreamText));
sl@0
   229
	RDbRowSet::RConstraint match;
sl@0
   230
	CleanupClosePushL(match);
sl@0
   231
	for (TInt id=ECash;id<=EPenny;++id)
sl@0
   232
		{
sl@0
   233
		Buf.Format(_L("id=%d"),id);
sl@0
   234
		Accs.FirstL();
sl@0
   235
		TEST(Accs.FindL(Accs.EForwards,Buf)>=0);
sl@0
   236
		Accs.GetL();
sl@0
   237
		TInt balance=Accs.ColInt(2);
sl@0
   238
		Buf.Format(_L("\nStatement for account %d: Previous balance %d\n"),id,balance);
sl@0
   239
		User::LeaveIfError(file.Write(Buf));
sl@0
   240
		Buf.Format(_L("from_id=%d or to_id=%d"),id,id);
sl@0
   241
		User::LeaveIfError(match.Open(Trans,Buf));
sl@0
   242
		for (Trans.BeginningL();Trans.NextL();)
sl@0
   243
			{
sl@0
   244
			if (Trans.MatchL(match))
sl@0
   245
				{
sl@0
   246
				Trans.GetL();
sl@0
   247
				TInt from=Trans.ColInt(2);
sl@0
   248
				TInt amount=Trans.ColInt(4);
sl@0
   249
				Buf.Format(_L("%04d: %6s %5d\n"),Trans.ColInt(1),from==id?_S("debit"):_S("credit"),amount);
sl@0
   250
				User::LeaveIfError(file.Write(Buf));
sl@0
   251
				if (from==id)
sl@0
   252
					balance-=amount;
sl@0
   253
				else
sl@0
   254
					balance+=amount;
sl@0
   255
				}
sl@0
   256
			}
sl@0
   257
		Buf.Format(_L("Closing balance %d\n"),balance);
sl@0
   258
		User::LeaveIfError(file.Write(Buf));
sl@0
   259
		Buf.Format(_L("Account balance %d\n"),Accs.ColInt(3));
sl@0
   260
		User::LeaveIfError(file.Write(Buf));
sl@0
   261
		}
sl@0
   262
	CleanupStack::PopAndDestroy(2);
sl@0
   263
	TEST(0);
sl@0
   264
	}
sl@0
   265
#endif
sl@0
   266
sl@0
   267
//
sl@0
   268
// Check that the database structure is fully intact
sl@0
   269
//
sl@0
   270
LOCAL_C void VerifyDatabaseL(CPersistentStore& aStore)
sl@0
   271
	{
sl@0
   272
	TheDatabase.OpenL(&aStore,aStore.Root());
sl@0
   273
// check any indexes
sl@0
   274
	TEST2(TheTable.Open(TheDatabase,KAccounts,TheTable.EReadOnly),KErrNone);
sl@0
   275
	TEST2(TheTable.CountL(),KAccountIDs);
sl@0
   276
	TInt r=TheTable.SetIndex(KAccountsID);
sl@0
   277
	if (r!=KErrCorrupt)
sl@0
   278
		{
sl@0
   279
		TEST2(r,KErrNone);
sl@0
   280
		TEST2(TheTable.CountL(),KAccountIDs);
sl@0
   281
		for (TInt id=ECash;id<=EPenny;++id)
sl@0
   282
			{
sl@0
   283
			TEST(TheTable.NextL());
sl@0
   284
			TheTable.GetL();
sl@0
   285
			TEST2(TheTable.ColInt(1),id);
sl@0
   286
			}
sl@0
   287
		TEST(!TheTable.NextL());
sl@0
   288
		}
sl@0
   289
	r=TheTable.SetIndex(KAccountsBalance);
sl@0
   290
	if (r!=KErrCorrupt)
sl@0
   291
		{
sl@0
   292
		TEST2(r,KErrNone);
sl@0
   293
		TEST2(TheTable.CountL(),KAccountIDs);
sl@0
   294
		TEST(TheTable.FirstL());
sl@0
   295
		TheTable.GetL();
sl@0
   296
		TInt last=TheTable.ColInt(2);
sl@0
   297
		for (TInt ii=1;ii<KAccountIDs;++ii)
sl@0
   298
			{
sl@0
   299
			TEST(TheTable.NextL());
sl@0
   300
			TheTable.GetL();
sl@0
   301
			TInt bal=TheTable.ColInt(2);
sl@0
   302
			TEST(bal>=last);
sl@0
   303
			last=bal;
sl@0
   304
			}
sl@0
   305
		TEST(!TheTable.NextL());
sl@0
   306
		}
sl@0
   307
	TheTable.Close();
sl@0
   308
	TEST2(TheTable.Open(TheDatabase,KTransactions,TheTable.EReadOnly),KErrNone);
sl@0
   309
	TInt count=TheTable.CountL();
sl@0
   310
	r=TheTable.SetIndex(KTransactionDate);
sl@0
   311
	if (r!=KErrCorrupt)
sl@0
   312
		{
sl@0
   313
		TEST2(r,KErrNone);
sl@0
   314
		TEST2(TheTable.CountL(),count);
sl@0
   315
		if (count)
sl@0
   316
			{
sl@0
   317
			TEST(TheTable.FirstL());
sl@0
   318
			TheTable.GetL();
sl@0
   319
			TInt last=TheTable.ColInt(1);
sl@0
   320
			while (--count!=0)
sl@0
   321
				{
sl@0
   322
				TEST(TheTable.NextL());
sl@0
   323
				TheTable.GetL();
sl@0
   324
				TInt date=TheTable.ColInt(1);
sl@0
   325
				TEST(date>last);
sl@0
   326
				last=date;
sl@0
   327
				}
sl@0
   328
			TEST(!TheTable.NextL());
sl@0
   329
			}
sl@0
   330
		else
sl@0
   331
			TEST(!TheTable.FirstL());
sl@0
   332
		}
sl@0
   333
	TheTable.Close();
sl@0
   334
// verify database integrity
sl@0
   335
	TInt balance[KAccountIDs];
sl@0
   336
	TEST2(Accs.Prepare(TheDatabase,_L("select id,statement_balance,balance from accounts"),Accs.EReadOnly),KErrNone);
sl@0
   337
	TEST2(Accs.CountL(),KAccountIDs);
sl@0
   338
	while (Accs.NextL())
sl@0
   339
		{
sl@0
   340
		Accs.GetL();
sl@0
   341
		TInt id=Accs.ColInt(1);
sl@0
   342
		balance[id]=Accs.ColInt(2);
sl@0
   343
		}
sl@0
   344
	TEST2(Trans.Prepare(TheDatabase,_L("select t_date,from_id,to_id,amount from Transactions"),Trans.EReadOnly),KErrNone);
sl@0
   345
	TInt transact=0;
sl@0
   346
	while (Trans.NextL())
sl@0
   347
		{
sl@0
   348
		++transact;
sl@0
   349
		Trans.GetL();
sl@0
   350
		TInt from=Trans.ColInt(2);
sl@0
   351
		TInt to=Trans.ColInt(3);
sl@0
   352
		TInt amount=Trans.ColInt(4);
sl@0
   353
		balance[from]-=amount;
sl@0
   354
		balance[to]+=amount;
sl@0
   355
		}
sl@0
   356
	TEST2(transact,Trans.CountL());
sl@0
   357
	if (NewCount!=-1 && transact!=NewCount)
sl@0
   358
		{
sl@0
   359
		TEST2(transact,OldCount);
sl@0
   360
		++ShotDuringCommit;
sl@0
   361
		}
sl@0
   362
	OldCount=NewCount=transact;
sl@0
   363
	TInt total=0;
sl@0
   364
	for (Accs.BeginningL();Accs.NextL();)
sl@0
   365
		{
sl@0
   366
		Accs.GetL();
sl@0
   367
		TInt id=Accs.ColInt(1);
sl@0
   368
#ifdef __DUMP_STATE
sl@0
   369
		if (balance[id]!=Accs.ColInt(3))
sl@0
   370
			DumpStateL();
sl@0
   371
#else
sl@0
   372
		TEST(balance[id]==Accs.ColInt(3));
sl@0
   373
#endif
sl@0
   374
		total+=balance[id];
sl@0
   375
		}
sl@0
   376
	TEST2(total,TotalMonies);
sl@0
   377
	Trans.Close();
sl@0
   378
	Accs.Close();
sl@0
   379
	TheDatabase.Close();
sl@0
   380
	}
sl@0
   381
sl@0
   382
LOCAL_C TInt Verify(CPersistentStore& aStore)
sl@0
   383
	{
sl@0
   384
	TRAPD(r,VerifyDatabaseL(aStore));
sl@0
   385
	return r;
sl@0
   386
	}
sl@0
   387
sl@0
   388
LOCAL_C TInt Recover(CPersistentStore& aStore)
sl@0
   389
	{
sl@0
   390
	TRAPD(r,TheDatabase.OpenL(&aStore,aStore.Root()));
sl@0
   391
	if (r==KErrNone)
sl@0
   392
		{
sl@0
   393
		r=TheDatabase.Recover();
sl@0
   394
		TheDatabase.Close();
sl@0
   395
		}
sl@0
   396
	return r;
sl@0
   397
	}
sl@0
   398
sl@0
   399
LOCAL_C void CompactL(CStreamStore& aStore)
sl@0
   400
	{
sl@0
   401
	TInt t=aStore.ReclaimL();
sl@0
   402
	Stopwatch.Start();
sl@0
   403
	t-=aStore.CompactL();
sl@0
   404
	TheTest.Printf(_L("  compacted %d byte(s) in"),t);
sl@0
   405
	Stopwatch.Print();
sl@0
   406
	aStore.CommitL();
sl@0
   407
	}
sl@0
   408
sl@0
   409
LOCAL_C TInt Compact(CStreamStore& aStore)
sl@0
   410
	{
sl@0
   411
	TRAPD(r,CompactL(aStore));
sl@0
   412
	return r;
sl@0
   413
	}
sl@0
   414
sl@0
   415
LOCAL_C TInt EndThread()
sl@0
   416
	{
sl@0
   417
	RunningTime+=RunTimer.Stop();
sl@0
   418
	if (TheStatus==KRequestPending)
sl@0
   419
		TheThread.Kill(1);
sl@0
   420
	User::WaitForRequest(TheStatus);
sl@0
   421
	TInt r;
sl@0
   422
	if (TheThread.ExitType()==EExitKill)
sl@0
   423
		r=TheThread.ExitReason();
sl@0
   424
	else
sl@0
   425
		r=TheStatus.Int();
sl@0
   426
	TheThread.Close();
sl@0
   427
	return r;
sl@0
   428
	}
sl@0
   429
sl@0
   430
//aTestExecutionTime - desired test execution time in minutes
sl@0
   431
LOCAL_C void RunTestL(TInt aTestExecutionTime = 0)
sl@0
   432
	{
sl@0
   433
	__ASSERT_ALWAYS(aTestExecutionTime >= 0, User::Invariant());
sl@0
   434
sl@0
   435
	RThread().SetPriority(EPriorityMore);
sl@0
   436
	TheTest.Start(_L("Create the database"));
sl@0
   437
	CreateDatabaseL();
sl@0
   438
sl@0
   439
	TTimeIntervalMinutes timeInterval(aTestExecutionTime);
sl@0
   440
sl@0
   441
	TTime timeCurrent;
sl@0
   442
	timeCurrent.UniversalTime();
sl@0
   443
	TTime timeEnd(timeCurrent);
sl@0
   444
	timeEnd += timeInterval;
sl@0
   445
sl@0
   446
	for (TBool condition=ETrue; condition; condition = aTestExecutionTime > 0 ? (timeCurrent < timeEnd) : ETrue)
sl@0
   447
		{
sl@0
   448
		TheTest.Next(_L("Main loop"));
sl@0
   449
		TheTest.Start(_L("Kick off the thread"));
sl@0
   450
		TEST2 (StartThread(TheThread,TheStatus),KErrNone);
sl@0
   451
		// random delay
sl@0
   452
		for (;;)
sl@0
   453
			{
sl@0
   454
			User::After(95000);
sl@0
   455
			if (TheStatus!=KRequestPending)
sl@0
   456
				break;
sl@0
   457
			if (Random(1000)<30)
sl@0
   458
				break;
sl@0
   459
			}
sl@0
   460
		TheTest.Next(_L("End the thread"));
sl@0
   461
		TInt exit=EndThread();
sl@0
   462
		if (exit!=1)
sl@0
   463
		    TheTest.Printf(_L("  thread failed with error %d\n"),exit);
sl@0
   464
//
sl@0
   465
		++Shot;
sl@0
   466
		CFileStore* store=NULL;
sl@0
   467
		for (TInt ii=0;;++ii)
sl@0
   468
			{
sl@0
   469
			TheTest.Printf(_L("Opening %d\r"),ii);
sl@0
   470
			TRAPD(r,store=CFileStore::OpenL(TheFs,KTestDatabase,EFileRead|EFileWrite|EFileWriteDirectIO));
sl@0
   471
			if (r==KErrNone)
sl@0
   472
				break;
sl@0
   473
			TEST2(r, KErrInUse);
sl@0
   474
			User::After(100000);
sl@0
   475
			}
sl@0
   476
		TheTest.Next(_L("Verify & Recover"));
sl@0
   477
		TEST2 (Verify(*store),KErrNone);
sl@0
   478
		TInt64 tps(TransId);
sl@0
   479
		tps*=1000u;
sl@0
   480
		tps/=RunningTime;
sl@0
   481
		TheTest.Printf(_L("    Iteration %d, TPS %d, during commit %d%%\n"),Shot,I64LOW(tps),(100*ShotDuringCommit)/Shot);
sl@0
   482
		TInt r=Recover(*store);
sl@0
   483
		if (r==KErrNoMemory || r==KErrDiskFull)
sl@0
   484
			{	// need to compact before completing recovery
sl@0
   485
			TheTest.Next(_L("No space, compacting"));
sl@0
   486
			TEST2 (Compact(*store),KErrNone);
sl@0
   487
			TheTest.Next(_L("Verify & Recover again"));
sl@0
   488
			TEST2 (Verify(*store),KErrNone);
sl@0
   489
			r=Recover(*store);
sl@0
   490
			}
sl@0
   491
		TEST2 (r,KErrNone);
sl@0
   492
		TheTest.Next(_L("Verify & Compact"));
sl@0
   493
//		TEST2 (Verify(*store),KErrNone);
sl@0
   494
		TEST2 (Compact(*store),KErrNone);
sl@0
   495
		TheTest.Next(_L("Verify"));
sl@0
   496
		TEST2 (Verify(*store),KErrNone);
sl@0
   497
//
sl@0
   498
		delete store;
sl@0
   499
		TheTest.End();
sl@0
   500
sl@0
   501
		timeCurrent.UniversalTime();
sl@0
   502
		}
sl@0
   503
	TheTest.End();
sl@0
   504
	}
sl@0
   505
sl@0
   506
/**
sl@0
   507
@SYMTestCaseID          SYSLIB-DBMS-CT-0636
sl@0
   508
@SYMTestCaseDesc        DBMS stess testing.
sl@0
   509
@SYMTestPriority        Medium
sl@0
   510
@SYMTestActions         Tests for verifying the database integrity.
sl@0
   511
@SYMTestExpectedResults Test must not fail
sl@0
   512
@SYMREQ                 REQ0000
sl@0
   513
*/
sl@0
   514
static void RunVerify()
sl@0
   515
	{
sl@0
   516
	TheTest.Start(_L(" @SYMTestCaseID:SYSLIB-DBMS-CT-0636 Open store "));
sl@0
   517
	CFileStore* store=NULL;
sl@0
   518
	TRAPD(r,store=CFileStore::OpenL(TheFs,KTestDatabase,EFileRead|EFileWrite|EFileWriteDirectIO));
sl@0
   519
	TEST2 (r,KErrNone);
sl@0
   520
	TheTest.Next(_L("Verify"));
sl@0
   521
	NewCount=-1;
sl@0
   522
	TotalMonies=KInitialCash + (EPenny-EJohn+1)*KInitialBalance;
sl@0
   523
	TEST2 (Verify(*store),KErrNone);
sl@0
   524
	TheTest.Next(_L("Recover"));
sl@0
   525
	TEST2 (Recover(*store),KErrNone);
sl@0
   526
	TheTest.Next(_L("Verify"));
sl@0
   527
	TEST2 (Verify(*store),KErrNone);
sl@0
   528
	delete store;
sl@0
   529
	TheTest.End();
sl@0
   530
	}
sl@0
   531
sl@0
   532
//
sl@0
   533
// Prepare the test directory.
sl@0
   534
//
sl@0
   535
LOCAL_C void setupTestDirectory()
sl@0
   536
    {
sl@0
   537
	TInt r=TheFs.Connect();
sl@0
   538
	TEST2(r,KErrNone);
sl@0
   539
//
sl@0
   540
	r=TheFs.MkDir(KTestDir);
sl@0
   541
	TEST(r==KErrNone || r==KErrAlreadyExists);
sl@0
   542
	r=TheFs.SetSessionPath(KTestDir);
sl@0
   543
	TEST2(r,KErrNone);
sl@0
   544
	}
sl@0
   545
sl@0
   546
//
sl@0
   547
// Initialise the cleanup stack.
sl@0
   548
//
sl@0
   549
LOCAL_C CTrapCleanup* setupCleanup()
sl@0
   550
    {
sl@0
   551
	CTrapCleanup* cleanup=CTrapCleanup::New();
sl@0
   552
	TEST(cleanup!=NULL);
sl@0
   553
	TRAPD(r,\
sl@0
   554
		{\
sl@0
   555
		for (TInt i=KTestCleanupStack;i>0;i--)\
sl@0
   556
			CleanupStack::PushL((TAny*)0);\
sl@0
   557
		CleanupStack::Pop(KTestCleanupStack);\
sl@0
   558
		});
sl@0
   559
	TEST2(r,KErrNone);
sl@0
   560
	return cleanup;
sl@0
   561
	}
sl@0
   562
sl@0
   563
//
sl@0
   564
// entry point
sl@0
   565
//
sl@0
   566
// Parameters usage:
sl@0
   567
//	t_stress [-v]|[0]|[<positive number>]
sl@0
   568
// Where:
sl@0
   569
//	-v - a verification test will be run;
sl@0
   570
//	0  - a stress test will be run for indefinite time;
sl@0
   571
//	<positive number>  - a stress test will be run for <positive number> minutes;
sl@0
   572
// If the test is run without arguments, the test execution time will be 10 minutes
sl@0
   573
// (KDefaultTestExecutionTime constant bellow).
sl@0
   574
GLDEF_C TInt E32Main()
sl@0
   575
    {
sl@0
   576
    TheTest.Title();
sl@0
   577
	setupTestDirectory();
sl@0
   578
	CTrapCleanup* cleanup=setupCleanup();
sl@0
   579
	__UHEAP_MARK;
sl@0
   580
//
sl@0
   581
	TBuf<100> cmd;
sl@0
   582
    User::CommandLine(cmd);
sl@0
   583
	TLex lex(cmd);
sl@0
   584
	TInt err = KErrNone;
sl@0
   585
	for(;;)
sl@0
   586
		{
sl@0
   587
		TPtrC arg(lex.NextToken());
sl@0
   588
		if(arg.Length() == 0)
sl@0
   589
			{
sl@0
   590
			const TInt KDefaultTestExecutionTime = 10;//default test execution time - minutes
sl@0
   591
			TRAP(err, RunTestL(KDefaultTestExecutionTime));
sl@0
   592
			break;
sl@0
   593
			}
sl@0
   594
		else if(arg.CompareF(_L("-v")) == 0)
sl@0
   595
			{
sl@0
   596
			RunVerify();
sl@0
   597
			break;
sl@0
   598
			}
sl@0
   599
		else
sl@0
   600
			{
sl@0
   601
			TInt32 testExecutionTime = 0;
sl@0
   602
			lex.Assign(arg);
sl@0
   603
			(void)lex.Val(testExecutionTime);
sl@0
   604
			TRAP(err, RunTestL(testExecutionTime));
sl@0
   605
			break;
sl@0
   606
			}
sl@0
   607
		}
sl@0
   608
	TInt err2 = TheFs.Delete(KTestDatabase);
sl@0
   609
	if(err2 != KErrNone)
sl@0
   610
		{
sl@0
   611
		RDebug::Print(_L("Error %d deleting \"%S\" file.\n"), err2, &KTestDatabase);
sl@0
   612
		}
sl@0
   613
	err2 = TheFs.Delete(KLogFile);
sl@0
   614
	if(err2 != KErrNone)
sl@0
   615
		{
sl@0
   616
		RDebug::Print(_L("Error %d deleting \"%S\" file.\n"), err2, &KLogFile);
sl@0
   617
		}
sl@0
   618
	TEST2(err, KErrNone);
sl@0
   619
//
sl@0
   620
	__UHEAP_MARKEND;
sl@0
   621
	delete cleanup;
sl@0
   622
	TheFs.Close();
sl@0
   623
	TheTest.Close();
sl@0
   624
	return 0;
sl@0
   625
    }