os/mm/mmtestenv/mmtestfw/Source/TestFramework/parseline.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) 2002-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
// This module contains CParseLine and CSuiteDll classes
sl@0
    15
// CParseLine contains the functions required to execute
sl@0
    16
// a line of test script file.
sl@0
    17
// CSuiteDll objects contains information about test suite
sl@0
    18
// dlls that have been loaded.
sl@0
    19
// 
sl@0
    20
//
sl@0
    21
sl@0
    22
// system includes
sl@0
    23
#include <f32file.h>
sl@0
    24
sl@0
    25
// test system includes
sl@0
    26
#include "TestFramework.h"
sl@0
    27
#include "script.h"
sl@0
    28
#include "parseline.h"
sl@0
    29
#include "Filename.h" 
sl@0
    30
sl@0
    31
#include "parseline.inl"
sl@0
    32
sl@0
    33
const TInt KTimeIncrement = 100*1000000; // max wait interval in micro-seconds (100s)
sl@0
    34
const TUint KMaxThreadAttempts = 128;	// max number of times to attempt to create a thread
sl@0
    35
sl@0
    36
/**
sl@0
    37
 *
sl@0
    38
 * File path literals
sl@0
    39
 *
sl@0
    40
 * @xxxx
sl@0
    41
 *
sl@0
    42
 */
sl@0
    43
_LIT(KTxtDLLpath, "c:\\;c:\\system\\libs;d:\\;d:\\system\\libs;e:\\;e:\\system\\libs;z:\\;z:\\system\\libs");
sl@0
    44
sl@0
    45
/**
sl@0
    46
 *
sl@0
    47
 * Script parameter defaults
sl@0
    48
 *
sl@0
    49
 * @xxxx
sl@0
    50
 *
sl@0
    51
 */
sl@0
    52
//const TInt KTestGuardTimerDefault = 1000L;	// EABI warning removal
sl@0
    53
const TInt KPanicGuardTimerDefault = 1000000L;
sl@0
    54
const TInt KPanicExitReasonDefault = 0;
sl@0
    55
sl@0
    56
/**
sl@0
    57
 *
sl@0
    58
 * CParseLine first-phase constructor
sl@0
    59
 *
sl@0
    60
 * @xxxx
sl@0
    61
 *
sl@0
    62
 */
sl@0
    63
CParseLine::CParseLine(const TDesC& aMatchString)
sl@0
    64
	:iTestVerdict(EPass), iMatchString(aMatchString)
sl@0
    65
	{
sl@0
    66
	}
sl@0
    67
sl@0
    68
/**
sl@0
    69
 *
sl@0
    70
 * CParseLine second-phase constructor 
sl@0
    71
 *
sl@0
    72
 * @param	"CScript* aScript"
sl@0
    73
 *			The script to be parsed
sl@0
    74
 *
sl@0
    75
 * @param	"CTestUtils* aTestUtils"
sl@0
    76
 *			The TestUtils object to use
sl@0
    77
 *
sl@0
    78
 * @param	"CLog* aLog"
sl@0
    79
 *			The logger to use
sl@0
    80
 *
sl@0
    81
 * @xxxx
sl@0
    82
 *
sl@0
    83
 */
sl@0
    84
void CParseLine::ConstructL(CScript* aScript, CTestUtils* aTestUtils, CLog* aLog, TInt64 aGuardTimer)
sl@0
    85
	{
sl@0
    86
	// create a new Array to store the test steps in
sl@0
    87
	iArrayLoadedSuiteDll = new(ELeave) CArrayPtrFlat<CSuiteDll>(1);
sl@0
    88
sl@0
    89
	iScript = aScript;
sl@0
    90
	iTestUtils = aTestUtils;
sl@0
    91
	iLog = aLog;
sl@0
    92
	iGuardTimer = aGuardTimer;
sl@0
    93
	iSeverity = ESevrAll;
sl@0
    94
	iBreakOnError = EFalse;
sl@0
    95
	}
sl@0
    96
sl@0
    97
/**
sl@0
    98
 *
sl@0
    99
 * CParseLine static constructor 
sl@0
   100
 *
sl@0
   101
 * @param	"CScript* aScript"
sl@0
   102
 *			The script to be parsed
sl@0
   103
 *
sl@0
   104
 * @param	"CTestUtils* aTestUtils"
sl@0
   105
 *			The TestUtils object to use
sl@0
   106
 *
sl@0
   107
 * @param	"CLog* aLog"
sl@0
   108
 *			The logger to use
sl@0
   109
 *
sl@0
   110
 * @xxxx
sl@0
   111
 *
sl@0
   112
 */
sl@0
   113
CParseLine* CParseLine::NewL(CScript* aScript, CTestUtils* aTestUtils, CLog* aLog, TInt64 aGuardTimer, const TDesC& aMatchString)
sl@0
   114
	{
sl@0
   115
	CParseLine* self = new(ELeave) CParseLine(aMatchString);
sl@0
   116
	CleanupStack::PushL(self);
sl@0
   117
	self->ConstructL(aScript, aTestUtils, aLog, aGuardTimer);
sl@0
   118
	CleanupStack::Pop();
sl@0
   119
	return self;
sl@0
   120
	}
sl@0
   121
sl@0
   122
/**
sl@0
   123
 *
sl@0
   124
 * CParseLine destructor 
sl@0
   125
 *
sl@0
   126
 * @xxxx
sl@0
   127
 *
sl@0
   128
 */
sl@0
   129
CParseLine::~CParseLine()
sl@0
   130
	{
sl@0
   131
sl@0
   132
	// unload DLLs and their records
sl@0
   133
	if (iArrayLoadedSuiteDll)
sl@0
   134
		{
sl@0
   135
		// delete all objects in iArrayLoadedSuiteDll
sl@0
   136
		// the destructors will unload any loaded DLLS
sl@0
   137
		iArrayLoadedSuiteDll->ResetAndDestroy();
sl@0
   138
		delete iArrayLoadedSuiteDll;
sl@0
   139
		}
sl@0
   140
sl@0
   141
	}
sl@0
   142
sl@0
   143
/**
sl@0
   144
 *
sl@0
   145
 * Process a single line from the script file.
sl@0
   146
 *
sl@0
   147
 * @param	"const TDesC8& aNarrowline"
sl@0
   148
 *			The script line
sl@0
   149
 *
sl@0
   150
 * @param	"TInt8 aLineNo"
sl@0
   151
 *			The script line number
sl@0
   152
 *
sl@0
   153
 * @xxxx
sl@0
   154
 *
sl@0
   155
 */
sl@0
   156
void CParseLine::ProcessLineL(const TDesC8& aNarrowline, TInt aLineNo)
sl@0
   157
	{
sl@0
   158
	// make a local unicode buffer
sl@0
   159
	TPtr16 lineBuf(REINTERPRET_CAST(TUint16*,User::AllocLC(KMaxLenScriptLine*2)), 0, KMaxLenScriptLine);
sl@0
   160
	lineBuf.Fill('\0', KMaxLenScriptLine);
sl@0
   161
sl@0
   162
	// convert the narrow script file to Unicode
sl@0
   163
	// TBC find a better way to do this
sl@0
   164
	CFileName* testnameU = CFileName::NewLC();
sl@0
   165
	testnameU->Copy(aNarrowline);
sl@0
   166
sl@0
   167
	// find the end of the line
sl@0
   168
	TInt end = testnameU->Locate('\n');
sl@0
   169
sl@0
   170
	// copy the line into lineBuf
sl@0
   171
	if ((end != KErrNotFound) && (end < KMaxLenScriptLine))
sl@0
   172
		lineBuf = testnameU->Left(end - 1);
sl@0
   173
	else
sl@0
   174
		lineBuf = testnameU->FileName();
sl@0
   175
sl@0
   176
	// destroy filename
sl@0
   177
	CleanupStack::PopAndDestroy(testnameU);
sl@0
   178
sl@0
   179
	// the parser relies on spaces between tokens. Commas are
sl@0
   180
	// allowed but are just replaced with spaces
sl@0
   181
	TInt findComma = lineBuf.Locate(TChar(','));
sl@0
   182
	while (findComma != KErrNotFound )
sl@0
   183
		{
sl@0
   184
		// found a comma so replace with space
sl@0
   185
		lineBuf.Replace(findComma, 1, _L(" "));
sl@0
   186
		findComma = lineBuf.Locate(TChar(','));
sl@0
   187
		}
sl@0
   188
sl@0
   189
	// for debugging display the line with a line no
sl@0
   190
#ifdef SCRIPT_DEBUG
sl@0
   191
	INFO_PRINTF3(_L("Line:%d %S "), aLineNo, &lineBuf);
sl@0
   192
#endif
sl@0
   193
sl@0
   194
	// if there has been a failure and the user has selected
sl@0
   195
	// x then the next commands in the script are skipped until
sl@0
   196
	// a test complete statement is found
sl@0
   197
	if (iBreakOnError)
sl@0
   198
		{
sl@0
   199
		if (lineBuf.FindF(_L("TEST_COMPLETE")) == 0)
sl@0
   200
			{
sl@0
   201
			TestComplete(lineBuf);
sl@0
   202
			// reset flag now test complete found
sl@0
   203
			iBreakOnError = EFalse;
sl@0
   204
			}
sl@0
   205
sl@0
   206
		CleanupStack::PopAndDestroy(); // linebuf
sl@0
   207
		// do not process the rest of the line
sl@0
   208
		return;
sl@0
   209
		}
sl@0
   210
sl@0
   211
	// check the line for command keywords
sl@0
   212
	if ((lineBuf.Find(_L("//")) == 0) || (lineBuf.Find(_L("#")) == 0))
sl@0
   213
		{
sl@0
   214
		// ignore comments
sl@0
   215
		}
sl@0
   216
	else if (lineBuf.FindF(_L("LOAD_SUITE")) == 0)
sl@0
   217
		{
sl@0
   218
		LoadSuiteL(lineBuf);
sl@0
   219
		}
sl@0
   220
	else if (lineBuf.FindF(_L("RUN_SCRIPT")) == 0)
sl@0
   221
		{
sl@0
   222
		RunScriptL(lineBuf);
sl@0
   223
		}
sl@0
   224
	else if (lineBuf.FindF(_L("RUN_TEST_STEP")) == 0)
sl@0
   225
		{
sl@0
   226
		RunTestStep(lineBuf);
sl@0
   227
		}
sl@0
   228
	else if (lineBuf.FindF(_L("RUN_PANIC_STEP")) == 0)
sl@0
   229
		{
sl@0
   230
		RunPanicTestStep(lineBuf);
sl@0
   231
		}
sl@0
   232
	else if (lineBuf.FindF(_L("RUN_TERMINATION_STEP")) == 0)
sl@0
   233
		{
sl@0
   234
		RunTerminationTestStep(lineBuf);
sl@0
   235
		}
sl@0
   236
	else if (lineBuf.FindF(_L("RUN_UTILS")) == 0)
sl@0
   237
		{
sl@0
   238
		RunUtil(lineBuf);
sl@0
   239
		}
sl@0
   240
	else if (lineBuf.FindF(_L("RUN_PROGRAM")) == 0)
sl@0
   241
		{
sl@0
   242
		RunProgram(lineBuf);
sl@0
   243
		}
sl@0
   244
	else if (lineBuf.FindF(_L("UNLOAD")) == 0)
sl@0
   245
		{
sl@0
   246
		Unload();
sl@0
   247
		}
sl@0
   248
	else if (lineBuf.FindF(_L("HEAP_MARK")) == 0)
sl@0
   249
		{
sl@0
   250
		HeapMark();
sl@0
   251
		}
sl@0
   252
	else if (lineBuf.FindF(_L("HEAP_CHECK")) == 0)
sl@0
   253
		{
sl@0
   254
		HeapCheck();
sl@0
   255
		}
sl@0
   256
	else if (lineBuf.FindF(_L("REQUEST_MARK")) == 0)
sl@0
   257
		{
sl@0
   258
		RequestMark();	
sl@0
   259
		}
sl@0
   260
	else if (lineBuf.FindF(_L("REQUEST_CHECK")) == 0)
sl@0
   261
		{
sl@0
   262
		RequestCheck();
sl@0
   263
		}
sl@0
   264
	else if (lineBuf.FindF(_L("HANDLES_MARK")) == 0)
sl@0
   265
		{
sl@0
   266
		HandlesMark();
sl@0
   267
		}
sl@0
   268
	else if (lineBuf.FindF(_L("HANDLES_CHECK")) == 0)
sl@0
   269
		{
sl@0
   270
		HandlesCheck();
sl@0
   271
		}
sl@0
   272
	else if (lineBuf.FindF(_L("PRINT")) == 0)
sl@0
   273
		{
sl@0
   274
		ScriptPrint(lineBuf);
sl@0
   275
		}
sl@0
   276
	else if (lineBuf.FindF(_L("DELAY")) == 0)
sl@0
   277
		{
sl@0
   278
		Delay(lineBuf);
sl@0
   279
		}
sl@0
   280
	else if (lineBuf.FindF(_L("SEVERITY")) == 0)
sl@0
   281
		{
sl@0
   282
		SetSeverity(lineBuf);
sl@0
   283
		}
sl@0
   284
	else if (lineBuf.FindF(_L("PAUSE_AT_END")) == 0)
sl@0
   285
		{
sl@0
   286
		// if implemented, add iScript->iPauseAtEnd = ETrue;
sl@0
   287
		WARN_PRINTF1(_L("Warning : PAUSE_AT_END not implemented"));
sl@0
   288
		}
sl@0
   289
	else if (lineBuf.FindF(_L("MULTITHREAD")) == 0)
sl@0
   290
		{
sl@0
   291
		WARN_PRINTF1(_L("Warning : MULTITHREAD keyword no longer required"));
sl@0
   292
		}
sl@0
   293
	else if (lineBuf.FindF(_L("SINGLETHREAD")) == 0)
sl@0
   294
		{
sl@0
   295
		ERR_PRINTF1(_L("Error : Single thread operation no longer supported"));
sl@0
   296
		}
sl@0
   297
	else if (lineBuf.FindF(_L("PAUSE")) == 0)
sl@0
   298
		{
sl@0
   299
		iScript->Pause();
sl@0
   300
		}
sl@0
   301
	else if (lineBuf.FindF(_L("BREAK_ON_ERROR")) == 0)
sl@0
   302
		{
sl@0
   303
		// if the current test verdict is not PASS
sl@0
   304
		// give the user the chance to quit
sl@0
   305
		if ( iTestVerdict != EPass )
sl@0
   306
			iBreakOnError = iScript->BreakOnError();
sl@0
   307
		}
sl@0
   308
	else if (lineBuf.FindF(_L("TEST_COMPLETE")) == 0)
sl@0
   309
		{
sl@0
   310
		// use Tlex to decode the cmd line
sl@0
   311
		TestComplete(lineBuf);
sl@0
   312
		}
sl@0
   313
	else if (lineBuf.FindF(_L("LOG_SETTINGS")) == 0)
sl@0
   314
		{
sl@0
   315
		// use Tlex to decode the cmd line
sl@0
   316
		LogSettings(lineBuf);
sl@0
   317
		}
sl@0
   318
	else if (lineBuf.Length() == 0)
sl@0
   319
		{
sl@0
   320
		// ignore blank lines
sl@0
   321
		}
sl@0
   322
	else
sl@0
   323
		{
sl@0
   324
		// failed to decode line
sl@0
   325
		ERR_PRINTF3(_L("Error in script line:%d - \'%S\'"), aLineNo, &lineBuf);
sl@0
   326
		}
sl@0
   327
sl@0
   328
	CleanupStack::PopAndDestroy(); // linebuf
sl@0
   329
	}
