os/mm/mmtestenv/mmtestfw/Source/TestFrameworkClient/TestStep.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
//
sl@0
    15
sl@0
    16
// EPOC includes
sl@0
    17
#include <e32base.h>
sl@0
    18
sl@0
    19
// Test system includes
sl@0
    20
#include <testframework.h>
sl@0
    21
#include "TestIniData.h"
sl@0
    22
#include "asyncscheduler.h"
sl@0
    23
sl@0
    24
// do not export if Unit Testing
sl@0
    25
#if defined (__TSU_TESTFRAMEWORK__)
sl@0
    26
#undef EXPORT_C
sl@0
    27
#define EXPORT_C
sl@0
    28
#endif
sl@0
    29
sl@0
    30
static const TUint16*const KVerdictString[] = // must match up with TVerdict
sl@0
    31
	{
sl@0
    32
	_S("EPass"),
sl@0
    33
	_S("EFail"),
sl@0
    34
	_S("EInconclusive"),
sl@0
    35
	_S("ETestSuiteError"),
sl@0
    36
	_S("EAbort"),
sl@0
    37
	_S("EKnownFailure")
sl@0
    38
	};
sl@0
    39
sl@0
    40
/**
sl@0
    41
 *
sl@0
    42
 * Test step constructor.
sl@0
    43
 *
sl@0
    44
 * @xxxx
sl@0
    45
 *
sl@0
    46
 */
sl@0
    47
EXPORT_C RTestStep::RTestStep()
sl@0
    48
	{
sl@0
    49
	iTestStepResult = EPass;
sl@0
    50
	iTestStepName.Zero();
sl@0
    51
	iSuite = NULL;
sl@0
    52
	iConfigData = NULL;
sl@0
    53
	iConfigDataAvailable = EFalse;
sl@0
    54
	iStackSize = KTestStackSize;
sl@0
    55
	iHeapSize = KMaxTestThreadHeapSize;
sl@0
    56
sl@0
    57
	// NB : RTestStep has no destructor
sl@0
    58
	}
sl@0
    59
sl@0
    60
/**
sl@0
    61
 *
sl@0
    62
 * CBase compatibility functionality.
sl@0
    63
 *
sl@0
    64
 * @xxxx
sl@0
    65
 *
sl@0
    66
 */
sl@0
    67
sl@0
    68
sl@0
    69
EXPORT_C TAny* RTestStep::operator new(TUint aSize, TAny* aBase) __NO_THROW
sl@0
    70
	{
sl@0
    71
	Mem::FillZ(aBase,aSize);
sl@0
    72
	return(aBase);
sl@0
    73
	}
sl@0
    74
sl@0
    75
sl@0
    76
EXPORT_C TAny* RTestStep::operator new(TUint aSize, TLeave)
sl@0
    77
	{
sl@0
    78
	return newL(aSize);		// will leave on alloc failure
sl@0
    79
	}
sl@0
    80
sl@0
    81
EXPORT_C TAny* RTestStep::operator new(TUint aSize) __NO_THROW
sl@0
    82
	{
sl@0
    83
	TAny* pM=User::Alloc(aSize);
sl@0
    84
	if (pM)
sl@0
    85
		Mem::FillZ(pM,aSize);
sl@0
    86
	return(pM);
sl@0
    87
	}
sl@0
    88
sl@0
    89
EXPORT_C TAny* RTestStep::newL(TUint aSize)
sl@0
    90
	{
sl@0
    91
	TAny* pM=User::AllocL(aSize);
sl@0
    92
	Mem::FillZ(pM,aSize);
sl@0
    93
	return pM;
sl@0
    94
	}
sl@0
    95
sl@0
    96
EXPORT_C TAny* RTestStep::operator new(TUint aSize,TUint anExtraSize) __NO_THROW
sl@0
    97
	{
sl@0
    98
	aSize+=anExtraSize;
sl@0
    99
	TAny *pM=User::Alloc(aSize);
sl@0
   100
	if (pM)
sl@0
   101
		Mem::FillZ(pM,aSize);
sl@0
   102
	return(pM);
sl@0
   103
	}
sl@0
   104
sl@0
   105
/**
sl@0
   106
 *
sl@0
   107
 * Pre-preamble for all test steps. This grows the cleanup stack to
sl@0
   108
 * allow for allocation errors.
sl@0
   109
 *
sl@0
   110
 * @xxxx
sl@0
   111
 *
sl@0
   112
 */
sl@0
   113
EXPORT_C void RTestStep::PreOpenL()
sl@0
   114
	{
sl@0
   115
	TAny* dummyPtr = NULL;
sl@0
   116
	for(TInt i = 0 ; i < 20 ; i++)
sl@0
   117
		CleanupStack::PushL(dummyPtr); // Grow the cleanup stack.
sl@0
   118
	CleanupStack::PopAndDestroy(20);
sl@0
   119
	}
sl@0
   120
sl@0
   121
/**
sl@0
   122
 *
sl@0
   123
 * Default empty implementation of OpenL (preamble).
sl@0
   124
 * Test steps can override this to implement required code.
sl@0
   125
 *
sl@0
   126
 * @return	"TVerdict"
sl@0
   127
 *			Result of preamble (should only be EPass or EInconclusive)
sl@0
   128
 *
sl@0
   129
 * @xxxx
sl@0
   130
 *
sl@0
   131
 */
sl@0
   132
EXPORT_C TVerdict RTestStep::OpenL()
sl@0
   133
	{
sl@0
   134
	// for backward compatibility with CTestStep
sl@0
   135
	return DoTestStepPreambleL();
sl@0
   136
	}
sl@0
   137
sl@0
   138
/**
sl@0
   139
 *
sl@0
   140
 * Default empty implementation of CleanupAfterOpenFail (preamble cleanup).
sl@0
   141
 * Test steps can override this to implement required code.
sl@0
   142
 *
sl@0
   143
 * @xxxx
sl@0
   144
 *
sl@0
   145
 */
sl@0
   146
EXPORT_C void RTestStep::CleanupAfterOpenFail()
sl@0
   147
	{
sl@0
   148
	// default empty implementation
sl@0
   149
	// a step should implement its own method if required
sl@0
   150
	}
sl@0
   151
sl@0
   152
/**
sl@0
   153
 *
sl@0
   154
 * Default empty implementation of Close (postamble)
sl@0
   155
 * Test steps can override this to implement required code.
sl@0
   156
 *
sl@0
   157
 * NB this does NOT leave - any leaves should be trapped and panicked.
sl@0
   158
 *
sl@0
   159
 * @xxxx
sl@0
   160
 *
sl@0
   161
 */
sl@0
   162
