os/persistentdata/loggingservices/eventlogger/test/src/t_logbadclient.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 <e32math.h>
sl@0
    19
#include "LogServShared.h"
sl@0
    20
#include "logservcli.h"
sl@0
    21
#include "t_logutil.h"
sl@0
    22
sl@0
    23
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    24
sl@0
    25
RTest TheTest(_L("t_logbadclient test"));
sl@0
    26
sl@0
    27
const TInt KTestIterCount = 5000;
sl@0
    28
const TInt KMaxDesArgLen = 1000;
sl@0
    29
enum TArgType 
sl@0
    30
	{
sl@0
    31
	EIntArgType, 
sl@0
    32
	ETextArgType, 
sl@0
    33
	EBinArgType, 
sl@0
    34
	ELastArgType
sl@0
    35
	};
sl@0
    36
sl@0
    37
const TLogServFunction KLogIpcMsgCodes[] = 
sl@0
    38
	{
sl@0
    39
	ELogOperationCancel, ELogOperationGetResult, ELogOperationInitiate, ELogNotify, ELogNotifyCancel, ELogViewCreate,
sl@0
    40
	ELogViewDelete, ELogViewCount, ELogViewOperationInitiate, ELogViewChangeNotificationsRequest,
sl@0
    41
	ELogViewChangeNotificationsCancel, ELogViewFetchChanges, ELogViewNotifyLockStatusChange, 
sl@0
    42
	ELogViewNotifyLockStatusChangeCancel, ELogNotifyExtended, ELogNotifyExtendedCancel, ELogNOTUSED   
sl@0
    43
	};
sl@0
    44
sl@0
    45
const TLogOperationType KLogOpTypes[] =
sl@0
    46
	{
sl@0
    47
	ELogOperationEventAdd, ELogOperationEventGet, ELogOperationEventChange, ELogOperationEventDelete,
sl@0
    48
	ELogOperationTypeAdd, ELogOperationTypeGet, ELogOperationTypeChange, ELogOperationTypeDelete,
sl@0
    49
	ELogOperationClearLog,  ELogOperationClearRecent, ELogOperationConfigGet,  ELogOperationConfigChange,
sl@0
    50
	ELogOperationMaintain, ELogOperationViewSetup, ELogOperationViewRemoveEvent, ELogOperationViewClearDuplicates,
sl@0
    51
	ELogOperationViewSetFlags, ELogOperationViewWindowFetch, (TLogOperationType)-100
sl@0
    52
	};
sl@0
    53
sl@0
    54
//////////////////////////////////////////////////////////////////////////////////////
sl@0
    55
sl@0
    56
//If the LogEng server crashes and the test receives KErrServerTerminated error, then the 
sl@0
    57
//next set will contain the last:
sl@0
    58
// - iteration number;
sl@0
    59
// - function code;
sl@0
    60
// - IPC arguments values;
sl@0
    61
struct TThreadData
sl@0
    62
	{
sl@0
    63
	TInt				iIteration;
sl@0
    64
	TInt 				iFunction;
sl@0
    65
	TArgType 			iArgType[KMaxMessageArguments];
sl@0
    66
	TInt 				iIntArg[KMaxMessageArguments];
sl@0
    67
	TBuf<KMaxDesArgLen> iTextArg[KMaxMessageArguments];
sl@0
    68
	TBuf8<KMaxDesArgLen> iBinArg[KMaxMessageArguments];
sl@0
    69
	TInt64 				iSeed;
sl@0
    70
	};
sl@0
    71
//////////////////////////////////////////////////////////////////////////////////////
sl@0
    72
sl@0
    73
_LIT(KPanicCategory, "SrvTerm");
sl@0
    74
_LIT(KPanicCategory2, "InvArg");
sl@0
    75
const TInt KPanicCode = 1111;
sl@0
    76
const TInt KPanicCode2 = 2222;
sl@0
    77
sl@0
    78
static TLogClientServerData TheLogIpcData;
sl@0
    79
static TPtrC8 TheLogIpcDataPtr((const TUint8*)&TheLogIpcData, sizeof(TheLogIpcData));
sl@0
    80
sl@0
    81
///////////////////////////////////////////////////////////////////////////////////////
sl@0
    82
sl@0
    83