sl@0
   330
sl@0
   331
/**
sl@0
   332
 *
sl@0
   333
 * Implements the TEST_COMPLETE script command.
sl@0
   334
 *
sl@0
   335
 * @param	"const TDesC& aText"
sl@0
   336
 *			The script line
sl@0
   337
 *
sl@0
   338
 * @xxxx
sl@0
   339
 *
sl@0
   340
 */
sl@0
   341
void CParseLine::TestComplete(const TDesC& aText)
sl@0
   342
	{
sl@0
   343
	// use Tlex to decode the cmd line
sl@0
   344
	TLex lex(aText);
sl@0
   345
sl@0
   346
	// start at the begining
sl@0
   347
	TPtrC token = lex.NextToken();
sl@0
   348
sl@0
   349
	// get suite name, if any
sl@0
   350
	token.Set(lex.NextToken());
sl@0
   351
sl@0
   352
	if (token.Length() != 0)
sl@0
   353
		{
sl@0
   354
		TBuf<KMaxLenTestSuiteName> currentSuiteName;
sl@0
   355
		currentSuiteName = token;
sl@0
   356
sl@0
   357
		// get step name, if any
sl@0
   358
		token.Set(lex.NextToken());
sl@0
   359
	
sl@0
   360
		if (token.Length() != 0)
sl@0
   361
			{	
sl@0
   362
			iCurrentStepName = token;
sl@0
   363
			iCurrentSuiteName = currentSuiteName;
sl@0
   364
			}
sl@0
   365
		else
sl@0
   366
			{
sl@0
   367
			// failed to decode line - require 0 or 2 parameters exactly
sl@0
   368
			// use last suite/step name, return fail
sl@0
   369
			ERR_PRINTF2(_L("Error in script line: \'%S\'"), &aText);
sl@0
   370
			iTestVerdict = EFail;
sl@0
   371
			}
sl@0
   372
		}
sl@0
   373
sl@0
   374
	if (!iSkip)
sl@0
   375
		{
sl@0
   376
		// add the current result to the script
sl@0
   377
		iScript->AddResult(iTestVerdict);		
sl@0
   378
		}
sl@0
   379
		
sl@0
   380
	// reset for next test
sl@0
   381
	iTestVerdict = EPass;
sl@0
   382
	}
sl@0
   383
sl@0
   384
/**
sl@0
   385
 *
sl@0
   386
 * Implements the PRINT script command.
sl@0
   387
 *
sl@0
   388
 * @param	"const TDesC& aText"
sl@0
   389
 *			The script line
sl@0
   390
 *
sl@0
   391
 * @xxxx
sl@0
   392
 *
sl@0
   393
 */
sl@0
   394
void CParseLine::ScriptPrint(const TDesC& aText)
sl@0
   395
	{
sl@0
   396
	// display the text after the PRINT and 1 space = 6
sl@0
   397
	INFO_PRINTF2(_L("%s "), (aText.Ptr() + 6));
sl@0
   398
	}
sl@0
   399
sl@0
   400
/**
sl@0
   401
 *
sl@0
   402
 * Implements the DELAY script command.
sl@0
   403
 *
sl@0
   404
 * @param	"const TDesC& aText"
sl@0
   405
 *			The script line
sl@0
   406
 *
sl@0
   407
 * @xxxx
sl@0
   408
 *
sl@0
   409
 */
sl@0
   410
void CParseLine::Delay(const TDesC& aText)
sl@0
   411
	{
sl@0
   412
	// if the test has already failed skip the delay
sl@0
   413
	if (iTestVerdict != EPass)
sl@0
   414
		{
sl@0
   415
		WARN_PRINTF1(_L("Skipped delay as test has already failed"));
sl@0
   416
		return;
sl@0
   417
		}
sl@0
   418
sl@0
   419
	// get the required time for the delay
sl@0
   420
	// first get the value as a string
sl@0
   421
	TLex timeOut(aText);
sl@0
   422
	timeOut.NextToken();
sl@0
   423
	TPtrC token = timeOut.NextToken();
sl@0
   424
sl@0
   425
	// convert the value into a TInt
sl@0
   426
	TLex lexTime(token);
sl@0
   427
	TInt64 guardTimerValue;
sl@0
   428
	if (lexTime.Val(guardTimerValue) != KErrNone  )
sl@0
   429
		{
sl@0
   430
		ERR_PRINTF2(_L("Error in guard timer value : could not decode \'%S\' as value"), 
sl@0
   431
					&token);
sl@0
   432
		return;
sl@0
   433
		}
sl@0
   434
sl@0
   435
	INFO_PRINTF2(_L("Delay for %ld mS"), guardTimerValue);
sl@0
   436
sl@0
   437
	// wait for the required delay
sl@0
   438
	User::After(I64INT(guardTimerValue) * 1000);
sl@0
   439
	}
sl@0
   440
	
sl@0
   441
/**
sl@0
   442
 *
sl@0
   443
 * Implements the SEVERITY script command.
sl@0
   444
 *
sl@0
   445
 * @param	"const TDesC& aText"
sl@0
   446
 *			The script line
sl@0
   447
 *
sl@0
   448
 * @xxxx
sl@0
   449
 *
sl@0
   450
 */
sl@0
   451
void CParseLine::SetSeverity(const TDesC& aText)
sl@0
   452
	{
sl@0
   453
	// get the required time for the delay
sl@0
   454
	// first get the value as a string
sl@0
   455
	TLex severityOut(aText);
sl@0
   456
	severityOut.NextToken();
sl@0
   457
	TPtrC token = severityOut.NextToken();
sl@0
   458
sl@0
   459
	// convert the value into a TInt
sl@0
   460
	TLex lexSeverity(token);
sl@0
   461
	TInt severityValue = ESevrAll;
sl@0
   462
	if (lexSeverity.Val(severityValue) != KErrNone)
sl@0
   463
		{
sl@0
   464
		ERR_PRINTF2(_L("Error in severity level value : could not decode \'%S\' as value"),
sl@0
   465
					&token);
sl@0
   466
		return;
sl@0
   467
		}
sl@0
   468
sl@0
   469
	// check severity value to ensure that only bitmasks in use are set...
sl@0
   470
	if(!LogSeverity::IsValid(severityValue))
sl@0
   471
		{
sl@0
   472
		ERR_PRINTF1(_L("Error in severity value : out of range"));
sl@0
   473
		return;
sl@0
   474
		}
sl@0
   475
	else
sl@0
   476
		{
sl@0
   477
		iSeverity = severityValue;
sl@0
   478
sl@0
   479
		TInt noOfDlls = iArrayLoadedSuiteDll->Count();
sl@0
   480
		for ( TInt i = 0; i < noOfDlls; i++)
sl@0
   481
			{
sl@0
   482
			CSuiteDll* ptrSuite = iArrayLoadedSuiteDll->At(i);
sl@0
   483
			CTestSuite* testSuite = ptrSuite->Suite();
sl@0
   484
			testSuite->SetSeverity(iSeverity);
sl@0
   485
			}
sl@0
   486
		}
sl@0
   487
sl@0
   488
	INFO_PRINTF2(_L("Severity is set to %d"), severityValue);
sl@0
   489
	}
sl@0
   490
sl@0
   491
/**
sl@0
   492
 *
sl@0
   493
 * Implements the RUN_SCRIPT script command.
sl@0
   494
 *
sl@0
   495
 * @param	"const TDesC& aText"
sl@0
   496
 *			The script line
sl@0
   497
 *
sl@0
   498
 * @xxxx
sl@0
   499
 *
sl@0
   500
 */
sl@0
   501
void CParseLine::RunScriptL(const TDesC& aText)
sl@0
   502
	{
sl@0
   503
	// use Tlex to decode the cmd line
sl@0
   504
	TLex lex(aText);
sl@0
   505
sl@0
   506
	// start at the begining
sl@0
   507
	TPtrC token=lex.NextToken();
sl@0
   508
sl@0
   509
	// step over the keyword
sl@0
   510
	token.Set(lex.NextToken());
sl@0
   511
sl@0
   512
	// format for printing
sl@0
   513
	INFO_PRINTF2(_L("RUN_SCRIPT %S"), &token);
sl@0
   514
sl@0
   515
	// create a new Script object (but use the current parser
sl@0
   516
	// as it has the dll loaded record)
sl@0
   517
	CScript* newScript=CScript::NewLC(this, iTestUtils, iLog, iGuardTimer, iMatchString);
sl@0
   518
sl@0
   519
	// read in the script file
sl@0
   520
	CFileName* scriptFileName = CFileName::NewLC();
sl@0
   521
	*scriptFileName = token;
sl@0
   522
sl@0
   523
	if (newScript->OpenScriptFile(scriptFileName))
sl@0
   524
		{
sl@0
   525
		// process it
sl@0
   526
		iTestVerdict = newScript->ExecuteScriptL();
sl@0
   527
sl@0
   528
		// don't bother logging verdicts for scripts - not really useful
sl@0
   529
		// add results from the new script to the owner script
sl@0
   530
		iScript->AddResult(newScript);
sl@0
   531
		}
sl@0
   532
	else
sl@0
   533
		{
sl@0
   534
		// failed to find script so verdict incloncusive
sl@0
   535
		iTestVerdict = EInconclusive;
sl@0
   536
		}
sl@0
   537
sl@0
   538
	CleanupStack::PopAndDestroy(scriptFileName);
sl@0
   539
	CleanupStack::PopAndDestroy(newScript);
sl@0
   540
	}
sl@0
   541
sl@0
   542
/**
sl@0
   543
 *
sl@0
   544
 * Implements the RUN_TEST_STEP script command.
sl@0
   545
 *
sl@0
   546
 * @param	"const TDesC& aText"
sl@0
   547
 *			The script line
sl@0
   548
 *
sl@0
   549
 * @xxxx
sl@0
   550
 *
sl@0
   551
 */
sl@0
   552
void CParseLine::RunTestStep(const TDesC& aText)
sl@0
   553
	{
sl@0
   554
	// use TLex to decode the cmd line
sl@0
   555
	TLex lex(aText);
sl@0
   556
sl@0
   557
	// step over keyword
sl@0
   558
	lex.NextToken();
sl@0
   559
sl@0
   560
	// get guard timer
sl@0
   561
	TPtrC timeout;
sl@0
   562
	timeout.Set(lex.NextToken());
sl@0
   563
sl@0
   564
	// get the other parameters
sl@0
   565
	TPtrC suite, step, config, name, paramSet;
sl@0
   566
	suite.Set(lex.NextToken());
sl@0
   567
	step.Set(lex.NextToken());
sl@0
   568
	config.Set(lex.NextToken());
sl@0
   569
	name.Set(lex.NextToken());
sl@0
   570
	if (name.Length()==0)
sl@0
   571
		{
sl@0
   572
		// name is optional, if not given use step 
sl@0
   573
		name.Set(step);
sl@0
   574
		}
sl@0
   575
	paramSet.Set(lex.NextToken()); 
sl@0
   576
	if (paramSet.Length()==0)
sl@0
   577
		{
sl@0
   578
		// paramSet is optional, if not given use name 
sl@0
   579
		paramSet.Set(name);
sl@0
   580
		}
sl@0
   581
sl@0
   582
	// save the name of the current test suite / step
sl@0
   583
	iCurrentSuiteName = suite;
sl@0
   584
	iCurrentStepName = name;
sl@0
   585
sl@0
   586
	TVerdict currentTestVerdict;
sl@0
   587
sl@0
   588
	INFO_PRINTF2(_L("<a name=\"%S\"</a>"),&name);
sl@0
   589
	
sl@0
   590
	if (iMatchString.Length()>0 && name.Match(iMatchString)<0)
sl@0
   591
		{
sl@0
   592
		// we have a match string but no match - so skip
sl@0
   593
		INFO_PRINTF2(_L("TEST_STEP:%S skipped"), &name);
sl@0
   594
		iSkip = ETrue;
sl@0
   595
		return;
sl@0
   596
		}
sl@0
   597
		
sl@0
   598
	iSkip = EFalse;
sl@0
   599
sl@0
   600
	// convert the guard timer value to a TInt64
sl@0
   601
	TLex lexTimeOut(timeout);
sl@0
   602
	TInt64 guardTimerValue;
sl@0
   603
	if (lexTimeOut.Val(guardTimerValue) != KErrNone)
sl@0
   604
		{
sl@0
   605
		ERR_PRINTF2(_L("Error in guard timer value: %S"),
sl@0
   606
					&timeout);
sl@0
   607
		currentTestVerdict = EInconclusive;
sl@0
   608
		}
sl@0
   609
sl@0
   610
	else
sl@0
   611
		{
sl@0
   612
		// override guard timer if necessary
sl@0
   613
		if((guardTimerValue == KNoGuardTimer) && (iGuardTimer != KNoGuardTimer))
sl@0
   614
			{
sl@0
   615
			INFO_PRINTF3(_L("Warning : Guard timer value overridden from %ld to %ld"),
sl@0
   616
							guardTimerValue, iGuardTimer);
sl@0
   617
			guardTimerValue = iGuardTimer;
sl@0
   618
			}
sl@0
   619
sl@0
   620
		// log the start of a test step
sl@0
   621
		INFO_PRINTF7(_L("RUN_TEST_STEP:%S (step:%S suite:%S timeout:%ldmS config:%S(%S))"),
sl@0
   622
					&name, &step, &suite, guardTimerValue, &config, &paramSet);
sl@0
   623
sl@0
   624
		// NOTE. Now running multithreaded all the time.
sl@0
   625
		currentTestVerdict = DoTestNewThread(suite, step, guardTimerValue, config, paramSet);
sl@0
   626
		}
sl@0
   627
sl@0
   628
	TPtrC verdictText = CLog::TestResultText(currentTestVerdict);
sl@0
   629
	
sl@0
   630
	INFO_PRINTF3(_L("TEST_STEP:%S returned:%S "), 
sl@0
   631
				&name, &verdictText);
sl@0
   632
sl@0
   633
	// this result is only significant if everything else has passed
sl@0
   634
	if (iTestVerdict == EPass)
sl@0
   635
		iTestVerdict = currentTestVerdict;
sl@0
   636
sl@0
   637
	}
sl@0
   638
sl@0
   639
/**
sl@0
   640
 *
sl@0
   641
 * Implements the RUN_PANIC_STEP script command.
sl@0
   642
 *
sl@0
   643
 * @param	"const TDesC& aText"
sl@0
   644
 *			The script line
sl@0
   645
 *
sl@0
   646
 * @xxxx
sl@0
   647
 *
sl@0
   648
 */
sl@0
   649
void CParseLine::RunPanicTestStep(const TDesC& aText)
sl@0
   650
	{
sl@0
   651
	// NOTE. RUN_PANIC_STEP now incorporates the panic reason and category
sl@0
   652
sl@0
   653
	// use Tlex to decode the cmd line
sl@0
   654
	TLex lex(aText);
sl@0
   655
sl@0
   656
	// start at the begining
sl@0
   657
	TPtrC timeout=lex.NextToken();
sl@0
   658
sl@0
   659
	// step over the keyword
sl@0
   660
	timeout.Set(lex.NextToken());
sl@0
   661
sl@0
   662
	// get the other parameters
sl@0
   663
	TPtrC suite, step;
sl@0
   664
	TPtrC category, reason;
sl@0
   665
	TPtrC config, name, paramSet;
sl@0
   666
sl@0
   667
	suite.Set(lex.NextToken());
sl@0
   668
	step.Set(lex.NextToken());
sl@0
   669
	category.Set(lex.NextToken());
sl@0
   670
	reason.Set(lex.NextToken());
sl@0
   671
	config.Set(lex.NextToken());
sl@0
   672
	name.Set(lex.NextToken());
sl@0
   673
	if (name.Length()==0)
sl@0
   674
		{
sl@0
   675
		// name is optional, if not given use step 
sl@0
   676
		name.Set(step);
sl@0
   677
		}
sl@0
   678
	paramSet.Set(lex.NextToken()); 
sl@0
   679
	if (paramSet.Length()==0)
sl@0
   680
		{
sl@0
   681
		// paramSet is optional, if not given use name 
sl@0
   682
		paramSet.Set(name);
sl@0
   683
		}
sl@0
   684
sl@0
   685
	if (iMatchString.Length()>0 && name.Match(iMatchString)<0)
sl@0
   686
		{
sl@0
   687
		// we have a match string but no match - so skip
sl@0
   688
		INFO_PRINTF2(_L("TEST_STEP:%S skipped"), &name);
sl@0
   689
		iSkip = ETrue;
sl@0
   690
		return;
sl@0
   691
		}
sl@0
   692
		
sl@0
   693
	iSkip = EFalse;
sl@0
   694
sl@0
   695
	// save the name of the current test suite / step
sl@0
   696
	iCurrentSuiteName = suite;
sl@0
   697
	iCurrentStepName = name;
sl@0
   698
sl@0
   699
	// convert the guard timer value to a TInt
sl@0
   700
	TLex lexTimeOut(timeout);
sl@0
   701
	TInt64 guardTimerValue;
sl@0
   702
	if (lexTimeOut.Val(guardTimerValue) != KErrNone)
sl@0
   703
		{
sl@0
   704
		ERR_PRINTF3(_L("Error in guard timer value:%S using default %dmS"), 
sl@0
   705
					&timeout, KPanicGuardTimerDefault);
sl@0
   706
		guardTimerValue = KPanicGuardTimerDefault;
sl@0
   707
		}
sl@0
   708
sl@0
   709
	// convert the exitReason value to a TInt
sl@0
   710
	TLex lexReason(reason);
sl@0
   711
	TInt exitReason;
sl@0
   712
	if (lexReason.Val(exitReason) != KErrNone)
sl@0
   713
		{
sl@0
   714
		ERR_PRINTF3(_L("Error in exitReason value:%S using default %d"), 
sl@0
   715
					&reason, KPanicExitReasonDefault);
sl@0
   716
		exitReason = KPanicExitReasonDefault;
sl@0
   717
		}
sl@0
   718
sl@0
   719
	// override guard timer if necessary
sl@0
   720
	if((guardTimerValue == KNoGuardTimer) && (iGuardTimer != KNoGuardTimer))
sl@0
   721
		{
sl@0
   722
		INFO_PRINTF3(_L("Warning : Guard timer value overridden from %ld to %ld"),
sl@0
   723
						guardTimerValue, iGuardTimer);
sl@0
   724
		guardTimerValue = iGuardTimer;
sl@0
   725
		}
sl@0
   726
sl@0
   727
	// log the start of a test step
sl@0
   728
	INFO_PRINTF9(_L("RUN_PANIC_STEP:%S (step:%S suite:%S timeout:%ldmS category:%S reason:%d config:%S(%S))"),
sl@0
   729
				&name, &step, &suite, guardTimerValue, &category, exitReason, &config, &paramSet);
sl@0
   730
sl@0
   731
	// run the test step
sl@0
   732
	TVerdict currentTestVerdict;
sl@0
   733
sl@0
   734
	// now running multithreaded all the time
sl@0
   735
	currentTestVerdict = DoPanicTest(suite, step, guardTimerValue,
sl@0
   736
									 category, exitReason, config, paramSet);
sl@0
   737
sl@0
   738
	TPtrC verdictText = CLog::TestResultText(currentTestVerdict);
sl@0
   739
	INFO_PRINTF3(_L("TEST_STEP:%S returned:%S "), 
sl@0
   740
				&name, &verdictText);
sl@0
   741
sl@0
   742
	// this result is only significant if every thing else has passed
sl@0
   743
	if (iTestVerdict == EPass)
sl@0
   744
		iTestVerdict = currentTestVerdict;
sl@0
   745
sl@0
   746
	}
sl@0
   747
	
sl@0
   748
/**
sl@0
   749
 *
sl@0
   750
 * Implements the RUN_TERMINATION_STEP script command.
sl@0
   751
 *
sl@0
   752
 * @param	"const TDesC& aText"
sl@0
   753
 *			The script line
sl@0
   754
 *
sl@0
   755
 * @xxxx
sl@0
   756
 *
sl@0
   757
 */
sl@0
   758
void CParseLine::RunTerminationTestStep(const TDesC& aText)
sl@0
   759
	{
sl@0
   760
	// use Tlex to decode the cmd line
sl@0
   761
	TLex lex(aText);
sl@0
   762
sl@0
   763
	// start at the begining
sl@0
   764
	TPtrC timeout=lex.NextToken();
sl@0
   765
sl@0
   766
	// step over the keyword
sl@0
   767
	timeout.Set(lex.NextToken());
sl@0
   768
sl@0
   769
	// get the other parameters
sl@0
   770
	TPtrC suite, step;
sl@0
   771
	TPtrC reason;
sl@0
   772
	TPtrC config;
sl@0
   773
sl@0
   774
	suite.Set(lex.NextToken());
sl@0
   775
	step.Set(lex.NextToken());
sl@0
   776
	reason.Set(lex.NextToken());
sl@0
   777
	config.Set(lex.NextToken());
sl@0
   778
sl@0
   779
	// save the name of the current test suite / step
sl@0
   780
	iCurrentSuiteName = suite;
sl@0
   781
	iCurrentStepName = step;
sl@0
   782
sl@0
   783
	// convert the guard timer value to a TInt
sl@0
   784
	TLex lexTimeOut(timeout);
sl@0
   785
	TInt64 guardTimerValue;
sl@0
   786
	if (lexTimeOut.Val(guardTimerValue) != KErrNone)
sl@0
   787
		{
sl@0
   788
		ERR_PRINTF3(_L("Error in guard timer value:%S using default %dmS"), 
sl@0
   789
					&timeout, KPanicGuardTimerDefault);
sl@0
   790
		guardTimerValue = KPanicGuardTimerDefault;
sl@0
   791
		}
sl@0
   792
sl@0
   793
	// convert the exitReason value to a TInt
sl@0
   794
	TLex lexReason(reason);
sl@0
   795
	TInt exitReason;
sl@0
   796
	if (lexReason.Val(exitReason) != KErrNone)
sl@0
   797
		{
sl@0
   798
		ERR_PRINTF3(_L("Error in exitReason value:%S using default %d"), 
sl@0
   799
					&reason, KPanicExitReasonDefault);
sl@0
   800
		exitReason = KPanicExitReasonDefault;
sl@0
   801
		}
sl@0
   802
sl@0
   803
	// override guard timer if necessary
sl@0
   804
	if((guardTimerValue == KNoGuardTimer) && (iGuardTimer != KNoGuardTimer))
sl@0
   805
	{
sl@0
   806
		INFO_PRINTF3(_L("Warning : Guard timer value overridden from %ld to %ld"),
sl@0
   807
						guardTimerValue, iGuardTimer);
sl@0
   808
		guardTimerValue = iGuardTimer;
sl@0
   809
	}
sl@0
   810
sl@0
   811
	// log the start of a test step
sl@0
   812
	INFO_PRINTF6(_L("RUN_TERMINATION_STEP:%S suite:%S timeout:%ldmS reason:%d config:%S"),
sl@0
   813
				&step, &suite, guardTimerValue, exitReason, &config);
sl@0
   814
sl@0
   815
	// run the test step
sl@0
   816
	TVerdict currentTestVerdict;
sl@0
   817
sl@0
   818
	// now running multithreaded all the time
sl@0
   819
	currentTestVerdict = DoTerminationTest(suite, step, guardTimerValue,
sl@0
   820
									 exitReason, config);
sl@0
   821
sl@0
   822
	TPtrC verdictText = CLog::TestResultText(currentTestVerdict);
sl@0
   823
	INFO_PRINTF3(_L("TEST_STEP:%S returned:%S "), 
sl@0
   824
				&step, &verdictText);
sl@0
   825
sl@0
   826
	// this result is only significant if every thing else has passed
sl@0
   827
	if (iTestVerdict == EPass)
sl@0
   828
		iTestVerdict = currentTestVerdict;
sl@0
   829
sl@0
   830
	}
sl@0
   831
sl@0
   832
/**
sl@0
   833
 *
sl@0
   834
 * Implements the RUN_UTILS script command.
sl@0
   835
 *
sl@0
   836
 * @param	"const TDesC& aText"
sl@0
   837
 *			The script line
sl@0
   838
 *
sl@0
   839
 * @xxxx
sl@0
   840
 *
sl@0
   841
 */
sl@0
   842
void CParseLine::RunUtil(const TDesC& aText)
sl@0
   843
	{
sl@0
   844
	// Call the utils
sl@0
   845
	iTestUtils->RunUtils(aText);
sl@0
   846
	}
sl@0
   847
sl@0
   848
/**
sl@0
   849
 *
sl@0
   850
 * Implements the REBOOT script command.
sl@0
   851
 *
sl@0
   852
 * @param	"const TDesC& aText"
sl@0
   853
 *			The script line
sl@0
   854
 *
sl@0
   855
 * @xxxx
sl@0
   856
 *
sl@0
   857
 */
sl@0
   858
void CParseLine::Reboot()
sl@0
   859
	{
sl@0
   860
	WARN_PRINTF1(_L("Warning : REBOOT command not implemented"));
sl@0
   861
	}
sl@0
   862
sl@0
   863