EXPORT_C void RTestStep::Close()
sl@0
   163
	{
sl@0
   164
	// for backward compatibility with CTestStep
sl@0
   165
	_LIT(KPanicStr, "RTestStep::Close");
sl@0
   166
	TVerdict ver = EPass;
sl@0
   167
	TRAPD(err, ver = DoTestStepPostambleL());
sl@0
   168
	if(err != KErrNone)
sl@0
   169
		User::Panic(KPanicStr, 0);
sl@0
   170
	// any DoTestStepPostambleL() which returns EInconclusive should be panicked
sl@0
   171
 	if(ver != EPass)
sl@0
   172
		User::Panic(KPanicStr, 1);
sl@0
   173
sl@0
   174
	}
sl@0
   175
sl@0
   176
/**
sl@0
   177
 *
sl@0
   178
 * Set the test suite
sl@0
   179
 *
sl@0
   180
 * @param	"CTestSuite*"
sl@0
   181
 *			The test suite
sl@0
   182
 *
sl@0
   183
 * @xxxx
sl@0
   184
 *
sl@0
   185
 */
sl@0
   186
EXPORT_C void RTestStep::SetSuite(CTestSuite* aSuite)
sl@0
   187
	{
sl@0
   188
	iSuite = aSuite;
sl@0
   189
	}
sl@0
   190
sl@0
   191
/**
sl@0
   192
 *
sl@0
   193
 * Set the test result
sl@0
   194
 *
sl@0
   195
 * @param	"TVerdict"
sl@0
   196
 *			The test result
sl@0
   197
 *
sl@0
   198
 * @xxxx
sl@0
   199
 *
sl@0
   200
 */
sl@0
   201
EXPORT_C void RTestStep::SetResult(TVerdict aResult)
sl@0
   202
	{
sl@0
   203
	iTestStepResult = aResult;
sl@0
   204
	}
sl@0
   205
sl@0
   206
/**
sl@0
   207
 *
sl@0
   208
 * Get the step name
sl@0
   209
 *
sl@0
   210
 * @return	"TPtrC"
sl@0
   211
 *			The step name
sl@0
   212
 *
sl@0
   213
 * @xxxx
sl@0
   214
 *
sl@0
   215
 */
sl@0
   216
EXPORT_C TPtrC RTestStep::StepName() const
sl@0
   217
	{
sl@0
   218
	return iTestStepName;
sl@0
   219
	}
sl@0
   220
sl@0
   221
/**
sl@0
   222
 *
sl@0
   223
 * General logging function for test steps.
sl@0
   224
 *
sl@0
   225
 * @param	"TRefByValue<const TDesC16> aFmt"
sl@0
   226
 *			Printf-style aFmt.
sl@0
   227
 *
sl@0
   228
 * @param	"..."
sl@0
   229
 *			Variable print parameters
sl@0
   230
 *
sl@0
   231
 * @xxxx
sl@0
   232
 *
sl@0
   233
 */
sl@0
   234
EXPORT_C void RTestStep::Log(TRefByValue<const TDesC16> aFmt, ...)
sl@0
   235
	{
sl@0
   236
    
sl@0
   237
	VA_LIST aList;
sl@0
   238
	VA_START(aList, aFmt);
sl@0
   239
sl@0
   240
	TIntegrationTestLog16Overflow iOverflow16;
sl@0
   241
sl@0
   242
	// decode formatted data for display on console
sl@0
   243
	TBuf<KMaxLogLineLength> lineBuf;
sl@0
   244
	lineBuf.AppendFormatList(aFmt, aList, &iOverflow16);
sl@0
   245
sl@0
   246
	// send the data to the log system via the suite
sl@0
   247
	iSuite->Log(_L("%S"),&lineBuf);
sl@0
   248
sl@0
   249
	VA_END(aList); 
sl@0
   250
sl@0
   251
	}
sl@0
   252
sl@0
   253
/**
sl@0
   254
 *
sl@0
   255
 * General logging function for test steps, with severity.
sl@0
   256
 *
sl@0
   257
 * @param	"TInt aSeverity"
sl@0
   258
 *			Severity level required to log
sl@0
   259
 *
sl@0
   260
 * @param	"TRefByValue<const TDesC16> aFmt"
sl@0
   261
 *			Printf-style aFmt.
sl@0
   262
 *
sl@0
   263
 * @param	"..."
sl@0
   264
 *			Variable print parameters
sl@0
   265
 *
sl@0
   266
 * @xxxx
sl@0
   267
 *
sl@0
   268
 */
sl@0
   269
EXPORT_C void RTestStep::Log( TInt aSeverity, TRefByValue<const TDesC16> aFmt, ... )
sl@0
   270
{
sl@0
   271
	VA_LIST aList;
sl@0
   272
	VA_START(aList, aFmt);
sl@0
   273
sl@0
   274
	TIntegrationTestLog16Overflow iOverflow16;
sl@0
   275
sl@0
   276
	// decode formatted data for display on console
sl@0
   277
	TBuf<KMaxLogLineLength> lineBuf;
sl@0
   278
	lineBuf.AppendFormatList(aFmt, aList, &iOverflow16);
sl@0
   279
sl@0
   280
	// send the data to the log system via the suite
sl@0
   281
	if(LogSeverity::IsActive(aSeverity, iSuite->Severity()))
sl@0
   282
		iSuite->Log(aSeverity, _L("%S"),&lineBuf);
sl@0
   283
sl@0
   284
	VA_END(aList); 
sl@0
   285
}
sl@0
   286
sl@0
   287
/**
sl@0
   288
 *
sl@0
   289
 * Traceable logging function for test steps.
sl@0
   290
 *
sl@0
   291
 * @param	"const TText8* aFile"
sl@0
   292
 *			Source code file name
sl@0
   293
 *
sl@0
   294
 * @param	"TInt aLine"
sl@0
   295
 *			Source code line
sl@0
   296
 *
sl@0
   297
 * @param	"TInt aSeverity"
sl@0
   298
 *			Severity level required to log
sl@0
   299
 *
sl@0
   300
 * @param	"TRefByValue<const TDesC16> aFmt"
sl@0
   301
 *			Printf-style format.
sl@0
   302
 *
sl@0
   303
 * @param	"..."
sl@0
   304
 *			Variable print parameters
sl@0
   305
 *
sl@0
   306
 * @xxxx
sl@0
   307
 *
sl@0
   308
 */
sl@0
   309
EXPORT_C void RTestStep::LogExtra(const TText8* aFile, TInt aLine, TInt aSeverity,
sl@0
   310
		TRefByValue<const TDesC16> aFmt,...)
