Update contrib.
1 // Copyright (c) 2009-2010 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 #include <featureuids.h>
20 #include "featurepanics.h"
21 #include <featurecontrol.h>
22 #include <featurenotifier.h>
23 #include "../src/inc/featmgrconfiguration.h"
24 #include "../src/inc/featmgrclientserver.h"
26 using namespace NFeature;
28 static RTest TheTest(_L("t_fmgrbadclient"));
30 const TInt KTestIterCount = 5000;
40 const TInt KMaxDesArgLen = 1000;
42 //If the FeatMgr server crashes and the test receives KErrServerTerminated error, then the
43 //next set will contain the last:
44 // - iteration number;
48 // - IPC arguments values;
53 TArgType iArgType[KMaxMessageArguments];
54 TInt iIntArg[KMaxMessageArguments];
55 TBuf<KMaxDesArgLen> iTextArg[KMaxMessageArguments];
56 TBuf8<KMaxDesArgLen> iBinArg[KMaxMessageArguments];
60 _LIT(KPanicCategory, "SrvTerm");
61 _LIT(KPanicCategory2, "InvArg");
62 const TInt KPanicCode = 1111;
63 const TInt KPanicCode2 = 2222;
65 ///////////////////////////////////////////////////////////////////////////////////////
67 //Deletes all created test files.
72 ///////////////////////////////////////////////////////////////////////////////////////
73 ///////////////////////////////////////////////////////////////////////////////////////
74 //Test macros and functions
75 void Check1(TInt aValue, TInt aLine, TBool aPrintThreadName = EFalse)
83 TName name = th.Name();
84 RDebug::Print(_L("*** Expression evaluated to false. Thread %S, Line %d\r\n"), &name, aLine);
88 TheTest.Printf(_L("*** Expression evaluated to false. Line %d\r\n"), aLine);
90 TheTest(EFalse, aLine);
93 void Check2(TInt aValue, TInt aExpected, TInt aLine, TBool aPrintThreadName = EFalse)
95 if(aValue != aExpected)
101 TName name = th.Name();
102 RDebug::Print(_L("*** Thread %S, Line %d Expected error: %d, got: %d\r\n"), &name, aLine, aExpected, aValue);
106 RDebug::Print(_L("*** Line %d, Expected error: %d, got: %d\r\n"), aLine, aExpected, aValue);
108 TheTest(EFalse, aLine);
111 #define TEST(arg) ::Check1((arg), __LINE__)
112 #define TEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__)
113 #define TTEST(arg) ::Check1((arg), __LINE__, ETrue)
114 #define TTEST2(aValue, aExpected) ::Check2(aValue, aExpected, __LINE__, ETrue)
116 //////////////////////////////////////////////////////////////////////////////////////////////////
118 static TInt StartFeatMgrServer()
121 const TUidType serverUid( KNullUid, KServerUid2, KNullUid );
122 TInt err = server.Create( KServerExeName, // FeatMgrServer.exe
123 KNullDesC, // A descriptor containing data passed as
124 // an argument to the thread function of
125 // the new process's main thread, when it
126 // is first scheduled.
127 serverUid, // FeatMgr server UID
128 EOwnerProcess ); // Ownership of this process handle
130 // Return error code if we couldn't create a process
131 if ( err == KErrNone )
133 // Rendezvous is used to detect server start
135 server.Rendezvous( stat );
137 if ( stat != KRequestPending )
139 server.Kill( KErrNone ); // Abort startup
143 server.Resume(); // Logon OK - start the server
146 User::WaitForRequest( stat ); // Wait for start or death
148 // We can't use the 'exit reason' if the server paniced as this
149 // is the panic 'reason' and may be '0' which cannot be distinguished
151 err = (server.ExitType() == EExitPanic)? KErrGeneral : stat.Int();
153 // We can close the handle now
160 //////////////////////////////////////////////////////////////////////////////////////////////////
161 ///////////////////////////// RTestFeatMgrSession ////////////////////////////////
162 //////////////////////////////////////////////////////////////////////////////////////////////////
164 class RTestFeatMgrSession : public RSessionBase
169 TInt SendReceive(TInt aFunction);
170 TInt SendReceive(TInt aFunction, const TIpcArgs& aArgs);
171 void SendReceive(TInt aFunction, const TIpcArgs& aArgs, TRequestStatus& aStatus);
174 TInt DoCreateSession();
177 TInt RTestFeatMgrSession::Connect()
179 TInt err = DoCreateSession();
180 if(err != KErrNone && err != KErrAlreadyExists)
187 void RTestFeatMgrSession::Close()
189 RSessionBase::Close();
192 TInt RTestFeatMgrSession::SendReceive(TInt aFunction)
194 return RSessionBase::SendReceive(aFunction);
197 TInt RTestFeatMgrSession::SendReceive(TInt aFunction, const TIpcArgs& aArgs)
199 return RSessionBase::SendReceive(aFunction, aArgs);
202 void RTestFeatMgrSession::SendReceive(TInt aFunction, const TIpcArgs& aArgs, TRequestStatus& aStatus)
204 RSessionBase::SendReceive(aFunction, aArgs, aStatus);
207 TInt RTestFeatMgrSession::DoCreateSession()
209 const TInt KRetry( 2 );
211 TInt retry( KRetry );
212 TInt err( KErrNone );
216 // Try to create a FeatMgr Server session
217 err = CreateSession(KServerProcessName,
218 TVersion(KServerVersionMajor, KServerVersionMinor, KServerVersionBuild),
221 if ( err != KErrNotFound && err != KErrServerTerminated )
223 // KErrNone or unrecoverable error
228 // Return code was KErrNotFound or KErrServerTerminated.
229 // Try to start a new FeatMgr Server
230 err = StartFeatMgrServer();
232 if ( err != KErrNone && err != KErrAlreadyExists )
234 // Unrecoverable error
245 void PrintIterationCount(TInt aIteration, TBool aFromThread = EFalse)
247 if((aIteration % 100) == 0)
251 TDateTime dt = time.DateTime();
253 tbuf.Format(_L("%02d:%02d:%02d.%06d"), dt.Hour(), dt.Minute(), dt.Second(), dt.MicroSecond());
256 RDebug::Print(_L("-----[%S] Test iterations: %d\r\n"), &tbuf, aIteration);
260 TheTest.Printf(_L("-----[%S] Test iterations: %d\r\n"), &tbuf, aIteration);
265 //Worker thread function.
266 //It behaves as a malicious client. Connects to the FeatMgr server. In each test iteration generates some random values
267 //for the function number, handle, IPC arguments. Then sends a command to the server using these
268 //randomly generated values. If the server crashes and the thread function receives KErrServerTerminated error,
269 //then the thread kills itself and the main thread will get KPanicCategory and KPanicCode as a reason for the
270 //worker thread's death. The last set of randomly generated values will be stored in the memory, pointed by aData argument.
271 TInt ThreadFunc1(void* aData)
275 CTrapCleanup* tc = CTrapCleanup::New();
278 TThreadData* p = static_cast <TThreadData*> (aData);
280 TThreadData& data = *p;
282 RTestFeatMgrSession sess;
283 TInt err = sess.Connect();
284 TTEST2(err, KErrNone);
286 while(++data.iIteration <= KTestIterCount)
288 PrintIterationCount(data.iIteration, ETrue);
290 data.iFunction = Math::Rand(data.iSeed) % (EFeatMgrSWIEnd + 1);//EFeatMgrSWIEnd - the last server message number (without resource checking IPCs))
291 for(TInt i=0;i<KMaxMessageArguments;++i)
293 //Initialize arguments
294 data.iArgType[i] = static_cast <TArgType> (Math::Rand(data.iSeed) % ELastArgType);
295 switch(data.iArgType[i])
298 data.iIntArg[i] = Math::Rand(data.iSeed) % 9711;
299 args.Set(i, data.iIntArg[i]);
303 TInt len = Math::Rand(data.iSeed) % KMaxDesArgLen;
304 data.iTextArg[i].SetLength(len);
305 args.Set(i, &data.iTextArg[i]);
310 TInt len = Math::Rand(data.iSeed) % KMaxDesArgLen;
311 data.iBinArg[i].SetLength(len);
312 args.Set(i, &data.iBinArg[i]);
316 User::Panic(KPanicCategory2, KPanicCode2);
321 User::SetJustInTime(EFalse);
323 if(data.iFunction == EFeatMgrReqNotify)
325 TRequestStatus status;
326 sess.SendReceive(data.iFunction, args, status);
327 if(status == KRequestPending)
329 err = sess.SendReceive(EFeatMgrReqNotifyCancelAll);
338 err = sess.SendReceive(data.iFunction, args);
340 if(err == KErrServerTerminated)
342 User::Panic(KPanicCategory, KPanicCode);
344 User::SetJustInTime(ETrue);
356 ////////////////////////////////////////////////////////////////////////////////////////////////////////////
359 @SYMTestCaseID PDS-EFM-CT-4065
361 @SYMTestPriority High
363 @SYMTestExpectedResults Test must not fail
368 TThreadData* p = new TThreadData;
370 TThreadData& data = *p;
374 data.iSeed = now.Int64();
376 _LIT(KThreadName, "WorkThrd");
378 for(data.iIteration=0;data.iIteration<KTestIterCount;++data.iIteration)
380 PrintIterationCount(data.iIteration);
381 //Run the malicious client (one worker theread which will try to crash the FeatMgr server)
383 TEST2(thread.Create(KThreadName, &ThreadFunc1, 0x2000, 0x1000, 0x10000, &data, EOwnerProcess), KErrNone);
384 TRequestStatus status;
385 thread.Logon(status);
386 TEST2(status.Int(), KRequestPending);
388 User::WaitForRequest(status);
389 User::SetJustInTime(ETrue); // enable debugger panic handling
390 if(thread.ExitType() == EExitPanic)
392 if(thread.ExitReason() == KPanicCode)
394 TheTest.Printf(_L("##Server terminated!\r\n"));
395 TheTest.Printf(_L("##Iteration=%d, Function(hex)=%X, Handle=%d\r\n"), data.iIteration, data.iFunction);
396 for(TInt i=0;i<KMaxMessageArguments;++i)
398 switch(data.iArgType[i])
401 TheTest.Printf(_L("##Arg %d, Integer, value=%d\r\n"), i, data.iIntArg[i]);
404 TheTest.Printf(_L("##Arg %d, Text, length=%d\r\n"), i, data.iTextArg[i].Length());
407 TheTest.Printf(_L("##Arg %d, Binary, length=%d\r\n"), i, data.iBinArg[i].Length());
410 TheTest.Printf(_L("##Arg %d, Invalid argument type: %d\r\n"), i, data.iArgType[i]);
419 User::SetJustInTime(ETrue); // enable debugger panic handling
425 //This test won't pass
426 TheTest.Start(_L("@SYMTestCaseID:PDS-EFM-CT-4065 Bad client test"));
434 CTrapCleanup* tc = CTrapCleanup::New();
439 TRAPD(err, DoTestsL());
441 TEST2(err, KErrNone);
450 User::Heap().Check();