/**
sl@0
   864
 *
sl@0
   865
 * Static function to call DoTestStep which is run
sl@0
   866
 * in a separate thread
sl@0
   867
 *
sl@0
   868
 * @param	"TAny* aPtr"
sl@0
   869
 *			The test step data
sl@0
   870
 *
sl@0
   871
 * @return "TInt"
sl@0
   872
 *			EPOC error code
sl@0
   873
 *
sl@0
   874
 * @xxxx
sl@0
   875
 *
sl@0
   876
 */
sl@0
   877
TInt CParseLine::ThreadFunctionL(TAny* aPtr)
sl@0
   878
	{
sl@0
   879
	TInt result = KErrNone;
sl@0
   880
sl@0
   881
	// get clean-up stack
sl@0
   882
  	CTrapCleanup* trapCleanup = CTrapCleanup::New();
sl@0
   883
sl@0
   884
	TRAPD(err, result = ThreadTrapFunctionL(aPtr));
sl@0
   885
sl@0
   886
	delete trapCleanup;
sl@0
   887
	return((err != KErrNone) ? err : result);
sl@0
   888
	}
sl@0
   889
sl@0
   890
/**
sl@0
   891
 *
sl@0
   892
 * Main function to call DoTestStep, called from within
sl@0
   893
 * a trap
sl@0
   894
 *
sl@0
   895
 * @param	"TAny* aPtr"
sl@0
   896
 *			The test step data
sl@0
   897
 *
sl@0
   898
 * @return "TInt"
sl@0
   899
 *			EPOC error code
sl@0
   900
 *
sl@0
   901
 * @xxxx
sl@0
   902
 *
sl@0
   903
 */
sl@0
   904
TInt CParseLine::ThreadTrapFunctionL(TAny* aPtr)
sl@0
   905
	{
sl@0
   906
	// get the data for the test
sl@0
   907
	CStepData* data = REINTERPRET_CAST(CStepData*, aPtr);
sl@0
   908
	CSuiteDll* suiteDll = data->SuiteDll();
sl@0
   909
	CTestSuite* testSuite = suiteDll->Suite();
sl@0
   910
sl@0
   911
	// setup local logger
sl@0
   912
	CLog* logClient	= CLog::NewLC();
sl@0
   913
	logClient->OpenLogFileL();
sl@0
   914
	testSuite->SetLogSystem(logClient);
sl@0
   915
sl@0
   916
	// do the test step 
sl@0
   917
	TVerdict result =  testSuite->DoTestStep(data->Step(), data->Config(), data->ParamSet());
sl@0
   918
sl@0
   919
	// NB it is the CALLING program's responsibility to save/restore the logger.
sl@0
   920
	// If the thread terminates prematurely, the logger is in an undefined state.
sl@0
   921
sl@0
   922
	CleanupStack::PopAndDestroy(logClient);
sl@0
   923
	testSuite->SetLogSystem(NULL);
sl@0
   924
sl@0
   925
	// return the test result
sl@0
   926
	return result;
sl@0
   927
	}
sl@0
   928
sl@0
   929
/**
sl@0
   930
 *
sl@0
   931
 * Do a test step in a new thread.
sl@0
   932
 *
sl@0
   933
 * @param	"const TDesC& aSuite"
sl@0
   934
 *			The test suite
sl@0
   935
 *
sl@0
   936
 * @param	"const TDesC& aStep"
sl@0
   937
 *			The test step
sl@0
   938
 *
sl@0
   939
 * @param	"TInt aGuardTimerValue"
sl@0
   940
 *			The guard timer value
sl@0
   941
 *
sl@0
   942
 * @param	"const TDesC& aConfig"
sl@0
   943
 *			The config data
sl@0
   944
 *
sl@0
   945
 * @return  "TVerdict"
sl@0
   946
 *			The test result
sl@0
   947
 *
sl@0
   948
 * @xxxx
sl@0
   949
 *
sl@0
   950
 */
sl@0
   951
TVerdict CParseLine::DoTestNewThread(const TDesC& aSuite, const TDesC& aStep, 
sl@0
   952
							TInt64 aGuardTimerValue, const TDesC& aConfig, const TDesC& aParamSet)
sl@0
   953
	{
sl@0
   954
	//	get the number of suites loaded
sl@0
   955
	TInt noOfDlls = iArrayLoadedSuiteDll->Count();
sl@0
   956
sl@0
   957
	// search the list of loaded test suite DLLs for the required one
sl@0
   958
	for (TInt i = 0; i < noOfDlls; i++)
sl@0
   959
		{
sl@0
   960
		CSuiteDll* ptrSuite = iArrayLoadedSuiteDll->At(i);
sl@0
   961
		TPtrC name = ptrSuite->Name();
sl@0
   962
sl@0
   963
		if (name.FindF(aSuite)!= KErrNotFound)
sl@0
   964
			{
sl@0
   965
			// reset step status
sl@0
   966
			CTestSuite* testSuite = ptrSuite->Suite();
sl@0
   967
			testSuite->SetStepStatus(EStepStatusNone);
sl@0
   968
sl@0
   969
			// store old log status, for restore at thread exit
sl@0
   970
			// NB we must do this here, as if thread times out, the log
sl@0
   971
			// is in an undefined state
sl@0
   972
			CLog* oldLogger = testSuite->LogSystem();
sl@0
   973
sl@0
   974
			CStepData* data = NULL;
sl@0
   975
			TRAPD(err, data = CStepData::NewL(aStep, aConfig, aParamSet, ptrSuite));
sl@0
   976
			if (err != KErrNone)
sl@0
   977
				{
sl@0
   978
				ERR_PRINTF2(_L("CStepData::NewL() left with error %d : unable to create test data!"), err);
sl@0
   979
				return EFail;
sl@0
   980
				}
sl@0
   981
			
sl@0
   982
			// get step's own stack and heap sizes
sl@0
   983
			TInt theHeapSize = KMaxTestThreadHeapSize;
sl@0
   984
			TInt theStackSize = KTestStackSize;
sl@0
   985
			GetHeapAndStackSize(data, &theHeapSize, &theStackSize);
sl@0
   986
sl@0
   987
			TInt res = KErrAlreadyExists;
sl@0
   988
			RThread newThread;
sl@0
   989
sl@0
   990
			TPtrC threadBaseName(_L("DoTestThread"));
sl@0
   991
			TBuf<32> threadName;
sl@0
   992
			
sl@0
   993
			// create a unique named test thread
sl@0
   994
			// this will leave if creation is not successful
sl@0
   995
			TRAP (res, CreateUniqueTestThreadL( threadBaseName, 
sl@0
   996
												threadName, 
sl@0
   997
												newThread, 
sl@0
   998
												ThreadFunctionL,
sl@0
   999
												theStackSize,
sl@0
  1000
												KMinHeapSize,
sl@0
  1001
												theHeapSize,
sl@0
  1002
												data ) );
sl@0
  1003
sl@0
  1004
			
sl@0
  1005
			if (res != KErrNone)
sl@0
  1006
				{
sl@0
  1007
				ERR_PRINTF2(_L("CreateUniqueTestThreadL() left with error %d : unable to create test thread "), res);
sl@0
  1008
				delete data;
sl@0
  1009
				data = NULL;
sl@0
  1010
				return EFail;
sl@0
  1011
				}
sl@0
  1012
sl@0
  1013
			// start clock
sl@0
  1014
			TTime testStart, testStop;
sl@0
  1015
			testStart.HomeTime();
sl@0
  1016
			
sl@0
  1017
			// start the thread and request the status
sl@0
  1018
			TRequestStatus threadStatus;
sl@0
  1019
			newThread.Logon(threadStatus);
sl@0
  1020
sl@0
  1021
			// if there is no guard timer value, don't time at all
sl@0
  1022
			if (aGuardTimerValue == KNoGuardTimer)
sl@0
  1023
				{
sl@0
  1024
				// no guard timer
sl@0
  1025
				newThread.Resume();
sl@0
  1026
				User::WaitForRequest(threadStatus);
sl@0
  1027
				}
sl@0
  1028
			else
sl@0
  1029
				{
sl@0
  1030
				// wait for either test thread or timer to end
sl@0
  1031
				RTimer guardTimer;
sl@0
  1032
				guardTimer.CreateLocal();			// create for this thread
sl@0
  1033
				TRequestStatus timerStatus;
sl@0
  1034
				newThread.Resume();
sl@0
  1035
sl@0
  1036
				// NB now using At() to allow 64-bit timer values
sl@0
  1037
				TInt64 guardTimerUsec = aGuardTimerValue * 1000;
sl@0
  1038
				TInt64 totalTime = 0;
sl@0
  1039
sl@0
  1040
				for (;;)
sl@0
  1041
					{
sl@0
  1042
					if (totalTime>=guardTimerUsec) // timeout has occured
sl@0
  1043
						break;
sl@0
  1044
					TInt timeout;
sl@0
  1045
sl@0
  1046
					if (totalTime+KTimeIncrement >= guardTimerUsec)
sl@0
  1047
						{
sl@0
  1048
						TInt64 temp = guardTimerUsec-totalTime;
sl@0
  1049
						timeout = I64INT(temp);
sl@0
  1050
						}
sl@0
  1051
					else
sl@0
  1052
						timeout = KTimeIncrement;
sl@0
  1053
					totalTime += timeout;
sl@0
  1054
					guardTimer.After(timerStatus, timeout);
sl@0
  1055
					User::WaitForRequest(threadStatus, timerStatus);
sl@0
  1056
					if (threadStatus!=KRequestPending) // normal exit
sl@0
  1057
						break;
sl@0
  1058
					}
sl@0
  1059
sl@0
  1060
				guardTimer.Cancel();
sl@0
  1061
				guardTimer.Close();
sl@0
  1062
				}
sl@0
  1063
sl@0
  1064
			// reset any file server error simulations
sl@0
  1065
			RFs fs;
sl@0
  1066
			TInt fsError = fs.Connect();
sl@0
  1067
			if (fsError == KErrNone)
sl@0
  1068
				{
sl@0
  1069
				fs.SetErrorCondition(KErrNone);
sl@0
  1070
				}
sl@0
  1071
			fs.Close();
sl@0
  1072
sl@0
  1073
			// restore logger
sl@0
  1074
			testSuite->SetLogSystem(oldLogger);
sl@0
  1075
sl@0
  1076
			// get the test result
sl@0
  1077
			TVerdict result = STATIC_CAST(TVerdict, threadStatus.Int());
sl@0
  1078
sl@0
  1079
			// check terminated ok
sl@0
  1080
			switch(newThread.ExitType())
sl@0
  1081
				{
sl@0
  1082
				case EExitTerminate:
sl@0
  1083
				case EExitKill:
sl@0
  1084
					break;
sl@0
  1085
				case EExitPanic:
sl@0
  1086
					{
sl@0
  1087
					TExitCategoryName exitCategory = newThread.ExitCategory();
sl@0
  1088
					TInt exitReason = newThread.ExitReason();
sl@0
  1089
 					ERR_PRINTF3(_L("Thread had a panic %S:%d"), &exitCategory, exitReason);
sl@0
  1090
sl@0
  1091
					result = EFail;
sl@0
  1092
					}
sl@0
  1093
					break;
sl@0
  1094
				case EExitPending:
sl@0
  1095
					// if the thread is still pending then the guard timer must have expired
sl@0
  1096
					ERR_PRINTF1(_L("Thread timed out"));
sl@0
  1097
					// kill the test step thread
sl@0
  1098
					newThread.Kill(1);
sl@0
  1099
					// give the OS time to cleanup devices, etc.
sl@0
  1100
					// NB if the thread dies, the postamble will NOT run
sl@0
  1101
					User::After(2000000);
sl@0
  1102
					result = EFail;
sl@0
  1103
					break;
sl@0
  1104
				default:
sl@0
  1105
					break;
sl@0
  1106
				}
sl@0
  1107
sl@0
  1108
			// done with the test thread
sl@0
  1109
			newThread.Close();
sl@0
  1110
sl@0
  1111
			// stop clock
sl@0
  1112
			testStop.HomeTime();
sl@0
  1113
sl@0
  1114
			TUint testDuration = I64INT(testStop.MicroSecondsFrom(testStart).Int64());
sl@0
  1115
			testDuration /= 1000; // to microseconds
sl@0
  1116
			TUint testDurationMsec = testDuration % 1000;
sl@0
  1117
			TUint testDurationSec = testDuration / 1000;
sl@0
  1118
			INFO_PRINTF3(_L("Test took %d.%03d sec"), testDurationSec, testDurationMsec);
sl@0
  1119
sl@0
  1120
			// return the test verdict
sl@0
  1121
			delete data;
sl@0
  1122
			data = NULL;
sl@0
  1123
			return result;
sl@0
  1124
			}
sl@0
  1125
		}
sl@0
  1126
sl@0
  1127
	// the required suite has not been found
sl@0
  1128
	ERR_PRINTF3(_L("Error in test step:%S - cannot find suite:%S" ),
sl@0
  1129
					&aStep, &aSuite);
sl@0
  1130
sl@0
  1131
	return ETestSuiteError;
sl@0
  1132
	}
sl@0
  1133
sl@0
  1134
/**
sl@0
  1135
 *
sl@0
  1136
 * Do a test step which is expected to panic.
sl@0
  1137
 *
sl@0
  1138
 * @param	"const TDesC& aSuite"
sl@0
  1139
 *			The test suite
sl@0
  1140
 *
sl@0
  1141
 * @param	"const TDesC& aStep"
sl@0
  1142
 *			The test step
sl@0
  1143
 *
sl@0
  1144
 * @param	"TInt aGuardTimerValue"
sl@0
  1145
 *			The guard timer value
sl@0
  1146
 *
sl@0
  1147
 * @param	"const TExitCategoryName aExitCategory"
sl@0
  1148
 *			The expected exit category
sl@0
  1149
 *
sl@0
  1150
 * @param	"TInt aExitReason"
sl@0
  1151
 *			The expected exit reason
sl@0
  1152
 *
sl@0
  1153
 * @param	"const TDesC& aConfig"
sl@0
  1154
 *			The config data
sl@0
  1155
 *
sl@0
  1156
 * @return "TVerdict"
sl@0
  1157
 *			The test result
sl@0
  1158
 *
sl@0
  1159
 * @xxxx
sl@0
  1160
 *
sl@0
  1161
 */