sl@0
   311
	{
sl@0
   312
	VA_LIST aList;
sl@0
   313
	VA_START(aList, aFmt);
sl@0
   314
sl@0
   315
	TIntegrationTestLog16Overflow iOverflow16;
sl@0
   316
sl@0
   317
	// decode formatted data for display on console
sl@0
   318
	TBuf<KMaxLogLineLength> lineBuf;
sl@0
   319
	lineBuf.AppendFormatList(aFmt, aList, &iOverflow16);
sl@0
   320
sl@0
   321
	// send the data to the log system via the suite
sl@0
   322
	if(LogSeverity::IsActive(aSeverity, iSuite->Severity()))
sl@0
   323
		iSuite->LogExtra(aFile, aLine, aSeverity, lineBuf);
sl@0
   324
sl@0
   325
	VA_END(aList); 
sl@0
   326
	}
sl@0
   327
	
sl@0
   328
/**
sl@0
   329
Set default paramSet
sl@0
   330
Test steps can use this when looking up configs, to provide a level of script control
sl@0
   331
*/
sl@0
   332
void RTestStep::SetDefaultParamSet(const TDesC& aParamSet)
sl@0
   333
	{
sl@0
   334
	iDefaultParamSet.Set(aParamSet);
sl@0
   335
	}
sl@0
   336
sl@0
   337
/**
sl@0
   338
 *
sl@0
   339
 * Load a configuration file.
sl@0
   340
 * If successful, data member iConfigDataAvailable is set.
sl@0
   341
 *
sl@0
   342
 * @param	"TPtrC aConfig"
sl@0
   343
 *			The configuration file name.
sl@0
   344
 *
sl@0
   345
 * @xxxx
sl@0
   346
 *
sl@0
   347
 */
sl@0
   348
EXPORT_C void RTestStep::LoadConfig(const TDesC& aConfig)
sl@0
   349
	{
sl@0
   350
sl@0
   351
	// if a config file supplied then use
sl@0
   352
	if (aConfig.Length() != 0)
sl@0
   353
		{
sl@0
   354
sl@0
   355
		// get the full pathname default drive name and extension
sl@0
   356
		_LIT(KRelated,"C:\\config.ini"); 
sl@0
   357
		TParse configFileName;
sl@0
   358
		TInt returnCode = configFileName.Set(aConfig, &KRelated, NULL);
sl@0
   359
sl@0
   360
		if (returnCode != KErrNone)
sl@0
   361
			{
sl@0
   362
			// error opening FileManager
sl@0
   363
			ERR_PRINTF2(_L("Error opening config file %S"), &(configFileName.FullName())); 
sl@0
   364
			}
sl@0
   365
sl@0
   366
		// create and load the CTestIniData object
sl@0
   367
		TRAPD(r, iConfigData = CTestIniData::NewL(configFileName.FullName()));
sl@0
   368
		
sl@0
   369
		// check if loaded ok
sl@0
   370
		if (r == KErrNone)
sl@0
   371
			{
sl@0
   372
			// loaded ok
sl@0
   373
			iConfigDataAvailable = ETrue;
sl@0
   374
			}
sl@0
   375
		else
sl@0
   376
			{
sl@0
   377
			// failed to load
sl@0
   378
			iConfigDataAvailable = EFalse;
sl@0
   379
			iConfigData = NULL;
sl@0
   380
sl@0
   381
			// report error 
sl@0
   382
			TPtrC errortxt = CLog::EpocErrorToText(r);
sl@0
   383
			ERR_PRINTF2(_L("Failed to load config data file - error %S"), &errortxt);
sl@0
   384
			}
sl@0
   385
		}
sl@0
   386
	}
sl@0
   387
sl@0
   388
/**
sl@0
   389
 *
sl@0
   390
 * Unload any existing configuration data.
sl@0
   391
 *
sl@0
   392
 * @xxxx
sl@0
   393
 *
sl@0
   394
 */
sl@0
   395
EXPORT_C void RTestStep::UnloadConfig()
sl@0
   396
	{
sl@0
   397
	iConfigDataAvailable = EFalse;
sl@0
   398
sl@0
   399
	// clean up Config data object
sl@0
   400
	delete iConfigData;
sl@0
   401
	iConfigData = NULL;
sl@0
   402
sl@0
   403
	}
sl@0
   404
sl@0
   405
/**
sl@0
   406
 *
sl@0
   407
 * Check the config file for a boolean value
sl@0
   408
 *
sl@0
   409
 * @param	"const TDesC &aSectName"
sl@0
   410
 *			Section name to check in
sl@0
   411
 *
sl@0
   412
 * @param	"const TDesC &aKeyName"
sl@0
   413
 *			Key name to check for
sl@0
   414
 *
sl@0
   415
 * @param	"Bool &aResult"
sl@0
   416
 *			TBool returned from config file
sl@0
   417
 *
sl@0
   418
 * @return	"TBool"
sl@0
   419
 *			Result (ETrue if found)
sl@0
   420
 *
sl@0
   421
 * @xxxx
sl@0
   422
 *
sl@0
   423
 */
sl@0
   424
EXPORT_C TBool RTestStep::GetBoolFromConfig(const TDesC &aSectName, const TDesC &aKeyName, TBool &aResult)
sl@0
   425
	{
sl@0
   426
	// check file available
sl@0
   427
	if (!iConfigDataAvailable)
sl@0
   428
		{
sl@0
   429
		ERR_PRINTF1(_L("No config file available"));
sl@0
   430
		return EFalse;
sl@0
   431
		}
sl@0
   432
sl@0
   433
	TBool ret = EFalse;
sl@0
   434
	TPtrC result;
sl@0
   435
sl@0
   436
	// get the value 
sl@0
   437
	ret = iConfigData->FindVar(aSectName, aKeyName, result);
sl@0
   438
sl@0
   439
	// if failed to decode display error
sl@0
   440
	if (!ret) 
sl@0
   441
		{
sl@0
   442
		// display error message
sl@0
   443
		ERR_PRINTF3(_L("Failed to read section:%S key:%S "),
sl@0
   444
				&aSectName, &aKeyName );
sl@0
   445
sl@0
   446
		// return fail
sl@0
   447
		return EFalse;
sl@0
   448
		}
sl@0
   449
sl@0
   450
	// set result as a TBool
sl@0
   451
	if (result.FindF(_L("true")) == KErrNotFound)
sl@0
   452
		aResult = EFalse;
sl@0
   453
	else
sl@0
   454
		aResult = ETrue;
sl@0
   455
	// return success
sl@0
   456
	return ETrue;
sl@0
   457
	}
sl@0
   458
sl@0
   459
/**
sl@0
   460
 *
sl@0
   461
 * Check the config file for a TInt value
sl@0
   462
 *
sl@0
   463
 * @param	"const TDesC &aSectName"
sl@0
   464
 *			Section name to check in
sl@0
   465
 *
sl@0
   466
 * @param	"const TDesC &aKeyName"
sl@0
   467
 *			Key name to check for
sl@0
   468
 *
sl@0
   469
 * @param	"TInt &aResult"
sl@0
   470
 *			TInt returned from config file
sl@0
   471
 *
sl@0
   472
 * @return	"TBool"
sl@0
   473
 *			Result (ETrue if found)
sl@0
   474
 *
sl@0
   475
 * @xxxx
sl@0
   476
 *
sl@0
   477
 */
sl@0
   478
EXPORT_C TBool RTestStep::GetIntFromConfig(const TDesC &aSectName, const TDesC &aKeyName, TInt &aResult)
sl@0
   479
	{
sl@0
   480
	// check file available
sl@0
   481
	if ( !iConfigDataAvailable )
sl@0
   482
		{
sl@0
   483
		ERR_PRINTF1(_L("No config file available"));
sl@0
   484
		return EFalse;
sl@0
   485
		}	
sl@0
   486
sl@0
   487
	TBool ret = EFalse;
sl@0
   488
	TPtrC result;
sl@0
   489
sl@0
   490
	// get the value 
sl@0
   491
	ret = iConfigData->FindVar(aSectName, aKeyName, result);
sl@0
   492
sl@0
   493
	// if failed to decode display error
sl@0
   494
	if (!ret) 
sl@0
   495
		{
sl@0
   496
		// display error message
sl@0
   497
		ERR_PRINTF3(_L("Failed to read section:%S key:%S "),
sl@0
   498
				&aSectName, &aKeyName );
sl@0
   499
sl@0
   500
		// return fail
sl@0
   501
		return EFalse;
sl@0
   502
		}
sl@0
   503
sl@0
   504
	// use TLex to convert to a TInt
sl@0
   505
	TLex lex(result);
sl@0
   506
	if (lex.Val(aResult) == KErrNone)
sl@0
   507
		return ETrue;
sl@0
   508
	else
sl@0
   509
		return EFalse;
sl@0
   510
}
sl@0
   511
sl@0
   512
/**
sl@0
   513
 *
sl@0
   514
 * Check the config file for a string TPtr value
sl@0
   515
 *
sl@0
   516
 * @param	"const TDesC &aSectName"
sl@0
   517
 *			Section name to check in
sl@0
   518
 *
sl@0
   519
 * @param	"const TDesC &aKeyName"
sl@0
   520
 *			Key name to check for
sl@0
   521
 *
sl@0
   522
 * @param	"TPtrC &aResult"
sl@0
   523
 *			String returned from config file
sl@0
   524
 *
sl@0
   525
 * @return	"TBool"
sl@0
   526
 *			Result (ETrue if found)
sl@0
   527
 *
sl@0
   528
 * @xxxx
sl@0
   529
 *
sl@0
   530
 */
sl@0
   531
EXPORT_C TBool RTestStep::GetStringFromConfig(const TDesC &aSectName, const TDesC &aKeyName, TPtrC &aResult)
sl@0
   532
	{
sl@0
   533
	// check file available
sl@0
   534
	if (!iConfigDataAvailable)
sl@0
   535
		{
sl@0
   536
		ERR_PRINTF1(_L("No config file available"));
sl@0
   537
		return EFalse;
sl@0
   538
		}	
sl@0
   539
sl@0
   540
	// get the value 
sl@0
   541
	TBool ret = iConfigData->FindVar(aSectName, aKeyName, aResult);
sl@0
   542
sl@0
   543
	// if failed to decode display error
sl@0
   544
	if (ret == EFalse) 
sl@0
   545
		{
sl@0
   546
		ERR_PRINTF3(_L("Failed to read section:%S key:%S "),
sl@0
   547
				&aSectName, &aKeyName );
sl@0
   548
		}
sl@0
   549
sl@0
   550
	return ret;
sl@0
   551
}
sl@0
   552
sl@0
   553
/**
sl@0
   554
 * Reads the value present from the test steps ini file within the mentioned section name and key name
sl@0
   555
 * Copies the value to the TInt reference passed in
sl@0
   556
 * @param aSectName - Section within the test steps ini file
sl@0
   557
 * @param aKeyName - Name of a key within a section
sl@0
   558
 * @return aResult - The integer value of the Hex input
sl@0
   559
 * @return TBool - ETrue for found, EFalse for not found 
sl@0
   560
 */	
sl@0
   561
EXPORT_C TBool RTestStep::GetHexFromConfig(const TDesC &aSectName,const TDesC &aKeyName,TInt &aResult)
sl@0
   562
	{
sl@0
   563
	TPtrC result;
sl@0
   564
	if(!iConfigData)
sl@0
   565
		return EFalse;
sl@0
   566
	if(!iConfigData->FindVar(aSectName, aKeyName, result))
sl@0
   567
		return EFalse;
sl@0
   568
	TLex lex(result);
sl@0
   569
	TInt err = lex.Val((TUint &)aResult, EHex);
sl@0
   570
	if(err)
sl@0
   571
		return EFalse;
sl@0
   572
	
sl@0
   573
	return(ETrue);
sl@0
   574
	}
sl@0
   575
	
sl@0
   576
/**
sl@0
   577
 *
sl@0
   578
 * Default empty implementation of DoTestStepPreambleL.
sl@0
   579
 * Test steps can override this to implement required code.
sl@0
   580
 *
sl@0
   581
 * @return	"TVerdict"
sl@0
   582
 *			Result of preamble (should only be EPass or EInconclusive)
sl@0
   583
 *
sl@0
   584
 * @xxxx
sl@0
   585
 *
sl@0
   586
 */
sl@0
   587
// for backward compatibility with CTestStep
sl@0
   588
EXPORT_C TVerdict RTestStep::DoTestStepPreambleL()
sl@0
   589
	{
sl@0
   590
	return EPass;
sl@0
   591
	}
sl@0
   592
sl@0
   593
/**
sl@0
   594
 *
sl@0
   595
 * Default empty implementation of DoTestStepPostambleL.
sl@0
   596
 * Test steps can override this to implement required code.
sl@0
   597
 *
sl@0
   598
 * @return	"TVerdict"
sl@0
   599
 *			Result of postamble (should only be EPass or EInconclusive)
sl@0
   600
 *
sl@0
   601
 * @xxxx
sl@0
   602
 *
sl@0
   603
 */
sl@0
   604
// for backward compatibility with CTestStep
sl@0
   605
EXPORT_C TVerdict RTestStep::DoTestStepPostambleL()
sl@0
   606
	{
sl@0
   607
	return EPass;
sl@0
   608
	}
sl@0
   609
sl@0
   610
