1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/mm/mmtestenv/mmtestfw/recog/TestFrameworkRecog.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,321 @@
1.4 +// Copyright (c) 2003-2009 Nokia Corporation and/or its subsidiary(-ies).
1.5 +// All rights reserved.
1.6 +// This component and the accompanying materials are made available
1.7 +// under the terms of "Eclipse Public License v1.0"
1.8 +// which accompanies this distribution, and is available
1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
1.10 +//
1.11 +// Initial Contributors:
1.12 +// Nokia Corporation - initial contribution.
1.13 +//
1.14 +// Contributors:
1.15 +//
1.16 +// Description:
1.17 +//
1.18 +
1.19 +#include <fbs.h>
1.20 +#include "TestFrameworkRecog.h"
1.21 +#include "TestFrameworkMain.h"
1.22 +
1.23 +#ifdef __WINS__
1.24 +#include <windows.h> // for ExitProcess
1.25 +#endif // __WINS__
1.26 +
1.27 +GLREF_C void StartupL();
1.28 +const TInt KThreadStackSize=0x2000; // 8KB
1.29 +const TInt KThreadInitHeapSize=0x1000; // 4KB
1.30 +const TInt KThreadMaxHeapSize=0x1000000; // 16MB
1.31 +const TInt KThreadStartupDelay=30000000; // 30 seconds
1.32 +const TInt KMaxLineLength=256; // max length of config file line
1.33 +_LIT(KLitConfigFileName, "C:\\MM\\AutorunTests.cfg");
1.34 +
1.35 +//
1.36 +
1.37 +CTestFrameworkRecognizer::CTestFrameworkRecognizer()
1.38 + : CApaDataRecognizerType(KUidTestFrameworkRecognizer, CApaDataRecognizerType::ENormal)
1.39 + {
1.40 + }
1.41 +
1.42 +CTestFrameworkRecognizer::~CTestFrameworkRecognizer()
1.43 + {
1.44 + delete iTestActive;
1.45 + }
1.46 +
1.47 +//
1.48 +// CApaDataRecognizerType stuff...
1.49 +TUint CTestFrameworkRecognizer::PreferredBufSize()
1.50 + {
1.51 + return 0;
1.52 + }
1.53 +
1.54 +TDataType CTestFrameworkRecognizer::SupportedDataTypeL(TInt /*aIndex*/) const
1.55 + {
1.56 + return TDataType();
1.57 + }
1.58 +
1.59 +void CTestFrameworkRecognizer::DoRecognizeL(const TDesC& /*aName*/, const TDesC8& /*aBuffer*/)
1.60 + {
1.61 + }
1.62 +
1.63 +//
1.64 +// Entry point of recognizer
1.65 +EXPORT_C CApaDataRecognizerType* CreateRecognizer()
1.66 + {
1.67 + CTestFrameworkRecognizer* self = new CTestFrameworkRecognizer();
1.68 + TRAPD(err, self->DoCreateL());
1.69 + return self;
1.70 + }
1.71 +
1.72 +void CTestFrameworkRecognizer::DoCreateL()
1.73 + {
1.74 + // Open the config file
1.75 + LoadConfigFileL(KLitConfigFileName);
1.76 +
1.77 + // If the RUN_SCRIPT command is present in the config file, run each test in a separate thread
1.78 + if (iRunScript)
1.79 + {
1.80 + // Create active object waiting on thread death
1.81 + iTestActive = new(ELeave) CTestFrameworkRecogActive(iTestScriptArray); // Takes ownership of iTestScriptArray
1.82 +
1.83 + // Create the first test thread
1.84 + iTestActive->CreateNextTestThread();
1.85 + }
1.86 + }
1.87 +
1.88 +void CTestFrameworkRecognizer::LoadConfigFileL(const TDesC& aFileName)
1.89 + {
1.90 + RFs fs;
1.91 + User::LeaveIfError(fs.Connect());
1.92 + CleanupClosePushL(fs);
1.93 +
1.94 + TEntry entry;
1.95 + User::LeaveIfError(fs.Entry(aFileName, entry));
1.96 +
1.97 + RFile file;
1.98 + User::LeaveIfError(file.Open(fs, aFileName, EFileRead));
1.99 + CleanupClosePushL(file);
1.100 +
1.101 + TInt size;
1.102 + User::LeaveIfError(file.Size(size));
1.103 + TUint8* fileData = (TUint8*)User::AllocLC(size);
1.104 + TPtr8 ptr(fileData, 0, size);
1.105 + User::LeaveIfError(file.Read(ptr));
1.106 +
1.107 + iTestScriptArray = new(ELeave) CTestScriptArray(4);
1.108 +
1.109 + // Process the config file
1.110 + TLex8 lex(ptr);
1.111 + while (!lex.Eos())
1.112 + {
1.113 + // skip any spaces
1.114 + while (lex.Peek() == ' ')
1.115 + lex.Inc();
1.116 + // mark the start of the line
1.117 + lex.Mark();
1.118 + // move to the next
1.119 + while (!lex.Eos() && lex.Peek() != '\n')
1.120 + lex.Inc();
1.121 + // step over \n
1.122 + if (lex.Peek() == '\n' )
1.123 + lex.Inc();
1.124 +
1.125 + // Process line
1.126 + TPtrC8 linePtr = lex.MarkedToken();
1.127 + ProcessLineL(linePtr);
1.128 + }
1.129 + CleanupStack::PopAndDestroy(fileData);
1.130 + CleanupStack::PopAndDestroy(2); // file, fs
1.131 + }
1.132 +
1.133 +void CTestFrameworkRecognizer::ProcessLineL(const TDesC8& aLine)
1.134 + {
1.135 + ASSERT(aLine.Length() <= KMaxLineLength);
1.136 + TBuf<KMaxLineLength> buf;
1.137 + buf.Copy(aLine);
1.138 + if (buf.Find(_L("//"))==0)
1.139 + {
1.140 + // ignore comments
1.141 + }
1.142 + else
1.143 + {
1.144 + // Get the script path and startup delay
1.145 + TLex lex(buf);
1.146 + if (!iRunScript)
1.147 + {
1.148 + if (lex.NextToken().Compare(_L("RUN_SCRIPT")) == 0)
1.149 + iRunScript = ETrue;
1.150 + }
1.151 + else
1.152 + {
1.153 + // Parse the parameters
1.154 + TTestScriptInfo info;
1.155 + info.iScriptPath = lex.NextToken();
1.156 + info.iThreadStartupDelay = 0;
1.157 +
1.158 + TPtrC token(lex.NextToken());
1.159 + while (token.Length())
1.160 + {
1.161 + if (token[0] == '-')
1.162 + {
1.163 + info.iParams.Append(token);
1.164 + info.iParams.Append(' ');
1.165 + }
1.166 + else
1.167 + {
1.168 + // Assume this to be the startup delay
1.169 + TLex tokenLex(token);
1.170 + User::LeaveIfError(tokenLex.Val(info.iThreadStartupDelay));
1.171 + if (info.iThreadStartupDelay < 0)
1.172 + info.iThreadStartupDelay = 0;
1.173 + }
1.174 + token.Set(lex.NextToken());
1.175 + }
1.176 +
1.177 + // Add the script info
1.178 + if (info.iScriptPath.Length())
1.179 + iTestScriptArray->AppendL(info);
1.180 + }
1.181 + }
1.182 + }
1.183 +
1.184 +//
1.185 +// CTestFrameworkRecogActive
1.186 +CTestFrameworkRecogActive::CTestFrameworkRecogActive(CTestScriptArray* aTestScriptArray)
1.187 + : CActive(EPriorityStandard), iTestScriptArray(aTestScriptArray)
1.188 + {
1.189 + CActiveScheduler::Add(this);
1.190 + iCurrentScript = -1;
1.191 + }
1.192 +
1.193 +CTestFrameworkRecogActive::~CTestFrameworkRecogActive()
1.194 + {
1.195 + delete iTestScriptArray;
1.196 + }
1.197 +
1.198 +TInt CTestFrameworkRecogActive::CreateNextTestThread()
1.199 + {
1.200 + // Create the next test in a separate thread
1.201 + iCurrentScript++;
1.202 + RThread thread;
1.203 + TBuf<16> threadName;
1.204 + threadName.Format(_L("TFR_THREAD_%d"), iCurrentScript);
1.205 +
1.206 + TInt err = thread.Create(threadName, &StartTestThreadFn, KThreadStackSize,
1.207 + &User::Heap(), this);
1.208 +
1.209 + if (err == KErrNone)
1.210 + {
1.211 + thread.Logon(iStatus);
1.212 + thread.Resume();
1.213 + SetActive();
1.214 + }
1.215 + return err;
1.216 + }
1.217 +
1.218 +void CTestFrameworkRecogActive::DoCancel()
1.219 + {
1.220 + }
1.221 +
1.222 +void CTestFrameworkRecogActive::RunL()
1.223 + {
1.224 + // This will run when the thread created in CreateNextTestThreadL dies
1.225 + //
1.226 + TInt err = KErrNone;
1.227 + if (iCurrentScript < (iTestScriptArray->Count() - 1))
1.228 + {
1.229 + err = CreateNextTestThread();
1.230 + }
1.231 + else
1.232 + {
1.233 + // Tests finished
1.234 + delete iTestScriptArray;
1.235 + iTestScriptArray = NULL;
1.236 +#ifdef __WINS__
1.237 + // Cause the emulator to exit
1.238 + ExitProcess(0);
1.239 +#endif // __WINS__
1.240 + }
1.241 + }
1.242 +
1.243 +TInt CTestFrameworkRecogActive::StartTestThreadFn(TAny* aPtr)
1.244 + {
1.245 + CTestFrameworkRecogActive* self = static_cast<CTestFrameworkRecogActive*>(aPtr);
1.246 + TRAPD(err, self->DoStartTestThreadL());
1.247 + return err;
1.248 + }
1.249 +
1.250 +void CTestFrameworkRecogActive::DoStartTestThreadL()
1.251 + {
1.252 + // Create the thread and wait until it's finished
1.253 + RThread thread;
1.254 + TBuf<16> threadName;
1.255 + threadName.Format(_L("TESTFRMRECOG_%d"), iCurrentScript);
1.256 +
1.257 + TInt err = thread.Create(threadName, &ThreadFunc, KThreadStackSize,
1.258 + &User::Heap(), this);
1.259 +
1.260 + if (err == KErrNone)
1.261 + {
1.262 + RSemaphore sem;
1.263 + err = sem.CreateGlobal(KRecogSemaphoreName, 0);
1.264 + if (err == KErrAlreadyExists)
1.265 + err = sem.OpenGlobal(KRecogSemaphoreName);
1.266 + if (err == KErrNone)
1.267 + {
1.268 + // Start the thread and wait for it to signal us that it's finished
1.269 + thread.Resume();
1.270 + sem.Wait();
1.271 + }
1.272 + }
1.273 + User::LeaveIfError(err);
1.274 + }
1.275 +
1.276 +
1.277 +TInt CTestFrameworkRecogActive::ThreadFunc(TAny* aPtr)
1.278 + {
1.279 + CTrapCleanup* cleanup = CTrapCleanup::New(); // get clean-up stack
1.280 + CTestFrameworkRecogActive* self = static_cast<CTestFrameworkRecogActive*>(aPtr);
1.281 + TRAPD(err, self->DoThreadFuncL());
1.282 + delete cleanup; // destroy clean-up stack
1.283 + return err;
1.284 + }
1.285 +
1.286 +void CTestFrameworkRecogActive::DoThreadFuncL()
1.287 + {
1.288 + // Run the test script, using filename held in iScriptPath
1.289 + CActiveScheduler* scheduler=new(ELeave) CActiveScheduler;
1.290 + CleanupStack::PushL(scheduler);
1.291 + CActiveScheduler::Install(scheduler);
1.292 +
1.293 +// Hurricane emulator only - start all services which we require to run tests.
1.294 +// Future enhancement :- add these startups to TestUtils
1.295 +#if defined(__WINS__)
1.296 + #ifndef EXCLUDE_FOR_UNITTEST
1.297 + FbsStartup();
1.298 + #endif // EXCLUDE_FOR_UNITTEST
1.299 +#endif
1.300 +
1.301 + // Get the current script
1.302 + const TTestScriptInfo& script = iTestScriptArray->At(iCurrentScript);
1.303 +
1.304 + // Delay for several seconds to allow vital bits of the emulator to start up (window server bits)
1.305 + User::After(script.iThreadStartupDelay);
1.306 +
1.307 + // Format the parameter to be passed to the test framework
1.308 + TFileName args;
1.309 + if (script.iParams.Length())
1.310 + {
1.311 + // Add the params
1.312 + args.Append(script.iParams);
1.313 + args.Append(' ');
1.314 + }
1.315 + args.Append(script.iScriptPath);
1.316 +
1.317 + // Run the script
1.318 + CTestFrameworkMain* tester = CTestFrameworkMain::NewLC();
1.319 + tester->StartTestingL(args);
1.320 +
1.321 + CleanupStack::PopAndDestroy(2, scheduler); // tester, scheduler
1.322 + }
1.323 +
1.324 +