void PrintIterationCount(TInt aIteration)
sl@0
    84
	{
sl@0
    85
	if((aIteration % 100) == 0)
sl@0
    86
		{
sl@0
    87
		TTime time;
sl@0
    88
		time.HomeTime();
sl@0
    89
		TDateTime dt = time.DateTime();
sl@0
    90
		TBuf<16> tbuf;
sl@0
    91
		tbuf.Format(_L("%02d:%02d:%02d.%06d"), dt.Hour(), dt.Minute(), dt.Second(), dt.MicroSecond());
sl@0
    92
		TheTest.Printf(_L("-----[%S] Test iterations: %d\r\n"), &tbuf, aIteration);
sl@0
    93
		}
sl@0
    94
	}
sl@0
    95
sl@0
    96
//////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    97
//////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    98
sl@0
    99
//Worker thread function.
sl@0
   100
//It behaves as a malicious client. Connects to the LogEng server. In each test iteration generates some random values
sl@0
   101
//for the function number, handle, handle type, IPC arguments. Then sends a command to the server using these
sl@0
   102
//randomly generated values. If the server crashes and the thread function receives KErrServerTerminated error,
sl@0
   103
//then the thread kills itself and the main thread will get KPanicCategory and KPanicCode as a reason for the
sl@0
   104
//worker thread's death. The last set of randomly generated values will be stored in the memory, pointed by aData argument.
sl@0
   105
