sl@0: // Copyright (c) 1997-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 the implementations of the classes sl@0: // which manage the internal structure of the registry. sl@0: // sl@0: // sl@0: sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include // for EStartupStateNonCritical sl@0: #include // for hash checking sl@0: #include // for RLoader sl@0: #include // for BaflUtils::FileExists sl@0: sl@0: #include "EComDebug.h" sl@0: #include sl@0: #include sl@0: #include sl@0: #include sl@0: #include "RegistryData.h" sl@0: #include "DowngradePath.h" sl@0: #include "DriveInfo.h" sl@0: #include "FileUtils.h" sl@0: #include "EComUidCodes.h" sl@0: #include "EComInternalErrorCodes.h" sl@0: #define UNUSED_VAR(a) a = a sl@0: sl@0: const TInt KRegVersion = -9999; sl@0: const TInt KDllExtensionLength=4; sl@0: _LIT(KDllExtension,".dll"); sl@0: // sl@0: //CRegistryData::TInterfaceStruct sl@0: sl@0: /** sl@0: Constructor for TInterfaceStruct sl@0: @post Its member variable is granulated as (1) sl@0: */ sl@0: CRegistryData::TInterfaceStruct::TInterfaceStruct(): sl@0: iImpData(1) sl@0: { sl@0: // Do nothing here sl@0: } sl@0: sl@0: /** sl@0: This method determins the order of two TInterfaceIndex objects sl@0: @param indexEntry1 first TInterfaceIndex object sl@0: @param indexEntry2 second TInterfaceIndex object sl@0: @return integer indicating the order of these two sl@0: @pre This object is fully constructed sl@0: */ sl@0: TInt CRegistryData::TInterfaceStruct::CompareInfUid(const TInterfaceIndex& indexEntry1, sl@0: const TInterfaceIndex& indexEntry2) sl@0: { sl@0: return CompareTUidValues(indexEntry1.iInterfaceUid.iUid,indexEntry2.iInterfaceUid.iUid); sl@0: } sl@0: sl@0: // sl@0: //CRegistryData::TImplStruct sl@0: sl@0: /** sl@0: Constructor for TImplStruct sl@0: @post Its member variables are initialised sl@0: */ sl@0: CRegistryData::TImplStruct::TImplStruct(): sl@0: iCurrentImpl(NULL), sl@0: iUnusedImpls(1) sl@0: { sl@0: // Do nothing here sl@0: } sl@0: sl@0: /** sl@0: This method determins the order of two TImplStruct objects sl@0: @param aEntry1 first TImplStruct object sl@0: @param aEntry2 second TImplStruct object sl@0: @return integer indicating the order of these two sl@0: @pre This object is fully constructed sl@0: */ sl@0: TInt CRegistryData::TImplStruct::CompareImplStructUid(const TImplStruct& aEntry1, sl@0: const TImplStruct& aEntry2) sl@0: { sl@0: return CompareTUidValues(aEntry1.iCurrentImpl->iImplInfo->ImplementationUid().iUid, sl@0: aEntry2.iCurrentImpl->iImplInfo->ImplementationUid().iUid); sl@0: } sl@0: sl@0: /** Comparer to determine the order of a TUid object (implementation UID) in relation to and a TImplStruct object. sl@0: @param aUid the Implementation UID to compare with. sl@0: @param aEntry the TImplStruct object to compare aUid against. sl@0: @return integer indicating the order of the two. sl@0: */ sl@0: TInt CRegistryData::TImplStruct::CompareUidAgainstImplStruct( sl@0: const TUid* aUid, sl@0: const TImplStruct& aEntry) sl@0: { sl@0: return CompareTUidValues(aUid->iUid,aEntry.iCurrentImpl->iImplInfo->ImplementationUid().iUid); sl@0: } sl@0: sl@0: // sl@0: // CRegistryData::CImplementationData class sl@0: sl@0: /** sl@0: Creates a new CImplemenatationData object. Note that CImplementationInformation needs to be constructed too, sl@0: for this object to be fully constructed. sl@0: @param aParent A pointer to the parent instance of CInterfaceData sl@0: @return A pointer to the newly created object. sl@0: @post This object is partly constructed and initialized and is on sl@0: the CleanupStack. sl@0: */ sl@0: CRegistryData::CImplementationData* CRegistryData::CImplementationData::NewLC(CInterfaceData* aParent) sl@0: { sl@0: CImplementationData* self=new(ELeave) CImplementationData(aParent); sl@0: CleanupStack::PushL(self); sl@0: return self; sl@0: } sl@0: sl@0: /** sl@0: Creates a new CImplemenatationData object. It takes parameters to create a CImplementationInformation object, sl@0: during 2nd phase construction and initializes iImplInfo to reference to this CImplementationInformation object. sl@0: @param aParent A pointer to the parent instance of CInterfaceData sl@0: @param aUid The unique Id of the new implementation sl@0: @param aVersion The version number of the new implementation sl@0: @param aName The display name of the new implementation sl@0: @param aDataType The data type supported by the new implementation sl@0: @param aOpaqueData Data for the new implementation which is not used by the ECom framework sl@0: @param aDrive The drive that the new implementation is on sl@0: @param aRomOnly The flag recording whether the new implementation may be loaded from ROM only sl@0: @param aRomBased The flag recording whether the new implementation is on ROM or is a later version of one on ROM sl@0: @return A pointer to the newly created object sl@0: @post This object is fully constructed and initialized. sl@0: */ sl@0: CRegistryData::CImplementationData* CRegistryData::CImplementationData::NewL(CInterfaceData* aParent, sl@0: TUid aUid, sl@0: TInt aVersion, sl@0: HBufC* aName, sl@0: HBufC8* aDataType, sl@0: HBufC8* aOpaqueData, sl@0: TDriveUnit aDrive, sl@0: TBool aRomOnly, sl@0: TBool aRomBased) sl@0: { sl@0: CImplementationData* self=new(ELeave) CImplementationData(aParent); sl@0: CleanupStack::PushL(self); sl@0: self->ConstructL(aUid, sl@0: aVersion, sl@0: aName, sl@0: aDataType, sl@0: aOpaqueData, sl@0: aDrive, sl@0: aRomOnly, sl@0: aRomBased); sl@0: CleanupStack::Pop(self); sl@0: return self; sl@0: } sl@0: sl@0: /** sl@0: Creates a new CImplemenatationData object. It takes parameters to create a CImplementationInformation object, sl@0: during 2nd phase construction and initializes iImplInfo to reference to this CImplementationInformation object. sl@0: @param aParent A pointer to the parent instance of CInterfaceData sl@0: @param aUid The unique Id of the new implementation sl@0: @param aVersion The version number of the new implementation sl@0: @param aName The display name of the new implementation sl@0: @param aDataType The data type supported by the new implementation sl@0: @param aOpaqueData Data for the new implementation which is not used by the ECom framework sl@0: @param aDrive The drive that the new implementation is on sl@0: @param aRomOnly The flag recording whether the new implementation may be loaded from ROM only sl@0: @param aRomBased The flag recording whether the new implementation is on ROM or is a later version of one on ROM sl@0: @param aExtendedInterfaces The pointer to the array recording the extended interfaces supported by this implementation. sl@0: NULL is available for PLUGIN without extended interfaces support. sl@0: @return A pointer to the newly created object sl@0: @post This object is fully constructed and initialized. sl@0: */ sl@0: CRegistryData::CImplementationData* CRegistryData::CImplementationData::NewL(CInterfaceData* aParent, sl@0: TUid aUid, sl@0: TInt aVersion, sl@0: HBufC* aName, sl@0: HBufC8* aDataType, sl@0: HBufC8* aOpaqueData, sl@0: TDriveUnit aDrive, sl@0: TBool aRomOnly, sl@0: TBool aRomBased, sl@0: RExtendedInterfacesArray* aExtendedInterfaces) sl@0: { sl@0: CImplementationData* self=new(ELeave) CImplementationData(aParent); sl@0: CleanupStack::PushL(self); sl@0: self->ConstructL(aUid, sl@0: aVersion, sl@0: aName, sl@0: aDataType, sl@0: aOpaqueData, sl@0: aDrive, sl@0: aRomOnly, sl@0: aRomBased, sl@0: aExtendedInterfaces); sl@0: CleanupStack::Pop(self); sl@0: return self; sl@0: } sl@0: sl@0: /** sl@0: Constructor for CImplementationData sl@0: @param aParent The parent interface data of this implementation sl@0: @post Its member variables are initialised sl@0: */ sl@0: CRegistryData::CImplementationData::CImplementationData( CInterfaceData* aParent): sl@0: CBase(), sl@0: iImplInfo(NULL), sl@0: iParent(aParent) sl@0: { sl@0: // Do nothing here sl@0: } sl@0: sl@0: /** sl@0: Destructor of CImplementationData sl@0: */ sl@0: CRegistryData::CImplementationData::~CImplementationData() sl@0: { sl@0: if(iImplInfo) sl@0: { sl@0: delete iImplInfo; sl@0: iImplInfo = NULL; sl@0: } sl@0: iParent = NULL; sl@0: } sl@0: sl@0: /** sl@0: The object's memory has been allocated. sl@0: @param aUid The unique Id of the new implementation sl@0: @param aVersion The version number of the new implementation sl@0: @param aName The display name of the new implementation sl@0: @param aDataType The data type supported by the new implementation sl@0: @param aOpaqueData Data for the new implementation which is not used by the ECom framework sl@0: @param aDrive The drive that the new implementation is on sl@0: @param aRomOnly The flag recording whether the new implementation may be loaded from ROM only sl@0: @param aRomBased The flag recording whether the new implementation is on ROM or is a later version of one on ROM sl@0: @pre This object is fully constructed. sl@0: @post This object is fully constructed and initialized. sl@0: */ sl@0: void CRegistryData::CImplementationData::ConstructL(TUid aUid, sl@0: TInt aVersion, sl@0: HBufC* aName, sl@0: HBufC8* aDataType, sl@0: HBufC8* aOpaqueData, sl@0: TDriveUnit aDrive, sl@0: TBool aRomOnly, sl@0: TBool aRomBased) sl@0: { sl@0: CImplementationInformation* newImpl = CImplementationInformation::NewL(aUid, sl@0: aVersion, sl@0: aName, sl@0: aDataType, sl@0: aOpaqueData, sl@0: aDrive, sl@0: aRomOnly, sl@0: aRomBased); sl@0: this->iImplInfo = newImpl; sl@0: } sl@0: sl@0: /** sl@0: The object's memory has been allocated. sl@0: @param aUid The unique Id of the new implementation sl@0: @param aVersion The version number of the new implementation sl@0: @param aName The display name of the new implementation sl@0: @param aDataType The data type supported by the new implementation sl@0: @param aOpaqueData Data for the new implementation which is not used by the ECom framework sl@0: @param aDrive The drive that the new implementation is on sl@0: @param aRomOnly The flag recording whether the new implementation may be loaded from ROM only sl@0: @param aRomBased The flag recording whether the new implementation is on ROM or is a later version of one on ROM sl@0: @param aExtendedInterfaces The pointer to the array recording the extended interfaces supported by this implementation. sl@0: NULL is available for PLUGIN without extended interfaces support. sl@0: @pre This object is fully constructed. sl@0: @post This object is fully constructed and initialized. sl@0: */ sl@0: void CRegistryData::CImplementationData::ConstructL(TUid aUid, sl@0: TInt aVersion, sl@0: HBufC* aName, sl@0: HBufC8* aDataType, sl@0: HBufC8* aOpaqueData, sl@0: TDriveUnit aDrive, sl@0: TBool aRomOnly, sl@0: TBool aRomBased, sl@0: RExtendedInterfacesArray* aExtendedInterfaces) sl@0: { sl@0: CImplementationInformation* newImpl = CImplementationInformation::NewL(aUid, sl@0: aVersion, sl@0: aName, sl@0: aDataType, sl@0: aOpaqueData, sl@0: aDrive, sl@0: aRomOnly, sl@0: aRomBased, sl@0: aExtendedInterfaces); sl@0: this->iImplInfo = newImpl; sl@0: } sl@0: sl@0: /** sl@0: Initialises member variable with the CImplementationInformation state specified in aStore. sl@0: @param aStore The stream to read the data from. sl@0: @pre This object is full constructed. sl@0: @post This object is set to the state specified in aStore. sl@0: */ sl@0: void CRegistryData::CImplementationData::InternalizeL(RReadStream& aStore) sl@0: { sl@0: if (iImplInfo) sl@0: { sl@0: delete iImplInfo; sl@0: iImplInfo = NULL; sl@0: } sl@0: CImplementationInformation* implInfo = 0; sl@0: implInfo=CImplementationInformation::NewLC(EFalse,aStore); sl@0: //as we never store the drive name we need to get this from the parent sl@0: implInfo->SetDrive(iParent->iParent->iParent->iDrive); sl@0: CleanupStack::Pop(1); sl@0: iImplInfo = implInfo; sl@0: implInfo = 0; sl@0: } sl@0: sl@0: sl@0: /** sl@0: Writes out the state of this member variable of type CImplementationInformation to aStore. sl@0: @param aStore The stream to store the data in. sl@0: @pre This object is fully constructed. sl@0: */ sl@0: void CRegistryData::CImplementationData::ExternalizeL(RWriteStream& aStore) const sl@0: { sl@0: iImplInfo->ExternalizeL(EFalse,aStore); sl@0: } sl@0: sl@0: /** sl@0: This method determines the order of two CImplementationData objects. sl@0: For backward compatibility reason, ECom allows different I/Fs using sl@0: the same implementation UID for when the device has an implementation UID sl@0: that is not unique. Although this is not a valid or supported situation ECOM sl@0: server should be robust. sl@0: @see FindImplementation sl@0: @see CompareTUidValues sl@0: @param aImpl1 first reference to CImplementationData object sl@0: @param aImpl2 second reference to CImplementationData object sl@0: @return integer indicating the order of these two sl@0: @pre This object is fully constructed sl@0: */ sl@0: TInt CRegistryData::CImplementationData::CompareImplUid(const CImplementationData& aImpl1, sl@0: const CImplementationData& aImpl2) sl@0: { sl@0: TInt ret = CompareTUidValues(aImpl1.iImplInfo->ImplementationUid().iUid, aImpl2.iImplInfo->ImplementationUid().iUid); sl@0: if (ret != 0) sl@0: { sl@0: return ret; sl@0: } sl@0: sl@0: // Implementation UIDs are equal, use I/F UID as tie breaker sl@0: return CompareTUidValues(aImpl1.iParent->iInterfaceUid.iUid, aImpl2.iParent->iInterfaceUid.iUid); sl@0: } sl@0: sl@0: /** sl@0: Similar to CompareImplUid above. This comparer only compare the Impl UID sl@0: and ignore the I/F UID part. sl@0: @param aImpl1 first reference to CImplementationData object sl@0: @param aImpl2 second reference to CImplementationData object sl@0: @return integer indicating the order of these two sl@0: @pre This object is fully constructed sl@0: */ sl@0: TInt CRegistryData::CImplementationData::CompareImplUidIgnoreIfUid(const CImplementationData& aImpl1, sl@0: const CImplementationData& aImpl2) sl@0: { sl@0: return CompareTUidValues(aImpl1.iImplInfo->ImplementationUid().iUid, sl@0: aImpl2.iImplInfo->ImplementationUid().iUid); sl@0: } sl@0: sl@0: /** sl@0: This method determines the order of an CImplementationData object in relation sl@0: to aUid. sl@0: @param aUid is really TUid. Cast it back before compare. sl@0: @param aImplData reference to CImplementationData object sl@0: @return integer indicating the order of the two. sl@0: */ sl@0: TInt CRegistryData::CImplementationData::CompareUidAgainstImplData( sl@0: const CImplementationData& aUid, sl@0: const CImplementationData& aImplData) sl@0: { sl@0: // The first argument aUid is really TUid. sl@0: const TUid* ImplUid = reinterpret_cast(&aUid); sl@0: return CompareTUidValues(ImplUid->iUid, sl@0: aImplData.iImplInfo->ImplementationUid().iUid); sl@0: } sl@0: sl@0: // sl@0: // CRegistryData::CInterfaceData class sl@0: sl@0: /** sl@0: Creates a new CInterfaceData object and leave it on the CleanupStack sl@0: @param aInterface The unique Id of this interface. sl@0: @param aParent A pointer to the parent dll data sl@0: @return A pointer to the newly created class. sl@0: @post This object is fully constructed and initialized and is on sl@0: the CleanupStack. sl@0: */ sl@0: CRegistryData::CInterfaceData* CRegistryData::CInterfaceData::NewLC(TUid aInterfaceUid,CDllData* aParent) sl@0: { sl@0: CInterfaceData* self=new(ELeave) CInterfaceData(aInterfaceUid,aParent); sl@0: CleanupStack::PushL(self); sl@0: self->ConstructL(); sl@0: return self; sl@0: } sl@0: sl@0: sl@0: /** sl@0: Creates a new CInterfaceData object using the supplied interface id sl@0: and leave it on the CleanupStack sl@0: @param aParent A pointer to the parent dll data sl@0: @return A pointer to the newly created class. sl@0: @post This object is fully constructed and initialized and is on sl@0: the CleanupStack. sl@0: */ sl@0: CRegistryData::CInterfaceData* CRegistryData::CInterfaceData::NewLC(CDllData* aParent) sl@0: { sl@0: CInterfaceData* self=new(ELeave) CInterfaceData(aParent); sl@0: CleanupStack::PushL(self); sl@0: self->ConstructL(); sl@0: return self; sl@0: } sl@0: sl@0: sl@0: CRegistryData::CInterfaceData::~CInterfaceData() sl@0: { sl@0: if(iImplementations) sl@0: { sl@0: iImplementations->ResetAndDestroy(); sl@0: delete iImplementations; sl@0: iImplementations = NULL; sl@0: } sl@0: iParent = NULL; sl@0: } sl@0: sl@0: /** sl@0: Adds the specified implementation to this interface in the registry. sl@0: @param aImplementation The implementation to add to this interface. sl@0: @pre This object is fully constructed. sl@0: @post aImplementation is added to the list of implementations for this interface. sl@0: */ sl@0: void CRegistryData::CInterfaceData::AddL(const CImplementationData* aImplementation) sl@0: { sl@0: #ifdef ECOM_TRACE sl@0: static int counter = 0; counter++; sl@0: __ECOM_TRACE6("ECOM: Implementation discovered (%04d) UID:0x%X interfaceUID:0x%X version:%d on drive:%d \"%S\"", counter, aImplementation->iImplInfo->ImplementationUid().iUid, aImplementation->iParent->iInterfaceUid.iUid, aImplementation->iImplInfo->Version(), (TInt)(aImplementation->iImplInfo->Drive()), &(aImplementation->iImplInfo->DisplayName())); sl@0: sl@0: #endif sl@0: sl@0: User::LeaveIfError(iImplementations->Append(aImplementation)); sl@0: } sl@0: sl@0: /** sl@0: Sets the uid of this interface to aInterfaceUid. sl@0: @param aInterfaceUid The Uid which this object should take. sl@0: @pre This object is fully constructed. sl@0: @post The Uid of this interface is set to aInterfaceUid sl@0: */ sl@0: void CRegistryData::CInterfaceData::SetInterfaceUid(TUid aInterfaceUid) sl@0: { sl@0: iInterfaceUid = aInterfaceUid; sl@0: } sl@0: sl@0: /** sl@0: Writes out this CInterfaceData to aStore. sl@0: @param aStore The stream to store the data in. sl@0: @pre The state of this object is stored in the stream aStore. sl@0: */ sl@0: void CRegistryData::CInterfaceData::ExternalizeL(RWriteStream& aStore) const sl@0: { sl@0: aStore.WriteInt32L(iInterfaceUid.iUid); sl@0: sl@0: if(iImplementations) sl@0: { sl@0: const TInt entryCount = iImplementations->Count(); sl@0: aStore.WriteInt32L(entryCount); sl@0: for(TInt i = 0; i < entryCount; ++i) sl@0: ((*iImplementations)[i])->ExternalizeL(aStore); sl@0: } sl@0: else sl@0: aStore.WriteInt32L(0); sl@0: } sl@0: sl@0: /** sl@0: Restores this CInterfaceData to the state specified in aStore. sl@0: @param aStore The stream to read the data from. sl@0: @param aPresent A boolean indicating whether the dll is still present sl@0: @pre This object is full constructed. sl@0: @post This object is set to the state specified in aStore. sl@0: */ sl@0: void CRegistryData::CInterfaceData::InternalizeL(RReadStream& aStore) sl@0: { sl@0: iInterfaceUid.iUid = aStore.ReadInt32L(); sl@0: sl@0: const TInt entryCount = aStore.ReadInt32L(); sl@0: if(entryCount < 0) sl@0: { sl@0: User::Leave(KErrCorrupt); sl@0: } sl@0: sl@0: for(TInt i = 0; i < entryCount; ++i) sl@0: { sl@0: CImplementationData* implementation = CImplementationData::NewLC(this); sl@0: implementation->InternalizeL(aStore); sl@0: AddL(implementation); sl@0: CleanupStack::Pop(implementation); //now owned by this interface sl@0: } sl@0: } sl@0: sl@0: /** sl@0: @param aParent A pointer to the parent dll data sl@0: */ sl@0: CRegistryData::CInterfaceData::CInterfaceData(CDllData* aParent) : sl@0: CBase(), sl@0: iParent(aParent) sl@0: { sl@0: // Do nothing here sl@0: } sl@0: sl@0: /** sl@0: @param aInterfaceUid The unique Id of this interface sl@0: @param aParent A pointer to the parent dll data sl@0: */ sl@0: CRegistryData::CInterfaceData::CInterfaceData(TUid aInterfaceUid, CDllData* aParent) : sl@0: CBase(), sl@0: iInterfaceUid(aInterfaceUid), sl@0: iParent(aParent) sl@0: { sl@0: // Do nothing here sl@0: } sl@0: sl@0: /** sl@0: Standard second phase construction function sl@0: @pre This object is fully constructed. sl@0: @post This object is fully constructed and initialized. sl@0: */ sl@0: void CRegistryData::CInterfaceData::ConstructL() sl@0: { sl@0: iImplementations = new(ELeave) RPointerArray; sl@0: } sl@0: sl@0: // sl@0: // CRegistryData::CDllData class sl@0: sl@0: /** sl@0: Creates a new CDllData object using aEntry and leaves it on the CleanupStack sl@0: @param aDllName the name of this dll sl@0: @param aDllModTime the modified time of this dll sl@0: @param aSecondUid Identifies type of the DLL. (PLUGIN or PLUGIN3) sl@0: @param aThirdUid Identifies a component uniquely. sl@0: @param aParent A pointer to the parent drive data sl@0: @return A pointer to the newly created object. sl@0: @post This object is fully constructed and initialized and on the CleanupStack. sl@0: */ sl@0: CRegistryData::CDllData* CRegistryData::CDllData::NewLC(const TDesC& aDllName,const TTime& aDllModTime,const TUid& aSecondUid,const TUid& aThirdUid,CDriveData* aParent) sl@0: { sl@0: CDllData* self=new(ELeave) CDllData(aParent); sl@0: CleanupStack::PushL(self); sl@0: self->ConstructL(aDllName,aDllModTime,aSecondUid,aThirdUid); sl@0: return self; sl@0: } sl@0: sl@0: /** sl@0: Creates a new CDllData object using aParent and leaves it on the CleanupStack sl@0: @param aParent A pointer to the parent drive data sl@0: @return A pointer to the newly created object. sl@0: @post This object is fully constructed and initialized and on the CleanupStack. sl@0: */ sl@0: CRegistryData::CDllData* CRegistryData::CDllData::NewLC( CDriveData* aParent) sl@0: { sl@0: CDllData* self=new(ELeave) CDllData(aParent); sl@0: CleanupStack::PushL(self); sl@0: self->ConstructL(); sl@0: return self; sl@0: } sl@0: sl@0: CRegistryData::CDllData::~CDllData() sl@0: { sl@0: if(iIfList) sl@0: { sl@0: // Clear the interface list and destroy its objects sl@0: iIfList->ResetAndDestroy(); sl@0: delete iIfList; sl@0: iIfList = NULL; sl@0: } sl@0: sl@0: // Unload this implementation dll. sl@0: delete iDllEntry; sl@0: iDllEntry = NULL; sl@0: iParent = NULL; sl@0: delete iRscFileExtension; sl@0: } sl@0: sl@0: /** sl@0: Adds the specified interface to this dll in the registry. sl@0: @param aInterface The interface to add to this dll sl@0: @pre This object is fully constructed. sl@0: @post aInterface is added to the list of interfaces in this dll. sl@0: */ sl@0: void CRegistryData::CDllData::AddL(const CInterfaceData* aInterface) sl@0: { sl@0: User::LeaveIfError(iIfList->Append(aInterface)); sl@0: } sl@0: sl@0: /** sl@0: Sets the resource extension for the plugin. Not set for read only internal drives. sl@0: @param aExt The resource extension to set sl@0: @pre This object is fully constructed. sl@0: @post aExt is added to the object. sl@0: */ sl@0: void CRegistryData::CDllData::SetResourceExtL(const TDesC& aExt) sl@0: { sl@0: delete iRscFileExtension; sl@0: iRscFileExtension = NULL; sl@0: iRscFileExtension = aExt.AllocL(); sl@0: } sl@0: sl@0: /** sl@0: Set the capability,the VID and do the Security check for this DLL. sl@0: @pre This object is fully constructed. sl@0: @return ETrue if the security check is done successfully. Otherwise EFalse is returned. sl@0: */ sl@0: TBool CRegistryData::CDllData::SaveSecurityInfoL() sl@0: { sl@0: iSecurityChecked = EFalse; sl@0: sl@0: //need to construct the full filename i.e. appending with the sl@0: //preconstructed drivepath name in CDriveData sl@0: TFileName dllFullName; sl@0: dllFullName.Append(iParent->iDrive.Name()); sl@0: dllFullName.Append(_L("\\sys\\bin\\")); sl@0: dllFullName.Append(iDllEntry->GetName()); sl@0: RLibrary::TInfoBuf infoBuf; sl@0: TInt ret = RLibrary::GetInfo(dllFullName, infoBuf); sl@0: if(ret != KErrNone) sl@0: { sl@0: return EFalse; sl@0: } sl@0: // Set the DLL's capability sl@0: iCapSet = infoBuf().iSecurityInfo.iCaps; sl@0: // Set the DLL's VID sl@0: iVid = infoBuf().iSecurityInfo.iVendorId; sl@0: // Now verify that SID identified in the resource file matches the SID of the Dll file sl@0: TBool match=iDllEntry->GetThirdUid() ==infoBuf().iSecurityInfo.iSecureId; sl@0: if (!match) sl@0: { sl@0: #ifdef __ECOM_TRACE sl@0: __ECOM_TRACE1("ERROR: Plugin SID Mismatch ERROR for %S.", &dllFullName); sl@0: #endif sl@0: return EFalse; sl@0: } sl@0: sl@0: // Now verify the two DLL's second Uid are of the same Uid type (used for distinguising between collection/collection3) sl@0: match=iDllEntry->GetSecondUid() ==infoBuf().iUids[1]; sl@0: if (!match) sl@0: { sl@0: #ifdef __ECOM_TRACE sl@0: __ECOM_TRACE1("ERROR: Plugin UID2 Mismatch ERROR for %S.", &iDllEntry->GetName()); sl@0: #endif sl@0: return EFalse; sl@0: } sl@0: sl@0: // On the emulator RLoader::CheckLibraryHash() returns KErrNotSupported. sl@0: // Also on the emulator RLoader does no hash checking for DLL's on removable drives. sl@0: // Therefore to be consistent ECOM does not do any hash checking itself on the emulator. sl@0: // Hence code is removed for emulator builds. sl@0: #if !defined(__WINSCW__) sl@0: // Verify hash is available sl@0: if(iParent->iParent->iCachedDriveInfo->DriveIsRemovableL(iParent->iDrive)) sl@0: { sl@0: RLoader loader; sl@0: TInt err = loader.Connect(); sl@0: if(err != KErrNone) sl@0: { sl@0: return EFalse; sl@0: } sl@0: err = loader.CheckLibraryHash(dllFullName, ETrue); sl@0: loader.Close(); sl@0: if(err != KErrNone) sl@0: { sl@0: #ifdef __ECOM_TRACE sl@0: __ECOM_TRACE2("INFO: Hash Check Failed for %S with error %d.", &dllFullName, err); sl@0: #endif sl@0: return EFalse; sl@0: } sl@0: } sl@0: #endif sl@0: iSecurityChecked = ETrue; sl@0: return ETrue; sl@0: } sl@0: sl@0: /** sl@0: Check whether security check has been performed if not go retrieve it. sl@0: @pre This object is fully constructed. sl@0: @return ETrue if the security check is done successfully. Otherwise EFalse is returned. sl@0: */ sl@0: TBool CRegistryData::CDllData::ProcessSecurityCheckL() sl@0: { sl@0: if(iSecurityChecked) sl@0: { sl@0: return ETrue; sl@0: } sl@0: return SaveSecurityInfoL(); sl@0: } sl@0: sl@0: sl@0: /** sl@0: Writes out the state of this CDllData to aStore. sl@0: @param aStore The stream to store the data in. sl@0: @pre This object is fully constructed. sl@0: */ sl@0: void CRegistryData::CDllData::ExternalizeL(RWriteStream& aStore) const sl@0: { sl@0: const CEComEntry& dllEntryData = *iDllEntry; sl@0: TInt size=dllEntryData.GetName().Length()-KDllExtensionLength; sl@0: aStore.WriteUint32L(size); sl@0: aStore.WriteL(dllEntryData.GetName(),size); sl@0: sl@0: aStore.WriteInt32L(dllEntryData.GetSecondUid().iUid); sl@0: aStore.WriteInt32L(dllEntryData.GetThirdUid().iUid); sl@0: TPckg modified(dllEntryData.GetModified()); sl@0: aStore.WriteL(modified); sl@0: sl@0: if(!iParent->iParent->iCachedDriveInfo->DriveIsReadOnlyInternalL(iParent->iDrive)) sl@0: { sl@0: if(iRscFileExtension) sl@0: { sl@0: size = iRscFileExtension->Length(); sl@0: aStore.WriteUint32L(size); sl@0: aStore.WriteL(iRscFileExtension->Des(),size); sl@0: } sl@0: else sl@0: { sl@0: aStore.WriteUint32L(0); sl@0: } sl@0: } sl@0: sl@0: if(iIfList) sl@0: { sl@0: const TInt entryCount = iIfList->Count(); sl@0: aStore.WriteInt32L(entryCount); sl@0: for(TInt i = 0; i < entryCount; ++i) sl@0: ((*iIfList)[i])->ExternalizeL(aStore); sl@0: } sl@0: else sl@0: aStore.WriteInt32L(0); sl@0: } sl@0: sl@0: /** sl@0: Restores this CDllData to the state specified in aStore. sl@0: @param aStore The stream to read the data from. sl@0: @pre This object is fully constructed. sl@0: @post The state of this object is restored to that specified in aStore. sl@0: */ sl@0: void CRegistryData::CDllData::InternalizeL(RReadStream& aStore) sl@0: { sl@0: TInt size=aStore.ReadUint32L(); sl@0: //The next "if" checks if size < 0 not size <= 0 because it seems the ECOM server externalizes sl@0: //the last file name as a string with length 0. If the fix is <= 0, then it makes it sl@0: //incompatible with the existing applications sl@0: if(size < 0 || size > KMaxFileName) sl@0: { sl@0: User::Leave(KErrCorrupt); sl@0: } sl@0: HBufC* name = HBufC::NewLC(size+KDllExtensionLength); sl@0: TPtr ptr=name->Des(); sl@0: aStore.ReadL(ptr,size); sl@0: ptr.Append(KDllExtension); sl@0: sl@0: TUid secondUid=TUid::Uid(aStore.ReadInt32L()); sl@0: TUid thirdUid=TUid::Uid(aStore.ReadInt32L()); sl@0: TTime dllModifiedTime; sl@0: TPckg modified(dllModifiedTime); sl@0: aStore.ReadL(modified); sl@0: sl@0: if(!iParent->iParent->iCachedDriveInfo->DriveIsReadOnlyInternalL(iParent->iDrive)) sl@0: { sl@0: size = aStore.ReadUint32L(); sl@0: if(size < 0 || size > KMaxFileName) sl@0: { sl@0: User::Leave(KErrCorrupt); sl@0: } sl@0: if(size) sl@0: { sl@0: iRscFileExtension = HBufC::NewL(size); sl@0: TPtr extPtr = iRscFileExtension->Des(); sl@0: aStore.ReadL(extPtr,size); sl@0: } sl@0: } sl@0: sl@0: const TInt entryCount = aStore.ReadInt32L(); sl@0: if(entryCount < 0) sl@0: { sl@0: User::Leave(KErrCorrupt); sl@0: } sl@0: sl@0: // Security check is deferred until the DLL is needed. sl@0: for(TInt i = 0; i < entryCount; ++i) sl@0: { sl@0: CInterfaceData* interface = CInterfaceData::NewLC(this); sl@0: interface->InternalizeL(aStore); sl@0: AddL(interface); sl@0: CleanupStack::Pop(interface); // now owned by dll sl@0: } sl@0: iDllEntry = CEComEntry::NewL(*name,secondUid,thirdUid); sl@0: iDllEntry->SetModified(dllModifiedTime); sl@0: CleanupStack::PopAndDestroy(name); sl@0: } sl@0: sl@0: /** sl@0: @param aParent The parent drive data of this implementation sl@0: */ sl@0: CRegistryData::CDllData::CDllData( CDriveData* aParent) : sl@0: CBase(), sl@0: iParent(aParent) sl@0: { sl@0: // Do nothing here sl@0: //Initialize empty capabilities here sl@0: iCapSet.SetEmpty(); sl@0: } sl@0: sl@0: /** sl@0: Standard second phase construction function. sl@0: @pre This object is fully constructed. sl@0: @post This object is fully constructed and initialized. sl@0: */ sl@0: void CRegistryData::CDllData::ConstructL() sl@0: { sl@0: iIfList = new(ELeave) RInterfaceList(2); sl@0: } sl@0: sl@0: /** sl@0: Standard second phase construction function with parameter. sl@0: @param aDllName the name of this dll sl@0: @param aDllModTime the modified time of this dll sl@0: @param aSecondUid Distinguishes between components having the same UID1 (which distinguishes between EXEs and DLLs) sl@0: UID2 identifies Interface Implementation Collections (collection and collection 3) sl@0: @param aThirdUid Identifies a component uniquely. In order to ensure that each binary that needs a distinguishing sl@0: UID is assigned a genuinely unique value. Symbian manages UID allocation through central database. sl@0: @pre This object is fully constructed. sl@0: @post This object is fully constructed and initialized. sl@0: */ sl@0: void CRegistryData::CDllData::ConstructL(const TDesC& aDllName,const TTime& aDllModTime,const TUid& aSecondUid,const TUid& aThirdUid) sl@0: { sl@0: iIfList = new(ELeave) RInterfaceList(2); sl@0: iDllEntry=CEComEntry::NewL(aDllName,aSecondUid,aThirdUid); sl@0: iDllEntry->SetModified(aDllModTime); sl@0: } sl@0: sl@0: sl@0: /** sl@0: Populate a caller-supplied TEntry instance with the data from this dll. sl@0: @param aEntry An entry to be populated (destination) sl@0: */ sl@0: void CRegistryData::CDllData::PopulateAnEntry(TEntry& aEntry) const sl@0: { sl@0: TPtr bufPtr=aEntry.iName.Des(); sl@0: bufPtr.Zero(); sl@0: bufPtr.Append(iParent->iDrive.Name()); sl@0: bufPtr.Append(_L("\\sys\\bin\\")); sl@0: bufPtr.Append(iDllEntry->GetName()); sl@0: aEntry.iType = TUidType(KDynamicLibraryUid,iDllEntry->GetSecondUid(),iDllEntry->GetThirdUid()); sl@0: aEntry.iModified= iDllEntry->GetModified(); sl@0: } sl@0: sl@0: sl@0: // sl@0: // CRegistryData::CDriveData class sl@0: sl@0: sl@0: /** sl@0: Creates a new CDriveData object and places it on the CleanupStack sl@0: @param aDrive Information on this drive sl@0: @param aParent A pointer to the parent registry data sl@0: @return A pointer to the newly created class. sl@0: @post This object is fully constructed and initialized and on the CleanupStack. sl@0: */ sl@0: CRegistryData::CDriveData* CRegistryData::CDriveData::NewLC(TDriveUnit aDrive, CRegistryData* aParent) sl@0: { sl@0: CDriveData* self=new(ELeave) CDriveData(aDrive,aParent); // calls c'tor sl@0: CleanupStack::PushL(self); // Make the construction safe by using the cleanup stack sl@0: self->ConstructL(); // Complete the 'construction'. sl@0: return self; sl@0: } sl@0: sl@0: /** sl@0: Destructor of CDriveData sl@0: */ sl@0: CRegistryData::CDriveData::~CDriveData() sl@0: { sl@0: if(iDllList) sl@0: { sl@0: // Clear the interface list and destroy its objects sl@0: iDllList->ResetAndDestroy(); sl@0: delete iDllList; sl@0: iDllList = NULL; sl@0: } sl@0: iParent=NULL; sl@0: } sl@0: sl@0: /** sl@0: Adds the specified CDllData to this drive data in the registry. sl@0: @param aDll The Dll to add to this drive sl@0: @pre This object is fully constructed. sl@0: @post aDll is added to the list of dll on this drive. sl@0: */ sl@0: void CRegistryData::CDriveData::AddL(const CDllData* aDll) sl@0: { sl@0: User::LeaveIfError(iDllList->Append(aDll)); sl@0: } sl@0: sl@0: /** sl@0: Returns the index of the DLL with this Uid in the Dll list. sl@0: @param aDllUid the UID of to be found DLL. sl@0: @return The index of this DLL in the DLL list. KErrNotFound if not found. sl@0: @pre This object is fully constructed. sl@0: */ sl@0: TInt CRegistryData::CDriveData::FindDllIndex(const TUid aDllUid) const sl@0: { sl@0: const TInt dllCount = iDllList->Count(); sl@0: for(TInt j = 0; j < dllCount; ++j) sl@0: { sl@0: // For each dll structure check the dll UID sl@0: CDllData* dll = (*iDllList)[j]; sl@0: if (aDllUid == dll->iDllEntry->GetThirdUid()) sl@0: { sl@0: // it is a match sl@0: return j; sl@0: } sl@0: } sl@0: return KErrNotFound; sl@0: } sl@0: sl@0: /** sl@0: Writes out the state of this CDriveData to file. sl@0: @param aFs A handle to an open file server session. sl@0: @param aDatFileName the dat file to persist to sl@0: @pre This object is fully constructed. sl@0: */ sl@0: void CRegistryData::CDriveData::ExternalizeL(RFs& aFs,const TDesC& aDatFileName) sl@0: { sl@0: // attempt to create the folders if they don't already exist sl@0: TInt mkDirError = aFs.MkDirAll(aDatFileName); sl@0: sl@0: if((mkDirError == KErrNone) || (mkDirError == KErrAlreadyExists)) sl@0: { sl@0: RFileWriteStream registryStream; sl@0: if(registryStream.Replace(aFs,aDatFileName, EFileWrite) == KErrNone) sl@0: { sl@0: CleanupClosePushL(registryStream); sl@0: // Write the version number as the first thing in the file stream. sl@0: registryStream.WriteInt32L(KRegVersion); sl@0: if(iDllList) sl@0: { sl@0: TInt entryCount = iDllList->Count(); sl@0: registryStream.WriteInt32L(entryCount); sl@0: // Now stream out the data sl@0: for(TInt i = 0; i < entryCount; ++i) sl@0: ((*iDllList)[i])->ExternalizeL(registryStream); sl@0: } sl@0: else sl@0: registryStream.WriteInt32L(0); sl@0: CleanupStack::PopAndDestroy(); // registryStream sl@0: } sl@0: } sl@0: else sl@0: { sl@0: User::Leave(mkDirError); sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Restores the state of this CDriveData from file. sl@0: @param aFs A handle to an open file server session. sl@0: @param aDatFileName the dat file name to internalize from sl@0: @pre This object is fully constructed. sl@0: @post The state of this object is restored to that specified sl@0: in the file on this drive. sl@0: */ sl@0: sl@0: void CRegistryData::CDriveData::InternalizeL(RFs& aFs,const TDesC& aDatFileName) sl@0: { sl@0: DoInternalizeL(aFs, aDatFileName); sl@0: } sl@0: /** sl@0: The method internalizes the data from aRegistryStream. sl@0: @param aFileName The ECOM registry file name. sl@0: @leave KErrNoMemory sl@0: @leave Some other system-wide error codes as well. sl@0: */ sl@0: void CRegistryData::CDriveData::DoInternalizeL(RFs& aFs, const TDesC& aFileName) sl@0: { sl@0: RFileReadStream registryStream; sl@0: User::LeaveIfError(registryStream.Open(aFs, aFileName, EFileRead)); sl@0: CleanupClosePushL(registryStream); sl@0: // Read in version number. If its not what expected - the file is treated as corrupted. sl@0: const TInt version = registryStream.ReadInt32L(); sl@0: if(version != KRegVersion) sl@0: { sl@0: User::Leave(KErrCorrupt); sl@0: } sl@0: const TInt entryCount = registryStream.ReadInt32L(); sl@0: if(entryCount < 0) sl@0: { sl@0: User::Leave(KErrCorrupt); sl@0: } sl@0: for(TInt i = 0; i < entryCount; ++i) sl@0: { sl@0: CDllData* dll = CDllData::NewLC(this); sl@0: dll->InternalizeL(registryStream); sl@0: // always add the DLL, DLL security check will be deferred sl@0: // until the DLL is needed. sl@0: AddL(dll); sl@0: CleanupStack::Pop(dll); // now owned by drive sl@0: } sl@0: CleanupStack::PopAndDestroy();//registryStream sl@0: } sl@0: sl@0: /** sl@0: Constructor for CDriveData sl@0: @param aDrive Information on this drive sl@0: @param aParent A pointer to the parent registry data sl@0: */ sl@0: CRegistryData::CDriveData::CDriveData(TDriveUnit aDrive, CRegistryData* aParent) : sl@0: CBase(), sl@0: sl@0: iDrive(aDrive), sl@0: iParent(aParent), sl@0: iDriveChanged(ETrue), sl@0: iRegistryChanged(EFalse) sl@0: { sl@0: } sl@0: sl@0: /** sl@0: Standard second phase construction function. sl@0: @pre This object is fully constructed. sl@0: @post This object is fully constructed and initialized. sl@0: */ sl@0: void CRegistryData::CDriveData::ConstructL() sl@0: { sl@0: iDllList = new(ELeave)TDll; sl@0: } sl@0: sl@0: // sl@0: // CRegistryData class sl@0: sl@0: /** sl@0: Standardized safe construction which leaves nothing on the cleanup stack. sl@0: This overload is used by CEComImplIndexPerfTest which plots sl@0: discovery time vs. granularity settings. sl@0: @leave KErrNoMemory. sl@0: @param aFs The open file session. sl@0: @param aInterfaceImplIndexGranularity granularity of main index. sl@0: @param aImplIndexGranularity granularity of auxiliary index. sl@0: @return The newly created instance of the registry data. sl@0: @post CRegistryData is fully constructed, and initialized. sl@0: */ sl@0: CRegistryData* CRegistryData::NewL(RFs& aFs, TInt aInterfaceImplIndexGranularity, TInt aImplIndexGranularity) sl@0: { sl@0: CRegistryData* self=new(ELeave) CRegistryData(aFs, aInterfaceImplIndexGranularity, aImplIndexGranularity); // calls c'tor sl@0: CleanupStack::PushL(self); // Make the construction safe by using the cleanup stack sl@0: self->ConstructL(); // Complete the 'construction'. sl@0: CleanupStack::Pop(self); sl@0: return self; sl@0: } sl@0: sl@0: /** sl@0: Standardized safe construction which leaves nothing on the cleanup stack. sl@0: This overload uses default granularities for the two indexes. These sl@0: default values were found to provide reasonably good performance at sl@0: the time of testing. sl@0: @leave KErrNoMemory. sl@0: @param aFs The open file session. sl@0: @return The newly created instance of the registry data. sl@0: @post CRegistryData is fully constructed, and initialized. sl@0: */ sl@0: CRegistryData* CRegistryData::NewL(RFs& aFs) sl@0: { sl@0: return NewL(aFs, KDefaultInterfaceImplIndexGranularity, KDefaultImplIndexGranularity); sl@0: } sl@0: sl@0: CRegistryData::~CRegistryData() sl@0: { sl@0: for(TInt index = 0; index < iInterfaceImplIndex.Count(); ++index) sl@0: { sl@0: iInterfaceImplIndex[index].Reset(); sl@0: } sl@0: iInterfaceImplIndex.Reset(); sl@0: sl@0: // Reset only. It does not own the pointers! sl@0: iImplIndex.Reset(); sl@0: sl@0: if(iRegistrations) sl@0: { sl@0: iRegistrations->ResetAndDestroy(); sl@0: delete iRegistrations; sl@0: iRegistrations = NULL; sl@0: } sl@0: // reset the cached language settings sl@0: RDowngradePath::Reset(); sl@0: sl@0: delete iCachedDriveInfo; sl@0: } sl@0: sl@0: /** sl@0: Lists the implementations given the interface UID. The filtered list is returned in the client provided sl@0: RImplDataArray parameter. sl@0: @param aInterfaceUid The Uid of the interface which the implementations should provide sl@0: @param aImplementationData Return value. The filtered list. sl@0: */ sl@0: void CRegistryData::ListImplementationsL(TUid aInterfaceUid, sl@0: RImplDataArray& aImplementationData) const sl@0: { sl@0: // If discoveries are taking place then the index is invalid so we cannot do sl@0: // this request sl@0: if(iCurrentlyDiscovering) sl@0: User::Leave(KEComErrListCurrentlyUnavailable); sl@0: sl@0: TBool securityPassed = EFalse; sl@0: //Do the security check sl@0: TInt index = 0; sl@0: while (!securityPassed) sl@0: { sl@0: aImplementationData.Reset(); sl@0: index = IndexedFind(aInterfaceUid); sl@0: if(index == KErrNotFound) sl@0: { sl@0: User::Leave(KEComErrNoInterfaceIdentified); sl@0: } sl@0: User::LeaveIfError(index); sl@0: sl@0: TImplContainerArray& implementationList = iInterfaceImplIndex[index].iImpData; sl@0: for (TInt j = 0; j < implementationList.Count(); j++) sl@0: { sl@0: CImplementationData* currentImplementation = implementationList[j].iCurrentImpl; sl@0: CDllData* dll = currentImplementation->iParent->iParent; sl@0: sl@0: securityPassed = dll->ProcessSecurityCheckL(); sl@0: if(!securityPassed) sl@0: { sl@0: // remove the implementations of the DLL from iInterfaceImplIndex sl@0: // and remove the DLL from its parent DLL list. sl@0: DeleteDllL(dll); sl@0: delete dll; sl@0: dll = NULL; sl@0: // don't continue processing implementations after we have deleted the DLL sl@0: // because the implementations list will have changed, so we need to sl@0: // re-do the IndexedFind sl@0: break; sl@0: } sl@0: else sl@0: { sl@0: User::LeaveIfError(aImplementationData.Append(currentImplementation)); sl@0: } sl@0: } sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Enables or disables the specified interface implementation within the registry. sl@0: @param aImplementationUid The interface implementation to change. sl@0: @param aState ETrue to enable the implementation, EFalse to disable it sl@0: @return KErrNone, KEComErrEnableFailed or KEComErrDisableFailed sl@0: @pre CRegistry is fully constructed, sl@0: @post The implementation is now enabled or disabled as described sl@0: by aState. sl@0: */ sl@0: TInt CRegistryData::SetEnabledState(TUid aImplementationUid, TBool aState) sl@0: { sl@0: CImplementationData* impData = NULL; sl@0: TUid dummy={0x00000000}; sl@0: sl@0: (void)FindImplementation(aImplementationUid, dummy, impData); sl@0: sl@0: if(impData != NULL) sl@0: { sl@0: impData->iImplInfo->SetDisabled((aState) ? EFalse : ETrue); sl@0: return KErrNone; sl@0: } sl@0: sl@0: return (aState) ? KEComErrEnableFailed : KEComErrDisableFailed; sl@0: } sl@0: sl@0: sl@0: /** sl@0: Temporarily uninstalls the interface implementation groups upon the specified drive. sl@0: (ie this can be undone later). sl@0: @param aDrive The identifier of the drive to uninstall. sl@0: @pre CRegistry is fully constructed sl@0: @post The registered implementation groupings sl@0: stored upon the specified drive are no sl@0: longer available for use. sl@0: */ sl@0: void CRegistryData::TemporaryUninstallL(const TDriveUnit& aDrive) sl@0: { sl@0: // Find the appropriate drive entry and remove it... sl@0: // Note : the drive may have already been taken offline, so no save is possible! sl@0: CDriveData* drive = NULL; sl@0: TInt driveIndex = FindDriveL(aDrive, drive); sl@0: if(driveIndex != KErrNotFound) sl@0: { sl@0: TInt dllIndex = drive->iDllList->Count(); sl@0: while(dllIndex > 0) sl@0: { sl@0: --dllIndex; sl@0: RemoveFromIndexL((*drive->iDllList)[dllIndex]); sl@0: } sl@0: iRegistrations->Remove(driveIndex); sl@0: delete drive; sl@0: sl@0: DriveChanged(aDrive, ETrue); sl@0: } sl@0: // The flag iRegistryChanged has been moved to each drive. sl@0: } sl@0: sl@0: /** sl@0: Undoes a temporary uninstall of the interface sl@0: implementation groups upon the specified drive. sl@0: (ie this can be undone later). sl@0: @leave KErrNoMemory, KErrReinstallFailed. sl@0: @param aDrive The identifier of the drive to reinstate. sl@0: @pre CRegistry is fully constructed sl@0: @post The registered implementation groupings sl@0: stored upon the specified drive are again sl@0: made available for use. sl@0: */ sl@0: void CRegistryData::UndoTemporaryUninstallL(const TDriveUnit& aDrive) sl@0: { sl@0: CDriveData* driveData = NULL; sl@0: // If the drive was not found then install it, otherwise do nothing because it sl@0: // is already there. NB. We could leave here but the problem with that is the sl@0: // use case where the drive has been mounted during registry load but the registrar sl@0: // is started afterwards, and makes calls to this method as it discovers drives sl@0: // for the first time. sl@0: if(FindDriveL(aDrive, driveData) == KErrNotFound) sl@0: { sl@0: // Add the drive entry sl@0: CDriveData* drive = CDriveData::NewLC(aDrive,this); sl@0: AddImplDataL(drive); sl@0: User::LeaveIfError(iRegistrations->Append(drive)); sl@0: CleanupStack::Pop(drive); // now owned by iRegistrations sl@0: sl@0: DriveChanged(aDrive, EFalse); sl@0: sl@0: // NB We DO need to set iRegistryChanged = ETrue because the idx file needs to be sl@0: // rewritten to include the reinstalled drive sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Update the staus of drives in the system sl@0: @param aDrive The drive that has changed sl@0: @param aDriveRemoved Indicates whether a drive has been removed or added sl@0: @pre CRegistry is fully constructed. sl@0: @post The state of the drive has been stored sl@0: */ sl@0: void CRegistryData::DriveChanged(const TDriveUnit& aDrive, TBool aDriveRemoved) sl@0: { sl@0: if(aDriveRemoved) sl@0: { sl@0: iRemovedDrives |= (0x1 << aDrive); sl@0: } sl@0: else sl@0: { sl@0: iRemovedDrives &= ~(0x1 << aDrive); sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Find if any Dll is regsitered in the drive unit. sl@0: @param aDrive The identifier of the drive to find if any Dll is registered. sl@0: @return ETrue if any Dll is registered in the drive, otherwise EFalse. sl@0: @pre CRegistry is fully constructed. sl@0: @post If any Dll is registered in the drive is returned if successfully. sl@0: */ sl@0: TBool CRegistryData::IsAnyDllRegisteredWithDriveL(const TDriveUnit& aDrive)const sl@0: { sl@0: CDriveData* driveData = NULL; sl@0: if(FindDriveL(aDrive, driveData) == KErrNotFound) sl@0: { sl@0: User::Leave(KEComErrDriveNotFound); sl@0: } sl@0: return driveData->iDllList->Count() > 0; sl@0: } sl@0: sl@0: /** sl@0: Determine if the specified implementation grouping already registered, but needs an update. sl@0: @param aDllUid The interface implementation collection data to add. sl@0: CRegistryData takes ownership of this object sl@0: so it should not be on the CleanupStack. sl@0: @param aModified The date of the last modification of the Interface sl@0: Implementation Collection. sl@0: @param aUpdate An output to indicate if the registry entry requires an update. sl@0: (Only valid if the return is ETrue). sl@0: @param aDriveData The drive data used to find DLL. sl@0: @return ETrue if the Interface Implementation Collection already has a registry entry. sl@0: EFalse otherwise. sl@0: @pre This object is fully constructed and there is also a valid drive entry sl@0: in the registry for aDrive sl@0: @post The new data is added to the registry sl@0: */ sl@0: TBool CRegistryData::IsRegisteredWithDate (TUid aDllUid, sl@0: const TTime& aModified, sl@0: TBool& aUpdate, sl@0: CDriveData* aDriveData) sl@0: { sl@0: // Find the appropriate drive entry for this sl@0: // Interface Implementation Collection sl@0: TBool found = EFalse; sl@0: TInt index = aDriveData->FindDllIndex(aDllUid); sl@0: if(index != KErrNotFound) sl@0: { sl@0: TDll* dllList = aDriveData->iDllList; sl@0: CDllData* dll = (*dllList)[index]; sl@0: const CEComEntry& dllEntry =*(dll->iDllEntry); sl@0: aUpdate = dllEntry.GetModified().Int64() < aModified.Int64(); sl@0: found = ETrue; sl@0: } sl@0: return found; sl@0: } sl@0: sl@0: /** sl@0: Adds data on a specific dll to the registry. The data to be added sl@0: is parsed by CRegistryParser. sl@0: @leave KErrNoMemory If the item could not be appended to the registry sl@0: @leave KEComErrDriveNotFound If aDrive is not a valid drive sl@0: @param aDrive The drive the registry data has been found on sl@0: @param aFoundDriveIndex The index of the registry data for the drive sl@0: the interface implementation collection data has been found on. sl@0: @param aDllData The dll data to add. CRegistryData takes ownership of this object sl@0: so it should not be on the CleanupStack. sl@0: @pre This object is fully constructed and there is also a valid drive entry sl@0: in the registry for aDrive sl@0: @post The new data is added to the registry sl@0: */ sl@0: void CRegistryData::AddDllDataL(const TDriveUnit& aDrive, TInt aFoundDriveIndex, CRegistryData::CDllData* aDllData) sl@0: { sl@0: const TInt regCount = iRegistrations->Count(); sl@0: if(regCount == 0) sl@0: User::Leave(KEComErrDriveNotFound); sl@0: sl@0: // Find the appropriate drive entry sl@0: CDriveData* drive = NULL; sl@0: if(aFoundDriveIndex == KErrNotFound) sl@0: { sl@0: if(FindDriveL(aDrive, drive) == KErrNotFound) sl@0: User::Leave(KEComErrDriveNotFound); sl@0: } sl@0: else sl@0: drive = (*iRegistrations)[aFoundDriveIndex]; sl@0: sl@0: // Append the new data to the list sl@0: TDll* dllList = drive->iDllList; sl@0: sl@0: // NOTE : This function MUST NOT leave after the following line because sl@0: // aDllData will potentially be 'owned' in two places (dllList and the caller) sl@0: // until we return and the caller can Pop. sl@0: TBool checkNeeded = ETrue; sl@0: TInt ifListCount=aDllData->iIfList->Count(); sl@0: for(TInt ifListCounter=0; ifListCounter < ifListCount; ifListCounter++) sl@0: { sl@0: // For each interface structure sl@0: CInterfaceData* interface = (*aDllData->iIfList)[ifListCounter]; sl@0: for(TInt impNum = 0; impNum < interface->iImplementations->Count(); ++impNum) sl@0: { sl@0: CImplementationData* implData = (*interface->iImplementations)[impNum]; sl@0: if(!InsertIntoIndexL(implData,checkNeeded)) sl@0: { sl@0: interface->iImplementations->Remove(impNum); //remove from implementation array sl@0: delete implData; //delete object sl@0: } sl@0: } sl@0: } sl@0: //add filtered list for legitimate implementations into registry sl@0: User::LeaveIfError(dllList->Append(aDllData)); sl@0: sl@0: drive->iRegistryChanged = ETrue; sl@0: } sl@0: sl@0: /** sl@0: Updates the data for a dll in the registry. sl@0: @leave KErrNoMemory If the indexes cannot be rebuilt sl@0: @param aFoundDriveIndex The index of the registry data for the drive sl@0: the interface implementation collection data has been found on. sl@0: @param aDllData The updated interface implementation collection data. sl@0: CRegistryData takes ownership of this object sl@0: so it should not be on the CleanupStack. sl@0: @pre This object is fully constructed sl@0: @post The dll data is updated in the registry sl@0: */ sl@0: void CRegistryData::UpdateDllDataL(const TDriveUnit& aDrive,TInt aFoundDriveIndex, CDllData* aDllData) sl@0: { sl@0: const TInt regCount = iRegistrations->Count(); sl@0: if(regCount == 0) sl@0: User::Leave(KEComErrDriveNotFound); sl@0: sl@0: // Find the appropriate drive entry sl@0: CDriveData* drive = (*iRegistrations)[aFoundDriveIndex]; sl@0: sl@0: TInt index = drive->FindDllIndex(aDllData->iDllEntry->GetThirdUid()); sl@0: sl@0: // find the dll entry and clear that too sl@0: if(index != KErrNotFound) sl@0: { sl@0: // Ok its a match, so remove this registry entry and call AddDllData() sl@0: //to add the new dll sl@0: TDll* dllList = drive->iDllList; sl@0: CDllData* olddll = (*dllList)[index]; sl@0: //remove 'old'implementations from iInterfaceImplIndex sl@0: RemoveFromIndexL(olddll); sl@0: //now remove old dll from the array and registry sl@0: dllList->Remove(index); sl@0: delete olddll; sl@0: AddDllDataL(aDrive,aFoundDriveIndex,aDllData); sl@0: drive->iRegistryChanged = ETrue; sl@0: sl@0: } sl@0: if (index == KErrNotFound) sl@0: { sl@0: __ECOM_TRACE("ECOM: PANIC in CRegistryData::UpdateDllDataL(), expected DLL data missing from ECOM registry"); sl@0: __ASSERT_DEBUG(EFalse, User::Panic (KEComServerPanicCategory, EEComPanic_CRegistryData_UpdateDllDataL_DllRegistrationMissing)); sl@0: } sl@0: } sl@0: sl@0: /** sl@0: The cue that newly discovered Dlls are about to be registered, sl@0: and, therefore, the registry index will be out of date until sl@0: DiscoveriesCompleteL is called. sl@0: @pre CRegistrar is fully constructed sl@0: @post iCurrentlyDiscovering is set and the index list is cleared. sl@0: */ sl@0: void CRegistryData::DiscoveriesBeginning() const sl@0: { sl@0: iCurrentlyDiscovering = ETrue; sl@0: // The registry is about to start changing so the index is now invalid sl@0: } sl@0: sl@0: /** sl@0: The cue to rebuild the registry indexes sl@0: @leave KErrNoMemory sl@0: @see CStore sl@0: @param aSuccessful Indicates whether discovery completed fully 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: @param aHasRegistryChanged The flag to indicate whether registry data has been changed sl@0: @pre CRegistrar is fully constructed sl@0: @post The internal access indexes have been rebuilt sl@0: */ sl@0: void CRegistryData::DiscoveriesCompleteL(TBool aSuccessful, TPluginProcessingTypeIdentifier aProcessingType, TBool& aHasRegistryChanged) sl@0: { sl@0: iCurrentlyDiscovering = EFalse; sl@0: if (!aSuccessful) sl@0: { sl@0: return; sl@0: } sl@0: //if the rediscovery is a result of iLanguageChanged, we should reset it to false sl@0: iLanguageChanged=EFalse; sl@0: // Warning: this method may be called from the CEComServer destructor. sl@0: // E.g. ecomservertest running OOM test on CEComServer::NewLC. sl@0: // The call stack: sl@0: // ~CEComServer delete iRegistrar, ~CRegistrar delete iDiscoverer, sl@0: // ~CDiscoverer calls CRegistrar::DiscoveriesComplete, sl@0: // then here. sl@0: // Hence if ! aSuccessful do not validate. sl@0: sl@0: if(aProcessingType != EPluginProcessingTypeCriticalOnly) sl@0: { sl@0: ValidateRegistryL(); sl@0: } sl@0: if(aProcessingType==EPluginProcessingTypeNonCriticalOnly || aProcessingType==EPluginProcessingTypeAll) sl@0: { sl@0: //first find whether there is any registry data changed sl@0: TInt regCount=iRegistrations->Count(); sl@0: for(TInt i = 0; i iRegistryChanged) sl@0: { sl@0: aHasRegistryChanged = ETrue; sl@0: //now having cached that the registry has changed we need sl@0: //to reset this flag in ALL the CDriveData, note that it is sl@0: //possible that more than one drive will have the flag set sl@0: (*iRegistrations)[i]->iRegistryChanged=EFalse; sl@0: } sl@0: } sl@0: //Also indicate registry change if any drives have been removed sl@0: aHasRegistryChanged |= (iRemovedDrives != 0); sl@0: iRemovedDrives = 0; sl@0: } sl@0: } sl@0: sl@0: /** sl@0: @fn SetDiscoveryFlag(const TDriveUnit aDrive) sl@0: Intended Usage : Set the flag to indicate the drive has change(s) sl@0: Error Condition : None. sl@0: @param aDrive The identifier of the drive changed. sl@0: @pre The CRegistrar must be fully constructed sl@0: @post The flag is set. sl@0: */ sl@0: sl@0: void CRegistryData::SetDiscoveryFlagL(const TDriveUnit& aDriveUnit) sl@0: { sl@0: CDriveData* drive = NULL; sl@0: TInt driveIndex = FindDriveL(aDriveUnit, drive); sl@0: if(driveIndex != KErrNotFound) sl@0: { sl@0: drive->iDriveChanged = ETrue; sl@0: } sl@0: sl@0: } sl@0: /** sl@0: Indicates whether the registry index is currently valid. The sl@0: index will not be valid if discoveries are currently taking place. sl@0: If the index is not currently valid then calls to sl@0: ListImplementationsL() cannot be serviced. sl@0: @return ETrue if the index is currently valid, EFalse otherwise. sl@0: @pre CRegistrar is fully constructed sl@0: */ sl@0: TBool CRegistryData::IndexValid() const sl@0: { sl@0: return !iCurrentlyDiscovering; sl@0: } sl@0: sl@0: /** sl@0: @param aCapabilitySet A capability set sl@0: @param aImplUid The Uid of the implementation for which info is required sl@0: @param aInterfaceUid The uid of the interface associated with aImplUid to find or less than sl@0: 0 if uid is unknown. sl@0: @param aEntry Output parameter which will contain the dll information sl@0: @param aImplInfo An output parameter. If the call succeeds it will point to the found implementation information, sl@0: NULL otherwise. sl@0: @param aIsOnRWDrive an output parameter. If the call is successful, this will sl@0: be set to ETrue if the implementation is on RW drive. EFalse if the sl@0: implementation is on ReadOnly drive. sl@0: @return KErrNone if the call succeeds, KErrNotFound - no implementation found, sl@0: KErrPermissionDenied - the caller has not enough capabilities to load the plugin. sl@0: @pre CRegistrar is fully constructed sl@0: */ sl@0: TInt CRegistryData::GetImplementationDllInfoForServer( sl@0: const TCapabilitySet& /*aCapabilitySet*/, sl@0: const TUid aImplUid, sl@0: const TUid aInterfaceUid, sl@0: TEntry& aEntry, sl@0: CImplementationInformation*& aImplInfo, sl@0: TBool& aIsOnRWDrive) const sl@0: { sl@0: aImplInfo = NULL; sl@0: CImplementationData* implData; sl@0: sl@0: TInt res = FindImplementation(aImplUid, aInterfaceUid, implData); sl@0: if (KErrNone != res) sl@0: { sl@0: return res; sl@0: } sl@0: sl@0: aImplInfo = implData->iImplInfo; sl@0: const CDllData* dll = implData->iParent->iParent; sl@0: dll->PopulateAnEntry(aEntry); sl@0: sl@0: TEComCachedDriveInfoIterator iter(*iCachedDriveInfo); sl@0: if (! iter.SetPos(dll->iParent->iDrive)) sl@0: { sl@0: res = KErrNotFound; sl@0: } sl@0: else sl@0: { sl@0: aIsOnRWDrive = iter.DriveIsWritable(); sl@0: } sl@0: return res; sl@0: } sl@0: sl@0: /** sl@0: @param aClientRequest A client request sl@0: @param aImplUid The Uid of the implementation for which info is required sl@0: @param aInterfaceUid The uid of the interface associated with aImplUid to find or less than sl@0: 0 if uid is unknown. sl@0: @param aEntry Output parameter which will contain the dll information sl@0: @param aImplInfo An output parameter. If the call succeeds it will point to the found implementation information, sl@0: NULL otherwise. sl@0: @param aSecurityCheckNeeded The bool value to identify whether the security check is needed here. The default value is false. sl@0: @return KErrNone if the call succeeds, KErrNotFound - no implementation found, sl@0: KErrPermissionDenied - the caller has not enough capabilities to load the plugin. sl@0: @pre CRegistrar is fully constructed sl@0: */ sl@0: TInt CRegistryData::GetImplementationDllInfoForClientL( sl@0: const TClientRequest& aClientRequest, sl@0: const TUid aImplUid, sl@0: const TUid aInterfaceUid, sl@0: TEntry& aEntry, sl@0: CImplementationInformation*& aImplInfo, sl@0: TBool aSecurityCheckNeeded)const sl@0: { sl@0: aImplInfo = NULL; sl@0: TInt res = KErrNotFound; sl@0: CImplementationData* implData; sl@0: if (!aSecurityCheckNeeded) sl@0: { sl@0: //The security check has been done already. All the invalid dll have been removed. sl@0: res = FindImplementation(aImplUid, aInterfaceUid, implData); sl@0: if (KErrNone != res) sl@0: { sl@0: return res; sl@0: } sl@0: CDllData* dll = implData->iParent->iParent; sl@0: dll->PopulateAnEntry(aEntry); sl@0: aImplInfo = implData->iImplInfo; sl@0: } sl@0: else sl@0: { sl@0: TBool securityPassed = EFalse; sl@0: while(!securityPassed) // go out of this loop either Security check is passed or no DLL found. sl@0: { sl@0: res = FindImplementation(aImplUid, aInterfaceUid, implData); sl@0: if (KErrNone != res) sl@0: { sl@0: return res; sl@0: } sl@0: CDllData* dll = implData->iParent->iParent; sl@0: // security check is deferred to here. sl@0: securityPassed = dll->ProcessSecurityCheckL(); sl@0: if(securityPassed) sl@0: { sl@0: if (!aClientRequest.CheckCapability(dll->iCapSet, *(implData->iImplInfo))) sl@0: { sl@0: return KErrPermissionDenied; sl@0: } sl@0: dll->PopulateAnEntry(aEntry); sl@0: aImplInfo = implData->iImplInfo; sl@0: } sl@0: else sl@0: { sl@0: // remove the implementations of the DLL from iInterfaceImplIndex sl@0: // and remove the DLL from its parent DLL list. sl@0: DeleteDllL(dll); sl@0: delete dll; sl@0: dll = NULL; sl@0: } sl@0: } sl@0: } sl@0: return res; sl@0: } sl@0: sl@0: /** sl@0: This method removes all implementations of This Dll from the iInterfaceImplIndex, sl@0: then delete this Dll from its parent Dll list. sl@0: @param aDllData the Dll to be cleaned. sl@0: @pre CRegistrar is fully constructed sl@0: */ sl@0: void CRegistryData::DeleteDllL(CDllData* aDllData) const sl@0: { sl@0: // find the index of the passed aDllData in its parent's Dll list. sl@0: TInt index = aDllData->iParent->FindDllIndex(aDllData->iDllEntry->GetThirdUid()); sl@0: sl@0: //removes all implementations of This Dll from the iInterfaceImplIndex sl@0: RemoveFromIndexL(aDllData); sl@0: if(index != KErrNotFound) sl@0: { sl@0: //now remove the dll from the array and registry sl@0: TDll* dllList = aDllData->iParent->iDllList; sl@0: dllList->Remove(index); sl@0: } sl@0: } sl@0: sl@0: sl@0: /** sl@0: Indicates whether the language downgrade path has changed. sl@0: This means we need to call NearestLanguageFile again sl@0: @return ETrue if the language downgrade path has changed, EFalse otherwise. sl@0: @pre CRegistrar is fully constructed sl@0: */ sl@0: TBool CRegistryData::HasLanguageChanged() const sl@0: { sl@0: return iLanguageChanged; sl@0: } sl@0: sl@0: sl@0: /** sl@0: @param aFs A handle to a connected file server. sl@0: */ sl@0: CRegistryData::CRegistryData(RFs& aFs, sl@0: TInt aInterfaceImplIndexGranularity, sl@0: TInt aImplIndexGranularity) : sl@0: CBase(), sl@0: iFs(aFs), sl@0: iInterfaceImplIndex(aInterfaceImplIndexGranularity), sl@0: iImplIndex(aImplIndexGranularity) sl@0: { sl@0: } sl@0: sl@0: sl@0: /** sl@0: Completes the safe construction of the CRegistryData object. sl@0: @leave KErrNoMemory. sl@0: @pre This object is constructed sl@0: @post This object is fully initialized sl@0: */ sl@0: void CRegistryData::ConstructL() sl@0: { sl@0: iCachedDriveInfo = CEComCachedDriveInfo::NewL(iFs); sl@0: sl@0: // Construction of the empty registration data structure here sl@0: iRegistrations = new(ELeave)TRegistration; sl@0: iSystemDrive=iFs.GetSystemDrive(); sl@0: // during construction we always need to call this function in sl@0: // order to initialise and store the language settings at boot time sl@0: TRAPD(err,iLanguageChanged=RDowngradePath::HasChangedL(iFs)) sl@0: if (err==KErrNoMemory) sl@0: { sl@0: User::LeaveNoMemory(); sl@0: } sl@0: } sl@0: sl@0: /** sl@0: To find the index entry for aImplementationUid. sl@0: @return The index of the item or KErrIndexEntryNotFound. sl@0: @pre This object is fully constructed sl@0: */ sl@0: TInt CRegistryData::IndexedFind(TUid aInterfaceUid) const sl@0: { sl@0: // Find the correct implementation sl@0: TInterfaceIndex key; sl@0: key.iInterfaceUid = aInterfaceUid; sl@0: return iInterfaceImplIndex.FindInOrder(key,TLinearOrder(TInterfaceStruct::CompareInfUid)); sl@0: } sl@0: sl@0: /** sl@0: Used by a TIdentityRelation to decide if two CDriveData match. sl@0: @return ETrue if the TDriveUnit inside the indexes match. sl@0: @param aIndexOne The first CDriveData to compare sl@0: @param aIndexTwo The second CDriveData to compare sl@0: */ sl@0: TBool CRegistryData::MatchOnDrive(const CRegistryData::CDriveData& aIndexOne, sl@0: const CRegistryData::CDriveData& aIndexTwo) sl@0: { sl@0: return aIndexOne.iDrive == aIndexTwo.iDrive; sl@0: } sl@0: sl@0: TInt CRegistryData::FindDriveL(const TDriveUnit& aDrive, sl@0: CRegistryData::CDriveData*& aDriveData)const sl@0: { sl@0: TInt index = KErrNotFound; sl@0: sl@0: // Set up the find parameters sl@0: TIdentityRelation identity(MatchOnDrive); sl@0: CDriveData* driveToMatch = CDriveData::NewLC(aDrive,const_cast(this)); sl@0: sl@0: index = iRegistrations->Find(driveToMatch, identity); sl@0: sl@0: CleanupStack::PopAndDestroy(driveToMatch); sl@0: sl@0: if(index != KErrNotFound) sl@0: aDriveData = (*iRegistrations)[index]; sl@0: else sl@0: aDriveData = NULL; sl@0: sl@0: return index; sl@0: } sl@0: sl@0: /** sl@0: Finds the instance of CImplementationData for given impl uid and interface uid if known. sl@0: @return KErrNone if impl found, otherwise KErrNotFound. sl@0: @param aImplUid The uid of the impl to find sl@0: @param aInterfaceUid The uid of the interface associated with the impl to find or less than sl@0: 0 if uid is unknown sl@0: @param aImplData Set to the instance CImplementationData found in the index or if sl@0: not found set to NULL. This argument is always overwritten. sl@0: */ sl@0: TInt CRegistryData::FindImplementation(const TUid aImplUid, const TUid aInterfaceUid, sl@0: CImplementationData*& aImplData) const sl@0: { sl@0: aImplData = NULL; sl@0: TInt i; sl@0: // Index is kept up to date even when discoveries are occurring sl@0: // therefore always search index for implementation sl@0: sl@0: // if aInterfaceUid is non zero use it to find position in index list sl@0: if(aInterfaceUid.iUid != 0) sl@0: { sl@0: i = IndexedFind(aInterfaceUid); sl@0: if(i == KErrNotFound) sl@0: { sl@0: return KErrNotFound; sl@0: } sl@0: sl@0: TInterfaceIndex listItem = iInterfaceImplIndex[i]; sl@0: i = listItem.iImpData.FindInOrder(aImplUid, sl@0: TImplStruct::CompareUidAgainstImplStruct); sl@0: if (i >= 0) sl@0: { sl@0: aImplData = listItem.iImpData[i].iCurrentImpl; sl@0: return KErrNone; sl@0: } sl@0: } sl@0: else // not given I/F UID. Use the iImplIndex. sl@0: { sl@0: // NB: 1. Impl UID should be globally unique. It is an error if impls sl@0: // have same Impl UID but different i/f UIDs. It means one of the sl@0: // plug-in supplier made an error in the .RSS file or it could be a sl@0: // deliberate attach. But to maintain backward compatibility, ECOM sl@0: // allows this error. sl@0: // (Multiple impls can have same i/f UID and same Impl UID. That is sl@0: // the upgrade situation and only one of them get stored in iImplIndex.) sl@0: sl@0: // 2. Entries in iImplIndex are ordered by Impl UID, and if duplicated, sl@0: // ordered by i/f UID. Here i/f UID is wild card. The situation is sl@0: // analogous to the array was built with InsertInOrderAllowRepeats. sl@0: // RPointerArray::SpecificFindInOrder is for ordered search in array sl@0: // with duplicates. Note: it is very expensive to instantiate sl@0: // a CImplementationData* just for binary search. Hence reinterpret sl@0: // cast the TUid* as CImplementationData*. CompareUidAgainstImplData sl@0: // knows to cast the first argument back to TUid. sl@0: i = iImplIndex.SpecificFindInOrder( sl@0: reinterpret_cast(&aImplUid), sl@0: TLinearOrder(CImplementationData::CompareUidAgainstImplData), sl@0: EArrayFindMode_First); sl@0: if (i == KErrNotFound) sl@0: { sl@0: return KErrNotFound; sl@0: } sl@0: sl@0: aImplData = iImplIndex[i]; sl@0: sl@0: // If duplicates exist, they are sorted according to i/f UID sl@0: // but we cannot take advantage of this order. To miniize risk sl@0: // of malware using duplicate Impl UID as DOS attack, ECOM applies sl@0: // the rule that ROMBased plug-in preferred over non-ROMBased, sl@0: // higher drive letter preferred over lower drive letter, and sl@0: // lastly lower i/f UID preferred over higher i/f UID. Must visit sl@0: // every duplicate to compare their ROMBasedness and driver letter sl@0: sl@0: TInt count = iImplIndex.Count(); sl@0: for (TInt j = i + 1; j < count; j++) sl@0: { sl@0: if ( iImplIndex[j]->iImplInfo->ImplementationUid().iUid != aImplUid.iUid ) sl@0: { sl@0: break; sl@0: } sl@0: sl@0: aImplData = SelectDuplicatedImpl(aImplData, iImplIndex[j]); sl@0: } sl@0: sl@0: return KErrNone; sl@0: } sl@0: sl@0: return KErrNotFound; sl@0: } sl@0: sl@0: /** sl@0: This function helps FindImplementation() to decide which of two sl@0: implementations with duplicated implementation UID to choose. sl@0: The selection rules are: sl@0: 1. ROM based plugins > non-ROM based. sl@0: 2. Higher drive letter > lower drive letter sl@0: 3. Lower I/F UID > Higher I/F UID (for backward compatibility reason) sl@0: sl@0: @param aImpl1 - one of the two implementations to compare. sl@0: @param aImpl2 - the other implementation to compare. sl@0: @return the preferred implementation. sl@0: */ sl@0: CRegistryData::CImplementationData* CRegistryData::SelectDuplicatedImpl( sl@0: const CImplementationData* aImpl1, sl@0: const CImplementationData* aImpl2) const sl@0: { sl@0: sl@0: #ifdef ECOM_TRACE sl@0: TPtrC oldName = aImpl1->iImplInfo->DisplayName().Left(60); sl@0: TPtrC newName = aImpl2->iImplInfo->DisplayName().Left(60); sl@0: TPtrC oldDll = aImpl1->iParent->iParent->iDllEntry->GetName(); sl@0: TPtrC newDll = aImpl2->iParent->iParent->iDllEntry->GetName(); sl@0: #endif sl@0: sl@0: const TInt KRomBasedFactor = 0x100; sl@0: sl@0: TInt drive1 = aImpl1->iImplInfo->Drive(); sl@0: if (aImpl1->iImplInfo->RomBased()) sl@0: { sl@0: drive1 |= KRomBasedFactor; sl@0: } sl@0: sl@0: TInt drive2 = aImpl2->iImplInfo->Drive(); sl@0: if (aImpl2->iImplInfo->RomBased()) sl@0: { sl@0: drive2 |= KRomBasedFactor; sl@0: } sl@0: sl@0: if (drive1 > drive2) sl@0: { sl@0: #ifdef ECOM_TRACE sl@0: if ((drive1 & KRomBasedFactor) && !(drive2 & KRomBasedFactor)) sl@0: { sl@0: __ECOM_TRACE5("ECOM dup impl UID resolution: rejected \"%S\" i/f UID 0x%X impl UID 0x%X, select ROM based %S over R/W %S", &newName, aImpl2->iParent->iInterfaceUid.iUid, aImpl2->iImplInfo->ImplementationUid().iUid, &oldDll, &newDll); sl@0: } sl@0: else sl@0: { sl@0: __ECOM_TRACE5("ECOM dup impl UID resolution: rejected \"%S\" i/f UID 0x%X impl UID 0x%X, select higher drive %S over %S", &newName, aImpl2->iParent->iInterfaceUid.iUid, aImpl2->iImplInfo->ImplementationUid().iUid, &oldDll, &newDll); sl@0: } sl@0: #endif sl@0: sl@0: return const_cast(aImpl1); sl@0: } sl@0: else if (drive1 < drive2) sl@0: { sl@0: #ifdef ECOM_TRACE sl@0: if ((drive2 & KRomBasedFactor) && !(drive1 & KRomBasedFactor)) sl@0: { sl@0: __ECOM_TRACE5("ECOM dup impl UID resolution: rejected \"%S\" i/f UID 0x%X impl UID 0x%X, select ROM based %S over R/W %S", &oldName, aImpl1->iParent->iInterfaceUid.iUid, aImpl1->iImplInfo->ImplementationUid().iUid, &newDll, &oldDll); sl@0: } sl@0: else sl@0: { sl@0: __ECOM_TRACE5("ECOM dup impl UID resolution: rejected \"%S\" i/f UID 0x%X impl UID 0x%X, select higher drive %S over %S", &oldName, aImpl1->iParent->iInterfaceUid.iUid, aImpl1->iImplInfo->ImplementationUid().iUid, &newDll, &oldDll); sl@0: } sl@0: #endif sl@0: sl@0: return const_cast(aImpl2); sl@0: } sl@0: // They are on the same drive. Choose the one with lower I/F UID. sl@0: else if (aImpl2->iParent->iInterfaceUid.iUid < aImpl1->iParent->iInterfaceUid.iUid) sl@0: { sl@0: #ifdef ECOM_TRACE sl@0: __ECOM_TRACE5("ECOM dup impl UID resolution: rejected \"%S\" i/f UID 0x%X impl UID 0x%X, select %S with lower i/f UID 0x%X", &oldName, aImpl1->iParent->iInterfaceUid.iUid, aImpl1->iImplInfo->ImplementationUid().iUid, &newDll, aImpl2->iParent->iInterfaceUid.iUid); sl@0: #endif sl@0: sl@0: return const_cast(aImpl2); sl@0: } sl@0: sl@0: #ifdef ECOM_TRACE sl@0: __ECOM_TRACE5("ECOM dup impl UID resolution: rejected \"%S\" i/f UID 0x%X impl UID 0x%X, select %S with lower i/f UID 0x%X", &newName, aImpl2->iParent->iInterfaceUid.iUid, aImpl2->iImplInfo->ImplementationUid().iUid, &oldDll, aImpl1->iParent->iInterfaceUid.iUid); sl@0: #endif sl@0: sl@0: return const_cast(aImpl1); sl@0: } sl@0: sl@0: /** sl@0: Checks each entry in the registry to ensure that both the RSC file and the sl@0: corresponding dll exist. If not the dll branch is removed sl@0: from the tree. If a drive branch is found which contains no dlls sl@0: it is also removed. sl@0: @pre This object is fully constructed. sl@0: @post Any out of date registry entries are removed from the tree. sl@0: */ sl@0: void CRegistryData::ValidateRegistryL() sl@0: { sl@0: TInt driveIndex = iRegistrations->Count(); sl@0: TFileName rscFileName; sl@0: while(driveIndex > 0) sl@0: { sl@0: --driveIndex; sl@0: CDriveData* drive = (*iRegistrations)[driveIndex]; sl@0: sl@0: if( !iCachedDriveInfo->DriveIsReadOnlyInternalL(drive->iDrive) && sl@0: drive->iDriveChanged) sl@0: { sl@0: TInt dllIndex = drive->iDllList->Count(); sl@0: while(dllIndex > 0) sl@0: { sl@0: --dllIndex; sl@0: CDllData* dll = (*drive->iDllList)[dllIndex]; sl@0: //reset the buffer first sl@0: rscFileName.Zero(); sl@0: rscFileName.Append(dll->iParent->iDrive.Name()); sl@0: rscFileName.Append(KEComResourceFilePath); sl@0: rscFileName.Append(dll->iDllEntry->GetName().Left(dll->iDllEntry->GetName().Length()-4)); sl@0: if(dll->iRscFileExtension) sl@0: { sl@0: rscFileName.Append(dll->iRscFileExtension->Des()); sl@0: } sl@0: sl@0: // check the existence of RSC file in resource\plugins\ directory. sl@0: // RSC filename is already a full name here. sl@0: TBool rscFileExistence = BaflUtils::FileExists(iFs, rscFileName); sl@0: sl@0: //save the security info, this will both check existence of the dll sl@0: //and cache the information for future use sl@0: TBool success=dll->SaveSecurityInfoL(); sl@0: // If this dll is not found or the corresponding RSC file is not found, then remove it from the registry sl@0: if(!success || !rscFileExistence) sl@0: { sl@0: //remove all implementations of this dll from iInterfaceImplIndex sl@0: RemoveFromIndexL(dll); sl@0: drive->iDllList->Remove(dllIndex); sl@0: //set flag to indicate registry data has been changed sl@0: drive->iRegistryChanged = ETrue; sl@0: delete dll; sl@0: } sl@0: } sl@0: } sl@0: } sl@0: } sl@0: sl@0: /** sl@0: Determines whether the new implmentation should be preferred over the existing implementation. sl@0: Validates later version implementations from R/W drives. sl@0: @param aOldImpl The array to append to sl@0: @param aNewImpl The item to append sl@0: @param aLegitimateImpl Flag, indicating whether current implementation is secure sl@0: @return The preferred implementation sl@0: @pre This object is fully constructed. sl@0: */ sl@0: CRegistryData::CImplementationData* CRegistryData::SelectPreferredImplL(CImplementationData* aOldImpl, sl@0: CImplementationData* aNewImpl, sl@0: TBool& aLegitimateImpl, sl@0: TBool aCheckIsNeeded) const sl@0: { sl@0: aLegitimateImpl = ETrue; sl@0: sl@0: TBool newIsRomOnly = aNewImpl->iImplInfo->RomOnly(); sl@0: TBool oldIsRomOnly = aOldImpl->iImplInfo->RomOnly(); sl@0: sl@0: /* In addition to selecting the highest version of an sl@0: implementation this check takes care of the following special sl@0: cases: sl@0: 1. Ensure that a higher-versioned RAM-based implementation sl@0: cannot override a ROM-based version sl@0: 2. Allows for the case where there are two versions of the sl@0: same ROM-only implementation on ROM, thus ensuring the sl@0: higher of the two versions is used. sl@0: */ sl@0: if(newIsRomOnly && !oldIsRomOnly) sl@0: { sl@0: return aNewImpl; sl@0: } sl@0: else if((newIsRomOnly && oldIsRomOnly)) sl@0: { sl@0: if(aNewImpl->iImplInfo->Version() > aOldImpl->iImplInfo->Version()) sl@0: { sl@0: return aNewImpl; sl@0: } sl@0: else if(aNewImpl->iImplInfo->Version() == aOldImpl->iImplInfo->Version()) sl@0: { sl@0: //any drive from Y-A has higher priority than Z drive sl@0: //any drive with a letter alphabetically greater has higher priority sl@0: if((aNewImpl->iImplInfo->Drive() != EDriveZ) && sl@0: (aOldImpl->iImplInfo->Drive() == EDriveZ || aNewImpl->iImplInfo->Drive() > aOldImpl->iImplInfo->Drive())) sl@0: { sl@0: return aNewImpl; sl@0: } sl@0: } sl@0: } sl@0: //all new implementations which are not flagged'RomOnly' sl@0: else if(!newIsRomOnly && !oldIsRomOnly) sl@0: { sl@0: TBool newIsReadOnly = iCachedDriveInfo->DriveIsReadOnlyInternalL(aNewImpl->iParent->iParent->iParent->iDrive); sl@0: TBool oldIsReadOnly = iCachedDriveInfo->DriveIsReadOnlyInternalL(aOldImpl->iParent->iParent->iParent->iDrive); sl@0: sl@0: if(aNewImpl->iImplInfo->Version() > aOldImpl->iImplInfo->Version()) sl@0: { sl@0: //filter for implementations on R/W drive trying to upgrade implementations on R/O drive sl@0: if((!newIsReadOnly && oldIsReadOnly) && aCheckIsNeeded) sl@0: { sl@0: //test if a later implementation version on R/W drive has the same filename as the current one before upgrading. sl@0: //If not,... sl@0: if(aOldImpl->iParent->iParent->iDllEntry->GetName().CompareF(aNewImpl->iParent->iParent->iDllEntry->GetName()) == 0) sl@0: { sl@0: return aNewImpl; sl@0: } sl@0: else sl@0: { sl@0: aLegitimateImpl = EFalse; //not secure sl@0: return aOldImpl; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: return aNewImpl; sl@0: } sl@0: } sl@0: else if(aNewImpl->iImplInfo->Version() == aOldImpl->iImplInfo->Version()) sl@0: { sl@0: //any drive from Y-A has higher priority than Z drive OR sl@0: //any drive with a letter alphabetically greater has higher priority sl@0: if((aNewImpl->iImplInfo->Drive() != EDriveZ) && sl@0: (aOldImpl->iImplInfo->Drive() == EDriveZ || aNewImpl->iImplInfo->Drive() > aOldImpl->iImplInfo->Drive())) sl@0: { sl@0: //filename check needs to performed on implementation from R/W drives, trying to sl@0: //upgrade implementations on R/O drive sl@0: if((!newIsReadOnly && oldIsReadOnly) && aCheckIsNeeded) sl@0: { sl@0: //test if a later implementation version on R/W drive has the same filename as the current one before upgrading. sl@0: //If not,... sl@0: if(aOldImpl->iParent->iParent->iDllEntry->GetName().CompareF(aNewImpl->iParent->iParent->iDllEntry->GetName()) == 0) sl@0: { sl@0: return aNewImpl; sl@0: } sl@0: else sl@0: { sl@0: aLegitimateImpl = EFalse; //not secure sl@0: return aOldImpl; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: return aNewImpl; sl@0: } sl@0: } sl@0: } sl@0: } sl@0: return aOldImpl; sl@0: } sl@0: sl@0: /** sl@0: This functions checks if a given implementation already exists in sl@0: the indexes. If it does exist, determine if the given implementation sl@0: should replace the existing one or not. sl@0: @param aIdxArray The container array of the interface to hold the implementation. sl@0: @param aNewImpl The implementation to filter. sl@0: @param aInsertMode whether aNewIMpl is a newcomer of the interface, or an sl@0: update of an existing implementation, or a older version of an sl@0: existing implementation. sl@0: @param aPosOfImplInArray return the index of aNewImpl in aIdxArray, sl@0: @param aLegitimateImpl Flag,indicating whether current implementation is secure sl@0: @pre This object is fully constructed. sl@0: @post none sl@0: */ sl@0: void sl@0: CRegistryData::FilterForLatestLegitimateImplL(TImplContainerArray& aIdxArray, sl@0: CImplementationData* aNewImpl, sl@0: TInsertImplMode& aInsertMode, sl@0: TInt& aPosOfImplInArray, sl@0: TBool& aLegitimateImpl, sl@0: TBool aCheckIsNeeded) sl@0: { sl@0: aInsertMode = EInsertImplUndefinedMode; sl@0: aLegitimateImpl = ETrue; sl@0: TImplStruct newImplStruct; sl@0: newImplStruct.iCurrentImpl = aNewImpl; sl@0: sl@0: TInt idxPos = aIdxArray.FindInOrder(newImplStruct, TLinearOrder (TImplStruct::CompareImplStructUid)); sl@0: sl@0: aPosOfImplInArray = idxPos; sl@0: sl@0: if(idxPos != KErrNotFound) sl@0: { sl@0: if(aNewImpl != SelectPreferredImplL(aIdxArray[idxPos].iCurrentImpl, sl@0: aNewImpl, sl@0: aLegitimateImpl, sl@0: aCheckIsNeeded)) sl@0: { sl@0: aInsertMode = EInsertImplAsUnusedImpl; sl@0: } sl@0: else sl@0: { sl@0: aInsertMode = EInsertImplAsUpgradeOfExistingImpl; sl@0: } sl@0: } sl@0: else sl@0: { sl@0: aInsertMode = EInsertImplAsNewcomerOfInterface; sl@0: } sl@0: } sl@0: sl@0: void CRegistryData::ResetTInterfaceIndex(TAny* aObject) sl@0: { sl@0: TInterfaceIndex* index=reinterpret_cast(aObject); sl@0: index->Reset(); sl@0: } sl@0: void CRegistryData::RemoveImplFromImplIndexCleanUp(TAny* aPtr) sl@0: { sl@0: TCleanupImplIndexEntry *aCleanup = (TCleanupImplIndexEntry*)aPtr; sl@0: aCleanup->iRegistryData->RemoveImplByAddrFromImplIndex(aCleanup->iImplEntry); sl@0: } sl@0: sl@0: /** sl@0: This method retrieves the data for security checks from the iInterfaceImplIndex sl@0: @param aImplPtr The new item to be checked sl@0: @param aCheckIsNeeded Boolean indicating, whether a filename check is needed sl@0: @pre This object is fully constructed. sl@0: @post aImplPtr has been checked and added to the index, or not, depending on the sl@0: outcome of the checks. sl@0: */ sl@0: TBool CRegistryData::InsertIntoIndexL(CImplementationData* aImplPtr, TBool aCheckIsNeeded) sl@0: { sl@0: TBool legitimateImpl = ETrue; sl@0: sl@0: TInterfaceIndex newIndexEl; sl@0: TInterfaceIndex* newElPtr; sl@0: //initialise sl@0: newIndexEl.iInterfaceUid = aImplPtr->iParent->iInterfaceUid; sl@0: CleanupStack::PushL(TCleanupItem(ResetTInterfaceIndex,&newIndexEl)); sl@0: sl@0: TInt indexPos = iInterfaceImplIndex.FindInOrder(newIndexEl,TLinearOrder(TInterfaceStruct::CompareInfUid)); sl@0: sl@0: sl@0: if(indexPos!=KErrNotFound) sl@0: newElPtr = &(iInterfaceImplIndex[indexPos]); sl@0: else sl@0: newElPtr = &newIndexEl; sl@0: sl@0: // For each implementation make sure we only have the latest version sl@0: TInt implIdxInContainerArray(KErrNotFound); sl@0: TInsertImplMode insertMode; sl@0: FilterForLatestLegitimateImplL(newElPtr->iImpData, sl@0: aImplPtr, sl@0: insertMode, sl@0: implIdxInContainerArray, sl@0: legitimateImpl, sl@0: aCheckIsNeeded); sl@0: sl@0: InsertImplInIndexesL(insertMode, sl@0: indexPos, sl@0: *newElPtr, sl@0: implIdxInContainerArray, sl@0: aImplPtr, sl@0: legitimateImpl); sl@0: sl@0: CleanupStack::Pop(); sl@0: sl@0: if (insertMode == EInsertImplAsUpgradeOfExistingImpl) sl@0: { sl@0: TUid ImplUid = aImplPtr->iImplInfo->ImplementationUid(); sl@0: iImplUpgradeCallBack.CallBack(ECallBackId_ImplUpgrade, &ImplUid); sl@0: } sl@0: sl@0: return legitimateImpl; sl@0: } sl@0: sl@0: /** Handle inserting a CImplementationData in iInterfaceImplIndex and sl@0: iImplIndex. sl@0: sl@0: @param aNewImpl the implementation to add to the indexes. sl@0: @param aInsertMode indicates whether the implementation is a newcover sl@0: of the interface, or is an update of existing implementation sl@0: or is an older version of an existing implementation. sl@0: @param aIfPosInInterfaceImplIndex is the index of the interface in sl@0: iInterfaceImplIndex. sl@0: @param aNewIfIndexEl the TInterfaceIndex object containing the implementation. sl@0: @param aImplPosInContainerArray is the index of the implementation in sl@0: the iImpData member of aNewIfIndexEl. sl@0: @param aLegitimateImpl indicate if the implementation passed security check or not. sl@0: @leave KErrNoMemory operation fails because the system is out of memory. sl@0: @leave KErrGeneral any non-specific programming error. sl@0: @leave KErrAlreadyExists the indexes already have an entry with the same sl@0: Impl. UID and Interface UID. sl@0: */ sl@0: void CRegistryData::InsertImplInIndexesL(TInsertImplMode aInsertMode, sl@0: TInt aIfPosInInterfaceImplIndex, sl@0: TInterfaceIndex& aNewIfIndexEl, sl@0: TInt aImplPosInContainerArray, sl@0: CImplementationData* aNewImpl, sl@0: TBool aLegitimateImpl) sl@0: { sl@0: if(aInsertMode == EInsertImplUndefinedMode) sl@0: { sl@0: // Will not happen because if FilterForLatestLegitimateImplL sl@0: // does not leave then insertMode is set to one of the valid sl@0: // values. If FilterForLatestLegitimateImplL leaves ecomserver exits. sl@0: User::Leave(KErrGeneral); sl@0: } sl@0: sl@0: // If not a legitimate implementation, aNewImpl will be deleted. sl@0: // Do not add it to the two indexes. sl@0: sl@0: if (!aLegitimateImpl) sl@0: { sl@0: return; sl@0: } sl@0: sl@0: TImplContainerArray& implContainerArray = aNewIfIndexEl.iImpData; sl@0: sl@0: if(aInsertMode == EInsertImplAsNewcomerOfInterface) sl@0: { sl@0: TImplStruct newImplStruct; sl@0: newImplStruct.iCurrentImpl = aNewImpl; sl@0: implContainerArray.InsertInOrderL(newImplStruct, TLinearOrder (TImplStruct::CompareImplStructUid)); sl@0: sl@0: #ifdef _DEBUG sl@0: // on debug build, check for duplicated implementation UID sl@0: // owned by different interfaces. sl@0: sl@0: TInt ii = iImplIndex.SpecificFindInOrder( sl@0: aNewImpl, sl@0: CImplementationData::CompareImplUidIgnoreIfUid, sl@0: EArrayFindMode_Any); sl@0: if (ii != KErrNotFound) sl@0: { sl@0: CImplementationData* otherImpl = iImplIndex[ii]; sl@0: TPtrC oldName = otherImpl->iImplInfo->DisplayName().Left(60); sl@0: TPtrC newName = aNewImpl->iImplInfo->DisplayName().Left(60); sl@0: __ECOM_TRACE5("** ECOM: error Impl UID %X, I/F UID %X DLL %S duplicating I/F %X DLL %S",\\ sl@0: aNewImpl->iImplInfo->ImplementationUid().iUid, \\ sl@0: aNewImpl->iParent->iInterfaceUid.iUid, &newName,\\ sl@0: otherImpl->iParent->iInterfaceUid.iUid, &oldName); sl@0: } sl@0: #endif sl@0: } sl@0: else if(aInsertMode == EInsertImplAsUpgradeOfExistingImpl) sl@0: { sl@0: TImplContainer& implContainer = implContainerArray[aImplPosInContainerArray]; sl@0: CImplementationData* oldImpl = implContainer.iCurrentImpl; sl@0: implContainer.iUnusedImpls.AppendL(oldImpl); sl@0: RemoveImplByAddrFromImplIndex(oldImpl); // ignore return code sl@0: implContainer.iCurrentImpl = aNewImpl; sl@0: sl@0: // We are replacing existing impl with aNewImpl. If existing sl@0: // corresponds to a ROM-based plug-in then aNewImpl is an sl@0: // update and qualifies as ROM-based under current policy. sl@0: if(oldImpl->iImplInfo->RomBased()) sl@0: { sl@0: aNewImpl->iImplInfo->SetRomBased(ETrue); sl@0: } sl@0: __ECOM_TRACE5("ECOM: Removed old implementation: UID:0x%X interfaceUID:0x%X version:%d on drive:%d \"%S\"", oldImpl->iImplInfo->ImplementationUid().iUid, oldImpl->iParent->iInterfaceUid.iUid, oldImpl->iImplInfo->Version(), (TInt)(oldImpl->iImplInfo->Drive()), &(oldImpl->iImplInfo->DisplayName())); sl@0: __ECOM_TRACE5("ECOM: Loaded new implementation: UID:0x%X interfaceUID:0x%X version:%d on drive:%d \"%S\"", aNewImpl->iImplInfo->ImplementationUid().iUid, aNewImpl->iParent->iInterfaceUid.iUid, aNewImpl->iImplInfo->Version(), (TInt)(aNewImpl->iImplInfo->Drive()), &(aNewImpl->iImplInfo->DisplayName())); sl@0: } sl@0: else if (aInsertMode == EInsertImplAsUnusedImpl) sl@0: { sl@0: TImplContainer& implContainer = implContainerArray[aImplPosInContainerArray]; sl@0: implContainer.iUnusedImpls.AppendL(aNewImpl); sl@0: sl@0: CImplementationData* currImpl = implContainer.iCurrentImpl; sl@0: __ECOM_TRACE5("ECOM: Kept old implementation: UID:0x%X interfaceUID:0x%X version:%d on drive:%d \"%S\"", currImpl->iImplInfo->ImplementationUid().iUid, currImpl->iParent->iInterfaceUid.iUid, currImpl->iImplInfo->Version(), (TInt)(currImpl->iImplInfo->Drive()), &(currImpl->iImplInfo->DisplayName())); sl@0: __ECOM_TRACE5("ECOM: Not loaded new implementation: UID:0x%X interfaceUID:0x%X version:%d on drive:%d \"%S\"", aNewImpl->iImplInfo->ImplementationUid().iUid, aNewImpl->iParent->iInterfaceUid.iUid, aNewImpl->iImplInfo->Version(), (TInt)(aNewImpl->iImplInfo->Drive()), &(aNewImpl->iImplInfo->DisplayName())); sl@0: // We are not replacing existing with aNewImpl. However, sl@0: // if aNewImpl corresponds to a ROM-based plug-in then the sl@0: // existing impl is an update and qualifies as ROM-based sl@0: // under current policy. sl@0: if(aNewImpl->iImplInfo->RomBased()) sl@0: { sl@0: currImpl->iImplInfo->SetRomBased(ETrue); sl@0: } sl@0: sl@0: return; // the trailing steps not applicable to downgrade situation sl@0: } sl@0: sl@0: User::LeaveIfError( InsertImplIntoImplIndex(aNewImpl) ); sl@0: sl@0: if(aIfPosInInterfaceImplIndex==KErrNotFound) sl@0: { sl@0: TCleanupImplIndexEntry aCleanup(this, aNewImpl); sl@0: CleanupStack::PushL(TCleanupItem(RemoveImplFromImplIndexCleanUp,&aCleanup)); sl@0: iInterfaceImplIndex.InsertInOrderL(aNewIfIndexEl, TLinearOrder(TInterfaceStruct::CompareInfUid)); sl@0: CleanupStack::Pop(); sl@0: } sl@0: } sl@0: sl@0: /** sl@0: This method takes removes all implementations of a plugin from the iInterfaceImplIndex sl@0: @param aDllData Reference to a fully constructed CDllData object sl@0: @pre This object is constructed sl@0: @post All implementations of the CDllData object, have been removed from the index sl@0: */ sl@0: void CRegistryData::RemoveFromIndexL(CDllData* aDllData) const sl@0: { sl@0: //remove 'old' implementations from iInterfaceImplIndex sl@0: TInt counter=0; sl@0: TImplStruct implStruct; sl@0: TInt ifListCount=aDllData->iIfList->Count(); sl@0: while(counter < ifListCount) sl@0: { sl@0: CInterfaceData* interface = (*aDllData->iIfList)[counter]; sl@0: TInterfaceIndex index; sl@0: index.iInterfaceUid = interface->iInterfaceUid; sl@0: sl@0: TInt indexPos = iInterfaceImplIndex.FindInOrder(index,TLinearOrder(TInterfaceStruct::CompareInfUid)); sl@0: TImplContainerArray* impDataArray = NULL; sl@0: TInt implCount=0; sl@0: sl@0: if(indexPos!=KErrNotFound) sl@0: { sl@0: impDataArray = &(iInterfaceImplIndex[indexPos].iImpData); sl@0: implCount=interface->iImplementations->Count(); sl@0: for(TInt i=0; iiImplementations)[i]; sl@0: TInt impPos = impDataArray->FindInOrder(implStruct, TLinearOrder (TImplStruct::CompareImplStructUid)); sl@0: if(impPos!=KErrNotFound) sl@0: { sl@0: TImplContainer& implContainer = (*impDataArray)[impPos]; sl@0: // remove from unused list if exists sl@0: TInt count = implContainer.iUnusedImpls.Count(); sl@0: while(count > 0) sl@0: { sl@0: --count; sl@0: if(implContainer.iUnusedImpls[count]->iParent->iParent == aDllData) sl@0: { sl@0: implContainer.iUnusedImpls.Remove(count); sl@0: } sl@0: } sl@0: sl@0: // update current entry sl@0: if(implContainer.iCurrentImpl->iParent->iParent == aDllData) sl@0: { sl@0: // do not care about the return code. sl@0: RemoveImplByAddrFromImplIndex(implContainer.iCurrentImpl); sl@0: sl@0: TInt implContainerUnusedImplCount=implContainer.iUnusedImpls.Count(); sl@0: // no unused impl's therefore no rollback and remove entry sl@0: if(implContainerUnusedImplCount == 0) sl@0: { sl@0: implContainer.Reset(); sl@0: impDataArray->Remove(impPos); sl@0: } sl@0: else sl@0: { sl@0: // Rollback to implementation from unused array sl@0: // copy first from unused array sl@0: TInt selectedPos = 0; sl@0: implContainer.iCurrentImpl = implContainer.iUnusedImpls[selectedPos]; sl@0: sl@0: // now check if any others in list should be preferred sl@0: for(count = 1;count < implContainerUnusedImplCount; count++) sl@0: { sl@0: // determine which implementation should be used. sl@0: // no security check required as this will have already been done when impl was sl@0: // first added to unused list. sl@0: TBool dummLegitimateImpl; sl@0: implContainer.iCurrentImpl = SelectPreferredImplL(implContainer.iCurrentImpl, sl@0: implContainer.iUnusedImpls[count], sl@0: dummLegitimateImpl, sl@0: EFalse); sl@0: if(implContainer.iCurrentImpl == implContainer.iUnusedImpls[count]) sl@0: { sl@0: selectedPos = count; sl@0: } sl@0: } sl@0: implContainer.iUnusedImpls.Remove(selectedPos); sl@0: User::LeaveIfError(InsertImplIntoImplIndex(implContainer.iCurrentImpl)); sl@0: #ifdef ECOM_TRACE sl@0: { sl@0: CImplementationData* p = implContainer.iCurrentImpl; sl@0: __ECOM_TRACE5("ECOM: unused implementation restored UID:0x%X interfaceUID:0x%X version:%d on drive:%d \"%S\"", p->iImplInfo->ImplementationUid().iUid, p->iParent->iInterfaceUid.iUid, p->iImplInfo->Version(), (TInt)(p->iImplInfo->Drive()), &(p->iImplInfo->DisplayName())); sl@0: } sl@0: #endif sl@0: } sl@0: } sl@0: } sl@0: //To make sure it is removed from the implIndex no matter what and that the return code is ignored. sl@0: //The previous removal is still required so that a subsequent InsertImplIntoImplIndex is possible sl@0: //for an implementation. e.g. an implementaion is on different drives and only one of htem was removed. sl@0: // The other one should now make it to the iImplIndex through InsertImplIntoImplIndex. If it wasn't sl@0: // removed, before the insert operation, it will fail since both the implementations hav the same sl@0: // impl UID. sl@0: RemoveImplByAddrFromImplIndex((*interface->iImplementations)[i]); sl@0: } sl@0: if(impDataArray->Count() == 0) sl@0: { sl@0: iInterfaceImplIndex[indexPos].Reset(); sl@0: iInterfaceImplIndex.Remove(indexPos); sl@0: } sl@0: } sl@0: counter++; sl@0: } sl@0: } sl@0: sl@0: /** sl@0: This method inserts implementations into the index sl@0: @param aDriveData Drive data in which we are searching the implementations sl@0: @pre This object is fully constructed. sl@0: @post all implemementations found, have been inserted into the index. sl@0: */ sl@0: void CRegistryData::AddImplDataL(CDriveData* aDriveData) sl@0: { sl@0: TBool checkIsNeeded = EFalse; sl@0: TInt dllCount=aDriveData->iDllList->Count(); sl@0: for(TInt dllCounter=0; dllCounter < dllCount;dllCounter++) sl@0: { sl@0: CDllData* aDllData = (*aDriveData->iDllList)[dllCounter]; sl@0: TInt interfaceCount=aDllData->iIfList->Count(); sl@0: for(TInt ifListCounter=0; ifListCounter < interfaceCount; ifListCounter++) sl@0: { sl@0: // For each interface structure sl@0: CInterfaceData* interface = (*aDllData->iIfList)[ifListCounter]; sl@0: TInt implementationCount=interface->iImplementations->Count(); sl@0: for(TInt impNum = 0; impNum < implementationCount; ++impNum) sl@0: { sl@0: CImplementationData* implData = (*interface->iImplementations)[impNum]; sl@0: TInt retValue = InsertIntoIndexL(implData, checkIsNeeded); sl@0: //we know that during internalize, all implementations are legitimate, sl@0: //as the check was already performed during discovery. No check is sl@0: //performed at this stage, therefore ignore return value, as it will always sl@0: //KErrNone sl@0: } sl@0: } sl@0: } sl@0: } sl@0: sl@0: sl@0: /** This method does a search by address and removes the specified entry from iImplIndex. sl@0: sl@0: @param aPtr is the entry to remove sl@0: @return True if aPtr is removed from iImplIndex. False if aPtr is not sl@0: in iImplIndex, i.e. nothing is removed. sl@0: */ sl@0: TBool CRegistryData::RemoveImplByAddrFromImplIndex(CImplementationData* aPtr) const sl@0: { sl@0: TInt aIdx = iImplIndex.Find(aPtr); sl@0: if (aIdx != KErrNotFound) sl@0: { sl@0: // This linear pointer search ensures a safe removal of the impl from iImplIndex so that it is free from a dead object. sl@0: sl@0: // The array does not own the pointer. Do not delete! sl@0: iImplIndex.Remove(aIdx); sl@0: return ETrue; sl@0: } sl@0: return EFalse; sl@0: } sl@0: sl@0: /** This method inserts the entry aNewImpl into iImplIndex. sl@0: sl@0: @param aNewImpl the item to add to iImplIndex. sl@0: @return KErrNone aNewImpl is successfully added to the index. sl@0: KErrAlreadyExists iImplIndex has an entry with the same sl@0: implementation UID and same parent Interface UID. sl@0: KErrNoMemory fail to insert due to out of memory. sl@0: Other system wide errors. sl@0: @pre aNewImpl is fully constructed. sl@0: */ sl@0: TInt CRegistryData::InsertImplIntoImplIndex(CImplementationData* aNewImpl) const sl@0: { sl@0: TLinearOrder ImplUidComparator(CImplementationData::CompareImplUid); sl@0: sl@0: return iImplIndex.InsertInOrder(aNewImpl, ImplUidComparator); sl@0: } sl@0: sl@0: /** This method checks whether the language downgrade path has been changed. sl@0: If it is, save language downgrade path information and set related flag true. sl@0: sl@0: @param aLanguageChanged the returned value to indicate language changed. sl@0: */ sl@0: void CRegistryData::LanguageChangedL(TBool& aLanguageChanged) sl@0: { sl@0: iLanguageChanged = RDowngradePath::HasChangedL(iFs); sl@0: aLanguageChanged = iLanguageChanged; sl@0: } sl@0: sl@0: /** setter sl@0: If need to unset the callback, use a TCallBackWithArg constructed with no sl@0: arguments. sl@0: */ sl@0: void CRegistryData::SetImplUpgradeCallBack(const TCallBackWithArg& aCallBack) sl@0: { sl@0: iImplUpgradeCallBack = aCallBack; sl@0: } sl@0: sl@0: #ifdef __ECOM_SERVER_PERFORMANCE__ sl@0: /** sl@0: This method calculates the drive, plugins, interfaces, implementations counts for the drive sl@0: type set sl@0: @param aType The drive type for which the counts should be calculated sl@0: @param aCounts Holds the calculated counts sl@0: @pre This object is constructed sl@0: */ sl@0: void CRegistryData::GetRegistryCountsL(TInt aType, RegistryCounts::TRegistryCounts& aCounts) const sl@0: { sl@0: aCounts.iDrives = 0; sl@0: aCounts.iImplementations = 0; sl@0: aCounts.iInterfaces = 0; sl@0: aCounts.iDlls = 0; sl@0: sl@0: for(TInt driveIndex = 0; driveIndex< iRegistrations->Count(); driveIndex++) sl@0: { sl@0: CDriveData* drive = (*iRegistrations)[driveIndex]; sl@0: TBool isReadOnly = iCachedDriveInfo->DriveIsReadOnlyInternalL(drive->iDrive); sl@0: sl@0: if((aType == RegistryCounts::TRegistryCounts::EAll) || sl@0: (aType == RegistryCounts::TRegistryCounts::ERoInternal && isReadOnly) || sl@0: (aType == RegistryCounts::TRegistryCounts::ENonRoInternal && !isReadOnly)) sl@0: { sl@0: aCounts.iDrives++; sl@0: aCounts.iDlls += drive->iDllList->Count(); sl@0: for(TInt dllIndex = 0; dllIndex < drive->iDllList->Count(); dllIndex++) sl@0: { sl@0: CDllData* dllList = (*drive->iDllList)[dllIndex]; sl@0: aCounts.iInterfaces += dllList->iIfList->Count(); sl@0: for(TInt ifIndex = 0; ifIndex < dllList->iIfList->Count(); ifIndex++) sl@0: { sl@0: CInterfaceData* ifList = (*dllList->iIfList)[ifIndex]; sl@0: aCounts.iImplementations += ifList->iImplementations->Count(); sl@0: } sl@0: } sl@0: } sl@0: } sl@0: } sl@0: sl@0: #endif // __ECOM_SERVER_PERFORMANCE__