/**
sl@0
   611
 *
sl@0
   612
 * Traceable Boolean condition tester.
sl@0
   613
 * If the condition is not true, record an error.
sl@0
   614
 *
sl@0
   615
 * @param	"TBool aCondition"
sl@0
   616
 *			Condition to be checked
sl@0
   617
 *
sl@0
   618
 * @param	"const TText8* aFile"
sl@0
   619
 *			Source code file name
sl@0
   620
 *
sl@0
   621
 * @param	"TInt aLine"
sl@0
   622
 *			Source code line
sl@0
   623
 *
sl@0
   624
 * @xxxx
sl@0
   625
 *
sl@0
   626
 */
sl@0
   627
EXPORT_C void RTestStep::TestBooleanTrue(TBool aCondition, const TText8* aFile, TInt aLine)
sl@0
   628
	{
sl@0
   629
sl@0
   630
	// check condition
sl@0
   631
	if (aCondition)
sl@0
   632
		return;
sl@0
   633
sl@0
   634
	// this is only relevant if the current result is pass
sl@0
   635
	if (iTestStepResult == EPass)
sl@0
   636
		iTestStepResult = EFail;
sl@0
   637
sl@0
   638
	// convert filename for log
sl@0
   639
	TBuf<KMaxLogFilenameLength> fileName;
sl@0
   640
	TPtrC8 fileName8(aFile);
sl@0
   641
	fileName.Copy(fileName8);  // TText8->TBuf16
sl@0
   642
sl@0
   643
	// display a log message
sl@0
   644
 	ERR_PRINTF3(_L("Test Failed in file:%S line:%d"), &fileName, aLine);
sl@0
   645
sl@0
   646
	}
sl@0
   647
sl@0
   648
/**
sl@0
   649
 *
sl@0
   650
 * Traceable Boolean condition tester.
sl@0
   651
 * If the condition is not true, record an error and leave.
sl@0
   652
 *
sl@0
   653
 * @param	"TBool aCondition"
sl@0
   654
 *			Condition to be checked
sl@0
   655
 *
sl@0
   656
 * @param	"const TText8* aFile"
sl@0
   657
 *			Source code file name
sl@0
   658
 *
sl@0
   659
 * @param	"TInt aLine"
sl@0
   660
 *			Source code line
sl@0
   661
 *
sl@0
   662
 * @xxxx
sl@0
   663
 *
sl@0
   664
 */
sl@0
   665
EXPORT_C void RTestStep::TestBooleanTrueL(TBool aCondition, const TText8* aFile, TInt aLine)
sl@0
   666
	{
sl@0
   667
sl@0
   668
	// check condition
sl@0
   669
	if (aCondition)
sl@0
   670
		return;
sl@0
   671
sl@0
   672
	// this is only relevant if the current result is pass
sl@0
   673
	if (iTestStepResult == EPass)
sl@0
   674
		iTestStepResult = EFail;
sl@0
   675
sl@0
   676
	// convert filename for log
sl@0
   677
	TBuf<KMaxLogFilenameLength> fileName;
sl@0
   678
	TPtrC8 fileName8(aFile);
sl@0
   679
	fileName.Copy(fileName8);  // TText8->TBuf16
sl@0
   680
sl@0
   681
	// display a log message
sl@0
   682
 	ERR_PRINTF3(_L("Test Failed in file:%S line:%d"), &fileName, aLine);
sl@0
   683
sl@0
   684
	// leave with error code
sl@0
   685
	User::Leave(KTestErrorCode);
sl@0
   686
sl@0
   687
	}
sl@0
   688
sl@0
   689
/**
sl@0
   690
 *
sl@0
   691
 * Traceable Boolean condition tester.
sl@0
   692
 * If the condition is not true, record an error with the supplied 
sl@0
   693
 * error code, and leave.
sl@0
   694
 *
sl@0
   695
 * @param	"TBool aCondition"
sl@0
   696
 *			Condition to be checked
sl@0
   697
 *
sl@0
   698
 * @param	"TInt aErrorCode"
sl@0
   699
 *			Supplied error code
sl@0
   700
 *
sl@0
   701
 * @param	"const TText8* aFile"
sl@0
   702
 *			Source code file name
sl@0
   703
 *
sl@0
   704
 * @param	"TInt aLine"
sl@0
   705
 *			Source code line
sl@0
   706
 *
sl@0
   707
 * @xxxx
sl@0
   708
 *
sl@0
   709
 */
sl@0
   710
EXPORT_C void RTestStep::TestBooleanTrueWithErrorCodeL(TBool aCondition, TInt aErrorCode, const TText8* aFile, TInt aLine)
sl@0
   711
	{
sl@0
   712
	// check condition
sl@0
   713
	if (aCondition)
sl@0
   714
		return;
sl@0
   715
sl@0
   716
	// this is only relevant if the current result is pass
sl@0
   717
	if (iTestStepResult == EPass)
sl@0
   718
		iTestStepResult = EFail;
sl@0
   719
sl@0
   720
	// convert filename for log
sl@0
   721
	TBuf<KMaxLogFilenameLength> fileName;
sl@0
   722
	TPtrC8 fileName8(aFile);
sl@0
   723
	fileName.Copy(fileName8);  // TText8->TBuf16
sl@0
   724
sl@0
   725
	// get the error text
sl@0
   726
	TPtrC errortxt = CLog::EpocErrorToText(aErrorCode);
sl@0
   727
sl@0
   728
	// display a log message
sl@0
   729
	ERR_PRINTF4(_L("Test Failed with error:%S in file:%S line:%d"),
sl@0
   730
			&errortxt, &fileName, aLine);
sl@0
   731
sl@0
   732
	// leave with error code
sl@0
   733
	User::Leave(aErrorCode);
sl@0
   734
	
sl@0
   735
	}
sl@0
   736
sl@0
   737
/**
sl@0
   738
 *
sl@0
   739
 * Traceable Boolean condition tester.
sl@0
   740
 * If the condition is not true, record an error with the supplied 
sl@0
   741
 * error code.
sl@0
   742
 *
sl@0
   743
 * @param	"TBool aCondition"
sl@0
   744
 *			Condition to be checked
sl@0
   745
 *
sl@0
   746
 * @param	"TInt aErrorCode"
sl@0
   747
 *			Supplied error code
sl@0
   748
 *
sl@0
   749
 * @param	"const TText8* aFile"
sl@0
   750
 *			Source code file name
sl@0
   751
 *
sl@0
   752
 * @param	"TInt aLine"
sl@0
   753
 *			Source code line
sl@0
   754
 *
sl@0
   755
 * @xxxx
sl@0
   756
 *
sl@0
   757
 */
sl@0
   758