sl@0
  1162
TVerdict CParseLine::DoPanicTest(const TDesC& aSuite, const TDesC& aStep, TInt64 aGuardTimerValue,
sl@0
  1163
									  const TExitCategoryName aExitCategory, TInt aExitReason, 
sl@0
  1164
									  const TDesC& aConfig, const TDesC& aParamSet)
sl@0
  1165
	{
sl@0
  1166
sl@0
  1167
	//	get the number of suites loaded
sl@0
  1168
	TInt noOfDlls = iArrayLoadedSuiteDll->Count();
sl@0
  1169
sl@0
  1170
	// search the list of loaded test suite DLLs for the required one
sl@0
  1171
	for (TInt i = 0; i < noOfDlls; i++)
sl@0
  1172
		{
sl@0
  1173
		CSuiteDll* ptrSuite = iArrayLoadedSuiteDll->At(i);
sl@0
  1174
		TPtrC name = ptrSuite->Name();
sl@0
  1175
sl@0
  1176
		if (name.FindF(aSuite)!= KErrNotFound)
sl@0
  1177
			{
sl@0
  1178
			// reset step status
sl@0
  1179
			CTestSuite* testSuite = ptrSuite->Suite();
sl@0
  1180
			testSuite->SetStepStatus(EStepStatusNone);
sl@0
  1181
sl@0
  1182
			// store old log status, for restore at thread exit
sl@0
  1183
			// NB we must do this here, as if thread times out, the log
sl@0
  1184
			// is in an undefined state
sl@0
  1185
			CLog* oldLogger = testSuite->LogSystem();
sl@0
  1186
sl@0
  1187
			CStepData* data = NULL;
sl@0
  1188
			TRAPD(err, data = CStepData::NewL(aStep, aConfig, aParamSet, ptrSuite));
sl@0
  1189
			if (err != KErrNone)
sl@0
  1190
				{
sl@0
  1191
				ERR_PRINTF2(_L("CStepData::NewL() left with error %d : unable to create test data!"), err);
sl@0
  1192
				return EFail;
sl@0
  1193
				}
sl@0
  1194
sl@0
  1195
			// get step's own stack and heap sizes
sl@0
  1196
			TInt theHeapSize = KMaxTestThreadHeapSize;
sl@0
  1197
			TInt theStackSize = KTestStackSize;
sl@0
  1198
			GetHeapAndStackSize(data, &theHeapSize, &theStackSize);
sl@0
  1199
sl@0
  1200
			TInt res = KErrAlreadyExists;
sl@0
  1201
			RThread newThread;
sl@0
  1202
sl@0
  1203
			// create a unique test name by appending a counter
sl@0
  1204
			TPtrC threadBaseName(_L("DoTestThread"));
sl@0
  1205
			TBuf<32> threadName;
sl@0
  1206
			
sl@0
  1207
			// create a unique named test thread
sl@0
  1208
			// this will leave if creation is not successful
sl@0
  1209
			TRAP (res, CreateUniqueTestThreadL( threadBaseName, 
sl@0
  1210
												threadName, 
sl@0
  1211
												newThread, 
sl@0
  1212
												ThreadFunctionL,
sl@0
  1213
												theStackSize,
sl@0
  1214
												KMinHeapSize,
sl@0
  1215
												theHeapSize,
sl@0
  1216
												data ) );
sl@0
  1217
			
sl@0
  1218
			if (res != KErrNone)
sl@0
  1219
				{
sl@0
  1220
				ERR_PRINTF2(_L("CreateUniqueTestThreadL() left with error %d : unable to create test thread "), res);
sl@0
  1221
				delete data;
sl@0
  1222
				data = NULL;
sl@0
  1223
				return EFail;
sl@0
  1224
				}
sl@0
  1225
sl@0
  1226
			// start clock
sl@0
  1227
			TTime testStart, testStop;
sl@0
  1228
			testStart.HomeTime();
sl@0
  1229
			
sl@0
  1230
			// start the thread and request the status
sl@0
  1231
			TRequestStatus threadStatus;
sl@0
  1232
			newThread.Logon(threadStatus);
sl@0
  1233
sl@0
  1234
			// if there is no guard timer value, don't time at all
sl@0
  1235
			if (aGuardTimerValue == KNoGuardTimer)
sl@0
  1236
				{
sl@0
  1237
				// no guard timer
sl@0
  1238
				newThread.Resume();
sl@0
  1239
				User::WaitForRequest(threadStatus);
sl@0
  1240
				}
sl@0
  1241
			else
sl@0
  1242
				{
sl@0
  1243
				// wait for either test thread or timer to end
sl@0
  1244
				RTimer guardTimer;
sl@0
  1245
				guardTimer.CreateLocal();			// create for this thread
sl@0
  1246
				TRequestStatus timerStatus;
sl@0
  1247
				newThread.Resume();
sl@0
  1248
sl@0
  1249
				// NB now using At() to allow 64-bit timer values
sl@0
  1250
				TInt64 guardTimerUsec = aGuardTimerValue * 1000;
sl@0
  1251
				TInt64 totalTime = 0;
sl@0
  1252
sl@0
  1253
				for (;;)
sl@0
  1254
					{
sl@0
  1255
					if (totalTime>=guardTimerUsec) // timeout has occured
sl@0
  1256
						break;
sl@0
  1257
					TInt timeout;
sl@0
  1258
sl@0
  1259
					if (totalTime+KTimeIncrement >= guardTimerUsec)
sl@0
  1260
						{
sl@0
  1261
						TInt64 temp = guardTimerUsec-totalTime;
sl@0
  1262
						timeout = I64INT(temp);
sl@0
  1263
						}
sl@0
  1264
					else
sl@0
  1265
						timeout = KTimeIncrement;
sl@0
  1266
						totalTime += timeout;
sl@0
  1267
					guardTimer.After(timerStatus, timeout);
sl@0
  1268
					User::WaitForRequest(threadStatus, timerStatus);
sl@0
  1269
					if (threadStatus!=KRequestPending) // normal exit
sl@0
  1270
						break;
sl@0
  1271
					}
sl@0
  1272
sl@0
  1273
				guardTimer.Cancel();
sl@0
  1274
				guardTimer.Close();
sl@0
  1275
				}
sl@0
  1276
sl@0
  1277
			// restore logger
sl@0
  1278
			testSuite->SetLogSystem(oldLogger);
sl@0
  1279
sl@0
  1280
			// get the test result
sl@0
  1281
			TVerdict result = STATIC_CAST(TVerdict, threadStatus.Int());
sl@0
  1282
sl@0
  1283
			// check terminated ok
sl@0
  1284
			switch(newThread.ExitType())
sl@0
  1285
				{
sl@0
  1286
				case EExitPanic:
sl@0
  1287
					{
sl@0
  1288
					TExitCategoryName exitCategory = newThread.ExitCategory();
sl@0
  1289
					TInt exitReason = newThread.ExitReason();
sl@0
  1290
					if((exitCategory != aExitCategory) || (exitReason != aExitReason && aExitReason != KNoPanicReason) )
sl@0
  1291
						{
sl@0
  1292
						ERR_PRINTF3(_L("Test step had an unexpected panic %S:%d and failed"),
sl@0
  1293
									&exitCategory, exitReason);
sl@0
  1294
						result = EFail;
sl@0
  1295
						}
sl@0
  1296
					else
sl@0
  1297
						{
sl@0
  1298
						// check here that the panic occurred within the test itself
sl@0
  1299
						CTestSuite* testSuite = ptrSuite->Suite();
sl@0
  1300
						TTestStepStatus status = testSuite->StepStatus();
sl@0
  1301
						switch(status)
sl@0
  1302
							{
sl@0
  1303
							case EStepStatusPreamble:
sl@0
  1304
								{
sl@0
  1305
								// thread panicked in the test itself - success
sl@0
  1306
								INFO_PRINTF3(_L("Test step had a panic %S:%d and passed"),
sl@0
  1307
											&exitCategory, exitReason);
sl@0
  1308
								result = EPass;
sl@0
  1309
								}
sl@0
  1310
								break;
sl@0
  1311
							case EStepStatusStart:
sl@0
  1312
								{
sl@0
  1313
								// thread panicked in preamble
sl@0
  1314
								ERR_PRINTF3(_L("Test step had a panic %S:%d in preamble"),
sl@0
  1315
											&exitCategory, exitReason);
sl@0
  1316
								result = EFail;
sl@0
  1317
								}
sl@0
  1318
								break;
sl@0
  1319
							case EStepStatusTest:
sl@0
  1320
								{
sl@0
  1321
								// thread panicked in postamble
sl@0
  1322
								ERR_PRINTF3(_L("Test step had a panic %S:%d in postamble"),
sl@0
  1323
											&exitCategory, exitReason);
sl@0
  1324
								result = EFail;
sl@0
  1325
								}
sl@0
  1326
								break;
sl@0
  1327
							default:
sl@0
  1328
								{
sl@0
  1329
								// thread panicked outside the test
sl@0
  1330
								ERR_PRINTF3(_L("Test step had a panic %S:%d outside the test"),
sl@0
  1331
											&exitCategory, exitReason);
sl@0
  1332
								result = EFail;
sl@0
  1333
								}
sl@0
  1334
								break;
sl@0
  1335
							}	// end switch
sl@0
  1336
						}
sl@0
  1337
					}
sl@0
  1338
					break;
sl@0
  1339
				case EExitPending:
sl@0
  1340
					// if the thread is still pending then the guard timer must have expired
sl@0
  1341
					ERR_PRINTF1(_L("Thread timed out"));
sl@0
  1342
					// kill the test step thread
sl@0
  1343
					newThread.Kill(1);
sl@0
  1344
					// give the OS time to cleanup devices, etc.
sl@0
  1345
					// NB if the thread dies, the postamble will NOT run
sl@0
  1346
					User::After(2000000);
sl@0
  1347
					result = EFail;
sl@0
  1348
					break;
sl@0
  1349
				case EExitTerminate:
sl@0
  1350
				case EExitKill:
sl@0
  1351
				default:
sl@0
  1352
					ERR_PRINTF1(_L("Test did not panic, so failed"));
sl@0
  1353
					result = EFail;
sl@0
  1354
					break;
sl@0
  1355
				}
sl@0
  1356
sl@0
  1357
			// done with the test thread
sl@0
  1358
			newThread.Close();
sl@0
  1359
sl@0
  1360
			// stop clock
sl@0
  1361
			testStop.HomeTime();
sl@0
  1362
sl@0
  1363
			TUint testDuration = I64INT(testStop.MicroSecondsFrom(testStart).Int64());
sl@0
  1364
			testDuration /= 1000; // to microseconds
sl@0
  1365
			TUint testDurationMsec = testDuration % 1000;
sl@0
  1366
			TUint testDurationSec = testDuration / 1000;
sl@0
  1367
			INFO_PRINTF3(_L("Test took %d.%03d sec"), testDurationSec, testDurationMsec);
sl@0
  1368
sl@0
  1369
			// return the test verdict
sl@0
  1370
			delete data;
sl@0
  1371
			data = NULL;
sl@0
  1372
			return result;
sl@0
  1373
			}
sl@0
  1374
		}
sl@0
  1375
sl@0
  1376
	// the required suite has not been found
sl@0
  1377
	ERR_PRINTF3(_L("Error in test step:%S - cannot find suite:%S"),
sl@0
  1378
				&aStep, &aSuite );
sl@0
  1379
sl@0
  1380
	return ETestSuiteError;
sl@0
  1381
	}
sl@0
  1382
	
sl@0
  1383
/**
sl@0
  1384
 *
sl@0
  1385
 * Do a test step which is expected to terminate.
sl@0
  1386
 *
sl@0
  1387
 * @param	"const TDesC& aSuite"
sl@0
  1388
 *			The test suite
sl@0
  1389
 *
sl@0
  1390
 * @param	"const TDesC& aStep"
sl@0
  1391
 *			The test step
sl@0
  1392
 *
sl@0
  1393
 * @param	"TInt aGuardTimerValue"
sl@0
  1394
 *			The guard timer value
sl@0
  1395
 *
sl@0
  1396
 * @param	"TInt aExitReason"
sl@0
  1397
 *			The expected exit reason
sl@0
  1398
 *
sl@0
  1399
 * @param	"const TDesC& aConfig"
sl@0
  1400
 *			The config data
sl@0
  1401
 *
sl@0
  1402
 * @return "TVerdict"
sl@0
  1403
 *			The test result
sl@0
  1404
 *
sl@0
  1405
 * @xxxx
sl@0
  1406
 *
sl@0
  1407
 */
sl@0
  1408
TVerdict CParseLine::DoTerminationTest(const TDesC& aSuite, const TDesC& aStep, TInt64 aGuardTimerValue,
sl@0
  1409
									  TInt aExitReason, const TDesC& aConfig)
