First public contribution.
1 // Copyright (c) 2003-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.
17 #include "TestFrameworkRecog.h"
18 #include "TestFrameworkMain.h"
21 #include <windows.h> // for ExitProcess
24 GLREF_C void StartupL();
25 const TInt KThreadStackSize=0x2000; // 8KB
26 const TInt KThreadInitHeapSize=0x1000; // 4KB
27 const TInt KThreadMaxHeapSize=0x1000000; // 16MB
28 const TInt KThreadStartupDelay=30000000; // 30 seconds
29 const TInt KMaxLineLength=256; // max length of config file line
30 _LIT(KLitConfigFileName, "C:\\MM\\AutorunTests.cfg");
34 CTestFrameworkRecognizer::CTestFrameworkRecognizer()
35 : CApaDataRecognizerType(KUidTestFrameworkRecognizer, CApaDataRecognizerType::ENormal)
39 CTestFrameworkRecognizer::~CTestFrameworkRecognizer()
45 // CApaDataRecognizerType stuff...
46 TUint CTestFrameworkRecognizer::PreferredBufSize()
51 TDataType CTestFrameworkRecognizer::SupportedDataTypeL(TInt /*aIndex*/) const
56 void CTestFrameworkRecognizer::DoRecognizeL(const TDesC& /*aName*/, const TDesC8& /*aBuffer*/)
61 // Entry point of recognizer
62 EXPORT_C CApaDataRecognizerType* CreateRecognizer()
64 CTestFrameworkRecognizer* self = new CTestFrameworkRecognizer();
65 TRAPD(err, self->DoCreateL());
69 void CTestFrameworkRecognizer::DoCreateL()
71 // Open the config file
72 LoadConfigFileL(KLitConfigFileName);
74 // If the RUN_SCRIPT command is present in the config file, run each test in a separate thread
77 // Create active object waiting on thread death
78 iTestActive = new(ELeave) CTestFrameworkRecogActive(iTestScriptArray); // Takes ownership of iTestScriptArray
80 // Create the first test thread
81 iTestActive->CreateNextTestThread();
85 void CTestFrameworkRecognizer::LoadConfigFileL(const TDesC& aFileName)
88 User::LeaveIfError(fs.Connect());
89 CleanupClosePushL(fs);
92 User::LeaveIfError(fs.Entry(aFileName, entry));
95 User::LeaveIfError(file.Open(fs, aFileName, EFileRead));
96 CleanupClosePushL(file);
99 User::LeaveIfError(file.Size(size));
100 TUint8* fileData = (TUint8*)User::AllocLC(size);
101 TPtr8 ptr(fileData, 0, size);
102 User::LeaveIfError(file.Read(ptr));
104 iTestScriptArray = new(ELeave) CTestScriptArray(4);
106 // Process the config file
111 while (lex.Peek() == ' ')
113 // mark the start of the line
116 while (!lex.Eos() && lex.Peek() != '\n')
119 if (lex.Peek() == '\n' )
123 TPtrC8 linePtr = lex.MarkedToken();
124 ProcessLineL(linePtr);
126 CleanupStack::PopAndDestroy(fileData);
127 CleanupStack::PopAndDestroy(2); // file, fs
130 void CTestFrameworkRecognizer::ProcessLineL(const TDesC8& aLine)
132 ASSERT(aLine.Length() <= KMaxLineLength);
133 TBuf<KMaxLineLength> buf;
135 if (buf.Find(_L("//"))==0)
141 // Get the script path and startup delay
145 if (lex.NextToken().Compare(_L("RUN_SCRIPT")) == 0)
150 // Parse the parameters
151 TTestScriptInfo info;
152 info.iScriptPath = lex.NextToken();
153 info.iThreadStartupDelay = 0;
155 TPtrC token(lex.NextToken());
156 while (token.Length())
160 info.iParams.Append(token);
161 info.iParams.Append(' ');
165 // Assume this to be the startup delay
166 TLex tokenLex(token);
167 User::LeaveIfError(tokenLex.Val(info.iThreadStartupDelay));
168 if (info.iThreadStartupDelay < 0)
169 info.iThreadStartupDelay = 0;
171 token.Set(lex.NextToken());
174 // Add the script info
175 if (info.iScriptPath.Length())
176 iTestScriptArray->AppendL(info);
182 // CTestFrameworkRecogActive
183 CTestFrameworkRecogActive::CTestFrameworkRecogActive(CTestScriptArray* aTestScriptArray)
184 : CActive(EPriorityStandard), iTestScriptArray(aTestScriptArray)
186 CActiveScheduler::Add(this);
190 CTestFrameworkRecogActive::~CTestFrameworkRecogActive()
192 delete iTestScriptArray;
195 TInt CTestFrameworkRecogActive::CreateNextTestThread()
197 // Create the next test in a separate thread
201 threadName.Format(_L("TFR_THREAD_%d"), iCurrentScript);
203 TInt err = thread.Create(threadName, &StartTestThreadFn, KThreadStackSize,
204 &User::Heap(), this);
208 thread.Logon(iStatus);
215 void CTestFrameworkRecogActive::DoCancel()
219 void CTestFrameworkRecogActive::RunL()
221 // This will run when the thread created in CreateNextTestThreadL dies
224 if (iCurrentScript < (iTestScriptArray->Count() - 1))
226 err = CreateNextTestThread();
231 delete iTestScriptArray;
232 iTestScriptArray = NULL;
234 // Cause the emulator to exit
240 TInt CTestFrameworkRecogActive::StartTestThreadFn(TAny* aPtr)
242 CTestFrameworkRecogActive* self = static_cast<CTestFrameworkRecogActive*>(aPtr);
243 TRAPD(err, self->DoStartTestThreadL());
247 void CTestFrameworkRecogActive::DoStartTestThreadL()
249 // Create the thread and wait until it's finished
252 threadName.Format(_L("TESTFRMRECOG_%d"), iCurrentScript);
254 TInt err = thread.Create(threadName, &ThreadFunc, KThreadStackSize,
255 &User::Heap(), this);
260 err = sem.CreateGlobal(KRecogSemaphoreName, 0);
261 if (err == KErrAlreadyExists)
262 err = sem.OpenGlobal(KRecogSemaphoreName);
265 // Start the thread and wait for it to signal us that it's finished
270 User::LeaveIfError(err);
274 TInt CTestFrameworkRecogActive::ThreadFunc(TAny* aPtr)
276 CTrapCleanup* cleanup = CTrapCleanup::New(); // get clean-up stack
277 CTestFrameworkRecogActive* self = static_cast<CTestFrameworkRecogActive*>(aPtr);
278 TRAPD(err, self->DoThreadFuncL());
279 delete cleanup; // destroy clean-up stack
283 void CTestFrameworkRecogActive::DoThreadFuncL()
285 // Run the test script, using filename held in iScriptPath
286 CActiveScheduler* scheduler=new(ELeave) CActiveScheduler;
287 CleanupStack::PushL(scheduler);
288 CActiveScheduler::Install(scheduler);
290 // Hurricane emulator only - start all services which we require to run tests.
291 // Future enhancement :- add these startups to TestUtils
292 #if defined(__WINS__)
293 #ifndef EXCLUDE_FOR_UNITTEST
295 #endif // EXCLUDE_FOR_UNITTEST
298 // Get the current script
299 const TTestScriptInfo& script = iTestScriptArray->At(iCurrentScript);
301 // Delay for several seconds to allow vital bits of the emulator to start up (window server bits)
302 User::After(script.iThreadStartupDelay);
304 // Format the parameter to be passed to the test framework
306 if (script.iParams.Length())
309 args.Append(script.iParams);
312 args.Append(script.iScriptPath);
315 CTestFrameworkMain* tester = CTestFrameworkMain::NewLC();
316 tester->StartTestingL(args);
318 CleanupStack::PopAndDestroy(2, scheduler); // tester, scheduler