EXPORT_C void RTestStep::TestBooleanTrueWithErrorCode(TBool aCondition, TInt aErrorCode, const TText8* aFile, TInt aLine)
sl@0
   759
	{
sl@0
   760
	// check condition
sl@0
   761
	if (aCondition)
sl@0
   762
		return;
sl@0
   763
sl@0
   764
	// this is only relevant if the current result is pass
sl@0
   765
	if (iTestStepResult == EPass)
sl@0
   766
		iTestStepResult = EFail;
sl@0
   767
sl@0
   768
	// convert filename for log
sl@0
   769
	TBuf<KMaxLogFilenameLength> fileName;
sl@0
   770
	TPtrC8 fileName8(aFile);
sl@0
   771
	fileName.Copy(fileName8);  // TText8->TBuf16
sl@0
   772
sl@0
   773
	// get the error text
sl@0
   774
	TPtrC errortxt = CLog::EpocErrorToText(aErrorCode);
sl@0
   775
sl@0
   776
	// display a log message
sl@0
   777
	ERR_PRINTF4(_L("Test Failed with error:%S in file:%S line:%d"),
sl@0
   778
			&errortxt, &fileName, aLine);
sl@0
   779
	}
sl@0
   780
sl@0
   781
/**
sl@0
   782
 *
sl@0
   783
 * Traceable checkpoint tester.
sl@0
   784
 * If the value does not match expected, record an error with supplied
sl@0
   785
 * text string, and leave.
sl@0
   786
 *
sl@0
   787
 * @param	"TInt aVal"
sl@0
   788
 *			Value to be checked
sl@0
   789
 *
sl@0
   790
 * @param	"TInt aExpectedVal"
sl@0
   791
 *			Value expected
sl@0
   792
 *
sl@0
   793
 * @param	"const TDesC& aText"
sl@0
   794
 *			Supplied text string
sl@0
   795
 *
sl@0
   796
 * @param	"const TText8* aFile"
sl@0
   797
 *			Source code file name
sl@0
   798
 *
sl@0
   799
 * @param	"TInt aLine"
sl@0
   800
 *			Source code line
sl@0
   801
 *
sl@0
   802
 * @xxxx
sl@0
   803
 *
sl@0
   804
 */
sl@0
   805
EXPORT_C void RTestStep::TestCheckPointCompareL(TInt aVal, TInt aExpectedVal, 
sl@0
   806
												  const TDesC& aText, const TText8* aFile, TInt aLine)
sl@0
   807
	{
sl@0
   808
	if(aVal != aExpectedVal)
sl@0
   809
		{
sl@0
   810
		// this is only relevant if the current result is pass
sl@0
   811
		if (iTestStepResult == EPass)
sl@0
   812
			iTestStepResult = EFail;
sl@0
   813
sl@0
   814
		// convert filename for log
sl@0
   815
		TBuf<KMaxLogFilenameLength> fileName;
sl@0
   816
		TPtrC8 fileName8(aFile);
sl@0
   817
		fileName.Copy(fileName8);  // TText8->TBuf16
sl@0
   818
sl@0
   819
		ERR_PRINTF6(_L("FAILED test:  Val = %d Exp Val = %d %S in file:%S line:%d"), 
sl@0
   820
			aVal, aExpectedVal, &aText, &fileName, aLine);
sl@0
   821
sl@0
   822
		User::Leave(aVal);
sl@0
   823
		}
sl@0
   824
	}
sl@0
   825
sl@0
   826
/** 
sl@0
   827
 *
sl@0
   828
 * Accessors for stack / heap size
sl@0
   829
 *
sl@0
   830
 * NB - These can only be set from within the RTestStep derived constructor itself - there
sl@0
   831
 * are no setter accessors.
sl@0
   832
 *
sl@0
   833
 * @xxxx
sl@0
   834
 *
sl@0
   835
 */
sl@0
   836
EXPORT_C TInt RTestStep::StackSize() const
sl@0
   837
{
sl@0
   838
	return iStackSize;
sl@0
   839
}
sl@0
   840
sl@0
   841
EXPORT_C TInt RTestStep::HeapSize() const
sl@0
   842
{
sl@0
   843
	return iHeapSize;
sl@0
   844
}
sl@0
   845
sl@0
   846
sl@0
   847
/** 
sl@0
   848
 *
sl@0
   849
 * Virtual destructor for CTestStep
sl@0
   850
 * Provided for backward compatibility ONLY
sl@0
   851
 *
sl@0
   852
 * @xxxx
sl@0
   853
 *
sl@0
   854
 */
sl@0
   855
EXPORT_C CTestStep::~CTestStep()
sl@0
   856
{
sl@0
   857
}
sl@0
   858
sl@0
   859
//
sl@0
   860
// RAsyncTestStep
sl@0
   861
//
sl@0
   862
sl@0
   863
EXPORT_C RAsyncTestStep::RAsyncTestStep():
sl@0
   864
	iReason (KErrNone),
sl@0
   865
	iResult (EPass),
sl@0
   866
	iStartAO (NULL),
sl@0
   867
	iActiveSchedulerWait (NULL),
sl@0
   868
	iStarted (EFalse)
sl@0
   869
	{
sl@0
   870
	}
sl@0
   871
sl@0
   872
EXPORT_C TVerdict RAsyncTestStep::DoTestStepL()
sl@0
   873
	{
sl@0
   874
	// allow recalls to same test step (?), so re-initialise the basic variables. Can't delete
sl@0
   875
	// as if in that case, old heap would be dead
sl@0
   876
	iReason = KErrNone;
sl@0
   877
	iResult = EPass;
sl@0
   878
	iScheduler = NULL;
sl@0
   879
	iActiveSchedulerWait = NULL;
sl@0
   880
	
sl@0
   881
	iScheduler = new (ELeave) CAsyncTestActiveScheduler(*this);	
sl@0
   882
	CActiveScheduler::Install(iScheduler);
sl@0
   883
	iActiveSchedulerWait = new (ELeave) CActiveSchedulerWait();
sl@0
   884
	
sl@0
   885
	// arrange for DoCallBack() to be called as the first thing. Use low priority to reduce
sl@0
   886
	// the overhead of an extra AO on everything else
sl@0
   887
	TCallBack callback (CallBack, this);
sl@0
   888
	iStartAO = NULL;
sl@0
   889
	iStartAO = new (ELeave) CAsyncCallBack (callback, CActive::EPriorityIdle); 
sl@0
   890
	iStartAO->Call();
sl@0
   891
	
sl@0
   892
	iStarted = ETrue; // obviously do this before we start, as can't once it is
sl@0
   893
	iActiveSchedulerWait->Start();
sl@0
   894
	return CheckTestResult();
sl@0
   895
	}
sl@0
   896
sl@0
   897