sl@0
  1410
	{
sl@0
  1411
sl@0
  1412
	//	get the number of suites loaded
sl@0
  1413
	TInt noOfDlls = iArrayLoadedSuiteDll->Count();
sl@0
  1414
sl@0
  1415
	// search the list of loaded test suite DLLs for the required one
sl@0
  1416
	for (TInt i = 0; i < noOfDlls; i++)
sl@0
  1417
		{
sl@0
  1418
		CSuiteDll* ptrSuite = iArrayLoadedSuiteDll->At(i);
sl@0
  1419
		TPtrC name = ptrSuite->Name();
sl@0
  1420
sl@0
  1421
		if (name.FindF(aSuite)!= KErrNotFound)
sl@0
  1422
			{
sl@0
  1423
			// reset step status
sl@0
  1424
			CTestSuite* testSuite = ptrSuite->Suite();
sl@0
  1425
			testSuite->SetStepStatus(EStepStatusNone);
sl@0
  1426
sl@0
  1427
			// store old log status, for restore at thread exit
sl@0
  1428
			// NB we must do this here, as if thread times out, the log
sl@0
  1429
			// is in an undefined state
sl@0
  1430
			CLog* oldLogger = testSuite->LogSystem();
sl@0
  1431
sl@0
  1432
			CStepData* data = NULL;
sl@0
  1433
			TRAPD(err, data = CStepData::NewL(aStep, aConfig, ptrSuite));
sl@0
  1434
			if (err != KErrNone)
sl@0
  1435
				{
sl@0
  1436
				ERR_PRINTF2(_L("CStepData::NewL() left with error %d : unable to create test data!"), err);
sl@0
  1437
				return EFail;
sl@0
  1438
				}
sl@0
  1439
sl@0
  1440
			// get step's own stack and heap sizes
sl@0
  1441
			TInt theHeapSize = KMaxTestThreadHeapSize;
sl@0
  1442
			TInt theStackSize = KTestStackSize;
sl@0
  1443
			GetHeapAndStackSize(data, &theHeapSize, &theStackSize);
sl@0
  1444
sl@0
  1445
			TInt res = KErrAlreadyExists;
sl@0
  1446
			RThread newThread;
sl@0
  1447
sl@0
  1448
			// create a unique test name by appending a counter
sl@0
  1449
			TPtrC threadBaseName(_L("DoTestThread"));
sl@0
  1450
			TBuf<32> threadName;
sl@0
  1451
			
sl@0
  1452
			// create a unique named test thread
sl@0
  1453
			// this will leave if creation is not successful
sl@0
  1454
			TRAP (res, CreateUniqueTestThreadL( threadBaseName, 
sl@0
  1455
												threadName, 
sl@0
  1456
												newThread, 
sl@0
  1457
												ThreadFunctionL,
sl@0
  1458
												theStackSize,
sl@0
  1459
												KMinHeapSize,
sl@0
  1460
												theHeapSize,
sl@0
  1461
												data ) );
sl@0
  1462
			
sl@0
  1463
			if (res != KErrNone)
sl@0
  1464
				{
sl@0
  1465
				ERR_PRINTF2(_L("CreateUniqueTestThreadL() left with error %d : unable to create test thread "), res);
sl@0
  1466
				delete data;
sl@0
  1467
				data = NULL;
sl@0
  1468
				return EFail;
sl@0
  1469
				}
sl@0
  1470
sl@0
  1471
			// start clock
sl@0
  1472
			TTime testStart, testStop;
sl@0
  1473
			testStart.HomeTime();
sl@0
  1474
			
sl@0
  1475
			// start the thread and request the status
sl@0
  1476
			TRequestStatus threadStatus;
sl@0
  1477
			newThread.Logon(threadStatus);
sl@0
  1478
sl@0
  1479
			// if there is no guard timer value, don't time at all
sl@0
  1480
			if (aGuardTimerValue == KNoGuardTimer)
sl@0
  1481
				{
sl@0
  1482
				// no guard timer
sl@0
  1483
				newThread.Resume();
sl@0
  1484
				User::WaitForRequest(threadStatus);
sl@0
  1485
				}
sl@0
  1486
			else
sl@0
  1487
				{
sl@0
  1488
				// wait for either test thread or timer to end
sl@0
  1489
				RTimer guardTimer;
sl@0
  1490
				guardTimer.CreateLocal();			// create for this thread
sl@0
  1491
				TRequestStatus timerStatus;
sl@0
  1492
				newThread.Resume();
sl@0
  1493
sl@0
  1494
				// NB now using At() to allow 64-bit timer values
sl@0
  1495
				TInt64 guardTimerUsec = aGuardTimerValue * 1000;
sl@0
  1496
				TInt64 totalTime = 0;
sl@0
  1497
sl@0
  1498
				for (;;)
sl@0
  1499
					{
sl@0
  1500
					if (totalTime>=guardTimerUsec) // timeout has occured
sl@0
  1501
						break;
sl@0
  1502
					TInt timeout;
sl@0
  1503
sl@0
  1504
					if (totalTime+KTimeIncrement >= guardTimerUsec)
sl@0
  1505
						{
sl@0
  1506
						TInt64 temp = guardTimerUsec-totalTime;
sl@0
  1507
						timeout = I64INT(temp);
sl@0
  1508
						}
sl@0
  1509
					else
sl@0
  1510
						timeout = KTimeIncrement;
sl@0
  1511
						totalTime += timeout;
sl@0
  1512
					guardTimer.After(timerStatus, timeout);
sl@0
  1513
					User::WaitForRequest(threadStatus, timerStatus);
sl@0
  1514
					if (threadStatus!=KRequestPending) // normal exit
sl@0
  1515
						break;
sl@0
  1516
					}
sl@0
  1517
sl@0
  1518
				guardTimer.Cancel();
sl@0
  1519
				guardTimer.Close();
sl@0
  1520
				}
sl@0
  1521
sl@0
  1522
			// restore logger
sl@0
  1523
			testSuite->SetLogSystem(oldLogger);
sl@0
  1524
sl@0
  1525
			// get the test result
sl@0
  1526
			TVerdict result = STATIC_CAST(TVerdict, threadStatus.Int());
sl@0
  1527
sl@0
  1528
			// check terminated ok
sl@0
  1529
			switch(newThread.ExitType())
sl@0
  1530
				{
sl@0
  1531
				case EExitTerminate:
sl@0
  1532
				case EExitKill:
sl@0
  1533
					{
sl@0
  1534
					TInt exitReason = newThread.ExitReason();
sl@0
  1535
					if(exitReason != aExitReason)
sl@0
  1536
						{
sl@0
  1537
						ERR_PRINTF2(_L("Test step had an unexpected exit reason:%d and failed"),
sl@0
  1538
									exitReason);
sl@0
  1539
						result = EFail;
sl@0
  1540
						}
sl@0
  1541
					else
sl@0
  1542
						{
sl@0
  1543
						// check here that the panic occurred within the test itself
sl@0
  1544
						CTestSuite* testSuite = ptrSuite->Suite();
sl@0
  1545
						TTestStepStatus status = testSuite->StepStatus();
sl@0
  1546
						switch(status)
sl@0
  1547
							{
sl@0
  1548
							case EStepStatusPreamble:
sl@0
  1549
								{
sl@0
  1550
								// thread terminated in the test itself - success
sl@0
  1551
								INFO_PRINTF2(_L("Test step had terminated:%d and passed"),
sl@0
  1552
											exitReason);
sl@0
  1553
								result = EPass;
sl@0
  1554
								}
sl@0
  1555
								break;
sl@0
  1556
							case EStepStatusStart:
sl@0
  1557
								{
sl@0
  1558
								// thread panicked in preamble
sl@0
  1559
								ERR_PRINTF2(_L("Test step had terminated:%d in preamble"),
sl@0
  1560
											exitReason);
sl@0
  1561
								result = EFail;
sl@0
  1562
								}
sl@0
  1563
								break;
sl@0
  1564
							case EStepStatusTest:
sl@0
  1565
								{
sl@0
  1566
								// thread panicked in postamble
sl@0
  1567
								ERR_PRINTF2(_L("Test step had terminated:%d in postamble"),
sl@0
  1568
											exitReason);
sl@0
  1569
								result = EFail;
sl@0
  1570
								}
sl@0
  1571
								break;
sl@0
  1572
							default:
sl@0
  1573
								{
sl@0
  1574
								// thread panicked outside the test
sl@0
  1575
								ERR_PRINTF2(_L("Test step had terminated:%d outside the test"),
sl@0
  1576
											exitReason);
sl@0
  1577
								result = EFail;
sl@0
  1578
								}
sl@0
  1579
								break;
sl@0
  1580
							}	// end switch
sl@0
  1581
						}
sl@0
  1582
					}
sl@0
  1583
					break;
sl@0
  1584
				case EExitPending:
sl@0
  1585
					// if the thread is still pending then the guard timer must have expired
sl@0
  1586
					ERR_PRINTF1(_L("Thread timed out"));
sl@0
  1587
					// kill the test step thread
sl@0
  1588
					newThread.Kill(1);
sl@0
  1589
					// give the OS time to cleanup devices, etc.
sl@0
  1590
					// NB if the thread dies, the postamble will NOT run
sl@0
  1591
					User::After(2000000);
sl@0
  1592
					result = EFail;
sl@0
  1593
					break;
sl@0
  1594
				case EExitPanic:
sl@0
  1595
				default:
sl@0
  1596
					ERR_PRINTF1(_L("Test did not terminate, so failed"));
sl@0
  1597
					result = EFail;
sl@0
  1598
					break;
sl@0
  1599
				}
sl@0
  1600
sl@0
  1601
			// done with the test thread
sl@0
  1602
			newThread.Close();
sl@0
  1603
sl@0
  1604
			// stop clock
sl@0
  1605
			testStop.HomeTime();
sl@0
  1606
sl@0
  1607
			TUint testDuration = I64INT(testStop.MicroSecondsFrom(testStart).Int64());
sl@0
  1608
			testDuration /= 1000; // to microseconds
sl@0
  1609
			TUint testDurationMsec = testDuration % 1000;
sl@0
  1610
			TUint testDurationSec = testDuration / 1000;
sl@0
  1611
			INFO_PRINTF3(_L("Test took %d.%03d sec"), testDurationSec, testDurationMsec);
sl@0
  1612
sl@0
  1613
			// return the test verdict
sl@0
  1614
			delete data;
sl@0
  1615
			data = NULL;
sl@0
  1616
			return result;
sl@0
  1617
			}
sl@0
  1618
		}
sl@0
  1619
sl@0
  1620
	// the required suite has not been found
sl@0
  1621
	ERR_PRINTF3(_L("Error in test step:%S - cannot find suite:%S"),
sl@0
  1622
				&aStep, &aSuite );
sl@0
  1623
sl@0
  1624
	return ETestSuiteError;
sl@0
  1625
	}
sl@0
  1626
sl@0
  1627
/**
sl@0
  1628
 *
sl@0
  1629
 * Gets a step's heap and stack size.
sl@0
  1630
 *
sl@0
  1631
 * @param	"const CStepData& aStepData"
sl@0
  1632
 *			The step data
sl@0
  1633
 * @param	"TInt* aHeapSize"
sl@0
  1634
 *			Returns the step's heap size
sl@0
  1635
 * @param	"TInt* aStackSize"
sl@0
  1636
 *			Returns the step's stack size
sl@0
  1637
 *
sl@0
  1638
 * @xxxx
sl@0
  1639
 *
sl@0
  1640
 */
sl@0
  1641
void CParseLine::GetHeapAndStackSize(const CStepData* aStepData, TInt* aHeapSize, TInt* aStackSize)
sl@0
  1642
	{
sl@0
  1643
	CSuiteDll* suiteDll = aStepData->SuiteDll();
sl@0
  1644
	CTestSuite* testSuite = suiteDll->Suite();
sl@0
  1645
	testSuite->GetHeapAndStackSize(aStepData->Step(), aHeapSize, aStackSize);
sl@0
  1646
	}
sl@0
  1647
sl@0
  1648
/**
sl@0
  1649
 *
sl@0
  1650
 * Implements the RUN_PROGRAM script command.
sl@0
  1651
 *
sl@0
  1652
 * @param	"const TDesC& aText"
sl@0
  1653
 *			The script line
sl@0
  1654
 *
sl@0
  1655
 * @xxxx
sl@0
  1656
 *
sl@0
  1657
 */
sl@0
  1658
void CParseLine::RunProgram(const TDesC& aText)
sl@0
  1659
	{
sl@0
  1660
	TPtrC param;
sl@0
  1661
sl@0
  1662
	// use Tlex to decode the cmd line
sl@0
  1663
	TLex lex(aText);
sl@0
  1664
sl@0
  1665
	// step over the keyword
sl@0
  1666
	lex.NextToken();
sl@0
  1667
sl@0
  1668
	// get program name	
sl@0
  1669
	TPtrC token;
sl@0
  1670
	token.Set(lex.NextToken());
sl@0
  1671
sl@0
  1672
	// get the parameters
sl@0
  1673
	param.Set(lex.NextToken());
sl@0
  1674
sl@0
  1675
	INFO_PRINTF1(_L("Run Program "));
sl@0
  1676
sl@0
  1677
sl@0
  1678
	// In the ARM build run program as a new process
sl@0
  1679
	// use the rest of the text as parameters
sl@0
  1680
	RProcess program;
sl@0
  1681
	TInt ret = program.Create(token, lex.Remainder());
sl@0
  1682
sl@0
  1683
	if (ret != KErrNone)
sl@0
  1684
		{
sl@0
  1685
		TPtrC errortxt = CLog::EpocErrorToText(ret);
sl@0
  1686
		ERR_PRINTF2(_L("Failed to start process - error %S"), &errortxt);
sl@0
  1687
		return;
sl@0
  1688
		}
sl@0
  1689
	else
sl@0
  1690
		{
sl@0
  1691
		INFO_PRINTF1(_L("Program started"));
sl@0
  1692
sl@0
  1693
		// start program
sl@0
  1694
		TRequestStatus threadStatus;
sl@0
  1695
		program.Logon(threadStatus);
sl@0
  1696
		program.Resume();
sl@0
  1697
sl@0
  1698
		// wait for guard timer
sl@0
  1699
		User::WaitForRequest(threadStatus);
sl@0
  1700
sl@0
  1701
		// check return type
sl@0
  1702
		if (program.ExitType() == EExitPanic)
sl@0
  1703
			INFO_PRINTF1(_L("Program returned EExitPanic"));
sl@0
  1704
		else if (program.ExitType() == EExitPending)
sl@0
  1705
			INFO_PRINTF1(_L("Program returned EExitPending"));
sl@0
  1706
		else
sl@0
  1707
			INFO_PRINTF1(_L("Program returned EExitTerminate"));
sl@0
  1708
		}
sl@0
  1709
	}