TInt ThreadFunc1(void* aData)
sl@0
   106
	{
sl@0
   107
	__UHEAP_MARK;
sl@0
   108
	
sl@0
   109
	CTrapCleanup* tc = CTrapCleanup::New();
sl@0
   110
	TTEST(tc != NULL);
sl@0
   111
sl@0
   112
	TThreadData* p = static_cast <TThreadData*> (aData);
sl@0
   113
	TTEST(p != NULL);
sl@0
   114
	TThreadData& data = *p;
sl@0
   115
	
sl@0
   116
	RLogSession sess;
sl@0
   117
	TInt err = sess.Connect();
sl@0
   118
	TTEST2(err, KErrNone);
sl@0
   119
sl@0
   120
	while(++data.iIteration <= KTestIterCount)
sl@0
   121
		{
sl@0
   122
		TIpcArgs args;
sl@0
   123
		const TInt KFnCnt = sizeof(KLogIpcMsgCodes) / sizeof(KLogIpcMsgCodes[0]);
sl@0
   124
		TInt fnIdx = Math::Rand(data.iSeed) % KFnCnt;
sl@0
   125
		data.iFunction = KLogIpcMsgCodes[fnIdx];
sl@0
   126
		PrintIterationCount(data.iIteration);
sl@0
   127
		for(TInt argIdx=0;argIdx<KMaxMessageArguments;++argIdx)
sl@0
   128
			{
sl@0
   129
			//Initialize arguments
sl@0
   130
			data.iArgType[argIdx] = EBinArgType;
sl@0
   131
			if(argIdx > 0)
sl@0
   132
				{
sl@0
   133
				data.iArgType[argIdx] = static_cast <TArgType> (Math::Rand(data.iSeed) % ELastArgType);
sl@0
   134
				}
sl@0
   135
			switch(data.iArgType[argIdx])
sl@0
   136
				{
sl@0
   137
				case EIntArgType:
sl@0
   138
					data.iIntArg[argIdx] = Math::Rand(data.iSeed) % 9711;
sl@0
   139
					args.Set(argIdx, data.iIntArg[argIdx]);
sl@0
   140
					break;
sl@0
   141
				case ETextArgType:
sl@0
   142
					{
sl@0
   143
					TInt len = Math::Rand(data.iSeed) % KMaxDesArgLen;	
sl@0
   144
					data.iTextArg[argIdx].SetLength(len);
sl@0
   145
					args.Set(argIdx, &data.iTextArg[argIdx]);
sl@0
   146
					}
sl@0
   147
					break;
sl@0
   148
				case EBinArgType:
sl@0
   149
					{
sl@0
   150
					if(argIdx == 0)
sl@0
   151
						{
sl@0
   152
						//The operations ids are guaranteed to be sequential by logeng.dll implementation.
sl@0
   153
						TheLogIpcData.iOperationId = data.iIteration;
sl@0
   154
						//if(Math::Rand(data.iSeed) & 1)
sl@0
   155
						//	{
sl@0
   156
						//	TheLogIpcData.iOperationId = 0;
sl@0
   157
						//	}
sl@0
   158
						const TInt KTypeCnt = sizeof(KLogOpTypes) / sizeof(KLogOpTypes[0]);
sl@0
   159
						TInt typeIdx = Math::Rand(data.iSeed) % KTypeCnt;
sl@0
   160
						TheLogIpcData.iOperationType = KLogOpTypes[typeIdx];
sl@0
   161
						TheLogIpcData.iDataSlot1 = Math::Rand(data.iSeed);
sl@0
   162
						TheLogIpcData.iDataSlot2 = Math::Rand(data.iSeed);
sl@0
   163
						args.Set(argIdx, &TheLogIpcDataPtr);
sl@0
   164
						}
sl@0
   165
					else
sl@0
   166
						{
sl@0
   167
						TInt len = Math::Rand(data.iSeed) % KMaxDesArgLen;	
sl@0
   168
						data.iBinArg[argIdx].SetLength(len);
sl@0
   169
						args.Set(argIdx, &data.iBinArg[argIdx]);
sl@0
   170
						}
sl@0
   171
					}
sl@0
   172
					break;
sl@0
   173
				default:
sl@0
   174
					User::Panic(KPanicCategory2, KPanicCode2);
sl@0
   175
					break;
sl@0
   176
				}
sl@0
   177
			}
sl@0
   178
		//Send arguments
sl@0
   179
		//RDebug::Print(_L("##data.iFunction=%d\r\n"), data.iFunction);
sl@0
   180
		TRequestStatus stat;
sl@0
   181
		sess.Send(data.iFunction, args, stat);
sl@0
   182
		if(stat.Int() == KErrServerTerminated)
sl@0
   183
			{
sl@0
   184
			User::Panic(KPanicCategory, KPanicCode);
sl@0
   185
			}
sl@0
   186
		else if(stat.Int() == KRequestPending)
sl@0
   187
			{
sl@0
   188
			if(data.iFunction == ELogOperationInitiate)
sl@0
   189
				{
sl@0
   190
				//RDebug::Print(_L("##ELogOperationGetResult\r\n"));
sl@0
   191
				err = sess.Send(ELogOperationGetResult, args);
sl@0
   192
				if(err == KErrServerTerminated)
sl@0
   193
					{
sl@0
   194
					User::Panic(KPanicCategory, KPanicCode);
sl@0
   195
					}
sl@0
   196
				}
sl@0
   197
			else
sl@0
   198
				{
sl@0
   199
				//Give some time to the LogEng server to do something with that async request, then cancel it.
sl@0
   200
				//Otherwise, on a multi-core hardware, the LogEnd server will end up with a long queue of
sl@0
   201
				//pending requests, not cleared if the client side thread is panic'd. It will be a complete chaos.
sl@0
   202
				//RDebug::Print(_L("##data.iFunction=%d, wait and cancel async request\r\n"), data.iFunction);
sl@0
   203
				User::After(100000);
sl@0
   204
				TRequestStatus* s = &stat;
sl@0
   205
				User::RequestComplete(s, KErrCancel);
sl@0
   206
				}
sl@0
   207
			//RDebug::Print(_L("##---err=%d\r\n"), err);
sl@0
   208
			}
sl@0
   209
		}
sl@0
   210
sl@0
   211
	sess.Close();
sl@0
   212
sl@0
   213
	delete tc;	
sl@0
   214
	
sl@0
   215
	__UHEAP_MARKEND;
sl@0
   216
	
sl@0
   217
	return KErrNone;		
sl@0
   218
	}
sl@0
   219
sl@0
   220
//////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   221
//////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   222
sl@0
   223
//Thread function to detect a crash in the server.
sl@0
   224
//The server should run for the duration of the test.
sl@0
   225
// return KErrAbort: If failure to start server
sl@0
   226
// return KErrServerTerminated: If server process is terminated 
sl@0
   227
sl@0
   228
