First public contribution.
1 // Copyright (c) 2002-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
19 // Test system includes
20 #include <testframework.h>
21 #include "TestIniData.h"
22 #include "asyncscheduler.h"
24 // do not export if Unit Testing
25 #if defined (__TSU_TESTFRAMEWORK__)
30 static const TUint16*const KVerdictString[] = // must match up with TVerdict
35 _S("ETestSuiteError"),
42 * Test step constructor.
47 EXPORT_C RTestStep::RTestStep()
49 iTestStepResult = EPass;
53 iConfigDataAvailable = EFalse;
54 iStackSize = KTestStackSize;
55 iHeapSize = KMaxTestThreadHeapSize;
57 // NB : RTestStep has no destructor
62 * CBase compatibility functionality.
69 EXPORT_C TAny* RTestStep::operator new(TUint aSize, TAny* aBase) __NO_THROW
71 Mem::FillZ(aBase,aSize);
76 EXPORT_C TAny* RTestStep::operator new(TUint aSize, TLeave)
78 return newL(aSize); // will leave on alloc failure
81 EXPORT_C TAny* RTestStep::operator new(TUint aSize) __NO_THROW
83 TAny* pM=User::Alloc(aSize);
89 EXPORT_C TAny* RTestStep::newL(TUint aSize)
91 TAny* pM=User::AllocL(aSize);
96 EXPORT_C TAny* RTestStep::operator new(TUint aSize,TUint anExtraSize) __NO_THROW
99 TAny *pM=User::Alloc(aSize);
101 Mem::FillZ(pM,aSize);
107 * Pre-preamble for all test steps. This grows the cleanup stack to
108 * allow for allocation errors.
113 EXPORT_C void RTestStep::PreOpenL()
115 TAny* dummyPtr = NULL;
116 for(TInt i = 0 ; i < 20 ; i++)
117 CleanupStack::PushL(dummyPtr); // Grow the cleanup stack.
118 CleanupStack::PopAndDestroy(20);
123 * Default empty implementation of OpenL (preamble).
124 * Test steps can override this to implement required code.
127 * Result of preamble (should only be EPass or EInconclusive)
132 EXPORT_C TVerdict RTestStep::OpenL()
134 // for backward compatibility with CTestStep
135 return DoTestStepPreambleL();
140 * Default empty implementation of CleanupAfterOpenFail (preamble cleanup).
141 * Test steps can override this to implement required code.
146 EXPORT_C void RTestStep::CleanupAfterOpenFail()
148 // default empty implementation
149 // a step should implement its own method if required
154 * Default empty implementation of Close (postamble)
155 * Test steps can override this to implement required code.
157 * NB this does NOT leave - any leaves should be trapped and panicked.
162 EXPORT_C void RTestStep::Close()
164 // for backward compatibility with CTestStep
165 _LIT(KPanicStr, "RTestStep::Close");
166 TVerdict ver = EPass;
167 TRAPD(err, ver = DoTestStepPostambleL());
169 User::Panic(KPanicStr, 0);
170 // any DoTestStepPostambleL() which returns EInconclusive should be panicked
172 User::Panic(KPanicStr, 1);
180 * @param "CTestSuite*"
186 EXPORT_C void RTestStep::SetSuite(CTestSuite* aSuite)
193 * Set the test result
201 EXPORT_C void RTestStep::SetResult(TVerdict aResult)
203 iTestStepResult = aResult;
216 EXPORT_C TPtrC RTestStep::StepName() const
218 return iTestStepName;
223 * General logging function for test steps.
225 * @param "TRefByValue<const TDesC16> aFmt"
229 * Variable print parameters
234 EXPORT_C void RTestStep::Log(TRefByValue<const TDesC16> aFmt, ...)
238 VA_START(aList, aFmt);
240 TIntegrationTestLog16Overflow iOverflow16;
242 // decode formatted data for display on console
243 TBuf<KMaxLogLineLength> lineBuf;
244 lineBuf.AppendFormatList(aFmt, aList, &iOverflow16);
246 // send the data to the log system via the suite
247 iSuite->Log(_L("%S"),&lineBuf);
255 * General logging function for test steps, with severity.
257 * @param "TInt aSeverity"
258 * Severity level required to log
260 * @param "TRefByValue<const TDesC16> aFmt"
264 * Variable print parameters
269 EXPORT_C void RTestStep::Log( TInt aSeverity, TRefByValue<const TDesC16> aFmt, ... )
272 VA_START(aList, aFmt);
274 TIntegrationTestLog16Overflow iOverflow16;
276 // decode formatted data for display on console
277 TBuf<KMaxLogLineLength> lineBuf;
278 lineBuf.AppendFormatList(aFmt, aList, &iOverflow16);
280 // send the data to the log system via the suite
281 if(LogSeverity::IsActive(aSeverity, iSuite->Severity()))
282 iSuite->Log(aSeverity, _L("%S"),&lineBuf);
289 * Traceable logging function for test steps.
291 * @param "const TText8* aFile"
292 * Source code file name
294 * @param "TInt aLine"
297 * @param "TInt aSeverity"
298 * Severity level required to log
300 * @param "TRefByValue<const TDesC16> aFmt"
301 * Printf-style format.
304 * Variable print parameters
309 EXPORT_C void RTestStep::LogExtra(const TText8* aFile, TInt aLine, TInt aSeverity,
310 TRefByValue<const TDesC16> aFmt,...)
313 VA_START(aList, aFmt);
315 TIntegrationTestLog16Overflow iOverflow16;
317 // decode formatted data for display on console
318 TBuf<KMaxLogLineLength> lineBuf;
319 lineBuf.AppendFormatList(aFmt, aList, &iOverflow16);
321 // send the data to the log system via the suite
322 if(LogSeverity::IsActive(aSeverity, iSuite->Severity()))
323 iSuite->LogExtra(aFile, aLine, aSeverity, lineBuf);
330 Test steps can use this when looking up configs, to provide a level of script control
332 void RTestStep::SetDefaultParamSet(const TDesC& aParamSet)
334 iDefaultParamSet.Set(aParamSet);
339 * Load a configuration file.
340 * If successful, data member iConfigDataAvailable is set.
342 * @param "TPtrC aConfig"
343 * The configuration file name.
348 EXPORT_C void RTestStep::LoadConfig(const TDesC& aConfig)
351 // if a config file supplied then use
352 if (aConfig.Length() != 0)
355 // get the full pathname default drive name and extension
356 _LIT(KRelated,"C:\\config.ini");
357 TParse configFileName;
358 TInt returnCode = configFileName.Set(aConfig, &KRelated, NULL);
360 if (returnCode != KErrNone)
362 // error opening FileManager
363 ERR_PRINTF2(_L("Error opening config file %S"), &(configFileName.FullName()));
366 // create and load the CTestIniData object
367 TRAPD(r, iConfigData = CTestIniData::NewL(configFileName.FullName()));
369 // check if loaded ok
373 iConfigDataAvailable = ETrue;
378 iConfigDataAvailable = EFalse;
382 TPtrC errortxt = CLog::EpocErrorToText(r);
383 ERR_PRINTF2(_L("Failed to load config data file - error %S"), &errortxt);
390 * Unload any existing configuration data.
395 EXPORT_C void RTestStep::UnloadConfig()
397 iConfigDataAvailable = EFalse;
399 // clean up Config data object
407 * Check the config file for a boolean value
409 * @param "const TDesC &aSectName"
410 * Section name to check in
412 * @param "const TDesC &aKeyName"
413 * Key name to check for
415 * @param "Bool &aResult"
416 * TBool returned from config file
419 * Result (ETrue if found)
424 EXPORT_C TBool RTestStep::GetBoolFromConfig(const TDesC &aSectName, const TDesC &aKeyName, TBool &aResult)
426 // check file available
427 if (!iConfigDataAvailable)
429 ERR_PRINTF1(_L("No config file available"));
437 ret = iConfigData->FindVar(aSectName, aKeyName, result);
439 // if failed to decode display error
442 // display error message
443 ERR_PRINTF3(_L("Failed to read section:%S key:%S "),
444 &aSectName, &aKeyName );
450 // set result as a TBool
451 if (result.FindF(_L("true")) == KErrNotFound)
461 * Check the config file for a TInt value
463 * @param "const TDesC &aSectName"
464 * Section name to check in
466 * @param "const TDesC &aKeyName"
467 * Key name to check for
469 * @param "TInt &aResult"
470 * TInt returned from config file
473 * Result (ETrue if found)
478 EXPORT_C TBool RTestStep::GetIntFromConfig(const TDesC &aSectName, const TDesC &aKeyName, TInt &aResult)
480 // check file available
481 if ( !iConfigDataAvailable )
483 ERR_PRINTF1(_L("No config file available"));
491 ret = iConfigData->FindVar(aSectName, aKeyName, result);
493 // if failed to decode display error
496 // display error message
497 ERR_PRINTF3(_L("Failed to read section:%S key:%S "),
498 &aSectName, &aKeyName );
504 // use TLex to convert to a TInt
506 if (lex.Val(aResult) == KErrNone)
514 * Check the config file for a string TPtr value
516 * @param "const TDesC &aSectName"
517 * Section name to check in
519 * @param "const TDesC &aKeyName"
520 * Key name to check for
522 * @param "TPtrC &aResult"
523 * String returned from config file
526 * Result (ETrue if found)
531 EXPORT_C TBool RTestStep::GetStringFromConfig(const TDesC &aSectName, const TDesC &aKeyName, TPtrC &aResult)
533 // check file available
534 if (!iConfigDataAvailable)
536 ERR_PRINTF1(_L("No config file available"));
541 TBool ret = iConfigData->FindVar(aSectName, aKeyName, aResult);
543 // if failed to decode display error
546 ERR_PRINTF3(_L("Failed to read section:%S key:%S "),
547 &aSectName, &aKeyName );
554 * Reads the value present from the test steps ini file within the mentioned section name and key name
555 * Copies the value to the TInt reference passed in
556 * @param aSectName - Section within the test steps ini file
557 * @param aKeyName - Name of a key within a section
558 * @return aResult - The integer value of the Hex input
559 * @return TBool - ETrue for found, EFalse for not found
561 EXPORT_C TBool RTestStep::GetHexFromConfig(const TDesC &aSectName,const TDesC &aKeyName,TInt &aResult)
566 if(!iConfigData->FindVar(aSectName, aKeyName, result))
569 TInt err = lex.Val((TUint &)aResult, EHex);
578 * Default empty implementation of DoTestStepPreambleL.
579 * Test steps can override this to implement required code.
582 * Result of preamble (should only be EPass or EInconclusive)
587 // for backward compatibility with CTestStep
588 EXPORT_C TVerdict RTestStep::DoTestStepPreambleL()
595 * Default empty implementation of DoTestStepPostambleL.
596 * Test steps can override this to implement required code.
599 * Result of postamble (should only be EPass or EInconclusive)
604 // for backward compatibility with CTestStep
605 EXPORT_C TVerdict RTestStep::DoTestStepPostambleL()
612 * Traceable Boolean condition tester.
613 * If the condition is not true, record an error.
615 * @param "TBool aCondition"
616 * Condition to be checked
618 * @param "const TText8* aFile"
619 * Source code file name
621 * @param "TInt aLine"
627 EXPORT_C void RTestStep::TestBooleanTrue(TBool aCondition, const TText8* aFile, TInt aLine)
634 // this is only relevant if the current result is pass
635 if (iTestStepResult == EPass)
636 iTestStepResult = EFail;
638 // convert filename for log
639 TBuf<KMaxLogFilenameLength> fileName;
640 TPtrC8 fileName8(aFile);
641 fileName.Copy(fileName8); // TText8->TBuf16
643 // display a log message
644 ERR_PRINTF3(_L("Test Failed in file:%S line:%d"), &fileName, aLine);
650 * Traceable Boolean condition tester.
651 * If the condition is not true, record an error and leave.
653 * @param "TBool aCondition"
654 * Condition to be checked
656 * @param "const TText8* aFile"
657 * Source code file name
659 * @param "TInt aLine"
665 EXPORT_C void RTestStep::TestBooleanTrueL(TBool aCondition, const TText8* aFile, TInt aLine)
672 // this is only relevant if the current result is pass
673 if (iTestStepResult == EPass)
674 iTestStepResult = EFail;
676 // convert filename for log
677 TBuf<KMaxLogFilenameLength> fileName;
678 TPtrC8 fileName8(aFile);
679 fileName.Copy(fileName8); // TText8->TBuf16
681 // display a log message
682 ERR_PRINTF3(_L("Test Failed in file:%S line:%d"), &fileName, aLine);
684 // leave with error code
685 User::Leave(KTestErrorCode);
691 * Traceable Boolean condition tester.
692 * If the condition is not true, record an error with the supplied
693 * error code, and leave.
695 * @param "TBool aCondition"
696 * Condition to be checked
698 * @param "TInt aErrorCode"
699 * Supplied error code
701 * @param "const TText8* aFile"
702 * Source code file name
704 * @param "TInt aLine"
710 EXPORT_C void RTestStep::TestBooleanTrueWithErrorCodeL(TBool aCondition, TInt aErrorCode, const TText8* aFile, TInt aLine)
716 // this is only relevant if the current result is pass
717 if (iTestStepResult == EPass)
718 iTestStepResult = EFail;
720 // convert filename for log
721 TBuf<KMaxLogFilenameLength> fileName;
722 TPtrC8 fileName8(aFile);
723 fileName.Copy(fileName8); // TText8->TBuf16
725 // get the error text
726 TPtrC errortxt = CLog::EpocErrorToText(aErrorCode);
728 // display a log message
729 ERR_PRINTF4(_L("Test Failed with error:%S in file:%S line:%d"),
730 &errortxt, &fileName, aLine);
732 // leave with error code
733 User::Leave(aErrorCode);
739 * Traceable Boolean condition tester.
740 * If the condition is not true, record an error with the supplied
743 * @param "TBool aCondition"
744 * Condition to be checked
746 * @param "TInt aErrorCode"
747 * Supplied error code
749 * @param "const TText8* aFile"
750 * Source code file name
752 * @param "TInt aLine"
758 EXPORT_C void RTestStep::TestBooleanTrueWithErrorCode(TBool aCondition, TInt aErrorCode, const TText8* aFile, TInt aLine)
764 // this is only relevant if the current result is pass
765 if (iTestStepResult == EPass)
766 iTestStepResult = EFail;
768 // convert filename for log
769 TBuf<KMaxLogFilenameLength> fileName;
770 TPtrC8 fileName8(aFile);
771 fileName.Copy(fileName8); // TText8->TBuf16
773 // get the error text
774 TPtrC errortxt = CLog::EpocErrorToText(aErrorCode);
776 // display a log message
777 ERR_PRINTF4(_L("Test Failed with error:%S in file:%S line:%d"),
778 &errortxt, &fileName, aLine);
783 * Traceable checkpoint tester.
784 * If the value does not match expected, record an error with supplied
785 * text string, and leave.
788 * Value to be checked
790 * @param "TInt aExpectedVal"
793 * @param "const TDesC& aText"
794 * Supplied text string
796 * @param "const TText8* aFile"
797 * Source code file name
799 * @param "TInt aLine"
805 EXPORT_C void RTestStep::TestCheckPointCompareL(TInt aVal, TInt aExpectedVal,
806 const TDesC& aText, const TText8* aFile, TInt aLine)
808 if(aVal != aExpectedVal)
810 // this is only relevant if the current result is pass
811 if (iTestStepResult == EPass)
812 iTestStepResult = EFail;
814 // convert filename for log
815 TBuf<KMaxLogFilenameLength> fileName;
816 TPtrC8 fileName8(aFile);
817 fileName.Copy(fileName8); // TText8->TBuf16
819 ERR_PRINTF6(_L("FAILED test: Val = %d Exp Val = %d %S in file:%S line:%d"),
820 aVal, aExpectedVal, &aText, &fileName, aLine);
828 * Accessors for stack / heap size
830 * NB - These can only be set from within the RTestStep derived constructor itself - there
831 * are no setter accessors.
836 EXPORT_C TInt RTestStep::StackSize() const
841 EXPORT_C TInt RTestStep::HeapSize() const
849 * Virtual destructor for CTestStep
850 * Provided for backward compatibility ONLY
855 EXPORT_C CTestStep::~CTestStep()
863 EXPORT_C RAsyncTestStep::RAsyncTestStep():
867 iActiveSchedulerWait (NULL),
872 EXPORT_C TVerdict RAsyncTestStep::DoTestStepL()
874 // allow recalls to same test step (?), so re-initialise the basic variables. Can't delete
875 // as if in that case, old heap would be dead
879 iActiveSchedulerWait = NULL;
881 iScheduler = new (ELeave) CAsyncTestActiveScheduler(*this);
882 CActiveScheduler::Install(iScheduler);
883 iActiveSchedulerWait = new (ELeave) CActiveSchedulerWait();
885 // arrange for DoCallBack() to be called as the first thing. Use low priority to reduce
886 // the overhead of an extra AO on everything else
887 TCallBack callback (CallBack, this);
889 iStartAO = new (ELeave) CAsyncCallBack (callback, CActive::EPriorityIdle);
892 iStarted = ETrue; // obviously do this before we start, as can't once it is
893 iActiveSchedulerWait->Start();
894 return CheckTestResult();
897 EXPORT_C TVerdict RAsyncTestStep::DoTestStepPostambleL()
900 delete iStartAO; // no need to Cancel
902 delete iActiveSchedulerWait;
903 iActiveSchedulerWait = NULL;
909 EXPORT_C void RAsyncTestStep::StopTest()
914 EXPORT_C void RAsyncTestStep::StopTest(TInt aReason)
916 TVerdict resultToUse = (aReason==KErrNone) ? EPass : EFail;
917 StopTest (aReason, resultToUse);
920 EXPORT_C void RAsyncTestStep::StopTest(TInt aReason, TVerdict aResult)
922 // note if stop is called multiple times, we record last
923 // non-KErrNone reason and last non-Pass result, but only actually stop once
924 if (aReason!=KErrNone)
932 iActiveSchedulerWait->AsyncStop();
936 EXPORT_C TInt RAsyncTestStep::Reason() const
941 EXPORT_C TVerdict RAsyncTestStep::Result() const
946 void RAsyncTestStep::SetResult(TVerdict aResult)
948 // remember the last non-Pass result
955 EXPORT_C TVerdict RAsyncTestStep::CheckTestResult()
957 TVerdict result = Result();
959 if (result!=EPass) // if the result is a Pass, even if error is too we treat as test pass
961 INFO_PRINTF3(_L("Failed test with error %d, result %s"), Reason(), KVerdictString[result]);
966 TInt RAsyncTestStep::CallBack(TAny* aPtr)
968 RAsyncTestStep* self = static_cast<RAsyncTestStep*> (aPtr);
969 self->DoCallBack(); // if this fails, it will stop itself
973 void RAsyncTestStep::DoCallBack()
975 TRAPD(error, KickoffTestL());
978 StopTest(error); // if kickoff fails, we stop here
982 void RAsyncTestStep::HandleError(TInt aError)
984 INFO_PRINTF2(_L("ActiveScheduler::Error(%d)"), aError);
992 CBusyTestUnit::CBusyTestUnit():
993 iPercentBusy(0), iThreadPriority(EPriorityNormal)
995 // non exported default constructor to enforce non-derivation
998 CBusyTestUnit::CBusyTestUnit(TInt aPercentBusy, TThreadPriority aThreadPriority):
999 iPercentBusy(aPercentBusy), iThreadPriority(aThreadPriority)
1001 ASSERT(aPercentBusy>=0 && aPercentBusy<=100); // assume this
1004 void CBusyTestUnit::ConstructL()
1008 EXPORT_C CBusyTestUnit* CBusyTestUnit::NewLC(TInt aPercentBusy, TThreadPriority aThreadPriority)
1010 CBusyTestUnit* self = new (ELeave) CBusyTestUnit(aPercentBusy, aThreadPriority);
1011 CleanupStack::PushL(self);
1016 EXPORT_C CBusyTestUnit* CBusyTestUnit::NewL(TInt aPercentBusy, TThreadPriority aThreadPriority)
1018 CBusyTestUnit* self = NewLC(aPercentBusy, aThreadPriority);
1019 CleanupStack::Pop(self);
1023 CBusyTestUnit::~CBusyTestUnit()
1028 EXPORT_C void CBusyTestUnit::Stop()
1036 if (iChildThread.Handle())
1038 // child thread created, so kill
1039 iChildThread.Kill(0);
1040 iChildThread.Close();
1044 EXPORT_C TInt CBusyTestUnit::Start()
1049 EXPORT_C TInt CBusyTestUnit::Start(TTimeIntervalMicroSeconds aRunFor)
1051 return Start(0, aRunFor);
1054 EXPORT_C TInt CBusyTestUnit::Start(TTimeIntervalMicroSeconds32 aDelayFor, TTimeIntervalMicroSeconds aRunFor)
1057 if (!aDelayFor.Int())
1064 iTimer = CPeriodic::NewL(CActive::EPriorityHigh);
1065 TCallBack callback(StaticTimerCallback, this);
1066 iTimer->Start(aDelayFor, 0, callback);
1071 TInt CBusyTestUnit::StaticTimerCallback(TAny* aPtr)
1073 CBusyTestUnit* self = static_cast<CBusyTestUnit*>(aPtr);
1074 return self->TimerCallback();
1077 TInt CBusyTestUnit::TimerCallback()
1079 // called on timer callback, so assume iTimer!
1081 // first stop and delete timer - don't need again this run
1083 // then kick off the thread
1084 TInt error = RunThread();
1085 // now delete the timer - do now as we've been called back by it!
1088 return error; // any error will stop the test, in theory
1091 TInt CBusyTestUnit::RunThread()
1093 TAny* paramPtr = this;
1094 TInt error = iChildThread.Create(KNullDesC, StartThread,
1095 KDefaultStackSize, NULL, paramPtr, EOwnerThread);
1098 iChildThread.SetPriority(iThreadPriority);
1099 iChildThread.Resume();
1104 TInt CBusyTestUnit::StartThread(TAny* aPtr)
1106 CBusyTestUnit* self = static_cast<CBusyTestUnit*>(aPtr);
1109 self->ThreadFunction();
1114 void CBusyTestUnit::ThreadFunction()
1116 // this runs in a separate thread and tries to use lots of CPU time up to percentage
1117 // nominally we run busy for loops for iPercentBusy/100ms, and then wait for rest of the
1118 // 100ms using User::After(). We keep doing this until we reach the target time, if there is
1121 const TInt KDefaultLoop = 10000;
1123 timeAtStart.UniversalTime(); // time of start
1125 const TInt KLoopInterval =100000; // 100ms - loop time
1126 TTimeIntervalMicroSeconds32 busyInterval(KLoopInterval*iPercentBusy/100); // how much of loop time to be busy
1127 TTimeIntervalMicroSeconds32 quietInterval(KLoopInterval-busyInterval.Int()); // how much of loop time to be quiet
1131 // the interval, for loops for the busy bit and then a User::After()
1132 TTime startOfInterval;
1133 startOfInterval.UniversalTime();
1136 // the busy bit - iBusyVariable is volatile so should never be optimised out
1137 for (TInt i=0; i<KDefaultLoop; i++)
1142 now.UniversalTime();
1143 if (startOfInterval + busyInterval < now)
1145 // we're passed the time
1149 User::After(quietInterval);
1151 if (iRunFor.Int64())
1153 // check to see if we are passed the interval given at Start()
1155 now.UniversalTime();
1156 if (timeAtStart + iRunFor < now)
1158 // we're passed the time