sl@0
  1710
sl@0
  1711
/**
sl@0
  1712
 *
sl@0
  1713
 * Implements the LOG_SETTINGS script command.
sl@0
  1714
 * Command format is LOG_SETTINGS "put src." (1/0),
sl@0
  1715
 * "HTML format" (1/0)
sl@0
  1716
 *
sl@0
  1717
 * @param	"const TDesC& aText"
sl@0
  1718
 *			The script line
sl@0
  1719
 *
sl@0
  1720
 * @xxxx
sl@0
  1721
 *
sl@0
  1722
 */
sl@0
  1723
void CParseLine::LogSettings(const TDesC& aText)
sl@0
  1724
	{
sl@0
  1725
	// use Tlex to decode the cmd line
sl@0
  1726
	TLex lex(aText);
sl@0
  1727
sl@0
  1728
	// start at the begining
sl@0
  1729
	TPtrC token=lex.NextToken();
sl@0
  1730
sl@0
  1731
	// step over the keyword
sl@0
  1732
	//Get information about source
sl@0
  1733
	token.Set(lex.NextToken());
sl@0
  1734
sl@0
  1735
	TLex srcLex(token);
sl@0
  1736
	TInt isSrc = ETrue; //Shall we put src information?
sl@0
  1737
	if (srcLex.Val(isSrc) != KErrNone)
sl@0
  1738
		{
sl@0
  1739
		ERR_PRINTF2(_L("Error in LOG_SETTINGS: could not decode >%S< as value(0/1)"),
sl@0
  1740
					&token);
sl@0
  1741
		}
sl@0
  1742
	else
sl@0
  1743
		{
sl@0
  1744
		iLog->SetPutSrcInfo(isSrc) ;
sl@0
  1745
		}
sl@0
  1746
	//Get information about format
sl@0
  1747
	token.Set(lex.NextToken());
sl@0
  1748
	TLex htmlLex(token);
sl@0
  1749
sl@0
  1750
	if (htmlLex.Val(isSrc) != KErrNone)
sl@0
  1751
		{
sl@0
  1752
		ERR_PRINTF2(_L("Error in LOG_SETTINGS: could not decode >%S< as value(0/1)"),
sl@0
  1753
					&token);
sl@0
  1754
		}
sl@0
  1755
	else
sl@0
  1756
		{
sl@0
  1757
		iLog->SetHtmlLogMode(isSrc);
sl@0
  1758
		}
sl@0
  1759
	}
sl@0
  1760
sl@0
  1761
sl@0
  1762
/**
sl@0
  1763
 *
sl@0
  1764
 * Implements the LOAD_SUITE script command.
sl@0
  1765
 * This function loads a required test suite DLL
sl@0
  1766
 * It also creates a CTestSuite object as a record
sl@0
  1767
 * of the loaded DLL.
sl@0
  1768
 *
sl@0
  1769
 * @param	"const TDesC& aText"
sl@0
  1770
 *			The script line
sl@0
  1771
 *
sl@0
  1772
 * @xxxx
sl@0
  1773
 *
sl@0
  1774
 */
sl@0
  1775
void CParseLine::LoadSuiteL(const TDesC& aText)
sl@0
  1776
	{
sl@0
  1777
	// use Tlex to decode the cmd line
sl@0
  1778
	TLex lex(aText);
sl@0
  1779
sl@0
  1780
	// step over the keyword
sl@0
  1781
	lex.NextToken();
sl@0
  1782
sl@0
  1783
	// get suite name
sl@0
  1784
	TPtrC token;
sl@0
  1785
	token.Set(lex.NextToken());
sl@0
  1786
sl@0
  1787
	// check not already loaded
sl@0
  1788
	// by searching the list of loaded test suite DLLs for the required one
sl@0
  1789
	// start with the number of suites loaded
sl@0
  1790
	TInt noOfDlls = iArrayLoadedSuiteDll->Count();
sl@0
  1791
	for (TInt i = 0; i < noOfDlls; i++)
sl@0
  1792
		{
sl@0
  1793
		CSuiteDll* ptrSuite = iArrayLoadedSuiteDll->At(i);
sl@0
  1794
		TPtrC name = ptrSuite->Name();
sl@0
  1795
sl@0
  1796
		// check the names
sl@0
  1797
		if (name.FindF(token) != KErrNotFound)
sl@0
  1798
			{
sl@0
  1799
			// this suite DLL is already loaded
sl@0
  1800
	 		WARN_PRINTF2(_L("Warning: Test suite %S already loaded - not re-loaded"), &token);
sl@0
  1801
			return;
sl@0
  1802
			}
sl@0
  1803
		}
sl@0
  1804
sl@0
  1805
sl@0
  1806
	// create a new CSuiteDll object to store info on loaded DLL
sl@0
  1807
	CSuiteDll* newRef = NULL;
sl@0
  1808
sl@0
  1809
	newRef = CSuiteDll::NewL(token, iLog);
sl@0
  1810
sl@0
  1811
	CTestSuite* testSuite = newRef->Suite();
sl@0
  1812
sl@0
  1813
	// set default severity and logging system
sl@0
  1814
	testSuite->SetSeverity(iSeverity);
sl@0
  1815
	testSuite->SetLogSystem(iLog);
sl@0
  1816
	
sl@0
  1817
	// add to data
sl@0
  1818
	iArrayLoadedSuiteDll->AppendL(newRef);
sl@0
  1819
	}
sl@0
  1820
sl@0
  1821
sl@0
  1822
/**
sl@0
  1823
 *
sl@0
  1824
 * Unload all the loaded DLLs
sl@0
  1825
 *
sl@0
  1826
 * @xxxx
sl@0
  1827
 *
sl@0
  1828
 */
sl@0
  1829
void CParseLine::Unload()
sl@0
  1830
	{
sl@0
  1831
	if (iArrayLoadedSuiteDll)
sl@0
  1832
		{
sl@0
  1833
		// unload all the loaded DLLS and their records
sl@0
  1834
		iArrayLoadedSuiteDll->ResetAndDestroy();
sl@0
  1835
		}
sl@0
  1836
	}
sl@0
  1837
sl@0
  1838
/**
sl@0
  1839
 *
sl@0
  1840
 * Mark the heap
sl@0
  1841
 *
sl@0
  1842
 * @xxxx
sl@0
  1843
 *
sl@0
  1844
 */
sl@0
  1845
void CParseLine::HeapMark()
sl@0
  1846
	{
sl@0
  1847
	ERR_PRINTF1(_L("Warning: Command HEAP_MARK no longer supported. Heap marking/checking should be done within test code"));
sl@0
  1848
sl@0
  1849
	// __UHEAP_MARK;
sl@0
  1850
	}
sl@0
  1851
sl@0
  1852
sl@0
  1853
/**
sl@0
  1854
 *
sl@0
  1855
 * Check the heap
sl@0
  1856
 *
sl@0
  1857
 * @xxxx
sl@0
  1858
 *
sl@0
  1859
 */
sl@0
  1860
void CParseLine::HeapCheck()
sl@0
  1861
	{
sl@0
  1862
	ERR_PRINTF1(_L("Warning: Command HEAP_CHECK no longer supported. Heap marking/checking should be done within test code"));
sl@0
  1863
sl@0
  1864
	// __UHEAP_MARKEND;
sl@0
  1865
	}
sl@0
  1866
sl@0
  1867
/**
sl@0
  1868
 *
sl@0
  1869
 * Mark request
sl@0
  1870
 *
sl@0
  1871
 * @xxxx
sl@0
  1872
 *
sl@0
  1873
 */
sl@0
  1874
void CParseLine::RequestMark()
sl@0
  1875
	{
sl@0
  1876
	// get number of outstanding requetsts on thread before we run the test
sl@0
  1877
	iReqsAtStart = RThread().RequestCount();
sl@0
  1878
	INFO_PRINTF2(_L("Requests at the start %d "),iReqsAtStart);
sl@0
  1879
	}
sl@0
  1880
sl@0
  1881
sl@0
  1882
/**
sl@0
  1883
 *
sl@0
  1884
 * Check request
sl@0
  1885
 *
sl@0
  1886
 * @xxxx
sl@0
  1887
 *
sl@0
  1888
 */
sl@0
  1889
void CParseLine::RequestCheck()
sl@0
  1890
	{
sl@0
  1891
	// check the number of outstanding requests against recorded value
sl@0
  1892
	INFO_PRINTF3(_L("Requests at the start %d now %d"),
sl@0
  1893
				iReqsAtStart, RThread().RequestCount());
sl@0
  1894
sl@0
  1895
	if (iReqsAtStart != RThread().RequestCount())
sl@0
  1896
		{
sl@0
  1897
		ERR_PRINTF1(_L("Test failed on requests count"));
sl@0
  1898
sl@0
  1899
		// this result is only significant if every thing else has passed
sl@0
  1900
		if (iTestVerdict == EPass)
sl@0
  1901
			iTestVerdict = EFail;
sl@0
  1902
sl@0
  1903
		}
sl@0
  1904
	}
sl@0
  1905
sl@0
  1906
sl@0
  1907
/**
sl@0
  1908
 *
sl@0
  1909
 * Mark number of handles
sl@0
  1910
 *
sl@0
  1911
 * @xxxx
sl@0
  1912
 *
sl@0
  1913
 */
sl@0
  1914
void CParseLine::HandlesMark()
sl@0
  1915
	{
sl@0
  1916
	// get number of Handles *before* we start the program
sl@0
  1917
	RThread().HandleCount(iProcessHandleCountBefore, iThreadHandleCountBefore);
sl@0
  1918
sl@0
  1919
	INFO_PRINTF3(_L("HandlesMark : process handle count %d thread handle count %d"),
sl@0
  1920
				iProcessHandleCountBefore,
sl@0
  1921
				iThreadHandleCountBefore);
sl@0
  1922
	}
sl@0
  1923
sl@0
  1924
/**
sl@0
  1925
 *
sl@0
  1926
 * Check number of handles
sl@0
  1927
 *
sl@0
  1928
 * @xxxx
sl@0
  1929
 *
sl@0
  1930
 */
sl@0
  1931
void CParseLine::HandlesCheck()
sl@0
  1932
	{
sl@0
  1933
	TInt processHandleCountAfter;
sl@0
  1934
	TInt threadHandleCountAfter;
sl@0
  1935
	RThread().HandleCount(processHandleCountAfter, threadHandleCountAfter);
sl@0
  1936
sl@0
  1937
	INFO_PRINTF3(_L("HandlesCheck : process handle count %d thread handle count %d"),
sl@0
  1938
				processHandleCountAfter,
sl@0
  1939
				threadHandleCountAfter);
sl@0
  1940
sl@0
  1941
	// check that we are closing all the threads
sl@0
  1942
	if(iThreadHandleCountBefore != threadHandleCountAfter)
sl@0
  1943
		{
sl@0
  1944
		ERR_PRINTF1(_L("Test failed on thread handle count"));
sl@0
  1945
sl@0
  1946
		// this result is only significant if everything else has passed
sl@0
  1947
		if (iTestVerdict == EPass)
sl@0
  1948
			iTestVerdict = EFail;
sl@0
  1949
		}
sl@0
  1950
sl@0
  1951
	// check that we are closing all the handles
sl@0
  1952
	if(iProcessHandleCountBefore != processHandleCountAfter)
sl@0
  1953
		{
sl@0
  1954
		ERR_PRINTF1(_L("Test failed on process handle count"));
sl@0
  1955
sl@0
  1956
		// this result is only significant if everything else has passed
sl@0
  1957
		if (iTestVerdict == EPass)
sl@0
  1958
			iTestVerdict = EFail;
sl@0
  1959
		}
sl@0
  1960
	}
sl@0
  1961
sl@0
  1962
/**
sl@0
  1963
 *
sl@0
  1964
 * Traceable logging function for parseline.
sl@0
  1965
 * To be called only with macros
sl@0
  1966
 *
sl@0
  1967
 * @param	"const TText8* aFile"
sl@0
  1968
 *			Source code file name
sl@0
  1969
 *
sl@0
  1970
 * @param	"TInt aLine"
sl@0
  1971
 *			Source code line
sl@0
  1972
 *
sl@0
  1973
 * @param	"TInt aSeverity"
sl@0
  1974
 *			Severity level required to log
sl@0
  1975
 *
sl@0
  1976
 * @param	"TRefByValue<const TDesC16> aFmt"
sl@0
  1977
 *			Printf-style format.
sl@0
  1978
 *
sl@0
  1979
 * @param	"..."
sl@0
  1980
 *			Variable print parameters
sl@0
  1981
 *
sl@0
  1982
 * @xxxx
sl@0
  1983
 *
sl@0
  1984
 */
sl@0
  1985
void CParseLine::LogExtra(const TText8* aFile, TInt aLine, TInt aSeverity,
sl@0
  1986
		TRefByValue<const TDesC16> aFmt,...)