TInt ServerWatcherFunc(TAny* /*aData*/)
sl@0
   229
	{
sl@0
   230
	__UHEAP_MARK;
sl@0
   231
sl@0
   232
	_LIT(KLogEngServerName,"LogServ*");
sl@0
   233
sl@0
   234
	TInt err;
sl@0
   235
sl@0
   236
	// Start the server if not already running
sl@0
   237
	RLogSession sess;
sl@0
   238
	err = sess.Connect();
sl@0
   239
	if (err != KErrNone)
sl@0
   240
		return KErrAbort;
sl@0
   241
	sess.Close();
sl@0
   242
sl@0
   243
	TFindProcess findProcess(KLogEngServerName);
sl@0
   244
	TFullName result;
sl@0
   245
	if ( findProcess.Next(result) != KErrNone )
sl@0
   246
		return KErrAbort;
sl@0
   247
	
sl@0
   248
	RProcess server;
sl@0
   249
	if( server.Open(findProcess, EOwnerProcess) != KErrNone)
sl@0
   250
		return KErrAbort;
sl@0
   251
sl@0
   252
	TRequestStatus status;
sl@0
   253
	server.Logon(status);	
sl@0
   254
 	User::WaitForRequest(status);
sl@0
   255
	
sl@0
   256
	server.Close();
sl@0
   257
	
sl@0
   258
	__UHEAP_MARKEND;
sl@0
   259
sl@0
   260
	return KErrServerTerminated;
sl@0
   261
	}
sl@0
   262
sl@0
   263
sl@0
   264
/**
sl@0
   265
@SYMTestCaseID			PDS-LOGENG-UT-4045
sl@0
   266
@SYMTestCaseDesc		In a loop, where the loop iterations are less than KTestIterCount (5000 at the moment), 
sl@0
   267
						the test creates a worker thread, which will behave as a malicious client. 
sl@0
   268
						If the worker thread crashes the LogEng server, then the worker thread
sl@0
   269
						dies notifying the main thread about the LogEng server crash. The main thread prints the 
sl@0
   270
						values used in the last IPC call and crashes the test.
sl@0
   271
@SYMTestPriority		High
sl@0
   272
@SYMTestActions			LogEng, Malicious client simulation test.
sl@0
   273
@SYMTestExpectedResults Test must not fail
sl@0
   274
@SYMREQ					REQ12746
sl@0
   275
*/	
sl@0
   276
