Update contrib.
1 // Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // This file contains the implementations of the classes
15 // which manage the internal structure of the registry.
23 #include <startup.hrh> // for EStartupStateNonCritical
24 #include <e32ldr.h> // for hash checking
25 #include <e32ldr_private.h> // for RLoader
26 #include <bautils.h> // for BaflUtils::FileExists
28 #include "EComDebug.h"
29 #include <ecom/ecom.h>
30 #include <ecom/ecomerrorcodes.h>
31 #include <ecom/ecomresolverparams.h>
32 #include <ecom/implementationinformation.h>
33 #include "RegistryData.h"
34 #include "DowngradePath.h"
35 #include "DriveInfo.h"
36 #include "FileUtils.h"
37 #include "EComUidCodes.h"
38 #include "EComInternalErrorCodes.h"
39 #define UNUSED_VAR(a) a = a
41 const TInt KRegVersion = -9999;
42 const TInt KDllExtensionLength=4;
43 _LIT(KDllExtension,".dll");
45 //CRegistryData::TInterfaceStruct
48 Constructor for TInterfaceStruct
49 @post Its member variable is granulated as (1)
51 CRegistryData::TInterfaceStruct::TInterfaceStruct():
58 This method determins the order of two TInterfaceIndex objects
59 @param indexEntry1 first TInterfaceIndex object
60 @param indexEntry2 second TInterfaceIndex object
61 @return integer indicating the order of these two
62 @pre This object is fully constructed
64 TInt CRegistryData::TInterfaceStruct::CompareInfUid(const TInterfaceIndex& indexEntry1,
65 const TInterfaceIndex& indexEntry2)
67 return CompareTUidValues(indexEntry1.iInterfaceUid.iUid,indexEntry2.iInterfaceUid.iUid);
71 //CRegistryData::TImplStruct
74 Constructor for TImplStruct
75 @post Its member variables are initialised
77 CRegistryData::TImplStruct::TImplStruct():
85 This method determins the order of two TImplStruct objects
86 @param aEntry1 first TImplStruct object
87 @param aEntry2 second TImplStruct object
88 @return integer indicating the order of these two
89 @pre This object is fully constructed
91 TInt CRegistryData::TImplStruct::CompareImplStructUid(const TImplStruct& aEntry1,
92 const TImplStruct& aEntry2)
94 return CompareTUidValues(aEntry1.iCurrentImpl->iImplInfo->ImplementationUid().iUid,
95 aEntry2.iCurrentImpl->iImplInfo->ImplementationUid().iUid);
98 /** Comparer to determine the order of a TUid object (implementation UID) in relation to and a TImplStruct object.
99 @param aUid the Implementation UID to compare with.
100 @param aEntry the TImplStruct object to compare aUid against.
101 @return integer indicating the order of the two.
103 TInt CRegistryData::TImplStruct::CompareUidAgainstImplStruct(
105 const TImplStruct& aEntry)
107 return CompareTUidValues(aUid->iUid,aEntry.iCurrentImpl->iImplInfo->ImplementationUid().iUid);
111 // CRegistryData::CImplementationData class
114 Creates a new CImplemenatationData object. Note that CImplementationInformation needs to be constructed too,
115 for this object to be fully constructed.
116 @param aParent A pointer to the parent instance of CInterfaceData
117 @return A pointer to the newly created object.
118 @post This object is partly constructed and initialized and is on
121 CRegistryData::CImplementationData* CRegistryData::CImplementationData::NewLC(CInterfaceData* aParent)
123 CImplementationData* self=new(ELeave) CImplementationData(aParent);
124 CleanupStack::PushL(self);
129 Creates a new CImplemenatationData object. It takes parameters to create a CImplementationInformation object,
130 during 2nd phase construction and initializes iImplInfo to reference to this CImplementationInformation object.
131 @param aParent A pointer to the parent instance of CInterfaceData
132 @param aUid The unique Id of the new implementation
133 @param aVersion The version number of the new implementation
134 @param aName The display name of the new implementation
135 @param aDataType The data type supported by the new implementation
136 @param aOpaqueData Data for the new implementation which is not used by the ECom framework
137 @param aDrive The drive that the new implementation is on
138 @param aRomOnly The flag recording whether the new implementation may be loaded from ROM only
139 @param aRomBased The flag recording whether the new implementation is on ROM or is a later version of one on ROM
140 @return A pointer to the newly created object
141 @post This object is fully constructed and initialized.
143 CRegistryData::CImplementationData* CRegistryData::CImplementationData::NewL(CInterfaceData* aParent,
153 CImplementationData* self=new(ELeave) CImplementationData(aParent);
154 CleanupStack::PushL(self);
155 self->ConstructL(aUid,
163 CleanupStack::Pop(self);
168 Creates a new CImplemenatationData object. It takes parameters to create a CImplementationInformation object,
169 during 2nd phase construction and initializes iImplInfo to reference to this CImplementationInformation object.
170 @param aParent A pointer to the parent instance of CInterfaceData
171 @param aUid The unique Id of the new implementation
172 @param aVersion The version number of the new implementation
173 @param aName The display name of the new implementation
174 @param aDataType The data type supported by the new implementation
175 @param aOpaqueData Data for the new implementation which is not used by the ECom framework
176 @param aDrive The drive that the new implementation is on
177 @param aRomOnly The flag recording whether the new implementation may be loaded from ROM only
178 @param aRomBased The flag recording whether the new implementation is on ROM or is a later version of one on ROM
179 @param aExtendedInterfaces The pointer to the array recording the extended interfaces supported by this implementation.
180 NULL is available for PLUGIN without extended interfaces support.
181 @return A pointer to the newly created object
182 @post This object is fully constructed and initialized.
184 CRegistryData::CImplementationData* CRegistryData::CImplementationData::NewL(CInterfaceData* aParent,
193 RExtendedInterfacesArray* aExtendedInterfaces)
195 CImplementationData* self=new(ELeave) CImplementationData(aParent);
196 CleanupStack::PushL(self);
197 self->ConstructL(aUid,
205 aExtendedInterfaces);
206 CleanupStack::Pop(self);
211 Constructor for CImplementationData
212 @param aParent The parent interface data of this implementation
213 @post Its member variables are initialised
215 CRegistryData::CImplementationData::CImplementationData( CInterfaceData* aParent):
224 Destructor of CImplementationData
226 CRegistryData::CImplementationData::~CImplementationData()
237 The object's memory has been allocated.
238 @param aUid The unique Id of the new implementation
239 @param aVersion The version number of the new implementation
240 @param aName The display name of the new implementation
241 @param aDataType The data type supported by the new implementation
242 @param aOpaqueData Data for the new implementation which is not used by the ECom framework
243 @param aDrive The drive that the new implementation is on
244 @param aRomOnly The flag recording whether the new implementation may be loaded from ROM only
245 @param aRomBased The flag recording whether the new implementation is on ROM or is a later version of one on ROM
246 @pre This object is fully constructed.
247 @post This object is fully constructed and initialized.
249 void CRegistryData::CImplementationData::ConstructL(TUid aUid,
258 CImplementationInformation* newImpl = CImplementationInformation::NewL(aUid,
266 this->iImplInfo = newImpl;
270 The object's memory has been allocated.
271 @param aUid The unique Id of the new implementation
272 @param aVersion The version number of the new implementation
273 @param aName The display name of the new implementation
274 @param aDataType The data type supported by the new implementation
275 @param aOpaqueData Data for the new implementation which is not used by the ECom framework
276 @param aDrive The drive that the new implementation is on
277 @param aRomOnly The flag recording whether the new implementation may be loaded from ROM only
278 @param aRomBased The flag recording whether the new implementation is on ROM or is a later version of one on ROM
279 @param aExtendedInterfaces The pointer to the array recording the extended interfaces supported by this implementation.
280 NULL is available for PLUGIN without extended interfaces support.
281 @pre This object is fully constructed.
282 @post This object is fully constructed and initialized.
284 void CRegistryData::CImplementationData::ConstructL(TUid aUid,
292 RExtendedInterfacesArray* aExtendedInterfaces)
294 CImplementationInformation* newImpl = CImplementationInformation::NewL(aUid,
302 aExtendedInterfaces);
303 this->iImplInfo = newImpl;
307 Initialises member variable with the CImplementationInformation state specified in aStore.
308 @param aStore The stream to read the data from.
309 @pre This object is full constructed.
310 @post This object is set to the state specified in aStore.
312 void CRegistryData::CImplementationData::InternalizeL(RReadStream& aStore)
319 CImplementationInformation* implInfo = 0;
320 implInfo=CImplementationInformation::NewLC(EFalse,aStore);
321 //as we never store the drive name we need to get this from the parent
322 implInfo->SetDrive(iParent->iParent->iParent->iDrive);
323 CleanupStack::Pop(1);
324 iImplInfo = implInfo;
330 Writes out the state of this member variable of type CImplementationInformation to aStore.
331 @param aStore The stream to store the data in.
332 @pre This object is fully constructed.
334 void CRegistryData::CImplementationData::ExternalizeL(RWriteStream& aStore) const
336 iImplInfo->ExternalizeL(EFalse,aStore);
340 This method determines the order of two CImplementationData objects.
341 For backward compatibility reason, ECom allows different I/Fs using
342 the same implementation UID for when the device has an implementation UID
343 that is not unique. Although this is not a valid or supported situation ECOM
344 server should be robust.
345 @see FindImplementation
346 @see CompareTUidValues
347 @param aImpl1 first reference to CImplementationData object
348 @param aImpl2 second reference to CImplementationData object
349 @return integer indicating the order of these two
350 @pre This object is fully constructed
352 TInt CRegistryData::CImplementationData::CompareImplUid(const CImplementationData& aImpl1,
353 const CImplementationData& aImpl2)
355 TInt ret = CompareTUidValues(aImpl1.iImplInfo->ImplementationUid().iUid, aImpl2.iImplInfo->ImplementationUid().iUid);
361 // Implementation UIDs are equal, use I/F UID as tie breaker
362 return CompareTUidValues(aImpl1.iParent->iInterfaceUid.iUid, aImpl2.iParent->iInterfaceUid.iUid);
366 Similar to CompareImplUid above. This comparer only compare the Impl UID
367 and ignore the I/F UID part.
368 @param aImpl1 first reference to CImplementationData object
369 @param aImpl2 second reference to CImplementationData object
370 @return integer indicating the order of these two
371 @pre This object is fully constructed
373 TInt CRegistryData::CImplementationData::CompareImplUidIgnoreIfUid(const CImplementationData& aImpl1,
374 const CImplementationData& aImpl2)
376 return CompareTUidValues(aImpl1.iImplInfo->ImplementationUid().iUid,
377 aImpl2.iImplInfo->ImplementationUid().iUid);
381 This method determines the order of an CImplementationData object in relation
383 @param aUid is really TUid. Cast it back before compare.
384 @param aImplData reference to CImplementationData object
385 @return integer indicating the order of the two.
387 TInt CRegistryData::CImplementationData::CompareUidAgainstImplData(
388 const CImplementationData& aUid,
389 const CImplementationData& aImplData)
391 // The first argument aUid is really TUid.
392 const TUid* ImplUid = reinterpret_cast<const TUid*>(&aUid);
393 return CompareTUidValues(ImplUid->iUid,
394 aImplData.iImplInfo->ImplementationUid().iUid);
398 // CRegistryData::CInterfaceData class
401 Creates a new CInterfaceData object and leave it on the CleanupStack
402 @param aInterface The unique Id of this interface.
403 @param aParent A pointer to the parent dll data
404 @return A pointer to the newly created class.
405 @post This object is fully constructed and initialized and is on
408 CRegistryData::CInterfaceData* CRegistryData::CInterfaceData::NewLC(TUid aInterfaceUid,CDllData* aParent)
410 CInterfaceData* self=new(ELeave) CInterfaceData(aInterfaceUid,aParent);
411 CleanupStack::PushL(self);
418 Creates a new CInterfaceData object using the supplied interface id
419 and leave it on the CleanupStack
420 @param aParent A pointer to the parent dll data
421 @return A pointer to the newly created class.
422 @post This object is fully constructed and initialized and is on
425 CRegistryData::CInterfaceData* CRegistryData::CInterfaceData::NewLC(CDllData* aParent)
427 CInterfaceData* self=new(ELeave) CInterfaceData(aParent);
428 CleanupStack::PushL(self);
434 CRegistryData::CInterfaceData::~CInterfaceData()
438 iImplementations->ResetAndDestroy();
439 delete iImplementations;
440 iImplementations = NULL;
446 Adds the specified implementation to this interface in the registry.
447 @param aImplementation The implementation to add to this interface.
448 @pre This object is fully constructed.
449 @post aImplementation is added to the list of implementations for this interface.
451 void CRegistryData::CInterfaceData::AddL(const CImplementationData* aImplementation)
454 static int counter = 0; counter++;
455 __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()));
459 User::LeaveIfError(iImplementations->Append(aImplementation));
463 Sets the uid of this interface to aInterfaceUid.
464 @param aInterfaceUid The Uid which this object should take.
465 @pre This object is fully constructed.
466 @post The Uid of this interface is set to aInterfaceUid
468 void CRegistryData::CInterfaceData::SetInterfaceUid(TUid aInterfaceUid)
470 iInterfaceUid = aInterfaceUid;
474 Writes out this CInterfaceData to aStore.
475 @param aStore The stream to store the data in.
476 @pre The state of this object is stored in the stream aStore.
478 void CRegistryData::CInterfaceData::ExternalizeL(RWriteStream& aStore) const
480 aStore.WriteInt32L(iInterfaceUid.iUid);
484 const TInt entryCount = iImplementations->Count();
485 aStore.WriteInt32L(entryCount);
486 for(TInt i = 0; i < entryCount; ++i)
487 ((*iImplementations)[i])->ExternalizeL(aStore);
490 aStore.WriteInt32L(0);
494 Restores this CInterfaceData to the state specified in aStore.
495 @param aStore The stream to read the data from.
496 @param aPresent A boolean indicating whether the dll is still present
497 @pre This object is full constructed.
498 @post This object is set to the state specified in aStore.
500 void CRegistryData::CInterfaceData::InternalizeL(RReadStream& aStore)
502 iInterfaceUid.iUid = aStore.ReadInt32L();
504 const TInt entryCount = aStore.ReadInt32L();
507 User::Leave(KErrCorrupt);
510 for(TInt i = 0; i < entryCount; ++i)
512 CImplementationData* implementation = CImplementationData::NewLC(this);
513 implementation->InternalizeL(aStore);
514 AddL(implementation);
515 CleanupStack::Pop(implementation); //now owned by this interface
520 @param aParent A pointer to the parent dll data
522 CRegistryData::CInterfaceData::CInterfaceData(CDllData* aParent) :
530 @param aInterfaceUid The unique Id of this interface
531 @param aParent A pointer to the parent dll data
533 CRegistryData::CInterfaceData::CInterfaceData(TUid aInterfaceUid, CDllData* aParent) :
535 iInterfaceUid(aInterfaceUid),
542 Standard second phase construction function
543 @pre This object is fully constructed.
544 @post This object is fully constructed and initialized.
546 void CRegistryData::CInterfaceData::ConstructL()
548 iImplementations = new(ELeave) RPointerArray<CImplementationData>;
552 // CRegistryData::CDllData class
555 Creates a new CDllData object using aEntry and leaves it on the CleanupStack
556 @param aDllName the name of this dll
557 @param aDllModTime the modified time of this dll
558 @param aSecondUid Identifies type of the DLL. (PLUGIN or PLUGIN3)
559 @param aThirdUid Identifies a component uniquely.
560 @param aParent A pointer to the parent drive data
561 @return A pointer to the newly created object.
562 @post This object is fully constructed and initialized and on the CleanupStack.
564 CRegistryData::CDllData* CRegistryData::CDllData::NewLC(const TDesC& aDllName,const TTime& aDllModTime,const TUid& aSecondUid,const TUid& aThirdUid,CDriveData* aParent)
566 CDllData* self=new(ELeave) CDllData(aParent);
567 CleanupStack::PushL(self);
568 self->ConstructL(aDllName,aDllModTime,aSecondUid,aThirdUid);
573 Creates a new CDllData object using aParent and leaves it on the CleanupStack
574 @param aParent A pointer to the parent drive data
575 @return A pointer to the newly created object.
576 @post This object is fully constructed and initialized and on the CleanupStack.
578 CRegistryData::CDllData* CRegistryData::CDllData::NewLC( CDriveData* aParent)
580 CDllData* self=new(ELeave) CDllData(aParent);
581 CleanupStack::PushL(self);
586 CRegistryData::CDllData::~CDllData()
590 // Clear the interface list and destroy its objects
591 iIfList->ResetAndDestroy();
596 // Unload this implementation dll.
600 delete iRscFileExtension;
604 Adds the specified interface to this dll in the registry.
605 @param aInterface The interface to add to this dll
606 @pre This object is fully constructed.
607 @post aInterface is added to the list of interfaces in this dll.
609 void CRegistryData::CDllData::AddL(const CInterfaceData* aInterface)
611 User::LeaveIfError(iIfList->Append(aInterface));
615 Sets the resource extension for the plugin. Not set for read only internal drives.
616 @param aExt The resource extension to set
617 @pre This object is fully constructed.
618 @post aExt is added to the object.
620 void CRegistryData::CDllData::SetResourceExtL(const TDesC& aExt)
622 delete iRscFileExtension;
623 iRscFileExtension = NULL;
624 iRscFileExtension = aExt.AllocL();
628 Set the capability,the VID and do the Security check for this DLL.
629 @pre This object is fully constructed.
630 @return ETrue if the security check is done successfully. Otherwise EFalse is returned.
632 TBool CRegistryData::CDllData::SaveSecurityInfoL()
634 iSecurityChecked = EFalse;
636 //need to construct the full filename i.e. appending with the
637 //preconstructed drivepath name in CDriveData
638 TFileName dllFullName;
639 dllFullName.Append(iParent->iDrive.Name());
640 dllFullName.Append(_L("\\sys\\bin\\"));
641 dllFullName.Append(iDllEntry->GetName());
642 RLibrary::TInfoBuf infoBuf;
643 TInt ret = RLibrary::GetInfo(dllFullName, infoBuf);
648 // Set the DLL's capability
649 iCapSet = infoBuf().iSecurityInfo.iCaps;
651 iVid = infoBuf().iSecurityInfo.iVendorId;
652 // Now verify that SID identified in the resource file matches the SID of the Dll file
653 TBool match=iDllEntry->GetThirdUid() ==infoBuf().iSecurityInfo.iSecureId;
657 __ECOM_TRACE1("ERROR: Plugin SID Mismatch ERROR for %S.", &dllFullName);
662 // Now verify the two DLL's second Uid are of the same Uid type (used for distinguising between collection/collection3)
663 match=iDllEntry->GetSecondUid() ==infoBuf().iUids[1];
667 __ECOM_TRACE1("ERROR: Plugin UID2 Mismatch ERROR for %S.", &iDllEntry->GetName());
672 // On the emulator RLoader::CheckLibraryHash() returns KErrNotSupported.
673 // Also on the emulator RLoader does no hash checking for DLL's on removable drives.
674 // Therefore to be consistent ECOM does not do any hash checking itself on the emulator.
675 // Hence code is removed for emulator builds.
676 #if !defined(__WINSCW__)
677 // Verify hash is available
678 if(iParent->iParent->iCachedDriveInfo->DriveIsRemovableL(iParent->iDrive))
681 TInt err = loader.Connect();
686 err = loader.CheckLibraryHash(dllFullName, ETrue);
691 __ECOM_TRACE2("INFO: Hash Check Failed for %S with error %d.", &dllFullName, err);
697 iSecurityChecked = ETrue;
702 Check whether security check has been performed if not go retrieve it.
703 @pre This object is fully constructed.
704 @return ETrue if the security check is done successfully. Otherwise EFalse is returned.
706 TBool CRegistryData::CDllData::ProcessSecurityCheckL()
712 return SaveSecurityInfoL();
717 Writes out the state of this CDllData to aStore.
718 @param aStore The stream to store the data in.
719 @pre This object is fully constructed.
721 void CRegistryData::CDllData::ExternalizeL(RWriteStream& aStore) const
723 const CEComEntry& dllEntryData = *iDllEntry;
724 TInt size=dllEntryData.GetName().Length()-KDllExtensionLength;
725 aStore.WriteUint32L(size);
726 aStore.WriteL(dllEntryData.GetName(),size);
728 aStore.WriteInt32L(dllEntryData.GetSecondUid().iUid);
729 aStore.WriteInt32L(dllEntryData.GetThirdUid().iUid);
730 TPckg<TTime> modified(dllEntryData.GetModified());
731 aStore.WriteL(modified);
733 if(!iParent->iParent->iCachedDriveInfo->DriveIsReadOnlyInternalL(iParent->iDrive))
735 if(iRscFileExtension)
737 size = iRscFileExtension->Length();
738 aStore.WriteUint32L(size);
739 aStore.WriteL(iRscFileExtension->Des(),size);
743 aStore.WriteUint32L(0);
749 const TInt entryCount = iIfList->Count();
750 aStore.WriteInt32L(entryCount);
751 for(TInt i = 0; i < entryCount; ++i)
752 ((*iIfList)[i])->ExternalizeL(aStore);
755 aStore.WriteInt32L(0);
759 Restores this CDllData to the state specified in aStore.
760 @param aStore The stream to read the data from.
761 @pre This object is fully constructed.
762 @post The state of this object is restored to that specified in aStore.
764 void CRegistryData::CDllData::InternalizeL(RReadStream& aStore)
766 TInt size=aStore.ReadUint32L();
767 //The next "if" checks if size < 0 not size <= 0 because it seems the ECOM server externalizes
768 //the last file name as a string with length 0. If the fix is <= 0, then it makes it
769 //incompatible with the existing applications
770 if(size < 0 || size > KMaxFileName)
772 User::Leave(KErrCorrupt);
774 HBufC* name = HBufC::NewLC(size+KDllExtensionLength);
775 TPtr ptr=name->Des();
776 aStore.ReadL(ptr,size);
777 ptr.Append(KDllExtension);
779 TUid secondUid=TUid::Uid(aStore.ReadInt32L());
780 TUid thirdUid=TUid::Uid(aStore.ReadInt32L());
781 TTime dllModifiedTime;
782 TPckg<TTime> modified(dllModifiedTime);
783 aStore.ReadL(modified);
785 if(!iParent->iParent->iCachedDriveInfo->DriveIsReadOnlyInternalL(iParent->iDrive))
787 size = aStore.ReadUint32L();
788 if(size < 0 || size > KMaxFileName)
790 User::Leave(KErrCorrupt);
794 iRscFileExtension = HBufC::NewL(size);
795 TPtr extPtr = iRscFileExtension->Des();
796 aStore.ReadL(extPtr,size);
800 const TInt entryCount = aStore.ReadInt32L();
803 User::Leave(KErrCorrupt);
806 // Security check is deferred until the DLL is needed.
807 for(TInt i = 0; i < entryCount; ++i)
809 CInterfaceData* interface = CInterfaceData::NewLC(this);
810 interface->InternalizeL(aStore);
812 CleanupStack::Pop(interface); // now owned by dll
814 iDllEntry = CEComEntry::NewL(*name,secondUid,thirdUid);
815 iDllEntry->SetModified(dllModifiedTime);
816 CleanupStack::PopAndDestroy(name);
820 @param aParent The parent drive data of this implementation
822 CRegistryData::CDllData::CDllData( CDriveData* aParent) :
827 //Initialize empty capabilities here
832 Standard second phase construction function.
833 @pre This object is fully constructed.
834 @post This object is fully constructed and initialized.
836 void CRegistryData::CDllData::ConstructL()
838 iIfList = new(ELeave) RInterfaceList(2);
842 Standard second phase construction function with parameter.
843 @param aDllName the name of this dll
844 @param aDllModTime the modified time of this dll
845 @param aSecondUid Distinguishes between components having the same UID1 (which distinguishes between EXEs and DLLs)
846 UID2 identifies Interface Implementation Collections (collection and collection 3)
847 @param aThirdUid Identifies a component uniquely. In order to ensure that each binary that needs a distinguishing
848 UID is assigned a genuinely unique value. Symbian manages UID allocation through central database.
849 @pre This object is fully constructed.
850 @post This object is fully constructed and initialized.
852 void CRegistryData::CDllData::ConstructL(const TDesC& aDllName,const TTime& aDllModTime,const TUid& aSecondUid,const TUid& aThirdUid)
854 iIfList = new(ELeave) RInterfaceList(2);
855 iDllEntry=CEComEntry::NewL(aDllName,aSecondUid,aThirdUid);
856 iDllEntry->SetModified(aDllModTime);
861 Populate a caller-supplied TEntry instance with the data from this dll.
862 @param aEntry An entry to be populated (destination)
864 void CRegistryData::CDllData::PopulateAnEntry(TEntry& aEntry) const
866 TPtr bufPtr=aEntry.iName.Des();
868 bufPtr.Append(iParent->iDrive.Name());
869 bufPtr.Append(_L("\\sys\\bin\\"));
870 bufPtr.Append(iDllEntry->GetName());
871 aEntry.iType = TUidType(KDynamicLibraryUid,iDllEntry->GetSecondUid(),iDllEntry->GetThirdUid());
872 aEntry.iModified= iDllEntry->GetModified();
877 // CRegistryData::CDriveData class
881 Creates a new CDriveData object and places it on the CleanupStack
882 @param aDrive Information on this drive
883 @param aParent A pointer to the parent registry data
884 @return A pointer to the newly created class.
885 @post This object is fully constructed and initialized and on the CleanupStack.
887 CRegistryData::CDriveData* CRegistryData::CDriveData::NewLC(TDriveUnit aDrive, CRegistryData* aParent)
889 CDriveData* self=new(ELeave) CDriveData(aDrive,aParent); // calls c'tor
890 CleanupStack::PushL(self); // Make the construction safe by using the cleanup stack
891 self->ConstructL(); // Complete the 'construction'.
896 Destructor of CDriveData
898 CRegistryData::CDriveData::~CDriveData()
902 // Clear the interface list and destroy its objects
903 iDllList->ResetAndDestroy();
911 Adds the specified CDllData to this drive data in the registry.
912 @param aDll The Dll to add to this drive
913 @pre This object is fully constructed.
914 @post aDll is added to the list of dll on this drive.
916 void CRegistryData::CDriveData::AddL(const CDllData* aDll)
918 User::LeaveIfError(iDllList->Append(aDll));
922 Returns the index of the DLL with this Uid in the Dll list.
923 @param aDllUid the UID of to be found DLL.
924 @return The index of this DLL in the DLL list. KErrNotFound if not found.
925 @pre This object is fully constructed.
927 TInt CRegistryData::CDriveData::FindDllIndex(const TUid aDllUid) const
929 const TInt dllCount = iDllList->Count();
930 for(TInt j = 0; j < dllCount; ++j)
932 // For each dll structure check the dll UID
933 CDllData* dll = (*iDllList)[j];
934 if (aDllUid == dll->iDllEntry->GetThirdUid())
944 Writes out the state of this CDriveData to file.
945 @param aFs A handle to an open file server session.
946 @param aDatFileName the dat file to persist to
947 @pre This object is fully constructed.
949 void CRegistryData::CDriveData::ExternalizeL(RFs& aFs,const TDesC& aDatFileName)
951 // attempt to create the folders if they don't already exist
952 TInt mkDirError = aFs.MkDirAll(aDatFileName);
954 if((mkDirError == KErrNone) || (mkDirError == KErrAlreadyExists))
956 RFileWriteStream registryStream;
957 if(registryStream.Replace(aFs,aDatFileName, EFileWrite) == KErrNone)
959 CleanupClosePushL(registryStream);
960 // Write the version number as the first thing in the file stream.
961 registryStream.WriteInt32L(KRegVersion);
964 TInt entryCount = iDllList->Count();
965 registryStream.WriteInt32L(entryCount);
966 // Now stream out the data
967 for(TInt i = 0; i < entryCount; ++i)
968 ((*iDllList)[i])->ExternalizeL(registryStream);
971 registryStream.WriteInt32L(0);
972 CleanupStack::PopAndDestroy(); // registryStream
977 User::Leave(mkDirError);
982 Restores the state of this CDriveData from file.
983 @param aFs A handle to an open file server session.
984 @param aDatFileName the dat file name to internalize from
985 @pre This object is fully constructed.
986 @post The state of this object is restored to that specified
987 in the file on this drive.
990 void CRegistryData::CDriveData::InternalizeL(RFs& aFs,const TDesC& aDatFileName)
992 DoInternalizeL(aFs, aDatFileName);
995 The method internalizes the data from aRegistryStream.
996 @param aFileName The ECOM registry file name.
998 @leave Some other system-wide error codes as well.
1000 void CRegistryData::CDriveData::DoInternalizeL(RFs& aFs, const TDesC& aFileName)
1002 RFileReadStream registryStream;
1003 User::LeaveIfError(registryStream.Open(aFs, aFileName, EFileRead));
1004 CleanupClosePushL(registryStream);
1005 // Read in version number. If its not what expected - the file is treated as corrupted.
1006 const TInt version = registryStream.ReadInt32L();
1007 if(version != KRegVersion)
1009 User::Leave(KErrCorrupt);
1011 const TInt entryCount = registryStream.ReadInt32L();
1014 User::Leave(KErrCorrupt);
1016 for(TInt i = 0; i < entryCount; ++i)
1018 CDllData* dll = CDllData::NewLC(this);
1019 dll->InternalizeL(registryStream);
1020 // always add the DLL, DLL security check will be deferred
1021 // until the DLL is needed.
1023 CleanupStack::Pop(dll); // now owned by drive
1025 CleanupStack::PopAndDestroy();//registryStream
1029 Constructor for CDriveData
1030 @param aDrive Information on this drive
1031 @param aParent A pointer to the parent registry data
1033 CRegistryData::CDriveData::CDriveData(TDriveUnit aDrive, CRegistryData* aParent) :
1038 iDriveChanged(ETrue),
1039 iRegistryChanged(EFalse)
1044 Standard second phase construction function.
1045 @pre This object is fully constructed.
1046 @post This object is fully constructed and initialized.
1048 void CRegistryData::CDriveData::ConstructL()
1050 iDllList = new(ELeave)TDll;
1054 // CRegistryData class
1057 Standardized safe construction which leaves nothing on the cleanup stack.
1058 This overload is used by CEComImplIndexPerfTest which plots
1059 discovery time vs. granularity settings.
1060 @leave KErrNoMemory.
1061 @param aFs The open file session.
1062 @param aInterfaceImplIndexGranularity granularity of main index.
1063 @param aImplIndexGranularity granularity of auxiliary index.
1064 @return The newly created instance of the registry data.
1065 @post CRegistryData is fully constructed, and initialized.
1067 CRegistryData* CRegistryData::NewL(RFs& aFs, TInt aInterfaceImplIndexGranularity, TInt aImplIndexGranularity)
1069 CRegistryData* self=new(ELeave) CRegistryData(aFs, aInterfaceImplIndexGranularity, aImplIndexGranularity); // calls c'tor
1070 CleanupStack::PushL(self); // Make the construction safe by using the cleanup stack
1071 self->ConstructL(); // Complete the 'construction'.
1072 CleanupStack::Pop(self);
1077 Standardized safe construction which leaves nothing on the cleanup stack.
1078 This overload uses default granularities for the two indexes. These
1079 default values were found to provide reasonably good performance at
1080 the time of testing.
1081 @leave KErrNoMemory.
1082 @param aFs The open file session.
1083 @return The newly created instance of the registry data.
1084 @post CRegistryData is fully constructed, and initialized.
1086 CRegistryData* CRegistryData::NewL(RFs& aFs)
1088 return NewL(aFs, KDefaultInterfaceImplIndexGranularity, KDefaultImplIndexGranularity);
1091 CRegistryData::~CRegistryData()
1093 for(TInt index = 0; index < iInterfaceImplIndex.Count(); ++index)
1095 iInterfaceImplIndex[index].Reset();
1097 iInterfaceImplIndex.Reset();
1099 // Reset only. It does not own the pointers!
1104 iRegistrations->ResetAndDestroy();
1105 delete iRegistrations;
1106 iRegistrations = NULL;
1108 // reset the cached language settings
1109 RDowngradePath::Reset();
1111 delete iCachedDriveInfo;
1115 Lists the implementations given the interface UID. The filtered list is returned in the client provided
1116 RImplDataArray parameter.
1117 @param aInterfaceUid The Uid of the interface which the implementations should provide
1118 @param aImplementationData Return value. The filtered list.
1120 void CRegistryData::ListImplementationsL(TUid aInterfaceUid,
1121 RImplDataArray& aImplementationData) const
1123 // If discoveries are taking place then the index is invalid so we cannot do
1125 if(iCurrentlyDiscovering)
1126 User::Leave(KEComErrListCurrentlyUnavailable);
1128 TBool securityPassed = EFalse;
1129 //Do the security check
1131 while (!securityPassed)
1133 aImplementationData.Reset();
1134 index = IndexedFind(aInterfaceUid);
1135 if(index == KErrNotFound)
1137 User::Leave(KEComErrNoInterfaceIdentified);
1139 User::LeaveIfError(index);
1141 TImplContainerArray& implementationList = iInterfaceImplIndex[index].iImpData;
1142 for (TInt j = 0; j < implementationList.Count(); j++)
1144 CImplementationData* currentImplementation = implementationList[j].iCurrentImpl;
1145 CDllData* dll = currentImplementation->iParent->iParent;
1147 securityPassed = dll->ProcessSecurityCheckL();
1150 // remove the implementations of the DLL from iInterfaceImplIndex
1151 // and remove the DLL from its parent DLL list.
1155 // don't continue processing implementations after we have deleted the DLL
1156 // because the implementations list will have changed, so we need to
1157 // re-do the IndexedFind
1162 User::LeaveIfError(aImplementationData.Append(currentImplementation));
1169 Enables or disables the specified interface implementation within the registry.
1170 @param aImplementationUid The interface implementation to change.
1171 @param aState ETrue to enable the implementation, EFalse to disable it
1172 @return KErrNone, KEComErrEnableFailed or KEComErrDisableFailed
1173 @pre CRegistry is fully constructed,
1174 @post The implementation is now enabled or disabled as described
1177 TInt CRegistryData::SetEnabledState(TUid aImplementationUid, TBool aState)
1179 CImplementationData* impData = NULL;
1180 TUid dummy={0x00000000};
1182 (void)FindImplementation(aImplementationUid, dummy, impData);
1186 impData->iImplInfo->SetDisabled((aState) ? EFalse : ETrue);
1190 return (aState) ? KEComErrEnableFailed : KEComErrDisableFailed;
1195 Temporarily uninstalls the interface implementation groups upon the specified drive.
1196 (ie this can be undone later).
1197 @param aDrive The identifier of the drive to uninstall.
1198 @pre CRegistry is fully constructed
1199 @post The registered implementation groupings
1200 stored upon the specified drive are no
1201 longer available for use.
1203 void CRegistryData::TemporaryUninstallL(const TDriveUnit& aDrive)
1205 // Find the appropriate drive entry and remove it...
1206 // Note : the drive may have already been taken offline, so no save is possible!
1207 CDriveData* drive = NULL;
1208 TInt driveIndex = FindDriveL(aDrive, drive);
1209 if(driveIndex != KErrNotFound)
1211 TInt dllIndex = drive->iDllList->Count();
1215 RemoveFromIndexL((*drive->iDllList)[dllIndex]);
1217 iRegistrations->Remove(driveIndex);
1220 DriveChanged(aDrive, ETrue);
1222 // The flag iRegistryChanged has been moved to each drive.
1226 Undoes a temporary uninstall of the interface
1227 implementation groups upon the specified drive.
1228 (ie this can be undone later).
1229 @leave KErrNoMemory, KErrReinstallFailed.
1230 @param aDrive The identifier of the drive to reinstate.
1231 @pre CRegistry is fully constructed
1232 @post The registered implementation groupings
1233 stored upon the specified drive are again
1234 made available for use.
1236 void CRegistryData::UndoTemporaryUninstallL(const TDriveUnit& aDrive)
1238 CDriveData* driveData = NULL;
1239 // If the drive was not found then install it, otherwise do nothing because it
1240 // is already there. NB. We could leave here but the problem with that is the
1241 // use case where the drive has been mounted during registry load but the registrar
1242 // is started afterwards, and makes calls to this method as it discovers drives
1243 // for the first time.
1244 if(FindDriveL(aDrive, driveData) == KErrNotFound)
1246 // Add the drive entry
1247 CDriveData* drive = CDriveData::NewLC(aDrive,this);
1248 AddImplDataL(drive);
1249 User::LeaveIfError(iRegistrations->Append(drive));
1250 CleanupStack::Pop(drive); // now owned by iRegistrations
1252 DriveChanged(aDrive, EFalse);
1254 // NB We DO need to set iRegistryChanged = ETrue because the idx file needs to be
1255 // rewritten to include the reinstalled drive
1260 Update the staus of drives in the system
1261 @param aDrive The drive that has changed
1262 @param aDriveRemoved Indicates whether a drive has been removed or added
1263 @pre CRegistry is fully constructed.
1264 @post The state of the drive has been stored
1266 void CRegistryData::DriveChanged(const TDriveUnit& aDrive, TBool aDriveRemoved)
1270 iRemovedDrives |= (0x1 << aDrive);
1274 iRemovedDrives &= ~(0x1 << aDrive);
1279 Find if any Dll is regsitered in the drive unit.
1280 @param aDrive The identifier of the drive to find if any Dll is registered.
1281 @return ETrue if any Dll is registered in the drive, otherwise EFalse.
1282 @pre CRegistry is fully constructed.
1283 @post If any Dll is registered in the drive is returned if successfully.
1285 TBool CRegistryData::IsAnyDllRegisteredWithDriveL(const TDriveUnit& aDrive)const
1287 CDriveData* driveData = NULL;
1288 if(FindDriveL(aDrive, driveData) == KErrNotFound)
1290 User::Leave(KEComErrDriveNotFound);
1292 return driveData->iDllList->Count() > 0;
1296 Determine if the specified implementation grouping already registered, but needs an update.
1297 @param aDllUid The interface implementation collection data to add.
1298 CRegistryData takes ownership of this object
1299 so it should not be on the CleanupStack.
1300 @param aModified The date of the last modification of the Interface
1301 Implementation Collection.
1302 @param aUpdate An output to indicate if the registry entry requires an update.
1303 (Only valid if the return is ETrue).
1304 @param aDriveData The drive data used to find DLL.
1305 @return ETrue if the Interface Implementation Collection already has a registry entry.
1307 @pre This object is fully constructed and there is also a valid drive entry
1308 in the registry for aDrive
1309 @post The new data is added to the registry
1311 TBool CRegistryData::IsRegisteredWithDate (TUid aDllUid,
1312 const TTime& aModified,
1314 CDriveData* aDriveData)
1316 // Find the appropriate drive entry for this
1317 // Interface Implementation Collection
1318 TBool found = EFalse;
1319 TInt index = aDriveData->FindDllIndex(aDllUid);
1320 if(index != KErrNotFound)
1322 TDll* dllList = aDriveData->iDllList;
1323 CDllData* dll = (*dllList)[index];
1324 const CEComEntry& dllEntry =*(dll->iDllEntry);
1325 aUpdate = dllEntry.GetModified().Int64() < aModified.Int64();
1332 Adds data on a specific dll to the registry. The data to be added
1333 is parsed by CRegistryParser.
1334 @leave KErrNoMemory If the item could not be appended to the registry
1335 @leave KEComErrDriveNotFound If aDrive is not a valid drive
1336 @param aDrive The drive the registry data has been found on
1337 @param aFoundDriveIndex The index of the registry data for the drive
1338 the interface implementation collection data has been found on.
1339 @param aDllData The dll data to add. CRegistryData takes ownership of this object
1340 so it should not be on the CleanupStack.
1341 @pre This object is fully constructed and there is also a valid drive entry
1342 in the registry for aDrive
1343 @post The new data is added to the registry
1345 void CRegistryData::AddDllDataL(const TDriveUnit& aDrive, TInt aFoundDriveIndex, CRegistryData::CDllData* aDllData)
1347 const TInt regCount = iRegistrations->Count();
1349 User::Leave(KEComErrDriveNotFound);
1351 // Find the appropriate drive entry
1352 CDriveData* drive = NULL;
1353 if(aFoundDriveIndex == KErrNotFound)
1355 if(FindDriveL(aDrive, drive) == KErrNotFound)
1356 User::Leave(KEComErrDriveNotFound);
1359 drive = (*iRegistrations)[aFoundDriveIndex];
1361 // Append the new data to the list
1362 TDll* dllList = drive->iDllList;
1364 // NOTE : This function MUST NOT leave after the following line because
1365 // aDllData will potentially be 'owned' in two places (dllList and the caller)
1366 // until we return and the caller can Pop.
1367 TBool checkNeeded = ETrue;
1368 TInt ifListCount=aDllData->iIfList->Count();
1369 for(TInt ifListCounter=0; ifListCounter < ifListCount; ifListCounter++)
1371 // For each interface structure
1372 CInterfaceData* interface = (*aDllData->iIfList)[ifListCounter];
1373 for(TInt impNum = 0; impNum < interface->iImplementations->Count(); ++impNum)
1375 CImplementationData* implData = (*interface->iImplementations)[impNum];
1376 if(!InsertIntoIndexL(implData,checkNeeded))
1378 interface->iImplementations->Remove(impNum); //remove from implementation array
1379 delete implData; //delete object
1383 //add filtered list for legitimate implementations into registry
1384 User::LeaveIfError(dllList->Append(aDllData));
1386 drive->iRegistryChanged = ETrue;
1390 Updates the data for a dll in the registry.
1391 @leave KErrNoMemory If the indexes cannot be rebuilt
1392 @param aFoundDriveIndex The index of the registry data for the drive
1393 the interface implementation collection data has been found on.
1394 @param aDllData The updated interface implementation collection data.
1395 CRegistryData takes ownership of this object
1396 so it should not be on the CleanupStack.
1397 @pre This object is fully constructed
1398 @post The dll data is updated in the registry
1400 void CRegistryData::UpdateDllDataL(const TDriveUnit& aDrive,TInt aFoundDriveIndex, CDllData* aDllData)
1402 const TInt regCount = iRegistrations->Count();
1404 User::Leave(KEComErrDriveNotFound);
1406 // Find the appropriate drive entry
1407 CDriveData* drive = (*iRegistrations)[aFoundDriveIndex];
1409 TInt index = drive->FindDllIndex(aDllData->iDllEntry->GetThirdUid());
1411 // find the dll entry and clear that too
1412 if(index != KErrNotFound)
1414 // Ok its a match, so remove this registry entry and call AddDllData()
1415 //to add the new dll
1416 TDll* dllList = drive->iDllList;
1417 CDllData* olddll = (*dllList)[index];
1418 //remove 'old'implementations from iInterfaceImplIndex
1419 RemoveFromIndexL(olddll);
1420 //now remove old dll from the array and registry
1421 dllList->Remove(index);
1423 AddDllDataL(aDrive,aFoundDriveIndex,aDllData);
1424 drive->iRegistryChanged = ETrue;
1427 if (index == KErrNotFound)
1429 __ECOM_TRACE("ECOM: PANIC in CRegistryData::UpdateDllDataL(), expected DLL data missing from ECOM registry");
1430 __ASSERT_DEBUG(EFalse, User::Panic (KEComServerPanicCategory, EEComPanic_CRegistryData_UpdateDllDataL_DllRegistrationMissing));
1435 The cue that newly discovered Dlls are about to be registered,
1436 and, therefore, the registry index will be out of date until
1437 DiscoveriesCompleteL is called.
1438 @pre CRegistrar is fully constructed
1439 @post iCurrentlyDiscovering is set and the index list is cleared.
1441 void CRegistryData::DiscoveriesBeginning() const
1443 iCurrentlyDiscovering = ETrue;
1444 // The registry is about to start changing so the index is now invalid
1448 The cue to rebuild the registry indexes
1451 @param aSuccessful Indicates whether discovery completed fully or not
1452 @param aProcessingType indicates the type of processing for plugins
1453 for ensuring that plugins are not processed multiple times
1454 during start-up phase
1455 @param aHasRegistryChanged The flag to indicate whether registry data has been changed
1456 @pre CRegistrar is fully constructed
1457 @post The internal access indexes have been rebuilt
1459 void CRegistryData::DiscoveriesCompleteL(TBool aSuccessful, TPluginProcessingTypeIdentifier aProcessingType, TBool& aHasRegistryChanged)
1461 iCurrentlyDiscovering = EFalse;
1466 //if the rediscovery is a result of iLanguageChanged, we should reset it to false
1467 iLanguageChanged=EFalse;
1468 // Warning: this method may be called from the CEComServer destructor.
1469 // E.g. ecomservertest running OOM test on CEComServer::NewLC.
1471 // ~CEComServer delete iRegistrar, ~CRegistrar delete iDiscoverer,
1472 // ~CDiscoverer calls CRegistrar::DiscoveriesComplete,
1474 // Hence if ! aSuccessful do not validate.
1476 if(aProcessingType != EPluginProcessingTypeCriticalOnly)
1478 ValidateRegistryL();
1480 if(aProcessingType==EPluginProcessingTypeNonCriticalOnly || aProcessingType==EPluginProcessingTypeAll)
1482 //first find whether there is any registry data changed
1483 TInt regCount=iRegistrations->Count();
1484 for(TInt i = 0; i <regCount; i++)
1486 if ((*iRegistrations)[i]->iRegistryChanged)
1488 aHasRegistryChanged = ETrue;
1489 //now having cached that the registry has changed we need
1490 //to reset this flag in ALL the CDriveData, note that it is
1491 //possible that more than one drive will have the flag set
1492 (*iRegistrations)[i]->iRegistryChanged=EFalse;
1495 //Also indicate registry change if any drives have been removed
1496 aHasRegistryChanged |= (iRemovedDrives != 0);
1502 @fn SetDiscoveryFlag(const TDriveUnit aDrive)
1503 Intended Usage : Set the flag to indicate the drive has change(s)
1504 Error Condition : None.
1505 @param aDrive The identifier of the drive changed.
1506 @pre The CRegistrar must be fully constructed
1507 @post The flag is set.
1510 void CRegistryData::SetDiscoveryFlagL(const TDriveUnit& aDriveUnit)
1512 CDriveData* drive = NULL;
1513 TInt driveIndex = FindDriveL(aDriveUnit, drive);
1514 if(driveIndex != KErrNotFound)
1516 drive->iDriveChanged = ETrue;
1521 Indicates whether the registry index is currently valid. The
1522 index will not be valid if discoveries are currently taking place.
1523 If the index is not currently valid then calls to
1524 ListImplementationsL() cannot be serviced.
1525 @return ETrue if the index is currently valid, EFalse otherwise.
1526 @pre CRegistrar is fully constructed
1528 TBool CRegistryData::IndexValid() const
1530 return !iCurrentlyDiscovering;
1534 @param aCapabilitySet A capability set
1535 @param aImplUid The Uid of the implementation for which info is required
1536 @param aInterfaceUid The uid of the interface associated with aImplUid to find or less than
1537 0 if uid is unknown.
1538 @param aEntry Output parameter which will contain the dll information
1539 @param aImplInfo An output parameter. If the call succeeds it will point to the found implementation information,
1541 @param aIsOnRWDrive an output parameter. If the call is successful, this will
1542 be set to ETrue if the implementation is on RW drive. EFalse if the
1543 implementation is on ReadOnly drive.
1544 @return KErrNone if the call succeeds, KErrNotFound - no implementation found,
1545 KErrPermissionDenied - the caller has not enough capabilities to load the plugin.
1546 @pre CRegistrar is fully constructed
1548 TInt CRegistryData::GetImplementationDllInfoForServer(
1549 const TCapabilitySet& /*aCapabilitySet*/,
1550 const TUid aImplUid,
1551 const TUid aInterfaceUid,
1553 CImplementationInformation*& aImplInfo,
1554 TBool& aIsOnRWDrive) const
1557 CImplementationData* implData;
1559 TInt res = FindImplementation(aImplUid, aInterfaceUid, implData);
1560 if (KErrNone != res)
1565 aImplInfo = implData->iImplInfo;
1566 const CDllData* dll = implData->iParent->iParent;
1567 dll->PopulateAnEntry(aEntry);
1569 TEComCachedDriveInfoIterator iter(*iCachedDriveInfo);
1570 if (! iter.SetPos(dll->iParent->iDrive))
1576 aIsOnRWDrive = iter.DriveIsWritable();
1582 @param aClientRequest A client request
1583 @param aImplUid The Uid of the implementation for which info is required
1584 @param aInterfaceUid The uid of the interface associated with aImplUid to find or less than
1585 0 if uid is unknown.
1586 @param aEntry Output parameter which will contain the dll information
1587 @param aImplInfo An output parameter. If the call succeeds it will point to the found implementation information,
1589 @param aSecurityCheckNeeded The bool value to identify whether the security check is needed here. The default value is false.
1590 @return KErrNone if the call succeeds, KErrNotFound - no implementation found,
1591 KErrPermissionDenied - the caller has not enough capabilities to load the plugin.
1592 @pre CRegistrar is fully constructed
1594 TInt CRegistryData::GetImplementationDllInfoForClientL(
1595 const TClientRequest& aClientRequest,
1596 const TUid aImplUid,
1597 const TUid aInterfaceUid,
1599 CImplementationInformation*& aImplInfo,
1600 TBool aSecurityCheckNeeded)const
1603 TInt res = KErrNotFound;
1604 CImplementationData* implData;
1605 if (!aSecurityCheckNeeded)
1607 //The security check has been done already. All the invalid dll have been removed.
1608 res = FindImplementation(aImplUid, aInterfaceUid, implData);
1609 if (KErrNone != res)
1613 CDllData* dll = implData->iParent->iParent;
1614 dll->PopulateAnEntry(aEntry);
1615 aImplInfo = implData->iImplInfo;
1619 TBool securityPassed = EFalse;
1620 while(!securityPassed) // go out of this loop either Security check is passed or no DLL found.
1622 res = FindImplementation(aImplUid, aInterfaceUid, implData);
1623 if (KErrNone != res)
1627 CDllData* dll = implData->iParent->iParent;
1628 // security check is deferred to here.
1629 securityPassed = dll->ProcessSecurityCheckL();
1632 if (!aClientRequest.CheckCapability(dll->iCapSet, *(implData->iImplInfo)))
1634 return KErrPermissionDenied;
1636 dll->PopulateAnEntry(aEntry);
1637 aImplInfo = implData->iImplInfo;
1641 // remove the implementations of the DLL from iInterfaceImplIndex
1642 // and remove the DLL from its parent DLL list.
1653 This method removes all implementations of This Dll from the iInterfaceImplIndex,
1654 then delete this Dll from its parent Dll list.
1655 @param aDllData the Dll to be cleaned.
1656 @pre CRegistrar is fully constructed
1658 void CRegistryData::DeleteDllL(CDllData* aDllData) const
1660 // find the index of the passed aDllData in its parent's Dll list.
1661 TInt index = aDllData->iParent->FindDllIndex(aDllData->iDllEntry->GetThirdUid());
1663 //removes all implementations of This Dll from the iInterfaceImplIndex
1664 RemoveFromIndexL(aDllData);
1665 if(index != KErrNotFound)
1667 //now remove the dll from the array and registry
1668 TDll* dllList = aDllData->iParent->iDllList;
1669 dllList->Remove(index);
1675 Indicates whether the language downgrade path has changed.
1676 This means we need to call NearestLanguageFile again
1677 @return ETrue if the language downgrade path has changed, EFalse otherwise.
1678 @pre CRegistrar is fully constructed
1680 TBool CRegistryData::HasLanguageChanged() const
1682 return iLanguageChanged;
1687 @param aFs A handle to a connected file server.
1689 CRegistryData::CRegistryData(RFs& aFs,
1690 TInt aInterfaceImplIndexGranularity,
1691 TInt aImplIndexGranularity) :
1694 iInterfaceImplIndex(aInterfaceImplIndexGranularity),
1695 iImplIndex(aImplIndexGranularity)
1701 Completes the safe construction of the CRegistryData object.
1702 @leave KErrNoMemory.
1703 @pre This object is constructed
1704 @post This object is fully initialized
1706 void CRegistryData::ConstructL()
1708 iCachedDriveInfo = CEComCachedDriveInfo::NewL(iFs);
1710 // Construction of the empty registration data structure here
1711 iRegistrations = new(ELeave)TRegistration;
1712 iSystemDrive=iFs.GetSystemDrive();
1713 // during construction we always need to call this function in
1714 // order to initialise and store the language settings at boot time
1715 TRAPD(err,iLanguageChanged=RDowngradePath::HasChangedL(iFs))
1716 if (err==KErrNoMemory)
1718 User::LeaveNoMemory();
1723 To find the index entry for aImplementationUid.
1724 @return The index of the item or KErrIndexEntryNotFound.
1725 @pre This object is fully constructed
1727 TInt CRegistryData::IndexedFind(TUid aInterfaceUid) const
1729 // Find the correct implementation
1730 TInterfaceIndex key;
1731 key.iInterfaceUid = aInterfaceUid;
1732 return iInterfaceImplIndex.FindInOrder(key,TLinearOrder<TInterfaceIndex>(TInterfaceStruct::CompareInfUid));
1736 Used by a TIdentityRelation to decide if two CDriveData match.
1737 @return ETrue if the TDriveUnit inside the indexes match.
1738 @param aIndexOne The first CDriveData to compare
1739 @param aIndexTwo The second CDriveData to compare
1741 TBool CRegistryData::MatchOnDrive(const CRegistryData::CDriveData& aIndexOne,
1742 const CRegistryData::CDriveData& aIndexTwo)
1744 return aIndexOne.iDrive == aIndexTwo.iDrive;
1747 TInt CRegistryData::FindDriveL(const TDriveUnit& aDrive,
1748 CRegistryData::CDriveData*& aDriveData)const
1750 TInt index = KErrNotFound;
1752 // Set up the find parameters
1753 TIdentityRelation<CRegistryData::CDriveData> identity(MatchOnDrive);
1754 CDriveData* driveToMatch = CDriveData::NewLC(aDrive,const_cast<CRegistryData*>(this));
1756 index = iRegistrations->Find(driveToMatch, identity);
1758 CleanupStack::PopAndDestroy(driveToMatch);
1760 if(index != KErrNotFound)
1761 aDriveData = (*iRegistrations)[index];
1769 Finds the instance of CImplementationData for given impl uid and interface uid if known.
1770 @return KErrNone if impl found, otherwise KErrNotFound.
1771 @param aImplUid The uid of the impl to find
1772 @param aInterfaceUid The uid of the interface associated with the impl to find or less than
1774 @param aImplData Set to the instance CImplementationData found in the index or if
1775 not found set to NULL. This argument is always overwritten.
1777 TInt CRegistryData::FindImplementation(const TUid aImplUid, const TUid aInterfaceUid,
1778 CImplementationData*& aImplData) const
1782 // Index is kept up to date even when discoveries are occurring
1783 // therefore always search index for implementation
1785 // if aInterfaceUid is non zero use it to find position in index list
1786 if(aInterfaceUid.iUid != 0)
1788 i = IndexedFind(aInterfaceUid);
1789 if(i == KErrNotFound)
1791 return KErrNotFound;
1794 TInterfaceIndex listItem = iInterfaceImplIndex[i];
1795 i = listItem.iImpData.FindInOrder(aImplUid,
1796 TImplStruct::CompareUidAgainstImplStruct);
1799 aImplData = listItem.iImpData[i].iCurrentImpl;
1803 else // not given I/F UID. Use the iImplIndex.
1805 // NB: 1. Impl UID should be globally unique. It is an error if impls
1806 // have same Impl UID but different i/f UIDs. It means one of the
1807 // plug-in supplier made an error in the .RSS file or it could be a
1808 // deliberate attach. But to maintain backward compatibility, ECOM
1809 // allows this error.
1810 // (Multiple impls can have same i/f UID and same Impl UID. That is
1811 // the upgrade situation and only one of them get stored in iImplIndex.)
1813 // 2. Entries in iImplIndex are ordered by Impl UID, and if duplicated,
1814 // ordered by i/f UID. Here i/f UID is wild card. The situation is
1815 // analogous to the array was built with InsertInOrderAllowRepeats.
1816 // RPointerArray::SpecificFindInOrder is for ordered search in array
1817 // with duplicates. Note: it is very expensive to instantiate
1818 // a CImplementationData* just for binary search. Hence reinterpret
1819 // cast the TUid* as CImplementationData*. CompareUidAgainstImplData
1820 // knows to cast the first argument back to TUid.
1821 i = iImplIndex.SpecificFindInOrder(
1822 reinterpret_cast<const CImplementationData*>(&aImplUid),
1823 TLinearOrder<CImplementationData>(CImplementationData::CompareUidAgainstImplData),
1824 EArrayFindMode_First);
1825 if (i == KErrNotFound)
1827 return KErrNotFound;
1830 aImplData = iImplIndex[i];
1832 // If duplicates exist, they are sorted according to i/f UID
1833 // but we cannot take advantage of this order. To miniize risk
1834 // of malware using duplicate Impl UID as DOS attack, ECOM applies
1835 // the rule that ROMBased plug-in preferred over non-ROMBased,
1836 // higher drive letter preferred over lower drive letter, and
1837 // lastly lower i/f UID preferred over higher i/f UID. Must visit
1838 // every duplicate to compare their ROMBasedness and driver letter
1840 TInt count = iImplIndex.Count();
1841 for (TInt j = i + 1; j < count; j++)
1843 if ( iImplIndex[j]->iImplInfo->ImplementationUid().iUid != aImplUid.iUid )
1848 aImplData = SelectDuplicatedImpl(aImplData, iImplIndex[j]);
1854 return KErrNotFound;
1858 This function helps FindImplementation() to decide which of two
1859 implementations with duplicated implementation UID to choose.
1860 The selection rules are:
1861 1. ROM based plugins > non-ROM based.
1862 2. Higher drive letter > lower drive letter
1863 3. Lower I/F UID > Higher I/F UID (for backward compatibility reason)
1865 @param aImpl1 - one of the two implementations to compare.
1866 @param aImpl2 - the other implementation to compare.
1867 @return the preferred implementation.
1869 CRegistryData::CImplementationData* CRegistryData::SelectDuplicatedImpl(
1870 const CImplementationData* aImpl1,
1871 const CImplementationData* aImpl2) const
1875 TPtrC oldName = aImpl1->iImplInfo->DisplayName().Left(60);
1876 TPtrC newName = aImpl2->iImplInfo->DisplayName().Left(60);
1877 TPtrC oldDll = aImpl1->iParent->iParent->iDllEntry->GetName();
1878 TPtrC newDll = aImpl2->iParent->iParent->iDllEntry->GetName();
1881 const TInt KRomBasedFactor = 0x100;
1883 TInt drive1 = aImpl1->iImplInfo->Drive();
1884 if (aImpl1->iImplInfo->RomBased())
1886 drive1 |= KRomBasedFactor;
1889 TInt drive2 = aImpl2->iImplInfo->Drive();
1890 if (aImpl2->iImplInfo->RomBased())
1892 drive2 |= KRomBasedFactor;
1895 if (drive1 > drive2)
1898 if ((drive1 & KRomBasedFactor) && !(drive2 & KRomBasedFactor))
1900 __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);
1904 __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);
1908 return const_cast<CImplementationData*>(aImpl1);
1910 else if (drive1 < drive2)
1913 if ((drive2 & KRomBasedFactor) && !(drive1 & KRomBasedFactor))
1915 __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);
1919 __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);
1923 return const_cast<CImplementationData*>(aImpl2);
1925 // They are on the same drive. Choose the one with lower I/F UID.
1926 else if (aImpl2->iParent->iInterfaceUid.iUid < aImpl1->iParent->iInterfaceUid.iUid)
1929 __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);
1932 return const_cast<CImplementationData*>(aImpl2);
1936 __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);
1939 return const_cast<CImplementationData*>(aImpl1);
1943 Checks each entry in the registry to ensure that both the RSC file and the
1944 corresponding dll exist. If not the dll branch is removed
1945 from the tree. If a drive branch is found which contains no dlls
1947 @pre This object is fully constructed.
1948 @post Any out of date registry entries are removed from the tree.
1950 void CRegistryData::ValidateRegistryL()
1952 TInt driveIndex = iRegistrations->Count();
1953 TFileName rscFileName;
1954 while(driveIndex > 0)
1957 CDriveData* drive = (*iRegistrations)[driveIndex];
1959 if( !iCachedDriveInfo->DriveIsReadOnlyInternalL(drive->iDrive) &&
1960 drive->iDriveChanged)
1962 TInt dllIndex = drive->iDllList->Count();
1966 CDllData* dll = (*drive->iDllList)[dllIndex];
1967 //reset the buffer first
1969 rscFileName.Append(dll->iParent->iDrive.Name());
1970 rscFileName.Append(KEComResourceFilePath);
1971 rscFileName.Append(dll->iDllEntry->GetName().Left(dll->iDllEntry->GetName().Length()-4));
1972 if(dll->iRscFileExtension)
1974 rscFileName.Append(dll->iRscFileExtension->Des());
1977 // check the existence of RSC file in resource\plugins\ directory.
1978 // RSC filename is already a full name here.
1979 TBool rscFileExistence = BaflUtils::FileExists(iFs, rscFileName);
1981 //save the security info, this will both check existence of the dll
1982 //and cache the information for future use
1983 TBool success=dll->SaveSecurityInfoL();
1984 // If this dll is not found or the corresponding RSC file is not found, then remove it from the registry
1985 if(!success || !rscFileExistence)
1987 //remove all implementations of this dll from iInterfaceImplIndex
1988 RemoveFromIndexL(dll);
1989 drive->iDllList->Remove(dllIndex);
1990 //set flag to indicate registry data has been changed
1991 drive->iRegistryChanged = ETrue;
2000 Determines whether the new implmentation should be preferred over the existing implementation.
2001 Validates later version implementations from R/W drives.
2002 @param aOldImpl The array to append to
2003 @param aNewImpl The item to append
2004 @param aLegitimateImpl Flag, indicating whether current implementation is secure
2005 @return The preferred implementation
2006 @pre This object is fully constructed.
2008 CRegistryData::CImplementationData* CRegistryData::SelectPreferredImplL(CImplementationData* aOldImpl,
2009 CImplementationData* aNewImpl,
2010 TBool& aLegitimateImpl,
2011 TBool aCheckIsNeeded) const
2013 aLegitimateImpl = ETrue;
2015 TBool newIsRomOnly = aNewImpl->iImplInfo->RomOnly();
2016 TBool oldIsRomOnly = aOldImpl->iImplInfo->RomOnly();
2018 /* In addition to selecting the highest version of an
2019 implementation this check takes care of the following special
2021 1. Ensure that a higher-versioned RAM-based implementation
2022 cannot override a ROM-based version
2023 2. Allows for the case where there are two versions of the
2024 same ROM-only implementation on ROM, thus ensuring the
2025 higher of the two versions is used.
2027 if(newIsRomOnly && !oldIsRomOnly)
2031 else if((newIsRomOnly && oldIsRomOnly))
2033 if(aNewImpl->iImplInfo->Version() > aOldImpl->iImplInfo->Version())
2037 else if(aNewImpl->iImplInfo->Version() == aOldImpl->iImplInfo->Version())
2039 //any drive from Y-A has higher priority than Z drive
2040 //any drive with a letter alphabetically greater has higher priority
2041 if((aNewImpl->iImplInfo->Drive() != EDriveZ) &&
2042 (aOldImpl->iImplInfo->Drive() == EDriveZ || aNewImpl->iImplInfo->Drive() > aOldImpl->iImplInfo->Drive()))
2048 //all new implementations which are not flagged'RomOnly'
2049 else if(!newIsRomOnly && !oldIsRomOnly)
2051 TBool newIsReadOnly = iCachedDriveInfo->DriveIsReadOnlyInternalL(aNewImpl->iParent->iParent->iParent->iDrive);
2052 TBool oldIsReadOnly = iCachedDriveInfo->DriveIsReadOnlyInternalL(aOldImpl->iParent->iParent->iParent->iDrive);
2054 if(aNewImpl->iImplInfo->Version() > aOldImpl->iImplInfo->Version())
2056 //filter for implementations on R/W drive trying to upgrade implementations on R/O drive
2057 if((!newIsReadOnly && oldIsReadOnly) && aCheckIsNeeded)
2059 //test if a later implementation version on R/W drive has the same filename as the current one before upgrading.
2061 if(aOldImpl->iParent->iParent->iDllEntry->GetName().CompareF(aNewImpl->iParent->iParent->iDllEntry->GetName()) == 0)
2067 aLegitimateImpl = EFalse; //not secure
2076 else if(aNewImpl->iImplInfo->Version() == aOldImpl->iImplInfo->Version())
2078 //any drive from Y-A has higher priority than Z drive OR
2079 //any drive with a letter alphabetically greater has higher priority
2080 if((aNewImpl->iImplInfo->Drive() != EDriveZ) &&
2081 (aOldImpl->iImplInfo->Drive() == EDriveZ || aNewImpl->iImplInfo->Drive() > aOldImpl->iImplInfo->Drive()))
2083 //filename check needs to performed on implementation from R/W drives, trying to
2084 //upgrade implementations on R/O drive
2085 if((!newIsReadOnly && oldIsReadOnly) && aCheckIsNeeded)
2087 //test if a later implementation version on R/W drive has the same filename as the current one before upgrading.
2089 if(aOldImpl->iParent->iParent->iDllEntry->GetName().CompareF(aNewImpl->iParent->iParent->iDllEntry->GetName()) == 0)
2095 aLegitimateImpl = EFalse; //not secure
2110 This functions checks if a given implementation already exists in
2111 the indexes. If it does exist, determine if the given implementation
2112 should replace the existing one or not.
2113 @param aIdxArray The container array of the interface to hold the implementation.
2114 @param aNewImpl The implementation to filter.
2115 @param aInsertMode whether aNewIMpl is a newcomer of the interface, or an
2116 update of an existing implementation, or a older version of an
2117 existing implementation.
2118 @param aPosOfImplInArray return the index of aNewImpl in aIdxArray,
2119 @param aLegitimateImpl Flag,indicating whether current implementation is secure
2120 @pre This object is fully constructed.
2124 CRegistryData::FilterForLatestLegitimateImplL(TImplContainerArray& aIdxArray,
2125 CImplementationData* aNewImpl,
2126 TInsertImplMode& aInsertMode,
2127 TInt& aPosOfImplInArray,
2128 TBool& aLegitimateImpl,
2129 TBool aCheckIsNeeded)
2131 aInsertMode = EInsertImplUndefinedMode;
2132 aLegitimateImpl = ETrue;
2133 TImplStruct newImplStruct;
2134 newImplStruct.iCurrentImpl = aNewImpl;
2136 TInt idxPos = aIdxArray.FindInOrder(newImplStruct, TLinearOrder<TImplStruct> (TImplStruct::CompareImplStructUid));
2138 aPosOfImplInArray = idxPos;
2140 if(idxPos != KErrNotFound)
2142 if(aNewImpl != SelectPreferredImplL(aIdxArray[idxPos].iCurrentImpl,
2147 aInsertMode = EInsertImplAsUnusedImpl;
2151 aInsertMode = EInsertImplAsUpgradeOfExistingImpl;
2156 aInsertMode = EInsertImplAsNewcomerOfInterface;
2160 void CRegistryData::ResetTInterfaceIndex(TAny* aObject)
2162 TInterfaceIndex* index=reinterpret_cast<TInterfaceIndex*>(aObject);
2165 void CRegistryData::RemoveImplFromImplIndexCleanUp(TAny* aPtr)
2167 TCleanupImplIndexEntry *aCleanup = (TCleanupImplIndexEntry*)aPtr;
2168 aCleanup->iRegistryData->RemoveImplByAddrFromImplIndex(aCleanup->iImplEntry);
2172 This method retrieves the data for security checks from the iInterfaceImplIndex
2173 @param aImplPtr The new item to be checked
2174 @param aCheckIsNeeded Boolean indicating, whether a filename check is needed
2175 @pre This object is fully constructed.
2176 @post aImplPtr has been checked and added to the index, or not, depending on the
2177 outcome of the checks.
2179 TBool CRegistryData::InsertIntoIndexL(CImplementationData* aImplPtr, TBool aCheckIsNeeded)
2181 TBool legitimateImpl = ETrue;
2183 TInterfaceIndex newIndexEl;
2184 TInterfaceIndex* newElPtr;
2186 newIndexEl.iInterfaceUid = aImplPtr->iParent->iInterfaceUid;
2187 CleanupStack::PushL(TCleanupItem(ResetTInterfaceIndex,&newIndexEl));
2189 TInt indexPos = iInterfaceImplIndex.FindInOrder(newIndexEl,TLinearOrder<TInterfaceIndex>(TInterfaceStruct::CompareInfUid));
2192 if(indexPos!=KErrNotFound)
2193 newElPtr = &(iInterfaceImplIndex[indexPos]);
2195 newElPtr = &newIndexEl;
2197 // For each implementation make sure we only have the latest version
2198 TInt implIdxInContainerArray(KErrNotFound);
2199 TInsertImplMode insertMode;
2200 FilterForLatestLegitimateImplL(newElPtr->iImpData,
2203 implIdxInContainerArray,
2207 InsertImplInIndexesL(insertMode,
2210 implIdxInContainerArray,
2214 CleanupStack::Pop();
2216 if (insertMode == EInsertImplAsUpgradeOfExistingImpl)
2218 TUid ImplUid = aImplPtr->iImplInfo->ImplementationUid();
2219 iImplUpgradeCallBack.CallBack(ECallBackId_ImplUpgrade, &ImplUid);
2222 return legitimateImpl;
2225 /** Handle inserting a CImplementationData in iInterfaceImplIndex and
2228 @param aNewImpl the implementation to add to the indexes.
2229 @param aInsertMode indicates whether the implementation is a newcover
2230 of the interface, or is an update of existing implementation
2231 or is an older version of an existing implementation.
2232 @param aIfPosInInterfaceImplIndex is the index of the interface in
2233 iInterfaceImplIndex.
2234 @param aNewIfIndexEl the TInterfaceIndex object containing the implementation.
2235 @param aImplPosInContainerArray is the index of the implementation in
2236 the iImpData member of aNewIfIndexEl.
2237 @param aLegitimateImpl indicate if the implementation passed security check or not.
2238 @leave KErrNoMemory operation fails because the system is out of memory.
2239 @leave KErrGeneral any non-specific programming error.
2240 @leave KErrAlreadyExists the indexes already have an entry with the same
2241 Impl. UID and Interface UID.
2243 void CRegistryData::InsertImplInIndexesL(TInsertImplMode aInsertMode,
2244 TInt aIfPosInInterfaceImplIndex,
2245 TInterfaceIndex& aNewIfIndexEl,
2246 TInt aImplPosInContainerArray,
2247 CImplementationData* aNewImpl,
2248 TBool aLegitimateImpl)
2250 if(aInsertMode == EInsertImplUndefinedMode)
2252 // Will not happen because if FilterForLatestLegitimateImplL
2253 // does not leave then insertMode is set to one of the valid
2254 // values. If FilterForLatestLegitimateImplL leaves ecomserver exits.
2255 User::Leave(KErrGeneral);
2258 // If not a legitimate implementation, aNewImpl will be deleted.
2259 // Do not add it to the two indexes.
2261 if (!aLegitimateImpl)
2266 TImplContainerArray& implContainerArray = aNewIfIndexEl.iImpData;
2268 if(aInsertMode == EInsertImplAsNewcomerOfInterface)
2270 TImplStruct newImplStruct;
2271 newImplStruct.iCurrentImpl = aNewImpl;
2272 implContainerArray.InsertInOrderL(newImplStruct, TLinearOrder<TImplStruct> (TImplStruct::CompareImplStructUid));
2275 // on debug build, check for duplicated implementation UID
2276 // owned by different interfaces.
2278 TInt ii = iImplIndex.SpecificFindInOrder(
2280 CImplementationData::CompareImplUidIgnoreIfUid,
2281 EArrayFindMode_Any);
2282 if (ii != KErrNotFound)
2284 CImplementationData* otherImpl = iImplIndex[ii];
2285 TPtrC oldName = otherImpl->iImplInfo->DisplayName().Left(60);
2286 TPtrC newName = aNewImpl->iImplInfo->DisplayName().Left(60);
2287 __ECOM_TRACE5("** ECOM: error Impl UID %X, I/F UID %X DLL %S duplicating I/F %X DLL %S",\\
2288 aNewImpl->iImplInfo->ImplementationUid().iUid, \\
2289 aNewImpl->iParent->iInterfaceUid.iUid, &newName,\\
2290 otherImpl->iParent->iInterfaceUid.iUid, &oldName);
2294 else if(aInsertMode == EInsertImplAsUpgradeOfExistingImpl)
2296 TImplContainer& implContainer = implContainerArray[aImplPosInContainerArray];
2297 CImplementationData* oldImpl = implContainer.iCurrentImpl;
2298 implContainer.iUnusedImpls.AppendL(oldImpl);
2299 RemoveImplByAddrFromImplIndex(oldImpl); // ignore return code
2300 implContainer.iCurrentImpl = aNewImpl;
2302 // We are replacing existing impl with aNewImpl. If existing
2303 // corresponds to a ROM-based plug-in then aNewImpl is an
2304 // update and qualifies as ROM-based under current policy.
2305 if(oldImpl->iImplInfo->RomBased())
2307 aNewImpl->iImplInfo->SetRomBased(ETrue);
2309 __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()));
2310 __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()));
2312 else if (aInsertMode == EInsertImplAsUnusedImpl)
2314 TImplContainer& implContainer = implContainerArray[aImplPosInContainerArray];
2315 implContainer.iUnusedImpls.AppendL(aNewImpl);
2317 CImplementationData* currImpl = implContainer.iCurrentImpl;
2318 __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()));
2319 __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()));
2320 // We are not replacing existing with aNewImpl. However,
2321 // if aNewImpl corresponds to a ROM-based plug-in then the
2322 // existing impl is an update and qualifies as ROM-based
2323 // under current policy.
2324 if(aNewImpl->iImplInfo->RomBased())
2326 currImpl->iImplInfo->SetRomBased(ETrue);
2329 return; // the trailing steps not applicable to downgrade situation
2332 User::LeaveIfError( InsertImplIntoImplIndex(aNewImpl) );
2334 if(aIfPosInInterfaceImplIndex==KErrNotFound)
2336 TCleanupImplIndexEntry aCleanup(this, aNewImpl);
2337 CleanupStack::PushL(TCleanupItem(RemoveImplFromImplIndexCleanUp,&aCleanup));
2338 iInterfaceImplIndex.InsertInOrderL(aNewIfIndexEl, TLinearOrder<TInterfaceIndex>(TInterfaceStruct::CompareInfUid));
2339 CleanupStack::Pop();
2344 This method takes removes all implementations of a plugin from the iInterfaceImplIndex
2345 @param aDllData Reference to a fully constructed CDllData object
2346 @pre This object is constructed
2347 @post All implementations of the CDllData object, have been removed from the index
2349 void CRegistryData::RemoveFromIndexL(CDllData* aDllData) const
2351 //remove 'old' implementations from iInterfaceImplIndex
2353 TImplStruct implStruct;
2354 TInt ifListCount=aDllData->iIfList->Count();
2355 while(counter < ifListCount)
2357 CInterfaceData* interface = (*aDllData->iIfList)[counter];
2358 TInterfaceIndex index;
2359 index.iInterfaceUid = interface->iInterfaceUid;
2361 TInt indexPos = iInterfaceImplIndex.FindInOrder(index,TLinearOrder<TInterfaceIndex>(TInterfaceStruct::CompareInfUid));
2362 TImplContainerArray* impDataArray = NULL;
2365 if(indexPos!=KErrNotFound)
2367 impDataArray = &(iInterfaceImplIndex[indexPos].iImpData);
2368 implCount=interface->iImplementations->Count();
2369 for(TInt i=0; i<implCount; i++)
2371 implStruct.iCurrentImpl = (*interface->iImplementations)[i];
2372 TInt impPos = impDataArray->FindInOrder(implStruct, TLinearOrder<CRegistryData::TImplStruct> (TImplStruct::CompareImplStructUid));
2373 if(impPos!=KErrNotFound)
2375 TImplContainer& implContainer = (*impDataArray)[impPos];
2376 // remove from unused list if exists
2377 TInt count = implContainer.iUnusedImpls.Count();
2381 if(implContainer.iUnusedImpls[count]->iParent->iParent == aDllData)
2383 implContainer.iUnusedImpls.Remove(count);
2387 // update current entry
2388 if(implContainer.iCurrentImpl->iParent->iParent == aDllData)
2390 // do not care about the return code.
2391 RemoveImplByAddrFromImplIndex(implContainer.iCurrentImpl);
2393 TInt implContainerUnusedImplCount=implContainer.iUnusedImpls.Count();
2394 // no unused impl's therefore no rollback and remove entry
2395 if(implContainerUnusedImplCount == 0)
2397 implContainer.Reset();
2398 impDataArray->Remove(impPos);
2402 // Rollback to implementation from unused array
2403 // copy first from unused array
2404 TInt selectedPos = 0;
2405 implContainer.iCurrentImpl = implContainer.iUnusedImpls[selectedPos];
2407 // now check if any others in list should be preferred
2408 for(count = 1;count < implContainerUnusedImplCount; count++)
2410 // determine which implementation should be used.
2411 // no security check required as this will have already been done when impl was
2412 // first added to unused list.
2413 TBool dummLegitimateImpl;
2414 implContainer.iCurrentImpl = SelectPreferredImplL(implContainer.iCurrentImpl,
2415 implContainer.iUnusedImpls[count],
2418 if(implContainer.iCurrentImpl == implContainer.iUnusedImpls[count])
2420 selectedPos = count;
2423 implContainer.iUnusedImpls.Remove(selectedPos);
2424 User::LeaveIfError(InsertImplIntoImplIndex(implContainer.iCurrentImpl));
2427 CImplementationData* p = implContainer.iCurrentImpl;
2428 __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()));
2434 //To make sure it is removed from the implIndex no matter what and that the return code is ignored.
2435 //The previous removal is still required so that a subsequent InsertImplIntoImplIndex is possible
2436 //for an implementation. e.g. an implementaion is on different drives and only one of htem was removed.
2437 // The other one should now make it to the iImplIndex through InsertImplIntoImplIndex. If it wasn't
2438 // removed, before the insert operation, it will fail since both the implementations hav the same
2440 RemoveImplByAddrFromImplIndex((*interface->iImplementations)[i]);
2442 if(impDataArray->Count() == 0)
2444 iInterfaceImplIndex[indexPos].Reset();
2445 iInterfaceImplIndex.Remove(indexPos);
2453 This method inserts implementations into the index
2454 @param aDriveData Drive data in which we are searching the implementations
2455 @pre This object is fully constructed.
2456 @post all implemementations found, have been inserted into the index.
2458 void CRegistryData::AddImplDataL(CDriveData* aDriveData)
2460 TBool checkIsNeeded = EFalse;
2461 TInt dllCount=aDriveData->iDllList->Count();
2462 for(TInt dllCounter=0; dllCounter < dllCount;dllCounter++)
2464 CDllData* aDllData = (*aDriveData->iDllList)[dllCounter];
2465 TInt interfaceCount=aDllData->iIfList->Count();
2466 for(TInt ifListCounter=0; ifListCounter < interfaceCount; ifListCounter++)
2468 // For each interface structure
2469 CInterfaceData* interface = (*aDllData->iIfList)[ifListCounter];
2470 TInt implementationCount=interface->iImplementations->Count();
2471 for(TInt impNum = 0; impNum < implementationCount; ++impNum)
2473 CImplementationData* implData = (*interface->iImplementations)[impNum];
2474 TInt retValue = InsertIntoIndexL(implData, checkIsNeeded);
2475 //we know that during internalize, all implementations are legitimate,
2476 //as the check was already performed during discovery. No check is
2477 //performed at this stage, therefore ignore return value, as it will always
2485 /** This method does a search by address and removes the specified entry from iImplIndex.
2487 @param aPtr is the entry to remove
2488 @return True if aPtr is removed from iImplIndex. False if aPtr is not
2489 in iImplIndex, i.e. nothing is removed.
2491 TBool CRegistryData::RemoveImplByAddrFromImplIndex(CImplementationData* aPtr) const
2493 TInt aIdx = iImplIndex.Find(aPtr);
2494 if (aIdx != KErrNotFound)
2496 // This linear pointer search ensures a safe removal of the impl from iImplIndex so that it is free from a dead object.
2498 // The array does not own the pointer. Do not delete!
2499 iImplIndex.Remove(aIdx);
2505 /** This method inserts the entry aNewImpl into iImplIndex.
2507 @param aNewImpl the item to add to iImplIndex.
2508 @return KErrNone aNewImpl is successfully added to the index.
2509 KErrAlreadyExists iImplIndex has an entry with the same
2510 implementation UID and same parent Interface UID.
2511 KErrNoMemory fail to insert due to out of memory.
2512 Other system wide errors.
2513 @pre aNewImpl is fully constructed.
2515 TInt CRegistryData::InsertImplIntoImplIndex(CImplementationData* aNewImpl) const
2517 TLinearOrder<CImplementationData> ImplUidComparator(CImplementationData::CompareImplUid);
2519 return iImplIndex.InsertInOrder(aNewImpl, ImplUidComparator);
2522 /** This method checks whether the language downgrade path has been changed.
2523 If it is, save language downgrade path information and set related flag true.
2525 @param aLanguageChanged the returned value to indicate language changed.
2527 void CRegistryData::LanguageChangedL(TBool& aLanguageChanged)
2529 iLanguageChanged = RDowngradePath::HasChangedL(iFs);
2530 aLanguageChanged = iLanguageChanged;
2534 If need to unset the callback, use a TCallBackWithArg constructed with no
2537 void CRegistryData::SetImplUpgradeCallBack(const TCallBackWithArg& aCallBack)
2539 iImplUpgradeCallBack = aCallBack;
2542 #ifdef __ECOM_SERVER_PERFORMANCE__
2544 This method calculates the drive, plugins, interfaces, implementations counts for the drive
2546 @param aType The drive type for which the counts should be calculated
2547 @param aCounts Holds the calculated counts
2548 @pre This object is constructed
2550 void CRegistryData::GetRegistryCountsL(TInt aType, RegistryCounts::TRegistryCounts& aCounts) const
2552 aCounts.iDrives = 0;
2553 aCounts.iImplementations = 0;
2554 aCounts.iInterfaces = 0;
2557 for(TInt driveIndex = 0; driveIndex< iRegistrations->Count(); driveIndex++)
2559 CDriveData* drive = (*iRegistrations)[driveIndex];
2560 TBool isReadOnly = iCachedDriveInfo->DriveIsReadOnlyInternalL(drive->iDrive);
2562 if((aType == RegistryCounts::TRegistryCounts::EAll) ||
2563 (aType == RegistryCounts::TRegistryCounts::ERoInternal && isReadOnly) ||
2564 (aType == RegistryCounts::TRegistryCounts::ENonRoInternal && !isReadOnly))
2567 aCounts.iDlls += drive->iDllList->Count();
2568 for(TInt dllIndex = 0; dllIndex < drive->iDllList->Count(); dllIndex++)
2570 CDllData* dllList = (*drive->iDllList)[dllIndex];
2571 aCounts.iInterfaces += dllList->iIfList->Count();
2572 for(TInt ifIndex = 0; ifIndex < dllList->iIfList->Count(); ifIndex++)
2574 CInterfaceData* ifList = (*dllList->iIfList)[ifIndex];
2575 aCounts.iImplementations += ifList->iImplementations->Count();
2582 #endif // __ECOM_SERVER_PERFORMANCE__