EXPORT_C TVerdict RAsyncTestStep::DoTestStepPostambleL()
sl@0
   898
	{
sl@0
   899
	CloseTest();
sl@0
   900
	delete iStartAO; // no need to Cancel 
sl@0
   901
	iStartAO = NULL;
sl@0
   902
	delete iActiveSchedulerWait;
sl@0
   903
	iActiveSchedulerWait = NULL;
sl@0
   904
	delete iScheduler; 
sl@0
   905
	iScheduler = NULL;
sl@0
   906
	return EPass;
sl@0
   907
	}
sl@0
   908
	
sl@0
   909
EXPORT_C void RAsyncTestStep::StopTest()
sl@0
   910
	{
sl@0
   911
	StopTest(KErrNone);
sl@0
   912
	}
sl@0
   913
	
sl@0
   914
EXPORT_C void RAsyncTestStep::StopTest(TInt aReason)
sl@0
   915
	{
sl@0
   916
	TVerdict resultToUse = (aReason==KErrNone) ? EPass : EFail;
sl@0
   917
	StopTest (aReason, resultToUse);
sl@0
   918
	}
sl@0
   919
sl@0
   920
EXPORT_C void RAsyncTestStep::StopTest(TInt aReason, TVerdict aResult)
sl@0
   921
	{
sl@0
   922
	// note if stop is called multiple times, we record last
sl@0
   923
	// non-KErrNone reason and last non-Pass result, but only actually stop once
sl@0
   924
	if (aReason!=KErrNone)
sl@0
   925
		{
sl@0
   926
		iReason = aReason;		
sl@0
   927
		}
sl@0
   928
	SetResult(aResult);
sl@0
   929
	if (iStarted)
sl@0
   930
		{
sl@0
   931
		iStarted = EFalse;
sl@0
   932
		iActiveSchedulerWait->AsyncStop();
sl@0
   933
		}
sl@0
   934
	}
sl@0
   935
	
sl@0
   936
EXPORT_C TInt RAsyncTestStep::Reason() const
sl@0
   937
	{
sl@0
   938
	return iReason;
sl@0
   939
	}
sl@0
   940
	
sl@0
   941
EXPORT_C TVerdict RAsyncTestStep::Result() const
sl@0
   942
	{
sl@0
   943
	return iResult;
sl@0
   944
	}
sl@0
   945
	
sl@0
   946
void RAsyncTestStep::SetResult(TVerdict aResult)
sl@0
   947
	{
sl@0
   948
	// remember the last non-Pass result
sl@0
   949
	if (aResult!=EPass)
sl@0
   950
		{
sl@0
   951
		iResult = aResult;
sl@0
   952
		}	
sl@0
   953
	}
sl@0
   954
	
sl@0
   955
EXPORT_C TVerdict RAsyncTestStep::CheckTestResult()
sl@0
   956
	{
sl@0
   957
	TVerdict result = Result();
sl@0
   958
	
sl@0
   959
	if (result!=EPass) // if the result is a Pass, even if error is too we treat as test pass
sl@0
   960
		{
sl@0
   961
		INFO_PRINTF3(_L("Failed test with error %d, result %s"), Reason(), KVerdictString[result]);
sl@0
   962
		}
sl@0
   963
	return result;
sl@0
   964
	}
sl@0
   965
sl@0
   966
TInt RAsyncTestStep::CallBack(TAny* aPtr)
sl@0
   967
	{
sl@0
   968
	RAsyncTestStep* self = static_cast<RAsyncTestStep*> (aPtr);
sl@0
   969
	self->DoCallBack(); // if this fails, it will stop itself
sl@0
   970
	return KErrNone;
sl@0
   971
	}
sl@0
   972
	
sl@0
   973
void RAsyncTestStep::DoCallBack()
sl@0
   974
	{
sl@0
   975
	TRAPD(error, KickoffTestL()); 
sl@0
   976
	if (error!=KErrNone)
sl@0
   977
		{
sl@0
   978
		StopTest(error); // if kickoff fails, we stop here
sl@0
   979
		}	
sl@0
   980
	}
sl@0
   981
sl@0
   982
void RAsyncTestStep::HandleError(TInt aError)
sl@0
   983
	{
sl@0
   984
	INFO_PRINTF2(_L("ActiveScheduler::Error(%d)"), aError);
sl@0
   985
	StopTest(aError);
sl@0
   986
	}
sl@0
   987
	
sl@0
   988
//
sl@0
   989
// CBusyTestUnit
sl@0
   990
//
sl@0
   991
sl@0
   992
CBusyTestUnit::CBusyTestUnit():
sl@0
   993
		iPercentBusy(0), iThreadPriority(EPriorityNormal)
sl@0
   994
	{
sl@0
   995
	// non exported default constructor to enforce non-derivation
sl@0
   996
	}
sl@0
   997
sl@0
   998
CBusyTestUnit::CBusyTestUnit(TInt aPercentBusy, TThreadPriority aThreadPriority):
sl@0
   999
		iPercentBusy(aPercentBusy), iThreadPriority(aThreadPriority)
sl@0
  1000
	{
sl@0
  1001
	ASSERT(aPercentBusy>=0 && aPercentBusy<=100); // assume this
sl@0
  1002
	}
sl@0
  1003
	
sl@0
  1004
void CBusyTestUnit::ConstructL()
sl@0
  1005
	{
sl@0
  1006
	}
sl@0
  1007
	
sl@0
  1008
EXPORT_C CBusyTestUnit* CBusyTestUnit::NewLC(TInt aPercentBusy, TThreadPriority aThreadPriority)
sl@0
  1009
	{
sl@0
  1010
	CBusyTestUnit* self = new (ELeave) CBusyTestUnit(aPercentBusy, aThreadPriority);
sl@0
  1011
	CleanupStack::PushL(self);
sl@0
  1012
	self->ConstructL();
sl@0
  1013
	return self;
sl@0
  1014
	}
sl@0
  1015
	
sl@0
  1016
EXPORT_C CBusyTestUnit* CBusyTestUnit::NewL(TInt aPercentBusy, TThreadPriority aThreadPriority)
sl@0
  1017
	{
sl@0
  1018
	CBusyTestUnit* self = NewLC(aPercentBusy, aThreadPriority);
sl@0
  1019
	CleanupStack::Pop(self);
sl@0
  1020
	return self;
sl@0
  1021
	}
sl@0
  1022
	
sl@0
  1023
CBusyTestUnit::~CBusyTestUnit()
sl@0
  1024
	{
sl@0
  1025
	Stop();
sl@0
  1026
	}
sl@0
  1027
	
sl@0
  1028
EXPORT_C void CBusyTestUnit::Stop()
sl@0
  1029
	{
sl@0
  1030
	if (iTimer)
sl@0
  1031
		{
sl@0
  1032
		iTimer->Cancel();
sl@0
  1033
		delete iTimer;
sl@0
  1034
		iTimer=NULL;
sl@0
  1035
		}
sl@0
  1036
	if (iChildThread.Handle())
sl@0
  1037
		{
sl@0
  1038
		// child thread created, so kill
sl@0
  1039
		iChildThread.Kill(0);
sl@0
  1040
		iChildThread.Close();
sl@0
  1041
		}
sl@0
  1042
	}