void BadClientTest()
sl@0
   277
	{
sl@0
   278
	// Start a thread to watch the server process
sl@0
   279
	RThread serverWatcher;
sl@0
   280
	TInt err = serverWatcher.Create(_L("ServerWatcher"), &ServerWatcherFunc, 0x2000, 0x1000, 0x10000, NULL, EOwnerProcess);
sl@0
   281
	TRequestStatus serverStatus;
sl@0
   282
	serverWatcher.Logon(serverStatus);
sl@0
   283
	serverWatcher.Resume();
sl@0
   284
sl@0
   285
	TThreadData* p = new TThreadData;
sl@0
   286
	TEST(p != NULL);
sl@0
   287
	TThreadData& data = *p;
sl@0
   288
	data.iFunction = 0;
sl@0
   289
	TTime now;
sl@0
   290
	now.UniversalTime();
sl@0
   291
	data.iSeed = now.Int64();
sl@0
   292
sl@0
   293
	for(data.iIteration=0;data.iIteration<KTestIterCount;++data.iIteration)
sl@0
   294
		{
sl@0
   295
		PrintIterationCount(data.iIteration);
sl@0
   296
		
sl@0
   297
		//Run the malicious client (one test thread which will try to crash the LogEng server)
sl@0
   298
		User::After(200000);
sl@0
   299
		_LIT(KTestThreadName, "TLBCThr");
sl@0
   300
		RThread thread;
sl@0
   301
		err = thread.Create(KTestThreadName, &ThreadFunc1, 0x2000, 0x1000, 0x10000, &data, EOwnerProcess);
sl@0
   302
		if(err == KErrAlreadyExists)
sl@0
   303
			{
sl@0
   304
			TheTest.Printf(_L("##Iteration %d. Function %d. Thread \"%S\" already exists!\r\n"), data.iIteration, data.iFunction, &KTestThreadName);
sl@0
   305
			for(TInt i=0;i<KMaxMessageArguments;++i)
sl@0
   306
				{
sl@0
   307
				TheTest.Printf(_L("##Arg %d, Type %d\r\n"), i + 1, data.iArgType[i]);
sl@0
   308
				switch(data.iArgType[i])
sl@0
   309
					{
sl@0
   310
					case EIntArgType:
sl@0
   311
					    TheTest.Printf(_L("Integer, value=%d\r\n"), data.iIntArg[i]);
sl@0
   312
						break;
sl@0
   313
					case ETextArgType:
sl@0
   314
					    TheTest.Printf(_L("Text, length=%d\r\n"), data.iTextArg[i].Length());
sl@0
   315
						break;
sl@0
   316
					case EBinArgType:
sl@0
   317
					    TheTest.Printf(_L("Binary, length=%d\r\n"), data.iBinArg[i].Length());
sl@0
   318
						break;
sl@0
   319
					default:
sl@0
   320
					    TheTest.Printf(_L("Invalid argument type: %d\r\n"), data.iArgType[i]);
sl@0
   321
						break;
sl@0
   322
					}
sl@0
   323
				}
sl@0
   324
			break;
sl@0
   325
			}
sl@0
   326
		TEST2(err, KErrNone);
sl@0
   327
		User::SetJustInTime(EFalse);		
sl@0
   328
		TRequestStatus status;
sl@0
   329
		thread.Logon(status);
sl@0
   330
		TEST2(status.Int(), KRequestPending);
sl@0
   331
		thread.Resume();
sl@0
   332
		User::WaitForRequest(status, serverStatus);
sl@0
   333
sl@0
   334
		// If the Server has crashed then we must fail		
sl@0
   335
		if (serverStatus != KRequestPending) 
sl@0
   336
			{
sl@0
   337
			TheTest.Printf(_L("##Iteration=%d, Function=%d, Status=%d, Server Status=%d\r\n"), data.iIteration, data.iFunction, status.Int(), serverStatus.Int());
sl@0
   338
			}
sl@0
   339
sl@0
   340
		TExitType exitType = thread.ExitType();
sl@0
   341
		TInt exitReason = thread.ExitReason();
sl@0
   342
		thread.Close();
sl@0
   343
		User::SetJustInTime(ETrue);
sl@0
   344
		
sl@0
   345
		if(exitType == EExitPanic || serverStatus != KRequestPending)
sl@0
   346
			{
sl@0
   347
			if(exitReason == KPanicCode || serverStatus != KRequestPending)
sl@0
   348
				{
sl@0
   349
				TheTest.Printf(_L("##Server terminated!\r\n"));
sl@0
   350
				TheTest.Printf(_L("##Iteration=%d, Function=%d, iOperationType=%d, iDataSlot1=%d, iDataSlot2=%d\r\n"), 
sl@0
   351
						data.iIteration, data.iFunction,
sl@0
   352
						TheLogIpcData.iOperationType,
sl@0
   353
						TheLogIpcData.iDataSlot1,
sl@0
   354
						TheLogIpcData.iDataSlot2);
sl@0
   355
				for(TInt i=0;i<KMaxMessageArguments;++i)
sl@0
   356
					{
sl@0
   357
					TheTest.Printf(_L("##Arg %d, Type %d\r\n"), i + 1, data.iArgType[i]);
sl@0
   358
					switch(data.iArgType[i])
sl@0
   359
						{
sl@0
   360
						case EIntArgType:
sl@0
   361
						    TheTest.Printf(_L("Integer, value=%d\r\n"), data.iIntArg[i]);
sl@0
   362
							break;
sl@0
   363
						case ETextArgType:
sl@0
   364
						    TheTest.Printf(_L("Text, length=%d\r\n"), data.iTextArg[i].Length());
sl@0
   365
							break;
sl@0
   366
						case EBinArgType:
sl@0
   367
						    TheTest.Printf(_L("Binary, length=%d\r\n"), data.iBinArg[i].Length());
sl@0
   368
							break;
sl@0
   369
						default:
sl@0
   370
						    TheTest.Printf(_L("Invalid argument type: %d\r\n"), data.iArgType[i]);
sl@0
   371
							break;
sl@0
   372
						}
sl@0
   373
					}
sl@0
   374
				TEST(0);
sl@0
   375
				}
sl@0
   376
			}
sl@0
   377
		}//for
sl@0
   378
	delete p;
sl@0
   379
sl@0
   380
	// Check to see if the server crashed and not detected by client
sl@0
   381
	TEST(serverStatus.Int() == KRequestPending);
sl@0
   382
	serverWatcher.Kill(KErrCancel);
sl@0
   383
	serverWatcher.Close();
sl@0
   384
		
sl@0
   385
	}
