sl@0: /* sl@0: * Copyright (c) 1998-2009 Nokia Corporation and/or its subsidiary(-ies). sl@0: * All rights reserved. sl@0: * This component and the accompanying materials are made available sl@0: * under the terms of the License "Eclipse Public License v1.0" sl@0: * which accompanies this distribution, and is available sl@0: * at the URL "http://www.eclipse.org/legal/epl-v10.html". sl@0: * sl@0: * Initial Contributors: sl@0: * Nokia Corporation - initial contribution. sl@0: * sl@0: * Contributors: sl@0: * sl@0: * Description: sl@0: * sl@0: */ sl@0: sl@0: sl@0: #include "t_testsetup.h" sl@0: #include "t_testactionspec.h" sl@0: #include "t_input.h" sl@0: #include "t_certstoreactionmemfail.h" sl@0: #include "tcancel.h" sl@0: #include "t_message.h" sl@0: #include "tScriptSetup.h" sl@0: #include "Thardcodedsetup.h" sl@0: #include "t_testhandler.h" sl@0: #include "t_output.h" sl@0: #include "tTestSpec.h" sl@0: #include "ttesthandlersettings.h" sl@0: #include "testexecuteinterface.h" sl@0: #include "t_logger.h" sl@0: #include "t_testrunner.h" sl@0: #include "t_dummyconsole.h" sl@0: sl@0: EXPORT_C HBufC* CTestSetup::GetArgument(TInt nPos) sl@0: { sl@0: // Get command line sl@0: HBufC *argv = HBufC::NewLC(User::CommandLineLength()); sl@0: TPtr cmd(argv->Des()); sl@0: User::CommandLine(cmd); sl@0: sl@0: TLex arguments(cmd); sl@0: TPtrC token; sl@0: token.Set(KNullDesC); sl@0: sl@0: // finds nth parameter that doesnt have a - sl@0: while(nPos >= 0 && !arguments.Eos()) sl@0: { sl@0: token.Set(arguments.NextToken()); sl@0: if(token.Length() > 0 && token[0] != '-') sl@0: nPos--; sl@0: } sl@0: sl@0: HBufC* result = token.AllocL(); sl@0: sl@0: CleanupStack::PopAndDestroy(argv); sl@0: sl@0: return result; sl@0: } sl@0: sl@0: void CTestSetup::InitFileserverSessionLC(RFs& aFs) sl@0: { sl@0: User::LeaveIfError(aFs.Connect()); sl@0: CleanupClosePushL(aFs); sl@0: sl@0: // enable tests to pass file handles between processes sl@0: aFs.ShareProtected(); sl@0: sl@0: // Create the private directory, to fix failures in tests that write to sl@0: // relative paths that were broken when the default path was changed to the sl@0: // private path in build 03429 sl@0: TFileName privatePath; sl@0: User::LeaveIfError(aFs.PrivatePath(privatePath)); sl@0: TInt err = aFs.MkDir(privatePath); sl@0: if (err != KErrNone && err != KErrAlreadyExists) sl@0: { sl@0: User::Leave(err); sl@0: } sl@0: } sl@0: sl@0: EXPORT_C void CTestSetup::CreateAndRunTestsL(TScriptTests theTestTypes[], sl@0: const TDesC& aScript, const TDesC& aLogFile, TBool aUseCommandLine, sl@0: CConsoleBase* aConsole, TBool* aResult) sl@0: { sl@0: LOG(_L("-- Test handler starting")); sl@0: sl@0: RFs fs; sl@0: sl@0: InitFileserverSessionLC(fs); sl@0: sl@0: CScriptSetup* testSetup = CScriptSetup::NewLC(aConsole); sl@0: CTestHandlerSettings* commandLineArgs = CTestHandlerSettings::NewLC(); sl@0: CTestSpec* testSpec = CTestSpec::NewL(); sl@0: CleanupStack::PushL(testSpec); sl@0: sl@0: TTestSummary summary; sl@0: if(testSetup->InitialiseL(fs, aScript, aLogFile, aUseCommandLine)) sl@0: { sl@0: sl@0: // Store the state of the heap and RFs resource count before the tests sl@0: sl@0: TInt initAllocCount = User::CountAllocCells(); sl@0: TInt initRFsCount = fs.ResourceCount(); sl@0: sl@0: // Store the initial count of process and thread handles sl@0: TInt initThreadHandleCount; sl@0: TInt initProcessHandleCount; sl@0: RThread().HandleCount(initProcessHandleCount, initThreadHandleCount); sl@0: sl@0: testSetup->LogFile().write(_L("
\r\n"));
sl@0: 		testSetup->LogFile().write(_L("State of the system before the tests:\n"));
sl@0: 		testSetup->LogFile().write(_L("\tRFs resource count: %d\n"), initRFsCount);
sl@0: 		testSetup->LogFile().write(_L("\tProcess handle count: %d\n"), initProcessHandleCount);
sl@0: 		testSetup->LogFile().write(_L("\tThread handle count: %d\n"), initThreadHandleCount);
sl@0: 			
sl@0: 		TRAPD(error, testSetup->SetupTestsL(fs, *testSpec, theTestTypes, *commandLineArgs))
sl@0: 			
sl@0: 		if(error==KErrNone)
sl@0: 			{
sl@0: 			CTestHandler* handler = CTestHandler::NewLC(fs, *testSpec, *commandLineArgs, 
sl@0: 														&testSetup->Console(),
sl@0: 														&testSetup->LogFile());
sl@0: 			handler->RunTestsL();
sl@0: 			summary = handler->Summary();
sl@0: 			testSpec->FreeAllTests();
sl@0: 			CleanupStack::PopAndDestroy(handler);
sl@0: 			
sl@0: 			// Do heap, RFs resource, process and thread handle balance checks	
sl@0: 			
sl@0: 			TInt finalRFsCount = fs.ResourceCount();
sl@0: 			
sl@0: 			testSetup->LogFile().write(_L("State of the system after the tests:\n"));
sl@0: 			testSetup->LogFile().write(_L("\tRFs resource count: %d\n"), finalRFsCount);
sl@0: 
sl@0: 			if (initRFsCount == finalRFsCount)
sl@0: 				{
sl@0: 				testSetup->LogFile().write(_L("\nRFs resource count ok: %d final\n\n"),
sl@0: 										   finalRFsCount);
sl@0: 				}
sl@0: 			else
sl@0: 				{
sl@0: 				testSetup->LogFile().write(_L("\nRFs resource count inbalance: %d final\n\n"),
sl@0: 										   finalRFsCount);
sl@0: 				++summary.iTestsFailed;
sl@0: 				}
sl@0: 	
sl@0: 
sl@0: 			TInt finalAllocCount = User::CountAllocCells();
sl@0: 			if (initAllocCount == finalAllocCount)
sl@0: 				{
sl@0: 				testSetup->LogFile().write(_L("\nHeap alloc count ok: %d final vs %d initial\n\n"),
sl@0: 										   finalAllocCount, initAllocCount);
sl@0: 				}
sl@0: 			else
sl@0: 				{
sl@0: 				testSetup->LogFile().write(_L("\nHeap alloc count inbalance: %d final vs %d initial\n\n"),
sl@0: 										   finalAllocCount, initAllocCount);
sl@0: 				++summary.iTestsFailed;
sl@0: 				}
sl@0: 
sl@0: 			TInt finalThreadHandleCount;
sl@0: 			TInt finalProcessHandleCount;
sl@0: 			RThread().HandleCount(finalProcessHandleCount, finalThreadHandleCount);
sl@0: 
sl@0: 			if (initProcessHandleCount == finalProcessHandleCount)
sl@0: 				{
sl@0: 				testSetup->LogFile().write(_L("\nProcess handle count ok: %d final vs %d initial\n\n"),
sl@0: 										   finalProcessHandleCount, initProcessHandleCount);
sl@0: 				}
sl@0: 			else
sl@0: 				{
sl@0: 				testSetup->LogFile().write(_L("\nProcess handle count imbalance: %d final vs %d initial\n\n"),
sl@0: 										   finalProcessHandleCount, initProcessHandleCount);
sl@0: 				++summary.iTestsFailed;
sl@0: 				}
sl@0: 
sl@0: 			if (initThreadHandleCount == finalThreadHandleCount)
sl@0: 				{
sl@0: 				testSetup->LogFile().write(_L("\nThread handle count ok: %d final vs %d initial\n\n"),
sl@0: 										   finalThreadHandleCount, initThreadHandleCount);
sl@0: 				}
sl@0: 			else
sl@0: 				{
sl@0: 				testSetup->LogFile().write(_L("\nThread handle count imbalance: %d final vs %d initial\n\n"),
sl@0: 										   finalThreadHandleCount, initThreadHandleCount);
sl@0: 				++summary.iTestsFailed;
sl@0: 				}
sl@0: 			++summary.iTestsRun;
sl@0: 			}
sl@0: 		
sl@0: 		// Set the result if required by caller
sl@0: 		if (aResult)
sl@0: 			{
sl@0: 			*aResult = summary.AllTestsPassed();
sl@0: 			}
sl@0: 
sl@0: 		summary.PrintL(testSetup->LogFile());
sl@0: 	
sl@0: 		// pauses runtime if command line requests it i.e. -w
sl@0: 		if(commandLineArgs->iWaitForKeyPressAtEnd)
sl@0: 			{
sl@0: 			testSetup->Console().Printf(_L("\nPress a key to quit"));
sl@0: 			testSetup->Console().Getch();
sl@0: 			}
sl@0: 
sl@0: 		}
sl@0: 
sl@0: 	CleanupStack::PopAndDestroy(4, &fs); // fs, testsetup, commandLineArgs and testspec
sl@0: 	// this MUST be the last thing to do incase objects being destructed on cleanup
sl@0: 	// have debug info
sl@0: 	//if (bTestSuccess)
sl@0: 	//	RDebug::RawPrint(_L("RTEST: SUCCESS : testhandler\n"));
sl@0: }
sl@0: 
sl@0: EXPORT_C void CTestSetup::CreateAndRunTestsL(THardcodedTests theTestTypes[], const TDesC& aDefaultLog)
sl@0: {
sl@0:     LOG(_L("-- Test handler starting")); 
sl@0:  
sl@0: 	RFs fs;
sl@0: 
sl@0: 	InitFileserverSessionLC(fs);
sl@0: 	
sl@0: 	InitFileserverSessionLC(fs);
sl@0: 	
sl@0: 	CHardcodedSetup* testSetup = CHardcodedSetup::NewLC();
sl@0: 	CTestHandlerSettings* commandLineArgs = CTestHandlerSettings::NewLC();
sl@0:  	CTestSpec* testSpec = CTestSpec::NewL();
sl@0: 	CleanupStack::PushL(testSpec);
sl@0: 
sl@0: 	TTestSummary summary;
sl@0: 	if(testSetup->InitialiseL(fs, aDefaultLog))
sl@0: 		{
sl@0: 		
sl@0: 		// Store the state of the heap and RFs resource count before the tests
sl@0: 		
sl@0: 		TInt initAllocCount = User::CountAllocCells();
sl@0: 		TInt initRFsCount = fs.ResourceCount();
sl@0: 				
sl@0: 		TRAPD(error, testSetup->SetupTestsL(fs, *testSpec, theTestTypes, *commandLineArgs));
sl@0: 			
sl@0: 		if(error==KErrNone)
sl@0: 			{
sl@0: 			CTestHandler* handler = CTestHandler::NewLC(fs, *testSpec, *commandLineArgs, 
sl@0: 														&testSetup->Console(),
sl@0: 														&testSetup->LogFile());
sl@0: 			handler->RunTestsL();
sl@0: 			summary = handler->Summary();
sl@0: 			testSpec->FreeAllTests();
sl@0: 			CleanupStack::PopAndDestroy(handler);
sl@0: 
sl@0: 			// Do heap and RFs resource balance checks
sl@0: 			
sl@0: 			TInt finalRFsCount = fs.ResourceCount();
sl@0: 			
sl@0: 			testSetup->LogFile().write(_L("State of the system after the tests:\n"));
sl@0: 			testSetup->LogFile().write(_L("\tRFs resource count: %d\n"), finalRFsCount);
sl@0: 
sl@0: 			if (initRFsCount == finalRFsCount)
sl@0: 				{
sl@0: 				testSetup->LogFile().write(_L("\nRFs resource count ok: %d final\n\n"),
sl@0: 										   finalRFsCount);
sl@0: 				}
sl@0: 			else
sl@0: 				{
sl@0: 				testSetup->LogFile().write(_L("\nRFs resource count inbalance: %d final\n\n"),
sl@0: 										   finalRFsCount);
sl@0: 				++summary.iTestsFailed;
sl@0: 				}
sl@0: 
sl@0: 			TInt finalAllocCount = User::CountAllocCells();
sl@0: 			if (initAllocCount == finalAllocCount)
sl@0: 				{
sl@0: 				testSetup->LogFile().write(_L("\nHeap alloc count ok: %d final vs %d initial\n\n"),
sl@0: 										   finalAllocCount, initAllocCount);
sl@0: 				}
sl@0: 			else
sl@0: 				{
sl@0: 				testSetup->LogFile().write(_L("\nHeap alloc count inbalance: %d final vs %d initial\n\n"),
sl@0: 										   finalAllocCount, initAllocCount);
sl@0: 				++summary.iTestsFailed;
sl@0: 				}
sl@0: 			++summary.iTestsRun;
sl@0: 			}
sl@0: 
sl@0: 		summary.PrintL(testSetup->LogFile());
sl@0: 
sl@0: 		// pauses runtime if command line requests it i.e. -w
sl@0: 		if(commandLineArgs->iWaitForKeyPressAtEnd)
sl@0: 			{
sl@0: 			testSetup->Console().Printf(_L("\nPress a key to quit"));
sl@0: 			testSetup->Console().Getch();
sl@0: 			}
sl@0: 		}
sl@0: 
sl@0: 	CleanupStack::PopAndDestroy(4, &fs); // fs, testsetup, commandLineArgs and testspec
sl@0: 	
sl@0: 	//if (bTestSuccess)
sl@0: 	//	RDebug::RawPrint(_L("RTEST: SUCCESS : testhandler\n"));
sl@0: }
sl@0: 
sl@0: //Check all flags is provided for understanability - Each grouping function called calls the next function
sl@0: // if it has succeeded
sl@0: TBool CTestSetup::CheckAllFlags(const CTestHandlerSettings& aCommandLineSettings, TInt& aScriptGroupings)
sl@0: 	{
sl@0: 	return CheckExhaustiveandSmoke(aCommandLineSettings,aScriptGroupings);
sl@0: 	}
sl@0: 
sl@0: TBool CTestSetup::CheckExhaustiveandSmoke(const CTestHandlerSettings& aCommandLineSettings, TInt& aScriptGroupings)
sl@0: 	{
sl@0: 	
sl@0: 	if (aCommandLineSettings.iExhaust || (aScriptGroupings & SMOKE))
sl@0: 		{
sl@0: 		//Do other flags
sl@0: 		return CheckSkipped(aCommandLineSettings, aScriptGroupings);
sl@0: 		}
sl@0: 	else
sl@0: 
sl@0: 		return EFalse;
sl@0: 	}
sl@0: 
sl@0: 
sl@0: TBool CTestSetup::CheckSkipped(const CTestHandlerSettings& aCommandLineSettings,
sl@0: 							   TInt& aScriptGroupings)
sl@0: 	{
sl@0: 	if (aCommandLineSettings.iSkip || !(aScriptGroupings & SKIP))
sl@0: 		{
sl@0: 		//Do other flags
sl@0: 		return CheckInteractive(aCommandLineSettings, aScriptGroupings);
sl@0: 		}
sl@0: 	else
sl@0: 		return EFalse;
sl@0: 	}
sl@0: 
sl@0: TBool CTestSetup::CheckInteractive(const CTestHandlerSettings& aCommandLineSettings,
sl@0: 								   TInt& aScriptGroupings)
sl@0: 	{
sl@0: 	if (aCommandLineSettings.iInt || !(aScriptGroupings & INTER))
sl@0: 		{
sl@0: 		//Do other flags
sl@0: 		return CheckOOMandCancel(aCommandLineSettings, aScriptGroupings);
sl@0: 		}
sl@0: 	else
sl@0: 		return EFalse;
sl@0: 	}
sl@0: 
sl@0: TBool CTestSetup::CheckOOMandCancel(const CTestHandlerSettings& aCommandLineSettings, 
sl@0: 									TInt& aScriptGroupings)
sl@0: 	{
sl@0: 
sl@0: 	if ((aScriptGroupings & (EXOOM | INOOM)) == (EXOOM | INOOM))
sl@0: 		User::Panic(_L("Test is both Included and Excluded from OOM"), 1);
sl@0: 
sl@0: 	if ((aScriptGroupings & (EXCANCEL | INCANCEL)) == (EXCANCEL | INCANCEL))
sl@0: 		User::Panic(_L("Test is both Included and Excluded from Cancel"), 1);
sl@0: 
sl@0: 	if ((aScriptGroupings & (INOOM | INCANCEL)) == (INOOM | INCANCEL))
sl@0: 		User::Panic(_L("Test is in both OOM and Cancel groups"), 1);
sl@0: 
sl@0: 	// Check to see whether -o set
sl@0: 	if (aCommandLineSettings.iOOM)
sl@0: 		{
sl@0: 		//Is the test in OOM group?
sl@0: 		if (aScriptGroupings & INOOM)
sl@0: 			return ETrue;
sl@0: 		else
sl@0: 		//Is the test excluded from OOM?
sl@0: 		if (aScriptGroupings & EXOOM)
sl@0: 			return EFalse;
sl@0: 		}
sl@0: 	else
sl@0: 	if (aScriptGroupings & INOOM)
sl@0: 			return EFalse;
sl@0: 
sl@0: 	// Check to see whether -c set
sl@0: 	if (aCommandLineSettings.iCancel)
sl@0: 		{
sl@0: 		//Is the test in CANCEL group?
sl@0: 		if (aScriptGroupings & INCANCEL)
sl@0: 			return ETrue;
sl@0: 		else
sl@0: 		//Is the test excluded from CANCEL?
sl@0: 		if (aScriptGroupings & EXCANCEL)
sl@0: 			return EFalse;
sl@0: 		}
sl@0: 	else
sl@0: 	if (aScriptGroupings & INCANCEL)
sl@0: 			return EFalse;
sl@0: 
sl@0: 
sl@0: 	return ETrue;
sl@0: 	}
sl@0: 
sl@0: 
sl@0: EXPORT_C CTestSetup::~CTestSetup()
sl@0: 	{
sl@0: 	delete iLogFile;
sl@0: 	delete iTestConsole;
sl@0: 	if (iConsoleOwned)
sl@0: 		{
sl@0: 		// Note that the heap count when this is deleted must be the same as
sl@0: 		// when the console was allocated, otherwise a panic will occur.  This
sl@0: 		// is due to the techview implementation of the console doing a heap
sl@0: 		// mark in its destructor.
sl@0: 		delete iConsole;
sl@0: 		}
sl@0: 	}
sl@0: 
sl@0: CTestSetup::CTestSetup(CConsoleBase* aConsole) 
sl@0: 	: iConsole(aConsole), iConsoleOwned(aConsole == NULL)
sl@0: 	{
sl@0: 	}
sl@0: 
sl@0: void CTestSetup::ConstructL()
sl@0: 	{
sl@0: 	if (iConsoleOwned)
sl@0: 		{
sl@0: 		iConsole = Console::NewL(_L("Test code"), TSize(KDefaultConsWidth, KDefaultConsHeight));
sl@0: 		}
sl@0: 
sl@0: 	// Currently the console passed to the test actions discards all output -
sl@0: 	// this is an attempt to make hardware tests run faster.  All information
sl@0: 	// should be written to the log file anyway, and I'd like to remove use of a
sl@0: 	// console as well as the log file.  -- jc
sl@0: 	iTestConsole = new (ELeave) CDummyConsole();
sl@0: 	}
sl@0: 
sl@0: void CTestSetup::OpenLogFileL(RFs &aFs, TInt nPos, const TDesC &aLogFile, TBool aUseCommandline)
sl@0: 	{
sl@0: 	HBufC* logFileName = NULL;
sl@0: 
sl@0: 	if (aUseCommandline)
sl@0: 		{
sl@0: 		logFileName = GetArgument(nPos);
sl@0: 		CleanupStack::PushL(logFileName);
sl@0: 		
sl@0: 		// check if logfile was specified on command line
sl@0: 		if(logFileName->Length()==0)
sl@0: 			{
sl@0: 			// empty so remove it
sl@0: 			CleanupStack::PopAndDestroy(logFileName);
sl@0: 			logFileName = NULL;
sl@0: 			}
sl@0: 		}
sl@0: 	
sl@0: 	if(logFileName == NULL)
sl@0: 		{
sl@0: 		if(aLogFile.Length()==0)
sl@0: 			{
sl@0: 			PRINTANDLOG(_L("No log file specified on command line and no default given"));
sl@0: 			User::Leave(KErrArgument);
sl@0: 			}
sl@0: 		else
sl@0: 			{
sl@0: 			// there is a default copy that
sl@0: 			logFileName = aLogFile.AllocL();
sl@0: 			CleanupStack::PushL(logFileName);
sl@0: 			}
sl@0: 		}
sl@0: 
sl@0: 	PRINTANDLOG1(_L("Log file: %S"), logFileName);
sl@0: 	
sl@0: 	RFile logFile;
sl@0: 
sl@0: 	// attempts to create directories incase they dont exist
sl@0: 	aFs.MkDirAll(*logFileName); // ignore errors
sl@0:  
sl@0: 	// write over any existing log
sl@0: 	User::LeaveIfError(logFile.Replace(aFs, *logFileName, EFileWrite));
sl@0: 
sl@0: 	// output goes only to the log file
sl@0: 	iLogFile = new(ELeave) FileOutput(logFile);
sl@0: 	CleanupStack::PopAndDestroy(logFileName);
sl@0: 	};