sl@0
  1043
	
sl@0
  1044
EXPORT_C TInt CBusyTestUnit::Start()
sl@0
  1045
	{
sl@0
  1046
	return Start(0);	
sl@0
  1047
	}
sl@0
  1048
	
sl@0
  1049
EXPORT_C TInt CBusyTestUnit::Start(TTimeIntervalMicroSeconds aRunFor)
sl@0
  1050
	{
sl@0
  1051
	return Start(0, aRunFor);
sl@0
  1052
	}
sl@0
  1053
	
sl@0
  1054
EXPORT_C TInt CBusyTestUnit::Start(TTimeIntervalMicroSeconds32 aDelayFor, TTimeIntervalMicroSeconds aRunFor)
sl@0
  1055
	{
sl@0
  1056
	iRunFor = aRunFor;
sl@0
  1057
	if (!aDelayFor.Int())
sl@0
  1058
		{
sl@0
  1059
		// run immediately
sl@0
  1060
		return RunThread();
sl@0
  1061
		}
sl@0
  1062
	else
sl@0
  1063
		{
sl@0
  1064
		iTimer = CPeriodic::NewL(CActive::EPriorityHigh);
sl@0
  1065
		TCallBack callback(StaticTimerCallback, this);
sl@0
  1066
		iTimer->Start(aDelayFor, 0, callback);
sl@0
  1067
		return KErrNone;
sl@0
  1068
		}
sl@0
  1069
	}
sl@0
  1070
	
sl@0
  1071
TInt CBusyTestUnit::StaticTimerCallback(TAny* aPtr)
sl@0
  1072
	{
sl@0
  1073
	CBusyTestUnit* self = static_cast<CBusyTestUnit*>(aPtr);
sl@0
  1074
	return self->TimerCallback();
sl@0
  1075
	}
sl@0
  1076
	
sl@0
  1077
TInt CBusyTestUnit::TimerCallback()
sl@0
  1078
	{
sl@0
  1079
	// called on timer callback, so assume iTimer!
sl@0
  1080
	ASSERT(iTimer);
sl@0
  1081
	// first stop and delete timer - don't need again this run
sl@0
  1082
	iTimer->Cancel();
sl@0
  1083
	// then kick off the thread
sl@0
  1084
	TInt error = RunThread(); 
sl@0
  1085
	// now delete the timer - do now as we've been called back by it!
sl@0
  1086
	delete iTimer;
sl@0
  1087
	iTimer = NULL;
sl@0
  1088
	return error; // any error will stop the test, in theory
sl@0
  1089
	}
sl@0
  1090
	
sl@0
  1091
TInt CBusyTestUnit::RunThread()
sl@0
  1092
	{
sl@0
  1093
	TAny* paramPtr = this;
sl@0
  1094
	TInt error = iChildThread.Create(KNullDesC, StartThread, 
sl@0
  1095
					KDefaultStackSize, NULL, paramPtr, EOwnerThread);
sl@0
  1096
	if (!error)
sl@0
  1097
		{
sl@0
  1098
		iChildThread.SetPriority(iThreadPriority);
sl@0
  1099
		iChildThread.Resume();
sl@0
  1100
		}
sl@0
  1101
	return error;
sl@0
  1102
	}
sl@0
  1103
	
sl@0
  1104
TInt CBusyTestUnit::StartThread(TAny* aPtr)
sl@0
  1105
	{
sl@0
  1106
	CBusyTestUnit* self = static_cast<CBusyTestUnit*>(aPtr);
sl@0
  1107
	if (self)
sl@0
  1108
		{
sl@0
  1109
		self->ThreadFunction();
sl@0
  1110
		}
sl@0
  1111
	return KErrNone;
sl@0
  1112
	}
sl@0
  1113
	
sl@0
  1114
void CBusyTestUnit::ThreadFunction()
sl@0
  1115
	{
sl@0
  1116
	// this runs in a separate thread and tries to use lots of CPU time up to percentage
sl@0
  1117
	// nominally we run busy for loops for iPercentBusy/100ms, and then wait for rest of the
sl@0
  1118
	// 100ms using User::After(). We keep doing this until we reach the target time, if there is
sl@0
  1119
	// one
sl@0
  1120
	
sl@0
  1121
	const TInt KDefaultLoop = 10000;
sl@0
  1122
	TTime timeAtStart; 
sl@0
  1123
	timeAtStart.UniversalTime(); // time of start
sl@0
  1124
	
sl@0
  1125
	const TInt KLoopInterval =100000; // 100ms - loop time
sl@0
  1126
	TTimeIntervalMicroSeconds32 busyInterval(KLoopInterval*iPercentBusy/100); // how much of loop time to be busy
sl@0
  1127
	TTimeIntervalMicroSeconds32 quietInterval(KLoopInterval-busyInterval.Int()); // how much of loop time to be quiet
sl@0
  1128
	
sl@0
  1129
	while (ETrue)
sl@0
  1130
		{
sl@0
  1131
		// the interval, for loops for the busy bit and then a User::After()
sl@0
  1132
		TTime startOfInterval;
sl@0
  1133
		startOfInterval.UniversalTime();
sl@0
  1134
		while (ETrue)
sl@0
  1135
			{
sl@0
  1136
			// the busy bit - iBusyVariable is volatile so should never be optimised out
sl@0
  1137
			for (TInt i=0; i<KDefaultLoop; i++)
sl@0
  1138
				{
sl@0
  1139
				iBusyVariable = i;
sl@0
  1140
				}
sl@0
  1141
			TTime now;
sl@0
  1142
			now.UniversalTime();
sl@0
  1143
			if (startOfInterval + busyInterval < now)
sl@0
  1144
				{
sl@0
  1145
				// we're passed the time
sl@0
  1146
				break;
sl@0
  1147
				}
sl@0
  1148
			}
sl@0
  1149
		User::After(quietInterval);		
sl@0
  1150
			
sl@0
  1151
		if (iRunFor.Int64())
sl@0
  1152
			{
sl@0
  1153
			// check to see if we are passed the interval given at Start()
sl@0
  1154
			TTime now;
sl@0
  1155
			now.UniversalTime();
sl@0
  1156
			if (timeAtStart + iRunFor < now)
sl@0
  1157
				{
sl@0
  1158
				// we're passed the time
sl@0
  1159
				break;
sl@0
  1160
				}
sl@0
  1161
			}
sl@0
  1162
		}
sl@0
  1163
	}
sl@0
  1164