sl@0
   386
sl@0
   387
/**
sl@0
   388
@SYMTestCaseID			PDS-LOGENG-UT-4044
sl@0
   389
@SYMTestCaseDesc		LogEng server startup - file I/O error simulation test.
sl@0
   390
						The test case shuts down the LogEng server in debug mode.
sl@0
   391
						Then attempts to connect to the server in a file I/O error simulation
sl@0
   392
						loop.
sl@0
   393
@SYMTestPriority		High
sl@0
   394
@SYMTestActions			LogEng server startup - file I/O error simulation test.
sl@0
   395
@SYMTestExpectedResults Test must not fail
sl@0
   396
@SYMREQ					REQ12746
sl@0
   397
*/	
sl@0
   398
void LogEngSrvStartupFileIoErrTest()
sl@0
   399
	{
sl@0
   400
	//Shut down the server if it is running
sl@0
   401
	RLogSession sess;
sl@0
   402
	TInt err = sess.Connect();
sl@0
   403
	TEST2(err, KErrNone);
sl@0
   404
sl@0
   405
	err = sess.Send(ELogMakeTransient, TIpcArgs(1));
sl@0
   406
	TEST2(err, KErrNone);
sl@0
   407
	sess.Close();
sl@0
   408
	
sl@0
   409
	//The shutdown delay is 2 seconds (defined in LogServShutdownTimer.h file). In this csase 5 seconds is more than enough.
sl@0
   410
	User::After(5000000);
sl@0
   411
	
sl@0
   412
	RFs fs;
sl@0
   413
	err = fs.Connect();
sl@0
   414
	TEST2(err, KErrNone);
sl@0
   415
sl@0
   416
	TBool finished = EFalse;
sl@0
   417
	TInt failCount = 0;
sl@0
   418
sl@0
   419
	while(!finished)
sl@0
   420
		{
sl@0
   421
		fs.SetErrorCondition(KErrCorrupt, ++failCount);
sl@0
   422
		TInt err = sess.Connect();
sl@0
   423
		fs.SetErrorCondition(KErrNone, 0);
sl@0
   424
		sess.Close();
sl@0
   425
		if(err == KErrNone)
sl@0
   426
			{
sl@0
   427
			finished = ETrue;
sl@0
   428
			}
sl@0
   429
		else
sl@0
   430
			{
sl@0
   431
			TEST2(err, KErrCorrupt);
sl@0
   432
			}
sl@0
   433
		}
sl@0
   434
sl@0
   435
	fs.Close();
sl@0
   436
	TheTest.Printf(_L("===LogEng Server Startup File I/O error simularion test succeeded at iteration %d\n"), failCount);
sl@0
   437
	}
sl@0
   438
sl@0
   439
void DoTests()
sl@0
   440
	{
sl@0
   441
	TheTest.Start(_L(" @SYMTestCaseID:PDS-LOGENG-UT-4045: Bad client test"));
sl@0
   442
	BadClientTest();
sl@0
   443
#ifdef _DEBUG	
sl@0
   444
	TheTest.Next(_L(" @SYMTestCaseID:PDS-LOGENG-UT-4044: LogEng Server Startup - File I/O error simulation test"));
sl@0
   445
	LogEngSrvStartupFileIoErrTest();
sl@0
   446
#endif	
sl@0
   447
	}
sl@0
   448
sl@0
   449
TInt E32Main()
sl@0
   450
	{
sl@0
   451
	TheTest.Title();
sl@0
   452
	
sl@0
   453
	CTrapCleanup* tc = CTrapCleanup::New();
sl@0
   454
	
sl@0
   455
	__UHEAP_MARK;
sl@0
   456
	
sl@0
   457
	DoTests();
sl@0
   458
sl@0
   459
	__UHEAP_MARKEND;
sl@0
   460
	
sl@0
   461
	TheTest.End();
sl@0
   462
	TheTest.Close();
sl@0
   463
	
sl@0
   464
	delete tc;
sl@0
   465
sl@0
   466
	User::Heap().Check();
sl@0
   467
	return KErrNone;
sl@0
   468
	}