sl@0
  1987
	{
sl@0
  1988
	VA_LIST aList;
sl@0
  1989
	VA_START(aList, aFmt);
sl@0
  1990
sl@0
  1991
	if(LogSeverity::IsActive(aSeverity, iSeverity))
sl@0
  1992
		{
sl@0
  1993
		if(iLog)
sl@0
  1994
			{
sl@0
  1995
			iLog->LogExtra(aFile, aLine, aSeverity, aFmt, aList);
sl@0
  1996
			}
sl@0
  1997
		}
sl@0
  1998
sl@0
  1999
	VA_END(aList);
sl@0
  2000
	}
sl@0
  2001
sl@0
  2002
/**
sl@0
  2003
 *
sl@0
  2004
 * Get current suite name.
sl@0
  2005
 *
sl@0
  2006
 * @return "TPtrC"
sl@0
  2007
 *			The suite name
sl@0
  2008
 *
sl@0
  2009
 * @xxxx
sl@0
  2010
 *
sl@0
  2011
 */
sl@0
  2012
TPtrC CParseLine::CurrentSuiteName() const
sl@0
  2013
	{
sl@0
  2014
	return iCurrentSuiteName;
sl@0
  2015
	}
sl@0
  2016
sl@0
  2017
/**
sl@0
  2018
 *
sl@0
  2019
 * Get current step name.
sl@0
  2020
 *
sl@0
  2021
 * @return "TPtrC"
sl@0
  2022
 *			The step name
sl@0
  2023
 *
sl@0
  2024
 * @xxxx
sl@0
  2025
 *
sl@0
  2026
 */
sl@0
  2027
TPtrC CParseLine::CurrentStepName() const
sl@0
  2028
	{
sl@0
  2029
	return iCurrentStepName;
sl@0
  2030
	}
sl@0
  2031
sl@0
  2032
/**
sl@0
  2033
 *
sl@0
  2034
 * Create a thread with a unique name from a base thread name
sl@0
  2035
 * e.g. "TestThread" may become "TestThread00000002"
sl@0
  2036
 * This test will leave instantly if an error other than 
sl@0
  2037
 * KErrAlreadyExists occurs. 
sl@0
  2038
 *
sl@0
  2039
 *
sl@0
  2040
 * @param	"TDesC& aBaseName"
sl@0
  2041
 *			The base name to use.  This will be modified to contain
sl@0
  2042
 *			the new unique thread name if creation is successful.
sl@0
  2043
 *
sl@0
  2044
 * @param	"TDes& aThreadName"
sl@0
  2045
 *			The thread name to use.  This will be modified to contain
sl@0
  2046
 *			the new unique thread name if creation is successful.  This must
sl@0
  2047
 *			be non NULL.
sl@0
  2048
 *
sl@0
  2049
 * @param	"RThread&"
sl@0
  2050
 *			An RThread which will be created.  This must not be a valid handle.
sl@0
  2051
 *
sl@0
  2052
 * @param	"TThreadFunction aFunction"
sl@0
  2053
 *			Thread function to use in RThread.
sl@0
  2054
 *
sl@0
  2055
 * @param	"TInt aStackSize"
sl@0
  2056
 *			The size of the new thread's stack.
sl@0
  2057
 *
sl@0
  2058
 * @param	"TInt aHeapMinSize"
sl@0
  2059
 *			The minimum size for the new thread's heap.
sl@0
  2060
 *
sl@0
  2061
 * @param	"TInt aHeapMaxSize"
sl@0
  2062
 *			The maximum size for the new thread's heap.
sl@0
  2063
 *
sl@0
  2064
 * @param	"TAny *aPtr"
sl@0
  2065
 *			Data to pass to the new thread.
sl@0
  2066
 *
sl@0
  2067
 * @leave	"function will leave with an error code if a thread cannot
sl@0
  2068
 *			be created after KMaxThreadAttempts tries."
sl@0
  2069
 *
sl@0
  2070
 * @xxxx
sl@0
  2071
 *
sl@0
  2072
 */
sl@0
  2073
void CParseLine::CreateUniqueTestThreadL(const TDesC& aBaseName, TDes& aThreadName, RThread& aTestThread, TThreadFunction aFunction, TInt aStackSize, TInt aHeapMinSize, TInt aHeapMaxSize, TAny *aPtr)
sl@0
  2074
	{	
sl@0
  2075
	TInt res = KErrAlreadyExists;
sl@0
  2076
	
sl@0
  2077
	// attempt to create a thread with the name aBaseName + counter.
sl@0
  2078
	for (TUint i = 0; i < KMaxThreadAttempts; i++)
sl@0
  2079
		{
sl@0
  2080
		// copy the base thread name
sl@0
  2081
		aThreadName.Copy(aBaseName);
sl@0
  2082
		
sl@0
  2083
		// append the current counter to the threadname
sl@0
  2084
		aThreadName.AppendNumFixedWidth(i, EDecimal, 8);
sl@0
  2085
			
sl@0
  2086
		// run in a new thread, with a new heap		
sl@0
  2087
		res = aTestThread.Create(aThreadName,
sl@0
  2088
				aFunction,
sl@0
  2089
				aStackSize,
sl@0
  2090
				aHeapMinSize,
sl@0
  2091
				aHeapMaxSize,
sl@0
  2092
				aPtr);
sl@0
  2093
					
sl@0
  2094
		// if thread created successfully then we have
sl@0
  2095
		// a unique threadname else if an error code other
sl@0
  2096
		// than KErrAlreadyExists occurs then exit immediately.
sl@0
  2097
		if ((res == KErrNone) || (res != KErrAlreadyExists)) 
sl@0
  2098
			break;
sl@0
  2099
		}
sl@0
  2100
		
sl@0
  2101
	User::LeaveIfError(res);
sl@0
  2102
	}
sl@0
  2103
sl@0
  2104
/**
sl@0
  2105
 *
sl@0
  2106
 * Static constructor for CSuiteDll.
sl@0
  2107
 *
sl@0
  2108
 *
sl@0
  2109
 * @return	"CSuiteDll*"
sl@0
  2110
 *			The constructed CSuiteDll
sl@0
  2111
 *
sl@0
  2112
 * @xxxx
sl@0
  2113
 *
sl@0
  2114
 */
sl@0
  2115
CSuiteDll* CSuiteDll::NewL(const TDesC& aName, CLog* aLog)
sl@0
  2116
	{
sl@0
  2117
	CSuiteDll* self = new(ELeave) CSuiteDll;
sl@0
  2118
	CleanupStack::PushL(self);
sl@0
  2119
	self->ConstructL(aName, aLog);
sl@0
  2120
	CleanupStack::Pop();
sl@0
  2121
	return self;
sl@0
  2122
	}
sl@0
  2123
sl@0
  2124
/**
sl@0
  2125
 *
sl@0
  2126
 * CSuiteDLL second-phase constructor
sl@0
  2127
 * Loads a test suite dll and saves the name and test
sl@0
  2128
 * suite pointers.
sl@0
  2129
 *
sl@0
  2130
 * @param "TDesC& aName"
sl@0
  2131
 *			The test suite name
sl@0
  2132
 *
sl@0
  2133
 * @param "CLog* aLog"
sl@0
  2134
 *			The logger to use
sl@0
  2135
 *
sl@0
  2136
 * @xxxx
sl@0
  2137
 *
sl@0
  2138
 */
sl@0
  2139
void CSuiteDll::ConstructL(const TDesC& aName, CLog* aLog)
sl@0
  2140
	{
sl@0
  2141
	iLog = aLog;
sl@0
  2142
sl@0
  2143
	User::Check();
sl@0
  2144
	// load DLL by name
sl@0
  2145
	TInt ret = iLibrary.Load(aName, KTxtDLLpath);
sl@0
  2146
sl@0
  2147
	User::Check();
sl@0
  2148
sl@0
  2149
	if (ret == KErrNotFound)
sl@0
  2150
		{
sl@0
  2151
		iLog->LogExtra(__FILE8__, __LINE__, ESevrErr, _L("Test suite %S was not found. Check any other DLLs required by %S"), &aName, &aName);
sl@0
  2152
		User::Leave(ret);
sl@0
  2153
		}
sl@0
  2154
	else if (ret != KErrNone)
sl@0
  2155
		{
sl@0
  2156
		iLog->LogExtra(__FILE8__, __LINE__, ESevrErr, _L("Test suite %S found but would not load. Check any other DLLs required by %S"), &aName, &aName);
sl@0
  2157
		User::Leave(ret);
sl@0
  2158
		}
sl@0
  2159
sl@0
  2160
	// save the name
sl@0
  2161
	iName.Copy(aName);
sl@0
  2162
sl@0
  2163
	// get the interface pointer at ordinal 1
sl@0
  2164
	const TInt KLibraryOrdinal = 1;
sl@0
  2165
	TLibraryFunction  entryL = iLibrary.Lookup(KLibraryOrdinal);
sl@0
  2166
sl@0
  2167
    // Call this interface pointer to create new CTestSuite
sl@0
  2168
	// If this call goes to the wrong function then the test
sl@0
  2169
	// suite does not have the correct function at ordinal 1.
sl@0
  2170
	// This is usually caused by an error in the def file.
sl@0
  2171
    iTestSuite = REINTERPRET_CAST(CTestSuite*, entryL());
sl@0
  2172
sl@0
  2173
    // NB :- Second-phase constructor for CTestSuite has already been called in entryL() above.
sl@0
  2174
	// There is no need to call it again!
sl@0
  2175
sl@0
  2176
	// set suite severity level
sl@0
  2177
	iTestSuite->SetSeverity(iLog->Severity());
sl@0
  2178
sl@0
  2179
	// get the version information
sl@0
  2180
	TPtrC versiontxt = iTestSuite->GetVersion();
sl@0
  2181
sl@0
  2182
	// add to log
sl@0
  2183
	iLog->LogExtra(__FILE8__, __LINE__, ESevrInfo, _L("LOAD_SUITE %S version %S loaded ok"),&aName, &versiontxt );
sl@0
  2184
	}
sl@0
  2185
sl@0
  2186
/**
sl@0
  2187
 *
sl@0
  2188
 * CSuiteDLL destructor
sl@0
  2189
 *
sl@0
  2190
 * @xxxx
sl@0
  2191
 *
sl@0
  2192
 */
sl@0
  2193
CSuiteDll::~CSuiteDll()
sl@0
  2194
	{
sl@0
  2195
	// delete the TestSuiteObject in the loaded DLL
sl@0
  2196
	delete iTestSuite;
sl@0
  2197
sl@0
  2198
	// close and unload the library
sl@0
  2199
	iLibrary.Close();
sl@0
  2200
	}
sl@0
  2201
sl@0
  2202
/**
sl@0
  2203
 *
sl@0
  2204
 * CSuiteDLL accessor : suite
sl@0
  2205
 *
sl@0
  2206
 * @return "CTestSuite*"
sl@0
  2207
 *			The test suite.
sl@0
  2208
 *
sl@0
  2209
 * @xxxx
sl@0
  2210
 *
sl@0
  2211
 */
sl@0
  2212
CTestSuite* CSuiteDll::Suite() const
sl@0
  2213
	{
sl@0
  2214
	return iTestSuite;
sl@0
  2215
	}
sl@0
  2216
sl@0
  2217
/**
sl@0
  2218
 *
sl@0
  2219
 * CSuiteDLL accessor : suite name
sl@0
  2220
 *
sl@0
  2221
 * @return "TPtrC"
sl@0
  2222
 *			The suite name.
sl@0
  2223
 *
sl@0
  2224
 * @xxxx
sl@0
  2225
 *
sl@0
  2226
 */
sl@0
  2227
TPtrC CSuiteDll::Name() const
sl@0
  2228
	{
sl@0
  2229
	return iName;
sl@0
  2230
	}
sl@0
  2231
sl@0
  2232
/**
sl@0
  2233
 *
sl@0
  2234
 * CStepData
sl@0
  2235
 *
sl@0
  2236
 * @xxxx
sl@0
  2237
 *
sl@0
  2238
 */
sl@0
  2239
CStepData* CStepData::NewL(const TDesC& aStep, const TDesC& aConfig, CSuiteDll* aSuite)
sl@0
  2240
	{
sl@0
  2241
	return NewL(aStep, aConfig, KNullDesC, aSuite);
sl@0
  2242
	}
sl@0
  2243
sl@0
  2244
CStepData* CStepData::NewL(const TDesC& aStep, const TDesC& aConfig, const TDesC& aParamSet, CSuiteDll* aSuite)
sl@0
  2245
	{
sl@0
  2246
	CStepData* self = new(ELeave) CStepData;
sl@0
  2247
	CleanupStack::PushL(self);
sl@0
  2248
	self->ConstructL(aStep, aConfig, aParamSet, aSuite);
sl@0
  2249
	CleanupStack::Pop();
sl@0
  2250
	return self;
sl@0
  2251
	}
sl@0
  2252
sl@0
  2253
CStepData::CStepData()
sl@0
  2254
	{
sl@0
  2255
	}
sl@0
  2256
sl@0
  2257
CStepData::~CStepData()
sl@0
  2258
	{
sl@0
  2259
	}
sl@0
  2260
sl@0
  2261
void CStepData::ConstructL(const TDesC& aStep, const TDesC& aConfig, const TDesC& aParamSet, CSuiteDll* aSuite)
sl@0
  2262
	{
sl@0
  2263
	iStep = aStep;
sl@0
  2264
	iConfig = aConfig;
sl@0
  2265
	iParamSet = aParamSet;
sl@0
  2266
	iSuite = aSuite;
sl@0
  2267
	}
sl@0
  2268
sl@0
  2269
const TDesC& CStepData::ParamSet() const
sl@0
  2270
	{
sl@0
  2271
	return iParamSet;
sl@0
  2272
	}
sl@0
  2273
sl@0
  2274