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.
18 #include "TestFrameworkServer.h"
20 #include "TestFrameworkServer.inl"
22 #include <testframeworkipc.h>
27 * Initiate server exit when the timer expires
32 void CTestFrameworkServerShutdown::RunL()
34 CActiveScheduler::Stop();
39 * Test Framework server static constructor
40 * NOTE : only one instance of the server process should run
41 * during any Test Framework run; all client sessions
42 * will use this one instance.
47 CMmfIpcServer* CTestFrameworkServer::NewL()
49 CTestFrameworkServer* s = new(ELeave) CTestFrameworkServer;
50 CleanupStack::PushL(s);
58 * Test Framework server first-phase constructor
63 CTestFrameworkServer::CTestFrameworkServer()
64 :CMmfIpcServer(0, ESharableSessions)
68 // set up so that iConsole is NULL. A call to OpenLogL will initialise the console
73 * Test Framework server second-phase constructor
74 * Starts the server; does not initialise the log
75 * (this must be done independently with a call to OpenLogL)
80 void CTestFrameworkServer::ConstructL()
82 StartL(KTestFrameworkServerName);
83 //Ensure that the server will exit even if the first client fails to connect
84 iShutdown.ConstructL();
87 iLogMode = 0; // initial value : not running any log
94 * Test Framework server destructor
99 CTestFrameworkServer::~CTestFrameworkServer()
106 * Create new server session.
108 * @param "const TVersion &aVersion"
109 * Server version (required)
111 * @return "CSharableSession*"
112 * The created server session.
117 CMmfIpcSession* CTestFrameworkServer::NewSessionL(const TVersion &aVersion) const
119 TVersion ver(KTestFrameworkServerMajorVersionNumber,
120 KTestFrameworkServerMinorVersionNumber,
121 KTestFrameworkServerBuildVersionNumber);
122 if (!User::QueryVersionSupported(ver, aVersion))
123 User::Leave(KErrNotSupported);
125 return new(ELeave) CTestFrameworkServerSession();
130 * Add a server session.
135 void CTestFrameworkServer::AddSession()
143 * Drop a server session.
148 void CTestFrameworkServer::DropSession()
150 if (--iSessionCount <= 0)
156 * Add an input window (at request from client).
158 * @param "CTestFrameworkServerSession* aOwner"
159 * Window owning server session
164 void CTestFrameworkServer::AddInputWindowL(CTestFrameworkServerSession* aOwner)
166 if (iInputWindow.HasOwner())
167 User::Leave(KErrAlreadyExists);
169 iInputWindow.SetOwner(aOwner);
171 rect.iTl = TPoint(0,0);
172 rect.iBr = TPoint(KConsFullScreen, KConsFullScreen);
173 iInputWindow.SetWinRectAndNotifyOwner(rect);
178 * Remove an input window (at request from client).
180 * @param "CTestFrameworkServerSession* aOwner"
181 * Window owning server session
186 void CTestFrameworkServer::RemoveWindow(CTestFrameworkServerSession* aOwner)
188 if (aOwner == iInputWindow.Owner())
190 iInputWindow.SetOwner(NULL);
198 * @param "const TDesC& aLogName"
201 * @param "TInt aLogMode"
202 * The log mode (as bitmask of TTestFrameworkLogMode)
207 void CTestFrameworkServer::OpenLogL(const TDesC& aLogName, TInt aLogMode)
209 // NB we need to check if a console is already open - if so, we do NOT
210 // create another one. Ditto with file / port.
212 if(aLogMode & ELogToConsole)
216 iConsole = CServerConsole::NewL(aLogName);
218 CConsoleBase* theConsole = iConsole->Console();
219 theConsole->Printf(_L("%S : Server log starting\n"), &aLogName);
221 iLogMode |= ELogToConsole;
223 if (aLogMode & ELogConsoleFull)
224 iLogMode |= ELogConsoleFull;
228 // NB relative paths will not work with TParse (there is no file server open).
229 // Exception is a bare filename (with no path) : this will be found in root of C:
231 // NOTE! We have no mechanism to notify this error. The console will display
232 // and then exit. The log file cannot be opened.
234 // TO BE ENHANCED - if console is made active, then we can pause
236 if(aLogMode & ELogToFile)
240 TRAPD(err, iFileLogger = CFileLogger::NewL());
243 // if we can't create a logger, we panic
244 User::Panic(_L("TestFrameworkServer"), 1);
247 _LIT(KLogPath, "C:\\Logs\\TestResults");
248 _LIT(KDefault, "C:\\.htm");
251 parseLogName.Set(aLogName, NULL, NULL);
253 TFileName logFilePath;
254 logFilePath = KLogPath;
256 if(parseLogName.PathPresent())
257 logFilePath.Append(parseLogName.Path());
259 logFilePath.Append(_L("\\"));
262 // overwrite extension if supplied with .htm
263 TParse logFileFullName;
264 TInt returnCode = logFileFullName.Set(KDefault, &logFilePath, &aLogName);
265 if (returnCode == KErrNone)
267 TInt ret = iFileLogger->Connect();
270 iFileLogger->CreateLog(logFilePath, logFileFullName.NameAndExt());
271 iLogMode |= ELogToFile;
277 if(aLogMode & ELogToPort)
279 // RDebug::Print will deal with the serial port, we don't do anything special here
280 iLogMode |= ELogToPort;
288 * @param "const TDesC& aMsg"
289 * The message string to write
291 * @param "TInt aLogMode"
292 * The log mode (as bitmask of TTestFrameworkLogMode)
297 void CTestFrameworkServer::WriteLog(const TDesC& aMsg, TInt aLogMode)
299 if(aLogMode & ELogToConsole)
303 CConsoleBase* theConsole = iConsole->Console();
304 theConsole->Printf(aMsg);
305 theConsole->Printf(_L("\n")); // add newline
309 if(aLogMode & ELogToFile)
313 iFileLogger->WriteLog(aMsg);
317 if(aLogMode & ELogToPort)
330 void CTestFrameworkServer::CloseLog()
332 if (iFileLogger != NULL)
333 iFileLogger->CloseLog();
343 * Get the log status.
346 * The log status (as bitmask of TTestFrameworkLogMode)
351 TInt CTestFrameworkServer::LogStatus() const
359 * Default window constructor (no owner)
371 * Default window constructor
373 * @param "CTestFrameworkServerSession* aOwner"
378 TWindow::TWindow(CTestFrameworkServerSession* aOwner)
385 * Set the window owner
387 * @param "CTestFrameworkServerSession* aOwner"
392 void TWindow::SetOwner(CTestFrameworkServerSession* aOwner)
399 * Set the window rectangle, and notify owner
401 * @param "const TRect& aWinRect"
402 * The window rectangle
406 void TWindow::SetWinRectAndNotifyOwner(const TRect& aWinRect)
409 iOwner->NotifyWindowChanged(aWinRect);
414 * Does this window have an owner?
417 * ETrue if window has an owner
421 TBool TWindow::HasOwner()
430 * Server session first-phase constructor
435 CTestFrameworkServerSession::CTestFrameworkServerSession()
441 * Create a server session.
444 * The server to add this session to
449 void CTestFrameworkServerSession::CreateL(const CMmfIpcServer& aServer)
451 CMmfIpcSession::CreateL(aServer); // does not leave
452 //Add session to server first. If anything leaves, it will be removed by the destructor
453 iServer = STATIC_CAST(CTestFrameworkServer*, (CONST_CAST(CMmfIpcServer*, &aServer)));
454 iServer->AddSession();
460 * Server session second-phase constructor
465 void CTestFrameworkServerSession::ConstructL()
471 * Server session destructor
476 CTestFrameworkServerSession::~CTestFrameworkServerSession()
478 iServer->RemoveWindow(this);
479 iServer->DropSession();
484 * Server session service function
487 * The message to be serviced.
492 void CTestFrameworkServerSession::ServiceL(const RMmfIpcMessage& aMessage)
494 switch (aMessage.Function())
496 case ECreateInputWindow:
497 SetOwnCopyOfWindowMessageL(aMessage);
498 iServer->AddInputWindowL(this);
500 case ENotifyIfWindowChange:
501 SetOwnCopyOfWindowMessageL(aMessage);
503 case ECancelNotifyIfWindowChange:
504 CompleteOwnCopyOfWindowMessage(KErrCancel);
505 aMessage.Complete(KErrNone);
512 TBuf<KMaxLogFilenameLength> msgBuf;
514 TInt r = MmfMessageUtilX::Read(aMessage, 0, msgBuf);
516 RunError(aMessage, r);
519 iServer->OpenLogL(msgBuf, aMessage.Int1());
520 aMessage.Complete(KErrNone);
527 TBuf<KMaxLogLineLength> msgBuf;
529 TInt r = MmfMessageUtilX::Read(aMessage, 0, msgBuf);
532 RunError(aMessage, r);
535 iServer->WriteLog(msgBuf, aMessage.Int1());
536 aMessage.Complete(KErrNone);
543 aMessage.Complete(KErrNone);
548 TPckgBuf<TInt> countPckg(iServer->LogStatus());
549 TInt r = MmfMessageUtilX::Write(aMessage, 0, countPckg);
551 RunError(aMessage, r);
553 aMessage.Complete(KErrNone);
558 // ? should panic here
565 * Error handler for server session.
566 * Completes the pending message using the error code
568 * @param "TInt aError"
572 * The error code (always KErrNone)
577 TInt CTestFrameworkServerSession::RunError(const RMmfIpcMessage& aMessage, TInt aError)
579 aMessage.Complete(aError);
585 * Set own copy of message.
586 * Helper function to enable window change notification.
589 * The message to be serviced.
594 void CTestFrameworkServerSession::SetOwnCopyOfWindowMessageL(const RMmfIpcMessage& aMessage)
596 if (iCanCompleteWindowMessage)
597 User::Leave(KErrAlreadyExists);
599 iWindowMessage = aMessage;
600 iCanCompleteWindowMessage = ETrue;
602 if (iNeedToNotifyClientOfWindowSizeChange)
604 TInt err = MmfMessageUtil::Write(iWindowMessage, 0, iWinRectBuf);
605 CompleteOwnCopyOfWindowMessage(err);
606 iNeedToNotifyClientOfWindowSizeChange = EFalse;
612 * Complete own copy of message.
613 * Helper function to enable window change notification.
615 * @param "TInt aReason"
616 * The message return code.
621 void CTestFrameworkServerSession::CompleteOwnCopyOfWindowMessage(TInt aReason)
623 if (iCanCompleteWindowMessage)
624 iWindowMessage.Complete(aReason);
625 iCanCompleteWindowMessage = EFalse;
630 * Window change notification.
632 * @param "const TRect& aWinRect"
633 * The window rectangle.
638 void CTestFrameworkServerSession::NotifyWindowChanged(const TRect& aWinRect)
640 iWinRectBuf() = aWinRect;
642 if (iCanCompleteWindowMessage)
644 TInt err = MmfMessageUtilX::Write(iWindowMessage, 0, iWinRectBuf);
645 CompleteOwnCopyOfWindowMessage(err);
649 iNeedToNotifyClientOfWindowSizeChange = ETrue;
653 // EXE/DLL initialisation
655 // NOTE! None of this should be built when compiling for Unit Testing
656 #if !defined (__TSU_TESTFRAMEWORK__)
661 * Perform all server initialisation, in particular creation of the
662 * scheduler and server; then run the scheduler
668 static void RunServerL()
670 // create and install the active scheduler we need
671 CActiveScheduler* s = new(ELeave) CActiveScheduler;
672 CleanupStack::PushL(s);
673 CActiveScheduler::Install(s);
675 // create the server (leave it on the cleanup stack)
676 CleanupStack::PushL(CTestFrameworkServer::NewL());
678 // Initialisation complete, now signal the client
679 RProcess::Rendezvous(KErrNone);
683 CActiveScheduler::Start();
685 // Cleanup the server and scheduler
686 CleanupStack::PopAndDestroy(2);
691 * Main entry point for the server thread
693 * @param "TTestFrameworkServerStart& aStart"
694 * The server starter.
699 static TInt RunServer()
702 CTrapCleanup* cleanup = CTrapCleanup::New();
703 TInt r = KErrNoMemory;
706 TRAP(r, RunServerL());
715 GLDEF_C TInt E32Main()
717 // Server process entry-point
718 // Recover the startup parameters and run the server
721 TInt r = RunServer();
726 #endif // __TSU_TESTFRAMEWORK__