sl@0: // Copyright (c) 2004-2009 Nokia Corporation and/or its subsidiary(-ies). sl@0: // All rights reserved. sl@0: // This component and the accompanying materials are made available sl@0: // under the terms of "Eclipse Public License v1.0" sl@0: // which accompanies this distribution, and is available sl@0: // at the URL "http://www.eclipse.org/legal/epl-v10.html". sl@0: // sl@0: // Initial Contributors: sl@0: // Nokia Corporation - initial contribution. sl@0: // sl@0: // Contributors: sl@0: // sl@0: // Description: sl@0: // This file contains test classes and their implementations sl@0: // to test production class CDiscoverer. Where necessary stubs sl@0: // are implemented to help in writing test harness using RTest. sl@0: // sl@0: // sl@0: sl@0: #include "Discoverer.h" sl@0: #include "EComErrorCodes.h" sl@0: #include "ImplementationInformation.h" sl@0: #include "DiscovererObserver.h" sl@0: #include "../EcomTestUtils/EcomTestUtils.h" sl@0: #include "EComUidCodes.h" sl@0: #include "RegistryData.h" sl@0: #include "DriveInfo.h" sl@0: sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include "baspi.h" sl@0: #include sl@0: sl@0: sl@0: //New RSC based plugins sl@0: _LIT(KNewResourceFileName, "C:\\resource\\plugins\\EComExample5.rsc"); sl@0: _LIT(KNewExampleDllFileName, "C:\\sys\\bin\\EComExample5.dll"); sl@0: _LIT(KNewResourceFileNameOnZ, "z:\\RAMOnly\\EComExample5.rsc"); sl@0: _LIT(KNewExampleDllFileNameOnZ, "z:\\RAMOnly\\EComExample5.dll"); sl@0: _LIT(KNewExampleDllFileNameOnly, "EComExample5.dll"); sl@0: _LIT(KNewResourceSearchPath, "\\resource\\plugins\\*"); sl@0: //PlugIn3 RSC based plugins sl@0: _LIT(KPlugIn3ResourceFileName, "C:\\resource\\plugins\\EComExample12.rsc"); sl@0: _LIT(KPlugIn3ExampleDllFileName, "C:\\sys\\bin\\EComExample12.dll"); sl@0: _LIT(KPlugIn3ResourceFileNameOnZ, "z:\\RAMOnly\\EComExample12.rsc"); sl@0: _LIT(KPlugIn3ExampleDllFileNameOnZ, "z:\\RAMOnly\\EComExample12.dll"); sl@0: _LIT(KPlugIn3ExampleDllFileNameOnly, "EComExample12.dll"); sl@0: sl@0: //Old DLL based plugins sl@0: sl@0: // spi test file sl@0: _LIT(KEComSpiTestFilePathAndName, "Z:\\Test\\Data\\EcomTest.spi"); sl@0: _LIT(KEComSpiTestFilePlugIn3PathAndName, "Z:\\Test\\Data\\EcomTestType3.spi"); sl@0: // file name detremined from example spi file sl@0: _LIT(KEComSpiTestFileDllPathAndName, "Z:\\sys\\bin\\EComExample5.dll"); sl@0: _LIT(KEComSpiTestFilePlugIn3DllPathAndName, "Z:\\sys\\bin\\EComExample12.dll"); sl@0: sl@0: // spi file sl@0: _LIT(KEComSPIFilePathAndName, "Z:\\private\\10009D8F\\Ecom.spi"); sl@0: sl@0: const TInt KOneSecond = 1000000; sl@0: sl@0: LOCAL_D RFs TheFs; sl@0: sl@0: LOCAL_D RTest test(_L("t_discoverer.exe")); sl@0: sl@0: LOCAL_D CTrapCleanup* TheTrapCleanup = NULL; sl@0: sl@0: LOCAL_D CActiveScheduler* TheActiveScheduler = NULL; sl@0: sl@0: // Used for supressing warning in OOM tests sl@0: #define __UNUSED_VAR(var) var = var sl@0: sl@0: // Used for OOM test sl@0: #define TEST_OOM_ERR if(err == KErrNoMemory) User::Leave(err) sl@0: sl@0: /** sl@0: Stub classes are provided to act as replacement members for the class sl@0: under test. This ensures the test class is constructed correctly and its sl@0: behaviour is also correct within the context of the test. It allows the sl@0: class to be tested in isolation. Other tests are available that test the sl@0: interoperability of the class sl@0: */ sl@0: class CObserverStub : public CBase, public MDiscovererObserver sl@0: { sl@0: public: sl@0: enum OSState sl@0: { sl@0: OSS_Undefined, sl@0: OSS_NoPlugins, sl@0: OSS_CriticalPlugins, sl@0: OSS_DiscoverNonCriticalPlugins, sl@0: OSS_AllPlugins sl@0: }; sl@0: static CObserverStub* NewL(); sl@0: virtual ~CObserverStub(); sl@0: sl@0: // Inherited from MDiscovererObserver sl@0: virtual void DiscoveriesBegin(); sl@0: virtual void RegisterDiscoveryL(const TDriveName& aDrive,CPluginBase*& aDirEntry,TBool aDatFileExists); sl@0: virtual void DiscoveriesComplete(TBool aSuccessful, TPluginProcessingTypeIdentifier aProcessingType); sl@0: virtual void DriveRemovedL(TDriveUnit aDrive); sl@0: virtual void DriveReinstatedL(TDriveUnit aDrive); sl@0: virtual void DriveIsSpiBasedL(const TDriveUnit& /*aDrive*/){} sl@0: virtual TBool NotifiedWithErrorCode(TInt aError); sl@0: virtual TBool IsAnyDllRegisteredWithDriveL(const TDriveUnit aDrive) const; sl@0: virtual void SetDiscoveryFlagL(const TDriveUnit &aDrive); sl@0: virtual void LanguageChangedL(TBool& aLanguageChanged); sl@0: sl@0: sl@0: // Result verifing methods sl@0: TBool IsDriveMounted(const TDriveUnit aDrive); sl@0: TBool IsEntryProcessed(CPluginBase*& aDirEntry); sl@0: //TBool IsSpiEntryProcessed(const TEntry& aEntry); sl@0: TBool IsDiscoveriesCompleteSuccessfully(); sl@0: void SetDiscoverer(CDiscoverer* aDiscoverer); sl@0: OSState GetState(); sl@0: sl@0: private: sl@0: CObserverStub(); sl@0: sl@0: private: sl@0: /** Base class ecom entry */ sl@0: CPluginBase* iEntryBase; sl@0: sl@0: /** Information of the dll which contains interface implementations */ sl@0: //TEComEntryDetails iEntry; sl@0: sl@0: /** Information of the spi file which contains interface implementations */ sl@0: //TEComSpiResourceEntryDetails iSpiEntry; sl@0: sl@0: /** iDriveUnit is drive mounted */ sl@0: TDriveUnit iDriveUnit; sl@0: sl@0: /** Flag to indicate whether iDriveUnit is mounted or not */ sl@0: TBool iDriveMounted; sl@0: sl@0: /** Flag to indicate whether iEntry is processed or not */ sl@0: TBool iEntryProcessed; sl@0: sl@0: /** Flag to indicate whether iSpiEntry is processed or not */ sl@0: TBool iSpiEntryProcessed; sl@0: sl@0: /** Flag to indicate whether Discovery process is Complete or not */ sl@0: TBool iDiscoveriesComplete; sl@0: sl@0: /** The current start-up state of this object */ sl@0: OSState iState; sl@0: sl@0: /** A reference to the CDiscoverer object that this class is observing*/ sl@0: CDiscoverer* iDiscoverer; sl@0: }; // End of CObserverStub declaration sl@0: sl@0: sl@0: /** sl@0: The TDiscoverer_StateAccessor class allows to access the private and protected sl@0: members of production code class CDiscoverer, as its a friend class. sl@0: */ sl@0: class TDiscoverer_StateAccessor sl@0: { sl@0: public: sl@0: sl@0: // Auxiliary functions that provide access to sl@0: // CDiscoverer private/protected members sl@0: void ScanDirectoryL(CDiscoverer& aDiscoverer, TDriveUnit aDriveUnit); sl@0: void ScanDirectoryCancel(CDiscoverer& aDiscoverer); sl@0: void CompleteNotificationProcessing(CDiscoverer& aDiscoverer); sl@0: void ValidateEntryL(CDiscoverer& aDiscoverer, sl@0: const TEntry& aEntry, sl@0: const TDriveName& aDriveName, sl@0: CPluginBase*& aEntryToFill, sl@0: TBool aIsRO); sl@0: void ValidateEntryL(CDiscoverer& aDiscoverer, sl@0: RResourceArchive& aRscArchive, sl@0: CPluginBase*& aEntryToFill); sl@0: void ProcessEntryL(CDiscoverer& aDiscoverer,const TDriveName& aDrive, CPluginBase*& aEntry, TBool aDatFileExists); sl@0: void DriveMountedL(CDiscoverer& aDiscoverer, const TDriveUnit aDrive); sl@0: void DriveUnmountedL(CDiscoverer& aDiscoverer, const TDriveUnit aDrive); sl@0: void SwiChangeNotificationL(CDiscoverer& aDiscoverer, TInt aSwiOperation); sl@0: void IdleScanningTimerRunErrorL(CDiscoverer& aDiscoverer, TInt aError); sl@0: void DirChangeNotifierRunErrorL(CDiscoverer& aDiscoverer, TInt aError); sl@0: void SwiChangeNotifierRunError(CDiscoverer& aDiscoverer, TInt aError); sl@0: sl@0: // Verification functions to check the state of CDiscoverer object against tests sl@0: TBool IsDriveMounted(CDiscoverer& aDiscoverer, const TDriveUnit aDrive); sl@0: TBool IsDiscovererActive(CDiscoverer& aDiscoverer); sl@0: TBool IsDirectoryScanCancelled(CDiscoverer& aDiscoverer); sl@0: TBool IsScanDirectoryComplete(CDiscoverer& aDiscoverer); sl@0: TBool IsScanDirectoryPending(CDiscoverer& aDiscoverer, TDriveUnit aDrive); sl@0: TUint PendingDriveListCount(CDiscoverer& aDiscoverer); sl@0: CDiscoverer::TDiscovererState State(CDiscoverer& aDiscoverer); sl@0: CObserverStub::OSState GetDiscovererObserverState(CDiscoverer& aDiscoverer); sl@0: void ScanDriveL(CDiscoverer& aDiscoverer, TDriveUnit aDrive, TBool aIsRO); sl@0: void LanguageChangedL(CDiscoverer& aDiscoverer); sl@0: }; sl@0: sl@0: /** sl@0: Scans plugin directories sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @return void sl@0: */ sl@0: void TDiscoverer_StateAccessor::ScanDirectoryL(CDiscoverer& aDiscoverer, TDriveUnit aDriveUnit) sl@0: { sl@0: aDiscoverer.RediscoveryScanDirectoryL(aDriveUnit); sl@0: } sl@0: sl@0: /** sl@0: Stops scanning of the plugin directories sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: */ sl@0: void TDiscoverer_StateAccessor::ScanDirectoryCancel(CDiscoverer& aDiscoverer) sl@0: { sl@0: aDiscoverer.ScanDirectoryCancel(); sl@0: } sl@0: sl@0: /** sl@0: Signals that the directory change has been fully processed. sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: */ sl@0: void TDiscoverer_StateAccessor::CompleteNotificationProcessing(CDiscoverer& aDiscoverer) sl@0: { sl@0: aDiscoverer.CompleteNotificationProcessing(); sl@0: } sl@0: sl@0: /** sl@0: Verifies that the discovered entry is valid. sl@0: Used to test the Interface Implementation Collection entry sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @param aEntry The plugin name sl@0: @param aDriveName The drive containing the entry sl@0: @param aEntryToFill On return points to complete sl@0: plugin file name(with path) i.e. aPath + aEntry sl@0: */ sl@0: void TDiscoverer_StateAccessor::ValidateEntryL(CDiscoverer& aDiscoverer, sl@0: const TEntry& aEntry, sl@0: const TDriveName& aDriveName, sl@0: CPluginBase*& aEntryToFill, sl@0: TBool aIsRO) sl@0: { sl@0: aDiscoverer.ValidateEntryL(aEntry, aDriveName, aEntryToFill, aIsRO); sl@0: } sl@0: sl@0: /** sl@0: Verifies that the discovered entry is valid. sl@0: Used to test the Interface Implementation Collection entry sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @param aRscArchive reference to the resource archive sl@0: @param aEntryToFill On return points to complete sl@0: plugin file name(with path) i.e. aPath + aEntry sl@0: */ sl@0: void TDiscoverer_StateAccessor::ValidateEntryL(CDiscoverer& aDiscoverer, sl@0: RResourceArchive& aRscArchive, sl@0: CPluginBase*& aEntryToFill) sl@0: { sl@0: aDiscoverer.ValidateEntryL(aRscArchive,aEntryToFill); sl@0: } sl@0: /** sl@0: Registers an Interface Implementation Collection sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @param aEntry This is the plugin name that needs to be registered sl@0: */ sl@0: void TDiscoverer_StateAccessor::ProcessEntryL(CDiscoverer& aDiscoverer,const TDriveName& aDrive, sl@0: CPluginBase*& aEntry, TBool aDatFileExists) sl@0: { sl@0: aDiscoverer.ProcessEntryL(aDrive,aEntry,aDatFileExists); sl@0: } sl@0: sl@0: /** sl@0: Notifies the discoverer of an SWI operation change sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @param aSwiOperation This new SWI state sl@0: */ sl@0: void TDiscoverer_StateAccessor::SwiChangeNotificationL(CDiscoverer& aDiscoverer, TInt aSwiOperation) sl@0: { sl@0: aDiscoverer.SwiChangeNotificationL(aSwiOperation); sl@0: } sl@0: sl@0: /** sl@0: Signals that a drive is available sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @param aDrive Drive that needs to be mounted sl@0: */ sl@0: void TDiscoverer_StateAccessor::DriveMountedL(CDiscoverer& aDiscoverer, sl@0: const TDriveUnit aDrive) sl@0: { sl@0: aDiscoverer.DriveMountedL(aDrive); sl@0: } sl@0: sl@0: /** sl@0: Signals that a drive is unavailable sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @param aDrive Drive that needs to be unmounted/removed sl@0: */ sl@0: void TDiscoverer_StateAccessor::DriveUnmountedL(CDiscoverer& aDiscoverer, sl@0: const TDriveUnit aDrive) sl@0: { sl@0: aDiscoverer.DriveUnmountedL(aDrive); sl@0: } sl@0: sl@0: /** sl@0: Checks whether CDiscoverer object has successfully completed with sl@0: the scanning of the plugin directories on all drives sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @return true if notification has been processed on all drives. sl@0: false if there is notification processing pending on any drive sl@0: */ sl@0: TBool TDiscoverer_StateAccessor::IsScanDirectoryComplete(CDiscoverer& aDiscoverer) sl@0: { sl@0: // check state of discoverer to see if all pending drives have been scanned. sl@0: return(aDiscoverer.State() == CDiscoverer::EDisc_AllPluginsDisc); sl@0: } sl@0: sl@0: /** sl@0: Checks whether a notification has been signaled but not processed on specified drive. sl@0: sl@0: Error Condition : Leaves with KErrNotFound if the drive is not registered. sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @param aDrive The drive to check sl@0: @return true if notification has been processed on the specified drive. sl@0: false if notification processing is pending sl@0: */ sl@0: TBool TDiscoverer_StateAccessor::IsScanDirectoryPending(CDiscoverer& aDiscoverer, TDriveUnit aDrive) sl@0: { sl@0: if(aDiscoverer.iScanningTimer->iPendingDriveList.Find(aDrive)!=KErrNotFound) sl@0: { sl@0: return ETrue; sl@0: } sl@0: else sl@0: { sl@0: return EFalse; sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Checks the current count of drives in the list waiting for processing. sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @return TUint sl@0: */ sl@0: TUint TDiscoverer_StateAccessor::PendingDriveListCount(CDiscoverer& aDiscoverer) sl@0: { sl@0: return(aDiscoverer.iScanningTimer->iPendingDriveList.Count()); sl@0: } sl@0: sl@0: /** sl@0: Checks that CDiscoverer object is not scanning the plugin directories. sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @return true if plugin dirctory scanning is stopped. sl@0: */ sl@0: TBool TDiscoverer_StateAccessor::IsDirectoryScanCancelled(CDiscoverer& aDiscoverer) sl@0: { sl@0: if(!aDiscoverer.iDirScanner) sl@0: { sl@0: return ETrue; sl@0: } sl@0: return EFalse; sl@0: } sl@0: sl@0: /** sl@0: Checks that CDiscoverer object is currently activated for the plugin dir scanning sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @return true if Discoverer object is tracking changes in the plugin directories in each drive. sl@0: */ sl@0: TBool TDiscoverer_StateAccessor::IsDiscovererActive(CDiscoverer& aDiscoverer) sl@0: { sl@0: //Return ETrue if both iScanningTimer & directory notifier(s) are active sl@0: TBool isActive = ETrue; sl@0: sl@0: for(TInt index = 0; index IsActive())) sl@0: { sl@0: // iRscDirNotifierList should be active, as each active object on list is constantly sl@0: // looking for changes in the plugin directories in each drive. sl@0: isActive = EFalse; sl@0: break; sl@0: } sl@0: } sl@0: return isActive; sl@0: } sl@0: sl@0: /** sl@0: Verifies whether given drive is registered sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @param aDrive Drive that needs to be verified for registration sl@0: @return return true if given drive is installed sl@0: */ sl@0: TBool TDiscoverer_StateAccessor::IsDriveMounted(CDiscoverer& aDiscoverer, sl@0: const TDriveUnit aDrive) sl@0: { sl@0: TBool isMounted = ETrue; sl@0: if(aDiscoverer.iDrivesDiscovered.Find(aDrive) == KErrNotFound) sl@0: { sl@0: isMounted = EFalse; sl@0: } sl@0: return isMounted; sl@0: } sl@0: sl@0: /** sl@0: Scans plugin directories sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @param aDrive The drive which is scanned sl@0: @param aIsRO Whether the drive is ready-only sl@0: @return true if plugin directorie in each drive get scaned successfully sl@0: */ sl@0: void TDiscoverer_StateAccessor::ScanDriveL(CDiscoverer& aDiscoverer, TDriveUnit aDrive, TBool aIsRO) sl@0: { sl@0: return aDiscoverer.iDirScanner->ScanDriveL(aDrive, aIsRO); sl@0: } sl@0: void TDiscoverer_StateAccessor::LanguageChangedL(CDiscoverer& aDiscoverer) sl@0: { sl@0: aDiscoverer.LanguageChangeNotificationL(); sl@0: } sl@0: /** sl@0: Retrieve the object's current state. sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @return TDiscovererState the current state of the CDiscoverer sl@0: class object under test sl@0: */ sl@0: CDiscoverer::TDiscovererState TDiscoverer_StateAccessor::State(CDiscoverer& aDiscoverer) sl@0: { sl@0: return aDiscoverer.State(); sl@0: } sl@0: sl@0: CObserverStub::OSState TDiscoverer_StateAccessor::GetDiscovererObserverState(CDiscoverer& aDiscoverer) sl@0: { sl@0: CObserverStub* discovererObserver = sl@0: static_cast(&aDiscoverer.iDiscovererObserver); sl@0: return discovererObserver->GetState(); sl@0: } sl@0: sl@0: /** sl@0: Call the CIdleScanningTimer RunError funtion sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @param aError The error code to pass to the RunError function sl@0: */ sl@0: void TDiscoverer_StateAccessor::IdleScanningTimerRunErrorL(CDiscoverer& aDiscoverer, TInt aError) sl@0: { sl@0: aDiscoverer.iScanningTimer->RunError(aError); sl@0: } sl@0: sl@0: /** sl@0: Call the CDirChangeNotifier RunError funtion sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @param aError The error code to pass to the RunError function sl@0: */ sl@0: void TDiscoverer_StateAccessor::DirChangeNotifierRunErrorL(CDiscoverer& aDiscoverer, TInt aError) sl@0: { sl@0: aDiscoverer.iRscDirNotifierList[0]->RunError(aError); sl@0: } sl@0: sl@0: /** sl@0: Call the CSwiChangeNotifier RunError funtion sl@0: sl@0: @param aDiscoverer The CDiscoverer class object under test sl@0: @param aError The error code to pass to the RunError function sl@0: */ sl@0: void TDiscoverer_StateAccessor::SwiChangeNotifierRunError(CDiscoverer& aDiscoverer, TInt aError) sl@0: { sl@0: aDiscoverer.iSwiChangeNotifier->RunError(aError); sl@0: } sl@0: sl@0: /** sl@0: Creates an instance of CObserverStub class. sl@0: sl@0: @return The new'ed object. sl@0: */ sl@0: CObserverStub* CObserverStub::NewL() sl@0: { sl@0: return new(ELeave) CObserverStub(); sl@0: } sl@0: sl@0: /** sl@0: Destructor sl@0: sl@0: @post This object is properly destroyed. sl@0: */ sl@0: CObserverStub::~CObserverStub() sl@0: { sl@0: // do nothing; sl@0: } sl@0: sl@0: /** sl@0: Default constructor. sl@0: sl@0: @post The object is properly constructed. sl@0: */ sl@0: CObserverStub::CObserverStub() sl@0: :CBase(), sl@0: iDriveUnit(EDriveC), sl@0: iDriveMounted(EFalse), sl@0: iEntryProcessed(EFalse), sl@0: iSpiEntryProcessed(EFalse), sl@0: iDiscoveriesComplete(EFalse), sl@0: iState(OSS_NoPlugins) sl@0: { sl@0: // do nothing; sl@0: } sl@0: sl@0: /** sl@0: MDiscovererObserver callback method, to signal that a discovery sl@0: session is starting. sl@0: */ sl@0: void CObserverStub::DiscoveriesBegin() sl@0: { sl@0: iDiscoveriesComplete = EFalse; sl@0: } sl@0: sl@0: void CObserverStub::SetDiscoveryFlagL(const TDriveUnit & /*aDrive*/) sl@0: { sl@0: iDiscoveriesComplete = ETrue; sl@0: } sl@0: /** sl@0: MDiscovererObserver callback method to register an Ecom Spi file. sl@0: sl@0: @param aEntry The Ecom file that needs to be registered sl@0: */ sl@0: void CObserverStub::RegisterDiscoveryL(const TDriveName& /* aDriveName */,CPluginBase*& aEntry,TBool /* aDatFileExists*/) sl@0: { sl@0: iEntryBase = aEntry; sl@0: iSpiEntryProcessed = ETrue; sl@0: } sl@0: sl@0: /** sl@0: MDiscovererObserver callback method, to signal that a discovery sl@0: session is complete. sl@0: sl@0: @param aSuccessful Indicates discoveries process completed successfully or not sl@0: @param aProcessingType indicates the type of processing for plugins sl@0: for ensuring that plugins are not processed multiple times sl@0: during start-up phase sl@0: */ sl@0: void CObserverStub::DiscoveriesComplete(TBool aSuccessful, TPluginProcessingTypeIdentifier aProcessingType) sl@0: { sl@0: iDiscoveriesComplete = aSuccessful; sl@0: sl@0: switch(iState) sl@0: { sl@0: case OSS_NoPlugins: sl@0: if(aProcessingType == EPluginProcessingTypeCriticalOnly && sl@0: iDiscoverer->State() == CDiscoverer::EDisc_CriticalPluginsDisc) sl@0: { sl@0: iState = OSS_CriticalPlugins; sl@0: } sl@0: else if(aProcessingType == EPluginProcessingTypeAll && sl@0: iDiscoverer->State() == CDiscoverer::EDisc_AllPluginsDisc) sl@0: { sl@0: iState = OSS_AllPlugins; sl@0: } sl@0: break; sl@0: case OSS_CriticalPlugins: sl@0: if(aProcessingType == EPluginProcessingTypeNonCriticalOnly && sl@0: iDiscoverer->State() == CDiscoverer::EDisc_AllPluginsDisc) sl@0: { sl@0: iState = OSS_AllPlugins; sl@0: } sl@0: break; sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Verifies whether given plugin entry is registered sl@0: sl@0: @param aDirEntry The Ecom plugin that to be checked for registration. sl@0: @return true if given plugin is registered sl@0: */ sl@0: TBool CObserverStub::IsEntryProcessed(CPluginBase*& aDirEntry) sl@0: { sl@0: if(iEntryProcessed && (iEntryBase->iDllThirdUid == aDirEntry->iDllThirdUid)) sl@0: { sl@0: return ETrue; sl@0: } sl@0: sl@0: return EFalse; sl@0: } sl@0: sl@0: /** sl@0: MDiscovererObserver callback method, to signal that a Drive sl@0: is removed/dismounted. sl@0: sl@0: @param aDrive The drive that is removed. sl@0: */ sl@0: void CObserverStub::DriveRemovedL(TDriveUnit aDrive) sl@0: { sl@0: iDriveMounted = EFalse; sl@0: iDriveUnit = aDrive; sl@0: } sl@0: sl@0: /** sl@0: MDiscovererObserver callback method, to signal that a Drive sl@0: is reinstalled. sl@0: sl@0: @param aDrive The drive that is available now. sl@0: */ sl@0: void CObserverStub::DriveReinstatedL(TDriveUnit aDrive) sl@0: { sl@0: iDriveMounted = ETrue; sl@0: iDriveUnit = aDrive; sl@0: } sl@0: sl@0: /** sl@0: MDiscovererObserver callback method, to signal that sl@0: during a discovery session an error has occured. sl@0: sl@0: @param aError The notification error code. sl@0: @return true if aError is one of the acceptable error codes sl@0: */ sl@0: TBool CObserverStub::NotifiedWithErrorCode(TInt aError) sl@0: { sl@0: // Test the safe error codes sl@0: return (aError == KErrNotReady || // Drive removed sl@0: aError == KErrPathNotFound); // Directory deleted sl@0: } sl@0: sl@0: /** sl@0: MDiscovererObserver callback method,to retrieve the drive unit's DAT sl@0: file infor. sl@0: sl@0: @param aDrive the identifier of the drive to retrieve DAT file infor from. sl@0: @return ETrue if DAT file exists on the drive unit, otherwise EFlase. sl@0: sl@0: */ sl@0: TBool CObserverStub::IsAnyDllRegisteredWithDriveL(const TDriveUnit /*aDrive*/) const sl@0: { sl@0: // To pass the build always returns EFalse to mean no Dll is discovered in the drive, sl@0: // it is not used in this test. sl@0: return EFalse; sl@0: } sl@0: sl@0: /** sl@0: Verification method that checks whether given drive is available sl@0: sl@0: @param aDrive The drive that to be checked. sl@0: @return true if given drive is registered sl@0: */ sl@0: TBool CObserverStub::IsDriveMounted(const TDriveUnit aDrive) sl@0: { sl@0: TBool mounted = EFalse; sl@0: if(iDriveMounted && (aDrive == iDriveUnit)) sl@0: { sl@0: mounted = ETrue; sl@0: } sl@0: return mounted; sl@0: } sl@0: sl@0: /** sl@0: Verification method that checks whether discovery process is completed successfully sl@0: sl@0: @return true if discovery process is completed successfully sl@0: */ sl@0: TBool CObserverStub::IsDiscoveriesCompleteSuccessfully() sl@0: { sl@0: return iDiscoveriesComplete; sl@0: } sl@0: sl@0: /** sl@0: Returns the current start-up state of this object sl@0: sl@0: @return The current start-up state of this object sl@0: */ sl@0: CObserverStub::OSState CObserverStub::GetState() sl@0: { sl@0: return iState; sl@0: } sl@0: sl@0: void CObserverStub::SetDiscoverer(CDiscoverer* aDiscoverer) sl@0: { sl@0: iDiscoverer = aDiscoverer; sl@0: } sl@0: sl@0: void CObserverStub::LanguageChangedL(TBool& aLanguageChanged) sl@0: { sl@0: aLanguageChanged = EFalse; sl@0: } sl@0: class CDiscovererTestShutdown : public CTimer sl@0: { sl@0: public: sl@0: inline CDiscovererTestShutdown(); sl@0: inline void ConstructL(); sl@0: inline void StartAfter(TTimeIntervalMicroSeconds32 aTimeInterval); sl@0: sl@0: private: sl@0: void RunL(); sl@0: }; sl@0: sl@0: inline CDiscovererTestShutdown::CDiscovererTestShutdown() sl@0: :CTimer(EPriorityHigh)// Priority set higher than CIdleScanningTimer sl@0: { sl@0: CActiveScheduler::Add(this); sl@0: } sl@0: sl@0: inline void CDiscovererTestShutdown::ConstructL() sl@0: { sl@0: CTimer::ConstructL(); sl@0: } sl@0: sl@0: inline void CDiscovererTestShutdown::StartAfter(TTimeIntervalMicroSeconds32 aTimeInterval) sl@0: { sl@0: After(aTimeInterval); sl@0: } sl@0: void CDiscovererTestShutdown::RunL() sl@0: { sl@0: CActiveScheduler::Stop(); sl@0: } sl@0: sl@0: /** sl@0: Test class for object CDiscoverer. sl@0: This class provides the parameters and behaviour that sl@0: allows this class to behave normally under a test sl@0: scenario. sl@0: */ sl@0: class CDiscovererTest : public CBase sl@0: { sl@0: public: sl@0: static CDiscovererTest* NewL(); sl@0: virtual ~CDiscovererTest(); sl@0: sl@0: void ResumeSuspendTestL(); sl@0: void DriveMountUnmountTestL(); sl@0: void ProcessEntryTestL(); sl@0: void ProcessSpiEntryTestL(); sl@0: void ValidateSpiEntryTestL(); sl@0: void ValidateSpiPluginsTestL(); sl@0: void ValidateEntryTestL(); sl@0: void ProcessEntryPlugIn3TestL(); sl@0: void ProcessSpiEntryPlugIn3TestL(); sl@0: void ValidateSpiEntryPlugIn3TestL(); sl@0: void ValidateEntryPlugIn3TestL(); sl@0: void ScanDirectoryIncrementTestL(); sl@0: void ScanDirectoryTestL(); sl@0: void ScanDirectoryCancelTestL(); sl@0: void StagedDiscoveryStateTransitionTestL(); sl@0: void AllAtOnceDiscoveryStateTransitionTestL(); sl@0: void MultipleNotificationProcessingTestL(); sl@0: void SWINotificationProcessingTestL(); sl@0: void IdleScanningTimerRunErrorL(); sl@0: void DirChangeNotifierRunErrorL(); sl@0: void SwiChangeNotifierRunError(); sl@0: void LanguageChangedNotificationTestL(); sl@0: sl@0: private: sl@0: CDiscovererTest(); sl@0: void ConstructL(); sl@0: sl@0: public: sl@0: /** The instance of the class under test */ sl@0: CDiscoverer* iDiscoverer; sl@0: sl@0: /** The instance of the state accessor interface */ sl@0: TDiscoverer_StateAccessor* iStateAccessor; sl@0: sl@0: /** The instance of the observer stub of the CDiscoverer */ sl@0: CObserverStub* iDiscovererObserver; sl@0: sl@0: /** Unique Id of the ECOM dll */ sl@0: TUid iDllUid; sl@0: sl@0: /** The drive on which interface implementations can be found */ sl@0: TDriveUnit iDriveUnit; sl@0: sl@0: /** Information on a dll which contains interface implementations */ sl@0: TEntry iDllEntry; sl@0: sl@0: /** A shutdown timer to manipulate the Active Scheduler of testing server */ sl@0: CDiscovererTestShutdown iShutdown; sl@0: sl@0: CEComCachedDriveInfo* iCachedDriveInfo; sl@0: }; sl@0: sl@0: /** sl@0: Standardised safe construction which sl@0: leaves nothing on the cleanup stack. sl@0: sl@0: @post CDiscovererTest is fully constructed and initialised sl@0: @return The new'ed object. sl@0: */ sl@0: CDiscovererTest* CDiscovererTest::NewL() sl@0: { sl@0: CDiscovererTest* self = new (ELeave) CDiscovererTest(); sl@0: CleanupStack::PushL( self ); sl@0: self->ConstructL(); sl@0: CleanupStack::Pop(self); sl@0: return self; sl@0: } sl@0: sl@0: /** sl@0: Default constructor. sl@0: sl@0: @post The object is properly constructed. sl@0: */ sl@0: CDiscovererTest::CDiscovererTest() sl@0: :CBase(), sl@0: iDriveUnit(EDriveC) sl@0: { sl@0: TUid uid1 = {0}; sl@0: TUid uid2 = {0}; sl@0: sl@0: iDllUid.iUid = 0x10009DB6; // Dll Uid for EComRomOnlyExampleOnC.dll sl@0: iDllEntry.iType = TUidType(uid1, uid2, iDllUid); sl@0: } sl@0: sl@0: /** sl@0: Destructor. sl@0: sl@0: @post This object is properly destroyed. sl@0: */ sl@0: CDiscovererTest::~CDiscovererTest() sl@0: { sl@0: delete iDiscoverer; sl@0: delete iStateAccessor; sl@0: delete iDiscovererObserver; sl@0: delete iCachedDriveInfo; sl@0: } sl@0: sl@0: /** sl@0: Standardized 2nd(Initialization) phase of two phase construction. sl@0: Creates supporting class objects for the execution of test. sl@0: sl@0: @post CDiscovererTest is fully constructed. sl@0: */ sl@0: void CDiscovererTest::ConstructL() sl@0: { sl@0: iStateAccessor = new(ELeave) TDiscoverer_StateAccessor; sl@0: iDiscovererObserver = CObserverStub::NewL(); sl@0: iDiscoverer = CDiscoverer::NewL(*iDiscovererObserver, TheFs); sl@0: iDiscovererObserver->SetDiscoverer(iDiscoverer); sl@0: iShutdown.ConstructL(); sl@0: iCachedDriveInfo = CEComCachedDriveInfo::NewL(TheFs); sl@0: } sl@0: sl@0: /** sl@0: The test executes by first suspending the discoverer then by sl@0: resuming it. sl@0: sl@0: @SYMTestCaseID SYSLIB-ECOM-CT-0716 sl@0: @SYMTestCaseDesc Tests for resuming and suspending Discoverer sl@0: @SYMTestPriority High sl@0: @SYMTestActions Suspend if Discoverer is active and then resume. Check for the error conditions(OOM) sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMREQ REQ0000 sl@0: */ sl@0: void CDiscovererTest::ResumeSuspendTestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0716 ")); sl@0: // Perform an initial discovery and start the notifiers. sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical); sl@0: sl@0: TBool isDiscActive = iStateAccessor->IsDiscovererActive(*iDiscoverer); sl@0: // Notifiers should be active. sl@0: test(isDiscActive); sl@0: sl@0: TInt err = iDiscoverer->Suspend(); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: sl@0: // Notifiers should still be active. sl@0: isDiscActive = iStateAccessor->IsDiscovererActive(*iDiscoverer); sl@0: test(isDiscActive); sl@0: sl@0: // Test that there are no drives pending a scan before we copy the plugin. sl@0: test(!iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveC)); sl@0: sl@0: // Raise a plugin change notification on C: drive. This simulates a plugin being copied. sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveC); sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveC); sl@0: sl@0: // Test that the drive we copied the plugin to still has a scan pending. sl@0: test(iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveC)); sl@0: sl@0: isDiscActive = iStateAccessor->IsDiscovererActive(*iDiscoverer); sl@0: // Notifiers should still be active. sl@0: test(isDiscActive); sl@0: sl@0: // Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process sl@0: // all pending notification. sl@0: iShutdown.StartAfter(KOneSecond * 3); sl@0: CActiveScheduler::Start(); sl@0: sl@0: // Check that notifications have not been processed. sl@0: test(!iStateAccessor->IsScanDirectoryComplete(*iDiscoverer)); sl@0: sl@0: err = iDiscoverer->Resume(); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: sl@0: isDiscActive = iStateAccessor->IsDiscovererActive(*iDiscoverer); sl@0: // Notifiers should still be active. sl@0: test(isDiscActive); sl@0: sl@0: // Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process sl@0: // all pending notification. sl@0: iShutdown.StartAfter(KOneSecond * 3); sl@0: CActiveScheduler::Start(); sl@0: sl@0: // Check if notifications have been processed on all drives. sl@0: test(iStateAccessor->IsScanDirectoryComplete(*iDiscoverer)); sl@0: } sl@0: sl@0: /** sl@0: The test executes by first making the test drive unmounted and sl@0: then testing for Mount and Unmount. sl@0: sl@0: @SYMTestCaseID SYSLIB-ECOM-CT-0717 sl@0: @SYMTestCaseDesc Tests for drive mount and unmount sl@0: @SYMTestPriority High sl@0: @SYMTestActions First remove the drive if its already there, then test for Mount then for unmount sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMREQ REQ0000 sl@0: */ sl@0: void CDiscovererTest::DriveMountUnmountTestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0717 ")); sl@0: TInt err; sl@0: TDriveUnit drive(EDriveC); sl@0: //First remove the drive if its already there, then test for Mount then for unmount sl@0: if(iStateAccessor->IsDriveMounted(*iDiscoverer, drive)) sl@0: { sl@0: TRAP(err, iStateAccessor->DriveUnmountedL(*iDiscoverer, drive)); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: } sl@0: sl@0: // Drive should not be present sl@0: test(!iStateAccessor->IsDriveMounted(*iDiscoverer, drive)); sl@0: sl@0: // Test Mount sl@0: TRAP(err, iStateAccessor->DriveMountedL(*iDiscoverer, drive)); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: sl@0: // CDiscoverer->DriveMountedL/DriveUnmountedL updates both itself and sl@0: // DiscovererObserver class for Mount/Unmount of drives. So testing for both sl@0: test(iStateAccessor->IsDriveMounted(*iDiscoverer, drive)); sl@0: test(iDiscovererObserver->IsDriveMounted(drive)); sl@0: sl@0: // Test Unmount sl@0: TRAP(err, iStateAccessor->DriveUnmountedL(*iDiscoverer, drive)); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone ); sl@0: sl@0: // CDiscoverer->DriveMountedL/DriveUnmountedL updates both itself and sl@0: // DiscovererObserver class for Mount/Unmount of drives. So testing for both sl@0: test(!iStateAccessor->IsDriveMounted(*iDiscoverer, drive)); sl@0: test(!iDiscovererObserver->IsDriveMounted(drive)); sl@0: } sl@0: sl@0: sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-UT-3559 sl@0: @SYMTestCaseDesc Tests for process entry of PLUGIN3 type. sl@0: @SYMTestPriority High sl@0: @SYMTestActions Call CDiscoverer::ProcessEntryL(). sl@0: @SYMTestExpectedResults The entry is registered successfully and no leave occurred. sl@0: @SYMEC EC43 sl@0: */ sl@0: void CDiscovererTest::ProcessEntryPlugIn3TestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-3559 ")); sl@0: CPluginBase* entryBase = NULL; sl@0: TInt err = KErrNone; sl@0: sl@0: //This part refers to testing of the entry containing the RSC sl@0: _LIT(KExampleRscFileNameOnly,"EComExample12.rsc"); sl@0: TDriveName driveName1(iDriveUnit.Name()); sl@0: TParse path1; sl@0: path1.Set(KNewResourceSearchPath(),NULL,&driveName1); sl@0: //Creating a entry that represents the Rsc disovered during scanning of \\resource\\plugins sl@0: TEntry rscEntry; sl@0: rscEntry.iName = KExampleRscFileNameOnly; sl@0: rscEntry.iType = TUidType(KNullUid,KUidInterfaceImplementationCollectionInfo,KNullUid); sl@0: entryBase = CSecurePlugin::NewL(TheFs,rscEntry,driveName1, EFalse); sl@0: CleanupStack::PushL(entryBase); sl@0: sl@0: // ProcessEntryL() updates entryToFill with information on a rsc specified by the other params. sl@0: TRAP(err, iStateAccessor->ProcessEntryL(*iDiscoverer,driveName1,entryBase,ETrue)); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: CleanupStack::PopAndDestroy(entryBase); sl@0: } sl@0: sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-UT-3560 sl@0: @SYMTestCaseDesc Check that the ProcessEntryL for spi data works correctly when using PLUGIN3 entries in the spi file. sl@0: @SYMTestPriority High sl@0: @SYMTestActions Call CDiscoverer::ProcessEntryL(). sl@0: @SYMTestExpectedResults ecomtesttype3.spi is processed successfully and no leave occurred. sl@0: @SYMEC EC43 sl@0: */ sl@0: void CDiscovererTest::ProcessSpiEntryPlugIn3TestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-3560 ")); sl@0: TEntry spiEntry; sl@0: test(TheFs.Entry(KEComSpiTestFilePlugIn3PathAndName, spiEntry) == KErrNone); sl@0: TParse spiPath; sl@0: spiPath.Set(KEComSpiTestFilePlugIn3PathAndName, NULL, NULL); sl@0: sl@0: RResourceArchive resourceArchive; sl@0: resourceArchive.OpenL(TheFs, KEComSpiTestFilePlugIn3PathAndName); sl@0: CleanupClosePushL(resourceArchive); sl@0: sl@0: // check SPI file type sl@0: TUid type = resourceArchive.Type(); sl@0: test(type == KEcomSpiFileTypeUid); sl@0: sl@0: // there is only 1 resource file sl@0: while(!resourceArchive.End()) sl@0: { sl@0: CPluginBase* entry = CSpiPlugin::NewL(resourceArchive); sl@0: CleanupStack::PushL(entry); sl@0: sl@0: TRAPD(err, iStateAccessor->ProcessEntryL(*iDiscoverer,spiPath.Drive(), entry,ETrue)); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: sl@0: CleanupStack::PopAndDestroy(entry); sl@0: entry = NULL; sl@0: } sl@0: CleanupStack::PopAndDestroy(&resourceArchive); sl@0: } sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-UT-3561 sl@0: @SYMTestCaseDesc Tests for the reference of the entry containing the DLL of PLUGIN3 type. sl@0: @SYMTestPriority High sl@0: @SYMTestActions Call CDiscoverer::ValidateEntryL(). Check the entry returned. sl@0: @SYMTestExpectedResults The entry is validated successfully and no leave occurred. sl@0: @SYMEC EC43 sl@0: */ sl@0: void CDiscovererTest::ValidateEntryPlugIn3TestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-3561 ")); sl@0: TInt err = KErrNone; sl@0: sl@0: //This part refers to testing of the entry containing the RSC sl@0: _LIT(KExampleRscFileNameOnly,"EComExample12.rsc"); sl@0: TDriveName driveName1(iDriveUnit.Name()); sl@0: CPluginBase *entryToFill1 = NULL; sl@0: //Creating an entry that represents the Rsc disovered during scanning of \\resource\\plugins sl@0: TEntry rscEntry; sl@0: rscEntry.iName = KExampleRscFileNameOnly; sl@0: rscEntry.iType = TUidType(KNullUid,KUidInterfaceImplementationCollectionInfo,KNullUid); sl@0: sl@0: TUid dllUid1 = {KDynamicLibraryUidValue}; sl@0: TUid example12Uid = {0x10009E3E}; sl@0: TUidType dllUidType = TUidType(dllUid1,KUidInterfaceImplementationCollection,example12Uid); sl@0: sl@0: //Now call ValidateEntryL() with this entry containg the rsc sl@0: TRAP(err,iStateAccessor->ValidateEntryL(*iDiscoverer,rscEntry,driveName1,entryToFill1, EFalse)); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: test(entryToFill1->iDllThirdUid == dllUidType[2]); sl@0: TParse dllParse; sl@0: dllParse.Set(KPlugIn3ExampleDllFileName,NULL,NULL); sl@0: test(entryToFill1->iDllName->CompareF(dllParse.NameAndExt()) == 0); sl@0: sl@0: delete entryToFill1; sl@0: } sl@0: sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-UT-3562 sl@0: @SYMTestCaseDesc Check that the ValidateEntryL for spi data works correctly when using PLUGIN3 entries in the spi file. sl@0: @SYMTestPriority High sl@0: @SYMTestActions Call CDiscoverer::ValidateEntryL(). Check the entry returned. sl@0: @SYMTestExpectedResults ecomtesttype3.spi file is validated successfully and no leave occurred. sl@0: @SYMEC EC43 sl@0: */ sl@0: void CDiscovererTest::ValidateSpiEntryPlugIn3TestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-3562 ")); sl@0: TEntry spiEntry; sl@0: test(TheFs.Entry(KEComSpiTestFilePlugIn3PathAndName, spiEntry) == KErrNone); sl@0: TParse spiPath; sl@0: spiPath.Set(KEComSpiTestFilePlugIn3PathAndName, NULL, NULL); sl@0: sl@0: TEntry TestEntry; sl@0: TUid uid1 = {0x10000079}; sl@0: TUid uid2 = {0x10009D93}; sl@0: TUid uid3 = {0x10009E3E}; // Uid for EComExample12.dll sl@0: TestEntry.iType = TUidType(uid1, uid2, uid3); sl@0: TestEntry.iName = KPlugIn3ExampleDllFileNameOnly; sl@0: sl@0: RResourceArchive resourceArchive; sl@0: resourceArchive.OpenL(TheFs, KEComSpiTestFilePlugIn3PathAndName); sl@0: CleanupClosePushL(resourceArchive); sl@0: sl@0: // check SPI file type sl@0: TUid type = resourceArchive.Type(); sl@0: test(type == KEcomSpiFileTypeUid); sl@0: sl@0: // there is only 1 resource file sl@0: while(!resourceArchive.End()) sl@0: { sl@0: CPluginBase* entryToFill = NULL; sl@0: TRAPD(err, iStateAccessor->ValidateEntryL(*iDiscoverer,resourceArchive,entryToFill)); sl@0: CleanupStack::PushL(entryToFill); sl@0: sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: test(entryToFill->iDllThirdUid == TestEntry.iType[2]); sl@0: sl@0: TFileName name1(KEComSpiTestFilePlugIn3DllPathAndName); sl@0: TFileName name2(*(entryToFill->iDllName)); sl@0: name1.LowerCase(); sl@0: name2.LowerCase(); sl@0: TParse dllparse; sl@0: dllparse.Set(name1,NULL,NULL); sl@0: test(dllparse.NameAndExt()== name2); sl@0: sl@0: CleanupStack::PopAndDestroy(entryToFill); // resourceFile, resourceName, entryToFill sl@0: entryToFill = NULL; sl@0: } sl@0: CleanupStack::PopAndDestroy(&resourceArchive); sl@0: } sl@0: sl@0: /** sl@0: The test executes by Registering an Interface Implementation Collection sl@0: and later verifing it sl@0: sl@0: @SYMTestCaseID SYSLIB-ECOM-CT-0718 sl@0: @SYMTestCaseDesc Tests for process entry. sl@0: @SYMTestPriority High sl@0: @SYMTestActions Register an interface implementation collection and later verifies it. sl@0: Check for OOM error. sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMREQ REQ0000 sl@0: */ sl@0: void CDiscovererTest::ProcessEntryTestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0718 ")); sl@0: CPluginBase* entryBase=NULL; sl@0: TInt err=KErrNone; sl@0: sl@0: //This part refers to testing of the entry containing the RSC sl@0: _LIT(KExampleRscFileNameOnly,"EComExample5.rsc"); sl@0: TDriveName driveName1(iDriveUnit.Name()); sl@0: TParse path1; sl@0: path1.Set(KNewResourceSearchPath(),NULL,&driveName1); sl@0: //Creating a entry that represents the Rsc disovered during scanning of \\resource\\plugins sl@0: TEntry rscEntry; sl@0: rscEntry.iName=KExampleRscFileNameOnly; sl@0: rscEntry.iType=TUidType(KNullUid,KUidInterfaceImplementationCollectionInfo,KNullUid); sl@0: TUid dllUid1 = {KDynamicLibraryUidValue}; sl@0: TUid example5Uid={0x101F847B}; sl@0: TUidType dllUidType=TUidType(dllUid1,KUidInterfaceImplementationCollection,example5Uid); sl@0: entryBase=CSecurePlugin::NewL(TheFs,rscEntry,driveName1, EFalse); sl@0: CleanupStack::PushL(entryBase); sl@0: sl@0: // ProcessEntryL() updates entryToFill with information on a rsc specified by the other params. sl@0: TRAP(err, iStateAccessor->ProcessEntryL(*iDiscoverer,driveName1,entryBase,ETrue)); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: CleanupStack::PopAndDestroy(entryBase); sl@0: entryBase=NULL; sl@0: sl@0: } sl@0: sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-CT-0091 sl@0: @SYMTestCaseDesc Check that the ProcessEntryL for spi data works correctly. sl@0: @SYMTestPriority High sl@0: @SYMTestActions Ensure ecomtest.spi can be processed successfully sl@0: and no leave occurred. sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMREQ REQ3655 sl@0: */ sl@0: void CDiscovererTest::ProcessSpiEntryTestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0091 ")); sl@0: TEntry spiEntry; sl@0: test(TheFs.Entry(KEComSpiTestFilePathAndName, spiEntry) == KErrNone); sl@0: TParse spiPath; sl@0: spiPath.Set(KEComSpiTestFilePathAndName, NULL, NULL); sl@0: sl@0: RResourceArchive resourceArchive; sl@0: resourceArchive.OpenL(TheFs, KEComSpiTestFilePathAndName); sl@0: CleanupClosePushL(resourceArchive); sl@0: sl@0: // check SPI file type sl@0: TUid type = resourceArchive.Type(); sl@0: test(type == KEcomSpiFileTypeUid); sl@0: sl@0: // there is only 1 resource file sl@0: while(!resourceArchive.End()) sl@0: { sl@0: CPluginBase* entry = CSpiPlugin::NewL(resourceArchive); sl@0: CleanupStack::PushL(entry); sl@0: sl@0: TRAPD(err, iStateAccessor->ProcessEntryL(*iDiscoverer,spiPath.Drive(), entry,ETrue)); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: sl@0: CleanupStack::PopAndDestroy(entry); sl@0: entry = NULL; sl@0: } sl@0: CleanupStack::PopAndDestroy(&resourceArchive); sl@0: } sl@0: /** sl@0: The test executes by checking whether discovered plugin entry is valid sl@0: sl@0: @SYMTestCaseID SYSLIB-ECOM-CT-0719 sl@0: @SYMTestCaseDesc Tests for plugin that resides on C drive sl@0: @SYMTestPriority High sl@0: @SYMTestActions Tests for the reference of the entry containing in the DLL sl@0: Check for OOM error. sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMREQ REQ0000 sl@0: */ sl@0: void CDiscovererTest::ValidateEntryTestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0719 ")); sl@0: TInt err=KErrNone; sl@0: sl@0: //This part refers to testing of the entry containing the RSC sl@0: _LIT(KExampleRscFileNameOnly,"EComExample5.rsc"); sl@0: TDriveName driveName1(iDriveUnit.Name()); sl@0: TParse path1; sl@0: CPluginBase *entryToFill1 = NULL; sl@0: path1.Set(KNewResourceSearchPath(),NULL,&driveName1); sl@0: //Creating a entry that represents the Rsc disovered during scanning of \\resource\\plugins sl@0: TEntry rscEntry; sl@0: rscEntry.iName=KExampleRscFileNameOnly; sl@0: rscEntry.iType=TUidType(KNullUid,KUidInterfaceImplementationCollectionInfo,KNullUid); sl@0: sl@0: TUid dllUid1 = {KDynamicLibraryUidValue}; sl@0: TUid example5Uid={0x101F847B}; sl@0: TUidType dllUidType=TUidType(dllUid1,KUidInterfaceImplementationCollection,example5Uid); sl@0: sl@0: //Now call ValidateEntryL() with this entry containg the rsc sl@0: TRAP(err,iStateAccessor->ValidateEntryL(*iDiscoverer,rscEntry,driveName1,entryToFill1, EFalse)); sl@0: TEST_OOM_ERR; sl@0: test(err==KErrNone); sl@0: test(entryToFill1->iDllThirdUid==dllUidType[2]); sl@0: TParse dllParse; sl@0: dllParse.Set(KNewExampleDllFileName,NULL,NULL); sl@0: test(entryToFill1->iDllName->CompareF(dllParse.NameAndExt())==0); sl@0: sl@0: delete entryToFill1; sl@0: } sl@0: sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-CT-0092 sl@0: @SYMTestCaseDesc Check that the ValidateEntryL for spi data works correctly. sl@0: @SYMTestPriority High sl@0: @SYMTestActions Ensure ecomtest.spi can be validated successfully sl@0: and no leave occurred. sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMREQ REQ3655 sl@0: */ sl@0: void CDiscovererTest::ValidateSpiEntryTestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0092 ")); sl@0: TEntry spiEntry; sl@0: test(TheFs.Entry(KEComSpiTestFilePathAndName, spiEntry) == KErrNone); sl@0: TParse spiPath; sl@0: spiPath.Set(KEComSpiTestFilePathAndName, NULL, NULL); sl@0: sl@0: TEntry TestEntry; sl@0: TUid uid1 = {0x10000079}; sl@0: TUid uid2 = {0x10009D8D}; sl@0: TUid uid3 = {0x101F847B}; // Uid for EComExample5.dll sl@0: TestEntry.iType = TUidType(uid1, uid2, uid3); sl@0: TestEntry.iName = KNewExampleDllFileNameOnly; sl@0: sl@0: RResourceArchive resourceArchive; sl@0: resourceArchive.OpenL(TheFs, KEComSpiTestFilePathAndName); sl@0: CleanupClosePushL(resourceArchive); sl@0: sl@0: // check SPI file type sl@0: TUid type = resourceArchive.Type(); sl@0: test(type == KEcomSpiFileTypeUid); sl@0: sl@0: // there is only 1 resource file sl@0: while(!resourceArchive.End()) sl@0: { sl@0: CPluginBase* entryToFill = NULL; sl@0: TRAPD(err, iStateAccessor->ValidateEntryL(*iDiscoverer,resourceArchive,entryToFill)); sl@0: CleanupStack::PushL(entryToFill); sl@0: sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: test(entryToFill->iDllThirdUid == TestEntry.iType[2]); sl@0: sl@0: TFileName name1(KEComSpiTestFileDllPathAndName); sl@0: TFileName name2(*(entryToFill->iDllName)); sl@0: name1.LowerCase(); sl@0: name2.LowerCase(); sl@0: TParse dllparse; sl@0: dllparse.Set(name1,NULL,NULL); sl@0: test(dllparse.NameAndExt()== name2); sl@0: sl@0: CleanupStack::PopAndDestroy(entryToFill); // resourceFile, resourceName, entryToFill sl@0: entryToFill = NULL; sl@0: } sl@0: CleanupStack::PopAndDestroy(&resourceArchive); sl@0: } sl@0: sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-CT-0093 sl@0: @SYMTestCaseDesc Check that there is DLL for each resource in SPI file sl@0: @SYMTestPriority High sl@0: @SYMTestActions Ensure ecom.spi contents i.e, rsc, actually exist on drive sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMREQ REQ3655 sl@0: */ sl@0: void CDiscovererTest::ValidateSpiPluginsTestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0093 ")); sl@0: __UHEAP_MARK; sl@0: sl@0: TEntry spiEntry; sl@0: TBool err = TheFs.Entry(KEComSPIFilePathAndName, spiEntry); sl@0: // It is possible for ecom.spi file not to exist. If it does not then no testing can be done. sl@0: if(err != KErrNone) sl@0: return; sl@0: sl@0: TParse spiPath; sl@0: spiPath.Set(KEComSPIFilePathAndName, NULL, NULL); sl@0: sl@0: //To read the SPI file sl@0: RResourceArchive resourceArchive; sl@0: resourceArchive.OpenL(TheFs, KEComSPIFilePathAndName); sl@0: CleanupClosePushL(resourceArchive); sl@0: CRegistryData::CDriveData* driveData = NULL; sl@0: //Check spi file type sl@0: TUid spiType = resourceArchive.Type(); sl@0: test(spiType == KEcomSpiFileTypeUid); sl@0: sl@0: while(!resourceArchive.End()) sl@0: { sl@0: CPluginBase* entryToFill = NULL; sl@0: TRAPD(err, iStateAccessor->ValidateEntryL(*iDiscoverer,resourceArchive,entryToFill)); sl@0: CleanupStack::PushL(entryToFill); sl@0: sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: sl@0: CRegistryData::CDllData * dllData=CRegistryData::CDllData::NewLC(*(entryToFill->iDllName),entryToFill->iDllModifiedTime,entryToFill->iDllSecondUid, entryToFill->iDllThirdUid,driveData); sl@0: sl@0: TBool successful=dllData->ProcessSecurityCheckL(); sl@0: if(!successful) sl@0: { sl@0: _LIT(KMessage,"ERROR: Plugin SID Mismatch ERROR for %S."); sl@0: RDebug::Print(KMessage, entryToFill->iDllName); sl@0: } sl@0: test(successful); sl@0: sl@0: CleanupStack::PopAndDestroy(dllData); sl@0: CleanupStack::PopAndDestroy(entryToFill); sl@0: entryToFill=NULL; sl@0: } sl@0: sl@0: CleanupStack::PopAndDestroy(&resourceArchive); sl@0: __UHEAP_MARKEND; sl@0: } sl@0: sl@0: sl@0: /** sl@0: Scans plugin directories and verifies successfull completion sl@0: sl@0: @SYMTestCaseID SYSLIB-ECOM-CT-0720 sl@0: @SYMTestCaseDesc Tests for scanning plugin directories and verifies successfull completion sl@0: @SYMTestPriority High sl@0: @SYMTestActions Scans registered plugin directories from A to Z drive sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMREQ REQ0000 sl@0: */ sl@0: void CDiscovererTest::ScanDirectoryIncrementTestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0720 ")); sl@0: TInt err; sl@0: // Scans registered plugin directories from A to Z drive sl@0: TEComCachedDriveInfoIterator iter(*iCachedDriveInfo); sl@0: for(iter.Last(); iter.InRange(); iter.Prev()) sl@0: { sl@0: TRAP(err, iStateAccessor->ScanDirectoryL(*iDiscoverer,iter.DriveNumber())); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: } sl@0: // After successful completion of ScanDirectoryL, DiscovererObserver is updated sl@0: // with DiscoveriesComplete successful sl@0: // DiscovererObserver is a stub class used in place of CRegistrar. Flags are used for the sl@0: // successfull execution of API sl@0: test(iDiscovererObserver->IsDiscoveriesCompleteSuccessfully()); sl@0: } sl@0: sl@0: /** sl@0: Stops scaning of plugin directories sl@0: sl@0: @SYMTestCaseID SYSLIB-ECOM-CT-0722 sl@0: @SYMTestCaseDesc Tests for stopping of scanning plugin directories sl@0: @SYMTestPriority High sl@0: @SYMTestActions Stops scaning of plugin directories. Check for OOM error. sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMREQ REQ0000 sl@0: */ sl@0: void CDiscovererTest::ScanDirectoryCancelTestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0722 ")); sl@0: TRAPD(err, iStateAccessor->ScanDirectoryCancel(*iDiscoverer)); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: test(iStateAccessor->IsDirectoryScanCancelled(*iDiscoverer)); sl@0: } sl@0: sl@0: /** sl@0: Scaning of plugin directories in Z: drive and C:drive, check these two drives is in drive list sl@0: sl@0: @SYMTestCaseID SYSLIB-ECOM-UT-1859 sl@0: @SYMTestCaseDesc Tests for scanning plugin directories in specific drives. sl@0: @SYMTestPriority High sl@0: @SYMTestActions scaning of plugin directories. Check for OOM error. sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMDEF DEF088454 sl@0: */ sl@0: sl@0: void CDiscovererTest::ScanDirectoryTestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-1859 ")); sl@0: EComPerformance::ResetEComPerfTimeRecords(); sl@0: sl@0: // Do scan on specific drive sl@0: TRAPD(err, iStateAccessor->ScanDriveL(*iDiscoverer, EDriveZ, ETrue)); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: sl@0: TRAP(err, iStateAccessor->ScanDriveL(*iDiscoverer, EDriveC, EFalse)); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: sl@0: // Test record information correct sl@0: test(EComPerformance::iEComPerfTimeRecordCount == 4); sl@0: test(EComPerformance::iEComPerfTimeRecords[0].iType == ECDiscovererRediscoveryScanDirectoryL); sl@0: test(EComPerformance::iEComPerfTimeRecords[0].iInfo == EDriveZ); sl@0: test(EComPerformance::iEComPerfTimeRecords[1].iType == ECDiscovererRediscoveryScanDirectoryL); sl@0: test(EComPerformance::iEComPerfTimeRecords[1].iInfo == EDriveZ); sl@0: sl@0: test(EComPerformance::iEComPerfTimeRecords[2].iType == ECDiscovererRediscoveryScanDirectoryL); sl@0: test(EComPerformance::iEComPerfTimeRecords[2].iInfo == EDriveC); sl@0: test(EComPerformance::iEComPerfTimeRecords[3].iType == ECDiscovererRediscoveryScanDirectoryL); sl@0: test(EComPerformance::iEComPerfTimeRecords[3].iInfo == EDriveC); sl@0: } sl@0: sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-CT-0185 sl@0: @SYMTestCaseDesc Check that SSA related states transition as expected when discovery is staged sl@0: @SYMTestPriority High sl@0: @SYMTestActions Use CDiscoverer::ProcessSSAEventL to start discovery in stages i.e. rom only sl@0: then non rom only. Check that the sl@0: CDiscoverer state is transitioning correctly at every step. sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMREQ REQ3655 sl@0: */ sl@0: void CDiscovererTest::StagedDiscoveryStateTransitionTestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0185 ")); sl@0: //After construction check that: sl@0: //- current state is EDisc_NoPluginsDisc sl@0: //- registry is empty. i.e. no plugins discovered sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_NoPluginsDisc); sl@0: sl@0: //Call ProcessSSAEvent with state set to various states that will sl@0: //not cause a transition. Check that: sl@0: //- current state has not changed sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateUndefined); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_NoPluginsDisc); sl@0: sl@0: //Call ProcessSSAEventL with EStartupStateCriticalStatic state. sl@0: //Check that sl@0: //- current state is EDisc_CriticalPluginsDisc sl@0: //- discoverer observer has processed RO plugins sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateCriticalStatic); sl@0: test(iStateAccessor->GetDiscovererObserverState(*iDiscoverer) == CObserverStub::OSS_CriticalPlugins); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_CriticalPluginsDisc); sl@0: sl@0: //Call ProcessSSAEvent with state set to various states that will sl@0: //not cause a transition. Check that: sl@0: //- current state has not changed sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateUndefined); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_CriticalPluginsDisc); sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateCriticalStatic); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_CriticalPluginsDisc); sl@0: sl@0: //Call ProcessSSAEventL with EStartupStateNonCritical state. Check that sl@0: //- current state is EDisc_AllPluginsDisc sl@0: //- discoverer observer has processed non-RO plugins sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical); sl@0: test(iStateAccessor->GetDiscovererObserverState(*iDiscoverer) == CObserverStub::OSS_AllPlugins); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc); sl@0: sl@0: //Call ProcessSSAEvent with state set to various states that will sl@0: //not cause a transition. Check that: sl@0: //- current state has not changed sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateUndefined); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc); sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateCriticalStatic); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc); sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateCriticalDynamic); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc); sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc); sl@0: sl@0: //Call ProcessDNEventL to indicate that current set of plugins is dirty. sl@0: //Check that: sl@0: //- current state is EDisc_PluginsDirty sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveC); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_PluginsDirty); sl@0: sl@0: //Call ProcessDNEventL to indicate that a rediscovery should take place. sl@0: //Check that: sl@0: //- current state is EDisc_AllPluginsDisc sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveC); sl@0: iStateAccessor->CompleteNotificationProcessing(*iDiscoverer); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc); sl@0: } sl@0: sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-CT-0186 sl@0: @SYMTestCaseDesc Check that SSA related states transition as expected when discovery is all at once sl@0: @SYMTestPriority High sl@0: @SYMTestActions Use CDiscoverer::ProcessSSAEventL to start discovery all at once. Check that the sl@0: CDiscoverer state is transitioning correctly at every step. sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMREQ REQ3655 sl@0: */ sl@0: void CDiscovererTest::AllAtOnceDiscoveryStateTransitionTestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0186 ")); sl@0: //After construction check that: sl@0: //- current state is EDisc_NoPluginsDisc sl@0: //- registry is empty. i.e. no plugins discovered sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_NoPluginsDisc); sl@0: sl@0: //Call ProcessSSAEvent with state set to various states that will sl@0: //not cause a transition. Check that: sl@0: //- current state has not changed sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateUndefined); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_NoPluginsDisc); sl@0: sl@0: //Call ProcessSSAEventL with EStartupStateNonCritical state. Check that sl@0: //- current state is EDisc_AllPluginsDisc sl@0: //- discoverer observer has processed non-RO plugins sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical); sl@0: test(iStateAccessor->GetDiscovererObserverState(*iDiscoverer) == CObserverStub::OSS_AllPlugins); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc); sl@0: sl@0: //Call ProcessSSAEvent with state set to various states that will sl@0: //not cause a transition. Check that: sl@0: //- current state has not changed sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateUndefined); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc); sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateCriticalStatic); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc); sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateCriticalDynamic); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc); sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc); sl@0: sl@0: //Call ProcessDNEventL to indicate that current set of plugins is dirty. sl@0: //Check that: sl@0: //- current state is EDisc_PluginsDirty sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveC); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_PluginsDirty); sl@0: sl@0: //Call ProcessDNEventL to indicate that a rediscovery should take place. sl@0: //Check that: sl@0: //- current state is EDisc_AllPluginsDisc sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveC); sl@0: iStateAccessor->CompleteNotificationProcessing(*iDiscoverer); sl@0: test(iStateAccessor->State(*iDiscoverer) == CDiscoverer::EDisc_AllPluginsDisc); sl@0: } sl@0: sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-UT-1796 sl@0: @SYMTestCaseDesc Tests multiple notification processing for sl@0: "INC087110: ECOM rescanning code could miss a drive?" and sl@0: "DEF088454: [PCB] ECOM CDiscoverer::CIdleScanningTimer::RunL() Performs Unnecessary Tasks" sl@0: @SYMTestPriority High sl@0: @SYMTestActions Create multiple notification request on both single drive and multiple drives. sl@0: Check if the notifications are ONLY processed on corresponding drive(s) by timer. sl@0: Check the notifications are processed properly by timer. sl@0: Check the state of discoverer was set correctly on completion. sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMDEF INC087110, DEF088454 sl@0: */ sl@0: void CDiscovererTest::MultipleNotificationProcessingTestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-1796 ")); sl@0: TBool pending = EFalse; sl@0: sl@0: // Multiple notification on single drive: sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical); sl@0: // Raise multiple notification on C: drive sl@0: for(TInt num = 0; num < 10; ++num) sl@0: { sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveC); sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveC); sl@0: } sl@0: sl@0: // Check only one pending drive in the list. sl@0: test(iStateAccessor->PendingDriveListCount(*iDiscoverer) == 1); sl@0: // Check the pending drive is C: drive sl@0: pending = iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveC); sl@0: test(pending); sl@0: sl@0: // Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process sl@0: // all pending notification. sl@0: iShutdown.StartAfter(KOneSecond * 3); sl@0: CActiveScheduler::Start(); sl@0: sl@0: // Check if notifications have been processed on all drives. sl@0: test(iStateAccessor->IsScanDirectoryComplete(*iDiscoverer)); sl@0: sl@0: // Multiple notification on multiple drives: sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical); sl@0: // Raise multiple notification on C: drive sl@0: for(TInt num = 0; num < 10; ++num) sl@0: { sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveC); sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveC); sl@0: } sl@0: #if defined(__WINSCW__) // X: drive on emulator sl@0: // Raise multiple notification on X: drive sl@0: for(TInt num = 0; num < 10; ++num) sl@0: { sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveX); sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveX); sl@0: } sl@0: sl@0: #else // E: drive on HW sl@0: // Raise multiple notification on E: drive sl@0: for(TInt num = 0; num < 10; ++num) sl@0: { sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveE); sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveE); sl@0: } sl@0: #endif // __WINSCW__ sl@0: sl@0: // Check only one pending drive in the list. sl@0: test(iStateAccessor->PendingDriveListCount(*iDiscoverer) == 2); sl@0: sl@0: // Check the pending drive are C: and X: on emulator, or C: and E: on hardware. sl@0: pending = iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveC); sl@0: test(pending); sl@0: #if defined(__WINSCW__) // X: drive on emulator sl@0: pending = iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveX); sl@0: test(pending); sl@0: #else // E: drive on HW sl@0: pending = iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveE); sl@0: test(pending); sl@0: #endif // __WINSCW__ sl@0: sl@0: // Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process sl@0: // all pending notification. sl@0: iShutdown.StartAfter(KOneSecond * 3); sl@0: CActiveScheduler::Start(); sl@0: sl@0: // Check if notifications have been processed on all drives. sl@0: test(iStateAccessor->IsScanDirectoryComplete(*iDiscoverer)); sl@0: } sl@0: sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-UT-3519 sl@0: @SYMTestCaseDesc Tests notification processing during SWI for sl@0: PDEF110201: Undesireable interaction between ECOM and SWI sl@0: @SYMTestActions Create notification request on drives. sl@0: Check the notifications are not processed by timer if SWI in progress. sl@0: Check that timer processes pending notifications after SWI completes sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMDEF PDEF110201 sl@0: */ sl@0: void CDiscovererTest::SWINotificationProcessingTestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-3519 ")); sl@0: sl@0: //Begin SWI sl@0: iStateAccessor->SwiChangeNotificationL(*iDiscoverer,ESASwisInstall); sl@0: sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical); sl@0: sl@0: // Raise a notification on C: drive sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveC); sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveC); sl@0: sl@0: // Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process sl@0: // all pending notification. sl@0: iShutdown.StartAfter(KOneSecond * 3); sl@0: CActiveScheduler::Start(); sl@0: sl@0: // Check that notifications have NOT been processed as SWI is in progress sl@0: test(!iStateAccessor->IsScanDirectoryComplete(*iDiscoverer)); sl@0: sl@0: //complete SWI sl@0: iStateAccessor->SwiChangeNotificationL(*iDiscoverer,ESASwisUninstall); sl@0: sl@0: sl@0: #if defined(__WINSCW__) // X: drive on emulator sl@0: // Raise notification on X: drive sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveX); sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveX); sl@0: #else // E: drive on HW sl@0: // Raise notification on E: drive sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified, EDriveE); sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover, EDriveE); sl@0: #endif // __WINSCW__ sl@0: sl@0: // Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process sl@0: // all pending notification. sl@0: iShutdown.StartAfter(KOneSecond * 3); sl@0: CActiveScheduler::Start(); sl@0: sl@0: // Check that notifications have NOT been processed as SWI is not finished sl@0: test(!iStateAccessor->IsScanDirectoryComplete(*iDiscoverer)); sl@0: sl@0: //finalise SWI sl@0: iStateAccessor->SwiChangeNotificationL(*iDiscoverer,ESASwisNone); sl@0: sl@0: // Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process sl@0: // all pending notification. sl@0: iShutdown.StartAfter(KOneSecond * 3); sl@0: CActiveScheduler::Start(); sl@0: sl@0: // Check that notifications have been processed on all drives sl@0: test(iStateAccessor->IsScanDirectoryComplete(*iDiscoverer)); sl@0: sl@0: //Begin SWI sl@0: iStateAccessor->SwiChangeNotificationL(*iDiscoverer,ESASwisInstall); sl@0: sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical); sl@0: sl@0: // Raise a notification sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsModified,EDriveC); sl@0: iDiscoverer->ProcessDNEventL(CDiscoverer::EPluginsRediscover,EDriveC); sl@0: sl@0: // Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process sl@0: // all pending notification. sl@0: iShutdown.StartAfter(KOneSecond * 3); sl@0: CActiveScheduler::Start(); sl@0: sl@0: // Check that notifications have NOT been processed as SWI is in progress sl@0: test(!iStateAccessor->IsScanDirectoryComplete(*iDiscoverer)); sl@0: sl@0: //Simulate an error in reading the P&S Variable - This should reset the sl@0: //SWI state to ESASwisNone in the same way as completing SWI sl@0: iStateAccessor->SwiChangeNotifierRunError(*iDiscoverer,KErrNotFound); sl@0: sl@0: // Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process sl@0: // all pending notification. sl@0: iShutdown.StartAfter(KOneSecond * 3); sl@0: CActiveScheduler::Start(); sl@0: sl@0: // Check that notifications have been processed on all drives sl@0: test(iStateAccessor->IsScanDirectoryComplete(*iDiscoverer)); sl@0: } sl@0: sl@0: /** sl@0: Call the TDiscoverer_StateAccessor::IdleScanningTimerRunErrorL funtion sl@0: passing in a reference to the CDiscoverer object and the error code sl@0: to be passed to the CIdleScanningTimer::RunError function sl@0: */ sl@0: void CDiscovererTest::IdleScanningTimerRunErrorL() sl@0: { sl@0: iStateAccessor->IdleScanningTimerRunErrorL(*iDiscoverer, EEComPanic_CDiscoverer_CIdleScanningTimer_RunError); sl@0: } sl@0: sl@0: /** sl@0: Call the TDiscoverer_StateAccessor::IdleScanningTimerRunErrorL funtion sl@0: passing in a reference to the CDiscoverer object and the error code sl@0: to be passed to the CDirChangeNotifier::RunError function sl@0: */ sl@0: void CDiscovererTest::DirChangeNotifierRunErrorL() sl@0: { sl@0: iStateAccessor->DirChangeNotifierRunErrorL(*iDiscoverer, EEComPanic_CDiscoverer_CDirChangeNotifier_RunError); sl@0: } sl@0: sl@0: /** sl@0: Call the TDiscoverer_StateAccessor::IdleScanningTimerRunErrorL funtion sl@0: passing in a reference to the CDiscoverer object and the error code sl@0: to be passed to the CSwiChangeNotifier::RunError function sl@0: */ sl@0: void CDiscovererTest::SwiChangeNotifierRunError() sl@0: { sl@0: iStateAccessor->SwiChangeNotifierRunError(*iDiscoverer, KErrNoMemory); sl@0: } sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-UT-3172 sl@0: @SYMTestCaseDesc Tests language switch notification processing for sl@0: "CR0902: Enable Dynamic Language Switching in APPARC, ECOM and EIKSTD" sl@0: @SYMTestPriority High sl@0: @SYMTestActions Call ProcessSSAEventL with EStartupStateNonCritical state. sl@0: Call DriveMountedL to build up drives in system. sl@0: Call LanguageChangedL to raise multiple notification on all drives in system. sl@0: Check if the notifications are processed on all drive(s) by timer. sl@0: Check the notifications are processed properly by timer. sl@0: Check the state of discoverer was set correctly on completion. sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMDEF CR0902 sl@0: */ sl@0: void CDiscovererTest::LanguageChangedNotificationTestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-UT-3172 ")); sl@0: TBool pending = EFalse; sl@0: sl@0: //Call ProcessSSAEventL with EStartupStateNonCritical state sl@0: iDiscoverer->ProcessSSAEventL(EStartupStateNonCritical); sl@0: sl@0: if(iStateAccessor->IsDriveMounted(*iDiscoverer, EDriveK)) sl@0: { sl@0: TRAPD(err, iStateAccessor->DriveUnmountedL(*iDiscoverer, EDriveK)); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: } sl@0: sl@0: // Mount drives on system sl@0: TRAPD(err, iStateAccessor->DriveMountedL(*iDiscoverer, EDriveZ)); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: sl@0: TRAP(err, iStateAccessor->DriveMountedL(*iDiscoverer, EDriveC)); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: sl@0: #if defined(__WINSCW__) // X: drive on emulator sl@0: TRAP(err, iStateAccessor->DriveMountedL(*iDiscoverer, EDriveX)); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: #else // E: drive on HW sl@0: TRAP(err, iStateAccessor->DriveMountedL(*iDiscoverer, EDriveE)); sl@0: TEST_OOM_ERR; sl@0: test(err == KErrNone); sl@0: #endif sl@0: sl@0: // Multiple notification on all drives: sl@0: iStateAccessor->LanguageChangedL(*iDiscoverer); sl@0: sl@0: // Check three pending drive in the list. sl@0: test(iStateAccessor->PendingDriveListCount(*iDiscoverer) == 3); sl@0: // Check the pending drive is Z: drive sl@0: pending = iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveZ); sl@0: test(pending); sl@0: sl@0: // Check the pending drive is C: drive sl@0: pending = iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveC); sl@0: test(pending); sl@0: sl@0: #if defined(__WINSCW__) // X: drive on emulator sl@0: pending = iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveX); sl@0: test(pending); sl@0: #else // E: drive on HW sl@0: pending = iStateAccessor->IsScanDirectoryPending(*iDiscoverer, EDriveE); sl@0: test(pending); sl@0: #endif // __WINSCW__ sl@0: sl@0: // Start Active Scheduler and shut it down in 3 secs, it will launch RunL of timer to process sl@0: // all pending notification. sl@0: iShutdown.StartAfter(KOneSecond * 3); sl@0: CActiveScheduler::Start(); sl@0: sl@0: // Check if notifications have been processed on all drives. sl@0: test(iStateAccessor->IsScanDirectoryComplete(*iDiscoverer)); sl@0: sl@0: } sl@0: sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-CT-0723 sl@0: @SYMTestCaseDesc Tests the creation and deletion of CDiscoverer sl@0: @SYMTestPriority High sl@0: @SYMTestActions Create and delete CDiscoverer object,checks for open handles sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMREQ REQ0000 sl@0: */ sl@0: LOCAL_C void CreateDeleteTestL() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0723 CreateDeleteTestL ")); sl@0: // sl@0: // Tests the Creating and deletion of CDiscoverer sl@0: // ------------------------------------------------------------------ sl@0: // sl@0: // Set up for heap leak checking sl@0: __UHEAP_MARK; sl@0: sl@0: //Check Thread handles leak sl@0: TInt startProcessHandleCount = 0; sl@0: TInt startThreadHandleCount = 0; sl@0: TInt endProcessHandleCount = 0; sl@0: TInt endThreadHandleCount = 0; sl@0: sl@0: RThread rThread; sl@0: rThread.HandleCount(startProcessHandleCount, startThreadHandleCount); sl@0: sl@0: //START CREATE DELETE TEST// sl@0: sl@0: CDiscovererTest* discTest = CDiscovererTest::NewL(); sl@0: sl@0: test(discTest != NULL); sl@0: sl@0: delete discTest; sl@0: sl@0: //END CREATE DELETE TEST// sl@0: sl@0: // Check for open handles sl@0: rThread.HandleCount(endProcessHandleCount, endThreadHandleCount); sl@0: test(startThreadHandleCount == endThreadHandleCount); sl@0: sl@0: //Test ends sl@0: __UHEAP_MARKEND; sl@0: } sl@0: sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-CT-0724 sl@0: @SYMTestCaseDesc OOM test for create and delete of CDiscoverer. sl@0: @SYMTestPriority High sl@0: @SYMTestActions Create and delete CDiscoverer object,checks for any memory leak. sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMREQ REQ0000 sl@0: */ sl@0: LOCAL_C void OOMCreateDeleteTest() sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0724 OOM CreateDeleteTest ")); sl@0: TInt err; sl@0: TInt failAt = 1; sl@0: __UNUSED_VAR(failAt); sl@0: sl@0: CDiscovererTest* discTest = NULL; sl@0: sl@0: do sl@0: { sl@0: __UHEAP_MARK; sl@0: // find out the number of open handles sl@0: TInt startProcessHandleCount; sl@0: TInt startThreadHandleCount; sl@0: RThread().HandleCount(startProcessHandleCount, startThreadHandleCount); sl@0: sl@0: __UHEAP_SETFAIL(RHeap::EDeterministic, failAt++); sl@0: sl@0: TRAP(err, discTest = CDiscovererTest::NewL()); sl@0: sl@0: __UHEAP_SETFAIL(RHeap::ENone, 0); sl@0: sl@0: delete discTest; sl@0: discTest = NULL; sl@0: sl@0: // check that no handles have leaked sl@0: TInt endProcessHandleCount; sl@0: TInt endThreadHandleCount; sl@0: RThread().HandleCount(endProcessHandleCount, endThreadHandleCount); sl@0: sl@0: test(startProcessHandleCount == endProcessHandleCount); sl@0: test(startThreadHandleCount == endThreadHandleCount); sl@0: sl@0: __UHEAP_MARKEND; sl@0: } sl@0: while(err == KErrNoMemory); sl@0: sl@0: test.Printf(_L("- Succeeded at heap failure rate of %i\n"), failAt); sl@0: test(err == KErrNone); sl@0: } sl@0: sl@0: // Type definition for pointer to member function. sl@0: // Used in calling the CDiscovererTest member function for testing. sl@0: typedef void (CDiscovererTest::*ClassFuncPtrL) (void); sl@0: sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-CT-0725 sl@0: @SYMTestCaseDesc Function to call all test functions sl@0: @SYMTestPriority High sl@0: @SYMTestActions Calls up test function of CDiscovererTest. sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMREQ REQ0000 sl@0: */ sl@0: /** sl@0: Wrapper function to call all test functions sl@0: sl@0: @param testFunc pointer to test function sl@0: @param aTestDesc test function name sl@0: */ sl@0: LOCAL_C void DoBasicTestL(ClassFuncPtrL testFuncL, const TDesC& aTestDesc) sl@0: { sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-0725 ")); sl@0: test.Next(aTestDesc); sl@0: sl@0: __UHEAP_MARK; sl@0: // find out the number of open handles sl@0: TInt startProcessHandleCount; sl@0: TInt startThreadHandleCount; sl@0: RThread().HandleCount(startProcessHandleCount, startThreadHandleCount); sl@0: sl@0: CDiscovererTest* discTest = CDiscovererTest::NewL(); sl@0: CleanupStack::PushL(discTest); sl@0: sl@0: (discTest->*testFuncL)(); sl@0: sl@0: CleanupStack::PopAndDestroy(discTest); sl@0: sl@0: // check that no handles have leaked sl@0: TInt endProcessHandleCount; sl@0: TInt endThreadHandleCount; sl@0: RThread().HandleCount(endProcessHandleCount, endThreadHandleCount); sl@0: sl@0: test(startProcessHandleCount == endProcessHandleCount); sl@0: test(startThreadHandleCount == endThreadHandleCount); sl@0: sl@0: __UHEAP_MARKEND; sl@0: } sl@0: sl@0: /** sl@0: Utility function to continually invoke a test function and cause memory allocation failures sl@0: sl@0: @param testFuncL pointer to OOM test function sl@0: @param tryCount specifies what value to start memory allocation failures at sl@0: @param increment how much to increase the point at which memory allocation will fail on each test attempt sl@0: @param stopCount the value of memory allocation failure to stop testing at sl@0: */ sl@0: TInt RunTestUnderOOMCondition(ClassFuncPtrL testFuncL, TInt tryCount, TInt increment, TInt stopCount) sl@0: { sl@0: TInt err = KErrNone; sl@0: sl@0: do sl@0: { sl@0: sl@0: __UHEAP_MARK; sl@0: // find out the number of open handles sl@0: TInt startProcessHandleCount; sl@0: TInt startThreadHandleCount; sl@0: RThread().HandleCount(startProcessHandleCount, startThreadHandleCount); sl@0: sl@0: CDiscovererTest* discTest = CDiscovererTest::NewL(); sl@0: CleanupStack::PushL(discTest); sl@0: sl@0: sl@0: __UHEAP_SETFAIL(RHeap::EDeterministic, tryCount); sl@0: TRAP(err, (discTest->*testFuncL)()); sl@0: __UHEAP_SETFAIL(RHeap::ENone, 0); sl@0: sl@0: sl@0: CleanupStack::PopAndDestroy(discTest); sl@0: discTest = NULL; sl@0: sl@0: // check that no handles have leaked sl@0: TInt endProcessHandleCount; sl@0: TInt endThreadHandleCount; sl@0: RThread().HandleCount(endProcessHandleCount, endThreadHandleCount); sl@0: sl@0: test(startProcessHandleCount == endProcessHandleCount); sl@0: test(startThreadHandleCount == endThreadHandleCount); sl@0: sl@0: __UHEAP_MARKEND; sl@0: sl@0: tryCount = tryCount + increment; sl@0: } while((err == KErrNoMemory) && (tryCount != stopCount)); sl@0: sl@0: tryCount = tryCount - increment; sl@0: sl@0: sl@0: if (err == KErrNoMemory) sl@0: { sl@0: // test has not yet been able to pass due to memory allocation failures. sl@0: return -1; sl@0: } sl@0: sl@0: test(err == KErrNone); sl@0: sl@0: // If enough memory has finally been allocated for the test to pass then return sl@0: // the memory allocation counter value. sl@0: test.Printf(_L("- server succeeded at heap failure rate of %i\n"), tryCount); sl@0: return tryCount; sl@0: } sl@0: sl@0: /** sl@0: Wrapper function to call all OOM test functions sl@0: sl@0: @param testFuncL pointer to OOM test function sl@0: @param aTestDesc test function name sl@0: */ sl@0: LOCAL_C void DoOOMTestL(ClassFuncPtrL testFuncL, const TDesC& aTestDesc) sl@0: { sl@0: test.Next(aTestDesc); sl@0: sl@0: TInt startCount = 1; sl@0: TInt increment = 1; sl@0: TInt successRate = -1; sl@0: TInt stopCount = 256; sl@0: sl@0: successRate = RunTestUnderOOMCondition(testFuncL, startCount, increment, stopCount); sl@0: sl@0: // (INC115057) sl@0: // When method CDiscoverer::CDirScanner::DoScanDriveL is encountered in a test hundreds sl@0: // of plug-in files will be scanned. It takes in the order of 3-4000 memory allocation failure sl@0: // loops (in techview context) before the scan is finished and the test can complete successfully. sl@0: // This will take over an hour. Instead determine the approximate failure point by sl@0: // testing with large increments between failures. Once a failure rate interval is found test sl@0: // for OOM conditions running up to it. sl@0: // This in effect means that we are checking OOM at the start and end of tests but skipping sl@0: // the scanning of each and every plugin in the middle of the test. sl@0: // (Note that CDiscoverer::CDirScanner::DoScanDriveL may return without leaving when it sl@0: // can't allocate TFileName. In this case it seems to this function that the test is sl@0: // successful. Therefore even if the successRate above indicates a pass the test is still sl@0: // re-run below with larger memory allocation failure valued to make sure that the test does sl@0: // in fact run to completion. sl@0: sl@0: startCount = 256; sl@0: increment = 256; sl@0: stopCount = -1; sl@0: successRate = RunTestUnderOOMCondition(testFuncL, startCount, increment, stopCount); sl@0: test(successRate > 0); sl@0: sl@0: if (successRate > 256) sl@0: { sl@0: startCount = successRate - 256; sl@0: increment = 1; sl@0: stopCount = -1; sl@0: successRate = RunTestUnderOOMCondition(testFuncL, startCount, increment, stopCount); sl@0: } sl@0: sl@0: test(successRate > 0); sl@0: } sl@0: sl@0: /** sl@0: Creates and installs active scheduler for this thread and calls sl@0: CDiscovererTest::IdleScanningTimerRunErrorL sl@0: sl@0: @param aDiscTest The CDiscovererTest object used to carry out the test sl@0: */ sl@0: LOCAL_C void DoIdleScanningTimerRunErrorTestL(CDiscovererTest* aDiscTest) sl@0: { sl@0: sl@0: // create and install the active scheduler we need sl@0: CActiveScheduler* scheduler=new(ELeave) CActiveScheduler; sl@0: CleanupStack::PushL(scheduler); sl@0: sl@0: CActiveScheduler::Install(scheduler); sl@0: sl@0: aDiscTest->IdleScanningTimerRunErrorL(); sl@0: sl@0: // Cleanup CDiscovererTest, TheFs and scheduler sl@0: CleanupStack::PopAndDestroy(scheduler); sl@0: sl@0: } sl@0: /** sl@0: Thread entry point for the test thread. Creates a CTrapCleanup and sl@0: calls DoIdleScanningTimerRunErrorTestL to carry out the test sl@0: sl@0: @param aDiscTest The CDiscovererTest object used to carry out the test sl@0: */ sl@0: LOCAL_C TInt IdleScanningTimerRunErrorThreadEntry(TAny* aDiscTest) sl@0: { sl@0: sl@0: CTrapCleanup* tc = CTrapCleanup::New(); sl@0: sl@0: CDiscovererTest *discTest = static_cast(aDiscTest); sl@0: sl@0: TRAPD(err,DoIdleScanningTimerRunErrorTestL(discTest)); sl@0: sl@0: delete tc; sl@0: sl@0: return err; sl@0: sl@0: } sl@0: sl@0: sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-CT-3165 sl@0: @SYMTestCaseDesc Check that the CIdleScanningTimer::RunError() works correctly. sl@0: @SYMTestPriority High sl@0: @SYMTestActions Create a new thread which will call RunError. Wait for the sl@0: thread to exit and check the thread exit type and reason sl@0: to verify behaviour sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMDEF DEF094675 sl@0: */ sl@0: LOCAL_C void IdleScanningTimer_RunErrorTest() sl@0: { sl@0: __UHEAP_MARK; sl@0: sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-3165 CIdleScanningTimer RunError test ")); sl@0: sl@0: _LIT(KStartThreadName,"CIdleScanningTimer RunError Thread"); sl@0: sl@0: //Disable JIT so that the Panic doesn't bring up a dialog sl@0: //and stop the test sl@0: TBool jitEnabled = User::JustInTime(); sl@0: User::SetJustInTime(EFalse); sl@0: sl@0: //Create a CDiscovererTest object to pass into the test thread sl@0: CDiscovererTest* discTest = CDiscovererTest::NewL(); sl@0: CleanupStack::PushL(discTest); sl@0: sl@0: //Create a new thread to run the test sl@0: RThread testThread; sl@0: testThread.Create(KStartThreadName, IdleScanningTimerRunErrorThreadEntry, sl@0: KDefaultStackSize,KMinHeapSize,KMinHeapSize,discTest); sl@0: TRequestStatus status; sl@0: testThread.Logon(status); sl@0: sl@0: sl@0: testThread.Resume(); sl@0: sl@0: //Wait for the thread to exit sl@0: User::WaitForRequest(status); sl@0: sl@0: //Obtain exit type and reason for test thread sl@0: TExitType exitType = testThread.ExitType(); sl@0: TInt exitReason = testThread.ExitReason(); sl@0: sl@0: //close the thread handle sl@0: testThread.Close(); sl@0: sl@0: CleanupStack::PopAndDestroy(discTest); sl@0: sl@0: //Set JIT back to original state sl@0: User::SetJustInTime(jitEnabled); sl@0: sl@0: //Verify the exit reason and exit code sl@0: test(exitType == EExitPanic); sl@0: test(exitReason == EEComPanic_CDiscoverer_CIdleScanningTimer_RunError); sl@0: sl@0: __UHEAP_MARKEND; sl@0: } sl@0: sl@0: /** sl@0: Creates and installs active scheduler for this thread and calls sl@0: CDiscovererTest::DirChangeNotifierRunErrorL sl@0: sl@0: @param aDiscTest The CDiscovererTest object used to carry out the test sl@0: */ sl@0: LOCAL_C void DoDirChangeNotifierRunErrorTestL(CDiscovererTest* aDiscTest) sl@0: { sl@0: sl@0: // create and install the active scheduler we need sl@0: CActiveScheduler* scheduler=new(ELeave) CActiveScheduler; sl@0: CleanupStack::PushL(scheduler); sl@0: sl@0: CActiveScheduler::Install(scheduler); sl@0: sl@0: //call the RunErrorL method which should panic sl@0: aDiscTest->DirChangeNotifierRunErrorL(); sl@0: sl@0: // Cleanup CDiscovererTest, TheFs and scheduler sl@0: CleanupStack::PopAndDestroy(scheduler); sl@0: sl@0: } sl@0: sl@0: /** sl@0: Thread entry point for the test thread. Creates a CTrapCleanup and sl@0: calls DoDirChangeNotifierRunErrorTestL to carry out the test sl@0: sl@0: @param aDiscTest The CDiscovererTest object used to carry out the test sl@0: */ sl@0: LOCAL_C TInt DirChangeNotifierRunErrorThreadEntry(TAny* aDiscTest) sl@0: { sl@0: sl@0: CTrapCleanup* tc = CTrapCleanup::New(); sl@0: sl@0: CDiscovererTest *discTest = static_cast(aDiscTest); sl@0: sl@0: TRAPD(err,DoDirChangeNotifierRunErrorTestL(discTest)); sl@0: sl@0: delete tc; sl@0: sl@0: return err; sl@0: sl@0: } sl@0: sl@0: /** sl@0: @SYMTestCaseID SYSLIB-ECOM-CT-3166 sl@0: @SYMTestCaseDesc Check that the CDirChangeNotifier::RunError() works correctly. sl@0: @SYMTestPriority High sl@0: @SYMTestActions Create a new thread which will call RunError. Wait for the sl@0: thread to exit and check the thread exit type and reason sl@0: to verify behaviour sl@0: @SYMTestExpectedResults The test must not fail. sl@0: @SYMDEF DEF094675 sl@0: */ sl@0: LOCAL_C void DirChangeNotifier_RunErrorTest() sl@0: { sl@0: __UHEAP_MARK; sl@0: sl@0: test.Next(_L(" @SYMTestCaseID:SYSLIB-ECOM-CT-3166 CDirChangeNotifier RunError test ")); sl@0: sl@0: _LIT(KStartThreadName,"CDirChangeNotifier RunError Thread"); sl@0: sl@0: //Disable JIT so that the Panic doesn't bring up a dialog sl@0: //and stop the test sl@0: TBool jitEnabled = User::JustInTime(); sl@0: User::SetJustInTime(EFalse); sl@0: sl@0: //Create a CDiscovererTest object to pass into the test thread sl@0: CDiscovererTest* discTest = CDiscovererTest::NewL(); sl@0: CleanupStack::PushL(discTest); sl@0: sl@0: //Create a new thread to run the test sl@0: RThread testThread; sl@0: testThread.Create(KStartThreadName, DirChangeNotifierRunErrorThreadEntry, sl@0: KDefaultStackSize,KMinHeapSize,KMinHeapSize,discTest); sl@0: TRequestStatus status; sl@0: testThread.Logon(status); sl@0: testThread.Resume(); sl@0: sl@0: //Wait for the thread to exit sl@0: User::WaitForRequest(status); sl@0: sl@0: //Obtain exit type and reason for test thread sl@0: TExitType exitType = testThread.ExitType(); sl@0: TInt exitReason = testThread.ExitReason(); sl@0: sl@0: //close the thread handle sl@0: testThread.Close(); sl@0: sl@0: CleanupStack::PopAndDestroy(discTest); sl@0: sl@0: //Set JIT back to original state sl@0: User::SetJustInTime(jitEnabled); sl@0: sl@0: //Verify the exit reason and exit code sl@0: test(exitType == EExitPanic); sl@0: test(exitReason == EEComPanic_CDiscoverer_CDirChangeNotifier_RunError); sl@0: sl@0: __UHEAP_MARKEND; sl@0: } sl@0: sl@0: TInt DoTestsL() sl@0: { sl@0: __UHEAP_MARK; sl@0: sl@0: // Basic tests sl@0: CreateDeleteTestL(); sl@0: DoBasicTestL(&CDiscovererTest::ResumeSuspendTestL, _L("ResumeSuspendTestL")); sl@0: DoBasicTestL(&CDiscovererTest::DriveMountUnmountTestL, _L("DriveMountUnmountTestL")); sl@0: DoBasicTestL(&CDiscovererTest::ProcessEntryTestL, _L("ProcessEntryTestL")); sl@0: DoBasicTestL(&CDiscovererTest::ValidateEntryTestL, _L("ValidateEntryTestL")); sl@0: DoBasicTestL(&CDiscovererTest::ProcessSpiEntryTestL, _L("ProcessSpiEntryTestL")); sl@0: DoBasicTestL(&CDiscovererTest::ValidateSpiEntryTestL, _L("ValidateSpiEntryTestL")); sl@0: DoBasicTestL(&CDiscovererTest::ProcessEntryPlugIn3TestL, _L("ProcessEntryPlugIn3TestL")); sl@0: DoBasicTestL(&CDiscovererTest::ValidateEntryPlugIn3TestL, _L("ValidateEntryPlugIn3TestL")); sl@0: DoBasicTestL(&CDiscovererTest::ProcessSpiEntryPlugIn3TestL, _L("ProcessSpiEntryPlugIn3TestL")); sl@0: DoBasicTestL(&CDiscovererTest::ValidateSpiEntryPlugIn3TestL, _L("ValidateSpiEntryPlugIn3TestL")); sl@0: DoBasicTestL(&CDiscovererTest::ValidateSpiPluginsTestL, _L("ValidateSpiPluginsTestL")); sl@0: DoBasicTestL(&CDiscovererTest::ScanDirectoryIncrementTestL, _L("ScanDirectoryIncrementTestL")); sl@0: DoBasicTestL(&CDiscovererTest::StagedDiscoveryStateTransitionTestL, _L("StagedDiscoveryStateTransitionTestL")); sl@0: DoBasicTestL(&CDiscovererTest::AllAtOnceDiscoveryStateTransitionTestL, _L("AllAtOnceDiscoveryStateTransitionTestL")); sl@0: DoBasicTestL(&CDiscovererTest::ScanDirectoryTestL, _L("ScanDirectoryTestL")); sl@0: DoBasicTestL(&CDiscovererTest::MultipleNotificationProcessingTestL, _L("MultipleNotificationProcessingTestL")); sl@0: DoBasicTestL(&CDiscovererTest::LanguageChangedNotificationTestL, _L("MLanguageChangedNotificationTestL")); sl@0: DoBasicTestL(&CDiscovererTest::SWINotificationProcessingTestL, _L("SWINotificationProcessingTestL")); sl@0: sl@0: //RunError tests sl@0: IdleScanningTimer_RunErrorTest(); sl@0: DirChangeNotifier_RunErrorTest(); sl@0: sl@0: // OOM tests sl@0: OOMCreateDeleteTest(); sl@0: // This test is not performed because the error from the final memory allocation failures are not sl@0: // propagated back to the calling function and cannot be dealt with. Therefore sl@0: // this test does not complete. However the normal test is performed above. sl@0: //DoOOMTestL(&CDiscovererTest::ResumeSuspendTestL, _L("OOM ResumeSuspendTestL")); sl@0: DoOOMTestL(&CDiscovererTest::DriveMountUnmountTestL, _L("OOM DriveMountUnmountTestL")); sl@0: DoOOMTestL(&CDiscovererTest::ProcessEntryTestL, _L("OOM ProcessEntryTestL")); sl@0: DoOOMTestL(&CDiscovererTest::ProcessSpiEntryTestL, _L("OOM ProcessSpiEntryTestL")); sl@0: DoOOMTestL(&CDiscovererTest::ValidateSpiEntryTestL, _L("OOM ValidateSpiEntryTestL")); sl@0: DoOOMTestL(&CDiscovererTest::ValidateEntryTestL, _L("OOM ValidateEntryTestL")); sl@0: DoOOMTestL(&CDiscovererTest::ProcessEntryPlugIn3TestL, _L("OOM ProcessEntryPlugIn3TestL")); sl@0: DoOOMTestL(&CDiscovererTest::ProcessSpiEntryPlugIn3TestL, _L("OOM ProcessSpiEntryPlugIn3TestL")); sl@0: DoOOMTestL(&CDiscovererTest::ValidateSpiEntryPlugIn3TestL, _L("OOM ValidateSpiEntryPlugIn3TestL")); sl@0: DoOOMTestL(&CDiscovererTest::ValidateEntryPlugIn3TestL, _L("OOM ValidateEntryPlugIn3TestL")); sl@0: DoOOMTestL(&CDiscovererTest::ScanDirectoryIncrementTestL, _L("OOM ScanDirectoryIncrementTestL")); sl@0: DoOOMTestL(&CDiscovererTest::StagedDiscoveryStateTransitionTestL, _L("StagedDiscoveryStateTransitionTestL")); sl@0: DoOOMTestL(&CDiscovererTest::AllAtOnceDiscoveryStateTransitionTestL, _L("AllAtOnceDiscoveryStateTransitionTestL")); sl@0: DoOOMTestL(&CDiscovererTest::ScanDirectoryTestL, _L("OOM ScanDirectoryTestL")); sl@0: // This test is not performed because the error from the final memory allocation failures are not sl@0: // propagated back to the calling function and cannot be dealt with. Therefore sl@0: // this test does not complete. However the normal test is performed above. sl@0: //DoOOMTestL(&CDiscovererTest::MultipleNotificationProcessingTestL, _L("MultipleNotificationProcessingTestL")); sl@0: // This test is not performed because the error from the final memory allocation failures are not sl@0: // propagated back to the calling function and cannot be dealt with. Therefore sl@0: // this test does not complete. However the normal test is performed above. sl@0: //DoOOMTestL(&CDiscovererTest::SWINotificationProcessingTestL, _L("OOM SWINotificationProcessingTestL")); sl@0: sl@0: __UHEAP_MARKEND; sl@0: return KErrNone; sl@0: } sl@0: sl@0: // Copy the Plugins to specific folder for testing purpose sl@0: LOCAL_C void CopyPlugins() sl@0: { sl@0: TInt err=KErrNone; sl@0: TRAP(err, EComTestUtils::FileManCopyFileL(KNewResourceFileNameOnZ, KNewResourceFileName)); sl@0: test(err==KErrNone); sl@0: TRAP(err, EComTestUtils::FileManCopyFileL(KNewExampleDllFileNameOnZ, KNewExampleDllFileName)); sl@0: test(err==KErrNone); sl@0: TRAP(err, EComTestUtils::FileManCopyFileL(KPlugIn3ResourceFileNameOnZ, KPlugIn3ResourceFileName)); sl@0: test(err==KErrNone); sl@0: TRAP(err, EComTestUtils::FileManCopyFileL(KPlugIn3ExampleDllFileNameOnZ, KPlugIn3ExampleDllFileName)); sl@0: test(err==KErrNone); sl@0: } sl@0: sl@0: // Deleting plugin from the RAM for cleanup purpose sl@0: inline LOCAL_C void DeleteTestPlugin() sl@0: { sl@0: TInt err=KErrNone; sl@0: TRAP(err, EComTestUtils::FileManDeleteFileL(KNewResourceFileName)); sl@0: TRAP(err, EComTestUtils::FileManDeleteFileL(KNewExampleDllFileName)); sl@0: TRAP(err, EComTestUtils::FileManDeleteFileL(KPlugIn3ResourceFileName)); sl@0: TRAP(err, EComTestUtils::FileManDeleteFileL(KPlugIn3ExampleDllFileName)); sl@0: } sl@0: sl@0: // sl@0: //Initialise the Active Scheduler sl@0: // sl@0: LOCAL_C void SetupL() sl@0: { sl@0: // Construct and install the Active Scheduler. The Active Schedular is needed sl@0: // by components used by this test as they are ActiveObjects. sl@0: TheActiveScheduler = new(ELeave)CActiveScheduler; sl@0: CActiveScheduler::Install(TheActiveScheduler); sl@0: } sl@0: sl@0: GLDEF_C TInt E32Main() sl@0: { sl@0: __UHEAP_MARK; sl@0: sl@0: test.Printf(_L("\n")); sl@0: test.Title(); sl@0: test.Start(_L("Discoverer Tests.")); sl@0: sl@0: TheTrapCleanup = CTrapCleanup::New(); sl@0: sl@0: TRAPD(err, SetupL()); sl@0: test(err == KErrNone); sl@0: sl@0: // Connect the file server instance sl@0: User::LeaveIfError(TheFs.Connect()); sl@0: sl@0: CopyPlugins(); sl@0: sl@0: // The reason for the folowing delay is: sl@0: // ECOM server could be already started. It means that when we copy some sl@0: // ECOM plugins from Z: to C: drive - ECOM server should look for and sl@0: // find the new ECOM plugins. The ECOM server uses for that an active object, sl@0: // which scans plugin directories. So the discovering service is asynchronous. sl@0: // We have to wait some time until it finishes. sl@0: // Otherwise ListImplementationsL could fail to find requested implementations. sl@0: User::After(KOneSecond * 3); sl@0: sl@0: sl@0: // Call the main tests sl@0: TRAP(err, DoTestsL()); sl@0: test(err == KErrNone); sl@0: sl@0: // Cleanup files. If the cleanup fails that is no problem, sl@0: // as any subsequent tests will replace them. The only downside sl@0: // would be the disk not being tidied sl@0: DeleteTestPlugin(); sl@0: sl@0: TheFs.Close(); sl@0: sl@0: delete TheActiveScheduler; sl@0: delete TheTrapCleanup; sl@0: sl@0: test.End(); sl@0: test.Close(); sl@0: sl@0: __UHEAP_MARKEND; sl@0: return (KErrNone); sl@0: } sl@0: