os/ossrv/lowlevellibsandfws/pluginfw/Framework/DiscovererTest/t_discoverer.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 2004-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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // This file contains test classes and their implementations
    15 // to test production class CDiscoverer. Where necessary stubs
    16 // are implemented to help in writing test harness using RTest.
    17 // 
    18 //
    19 
    20 #include "Discoverer.h"
    21 #include "EComErrorCodes.h"
    22 #include "ImplementationInformation.h"
    23 #include "DiscovererObserver.h"
    24 #include "../EcomTestUtils/EcomTestUtils.h"
    25 #include "EComUidCodes.h"
    26 #include "RegistryData.h"
    27 #include "DriveInfo.h"
    28 
    29 #include <ecom/ecompanics.h>
    30 #include <e32test.h>
    31 #include <bautils.h>
    32 #include <s32file.h>
    33 #include <barsread2.h>
    34 #include <e32uid.h>
    35 #include <startup.hrh>
    36 #include "baspi.h"
    37 #include <sacls.h>
    38 
    39 
    40 //New RSC based plugins
    41 _LIT(KNewResourceFileName,		 		"C:\\resource\\plugins\\EComExample5.rsc");
    42 _LIT(KNewExampleDllFileName,	 		"C:\\sys\\bin\\EComExample5.dll");
    43 _LIT(KNewResourceFileNameOnZ,   		"z:\\RAMOnly\\EComExample5.rsc");
    44 _LIT(KNewExampleDllFileNameOnZ, 		"z:\\RAMOnly\\EComExample5.dll");
    45 _LIT(KNewExampleDllFileNameOnly,		"EComExample5.dll");
    46 _LIT(KNewResourceSearchPath,			"\\resource\\plugins\\*");
    47 //PlugIn3 RSC based plugins
    48 _LIT(KPlugIn3ResourceFileName,		 		"C:\\resource\\plugins\\EComExample12.rsc");
    49 _LIT(KPlugIn3ExampleDllFileName,	 		"C:\\sys\\bin\\EComExample12.dll");
    50 _LIT(KPlugIn3ResourceFileNameOnZ,   		"z:\\RAMOnly\\EComExample12.rsc");
    51 _LIT(KPlugIn3ExampleDllFileNameOnZ, 		"z:\\RAMOnly\\EComExample12.dll");
    52 _LIT(KPlugIn3ExampleDllFileNameOnly,		"EComExample12.dll");
    53 
    54 //Old DLL based plugins
    55 
    56 // spi test file
    57 _LIT(KEComSpiTestFilePathAndName, "Z:\\Test\\Data\\EcomTest.spi");
    58 _LIT(KEComSpiTestFilePlugIn3PathAndName, "Z:\\Test\\Data\\EcomTestType3.spi");
    59 // file name detremined from example spi file
    60 _LIT(KEComSpiTestFileDllPathAndName, "Z:\\sys\\bin\\EComExample5.dll");
    61 _LIT(KEComSpiTestFilePlugIn3DllPathAndName, "Z:\\sys\\bin\\EComExample12.dll");
    62 
    63 // spi file
    64 _LIT(KEComSPIFilePathAndName, "Z:\\private\\10009D8F\\Ecom.spi");
    65 
    66 const TInt KOneSecond = 1000000;
    67 
    68 LOCAL_D RFs	TheFs;
    69 
    70 LOCAL_D RTest test(_L("t_discoverer.exe"));
    71 
    72 LOCAL_D CTrapCleanup* 	  TheTrapCleanup 	 = NULL;
    73 
    74 LOCAL_D CActiveScheduler* TheActiveScheduler = NULL;
    75 
    76 // Used for supressing warning in OOM tests
    77 #define __UNUSED_VAR(var) var = var
    78 
    79 // Used for OOM test
    80 #define TEST_OOM_ERR if(err == KErrNoMemory) User::Leave(err)
    81 
    82 /**
    83 Stub classes are provided to act as replacement members for the class
    84 under test. This ensures the test class is constructed correctly and its
    85 behaviour is also correct within the context of the test. It allows the
    86 class to be tested in isolation. Other tests are available that test the
    87 interoperability of the class
    88 */
    89 class CObserverStub : public CBase, public MDiscovererObserver
    90 {
    91 public:
    92 	enum OSState
    93 		{
    94 		OSS_Undefined,
    95 		OSS_NoPlugins,
    96 		OSS_CriticalPlugins,
    97 		OSS_DiscoverNonCriticalPlugins,
    98 		OSS_AllPlugins
    99 		};
   100 	static CObserverStub* NewL();
   101 	virtual ~CObserverStub();
   102 
   103 	// Inherited from MDiscovererObserver
   104 	virtual void DiscoveriesBegin();
   105 	virtual void RegisterDiscoveryL(const TDriveName& aDrive,CPluginBase*& aDirEntry,TBool aDatFileExists);
   106 	virtual void DiscoveriesComplete(TBool aSuccessful, TPluginProcessingTypeIdentifier aProcessingType);
   107 	virtual void DriveRemovedL(TDriveUnit aDrive);
   108 	virtual void DriveReinstatedL(TDriveUnit aDrive);
   109 	virtual void DriveIsSpiBasedL(const TDriveUnit& /*aDrive*/){}
   110 	virtual TBool NotifiedWithErrorCode(TInt aError);
   111 	virtual TBool IsAnyDllRegisteredWithDriveL(const TDriveUnit aDrive) const;
   112 	virtual void SetDiscoveryFlagL(const TDriveUnit &aDrive);
   113 	virtual void LanguageChangedL(TBool& aLanguageChanged);
   114 
   115 
   116 	// Result verifing methods
   117 	TBool IsDriveMounted(const TDriveUnit aDrive);
   118 	TBool IsEntryProcessed(CPluginBase*& aDirEntry);
   119 	//TBool IsSpiEntryProcessed(const TEntry& aEntry);
   120 	TBool IsDiscoveriesCompleteSuccessfully();
   121 	void SetDiscoverer(CDiscoverer* aDiscoverer);
   122 	OSState GetState();
   123 
   124 private:
   125 	CObserverStub();
   126 
   127 private:
   128 	/** Base class ecom entry */
   129 	CPluginBase* iEntryBase;
   130 
   131 	/** Information of the dll which contains interface implementations */
   132 	//TEComEntryDetails		iEntry;
   133 
   134 	/** Information of the spi file which contains interface implementations */
   135 	//TEComSpiResourceEntryDetails		iSpiEntry;
   136 
   137 	/** iDriveUnit is drive mounted */
   138 	TDriveUnit	iDriveUnit;
   139 
   140 	/** Flag to indicate whether iDriveUnit is mounted or not */
   141 	TBool		iDriveMounted;
   142 
   143 	/** Flag to indicate whether iEntry is processed or not */
   144 	TBool		iEntryProcessed;
   145 
   146 	/** Flag to indicate whether iSpiEntry is processed or not */
   147 	TBool		iSpiEntryProcessed;
   148 
   149 	/** Flag to indicate whether Discovery process is Complete or not */
   150 	TBool		iDiscoveriesComplete;
   151 
   152 	/** The current start-up state of this object */
   153 	OSState     iState;
   154 
   155 	/** A reference to the CDiscoverer object that this class is observing*/
   156 	CDiscoverer* iDiscoverer;
   157 };  // End of CObserverStub declaration
   158 
   159 
   160 /**
   161 The TDiscoverer_StateAccessor class allows to access the private and protected
   162 members of production code class CDiscoverer, as its a friend class.
   163 */
   164 class TDiscoverer_StateAccessor
   165 	{
   166 public:
   167 
   168 	// Auxiliary functions that provide access to
   169 	// CDiscoverer private/protected members
   170 	void ScanDirectoryL(CDiscoverer& aDiscoverer, TDriveUnit aDriveUnit);
   171 	void ScanDirectoryCancel(CDiscoverer& aDiscoverer);
   172 	void CompleteNotificationProcessing(CDiscoverer& aDiscoverer);
   173 	void ValidateEntryL(CDiscoverer& aDiscoverer,
   174 						const TEntry& aEntry,
   175 						const TDriveName& aDriveName,
   176 						CPluginBase*& aEntryToFill,
   177 						TBool aIsRO);
   178 	void ValidateEntryL(CDiscoverer& aDiscoverer,
   179 						RResourceArchive& aRscArchive,
   180 						CPluginBase*& aEntryToFill);
   181 	void ProcessEntryL(CDiscoverer& aDiscoverer,const TDriveName& aDrive, CPluginBase*& aEntry, TBool aDatFileExists);
   182 	void DriveMountedL(CDiscoverer& aDiscoverer, const TDriveUnit aDrive);
   183 	void DriveUnmountedL(CDiscoverer& aDiscoverer, const TDriveUnit aDrive);
   184 	void SwiChangeNotificationL(CDiscoverer& aDiscoverer, TInt aSwiOperation);
   185 	void IdleScanningTimerRunErrorL(CDiscoverer& aDiscoverer, TInt aError);
   186 	void DirChangeNotifierRunErrorL(CDiscoverer& aDiscoverer, TInt aError);
   187 	void SwiChangeNotifierRunError(CDiscoverer& aDiscoverer, TInt aError);
   188 
   189 	// Verification functions to check the state of CDiscoverer object against tests
   190 	TBool IsDriveMounted(CDiscoverer& aDiscoverer, const TDriveUnit aDrive);
   191 	TBool IsDiscovererActive(CDiscoverer& aDiscoverer);
   192 	TBool IsDirectoryScanCancelled(CDiscoverer& aDiscoverer);
   193 	TBool IsScanDirectoryComplete(CDiscoverer& aDiscoverer);
   194 	TBool IsScanDirectoryPending(CDiscoverer& aDiscoverer, TDriveUnit aDrive);
   195 	TUint PendingDriveListCount(CDiscoverer& aDiscoverer);
   196 	CDiscoverer::TDiscovererState State(CDiscoverer& aDiscoverer);
   197 	CObserverStub::OSState GetDiscovererObserverState(CDiscoverer& aDiscoverer);
   198 	void ScanDriveL(CDiscoverer& aDiscoverer, TDriveUnit aDrive,  TBool aIsRO);
   199 	void LanguageChangedL(CDiscoverer& aDiscoverer);
   200 	};
   201 
   202 /**
   203 Scans plugin directories
   204 
   205 @param		aDiscoverer The CDiscoverer class object under test
   206 @return		void
   207 */
   208 void TDiscoverer_StateAccessor::ScanDirectoryL(CDiscoverer& aDiscoverer, TDriveUnit aDriveUnit)
   209 	{
   210 	aDiscoverer.RediscoveryScanDirectoryL(aDriveUnit);
   211 	}
   212 
   213 /**
   214 Stops scanning of the plugin directories
   215 
   216 @param		aDiscoverer The CDiscoverer class object under test
   217 */
   218 void TDiscoverer_StateAccessor::ScanDirectoryCancel(CDiscoverer& aDiscoverer)
   219 	{
   220 	aDiscoverer.ScanDirectoryCancel();
   221 	}
   222 
   223 /**
   224 Signals that the directory change has been fully processed.
   225 
   226 @param		aDiscoverer The CDiscoverer class object under test
   227 */
   228 void TDiscoverer_StateAccessor::CompleteNotificationProcessing(CDiscoverer& aDiscoverer)
   229 	{
   230 	aDiscoverer.CompleteNotificationProcessing();
   231 	}
   232 
   233 /**
   234 Verifies that the discovered entry is valid.
   235 Used to test the Interface Implementation Collection entry
   236 
   237 @param		aDiscoverer The CDiscoverer class object under test
   238 @param		aEntry The plugin name
   239 @param		aDriveName The drive containing the entry
   240 @param		aEntryToFill On return points to complete
   241 			plugin file name(with path) i.e. aPath + aEntry
   242 */
   243 void TDiscoverer_StateAccessor::ValidateEntryL(CDiscoverer& aDiscoverer,
   244 											   const TEntry& aEntry,
   245 											   const TDriveName& aDriveName,
   246 											   CPluginBase*& aEntryToFill,
   247 											   TBool aIsRO)
   248 	{
   249 	aDiscoverer.ValidateEntryL(aEntry, aDriveName, aEntryToFill, aIsRO);
   250 	}
   251 
   252 /**
   253 Verifies that the discovered entry is valid.
   254 Used to test the Interface Implementation Collection entry
   255 
   256 @param		aDiscoverer The CDiscoverer class object under test
   257 @param		aRscArchive reference to the resource archive
   258 @param		aEntryToFill On return points to complete
   259 			plugin file name(with path) i.e. aPath + aEntry
   260 */
   261 void TDiscoverer_StateAccessor::ValidateEntryL(CDiscoverer& aDiscoverer,
   262 											RResourceArchive& aRscArchive,
   263 											CPluginBase*& aEntryToFill)
   264 	{
   265 	aDiscoverer.ValidateEntryL(aRscArchive,aEntryToFill);
   266 	}
   267 /**
   268 Registers an Interface Implementation Collection
   269 
   270 @param		aDiscoverer The CDiscoverer class object under test
   271 @param		aEntry This is the plugin name that needs to be registered
   272 */
   273 void TDiscoverer_StateAccessor::ProcessEntryL(CDiscoverer& aDiscoverer,const TDriveName& aDrive,
   274 										  CPluginBase*& aEntry, TBool aDatFileExists)
   275 	{
   276 	aDiscoverer.ProcessEntryL(aDrive,aEntry,aDatFileExists);
   277 	}
   278 
   279 /**
   280 Notifies the discoverer of an SWI operation change
   281 
   282 @param		aDiscoverer The CDiscoverer class object under test
   283 @param		aSwiOperation This new SWI state
   284 */
   285 void TDiscoverer_StateAccessor::SwiChangeNotificationL(CDiscoverer& aDiscoverer, TInt aSwiOperation)
   286 	{
   287 	aDiscoverer.SwiChangeNotificationL(aSwiOperation);
   288 	}
   289 
   290 /**
   291 Signals that a drive is available
   292 
   293 @param		aDiscoverer The CDiscoverer class object under test
   294 @param		aDrive Drive that needs to be mounted
   295 */
   296 void TDiscoverer_StateAccessor::DriveMountedL(CDiscoverer& aDiscoverer,
   297 											  const TDriveUnit aDrive)
   298 	{
   299 	aDiscoverer.DriveMountedL(aDrive);
   300 	}
   301 
   302 /**
   303 Signals that a drive is unavailable
   304 
   305 @param		aDiscoverer The CDiscoverer class object under test
   306 @param		aDrive Drive that needs to be unmounted/removed
   307 */
   308 void TDiscoverer_StateAccessor::DriveUnmountedL(CDiscoverer& aDiscoverer,
   309 												const TDriveUnit aDrive)
   310 	{
   311 	aDiscoverer.DriveUnmountedL(aDrive);
   312 	}
   313 
   314 /**
   315 Checks whether CDiscoverer object has successfully completed with
   316 the scanning of the plugin directories on all drives
   317 
   318 @param		aDiscoverer The CDiscoverer class object under test
   319 @return		true if notification has been processed on all drives.
   320 			false if there is notification processing pending on any drive
   321 */
   322 TBool TDiscoverer_StateAccessor::IsScanDirectoryComplete(CDiscoverer& aDiscoverer)
   323 	{
   324 	// check state of discoverer to see if all pending drives have been scanned.
   325 	return(aDiscoverer.State() == CDiscoverer::EDisc_AllPluginsDisc);
   326 	}
   327 
   328 /**
   329 Checks whether a notification has been signaled but not processed on specified drive.
   330 
   331 Error Condition	: Leaves with KErrNotFound if the drive is not registered.
   332 @param		aDiscoverer The CDiscoverer class object under test
   333 @param		aDrive The drive to check
   334 @return		true if notification has been processed on the specified drive.
   335 			false if notification processing is pending
   336 */
   337 TBool TDiscoverer_StateAccessor::IsScanDirectoryPending(CDiscoverer& aDiscoverer, TDriveUnit aDrive)
   338 	{
   339 	if(aDiscoverer.iScanningTimer->iPendingDriveList.Find(aDrive)!=KErrNotFound)
   340 		{
   341 		return ETrue;
   342 		}
   343 	else
   344 		{
   345 		return EFalse;
   346 		}
   347 	}
   348 
   349 /**
   350 Checks the current count of drives in the list waiting for processing.
   351 
   352 @param		aDiscoverer The CDiscoverer class object under test
   353 @return		TUint
   354 */
   355 TUint TDiscoverer_StateAccessor::PendingDriveListCount(CDiscoverer& aDiscoverer)
   356 	{
   357 	return(aDiscoverer.iScanningTimer->iPendingDriveList.Count());
   358 	}
   359 
   360 /**
   361 Checks that CDiscoverer object is not scanning the plugin directories.
   362 
   363 @param		aDiscoverer The CDiscoverer class object under test
   364 @return		true if plugin dirctory scanning is stopped.
   365 */
   366 TBool TDiscoverer_StateAccessor::IsDirectoryScanCancelled(CDiscoverer& aDiscoverer)
   367 	{
   368 	if(!aDiscoverer.iDirScanner)
   369 		{
   370 		return ETrue;
   371 		}
   372 	return EFalse;
   373 	}
   374 
   375 /**
   376 Checks that CDiscoverer object is currently activated for the plugin dir scanning
   377 
   378 @param		aDiscoverer The CDiscoverer class object under test
   379 @return		true if Discoverer object is tracking changes in the plugin directories in each drive.
   380 */
   381 TBool TDiscoverer_StateAccessor::IsDiscovererActive(CDiscoverer& aDiscoverer)
   382 	{
   383 	//Return ETrue if both iScanningTimer & directory notifier(s) are active
   384 	TBool isActive = ETrue;
   385 
   386 	  for(TInt index = 0; index <aDiscoverer.iRscDirNotifierList.Count(); index++)
   387 	  	{
   388 	  	if(!(aDiscoverer.iRscDirNotifierList[index]&& aDiscoverer.iRscDirNotifierList[index]->IsActive()))
   389 			{
   390 			// iRscDirNotifierList should be active, as each active object on list is constantly
   391 			// looking for changes in the plugin directories in each drive.
   392 			isActive = EFalse;
   393 			break;
   394 			}
   395 		}
   396 	return isActive;
   397 	}
   398 
   399 /**
   400 Verifies whether given drive is registered
   401 
   402 @param		aDiscoverer The CDiscoverer class object under test
   403 @param		aDrive Drive that needs to be verified for registration
   404 @return		return true if given drive is installed
   405 */
   406 TBool TDiscoverer_StateAccessor::IsDriveMounted(CDiscoverer& aDiscoverer,
   407 												const TDriveUnit aDrive)
   408 	{
   409 	TBool isMounted = ETrue;
   410 	if(aDiscoverer.iDrivesDiscovered.Find(aDrive) == KErrNotFound)
   411 		{
   412 		isMounted = EFalse;
   413 		}
   414 	return isMounted;
   415 	}
   416 
   417 /**
   418 Scans plugin directories
   419 
   420 @param		aDiscoverer The CDiscoverer class object under test
   421 @param 		aDrive	The drive which is scanned
   422 @param		aIsRO	Whether the drive is ready-only
   423 @return		true if plugin directorie in each drive get scaned successfully
   424 */
   425 void TDiscoverer_StateAccessor::ScanDriveL(CDiscoverer& aDiscoverer, TDriveUnit aDrive,  TBool aIsRO)
   426 	{
   427 	return aDiscoverer.iDirScanner->ScanDriveL(aDrive, aIsRO);
   428 	}
   429 void TDiscoverer_StateAccessor::LanguageChangedL(CDiscoverer& aDiscoverer)
   430 	{
   431 	aDiscoverer.LanguageChangeNotificationL();
   432 	}
   433 /**
   434 Retrieve the object's current state.
   435 
   436 @param		aDiscoverer The CDiscoverer class object under test
   437 @return 	TDiscovererState the current state of the CDiscoverer
   438 class object under test
   439 */
   440 CDiscoverer::TDiscovererState TDiscoverer_StateAccessor::State(CDiscoverer& aDiscoverer)
   441 	{
   442 	return aDiscoverer.State();
   443 	}
   444 
   445 CObserverStub::OSState TDiscoverer_StateAccessor::GetDiscovererObserverState(CDiscoverer& aDiscoverer)
   446 	{
   447 	CObserverStub* discovererObserver =
   448 		static_cast<CObserverStub*>(&aDiscoverer.iDiscovererObserver);
   449 	return discovererObserver->GetState();
   450 	}
   451 
   452 /**
   453 Call the CIdleScanningTimer RunError funtion
   454 
   455 @param		aDiscoverer The CDiscoverer class object under test
   456 @param 		aError The error code to pass to the RunError function
   457 */
   458 void TDiscoverer_StateAccessor::IdleScanningTimerRunErrorL(CDiscoverer& aDiscoverer, TInt aError)
   459 	{
   460 	aDiscoverer.iScanningTimer->RunError(aError);
   461 	}
   462 
   463 /**
   464 Call the CDirChangeNotifier RunError funtion
   465 
   466 @param		aDiscoverer The CDiscoverer class object under test
   467 @param 		aError The error code to pass to the RunError function
   468 */
   469 void TDiscoverer_StateAccessor::DirChangeNotifierRunErrorL(CDiscoverer& aDiscoverer, TInt aError)
   470 	{
   471 	aDiscoverer.iRscDirNotifierList[0]->RunError(aError);
   472 	}
   473 
   474 /**
   475 Call the CSwiChangeNotifier RunError funtion
   476 
   477 @param		aDiscoverer The CDiscoverer class object under test
   478 @param 		aError The error code to pass to the RunError function
   479 */
   480 void TDiscoverer_StateAccessor::SwiChangeNotifierRunError(CDiscoverer& aDiscoverer, TInt aError)
   481 	{
   482 	aDiscoverer.iSwiChangeNotifier->RunError(aError);
   483 	}
   484 
   485 /**
   486 Creates an instance of CObserverStub class.
   487 
   488 @return		The new'ed object.
   489 */
   490 CObserverStub* CObserverStub::NewL()
   491 	{
   492 	return new(ELeave) CObserverStub();
   493 	}
   494 
   495 /**
   496 Destructor
   497 
   498 @post		This object is properly destroyed.
   499 */
   500 CObserverStub::~CObserverStub()
   501 	{
   502 	// do nothing;
   503 	}
   504 
   505 /**
   506 Default constructor.
   507 
   508 @post		The object is properly constructed.
   509 */
   510 CObserverStub::CObserverStub()
   511 				:CBase(),
   512 				iDriveUnit(EDriveC),
   513 				iDriveMounted(EFalse),
   514 				iEntryProcessed(EFalse),
   515 				iSpiEntryProcessed(EFalse),
   516 				iDiscoveriesComplete(EFalse),
   517 				iState(OSS_NoPlugins)
   518 	{
   519 	// do nothing;
   520 	}
   521 
   522 /**
   523 MDiscovererObserver callback method, to signal that a discovery
   524 session is starting.
   525 */
   526 void CObserverStub::DiscoveriesBegin()
   527 	{
   528 	iDiscoveriesComplete = EFalse;
   529 	}
   530 
   531 void CObserverStub::SetDiscoveryFlagL(const TDriveUnit & /*aDrive*/)
   532 	{
   533 	iDiscoveriesComplete = ETrue;
   534 	}
   535 /**
   536 MDiscovererObserver callback method to register an Ecom Spi file.
   537 
   538 @param		aEntry The Ecom file  that needs to be registered
   539 */
   540 void CObserverStub::RegisterDiscoveryL(const TDriveName& /* aDriveName */,CPluginBase*& aEntry,TBool /* aDatFileExists*/)
   541 	{
   542 	iEntryBase = aEntry;
   543 	iSpiEntryProcessed = ETrue;
   544 	}
   545 
   546 /**
   547 MDiscovererObserver callback method, to signal that a discovery
   548 session is complete.
   549 
   550 @param		aSuccessful Indicates discoveries process completed successfully or not
   551 @param		aProcessingType indicates the type of processing for plugins
   552 			for ensuring that plugins are not processed multiple times
   553 			during start-up phase
   554 */
   555 void CObserverStub::DiscoveriesComplete(TBool aSuccessful, TPluginProcessingTypeIdentifier aProcessingType)
   556 	{
   557 	iDiscoveriesComplete = aSuccessful;
   558 
   559 	switch(iState)
   560 		{
   561 		case OSS_NoPlugins:
   562 			if(aProcessingType == EPluginProcessingTypeCriticalOnly &&
   563 			   iDiscoverer->State() == CDiscoverer::EDisc_CriticalPluginsDisc)
   564 				{
   565 				iState = OSS_CriticalPlugins;
   566 				}
   567 			else if(aProcessingType == EPluginProcessingTypeAll &&
   568 				    iDiscoverer->State() == CDiscoverer::EDisc_AllPluginsDisc)
   569 				{
   570 				iState = OSS_AllPlugins;
   571 				}
   572 			break;
   573 		case OSS_CriticalPlugins:
   574 			if(aProcessingType == EPluginProcessingTypeNonCriticalOnly &&
   575 			   iDiscoverer->State() == CDiscoverer::EDisc_AllPluginsDisc)
   576 				{
   577 				iState = OSS_AllPlugins;
   578 				}
   579 			break;
   580 		}
   581 	}
   582 
   583 /**
   584 Verifies whether given plugin entry is registered
   585 
   586 @param		aDirEntry The Ecom plugin that to be checked for registration.
   587 @return		true if given plugin is registered
   588 */
   589 TBool CObserverStub::IsEntryProcessed(CPluginBase*& aDirEntry)
   590 	{
   591 	if(iEntryProcessed && (iEntryBase->iDllThirdUid == aDirEntry->iDllThirdUid))
   592 		{
   593 		return ETrue;
   594 		}
   595 
   596 	return EFalse;
   597 	}
   598 
   599 /**
   600 MDiscovererObserver callback method, to signal that a Drive
   601 is removed/dismounted.
   602 
   603 @param		aDrive The drive that is removed.
   604 */
   605 void CObserverStub::DriveRemovedL(TDriveUnit aDrive)
   606 	{
   607 	iDriveMounted = EFalse;
   608 	iDriveUnit = aDrive;
   609 	}
   610 
   611 /**
   612 MDiscovererObserver callback method, to signal that a Drive
   613 is reinstalled.
   614 
   615 @param		aDrive The drive that is available now.
   616 */
   617 void CObserverStub::DriveReinstatedL(TDriveUnit aDrive)
   618 	{
   619 	iDriveMounted = ETrue;
   620 	iDriveUnit = aDrive;
   621 	}
   622 
   623 /**
   624 MDiscovererObserver callback method, to signal that
   625 during a discovery session an error has occured.
   626 
   627 @param		aError The notification error code.
   628 @return		true if aError is one of the acceptable error codes
   629 */
   630 TBool CObserverStub::NotifiedWithErrorCode(TInt aError)
   631 	{
   632 	// Test the safe error codes
   633 	return (aError == KErrNotReady ||		// Drive removed
   634 			aError == KErrPathNotFound);	// Directory deleted
   635 	}
   636 
   637 /**
   638 MDiscovererObserver callback method,to retrieve the drive unit's DAT
   639 file infor.
   640 
   641 @param 		aDrive the identifier of the drive to retrieve DAT file infor from.
   642 @return		ETrue if DAT file exists on the drive unit, otherwise EFlase.
   643 
   644 */
   645 TBool CObserverStub::IsAnyDllRegisteredWithDriveL(const TDriveUnit /*aDrive*/) const
   646 	{
   647 	// To pass the build always returns EFalse to mean no Dll is discovered in the drive,
   648 	// it is not used in this test.
   649 	return EFalse;
   650 	}
   651 
   652 /**
   653 Verification method that checks whether given drive is available
   654 
   655 @param		aDrive The drive that to be checked.
   656 @return		true if given drive is registered
   657 */
   658 TBool CObserverStub::IsDriveMounted(const TDriveUnit aDrive)
   659 	{
   660 	TBool mounted = EFalse;
   661 	if(iDriveMounted && (aDrive == iDriveUnit))
   662 		{
   663 		mounted = ETrue;
   664 		}
   665 	return mounted;
   666 	}
   667 
   668 /**
   669 Verification method that checks whether discovery process is completed successfully
   670 
   671 @return		true if discovery process is completed successfully
   672 */
   673 TBool CObserverStub::IsDiscoveriesCompleteSuccessfully()
   674 	{
   675 	return iDiscoveriesComplete;
   676 	}
   677 
   678 /**
   679 Returns the current start-up state of this object
   680 
   681 @return	    The current start-up state of this object
   682 */
   683 CObserverStub::OSState CObserverStub::GetState()
   684 	{
   685 	return iState;
   686 	}
   687 
   688 void CObserverStub::SetDiscoverer(CDiscoverer* aDiscoverer)
   689 	{
   690 	iDiscoverer = aDiscoverer;
   691 	}
   692 
   693 void CObserverStub::LanguageChangedL(TBool& aLanguageChanged)
   694 	{
   695 	aLanguageChanged = EFalse;
   696 	}
   697 class CDiscovererTestShutdown : public CTimer
   698 	{
   699 public:
   700 	inline CDiscovererTestShutdown();
   701 	inline void ConstructL();
   702 	inline void StartAfter(TTimeIntervalMicroSeconds32 aTimeInterval);
   703 
   704 private:
   705 	void RunL();
   706 	};
   707 
   708 inline CDiscovererTestShutdown::CDiscovererTestShutdown()
   709 :CTimer(EPriorityHigh)// Priority set higher than CIdleScanningTimer
   710 	{
   711 	CActiveScheduler::Add(this);
   712 	}
   713 
   714 inline void CDiscovererTestShutdown::ConstructL()
   715 	{
   716 	CTimer::ConstructL();
   717 	}
   718 
   719 inline void CDiscovererTestShutdown::StartAfter(TTimeIntervalMicroSeconds32 aTimeInterval)
   720 	{
   721 	After(aTimeInterval);
   722 	}
   723 void CDiscovererTestShutdown::RunL()
   724 	{
   725 	CActiveScheduler::Stop();
   726 	}
   727 
   728 /**
   729 Test class for object CDiscoverer.
   730 This class provides the parameters and behaviour that
   731 allows this class to behave normally under a test
   732 scenario.
   733 */
   734 class CDiscovererTest : public CBase
   735 	{
   736 public:
   737 	static CDiscovererTest* NewL();
   738 	virtual ~CDiscovererTest();
   739 
   740 	void ResumeSuspendTestL();
   741 	void DriveMountUnmountTestL();
   742 	void ProcessEntryTestL();
   743 	void ProcessSpiEntryTestL();
   744 	void ValidateSpiEntryTestL();
   745 	void ValidateSpiPluginsTestL();
   746 	void ValidateEntryTestL();
   747 	void ProcessEntryPlugIn3TestL();
   748 	void ProcessSpiEntryPlugIn3TestL();
   749 	void ValidateSpiEntryPlugIn3TestL();
   750 	void ValidateEntryPlugIn3TestL();
   751 	void ScanDirectoryIncrementTestL();
   752 	void ScanDirectoryTestL();
   753 	void ScanDirectoryCancelTestL();
   754 	void StagedDiscoveryStateTransitionTestL();
   755 	void AllAtOnceDiscoveryStateTransitionTestL();
   756 	void MultipleNotificationProcessingTestL();
   757 	void SWINotificationProcessingTestL();
   758 	void IdleScanningTimerRunErrorL();
   759 	void DirChangeNotifierRunErrorL();
   760 	void SwiChangeNotifierRunError();
   761 	void LanguageChangedNotificationTestL();
   762 
   763 private:
   764 	CDiscovererTest();
   765 	void ConstructL();
   766 
   767 public:
   768 	/** The instance of the class under test */
   769 	CDiscoverer* iDiscoverer;
   770 
   771 	/** The instance of the state accessor interface */
   772 	TDiscoverer_StateAccessor* iStateAccessor;
   773 
   774 	/** The instance of the observer stub of the CDiscoverer */
   775 	CObserverStub* iDiscovererObserver;
   776 
   777 	/** Unique Id of the ECOM dll */
   778 	TUid		iDllUid;
   779 
   780 	/** The drive on which interface implementations can be found */
   781 	TDriveUnit	iDriveUnit;
   782 
   783 	/** Information on a dll which contains interface implementations */
   784 	TEntry		iDllEntry;
   785 
   786 	/** A shutdown timer to manipulate the Active Scheduler of testing server */
   787 	CDiscovererTestShutdown	iShutdown;
   788 
   789 	CEComCachedDriveInfo* iCachedDriveInfo;
   790 	};
   791 
   792 /**
   793 Standardised safe construction which
   794 leaves nothing on the cleanup stack.
   795 
   796 @post			CDiscovererTest is fully constructed and initialised
   797 @return			The new'ed object.
   798 */
   799 CDiscovererTest* CDiscovererTest::NewL()
   800 	{
   801 	CDiscovererTest* self = new (ELeave) CDiscovererTest();
   802 	CleanupStack::PushL( self );
   803 	self->ConstructL();
   804 	CleanupStack::Pop(self);
   805 	return self;
   806 	}
   807 
   808 /**
   809 Default constructor.
   810 
   811 @post	The object is properly constructed.
   812 */
   813 CDiscovererTest::CDiscovererTest()
   814 				:CBase(),
   815 				 iDriveUnit(EDriveC)
   816 	{
   817 	TUid uid1 = {0};
   818 	TUid uid2 = {0};
   819 
   820 	iDllUid.iUid	= 0x10009DB6; // Dll Uid for EComRomOnlyExampleOnC.dll
   821 	iDllEntry.iType = TUidType(uid1, uid2, iDllUid);
   822 	}
   823 
   824 /**
   825 Destructor.
   826 
   827 @post		This object is properly destroyed.
   828 */
   829 CDiscovererTest::~CDiscovererTest()
   830 	{
   831 	delete iDiscoverer;
   832 	delete iStateAccessor;
   833 	delete iDiscovererObserver;
   834 	delete iCachedDriveInfo;
   835 	}
   836 
   837 /**
   838 Standardized 2nd(Initialization) phase of two phase construction.
   839 Creates supporting class objects for the execution of test.
   840 
   841 @post			CDiscovererTest is fully constructed.
   842 */
   843 void CDiscovererTest::ConstructL()
   844 	{
   845 	iStateAccessor		= new(ELeave) TDiscoverer_StateAccessor;
   846 	iDiscovererObserver	= CObserverStub::NewL();
   847 	iDiscoverer			= CDiscoverer::NewL(*iDiscovererObserver, TheFs);
   848 	iDiscovererObserver->SetDiscoverer(iDiscoverer);
   849 	iShutdown.ConstructL();
   850 	iCachedDriveInfo = CEComCachedDriveInfo::NewL(TheFs);
   851 	}
   852 
   853 /**
   854 The test executes by first suspending the discoverer then by
   855 resuming it.
   856 
   857 @SYMTestCaseID          SYSLIB-ECOM-CT-0716
   858 @SYMTestCaseDesc	    Tests for resuming and suspending Discoverer
   859 @SYMTestPriority 	    High
   860 @SYMTestActions  	    Suspend if Discoverer is active and then resume. Check for the error conditions(OOM)
   861 @SYMTestExpectedResults The test must not fail.
   862 @SYMREQ                 REQ0000
   863 */
   864 void CDiscovererTest::ResumeSuspendTestL()
   865 	{
   866 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0716 "));
   867 	// Perform an initial discovery and start the notifiers.
   868 	iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical);
   869 
   870 	TBool isDiscActive = iStateAccessor->IsDiscovererActive(*iDiscoverer);
   871 	// Notifiers should be active.
   872 	test(isDiscActive);
   873 
   874 	TInt err = iDiscoverer->Suspend();
   875 	TEST_OOM_ERR;
   876 	test(err == KErrNone);
   877 
   878 	// Notifiers should still be active.
   879 	isDiscActive = iStateAccessor->IsDiscovererActive(*iDiscoverer);
   880 	test(isDiscActive);
   881 
   882 	// Test that there are no drives pending a scan before we copy the plugin.
   883 	test(!iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveC));
   884 
   885 	// Raise a plugin change notification on C: drive. This simulates a plugin being copied.
   886 	iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveC);
   887 	iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveC);
   888 
   889 	// Test that the drive we copied the plugin to still has a scan pending.
   890 	test(iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveC));
   891 
   892 	isDiscActive = iStateAccessor->IsDiscovererActive(*iDiscoverer);
   893 	// Notifiers should still be active.
   894 	test(isDiscActive);
   895 
   896 	// Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process
   897 	// all pending notification.
   898 	iShutdown.StartAfter(KOneSecond * 3);
   899 	CActiveScheduler::Start();
   900 
   901 	// Check that notifications have not been processed.
   902 	test(!iStateAccessor->IsScanDirectoryComplete(*iDiscoverer));
   903 
   904 	err = iDiscoverer->Resume();
   905 	TEST_OOM_ERR;
   906 	test(err == KErrNone);
   907 
   908 	isDiscActive = iStateAccessor->IsDiscovererActive(*iDiscoverer);
   909 	// Notifiers should still be active.
   910 	test(isDiscActive);
   911 
   912 	// Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process
   913 	// all pending notification.
   914 	iShutdown.StartAfter(KOneSecond * 3);
   915 	CActiveScheduler::Start();
   916 
   917 	// Check if notifications have been processed on all drives.
   918 	test(iStateAccessor->IsScanDirectoryComplete(*iDiscoverer));
   919 	}
   920 
   921 /**
   922 The test executes by first making the test drive unmounted and
   923 then testing for Mount and Unmount.
   924 
   925 @SYMTestCaseID          SYSLIB-ECOM-CT-0717
   926 @SYMTestCaseDesc	    Tests for drive mount and unmount
   927 @SYMTestPriority 	    High
   928 @SYMTestActions  	    First remove the drive if its already there, then test for Mount then for unmount
   929 @SYMTestExpectedResults The test must not fail.
   930 @SYMREQ                 REQ0000
   931 */
   932 void CDiscovererTest::DriveMountUnmountTestL()
   933 	{
   934 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0717 "));
   935 	TInt err;
   936 	TDriveUnit drive(EDriveC);
   937 	//First remove the drive if its already there, then test for Mount then for unmount
   938 	if(iStateAccessor->IsDriveMounted(*iDiscoverer, drive))
   939 		{
   940 		TRAP(err, iStateAccessor->DriveUnmountedL(*iDiscoverer, drive));
   941 		TEST_OOM_ERR;
   942 		test(err == KErrNone);
   943 		}
   944 
   945 	// Drive should not be present
   946 	test(!iStateAccessor->IsDriveMounted(*iDiscoverer, drive));
   947 
   948 	// Test Mount
   949 	TRAP(err, iStateAccessor->DriveMountedL(*iDiscoverer, drive));
   950 	TEST_OOM_ERR;
   951 	test(err == KErrNone);
   952 
   953 	// CDiscoverer->DriveMountedL/DriveUnmountedL updates both itself and
   954 	// DiscovererObserver class for Mount/Unmount of drives. So testing for both
   955 	test(iStateAccessor->IsDriveMounted(*iDiscoverer, drive));
   956 	test(iDiscovererObserver->IsDriveMounted(drive));
   957 
   958 	// Test Unmount
   959 	TRAP(err, iStateAccessor->DriveUnmountedL(*iDiscoverer, drive));
   960 	TEST_OOM_ERR;
   961 	test(err == KErrNone );
   962 
   963 	// CDiscoverer->DriveMountedL/DriveUnmountedL updates both itself and
   964 	// DiscovererObserver class for Mount/Unmount of drives. So testing for both
   965 	test(!iStateAccessor->IsDriveMounted(*iDiscoverer, drive));
   966 	test(!iDiscovererObserver->IsDriveMounted(drive));
   967 	}
   968 
   969 
   970 /**
   971 @SYMTestCaseID          SYSLIB-ECOM-UT-3559
   972 @SYMTestCaseDesc	    Tests for process entry of PLUGIN3 type.
   973 @SYMTestPriority 	    High
   974 @SYMTestActions  	    Call CDiscoverer::ProcessEntryL().
   975 @SYMTestExpectedResults The entry is registered successfully and no leave occurred.
   976 @SYMEC	                EC43
   977 */
   978 void CDiscovererTest::ProcessEntryPlugIn3TestL()
   979 	{
   980 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-3559 "));
   981 	CPluginBase* entryBase = NULL;
   982 	TInt err = KErrNone;
   983 
   984 	//This part refers to testing of the entry containing the RSC
   985 	_LIT(KExampleRscFileNameOnly,"EComExample12.rsc");
   986 	TDriveName driveName1(iDriveUnit.Name());
   987 	TParse path1;
   988 	path1.Set(KNewResourceSearchPath(),NULL,&driveName1);
   989 	//Creating a entry that represents the Rsc disovered during scanning of \\resource\\plugins
   990 	TEntry rscEntry;
   991 	rscEntry.iName = KExampleRscFileNameOnly;
   992 	rscEntry.iType = TUidType(KNullUid,KUidInterfaceImplementationCollectionInfo,KNullUid);
   993 	entryBase = CSecurePlugin::NewL(TheFs,rscEntry,driveName1, EFalse);
   994 	CleanupStack::PushL(entryBase);
   995 
   996 	// ProcessEntryL() updates entryToFill with information on a rsc specified by the other params.
   997 	TRAP(err, iStateAccessor->ProcessEntryL(*iDiscoverer,driveName1,entryBase,ETrue));
   998 	TEST_OOM_ERR;
   999 	test(err == KErrNone);
  1000 	CleanupStack::PopAndDestroy(entryBase);
  1001 	}
  1002 
  1003 /**
  1004 @SYMTestCaseID			SYSLIB-ECOM-UT-3560
  1005 @SYMTestCaseDesc 		Check that the ProcessEntryL for spi data works correctly when using PLUGIN3 entries in the spi file.
  1006 @SYMTestPriority 		High
  1007 @SYMTestActions  		Call CDiscoverer::ProcessEntryL().
  1008 @SYMTestExpectedResults ecomtesttype3.spi is processed successfully and no leave occurred.
  1009 @SYMEC					EC43
  1010 */
  1011 void CDiscovererTest::ProcessSpiEntryPlugIn3TestL()
  1012 	{
  1013 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-3560 "));
  1014 	TEntry spiEntry;
  1015 	test(TheFs.Entry(KEComSpiTestFilePlugIn3PathAndName, spiEntry) == KErrNone);
  1016 	TParse spiPath;
  1017 	spiPath.Set(KEComSpiTestFilePlugIn3PathAndName, NULL, NULL);
  1018 
  1019 	RResourceArchive resourceArchive;
  1020 	resourceArchive.OpenL(TheFs, KEComSpiTestFilePlugIn3PathAndName);
  1021 	CleanupClosePushL(resourceArchive);
  1022 
  1023 	// check SPI file type
  1024 	TUid type = resourceArchive.Type();
  1025 	test(type == KEcomSpiFileTypeUid);
  1026 
  1027 	// there is only 1 resource file
  1028 	while(!resourceArchive.End())
  1029 		{
  1030 		CPluginBase* entry = CSpiPlugin::NewL(resourceArchive);
  1031 		CleanupStack::PushL(entry);
  1032 
  1033 		TRAPD(err, iStateAccessor->ProcessEntryL(*iDiscoverer,spiPath.Drive(), entry,ETrue));
  1034 		TEST_OOM_ERR;
  1035 		test(err == KErrNone);
  1036 
  1037 		CleanupStack::PopAndDestroy(entry);
  1038 		entry = NULL;
  1039 		}
  1040 	CleanupStack::PopAndDestroy(&resourceArchive);
  1041 	}
  1042 /**
  1043 @SYMTestCaseID          SYSLIB-ECOM-UT-3561
  1044 @SYMTestCaseDesc	    Tests for the reference of the entry containing the DLL of PLUGIN3 type.
  1045 @SYMTestPriority 	    High
  1046 @SYMTestActions  	    Call CDiscoverer::ValidateEntryL(). Check the entry returned.
  1047 @SYMTestExpectedResults The entry is validated successfully and no leave occurred.
  1048 @SYMEC	                EC43
  1049 */
  1050 void CDiscovererTest::ValidateEntryPlugIn3TestL()
  1051 	{
  1052 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-3561 "));
  1053 	TInt err = KErrNone;
  1054 
  1055 	//This part refers to testing of the entry containing the RSC
  1056 	_LIT(KExampleRscFileNameOnly,"EComExample12.rsc");
  1057 	TDriveName driveName1(iDriveUnit.Name());
  1058 	CPluginBase *entryToFill1 = NULL;
  1059 	//Creating an entry that represents the Rsc disovered during scanning of \\resource\\plugins
  1060 	TEntry rscEntry;
  1061 	rscEntry.iName = KExampleRscFileNameOnly;
  1062 	rscEntry.iType = TUidType(KNullUid,KUidInterfaceImplementationCollectionInfo,KNullUid);
  1063 
  1064 	TUid dllUid1 = {KDynamicLibraryUidValue};
  1065 	TUid example12Uid = {0x10009E3E};
  1066 	TUidType dllUidType = TUidType(dllUid1,KUidInterfaceImplementationCollection,example12Uid);
  1067 
  1068 	//Now call ValidateEntryL() with this entry containg the rsc
  1069 	TRAP(err,iStateAccessor->ValidateEntryL(*iDiscoverer,rscEntry,driveName1,entryToFill1, EFalse));
  1070 	TEST_OOM_ERR;
  1071 	test(err == KErrNone);
  1072 	test(entryToFill1->iDllThirdUid == dllUidType[2]);
  1073 	TParse dllParse;
  1074 	dllParse.Set(KPlugIn3ExampleDllFileName,NULL,NULL);
  1075 	test(entryToFill1->iDllName->CompareF(dllParse.NameAndExt()) == 0);
  1076 
  1077 	delete entryToFill1;
  1078 	}
  1079 
  1080 /**
  1081 @SYMTestCaseID			SYSLIB-ECOM-UT-3562
  1082 @SYMTestCaseDesc 		Check that the ValidateEntryL for spi data works correctly when using PLUGIN3 entries in the spi file.
  1083 @SYMTestPriority 		High
  1084 @SYMTestActions  		Call CDiscoverer::ValidateEntryL(). Check the entry returned.
  1085 @SYMTestExpectedResults ecomtesttype3.spi file is validated successfully and no leave occurred.
  1086 @SYMEC					EC43
  1087 */
  1088 void CDiscovererTest::ValidateSpiEntryPlugIn3TestL()
  1089 	{
  1090 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-3562 "));
  1091 	TEntry spiEntry;
  1092 	test(TheFs.Entry(KEComSpiTestFilePlugIn3PathAndName, spiEntry) == KErrNone);
  1093 	TParse spiPath;
  1094 	spiPath.Set(KEComSpiTestFilePlugIn3PathAndName, NULL, NULL);
  1095 
  1096 	TEntry TestEntry;
  1097 	TUid uid1 = {0x10000079};
  1098 	TUid uid2 = {0x10009D93};
  1099 	TUid uid3 = {0x10009E3E}; // Uid for EComExample12.dll
  1100 	TestEntry.iType = TUidType(uid1, uid2, uid3);
  1101 	TestEntry.iName = KPlugIn3ExampleDllFileNameOnly;
  1102 
  1103 	RResourceArchive resourceArchive;
  1104 	resourceArchive.OpenL(TheFs, KEComSpiTestFilePlugIn3PathAndName);
  1105 	CleanupClosePushL(resourceArchive);
  1106 
  1107 	// check SPI file type
  1108 	TUid type = resourceArchive.Type();
  1109 	test(type == KEcomSpiFileTypeUid);
  1110 
  1111 	// there is only 1 resource file
  1112 	while(!resourceArchive.End())
  1113 		{
  1114 		CPluginBase* entryToFill = NULL;
  1115 		TRAPD(err, iStateAccessor->ValidateEntryL(*iDiscoverer,resourceArchive,entryToFill));
  1116 		CleanupStack::PushL(entryToFill);
  1117 
  1118 		TEST_OOM_ERR;
  1119 		test(err == KErrNone);
  1120 		test(entryToFill->iDllThirdUid == TestEntry.iType[2]);
  1121 
  1122 		TFileName name1(KEComSpiTestFilePlugIn3DllPathAndName);
  1123 		TFileName name2(*(entryToFill->iDllName));
  1124 		name1.LowerCase();
  1125 		name2.LowerCase();
  1126 		TParse dllparse;
  1127 		dllparse.Set(name1,NULL,NULL);
  1128 		test(dllparse.NameAndExt()== name2);
  1129 
  1130 		CleanupStack::PopAndDestroy(entryToFill); // resourceFile, resourceName, entryToFill
  1131 		entryToFill = NULL;
  1132 		}
  1133 	CleanupStack::PopAndDestroy(&resourceArchive);
  1134 	}
  1135 
  1136 /**
  1137 The test executes by Registering an Interface Implementation Collection
  1138 and later verifing it
  1139 
  1140 @SYMTestCaseID          SYSLIB-ECOM-CT-0718
  1141 @SYMTestCaseDesc	    Tests for process entry.
  1142 @SYMTestPriority 	    High
  1143 @SYMTestActions  	    Register an interface implementation collection and later verifies it.
  1144                         Check for OOM error.
  1145 @SYMTestExpectedResults The test must not fail.
  1146 @SYMREQ                 REQ0000
  1147 */
  1148 void CDiscovererTest::ProcessEntryTestL()
  1149 	{
  1150 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0718 "));
  1151 	CPluginBase* entryBase=NULL;
  1152 	TInt err=KErrNone;
  1153 
  1154 	//This part refers to testing of the entry containing the RSC
  1155 	_LIT(KExampleRscFileNameOnly,"EComExample5.rsc");
  1156 	TDriveName driveName1(iDriveUnit.Name());
  1157 	TParse path1;
  1158 	path1.Set(KNewResourceSearchPath(),NULL,&driveName1);
  1159 	//Creating a entry that represents the Rsc disovered during scanning of \\resource\\plugins
  1160 	TEntry rscEntry;
  1161 	rscEntry.iName=KExampleRscFileNameOnly;
  1162 	rscEntry.iType=TUidType(KNullUid,KUidInterfaceImplementationCollectionInfo,KNullUid);
  1163 	TUid dllUid1 = {KDynamicLibraryUidValue};
  1164 	TUid example5Uid={0x101F847B};
  1165 	TUidType dllUidType=TUidType(dllUid1,KUidInterfaceImplementationCollection,example5Uid);
  1166 	entryBase=CSecurePlugin::NewL(TheFs,rscEntry,driveName1, EFalse);
  1167 	CleanupStack::PushL(entryBase);
  1168 
  1169 	// ProcessEntryL() updates entryToFill with information on a rsc specified by the other params.
  1170 	TRAP(err, iStateAccessor->ProcessEntryL(*iDiscoverer,driveName1,entryBase,ETrue));
  1171 	TEST_OOM_ERR;
  1172 	test(err == KErrNone);
  1173 	CleanupStack::PopAndDestroy(entryBase);
  1174 	entryBase=NULL;
  1175 
  1176 	}
  1177 
  1178 /**
  1179 @SYMTestCaseID		SYSLIB-ECOM-CT-0091
  1180 @SYMTestCaseDesc 	Check that the ProcessEntryL for spi data works correctly.
  1181 @SYMTestPriority 	High
  1182 @SYMTestActions  	Ensure ecomtest.spi can be processed successfully
  1183 					and no leave occurred.
  1184 @SYMTestExpectedResults The test must not fail.
  1185 @SYMREQ REQ3655
  1186 */
  1187 void CDiscovererTest::ProcessSpiEntryTestL()
  1188 	{
  1189 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0091 "));
  1190 	TEntry spiEntry;
  1191 	test(TheFs.Entry(KEComSpiTestFilePathAndName, spiEntry) == KErrNone);
  1192 	TParse spiPath;
  1193 	spiPath.Set(KEComSpiTestFilePathAndName, NULL, NULL);
  1194 
  1195 	RResourceArchive resourceArchive;
  1196 	resourceArchive.OpenL(TheFs, KEComSpiTestFilePathAndName);
  1197 	CleanupClosePushL(resourceArchive);
  1198 
  1199 	// check SPI file type
  1200 	TUid type = resourceArchive.Type();
  1201 	test(type == KEcomSpiFileTypeUid);
  1202 
  1203 	// there is only 1 resource file
  1204 	while(!resourceArchive.End())
  1205 		{
  1206 		CPluginBase* entry = CSpiPlugin::NewL(resourceArchive);
  1207 		CleanupStack::PushL(entry);
  1208 
  1209 		TRAPD(err, iStateAccessor->ProcessEntryL(*iDiscoverer,spiPath.Drive(), entry,ETrue));
  1210 		TEST_OOM_ERR;
  1211 		test(err == KErrNone);
  1212 
  1213 		CleanupStack::PopAndDestroy(entry);
  1214 		entry = NULL;
  1215 		}
  1216 	CleanupStack::PopAndDestroy(&resourceArchive);
  1217 	}
  1218 /**
  1219 The test executes by checking whether discovered plugin entry is valid
  1220 
  1221 @SYMTestCaseID          SYSLIB-ECOM-CT-0719
  1222 @SYMTestCaseDesc	    Tests for plugin that resides on C drive
  1223 @SYMTestPriority 	    High
  1224 @SYMTestActions  	    Tests for the reference of the entry containing in the DLL
  1225                         Check for OOM error.
  1226 @SYMTestExpectedResults The test must not fail.
  1227 @SYMREQ                 REQ0000
  1228 */
  1229 void CDiscovererTest::ValidateEntryTestL()
  1230 	{
  1231 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0719 "));
  1232 	TInt err=KErrNone;
  1233 
  1234 	//This part refers to testing of the entry containing the RSC
  1235 	_LIT(KExampleRscFileNameOnly,"EComExample5.rsc");
  1236 	TDriveName driveName1(iDriveUnit.Name());
  1237 	TParse path1;
  1238 	CPluginBase *entryToFill1 = NULL;
  1239 	path1.Set(KNewResourceSearchPath(),NULL,&driveName1);
  1240 	//Creating a entry that represents the Rsc disovered during scanning of \\resource\\plugins
  1241 	TEntry rscEntry;
  1242 	rscEntry.iName=KExampleRscFileNameOnly;
  1243 	rscEntry.iType=TUidType(KNullUid,KUidInterfaceImplementationCollectionInfo,KNullUid);
  1244 
  1245 	TUid dllUid1 = {KDynamicLibraryUidValue};
  1246 	TUid example5Uid={0x101F847B};
  1247 	TUidType dllUidType=TUidType(dllUid1,KUidInterfaceImplementationCollection,example5Uid);
  1248 
  1249 	//Now call ValidateEntryL() with this entry containg the rsc
  1250 	TRAP(err,iStateAccessor->ValidateEntryL(*iDiscoverer,rscEntry,driveName1,entryToFill1, EFalse));
  1251 	TEST_OOM_ERR;
  1252 	test(err==KErrNone);
  1253 	test(entryToFill1->iDllThirdUid==dllUidType[2]);
  1254 	TParse dllParse;
  1255 	dllParse.Set(KNewExampleDllFileName,NULL,NULL);
  1256 	test(entryToFill1->iDllName->CompareF(dllParse.NameAndExt())==0);
  1257 
  1258 	delete entryToFill1;
  1259 	}
  1260 
  1261 /**
  1262 @SYMTestCaseID		SYSLIB-ECOM-CT-0092
  1263 @SYMTestCaseDesc 	Check that the ValidateEntryL for spi data works correctly.
  1264 @SYMTestPriority 	High
  1265 @SYMTestActions  	Ensure ecomtest.spi can be validated successfully
  1266 					and no leave occurred.
  1267 @SYMTestExpectedResults The test must not fail.
  1268 @SYMREQ REQ3655
  1269 */
  1270 void CDiscovererTest::ValidateSpiEntryTestL()
  1271 	{
  1272 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0092 "));
  1273 	TEntry spiEntry;
  1274 	test(TheFs.Entry(KEComSpiTestFilePathAndName, spiEntry) == KErrNone);
  1275 	TParse spiPath;
  1276 	spiPath.Set(KEComSpiTestFilePathAndName, NULL, NULL);
  1277 
  1278 	TEntry TestEntry;
  1279 	TUid uid1 = {0x10000079};
  1280 	TUid uid2 = {0x10009D8D};
  1281 	TUid	 uid3 = {0x101F847B}; // Uid for EComExample5.dll
  1282 	TestEntry.iType = TUidType(uid1, uid2, uid3);
  1283 	TestEntry.iName = KNewExampleDllFileNameOnly;
  1284 
  1285 	RResourceArchive resourceArchive;
  1286 	resourceArchive.OpenL(TheFs, KEComSpiTestFilePathAndName);
  1287 	CleanupClosePushL(resourceArchive);
  1288 
  1289 	// check SPI file type
  1290 	TUid type = resourceArchive.Type();
  1291 	test(type == KEcomSpiFileTypeUid);
  1292 
  1293 	// there is only 1 resource file
  1294 	while(!resourceArchive.End())
  1295 		{
  1296 		CPluginBase* entryToFill = NULL;
  1297 		TRAPD(err, iStateAccessor->ValidateEntryL(*iDiscoverer,resourceArchive,entryToFill));
  1298 		CleanupStack::PushL(entryToFill);
  1299 
  1300 		TEST_OOM_ERR;
  1301 		test(err == KErrNone);
  1302 		test(entryToFill->iDllThirdUid == TestEntry.iType[2]);
  1303 
  1304 		TFileName name1(KEComSpiTestFileDllPathAndName);
  1305 		TFileName name2(*(entryToFill->iDllName));
  1306 		name1.LowerCase();
  1307 		name2.LowerCase();
  1308 		TParse dllparse;
  1309 		dllparse.Set(name1,NULL,NULL);
  1310 		test(dllparse.NameAndExt()== name2);
  1311 
  1312 		CleanupStack::PopAndDestroy(entryToFill); // resourceFile, resourceName, entryToFill
  1313 		entryToFill = NULL;
  1314 		}
  1315 	CleanupStack::PopAndDestroy(&resourceArchive);
  1316 	}
  1317 
  1318 /**
  1319 @SYMTestCaseID		SYSLIB-ECOM-CT-0093
  1320 @SYMTestCaseDesc 	Check that there is DLL for each resource in SPI file
  1321 @SYMTestPriority 	High
  1322 @SYMTestActions  	Ensure ecom.spi contents i.e, rsc, actually exist on drive
  1323 @SYMTestExpectedResults The test must not fail.
  1324 @SYMREQ REQ3655
  1325 */
  1326 void CDiscovererTest::ValidateSpiPluginsTestL()
  1327 	{
  1328 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0093 "));
  1329 	__UHEAP_MARK;
  1330 
  1331 	TEntry spiEntry;
  1332 	TBool err = TheFs.Entry(KEComSPIFilePathAndName, spiEntry);
  1333 	// It is possible for ecom.spi file not to exist. If it does not then no testing can be done.
  1334 	if(err != KErrNone)
  1335 		return;
  1336 
  1337 	TParse spiPath;
  1338 	spiPath.Set(KEComSPIFilePathAndName, NULL, NULL);
  1339 
  1340 	//To read the SPI file
  1341 	RResourceArchive resourceArchive;
  1342 	resourceArchive.OpenL(TheFs, KEComSPIFilePathAndName);
  1343 	CleanupClosePushL(resourceArchive);
  1344 	CRegistryData::CDriveData* driveData = NULL;
  1345 	//Check spi file type
  1346 	TUid spiType = resourceArchive.Type();
  1347 	test(spiType == KEcomSpiFileTypeUid);
  1348 
  1349 	while(!resourceArchive.End())
  1350 		{
  1351 		CPluginBase* entryToFill = NULL;
  1352 		TRAPD(err, iStateAccessor->ValidateEntryL(*iDiscoverer,resourceArchive,entryToFill));
  1353 		CleanupStack::PushL(entryToFill);
  1354 
  1355 		TEST_OOM_ERR;
  1356 		test(err == KErrNone);
  1357 
  1358 		CRegistryData::CDllData * dllData=CRegistryData::CDllData::NewLC(*(entryToFill->iDllName),entryToFill->iDllModifiedTime,entryToFill->iDllSecondUid, entryToFill->iDllThirdUid,driveData);
  1359 
  1360 		TBool successful=dllData->ProcessSecurityCheckL();
  1361 		if(!successful)
  1362 		{
  1363 			_LIT(KMessage,"ERROR: Plugin SID Mismatch ERROR for %S.");
  1364 			RDebug::Print(KMessage, entryToFill->iDllName);
  1365 		}
  1366 		test(successful);
  1367 
  1368 		CleanupStack::PopAndDestroy(dllData);
  1369 		CleanupStack::PopAndDestroy(entryToFill);
  1370 		entryToFill=NULL;
  1371 		}
  1372 
  1373 		CleanupStack::PopAndDestroy(&resourceArchive);
  1374 	__UHEAP_MARKEND;
  1375 	}
  1376 
  1377 
  1378 /**
  1379 Scans plugin directories and verifies successfull completion
  1380 
  1381 @SYMTestCaseID          SYSLIB-ECOM-CT-0720
  1382 @SYMTestCaseDesc	    Tests for scanning plugin directories and verifies successfull completion
  1383 @SYMTestPriority 	    High
  1384 @SYMTestActions  	    Scans registered plugin directories from A to Z drive
  1385 @SYMTestExpectedResults The test must not fail.
  1386 @SYMREQ                 REQ0000
  1387 */
  1388 void CDiscovererTest::ScanDirectoryIncrementTestL()
  1389 	{
  1390 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0720 "));
  1391 	TInt err;
  1392 	// Scans registered plugin directories from A to Z drive
  1393 	TEComCachedDriveInfoIterator iter(*iCachedDriveInfo);
  1394 	for(iter.Last(); iter.InRange(); iter.Prev())
  1395 		{
  1396 		TRAP(err, iStateAccessor->ScanDirectoryL(*iDiscoverer,iter.DriveNumber()));
  1397 		TEST_OOM_ERR;
  1398 		test(err == KErrNone);
  1399 		}
  1400 	// After successful completion of ScanDirectoryL, DiscovererObserver is updated
  1401 	// with DiscoveriesComplete successful
  1402 	// DiscovererObserver is a stub class used in place of CRegistrar. Flags are used for the
  1403 	// successfull execution of API
  1404 	test(iDiscovererObserver->IsDiscoveriesCompleteSuccessfully());
  1405 	}
  1406 
  1407 /**
  1408 Stops scaning of plugin directories
  1409 
  1410 @SYMTestCaseID          SYSLIB-ECOM-CT-0722
  1411 @SYMTestCaseDesc	    Tests for stopping of scanning plugin directories
  1412 @SYMTestPriority 	    High
  1413 @SYMTestActions  	    Stops scaning of plugin directories. Check for OOM error.
  1414 @SYMTestExpectedResults The test must not fail.
  1415 @SYMREQ                 REQ0000
  1416 */
  1417 void CDiscovererTest::ScanDirectoryCancelTestL()
  1418 	{
  1419 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0722 "));
  1420 	TRAPD(err, iStateAccessor->ScanDirectoryCancel(*iDiscoverer));
  1421 	TEST_OOM_ERR;
  1422 	test(err == KErrNone);
  1423 	test(iStateAccessor->IsDirectoryScanCancelled(*iDiscoverer));
  1424 	}
  1425 
  1426 /**
  1427 Scaning of plugin directories in Z: drive and C:drive, check these two drives is in drive list
  1428 
  1429 @SYMTestCaseID          SYSLIB-ECOM-UT-1859
  1430 @SYMTestCaseDesc	    Tests for scanning plugin directories in specific drives.
  1431 @SYMTestPriority 	    High
  1432 @SYMTestActions  	    scaning of plugin directories. Check for OOM error.
  1433 @SYMTestExpectedResults The test must not fail.
  1434 @SYMDEF                 DEF088454
  1435 */
  1436 
  1437 void CDiscovererTest::ScanDirectoryTestL()
  1438 	{
  1439 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-1859 "));
  1440 	EComPerformance::ResetEComPerfTimeRecords();
  1441 
  1442 	// Do scan on specific drive
  1443 	TRAPD(err, iStateAccessor->ScanDriveL(*iDiscoverer, EDriveZ, ETrue));
  1444 	TEST_OOM_ERR;
  1445 	test(err == KErrNone);
  1446 
  1447 	TRAP(err, iStateAccessor->ScanDriveL(*iDiscoverer, EDriveC, EFalse));
  1448 	TEST_OOM_ERR;
  1449 	test(err == KErrNone);
  1450 
  1451 	// Test record information correct
  1452 	test(EComPerformance::iEComPerfTimeRecordCount == 4);
  1453 	test(EComPerformance::iEComPerfTimeRecords[0].iType == ECDiscovererRediscoveryScanDirectoryL);
  1454 	test(EComPerformance::iEComPerfTimeRecords[0].iInfo == EDriveZ);
  1455 	test(EComPerformance::iEComPerfTimeRecords[1].iType == ECDiscovererRediscoveryScanDirectoryL);
  1456 	test(EComPerformance::iEComPerfTimeRecords[1].iInfo == EDriveZ);
  1457 
  1458 	test(EComPerformance::iEComPerfTimeRecords[2].iType == ECDiscovererRediscoveryScanDirectoryL);
  1459 	test(EComPerformance::iEComPerfTimeRecords[2].iInfo == EDriveC);
  1460 	test(EComPerformance::iEComPerfTimeRecords[3].iType == ECDiscovererRediscoveryScanDirectoryL);
  1461 	test(EComPerformance::iEComPerfTimeRecords[3].iInfo == EDriveC);
  1462 	}
  1463 
  1464 /**
  1465 @SYMTestCaseID		SYSLIB-ECOM-CT-0185
  1466 @SYMTestCaseDesc 	Check that SSA related states transition as expected when discovery is staged
  1467 @SYMTestPriority 	High
  1468 @SYMTestActions  	Use CDiscoverer::ProcessSSAEventL to start discovery in stages i.e. rom only
  1469 then non rom only. Check that the
  1470 CDiscoverer state is transitioning correctly at every step.
  1471 @SYMTestExpectedResults The test must not fail.
  1472 @SYMREQ REQ3655
  1473 */
  1474 void CDiscovererTest::StagedDiscoveryStateTransitionTestL()
  1475 	{
  1476 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0185 "));
  1477 	//After construction check that:
  1478 	//- current state is EDisc_NoPluginsDisc
  1479 	//- registry is empty. i.e. no plugins discovered
  1480 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_NoPluginsDisc);
  1481 
  1482 	//Call ProcessSSAEvent with state set to various states that will
  1483 	//not cause a transition. Check that:
  1484 	//- current state has not changed
  1485 	iDiscoverer->ProcessSSAEventL(EStartupStateUndefined);
  1486 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_NoPluginsDisc);
  1487 
  1488 	//Call ProcessSSAEventL with EStartupStateCriticalStatic state.
  1489 	//Check that
  1490 	//- current state is EDisc_CriticalPluginsDisc
  1491 	//- discoverer observer has processed RO plugins
  1492 	iDiscoverer->ProcessSSAEventL(EStartupStateCriticalStatic);
  1493 	test(iStateAccessor->GetDiscovererObserverState(*iDiscoverer) == CObserverStub::OSS_CriticalPlugins);
  1494 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_CriticalPluginsDisc);
  1495 
  1496 	//Call ProcessSSAEvent with state set to various states that will
  1497 	//not cause a transition. Check that:
  1498 	//- current state has not changed
  1499 	iDiscoverer->ProcessSSAEventL(EStartupStateUndefined);
  1500 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_CriticalPluginsDisc);
  1501 	iDiscoverer->ProcessSSAEventL(EStartupStateCriticalStatic);
  1502 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_CriticalPluginsDisc);
  1503 
  1504 	//Call ProcessSSAEventL with EStartupStateNonCritical state. Check that
  1505 	//- current state is EDisc_AllPluginsDisc
  1506 	//- discoverer observer has processed non-RO plugins
  1507 	iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical);
  1508 	test(iStateAccessor->GetDiscovererObserverState(*iDiscoverer) == CObserverStub::OSS_AllPlugins);
  1509 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc);
  1510 
  1511 	//Call ProcessSSAEvent with state set to various states that will
  1512 	//not cause a transition. Check that:
  1513 	//- current state has not changed
  1514 	iDiscoverer->ProcessSSAEventL(EStartupStateUndefined);
  1515 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc);
  1516 	iDiscoverer->ProcessSSAEventL(EStartupStateCriticalStatic);
  1517 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc);
  1518 	iDiscoverer->ProcessSSAEventL(EStartupStateCriticalDynamic);
  1519 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc);
  1520 	iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical);
  1521 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc);
  1522 
  1523 	//Call ProcessDNEventL to indicate that current set of plugins is dirty.
  1524 	//Check that:
  1525 	//- current state is EDisc_PluginsDirty
  1526 	iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveC);
  1527 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_PluginsDirty);
  1528 
  1529 	//Call ProcessDNEventL to indicate that a rediscovery should take place.
  1530 	//Check that:
  1531 	//- current state is EDisc_AllPluginsDisc
  1532 	iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveC);
  1533 	iStateAccessor->CompleteNotificationProcessing(*iDiscoverer);
  1534 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc);
  1535 	}
  1536 
  1537 /**
  1538 @SYMTestCaseID		SYSLIB-ECOM-CT-0186
  1539 @SYMTestCaseDesc 	Check that SSA related states transition as expected when discovery is all at once
  1540 @SYMTestPriority 	High
  1541 @SYMTestActions  	Use CDiscoverer::ProcessSSAEventL to start discovery all at once. Check that the
  1542 CDiscoverer state is transitioning correctly at every step.
  1543 @SYMTestExpectedResults The test must not fail.
  1544 @SYMREQ REQ3655
  1545 */
  1546 void CDiscovererTest::AllAtOnceDiscoveryStateTransitionTestL()
  1547 	{
  1548 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0186 "));
  1549 	//After construction check that:
  1550 	//- current state is EDisc_NoPluginsDisc
  1551 	//- registry is empty. i.e. no plugins discovered
  1552 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_NoPluginsDisc);
  1553 
  1554 	//Call ProcessSSAEvent with state set to various states that will
  1555 	//not cause a transition. Check that:
  1556 	//- current state has not changed
  1557 	iDiscoverer->ProcessSSAEventL(EStartupStateUndefined);
  1558 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_NoPluginsDisc);
  1559 
  1560 	//Call ProcessSSAEventL with EStartupStateNonCritical state. Check that
  1561 	//- current state is EDisc_AllPluginsDisc
  1562 	//- discoverer observer has processed non-RO plugins
  1563 	iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical);
  1564 	test(iStateAccessor->GetDiscovererObserverState(*iDiscoverer) == CObserverStub::OSS_AllPlugins);
  1565 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc);
  1566 
  1567 	//Call ProcessSSAEvent with state set to various states that will
  1568 	//not cause a transition. Check that:
  1569 	//- current state has not changed
  1570 	iDiscoverer->ProcessSSAEventL(EStartupStateUndefined);
  1571 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc);
  1572 	iDiscoverer->ProcessSSAEventL(EStartupStateCriticalStatic);
  1573 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc);
  1574 	iDiscoverer->ProcessSSAEventL(EStartupStateCriticalDynamic);
  1575 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc);
  1576 	iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical);
  1577 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc);
  1578 
  1579 	//Call ProcessDNEventL to indicate that current set of plugins is dirty.
  1580 	//Check that:
  1581 	//- current state is EDisc_PluginsDirty
  1582 	iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveC);
  1583 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_PluginsDirty);
  1584 
  1585 	//Call ProcessDNEventL to indicate that a rediscovery should take place.
  1586 	//Check that:
  1587 	//- current state is EDisc_AllPluginsDisc
  1588 	iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveC);
  1589 	iStateAccessor->CompleteNotificationProcessing(*iDiscoverer);
  1590 	test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc);
  1591 	}
  1592 
  1593 /**
  1594 @SYMTestCaseID          SYSLIB-ECOM-UT-1796
  1595 @SYMTestCaseDesc	    Tests multiple notification processing for
  1596 						"INC087110: ECOM rescanning code could miss a drive?" and
  1597 						"DEF088454: [PCB] ECOM CDiscoverer::CIdleScanningTimer::RunL() Performs Unnecessary Tasks"
  1598 @SYMTestPriority 	    High
  1599 @SYMTestActions  	    Create multiple notification request on both single drive and multiple drives.
  1600 						Check if the notifications are ONLY processed on corresponding drive(s) by timer.
  1601 						Check the notifications are processed properly by timer.
  1602 						Check the state of discoverer was set correctly on completion.
  1603 @SYMTestExpectedResults The test must not fail.
  1604 @SYMDEF                 INC087110, DEF088454
  1605 */
  1606 void CDiscovererTest::MultipleNotificationProcessingTestL()
  1607 	{
  1608 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-1796 "));
  1609 	TBool pending = EFalse;
  1610 
  1611 	// Multiple notification on single drive:
  1612 	iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical);
  1613 	// Raise multiple notification on C: drive
  1614 	for(TInt num = 0; num < 10; ++num)
  1615 		{
  1616 		iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveC);
  1617 		iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveC);
  1618 		}
  1619 
  1620 	// Check only one pending drive in the list.
  1621 	test(iStateAccessor->PendingDriveListCount(*iDiscoverer) == 1);
  1622 	// Check the pending drive is C: drive
  1623 	pending = iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveC);
  1624 	test(pending);
  1625 
  1626 	// Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process
  1627 	// all pending notification.
  1628 	iShutdown.StartAfter(KOneSecond * 3);
  1629 	CActiveScheduler::Start();
  1630 
  1631 	// Check if notifications have been processed on all drives.
  1632 	test(iStateAccessor->IsScanDirectoryComplete(*iDiscoverer));
  1633 
  1634 	// Multiple notification on multiple drives:
  1635 	iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical);
  1636 	// Raise multiple notification on C: drive
  1637 	for(TInt num = 0; num < 10; ++num)
  1638 		{
  1639 		iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveC);
  1640 		iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveC);
  1641 		}
  1642 #if defined(__WINSCW__) // X: drive on emulator
  1643 	// Raise multiple notification on X: drive
  1644 	for(TInt num = 0; num < 10; ++num)
  1645 		{
  1646 		iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveX);
  1647 		iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveX);
  1648 		}
  1649 
  1650 #else // E: drive on HW
  1651 	// Raise multiple notification on E: drive
  1652 	for(TInt num = 0; num < 10; ++num)
  1653 		{
  1654 		iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveE);
  1655 		iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveE);
  1656 		}
  1657 #endif	// __WINSCW__
  1658 
  1659 	// Check only one pending drive in the list.
  1660 	test(iStateAccessor->PendingDriveListCount(*iDiscoverer) == 2);
  1661 
  1662 	// Check the pending drive are C: and X: on emulator, or C: and E: on hardware.
  1663 	pending = iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveC);
  1664 	test(pending);
  1665 #if defined(__WINSCW__) // X: drive on emulator
  1666 	pending = iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveX);
  1667 	test(pending);
  1668 #else // E: drive on HW
  1669 	pending = iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveE);
  1670 	test(pending);
  1671 #endif	// __WINSCW__
  1672 
  1673 	// Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process
  1674 	// all pending notification.
  1675 	iShutdown.StartAfter(KOneSecond * 3);
  1676 	CActiveScheduler::Start();
  1677 
  1678 	// Check if notifications have been processed on all drives.
  1679 	test(iStateAccessor->IsScanDirectoryComplete(*iDiscoverer));
  1680 	}
  1681 
  1682 /**
  1683 @SYMTestCaseID          SYSLIB-ECOM-UT-3519
  1684 @SYMTestCaseDesc	    Tests notification processing during SWI for
  1685 						PDEF110201: Undesireable interaction between ECOM and SWI
  1686 @SYMTestActions  	    Create notification request on  drives.
  1687 						Check the notifications are not processed by timer if SWI in progress.
  1688 						Check that timer processes pending notifications after SWI completes
  1689 @SYMTestExpectedResults The test must not fail.
  1690 @SYMDEF                 PDEF110201
  1691 */
  1692 void CDiscovererTest::SWINotificationProcessingTestL()
  1693 	{
  1694 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-3519 "));
  1695 
  1696 	//Begin SWI
  1697 	iStateAccessor->SwiChangeNotificationL(*iDiscoverer,ESASwisInstall);
  1698 
  1699 	iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical);
  1700 
  1701 	// Raise a notification on C: drive
  1702 	iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveC);
  1703 	iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveC);
  1704 
  1705 	// Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process
  1706 	// all pending notification.
  1707 	iShutdown.StartAfter(KOneSecond * 3);
  1708 	CActiveScheduler::Start();
  1709 
  1710 	// Check that notifications have NOT been processed as SWI is in progress
  1711 	test(!iStateAccessor->IsScanDirectoryComplete(*iDiscoverer));
  1712 
  1713 	//complete SWI
  1714 	iStateAccessor->SwiChangeNotificationL(*iDiscoverer,ESASwisUninstall);
  1715 
  1716 
  1717 #if defined(__WINSCW__) // X: drive on emulator
  1718 	// Raise  notification on X: drive
  1719 	iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveX);
  1720 	iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveX);
  1721 #else // E: drive on HW
  1722 	// Raise notification on E: drive
  1723 	iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveE);
  1724 	iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveE);
  1725 #endif	// __WINSCW__
  1726 
  1727 	// Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process
  1728 	// all pending notification.
  1729 	iShutdown.StartAfter(KOneSecond * 3);
  1730 	CActiveScheduler::Start();
  1731 
  1732 	// Check that notifications have NOT been processed as SWI is not finished
  1733 	test(!iStateAccessor->IsScanDirectoryComplete(*iDiscoverer));
  1734 
  1735 	//finalise SWI
  1736 	iStateAccessor->SwiChangeNotificationL(*iDiscoverer,ESASwisNone);
  1737 
  1738 	// Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process
  1739 	// all pending notification.
  1740 	iShutdown.StartAfter(KOneSecond * 3);
  1741 	CActiveScheduler::Start();
  1742 
  1743 	// Check that notifications have been processed on all drives
  1744 	test(iStateAccessor->IsScanDirectoryComplete(*iDiscoverer));
  1745 
  1746 	//Begin SWI
  1747 	iStateAccessor->SwiChangeNotificationL(*iDiscoverer,ESASwisInstall);
  1748 
  1749 	iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical);
  1750 
  1751 	// Raise a notification
  1752 	iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified,EDriveC);
  1753 	iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover,EDriveC);
  1754 
  1755 	// Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process
  1756 	// all pending notification.
  1757 	iShutdown.StartAfter(KOneSecond * 3);
  1758 	CActiveScheduler::Start();
  1759 
  1760 	// Check that notifications have NOT been processed as SWI is in progress
  1761 	test(!iStateAccessor->IsScanDirectoryComplete(*iDiscoverer));
  1762 
  1763 	//Simulate an error in reading the P&S Variable - This should reset the
  1764 	//SWI state to ESASwisNone in the same way as completing SWI
  1765 	iStateAccessor->SwiChangeNotifierRunError(*iDiscoverer,KErrNotFound);
  1766 
  1767 	// Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process
  1768 	// all pending notification.
  1769 	iShutdown.StartAfter(KOneSecond * 3);
  1770 	CActiveScheduler::Start();
  1771 
  1772 	// Check that notifications have been processed on all drives
  1773 	test(iStateAccessor->IsScanDirectoryComplete(*iDiscoverer));
  1774 	}
  1775 
  1776 /**
  1777 Call the TDiscoverer_StateAccessor::IdleScanningTimerRunErrorL funtion
  1778 passing in a reference to the CDiscoverer object and the error code
  1779 to be passed to the CIdleScanningTimer::RunError function
  1780 */
  1781 void CDiscovererTest::IdleScanningTimerRunErrorL()
  1782 	{
  1783 	iStateAccessor->IdleScanningTimerRunErrorL(*iDiscoverer, EEComPanic_CDiscoverer_CIdleScanningTimer_RunError);
  1784 	}
  1785 
  1786 /**
  1787 Call the TDiscoverer_StateAccessor::IdleScanningTimerRunErrorL funtion
  1788 passing in a reference to the CDiscoverer object and the error code
  1789 to be passed to the CDirChangeNotifier::RunError function
  1790 */
  1791 void CDiscovererTest::DirChangeNotifierRunErrorL()
  1792 	{
  1793 	iStateAccessor->DirChangeNotifierRunErrorL(*iDiscoverer, EEComPanic_CDiscoverer_CDirChangeNotifier_RunError);
  1794 	}
  1795 
  1796 /**
  1797 Call the TDiscoverer_StateAccessor::IdleScanningTimerRunErrorL funtion
  1798 passing in a reference to the CDiscoverer object and the error code
  1799 to be passed to the CSwiChangeNotifier::RunError function
  1800 */
  1801 void CDiscovererTest::SwiChangeNotifierRunError()
  1802 	{
  1803 	iStateAccessor->SwiChangeNotifierRunError(*iDiscoverer, KErrNoMemory);
  1804 	}
  1805 /**
  1806 @SYMTestCaseID          SYSLIB-ECOM-UT-3172
  1807 @SYMTestCaseDesc	    Tests language switch notification processing for
  1808 						"CR0902: Enable Dynamic Language Switching in APPARC, ECOM and EIKSTD"
  1809 @SYMTestPriority 	    High
  1810 @SYMTestActions  	    Call ProcessSSAEventL with EStartupStateNonCritical state.
  1811 						Call DriveMountedL to build up drives in system.
  1812 						Call LanguageChangedL to raise multiple notification on all drives in system.
  1813 						Check if the notifications are processed on all drive(s) by timer.
  1814 						Check the notifications are processed properly by timer.
  1815 						Check the state of discoverer was set correctly on completion.
  1816 @SYMTestExpectedResults The test must not fail.
  1817 @SYMDEF                 CR0902
  1818 */
  1819 void CDiscovererTest::LanguageChangedNotificationTestL()
  1820 	{
  1821 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-3172 "));
  1822 	TBool pending = EFalse;
  1823 
  1824 	//Call ProcessSSAEventL with EStartupStateNonCritical state
  1825 	iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical);
  1826 
  1827 	if(iStateAccessor->IsDriveMounted(*iDiscoverer, EDriveK))
  1828 		{
  1829 		TRAPD(err, iStateAccessor->DriveUnmountedL(*iDiscoverer, EDriveK));
  1830 		TEST_OOM_ERR;
  1831 		test(err == KErrNone);
  1832 		}
  1833 
  1834 	// Mount drives on system
  1835 	TRAPD(err, iStateAccessor->DriveMountedL(*iDiscoverer, EDriveZ));
  1836 	TEST_OOM_ERR;
  1837 	test(err == KErrNone);
  1838 
  1839 	TRAP(err, iStateAccessor->DriveMountedL(*iDiscoverer, EDriveC));
  1840 	TEST_OOM_ERR;
  1841 	test(err == KErrNone);
  1842 
  1843 #if defined(__WINSCW__) // X: drive on emulator
  1844 	TRAP(err, iStateAccessor->DriveMountedL(*iDiscoverer, EDriveX));
  1845 	TEST_OOM_ERR;
  1846 	test(err == KErrNone);
  1847 #else // E: drive on HW
  1848 	TRAP(err, iStateAccessor->DriveMountedL(*iDiscoverer, EDriveE));
  1849 	TEST_OOM_ERR;
  1850 	test(err == KErrNone);
  1851 #endif
  1852 
  1853 	// Multiple notification on all drives:
  1854 	iStateAccessor->LanguageChangedL(*iDiscoverer);
  1855 
  1856 	// Check three pending drive in the list.
  1857 	test(iStateAccessor->PendingDriveListCount(*iDiscoverer) == 3);
  1858 	// Check the pending drive is Z: drive
  1859 	pending = iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveZ);
  1860 	test(pending);
  1861 
  1862 	// Check the pending drive is C: drive
  1863 	pending = iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveC);
  1864 	test(pending);
  1865 
  1866 #if defined(__WINSCW__) // X: drive on emulator
  1867 	pending = iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveX);
  1868 	test(pending);
  1869 #else // E: drive on HW
  1870 	pending = iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveE);
  1871 	test(pending);
  1872 #endif	// __WINSCW__
  1873 
  1874 	// Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process
  1875 	// all pending notification.
  1876 	iShutdown.StartAfter(KOneSecond * 3);
  1877 	CActiveScheduler::Start();
  1878 
  1879 	// Check if notifications have been processed on all drives.
  1880 	test(iStateAccessor->IsScanDirectoryComplete(*iDiscoverer));
  1881 
  1882 	}
  1883 
  1884 /**
  1885 @SYMTestCaseID          SYSLIB-ECOM-CT-0723
  1886 @SYMTestCaseDesc	    Tests the creation and deletion of CDiscoverer
  1887 @SYMTestPriority 	    High
  1888 @SYMTestActions  	    Create and delete CDiscoverer object,checks for open handles
  1889 @SYMTestExpectedResults The test must not fail.
  1890 @SYMREQ                 REQ0000
  1891 */
  1892 LOCAL_C void CreateDeleteTestL()
  1893 	{
  1894 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0723 CreateDeleteTestL "));
  1895 	//
  1896 	// Tests the Creating and deletion of CDiscoverer
  1897 	// ------------------------------------------------------------------
  1898 	//
  1899 	// Set up for heap leak checking
  1900 	__UHEAP_MARK;
  1901 
  1902 	//Check Thread handles leak
  1903 	TInt startProcessHandleCount = 0;
  1904 	TInt startThreadHandleCount = 0;
  1905 	TInt endProcessHandleCount = 0;
  1906 	TInt endThreadHandleCount = 0;
  1907 
  1908 	RThread rThread;
  1909 	rThread.HandleCount(startProcessHandleCount, startThreadHandleCount);
  1910 
  1911 	//START CREATE DELETE TEST//
  1912 
  1913 	CDiscovererTest* discTest = CDiscovererTest::NewL();
  1914 
  1915 	test(discTest != NULL);
  1916 
  1917 	delete discTest;
  1918 
  1919 	//END CREATE DELETE TEST//
  1920 
  1921 	// Check for open handles
  1922 	rThread.HandleCount(endProcessHandleCount, endThreadHandleCount);
  1923 	test(startThreadHandleCount == endThreadHandleCount);
  1924 
  1925 	//Test ends
  1926 	__UHEAP_MARKEND;
  1927 	}
  1928 
  1929 /**
  1930 @SYMTestCaseID          SYSLIB-ECOM-CT-0724
  1931 @SYMTestCaseDesc	    OOM test for create and delete of CDiscoverer.
  1932 @SYMTestPriority 	    High
  1933 @SYMTestActions  	    Create and delete CDiscoverer object,checks for any memory leak.
  1934 @SYMTestExpectedResults The test must not fail.
  1935 @SYMREQ                 REQ0000
  1936 */
  1937 LOCAL_C void OOMCreateDeleteTest()
  1938 	{
  1939 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0724 OOM CreateDeleteTest "));
  1940 	TInt err;
  1941 	TInt failAt = 1;
  1942 	__UNUSED_VAR(failAt);
  1943 
  1944 	CDiscovererTest* discTest = NULL;
  1945 
  1946 	do
  1947 		{
  1948 		__UHEAP_MARK;
  1949   		// find out the number of open handles
  1950 		TInt startProcessHandleCount;
  1951 		TInt startThreadHandleCount;
  1952 		RThread().HandleCount(startProcessHandleCount, startThreadHandleCount);
  1953 
  1954 		__UHEAP_SETFAIL(RHeap::EDeterministic, failAt++);
  1955 
  1956 		TRAP(err, discTest = CDiscovererTest::NewL());
  1957 
  1958 		__UHEAP_SETFAIL(RHeap::ENone, 0);
  1959 
  1960 		delete discTest;
  1961 		discTest = NULL;
  1962 
  1963 		// check that no handles have leaked
  1964 		TInt endProcessHandleCount;
  1965 		TInt endThreadHandleCount;
  1966 		RThread().HandleCount(endProcessHandleCount, endThreadHandleCount);
  1967 
  1968 		test(startProcessHandleCount == endProcessHandleCount);
  1969 		test(startThreadHandleCount  == endThreadHandleCount);
  1970 
  1971 		__UHEAP_MARKEND;
  1972 		}
  1973 	while(err == KErrNoMemory);
  1974 
  1975 	test.Printf(_L("- Succeeded at heap failure rate of %i\n"), failAt);
  1976 	test(err == KErrNone);
  1977 	}
  1978 
  1979 // Type definition for pointer to member function.
  1980 // Used in calling the CDiscovererTest member function for testing.
  1981 typedef void (CDiscovererTest::*ClassFuncPtrL) (void);
  1982 
  1983 /**
  1984 @SYMTestCaseID          SYSLIB-ECOM-CT-0725
  1985 @SYMTestCaseDesc	    Function to call all test functions
  1986 @SYMTestPriority 	    High
  1987 @SYMTestActions  	    Calls up test function of CDiscovererTest.
  1988 @SYMTestExpectedResults The test must not fail.
  1989 @SYMREQ                 REQ0000
  1990 */
  1991 /**
  1992 Wrapper function to call all test functions
  1993 
  1994 @param		testFunc pointer to test function
  1995 @param		aTestDesc test function name
  1996 */
  1997 LOCAL_C void DoBasicTestL(ClassFuncPtrL testFuncL, const TDesC& aTestDesc)
  1998 	{
  1999 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0725 "));
  2000 	test.Next(aTestDesc);
  2001 
  2002 	__UHEAP_MARK;
  2003   	// find out the number of open handles
  2004 	TInt startProcessHandleCount;
  2005 	TInt startThreadHandleCount;
  2006 	RThread().HandleCount(startProcessHandleCount, startThreadHandleCount);
  2007 
  2008 	CDiscovererTest* discTest = CDiscovererTest::NewL();
  2009 	CleanupStack::PushL(discTest);
  2010 
  2011 	(discTest->*testFuncL)();
  2012 
  2013 	CleanupStack::PopAndDestroy(discTest);
  2014 
  2015 	// check that no handles have leaked
  2016 	TInt endProcessHandleCount;
  2017 	TInt endThreadHandleCount;
  2018 	RThread().HandleCount(endProcessHandleCount, endThreadHandleCount);
  2019 
  2020 	test(startProcessHandleCount == endProcessHandleCount);
  2021 	test(startThreadHandleCount  == endThreadHandleCount);
  2022 
  2023 	__UHEAP_MARKEND;
  2024 	}
  2025 
  2026 /**
  2027 Utility function to continually invoke a test function and cause memory allocation failures
  2028 
  2029 @param		testFuncL pointer to OOM test function
  2030 @param	 	tryCount specifies what value to start memory allocation failures at
  2031 @param		increment how much to increase the point at which memory allocation will fail on each test attempt
  2032 @param		stopCount the value of memory allocation failure to stop testing at
  2033 */
  2034 TInt RunTestUnderOOMCondition(ClassFuncPtrL testFuncL, TInt tryCount, TInt increment, TInt stopCount)
  2035 {
  2036 	TInt err = KErrNone;
  2037 
  2038 	do
  2039 		{
  2040 
  2041 		__UHEAP_MARK;
  2042   		// find out the number of open handles
  2043 		TInt startProcessHandleCount;
  2044 		TInt startThreadHandleCount;
  2045 		RThread().HandleCount(startProcessHandleCount, startThreadHandleCount);
  2046 
  2047 		CDiscovererTest* discTest = CDiscovererTest::NewL();
  2048 		CleanupStack::PushL(discTest);
  2049 
  2050 
  2051 		__UHEAP_SETFAIL(RHeap::EDeterministic, tryCount);
  2052 		TRAP(err, (discTest->*testFuncL)());
  2053 		__UHEAP_SETFAIL(RHeap::ENone, 0);
  2054 
  2055 
  2056 		CleanupStack::PopAndDestroy(discTest);
  2057 		discTest = NULL;
  2058 
  2059 		// check that no handles have leaked
  2060 		TInt endProcessHandleCount;
  2061 		TInt endThreadHandleCount;
  2062 		RThread().HandleCount(endProcessHandleCount, endThreadHandleCount);
  2063 
  2064 		test(startProcessHandleCount == endProcessHandleCount);
  2065 		test(startThreadHandleCount  == endThreadHandleCount);
  2066 
  2067 		__UHEAP_MARKEND;
  2068 
  2069 	 	tryCount = tryCount + increment;
  2070 		} while((err == KErrNoMemory) && (tryCount != stopCount));
  2071 
  2072 		tryCount = tryCount - increment;
  2073 
  2074 
  2075 		if (err == KErrNoMemory)
  2076 		{
  2077 			// test has not yet been able to pass due to memory allocation failures.
  2078 			return -1;
  2079 		}
  2080 
  2081 		test(err == KErrNone);
  2082 
  2083 	 	// If enough memory has finally been allocated for the test to pass then return
  2084 		// the memory allocation counter value.
  2085 		test.Printf(_L("- server succeeded at heap failure rate of %i\n"), tryCount);
  2086 		return tryCount;
  2087 }
  2088 
  2089 /**
  2090 Wrapper function to call all OOM test functions
  2091 
  2092 @param		testFuncL pointer to OOM test function
  2093 @param		aTestDesc test function name
  2094 */
  2095 LOCAL_C void DoOOMTestL(ClassFuncPtrL testFuncL, const TDesC& aTestDesc)
  2096 	{
  2097 	test.Next(aTestDesc);
  2098 
  2099 	TInt startCount = 1;
  2100 	TInt increment = 1;
  2101 	TInt successRate = -1;
  2102 	TInt stopCount = 256;
  2103 
  2104  	successRate = RunTestUnderOOMCondition(testFuncL, startCount, increment, stopCount);
  2105 
  2106  	// (INC115057)
  2107  	// When method CDiscoverer::CDirScanner::DoScanDriveL is encountered in a test hundreds
  2108  	// of plug-in files will be scanned. It takes in the order of 3-4000 memory allocation failure
  2109  	// loops (in techview context) before the scan is finished and the test can complete successfully.
  2110  	// This will take over an hour. Instead determine the approximate failure point by
  2111  	// testing with large increments between failures. Once a failure rate interval is found test
  2112  	// for OOM conditions running up to it.
  2113  	// This in effect means that we are checking OOM at the start and end of tests but skipping
  2114  	// the scanning of each and every plugin in the middle of the test.
  2115  	// (Note that CDiscoverer::CDirScanner::DoScanDriveL may return without leaving when it
  2116  	// can't allocate TFileName. In this case it seems to this function that the test is
  2117  	// successful. Therefore even if the successRate above indicates a pass the test is still
  2118  	// re-run below with larger memory allocation failure valued to make sure that the test does
  2119  	// in fact run to completion.
  2120 
  2121  		startCount = 256;
  2122  		increment = 256;
  2123  		stopCount = -1;
  2124  		successRate = RunTestUnderOOMCondition(testFuncL, startCount, increment, stopCount);
  2125  		test(successRate > 0);
  2126 
  2127  		if (successRate > 256)
  2128  		{
  2129  			startCount = successRate - 256;
  2130  			increment = 1;
  2131  			stopCount = -1;
  2132  			successRate = RunTestUnderOOMCondition(testFuncL, startCount, increment, stopCount);
  2133  		}
  2134 
  2135  	test(successRate > 0);
  2136 	}
  2137 
  2138 /**
  2139 Creates and installs active scheduler for this thread and calls
  2140 CDiscovererTest::IdleScanningTimerRunErrorL
  2141 
  2142 @param		aDiscTest The CDiscovererTest object used to carry out the test
  2143 */
  2144 LOCAL_C	void DoIdleScanningTimerRunErrorTestL(CDiscovererTest* aDiscTest)
  2145 	{
  2146 
  2147 	// create and install the active scheduler we need
  2148 	CActiveScheduler* scheduler=new(ELeave) CActiveScheduler;
  2149 	CleanupStack::PushL(scheduler);
  2150 
  2151 	CActiveScheduler::Install(scheduler);
  2152 
  2153 	aDiscTest->IdleScanningTimerRunErrorL();
  2154 
  2155 	// Cleanup CDiscovererTest, TheFs and scheduler
  2156 	CleanupStack::PopAndDestroy(scheduler);
  2157 
  2158 	}
  2159 /**
  2160 Thread entry point for the test thread.  Creates a CTrapCleanup and
  2161 calls  DoIdleScanningTimerRunErrorTestL to carry out the test
  2162 
  2163 @param		aDiscTest The CDiscovererTest object used to carry out the test
  2164 */
  2165 LOCAL_C TInt IdleScanningTimerRunErrorThreadEntry(TAny* aDiscTest)
  2166 	{
  2167 
  2168 	CTrapCleanup* tc = CTrapCleanup::New();
  2169 
  2170 	CDiscovererTest *discTest = static_cast<CDiscovererTest*>(aDiscTest);
  2171 
  2172 	TRAPD(err,DoIdleScanningTimerRunErrorTestL(discTest));
  2173 
  2174 	delete tc;
  2175 
  2176 	return err;
  2177 
  2178 	}
  2179 
  2180 
  2181 /**
  2182 @SYMTestCaseID		SYSLIB-ECOM-CT-3165
  2183 @SYMTestCaseDesc 	Check that the CIdleScanningTimer::RunError() works correctly.
  2184 @SYMTestPriority 	High
  2185 @SYMTestActions  	Create a new thread which will call RunError.  Wait for the
  2186 					thread to exit and check the thread exit type and reason
  2187 					to verify behaviour
  2188 @SYMTestExpectedResults The test must not fail.
  2189 @SYMDEF DEF094675
  2190 */
  2191 LOCAL_C void IdleScanningTimer_RunErrorTest()
  2192 	{
  2193 	__UHEAP_MARK;
  2194 
  2195 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-3165 CIdleScanningTimer RunError test "));
  2196 
  2197 	_LIT(KStartThreadName,"CIdleScanningTimer RunError Thread");
  2198 
  2199 	//Disable JIT so that the Panic doesn't bring up a dialog
  2200 	//and stop the test
  2201 	TBool jitEnabled = User::JustInTime();
  2202 	User::SetJustInTime(EFalse);
  2203 
  2204 	//Create a CDiscovererTest object to pass into the test thread
  2205 	CDiscovererTest* discTest = CDiscovererTest::NewL();
  2206 	CleanupStack::PushL(discTest);
  2207 
  2208 	//Create a new thread to run the test
  2209 	RThread testThread;
  2210 	testThread.Create(KStartThreadName, IdleScanningTimerRunErrorThreadEntry,
  2211 					KDefaultStackSize,KMinHeapSize,KMinHeapSize,discTest);
  2212 	TRequestStatus status;
  2213 	testThread.Logon(status);
  2214 
  2215 
  2216 	testThread.Resume();
  2217 
  2218 	//Wait for the thread to exit
  2219 	User::WaitForRequest(status);
  2220 
  2221 	//Obtain exit type and reason for test thread
  2222 	TExitType exitType = testThread.ExitType();
  2223 	TInt exitReason = testThread.ExitReason();
  2224 
  2225 	//close the thread handle
  2226 	testThread.Close();
  2227 
  2228 	CleanupStack::PopAndDestroy(discTest);
  2229 
  2230 	//Set JIT back to original state
  2231 	User::SetJustInTime(jitEnabled);
  2232 
  2233 	//Verify the exit reason and exit code
  2234 	test(exitType == EExitPanic);
  2235 	test(exitReason == EEComPanic_CDiscoverer_CIdleScanningTimer_RunError);
  2236 
  2237 	__UHEAP_MARKEND;
  2238 	}
  2239 
  2240 /**
  2241 Creates and installs active scheduler for this thread and calls
  2242 CDiscovererTest::DirChangeNotifierRunErrorL
  2243 
  2244 @param		aDiscTest The CDiscovererTest object used to carry out the test
  2245 */
  2246 LOCAL_C void DoDirChangeNotifierRunErrorTestL(CDiscovererTest* aDiscTest)
  2247 	{
  2248 
  2249 	// create and install the active scheduler we need
  2250 	CActiveScheduler* scheduler=new(ELeave) CActiveScheduler;
  2251 	CleanupStack::PushL(scheduler);
  2252 
  2253 	CActiveScheduler::Install(scheduler);
  2254 
  2255 	//call the RunErrorL method which should panic
  2256 	aDiscTest->DirChangeNotifierRunErrorL();
  2257 
  2258 	// Cleanup CDiscovererTest, TheFs and scheduler
  2259 	CleanupStack::PopAndDestroy(scheduler);
  2260 
  2261 	}
  2262 
  2263 /**
  2264 Thread entry point for the test thread.  Creates a CTrapCleanup and
  2265 calls  DoDirChangeNotifierRunErrorTestL to carry out the test
  2266 
  2267 @param		aDiscTest The CDiscovererTest object used to carry out the test
  2268 */
  2269 LOCAL_C TInt DirChangeNotifierRunErrorThreadEntry(TAny* aDiscTest)
  2270 	{
  2271 
  2272 	CTrapCleanup* tc = CTrapCleanup::New();
  2273 
  2274 	CDiscovererTest *discTest = static_cast<CDiscovererTest*>(aDiscTest);
  2275 
  2276 	TRAPD(err,DoDirChangeNotifierRunErrorTestL(discTest));
  2277 
  2278 	delete tc;
  2279 
  2280 	return err;
  2281 
  2282 	}
  2283 
  2284 /**
  2285 @SYMTestCaseID		SYSLIB-ECOM-CT-3166
  2286 @SYMTestCaseDesc 	Check that the CDirChangeNotifier::RunError() works correctly.
  2287 @SYMTestPriority 	High
  2288 @SYMTestActions  	Create a new thread which will call RunError.  Wait for the
  2289 					thread to exit and check the thread exit type and reason
  2290 					to verify behaviour
  2291 @SYMTestExpectedResults The test must not fail.
  2292 @SYMDEF DEF094675
  2293 */
  2294 LOCAL_C void DirChangeNotifier_RunErrorTest()
  2295 	{
  2296 	__UHEAP_MARK;
  2297 
  2298 	test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-3166 CDirChangeNotifier RunError test "));
  2299 
  2300 	_LIT(KStartThreadName,"CDirChangeNotifier RunError Thread");
  2301 
  2302 	//Disable JIT so that the Panic doesn't bring up a dialog
  2303 	//and stop the test
  2304 	TBool jitEnabled = User::JustInTime();
  2305 	User::SetJustInTime(EFalse);
  2306 
  2307 	//Create a CDiscovererTest object to pass into the test thread
  2308 	CDiscovererTest* discTest = CDiscovererTest::NewL();
  2309 	CleanupStack::PushL(discTest);
  2310 
  2311 	//Create a new thread to run the test
  2312 	RThread testThread;
  2313 	testThread.Create(KStartThreadName, DirChangeNotifierRunErrorThreadEntry,
  2314 					KDefaultStackSize,KMinHeapSize,KMinHeapSize,discTest);
  2315 	TRequestStatus status;
  2316 	testThread.Logon(status);
  2317 	testThread.Resume();
  2318 
  2319 	//Wait for the thread to exit
  2320 	User::WaitForRequest(status);
  2321 
  2322 	//Obtain exit type and reason for test thread
  2323 	TExitType exitType = testThread.ExitType();
  2324 	TInt exitReason = testThread.ExitReason();
  2325 
  2326 	//close the thread handle
  2327 	testThread.Close();
  2328 
  2329 	CleanupStack::PopAndDestroy(discTest);
  2330 
  2331 	//Set JIT back to original state
  2332 	User::SetJustInTime(jitEnabled);
  2333 
  2334 	//Verify the exit reason and exit code
  2335 	test(exitType == EExitPanic);
  2336 	test(exitReason == EEComPanic_CDiscoverer_CDirChangeNotifier_RunError);
  2337 
  2338 	__UHEAP_MARKEND;
  2339 	}
  2340 
  2341 TInt DoTestsL()
  2342 	{
  2343 	__UHEAP_MARK;
  2344 
  2345 	// Basic tests
  2346 	CreateDeleteTestL();
  2347 	DoBasicTestL(&CDiscovererTest::ResumeSuspendTestL, _L("ResumeSuspendTestL"));
  2348 	DoBasicTestL(&CDiscovererTest::DriveMountUnmountTestL, _L("DriveMountUnmountTestL"));
  2349 	DoBasicTestL(&CDiscovererTest::ProcessEntryTestL, _L("ProcessEntryTestL"));
  2350 	DoBasicTestL(&CDiscovererTest::ValidateEntryTestL, _L("ValidateEntryTestL"));
  2351 	DoBasicTestL(&CDiscovererTest::ProcessSpiEntryTestL, _L("ProcessSpiEntryTestL"));
  2352 	DoBasicTestL(&CDiscovererTest::ValidateSpiEntryTestL, _L("ValidateSpiEntryTestL"));
  2353 	DoBasicTestL(&CDiscovererTest::ProcessEntryPlugIn3TestL, _L("ProcessEntryPlugIn3TestL"));
  2354 	DoBasicTestL(&CDiscovererTest::ValidateEntryPlugIn3TestL, _L("ValidateEntryPlugIn3TestL"));
  2355 	DoBasicTestL(&CDiscovererTest::ProcessSpiEntryPlugIn3TestL, _L("ProcessSpiEntryPlugIn3TestL"));
  2356 	DoBasicTestL(&CDiscovererTest::ValidateSpiEntryPlugIn3TestL, _L("ValidateSpiEntryPlugIn3TestL"));
  2357 	DoBasicTestL(&CDiscovererTest::ValidateSpiPluginsTestL, _L("ValidateSpiPluginsTestL"));
  2358 	DoBasicTestL(&CDiscovererTest::ScanDirectoryIncrementTestL, _L("ScanDirectoryIncrementTestL"));
  2359 	DoBasicTestL(&CDiscovererTest::StagedDiscoveryStateTransitionTestL, _L("StagedDiscoveryStateTransitionTestL"));
  2360 	DoBasicTestL(&CDiscovererTest::AllAtOnceDiscoveryStateTransitionTestL, _L("AllAtOnceDiscoveryStateTransitionTestL"));
  2361 	DoBasicTestL(&CDiscovererTest::ScanDirectoryTestL, _L("ScanDirectoryTestL"));
  2362 	DoBasicTestL(&CDiscovererTest::MultipleNotificationProcessingTestL, _L("MultipleNotificationProcessingTestL"));
  2363 	DoBasicTestL(&CDiscovererTest::LanguageChangedNotificationTestL, _L("MLanguageChangedNotificationTestL"));
  2364 	DoBasicTestL(&CDiscovererTest::SWINotificationProcessingTestL, _L("SWINotificationProcessingTestL"));
  2365 
  2366 	//RunError tests
  2367 	IdleScanningTimer_RunErrorTest();
  2368 	DirChangeNotifier_RunErrorTest();
  2369 
  2370 	// OOM tests
  2371 	OOMCreateDeleteTest();
  2372 	// This test is not performed because the error from the final memory allocation failures are not
  2373 	// propagated back to the calling function and cannot be dealt with. Therefore
  2374 	// this test does not complete. However the normal test is performed above.
  2375 	//DoOOMTestL(&CDiscovererTest::ResumeSuspendTestL, _L("OOM ResumeSuspendTestL"));
  2376 	DoOOMTestL(&CDiscovererTest::DriveMountUnmountTestL, _L("OOM DriveMountUnmountTestL"));
  2377 	DoOOMTestL(&CDiscovererTest::ProcessEntryTestL, _L("OOM ProcessEntryTestL"));
  2378 	DoOOMTestL(&CDiscovererTest::ProcessSpiEntryTestL, _L("OOM ProcessSpiEntryTestL"));
  2379 	DoOOMTestL(&CDiscovererTest::ValidateSpiEntryTestL, _L("OOM ValidateSpiEntryTestL"));
  2380 	DoOOMTestL(&CDiscovererTest::ValidateEntryTestL, _L("OOM ValidateEntryTestL"));
  2381 	DoOOMTestL(&CDiscovererTest::ProcessEntryPlugIn3TestL, _L("OOM ProcessEntryPlugIn3TestL"));
  2382 	DoOOMTestL(&CDiscovererTest::ProcessSpiEntryPlugIn3TestL, _L("OOM ProcessSpiEntryPlugIn3TestL"));
  2383 	DoOOMTestL(&CDiscovererTest::ValidateSpiEntryPlugIn3TestL, _L("OOM ValidateSpiEntryPlugIn3TestL"));
  2384 	DoOOMTestL(&CDiscovererTest::ValidateEntryPlugIn3TestL, _L("OOM ValidateEntryPlugIn3TestL"));
  2385 	DoOOMTestL(&CDiscovererTest::ScanDirectoryIncrementTestL, _L("OOM ScanDirectoryIncrementTestL"));
  2386 	DoOOMTestL(&CDiscovererTest::StagedDiscoveryStateTransitionTestL, _L("StagedDiscoveryStateTransitionTestL"));
  2387 	DoOOMTestL(&CDiscovererTest::AllAtOnceDiscoveryStateTransitionTestL, _L("AllAtOnceDiscoveryStateTransitionTestL"));
  2388 	DoOOMTestL(&CDiscovererTest::ScanDirectoryTestL, _L("OOM ScanDirectoryTestL"));
  2389 	// This test is not performed because the error from the final memory allocation failures are not
  2390 	// propagated back to the calling function and cannot be dealt with. Therefore
  2391 	// this test does not complete. However the normal test is performed above.
  2392 	//DoOOMTestL(&CDiscovererTest::MultipleNotificationProcessingTestL, _L("MultipleNotificationProcessingTestL"));
  2393 	// This test is not performed because the error from the final memory allocation failures are not
  2394 	// propagated back to the calling function and cannot be dealt with. Therefore
  2395 	// this test does not complete. However the normal test is performed above.
  2396 	//DoOOMTestL(&CDiscovererTest::SWINotificationProcessingTestL, _L("OOM SWINotificationProcessingTestL"));
  2397 
  2398 	__UHEAP_MARKEND;
  2399 	return KErrNone;
  2400 	}
  2401 
  2402 // Copy the Plugins to specific folder for testing purpose
  2403 LOCAL_C void CopyPlugins()
  2404 	{
  2405 	TInt err=KErrNone;
  2406 	TRAP(err, EComTestUtils::FileManCopyFileL(KNewResourceFileNameOnZ, KNewResourceFileName));
  2407 	test(err==KErrNone);
  2408 	TRAP(err, EComTestUtils::FileManCopyFileL(KNewExampleDllFileNameOnZ, KNewExampleDllFileName));
  2409 	test(err==KErrNone);
  2410 	TRAP(err, EComTestUtils::FileManCopyFileL(KPlugIn3ResourceFileNameOnZ, KPlugIn3ResourceFileName));
  2411 	test(err==KErrNone);
  2412 	TRAP(err, EComTestUtils::FileManCopyFileL(KPlugIn3ExampleDllFileNameOnZ, KPlugIn3ExampleDllFileName));
  2413 	test(err==KErrNone);
  2414 	}
  2415 
  2416 // Deleting plugin from the RAM for cleanup purpose
  2417 inline LOCAL_C void DeleteTestPlugin()
  2418 	{
  2419 	TInt err=KErrNone;
  2420 	TRAP(err, EComTestUtils::FileManDeleteFileL(KNewResourceFileName));
  2421 	TRAP(err, EComTestUtils::FileManDeleteFileL(KNewExampleDllFileName));
  2422 	TRAP(err, EComTestUtils::FileManDeleteFileL(KPlugIn3ResourceFileName));
  2423 	TRAP(err, EComTestUtils::FileManDeleteFileL(KPlugIn3ExampleDllFileName));
  2424 	}
  2425 
  2426 //
  2427 //Initialise the Active Scheduler
  2428 //
  2429 LOCAL_C void SetupL()
  2430 	{
  2431 	// Construct and install the Active Scheduler. The Active Schedular is needed
  2432 	// by components used by this test as they are ActiveObjects.
  2433 	TheActiveScheduler = new(ELeave)CActiveScheduler;
  2434 	CActiveScheduler::Install(TheActiveScheduler);
  2435 	}
  2436 
  2437 GLDEF_C TInt E32Main()
  2438 	{
  2439 	__UHEAP_MARK;
  2440 
  2441 	test.Printf(_L("\n"));
  2442 	test.Title();
  2443 	test.Start(_L("Discoverer Tests."));
  2444 
  2445 	TheTrapCleanup = CTrapCleanup::New();
  2446 
  2447 	TRAPD(err, SetupL());
  2448 	test(err == KErrNone);
  2449 
  2450 	// Connect the file server instance
  2451 	User::LeaveIfError(TheFs.Connect());
  2452 
  2453 	CopyPlugins();
  2454 
  2455 	// The reason for the folowing delay is:
  2456 	// ECOM server could be already started. It means that when we copy some
  2457 	// ECOM plugins from Z: to C: drive - ECOM server should look for and
  2458 	// find the new ECOM plugins. The ECOM server uses for that an active object,
  2459 	// which scans plugin directories. So the discovering service is asynchronous.
  2460 	// We have to wait some time until it finishes.
  2461 	// Otherwise ListImplementationsL could fail to find requested implementations.
  2462 	User::After(KOneSecond * 3);
  2463 
  2464 
  2465 	// Call the main tests
  2466 	TRAP(err, DoTestsL());
  2467 	test(err == KErrNone);
  2468 
  2469 	// Cleanup files. If the cleanup fails that is no problem,
  2470 	// as any subsequent tests will replace them. The only downside
  2471 	// would be the disk not being tidied
  2472 	DeleteTestPlugin();
  2473 
  2474 	TheFs.Close();
  2475 
  2476 	delete TheActiveScheduler;
  2477 	delete TheTrapCleanup;
  2478 
  2479 	test.End();
  2480 	test.Close();
  2481 
  2482 	__UHEAP_MARKEND;
  2483 	return (KErrNone);
  2484 	}
  2485