os/ossrv/lowlevellibsandfws/pluginfw/Framework/frame/RegistryData.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // This file contains the implementations of the classes
    15 // which manage the internal structure of the registry.
    16 // 
    17 //
    18 
    19 #include <e32base.h>
    20 #include <e32uid.h>
    21 #include <s32stor.h>
    22 #include <s32file.h>
    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
    27 
    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
    40 
    41 const TInt KRegVersion = -9999;
    42 const TInt KDllExtensionLength=4;
    43 _LIT(KDllExtension,".dll");
    44 //
    45 //CRegistryData::TInterfaceStruct
    46 
    47 /**
    48 Constructor for TInterfaceStruct
    49 @post			Its member variable is granulated as (1)
    50 */
    51 CRegistryData::TInterfaceStruct::TInterfaceStruct():
    52 	iImpData(1)
    53 	{
    54 	// Do nothing here
    55 	}
    56 
    57 /**
    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
    63 */
    64 TInt CRegistryData::TInterfaceStruct::CompareInfUid(const TInterfaceIndex& indexEntry1,
    65 													const TInterfaceIndex& indexEntry2)
    66 	{
    67 	return CompareTUidValues(indexEntry1.iInterfaceUid.iUid,indexEntry2.iInterfaceUid.iUid);
    68 	}
    69 
    70 //
    71 //CRegistryData::TImplStruct
    72 
    73 /**
    74 Constructor for TImplStruct
    75 @post			Its member variables are initialised
    76 */
    77 CRegistryData::TImplStruct::TImplStruct():
    78 	iCurrentImpl(NULL),
    79 	iUnusedImpls(1)
    80 	{
    81 	// Do nothing here
    82 	}
    83 	
    84 /**
    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
    90 */
    91  TInt CRegistryData::TImplStruct::CompareImplStructUid(const TImplStruct& aEntry1,
    92 													const TImplStruct& aEntry2)
    93 	{
    94 	return CompareTUidValues(aEntry1.iCurrentImpl->iImplInfo->ImplementationUid().iUid,
    95 						aEntry2.iCurrentImpl->iImplInfo->ImplementationUid().iUid);
    96 	}
    97 	
    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.
   102 */
   103 TInt CRegistryData::TImplStruct::CompareUidAgainstImplStruct(
   104 	const TUid* aUid,
   105 	const TImplStruct& aEntry)
   106 	{
   107 	return CompareTUidValues(aUid->iUid,aEntry.iCurrentImpl->iImplInfo->ImplementationUid().iUid);
   108 	}
   109 
   110 //
   111 // CRegistryData::CImplementationData class
   112 
   113 /**
   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 
   119 				the CleanupStack.
   120 */
   121 CRegistryData::CImplementationData* CRegistryData::CImplementationData::NewLC(CInterfaceData* aParent)
   122 	{
   123 	CImplementationData* self=new(ELeave) CImplementationData(aParent);
   124 	CleanupStack::PushL(self);
   125 	return self;
   126 	}
   127 
   128 /**
   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.
   142 */
   143 CRegistryData::CImplementationData* CRegistryData::CImplementationData::NewL(CInterfaceData* aParent,
   144 																				TUid	aUid,
   145 																				TInt	aVersion, 
   146 																				HBufC*  aName,
   147 																				HBufC8* aDataType,
   148 																				HBufC8* aOpaqueData,
   149 																				TDriveUnit aDrive,
   150 																				TBool aRomOnly,
   151 																				TBool aRomBased)
   152 	{
   153 	CImplementationData* self=new(ELeave) CImplementationData(aParent);
   154 	CleanupStack::PushL(self);
   155 	self->ConstructL(aUid,
   156 					aVersion,
   157 					aName,
   158 					aDataType,
   159 					aOpaqueData,
   160 					aDrive,
   161 					aRomOnly,
   162 					aRomBased);
   163 	CleanupStack::Pop(self);
   164 	return self;
   165 	}
   166 
   167 /**
   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.
   183 */
   184 CRegistryData::CImplementationData* CRegistryData::CImplementationData::NewL(CInterfaceData* aParent,
   185 																				TUid	aUid,
   186 																				TInt	aVersion, 
   187 																				HBufC*  aName,
   188 																				HBufC8* aDataType,
   189 																				HBufC8* aOpaqueData,
   190 																				TDriveUnit aDrive,
   191 																				TBool aRomOnly,
   192 																				TBool aRomBased,
   193 																				RExtendedInterfacesArray* aExtendedInterfaces)
   194 	{
   195 	CImplementationData* self=new(ELeave) CImplementationData(aParent);
   196 	CleanupStack::PushL(self);
   197 	self->ConstructL(aUid,
   198 					aVersion,
   199 					aName,
   200 					aDataType,
   201 					aOpaqueData,
   202 					aDrive,
   203 					aRomOnly,
   204 					aRomBased,
   205 					aExtendedInterfaces);
   206 	CleanupStack::Pop(self);
   207 	return self;
   208 	}
   209 
   210 /**
   211 Constructor for CImplementationData
   212 @param			aParent The parent interface data of this implementation
   213 @post			Its member variables are initialised
   214 */
   215 CRegistryData::CImplementationData::CImplementationData( CInterfaceData* aParent):
   216 	CBase(),
   217 	iImplInfo(NULL),
   218 	iParent(aParent)
   219 	{
   220 	// Do nothing here
   221 	}
   222 	
   223 /**
   224 Destructor of CImplementationData
   225 */	
   226 CRegistryData::CImplementationData::~CImplementationData()
   227 	{
   228 	if(iImplInfo)
   229 		{
   230 		delete iImplInfo;
   231 		iImplInfo = NULL;	
   232 		}	
   233 	iParent = NULL;		
   234 	}
   235 
   236 /**
   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.
   248 */
   249 void CRegistryData::CImplementationData::ConstructL(TUid	aUid,
   250 													TInt	aVersion, 
   251 													HBufC*  aName,
   252 													HBufC8* aDataType,
   253 													HBufC8* aOpaqueData,
   254 													TDriveUnit aDrive,
   255 													TBool aRomOnly,
   256 													TBool aRomBased)
   257 	{
   258 	CImplementationInformation* newImpl = CImplementationInformation::NewL(aUid, 
   259 													 						aVersion, 
   260 													 						aName, 
   261 															 				aDataType,
   262 																			aOpaqueData,
   263 														 					aDrive,
   264 														 					aRomOnly,
   265 													 						aRomBased);
   266 	this->iImplInfo = newImpl;
   267 	}
   268 
   269 /**
   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.
   283 */
   284 void CRegistryData::CImplementationData::ConstructL(TUid	aUid,
   285 													TInt	aVersion, 
   286 													HBufC*  aName,
   287 													HBufC8* aDataType,
   288 													HBufC8* aOpaqueData,
   289 													TDriveUnit aDrive,
   290 													TBool aRomOnly,
   291 													TBool aRomBased,
   292 													RExtendedInterfacesArray* aExtendedInterfaces)
   293 	{
   294 	CImplementationInformation* newImpl = CImplementationInformation::NewL(aUid, 
   295 													 						aVersion, 
   296 													 						aName, 
   297 															 				aDataType,
   298 																			aOpaqueData,
   299 														 					aDrive,
   300 														 					aRomOnly,
   301 													 						aRomBased,
   302 													 						aExtendedInterfaces);
   303 	this->iImplInfo = newImpl;
   304 	}
   305 
   306 /**
   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.
   311 */
   312 void CRegistryData::CImplementationData::InternalizeL(RReadStream& aStore)
   313 	{
   314 	if (iImplInfo)
   315 		{
   316 		delete iImplInfo;
   317 		iImplInfo = NULL;
   318 		}
   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;
   325 	implInfo = 0;
   326 	}
   327 
   328 
   329 /**
   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.
   333 */
   334 void CRegistryData::CImplementationData::ExternalizeL(RWriteStream& aStore) const
   335 	{
   336 	iImplInfo->ExternalizeL(EFalse,aStore);
   337 	}
   338 
   339 /**
   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
   351 */	
   352 TInt CRegistryData::CImplementationData::CompareImplUid(const CImplementationData& aImpl1,
   353 													const CImplementationData& aImpl2)
   354 	{
   355 	TInt ret = CompareTUidValues(aImpl1.iImplInfo->ImplementationUid().iUid, aImpl2.iImplInfo->ImplementationUid().iUid);
   356 	if (ret != 0)
   357 		{
   358 		return ret;
   359 		}
   360 
   361 	// Implementation UIDs are equal, use I/F UID as tie breaker
   362 	return CompareTUidValues(aImpl1.iParent->iInterfaceUid.iUid, aImpl2.iParent->iInterfaceUid.iUid);
   363 	}
   364 	
   365 /**
   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
   372 */	
   373 TInt CRegistryData::CImplementationData::CompareImplUidIgnoreIfUid(const CImplementationData& aImpl1,
   374 													const CImplementationData& aImpl2)
   375 	{
   376 	return CompareTUidValues(aImpl1.iImplInfo->ImplementationUid().iUid,
   377 						aImpl2.iImplInfo->ImplementationUid().iUid);
   378 	}
   379 	
   380 /**
   381 This method determines the order of an CImplementationData object in relation
   382 to aUid.
   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.
   386 */	
   387 TInt CRegistryData::CImplementationData::CompareUidAgainstImplData(
   388 											const CImplementationData& aUid,
   389 											const CImplementationData& aImplData)
   390 	{
   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);
   395 	}
   396 
   397 //
   398 // CRegistryData::CInterfaceData class
   399 
   400 /**
   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 
   406 				the CleanupStack.
   407 */
   408 CRegistryData::CInterfaceData* CRegistryData::CInterfaceData::NewLC(TUid aInterfaceUid,CDllData* aParent)
   409 	{
   410 	CInterfaceData* self=new(ELeave) CInterfaceData(aInterfaceUid,aParent);
   411 	CleanupStack::PushL(self);
   412 	self->ConstructL();
   413 	return self;
   414 	}
   415 
   416 
   417 /**
   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 
   423 				the CleanupStack.
   424 */
   425 CRegistryData::CInterfaceData* CRegistryData::CInterfaceData::NewLC(CDllData* aParent)
   426 	{
   427 	CInterfaceData* self=new(ELeave) CInterfaceData(aParent);
   428 	CleanupStack::PushL(self);
   429 	self->ConstructL();
   430 	return self;
   431 	}
   432 
   433 
   434 CRegistryData::CInterfaceData::~CInterfaceData()
   435 	{
   436 	if(iImplementations)
   437 		{
   438 		iImplementations->ResetAndDestroy();
   439 		delete iImplementations;
   440 		iImplementations = NULL;
   441 		}
   442 	iParent = NULL;
   443 	}
   444 
   445 /**
   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.
   450 */
   451 void CRegistryData::CInterfaceData::AddL(const CImplementationData* aImplementation)
   452 	{
   453 #ifdef ECOM_TRACE
   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()));
   456 
   457 #endif
   458 
   459 	User::LeaveIfError(iImplementations->Append(aImplementation));
   460 	}
   461 
   462 /**
   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
   467 */
   468 void CRegistryData::CInterfaceData::SetInterfaceUid(TUid aInterfaceUid)
   469 	{
   470 	iInterfaceUid = aInterfaceUid;
   471 	}
   472 
   473 /**
   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.
   477 */
   478 void CRegistryData::CInterfaceData::ExternalizeL(RWriteStream& aStore) const
   479 	{
   480 	aStore.WriteInt32L(iInterfaceUid.iUid);
   481 
   482 	if(iImplementations)
   483 		{
   484 		const TInt entryCount = iImplementations->Count();
   485 		aStore.WriteInt32L(entryCount);
   486 		for(TInt i = 0; i < entryCount; ++i)
   487 			((*iImplementations)[i])->ExternalizeL(aStore);
   488 		}
   489 	else
   490 		aStore.WriteInt32L(0);
   491 	}
   492 
   493 /**
   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.
   499 */
   500 void CRegistryData::CInterfaceData::InternalizeL(RReadStream& aStore)
   501 	{
   502 	iInterfaceUid.iUid = aStore.ReadInt32L();
   503 
   504 	const TInt entryCount = aStore.ReadInt32L();
   505     if(entryCount < 0)
   506         {
   507         User::Leave(KErrCorrupt);
   508         }
   509 	
   510 	for(TInt i = 0; i < entryCount; ++i)
   511 		{
   512 		CImplementationData* implementation = CImplementationData::NewLC(this);
   513 		implementation->InternalizeL(aStore);
   514 		AddL(implementation);
   515 		CleanupStack::Pop(implementation);		//now owned by this interface
   516 		}
   517 	}
   518 
   519 /**
   520 @param			aParent A pointer to the parent dll data
   521 */
   522 CRegistryData::CInterfaceData::CInterfaceData(CDllData* aParent) :
   523 	CBase(),
   524 	iParent(aParent)
   525 	{
   526 	// Do nothing here
   527 	}
   528 
   529 /**
   530 @param			aInterfaceUid The unique Id of this interface
   531 @param			aParent A pointer to the parent dll data			
   532 */
   533 CRegistryData::CInterfaceData::CInterfaceData(TUid aInterfaceUid, CDllData* aParent) :
   534 	CBase(),
   535 	iInterfaceUid(aInterfaceUid),
   536 	iParent(aParent)
   537 	{
   538 	// Do nothing here
   539 	}
   540 
   541 /**
   542 Standard second phase construction function
   543 @pre 			This object is fully constructed.
   544 @post			This object is fully constructed and initialized.
   545 */
   546 void CRegistryData::CInterfaceData::ConstructL()
   547 	{
   548 	iImplementations = new(ELeave) RPointerArray<CImplementationData>;
   549 	}
   550 
   551 //
   552 // CRegistryData::CDllData class
   553 
   554 /**
   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.
   563 */
   564 CRegistryData::CDllData* CRegistryData::CDllData::NewLC(const TDesC& aDllName,const TTime& aDllModTime,const TUid& aSecondUid,const TUid& aThirdUid,CDriveData* aParent)
   565 	{
   566 	CDllData* self=new(ELeave) CDllData(aParent);
   567 	CleanupStack::PushL(self);
   568 	self->ConstructL(aDllName,aDllModTime,aSecondUid,aThirdUid);
   569 	return self;
   570 	}
   571 
   572 /**
   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.
   577 */
   578 CRegistryData::CDllData* CRegistryData::CDllData::NewLC( CDriveData* aParent)
   579 	{
   580 	CDllData* self=new(ELeave) CDllData(aParent);
   581 	CleanupStack::PushL(self);
   582 	self->ConstructL();
   583 	return self;
   584 	}
   585 	
   586 CRegistryData::CDllData::~CDllData()
   587 	{
   588 	if(iIfList)
   589 		{
   590 		// Clear the interface list and destroy its objects
   591 		iIfList->ResetAndDestroy();
   592 		delete iIfList;
   593 		iIfList = NULL;
   594 		}
   595 
   596 	// Unload this implementation dll.
   597 	delete iDllEntry;
   598 	iDllEntry = NULL;
   599 	iParent = NULL;
   600 	delete iRscFileExtension;
   601 	}
   602 
   603 /**
   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.
   608 */
   609 void CRegistryData::CDllData::AddL(const CInterfaceData* aInterface)
   610 	{
   611 	User::LeaveIfError(iIfList->Append(aInterface));
   612 	}
   613 
   614 /**
   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.
   619 */
   620 void CRegistryData::CDllData::SetResourceExtL(const TDesC& aExt)
   621 	{
   622 	delete iRscFileExtension;
   623 	iRscFileExtension = NULL;
   624 	iRscFileExtension = aExt.AllocL();
   625 	}
   626 	
   627 /**
   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. 
   631 */
   632 TBool CRegistryData::CDllData::SaveSecurityInfoL()
   633 	{
   634 	iSecurityChecked = EFalse;
   635 
   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);
   644 	if(ret != KErrNone)
   645 		{
   646 		return EFalse;
   647 		}
   648 	// Set the DLL's capability
   649 	iCapSet = infoBuf().iSecurityInfo.iCaps;
   650 	// Set the DLL's VID
   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;
   654 	if (!match)
   655 		{
   656 #ifdef __ECOM_TRACE 
   657 		__ECOM_TRACE1("ERROR: Plugin SID Mismatch ERROR for %S.", &dllFullName);
   658 #endif
   659 		return EFalse;
   660 		}
   661 
   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];
   664 	if (!match)
   665 		{
   666 #ifdef __ECOM_TRACE 
   667 		__ECOM_TRACE1("ERROR: Plugin UID2 Mismatch ERROR for %S.", &iDllEntry->GetName());
   668 #endif
   669 		return EFalse;
   670 		}
   671 
   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))
   679 		{
   680 		RLoader loader;
   681 		TInt err = loader.Connect();
   682 		if(err != KErrNone)
   683 			{
   684 			return EFalse;
   685 			}
   686 		err = loader.CheckLibraryHash(dllFullName, ETrue);
   687 		loader.Close();
   688 		if(err != KErrNone)
   689 			{
   690 #ifdef __ECOM_TRACE 
   691 		__ECOM_TRACE2("INFO: Hash Check Failed for %S with error %d.", &dllFullName, err);
   692 #endif
   693 			return EFalse;
   694 			}
   695 		}
   696 #endif
   697 	iSecurityChecked = ETrue;
   698 	return ETrue;
   699 	}
   700 	
   701 /**
   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. 
   705 */
   706 TBool CRegistryData::CDllData::ProcessSecurityCheckL()
   707 	{
   708 	if(iSecurityChecked)
   709 		{
   710 		return ETrue;
   711 		}
   712 	return SaveSecurityInfoL();
   713 	}
   714 
   715 
   716 /**
   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.
   720 */
   721 void CRegistryData::CDllData::ExternalizeL(RWriteStream& aStore) const
   722    	{
   723  	const CEComEntry& dllEntryData = *iDllEntry;
   724 	TInt size=dllEntryData.GetName().Length()-KDllExtensionLength;
   725  	aStore.WriteUint32L(size);
   726 	aStore.WriteL(dllEntryData.GetName(),size);
   727 
   728 	aStore.WriteInt32L(dllEntryData.GetSecondUid().iUid);	
   729   	aStore.WriteInt32L(dllEntryData.GetThirdUid().iUid);
   730     TPckg<TTime> modified(dllEntryData.GetModified());	
   731  	aStore.WriteL(modified);
   732 	
   733 	if(!iParent->iParent->iCachedDriveInfo->DriveIsReadOnlyInternalL(iParent->iDrive))
   734 		{
   735 		if(iRscFileExtension)
   736 			{
   737 			size = iRscFileExtension->Length();		
   738 			aStore.WriteUint32L(size);
   739 			aStore.WriteL(iRscFileExtension->Des(),size);
   740 			}
   741 		else
   742 			{
   743 			aStore.WriteUint32L(0);		
   744 			}
   745 		}
   746 
   747    	if(iIfList)
   748    		{
   749    		const TInt entryCount = iIfList->Count();
   750    		aStore.WriteInt32L(entryCount);
   751    		for(TInt i = 0; i < entryCount; ++i)
   752    			((*iIfList)[i])->ExternalizeL(aStore);
   753    		}
   754    	else
   755    		aStore.WriteInt32L(0);
   756 	}
   757 
   758 /**
   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.
   763 */
   764 void CRegistryData::CDllData::InternalizeL(RReadStream& aStore)
   765  	{	
   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)
   771         {
   772         User::Leave(KErrCorrupt);
   773         }
   774 	HBufC* name = HBufC::NewLC(size+KDllExtensionLength);
   775 	TPtr ptr=name->Des();
   776 	aStore.ReadL(ptr,size);
   777 	ptr.Append(KDllExtension);
   778 
   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);
   784 
   785 	if(!iParent->iParent->iCachedDriveInfo->DriveIsReadOnlyInternalL(iParent->iDrive))
   786 		{
   787 		size = aStore.ReadUint32L();
   788 		if(size < 0 || size > KMaxFileName)
   789 		        {
   790        		 User::Leave(KErrCorrupt);
   791        		 }
   792 		if(size)
   793 			{
   794 			iRscFileExtension = HBufC::NewL(size);
   795 			TPtr extPtr = iRscFileExtension->Des();
   796 			aStore.ReadL(extPtr,size);
   797 			}
   798 		}
   799 	
   800    	const TInt entryCount = aStore.ReadInt32L();
   801 	if(entryCount < 0)
   802 		{
   803 		User::Leave(KErrCorrupt);
   804 		}
   805 	
   806    	// Security check is deferred until the DLL is needed.
   807 	for(TInt i = 0; i < entryCount; ++i)
   808 	   	{
   809 	   	CInterfaceData* interface = CInterfaceData::NewLC(this);
   810 		interface->InternalizeL(aStore);
   811 		AddL(interface);
   812 	   	CleanupStack::Pop(interface);	// now owned by dll	
   813 		}  	
   814 	iDllEntry = CEComEntry::NewL(*name,secondUid,thirdUid);
   815 	iDllEntry->SetModified(dllModifiedTime);
   816  	CleanupStack::PopAndDestroy(name);
   817 	}
   818 	
   819 /**
   820 @param			aParent The parent drive data of this implementation
   821 */	
   822 CRegistryData::CDllData::CDllData( CDriveData* aParent) :
   823 	CBase(),
   824 	iParent(aParent)	
   825 	{
   826 	// Do nothing here
   827 	//Initialize empty capabilities here
   828 	iCapSet.SetEmpty();
   829 	}
   830 
   831 /**
   832 Standard second phase construction function.
   833 @pre 			This object is fully constructed.
   834 @post			This object is fully constructed and initialized.
   835 */
   836 void CRegistryData::CDllData::ConstructL()
   837 	{
   838 	iIfList = new(ELeave) RInterfaceList(2);
   839 	}
   840 
   841 /**
   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.
   851 */
   852 void CRegistryData::CDllData::ConstructL(const TDesC& aDllName,const TTime& aDllModTime,const TUid& aSecondUid,const TUid& aThirdUid)
   853 	{
   854 	iIfList = new(ELeave) RInterfaceList(2);
   855 	iDllEntry=CEComEntry::NewL(aDllName,aSecondUid,aThirdUid);
   856 	iDllEntry->SetModified(aDllModTime);
   857 	}
   858 	
   859 	
   860 /**
   861 Populate a caller-supplied TEntry instance with the data from this dll.
   862 @param			aEntry An entry to be populated (destination)
   863 */
   864 void CRegistryData::CDllData::PopulateAnEntry(TEntry& aEntry) const
   865 	{
   866 	TPtr bufPtr=aEntry.iName.Des();
   867 	bufPtr.Zero();
   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();
   873 	}
   874 
   875 
   876 //
   877 // CRegistryData::CDriveData class
   878 
   879 
   880 /**
   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.
   886 */
   887 CRegistryData::CDriveData* CRegistryData::CDriveData::NewLC(TDriveUnit aDrive, CRegistryData* aParent)
   888 	{
   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'.
   892 	return self;
   893 	}
   894 
   895 /**
   896 Destructor of CDriveData
   897 */
   898 CRegistryData::CDriveData::~CDriveData()
   899 	{
   900 	if(iDllList)
   901 		{
   902 		// Clear the interface list and destroy its objects
   903 		iDllList->ResetAndDestroy();
   904 		delete iDllList;
   905 		iDllList = NULL;
   906 		}
   907 	iParent=NULL;
   908 	}
   909 
   910 /**
   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.
   915 */
   916 void CRegistryData::CDriveData::AddL(const CDllData* aDll)
   917 	{
   918 	User::LeaveIfError(iDllList->Append(aDll));
   919 	}
   920 
   921 /**
   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.
   926 */
   927 TInt CRegistryData::CDriveData::FindDllIndex(const TUid aDllUid) const
   928 	{
   929 	const TInt dllCount = iDllList->Count();
   930 	for(TInt j = 0; j < dllCount; ++j)
   931 		{
   932 		// For each dll structure check the dll UID
   933 		CDllData* dll = (*iDllList)[j];
   934 		if (aDllUid == dll->iDllEntry->GetThirdUid())
   935 			{
   936 			// it is a match 
   937  			return j;	 				
   938 			}
   939 		}
   940 	return KErrNotFound;
   941 	}
   942 	
   943 /**
   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.
   948 */
   949 void CRegistryData::CDriveData::ExternalizeL(RFs& aFs,const TDesC& aDatFileName) 
   950 	{
   951 	// attempt to create the folders if they don't already exist
   952 	TInt mkDirError = aFs.MkDirAll(aDatFileName);
   953 		
   954 	if((mkDirError == KErrNone) || (mkDirError == KErrAlreadyExists))
   955 		{
   956 		RFileWriteStream registryStream;
   957 		if(registryStream.Replace(aFs,aDatFileName, EFileWrite) == KErrNone)
   958 			{
   959 			CleanupClosePushL(registryStream);
   960 			// Write the version number as the first thing in the file stream.
   961 			registryStream.WriteInt32L(KRegVersion);				
   962 			if(iDllList)
   963 				{
   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);
   969 				}
   970 			else
   971 				registryStream.WriteInt32L(0);
   972 			CleanupStack::PopAndDestroy();	// registryStream
   973 			}
   974 		}
   975 	else
   976 		{
   977 		User::Leave(mkDirError);
   978 		}
   979 	}
   980 
   981 /**
   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.
   988 */
   989 
   990 void CRegistryData::CDriveData::InternalizeL(RFs& aFs,const TDesC& aDatFileName)
   991 	{
   992 	DoInternalizeL(aFs, aDatFileName);  
   993 	}
   994 /**
   995 The method internalizes the data from aRegistryStream.
   996 @param aFileName The ECOM registry file name.
   997 @leave KErrNoMemory
   998 @leave Some other system-wide error codes as well.
   999 */
  1000 void CRegistryData::CDriveData::DoInternalizeL(RFs& aFs, const TDesC& aFileName)
  1001 	{
  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)
  1008 		{
  1009 		User::Leave(KErrCorrupt);
  1010 		}	
  1011 	const TInt entryCount = registryStream.ReadInt32L();
  1012     if(entryCount < 0)
  1013         {
  1014         User::Leave(KErrCorrupt);
  1015         }
  1016 	for(TInt i = 0; i < entryCount; ++i)
  1017 		{
  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.		
  1022 		AddL(dll);
  1023 		CleanupStack::Pop(dll);	// now owned by drive	
  1024 		}
  1025 	CleanupStack::PopAndDestroy();//registryStream
  1026 	}
  1027 	
  1028 /**	
  1029 Constructor for CDriveData
  1030 @param			aDrive Information on this drive
  1031 @param			aParent A pointer to the parent registry data
  1032 */	
  1033 CRegistryData::CDriveData::CDriveData(TDriveUnit aDrive, CRegistryData* aParent) :
  1034 	CBase(), 
  1035 
  1036 	iDrive(aDrive),
  1037 	iParent(aParent),
  1038 	iDriveChanged(ETrue),
  1039 	iRegistryChanged(EFalse)
  1040 	{
  1041 	}
  1042 
  1043 /**
  1044 Standard second phase construction function.
  1045 @pre 			This object is fully constructed.
  1046 @post			This object is fully constructed and initialized.
  1047 */
  1048 void CRegistryData::CDriveData::ConstructL()
  1049 	{
  1050 	iDllList = new(ELeave)TDll;
  1051 	}
  1052 	
  1053 //
  1054 // CRegistryData class
  1055 
  1056 /**
  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.
  1066 */
  1067 CRegistryData* CRegistryData::NewL(RFs& aFs, TInt aInterfaceImplIndexGranularity, TInt aImplIndexGranularity)
  1068 	{
  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);
  1073 	return self;
  1074 	}
  1075 
  1076 /**
  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.
  1085 */
  1086 CRegistryData* CRegistryData::NewL(RFs& aFs)
  1087 	{
  1088 	return NewL(aFs, KDefaultInterfaceImplIndexGranularity, KDefaultImplIndexGranularity);
  1089 	}
  1090 
  1091 CRegistryData::~CRegistryData()
  1092 	{
  1093 	for(TInt index = 0; index < iInterfaceImplIndex.Count(); ++index)
  1094 		{
  1095 		iInterfaceImplIndex[index].Reset();
  1096 		}
  1097 	iInterfaceImplIndex.Reset();
  1098 	
  1099 	// Reset only. It does not own the pointers!
  1100 	iImplIndex.Reset();
  1101 
  1102 	if(iRegistrations)
  1103 		{
  1104 		iRegistrations->ResetAndDestroy();
  1105 		delete iRegistrations;
  1106 		iRegistrations = NULL;
  1107 		}
  1108 	// reset the cached language settings
  1109 	RDowngradePath::Reset();
  1110 
  1111 	delete iCachedDriveInfo;
  1112 	}
  1113 
  1114 /**
  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.
  1119 */
  1120 void CRegistryData::ListImplementationsL(TUid aInterfaceUid,
  1121 										 RImplDataArray& aImplementationData) const
  1122 	{
  1123 	// If discoveries are taking place then the index is invalid so we cannot do
  1124 	// this request
  1125 	if(iCurrentlyDiscovering)
  1126 		User::Leave(KEComErrListCurrentlyUnavailable);
  1127     
  1128 	TBool securityPassed = EFalse;
  1129 	//Do the security check
  1130 	TInt index = 0;
  1131 	while (!securityPassed)
  1132 		{
  1133 		aImplementationData.Reset();
  1134 		index = IndexedFind(aInterfaceUid);
  1135 		if(index == KErrNotFound)
  1136 			{
  1137 			User::Leave(KEComErrNoInterfaceIdentified);
  1138 			}
  1139 		User::LeaveIfError(index);
  1140 	
  1141 		TImplContainerArray& implementationList = iInterfaceImplIndex[index].iImpData;
  1142 		for (TInt j = 0; j < implementationList.Count(); j++)
  1143 			{
  1144 			CImplementationData* currentImplementation = implementationList[j].iCurrentImpl;
  1145 			CDllData* dll = currentImplementation->iParent->iParent;
  1146 			
  1147 			securityPassed = dll->ProcessSecurityCheckL();
  1148 			if(!securityPassed)
  1149 				{
  1150 				// remove the implementations of the DLL from iInterfaceImplIndex
  1151 				// and remove the DLL from its parent DLL list.
  1152 				DeleteDllL(dll);
  1153 				delete dll;
  1154 				dll = NULL;
  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
  1158 				break;	
  1159 				}
  1160 			else
  1161 				{
  1162 				User::LeaveIfError(aImplementationData.Append(currentImplementation));
  1163 				}	
  1164 			}
  1165 		}
  1166 	}
  1167 
  1168 /**
  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 
  1175 				by aState.
  1176 */
  1177 TInt CRegistryData::SetEnabledState(TUid aImplementationUid, TBool aState)
  1178 	{
  1179 	CImplementationData* impData = NULL;
  1180 	TUid dummy={0x00000000};
  1181 	
  1182 	(void)FindImplementation(aImplementationUid, dummy, impData);
  1183 
  1184 	if(impData != NULL)
  1185 		{
  1186 		impData->iImplInfo->SetDisabled((aState) ? EFalse : ETrue);
  1187 		return KErrNone;
  1188 		}
  1189 
  1190     return (aState) ? KEComErrEnableFailed : KEComErrDisableFailed;
  1191 	}
  1192 
  1193 
  1194 /**
  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.
  1202 */
  1203 void CRegistryData::TemporaryUninstallL(const TDriveUnit& aDrive)
  1204 	{
  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)
  1210 		{
  1211     	TInt dllIndex = drive->iDllList->Count();
  1212 		while(dllIndex > 0)
  1213 			{
  1214 			--dllIndex;
  1215 			RemoveFromIndexL((*drive->iDllList)[dllIndex]);
  1216 			}       
  1217 		iRegistrations->Remove(driveIndex);
  1218 		delete drive;
  1219 
  1220 		DriveChanged(aDrive, ETrue);
  1221 		}
  1222 	// The flag iRegistryChanged has been moved to each drive.
  1223 	}
  1224 
  1225 /**
  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.
  1235 */
  1236 void CRegistryData::UndoTemporaryUninstallL(const TDriveUnit& aDrive)
  1237 	{
  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)
  1245 		{
  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
  1251 
  1252 		DriveChanged(aDrive, EFalse);
  1253 
  1254 		// NB We DO need to set iRegistryChanged = ETrue because the idx file needs to be
  1255 		// rewritten to include the reinstalled drive
  1256 		}
  1257 	}
  1258 	
  1259 /** 
  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
  1265 */
  1266 void CRegistryData::DriveChanged(const TDriveUnit& aDrive, TBool aDriveRemoved)
  1267 	{
  1268 	if(aDriveRemoved)
  1269 		{
  1270 		iRemovedDrives |= (0x1 << aDrive);
  1271 		}
  1272 	else
  1273 		{
  1274 		iRemovedDrives &= ~(0x1 << aDrive);
  1275 		}
  1276 	}
  1277 
  1278 /** 
  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.
  1284 */
  1285 TBool CRegistryData::IsAnyDllRegisteredWithDriveL(const TDriveUnit& aDrive)const
  1286 	{
  1287 	CDriveData* driveData = NULL;
  1288 	if(FindDriveL(aDrive, driveData) == KErrNotFound)
  1289 		{
  1290 		User::Leave(KEComErrDriveNotFound);	
  1291 		}
  1292 	return driveData->iDllList->Count() > 0;
  1293 	}
  1294 	
  1295 /**
  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.
  1306 				EFalse otherwise.
  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
  1310 */
  1311 TBool CRegistryData::IsRegisteredWithDate (TUid aDllUid, 
  1312 										   	const TTime& aModified, 
  1313 										   	TBool& aUpdate, 
  1314 										   	CDriveData* aDriveData)
  1315 	{
  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)
  1321 		{
  1322 		TDll* dllList = aDriveData->iDllList;
  1323 		CDllData* dll = (*dllList)[index];
  1324 		const CEComEntry& dllEntry =*(dll->iDllEntry);	
  1325 		aUpdate = dllEntry.GetModified().Int64() < aModified.Int64(); 
  1326 		found = ETrue;
  1327 		}
  1328 	return found;
  1329 	}
  1330 
  1331 /**
  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
  1344 */
  1345 void CRegistryData::AddDllDataL(const TDriveUnit& aDrive, TInt aFoundDriveIndex, CRegistryData::CDllData* aDllData)
  1346 	{
  1347 	const TInt regCount = iRegistrations->Count();
  1348 	if(regCount == 0)
  1349 		User::Leave(KEComErrDriveNotFound);
  1350 
  1351 	// Find the appropriate drive entry
  1352 	CDriveData* drive = NULL;
  1353 	if(aFoundDriveIndex == KErrNotFound)
  1354 		{
  1355 		if(FindDriveL(aDrive, drive) == KErrNotFound)
  1356 			User::Leave(KEComErrDriveNotFound);
  1357 		}
  1358 	else
  1359 		drive = (*iRegistrations)[aFoundDriveIndex];
  1360 	
  1361 	// Append the new data to the list
  1362 	TDll* dllList = drive->iDllList;
  1363 	
  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++)
  1370 		{
  1371 		// For each interface structure
  1372 		CInterfaceData* interface = (*aDllData->iIfList)[ifListCounter];
  1373 		for(TInt impNum = 0; impNum < interface->iImplementations->Count(); ++impNum)
  1374 			{
  1375 			CImplementationData* implData = (*interface->iImplementations)[impNum];
  1376 			if(!InsertIntoIndexL(implData,checkNeeded))
  1377 				{
  1378 				interface->iImplementations->Remove(impNum); //remove from implementation array
  1379 				delete implData; //delete object
  1380 				}	
  1381 			}
  1382 		}
  1383 	//add filtered list for legitimate implementations into registry
  1384 	User::LeaveIfError(dllList->Append(aDllData)); 
  1385 		
  1386 	drive->iRegistryChanged = ETrue;	
  1387 	}
  1388 
  1389 /**
  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
  1399 */
  1400 void CRegistryData::UpdateDllDataL(const TDriveUnit& aDrive,TInt aFoundDriveIndex, CDllData* aDllData)
  1401 	{
  1402 	const TInt regCount = iRegistrations->Count();
  1403 	if(regCount == 0)
  1404 		User::Leave(KEComErrDriveNotFound);
  1405 
  1406 	// Find the appropriate drive entry
  1407 	CDriveData* drive = (*iRegistrations)[aFoundDriveIndex];
  1408 	
  1409 	TInt index = drive->FindDllIndex(aDllData->iDllEntry->GetThirdUid());
  1410 	
  1411 	// find the dll entry and clear that too
  1412 	if(index != KErrNotFound)
  1413 		{
  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);
  1422    		delete olddll;
  1423    		AddDllDataL(aDrive,aFoundDriveIndex,aDllData);
  1424    		drive->iRegistryChanged = ETrue;
  1425 
  1426 		}
  1427 	if (index == KErrNotFound)
  1428 	    {
  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));
  1431 	    }
  1432 	}
  1433 
  1434 /**
  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.
  1440 */
  1441 void CRegistryData::DiscoveriesBeginning() const
  1442 	{
  1443 	iCurrentlyDiscovering = ETrue;
  1444 	// The registry is about to start changing so the index is now invalid
  1445 	}
  1446 
  1447 /**
  1448 The cue to rebuild the registry indexes
  1449 @leave			KErrNoMemory
  1450 @see 			CStore
  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
  1458 */
  1459 void CRegistryData::DiscoveriesCompleteL(TBool aSuccessful, TPluginProcessingTypeIdentifier aProcessingType, TBool& aHasRegistryChanged)
  1460 	{
  1461 	iCurrentlyDiscovering = EFalse;
  1462 	if (!aSuccessful)
  1463 		{
  1464 		return;
  1465 		}
  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.
  1470 	// The call stack: 
  1471 	// ~CEComServer delete iRegistrar, ~CRegistrar delete iDiscoverer,
  1472 	// ~CDiscoverer calls CRegistrar::DiscoveriesComplete,
  1473 	// then here.
  1474     // Hence if ! aSuccessful do not validate.
  1475 
  1476 	if(aProcessingType != EPluginProcessingTypeCriticalOnly)
  1477 		{
  1478 		ValidateRegistryL();
  1479 		}
  1480 	if(aProcessingType==EPluginProcessingTypeNonCriticalOnly || aProcessingType==EPluginProcessingTypeAll)
  1481 		{
  1482 		//first find whether there is any registry data changed
  1483 		TInt regCount=iRegistrations->Count();
  1484 		for(TInt i = 0; i <regCount; i++)
  1485 			{
  1486 			if ((*iRegistrations)[i]->iRegistryChanged)
  1487 				{
  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;
  1493 				}
  1494 			}
  1495 			//Also indicate registry change if any drives have been removed
  1496 			aHasRegistryChanged |= (iRemovedDrives != 0);
  1497 			iRemovedDrives = 0;
  1498 		}
  1499 	}
  1500 
  1501 /**
  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.				 
  1508 */
  1509 
  1510 void CRegistryData::SetDiscoveryFlagL(const TDriveUnit& aDriveUnit)
  1511 	{
  1512 	CDriveData* drive = NULL;
  1513 	TInt driveIndex = FindDriveL(aDriveUnit, drive);
  1514 	if(driveIndex != KErrNotFound)
  1515 		{
  1516 		drive->iDriveChanged = ETrue;
  1517 		}
  1518 	
  1519 	}
  1520 /**
  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
  1527 */
  1528 TBool CRegistryData::IndexValid() const
  1529 	{
  1530 	return !iCurrentlyDiscovering;
  1531 	}
  1532 
  1533 /**
  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,
  1540 				NULL otherwise.
  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
  1547 */
  1548 TInt CRegistryData::GetImplementationDllInfoForServer(
  1549 	const TCapabilitySet& /*aCapabilitySet*/,
  1550 	const TUid aImplUid,
  1551 	const TUid aInterfaceUid,	
  1552 	TEntry& aEntry,
  1553 	CImplementationInformation*& aImplInfo,
  1554 	TBool& aIsOnRWDrive) const
  1555 	{
  1556 	aImplInfo = NULL;
  1557 	CImplementationData* implData;
  1558 	
  1559 	TInt res = FindImplementation(aImplUid, aInterfaceUid, implData);
  1560 	if (KErrNone != res)
  1561 		{
  1562 		return res;
  1563 		}
  1564 	
  1565 	aImplInfo = implData->iImplInfo;
  1566 	const CDllData* dll = implData->iParent->iParent;
  1567 	dll->PopulateAnEntry(aEntry);
  1568 
  1569 	TEComCachedDriveInfoIterator iter(*iCachedDriveInfo);
  1570 	if (! iter.SetPos(dll->iParent->iDrive))
  1571 		{
  1572 		res = KErrNotFound;
  1573 		}
  1574 	else
  1575 		{
  1576 		aIsOnRWDrive = iter.DriveIsWritable();
  1577 		}
  1578 	return res;
  1579 	}
  1580 
  1581 /**
  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,
  1588 				NULL otherwise.
  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
  1593 */
  1594 TInt CRegistryData::GetImplementationDllInfoForClientL(
  1595 	const TClientRequest& aClientRequest,
  1596 	const TUid aImplUid,
  1597 	const TUid aInterfaceUid,	
  1598 	TEntry& aEntry,
  1599 	CImplementationInformation*& aImplInfo,
  1600 	TBool aSecurityCheckNeeded)const
  1601 	{
  1602 	aImplInfo = NULL;
  1603 	TInt res = KErrNotFound;
  1604 	CImplementationData* implData;
  1605 	if (!aSecurityCheckNeeded)
  1606 		{
  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)
  1610 			{
  1611 			return res;
  1612 			}
  1613 		CDllData* dll = implData->iParent->iParent;
  1614 		dll->PopulateAnEntry(aEntry);
  1615 		aImplInfo = implData->iImplInfo;
  1616 		}
  1617 	else
  1618 		{
  1619 		TBool securityPassed = EFalse;
  1620 		while(!securityPassed) // go out of this loop either Security check is passed or no DLL found.
  1621 			{
  1622 			res = FindImplementation(aImplUid, aInterfaceUid, implData);
  1623 			if (KErrNone != res)
  1624 				{
  1625 				return res;
  1626 				}
  1627 			CDllData* dll = implData->iParent->iParent;
  1628 			// security check is deferred to here.
  1629 			securityPassed = dll->ProcessSecurityCheckL();
  1630 			if(securityPassed)
  1631 				{
  1632 				if (!aClientRequest.CheckCapability(dll->iCapSet, *(implData->iImplInfo)))
  1633 					{
  1634 					return KErrPermissionDenied;
  1635 					}
  1636 				dll->PopulateAnEntry(aEntry);
  1637 				aImplInfo = implData->iImplInfo;
  1638 				}
  1639 			else
  1640 				{
  1641 				// remove the implementations of the DLL from iInterfaceImplIndex
  1642 				// and remove the DLL from its parent DLL list.
  1643 				DeleteDllL(dll);
  1644 				delete dll;
  1645 				dll = NULL;
  1646 				}
  1647 			}
  1648 		}
  1649 	return res;
  1650 	}
  1651 
  1652 /**
  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	
  1657 */
  1658 void CRegistryData::DeleteDllL(CDllData* aDllData) const
  1659 	{
  1660 	// find the index of the passed aDllData in its parent's Dll list.
  1661 	TInt index = aDllData->iParent->FindDllIndex(aDllData->iDllEntry->GetThirdUid());
  1662 	
  1663 	//removes all implementations of This Dll from the iInterfaceImplIndex
  1664 	RemoveFromIndexL(aDllData);
  1665 	if(index != KErrNotFound)
  1666 		{
  1667 		//now remove the dll from the array and registry
  1668 		TDll* dllList = aDllData->iParent->iDllList;
  1669 		dllList->Remove(index);
  1670 		}
  1671 	}
  1672 
  1673 
  1674 /**
  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
  1679 */
  1680 TBool CRegistryData::HasLanguageChanged() const
  1681 	{
  1682 	return iLanguageChanged;
  1683 	}
  1684 
  1685 	
  1686 /**
  1687 @param 			aFs A handle to a connected file server.
  1688 */
  1689 CRegistryData::CRegistryData(RFs& aFs,
  1690 							 TInt aInterfaceImplIndexGranularity,
  1691 							 TInt aImplIndexGranularity) :
  1692 	CBase(), 
  1693 	iFs(aFs),
  1694 	iInterfaceImplIndex(aInterfaceImplIndexGranularity),
  1695 	iImplIndex(aImplIndexGranularity)
  1696 	{
  1697 	}
  1698 
  1699 
  1700 /**
  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
  1705 */
  1706 void CRegistryData::ConstructL()
  1707 	{
  1708 	iCachedDriveInfo = CEComCachedDriveInfo::NewL(iFs);
  1709 
  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)
  1717 		{
  1718 		User::LeaveNoMemory();
  1719 		}
  1720 	}
  1721 
  1722 /**
  1723 To find the index entry for aImplementationUid.
  1724 @return			The index of the item or KErrIndexEntryNotFound. 
  1725 @pre 			This object is fully constructed
  1726 */
  1727 TInt CRegistryData::IndexedFind(TUid aInterfaceUid) const
  1728 	{
  1729 	// Find the correct implementation
  1730 	TInterfaceIndex key;
  1731 	key.iInterfaceUid = aInterfaceUid;
  1732 	return iInterfaceImplIndex.FindInOrder(key,TLinearOrder<TInterfaceIndex>(TInterfaceStruct::CompareInfUid));
  1733 	}
  1734 
  1735 /**
  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
  1740 */
  1741 TBool CRegistryData::MatchOnDrive(const CRegistryData::CDriveData& aIndexOne, 
  1742 								  const CRegistryData::CDriveData& aIndexTwo)
  1743 	{
  1744 	return aIndexOne.iDrive == aIndexTwo.iDrive;
  1745 	}
  1746 
  1747 TInt CRegistryData::FindDriveL(const TDriveUnit& aDrive,
  1748 							   CRegistryData::CDriveData*& aDriveData)const 
  1749 	{
  1750 	TInt index = KErrNotFound;
  1751 
  1752 	// Set up the find parameters
  1753 	TIdentityRelation<CRegistryData::CDriveData> identity(MatchOnDrive);
  1754     CDriveData* driveToMatch = CDriveData::NewLC(aDrive,const_cast<CRegistryData*>(this));
  1755 
  1756 	index = iRegistrations->Find(driveToMatch, identity);
  1757 	
  1758 	CleanupStack::PopAndDestroy(driveToMatch);
  1759 
  1760 	if(index != KErrNotFound)
  1761 		aDriveData = (*iRegistrations)[index];
  1762 	else
  1763 		aDriveData = NULL;
  1764 
  1765 	return index;
  1766 	}
  1767 
  1768 /**
  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
  1773 				0 if uid is unknown
  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.
  1776 */
  1777 TInt CRegistryData::FindImplementation(const TUid aImplUid, const TUid aInterfaceUid, 
  1778 												CImplementationData*& aImplData) const
  1779 	{
  1780 	aImplData = NULL;
  1781 	TInt i;
  1782 	// Index is kept up to date even when discoveries are occurring
  1783 	// therefore always search index for implementation
  1784 
  1785 	// if aInterfaceUid is non zero use it to find position in index list
  1786 	if(aInterfaceUid.iUid != 0)
  1787 		{
  1788 		i = IndexedFind(aInterfaceUid);
  1789 		if(i == KErrNotFound)
  1790 			{
  1791 			return KErrNotFound;
  1792 			}
  1793 		
  1794 		TInterfaceIndex listItem = iInterfaceImplIndex[i];
  1795 		i = listItem.iImpData.FindInOrder(aImplUid,
  1796 			TImplStruct::CompareUidAgainstImplStruct);
  1797 		if (i >= 0)
  1798 			{
  1799 			aImplData = listItem.iImpData[i].iCurrentImpl;
  1800 			return KErrNone;
  1801 			}
  1802 		}
  1803 	else // not given I/F UID. Use the iImplIndex.
  1804 		{
  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.)
  1812 
  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)
  1826 			{
  1827 			return KErrNotFound;
  1828 			}
  1829 
  1830 		aImplData = iImplIndex[i];
  1831 
  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
  1839 
  1840 		TInt count = iImplIndex.Count();
  1841 		for (TInt j = i + 1; j < count; j++)
  1842 			{
  1843 			if ( iImplIndex[j]->iImplInfo->ImplementationUid().iUid != aImplUid.iUid )
  1844 				{
  1845 				break;
  1846 				}
  1847 
  1848 			aImplData = SelectDuplicatedImpl(aImplData, iImplIndex[j]);
  1849 			}
  1850 
  1851 		return KErrNone;
  1852 		}
  1853 		
  1854 	return KErrNotFound;
  1855 	}
  1856 
  1857 /**
  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)
  1864 
  1865 @param aImpl1 - one of the two implementations to compare.
  1866 @param aImpl2 - the other implementation to compare.
  1867 @return the preferred implementation.
  1868 */
  1869 CRegistryData::CImplementationData* CRegistryData::SelectDuplicatedImpl(
  1870 	const CImplementationData* aImpl1, 
  1871 	const CImplementationData* aImpl2) const
  1872 	{
  1873 
  1874 #ifdef ECOM_TRACE
  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();
  1879 #endif
  1880 
  1881 	const TInt KRomBasedFactor = 0x100;
  1882 
  1883 	TInt drive1 = aImpl1->iImplInfo->Drive();
  1884 	if (aImpl1->iImplInfo->RomBased())
  1885 		{
  1886 		drive1 |= KRomBasedFactor;
  1887 		}
  1888 
  1889 	TInt drive2 = aImpl2->iImplInfo->Drive();
  1890 	if (aImpl2->iImplInfo->RomBased())
  1891 		{
  1892 		drive2 |= KRomBasedFactor;
  1893 		}
  1894 
  1895 	if (drive1 > drive2)
  1896 		{
  1897 #ifdef ECOM_TRACE
  1898 		if ((drive1 & KRomBasedFactor) && !(drive2 & KRomBasedFactor))
  1899 			{
  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);
  1901 			}
  1902 		else
  1903 			{
  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);
  1905 			}
  1906 #endif
  1907 
  1908 		return const_cast<CImplementationData*>(aImpl1);
  1909 		}
  1910 	else if (drive1 < drive2)
  1911 		{
  1912 #ifdef ECOM_TRACE
  1913 		if ((drive2 & KRomBasedFactor) && !(drive1 & KRomBasedFactor))
  1914 			{
  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);
  1916 			}
  1917 		else
  1918 			{
  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);
  1920 			}
  1921 #endif
  1922 
  1923 		return const_cast<CImplementationData*>(aImpl2);
  1924 		}
  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) 
  1927 		{
  1928 #ifdef ECOM_TRACE
  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);
  1930 #endif
  1931 
  1932 		return const_cast<CImplementationData*>(aImpl2);
  1933 		}
  1934 
  1935 #ifdef ECOM_TRACE
  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);
  1937 #endif
  1938 
  1939 	return const_cast<CImplementationData*>(aImpl1);
  1940 	}
  1941 
  1942 /**
  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
  1946 it is also removed.
  1947 @pre 			This object is fully constructed.
  1948 @post			Any out of date registry entries are removed from the tree.
  1949 */
  1950 void CRegistryData::ValidateRegistryL()
  1951 	{
  1952 	TInt driveIndex = iRegistrations->Count();
  1953 	TFileName rscFileName;
  1954 	while(driveIndex > 0)
  1955 		{
  1956 		--driveIndex;
  1957 		CDriveData* drive = (*iRegistrations)[driveIndex];
  1958 		
  1959 		if( !iCachedDriveInfo->DriveIsReadOnlyInternalL(drive->iDrive) &&
  1960 			drive->iDriveChanged)
  1961 			{
  1962 			TInt dllIndex = drive->iDllList->Count();	
  1963 			while(dllIndex > 0)
  1964 				{
  1965 				--dllIndex;
  1966 				CDllData* dll = (*drive->iDllList)[dllIndex];
  1967 				//reset the buffer first			
  1968 				rscFileName.Zero();
  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)
  1973 					{
  1974 					rscFileName.Append(dll->iRscFileExtension->Des());
  1975 					}
  1976 			
  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);
  1980 
  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)
  1986 					{
  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;
  1992 					delete dll;
  1993 					}
  1994 				}
  1995 			}
  1996 		}
  1997 	}
  1998 
  1999 /**
  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.
  2007 */
  2008 CRegistryData::CImplementationData* CRegistryData::SelectPreferredImplL(CImplementationData* aOldImpl,
  2009 																CImplementationData* aNewImpl,
  2010 																TBool& aLegitimateImpl,
  2011 																TBool aCheckIsNeeded) const
  2012 	{
  2013 	aLegitimateImpl = ETrue;
  2014 
  2015 	TBool newIsRomOnly = aNewImpl->iImplInfo->RomOnly();
  2016 	TBool oldIsRomOnly = aOldImpl->iImplInfo->RomOnly();
  2017 		
  2018 	/* 	In addition to selecting the highest version of an 
  2019 		implementation this check takes care of the following special 
  2020 		cases:
  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.
  2026 	*/
  2027 	if(newIsRomOnly && !oldIsRomOnly)
  2028 		{
  2029 		return aNewImpl;
  2030 		}
  2031 	else if((newIsRomOnly && oldIsRomOnly))
  2032 		{
  2033 		if(aNewImpl->iImplInfo->Version() > aOldImpl->iImplInfo->Version())
  2034 			{
  2035 			return aNewImpl;
  2036 			}
  2037 		else if(aNewImpl->iImplInfo->Version() == aOldImpl->iImplInfo->Version())
  2038 			{
  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()))
  2043                 {
  2044 				return aNewImpl;
  2045 				}
  2046 			}
  2047 		}
  2048 	//all new implementations which are not flagged'RomOnly'
  2049 	else if(!newIsRomOnly && !oldIsRomOnly)
  2050 		{
  2051 		TBool newIsReadOnly = iCachedDriveInfo->DriveIsReadOnlyInternalL(aNewImpl->iParent->iParent->iParent->iDrive);
  2052 		TBool oldIsReadOnly = iCachedDriveInfo->DriveIsReadOnlyInternalL(aOldImpl->iParent->iParent->iParent->iDrive);
  2053 			
  2054 		if(aNewImpl->iImplInfo->Version() > aOldImpl->iImplInfo->Version())
  2055 			{
  2056 			//filter for implementations on R/W drive trying to upgrade implementations on R/O drive
  2057 			if((!newIsReadOnly && oldIsReadOnly) && aCheckIsNeeded)
  2058 				{						
  2059 				//test if a later implementation version on R/W drive has the same filename as the current one before upgrading.
  2060 				//If not,...
  2061 				if(aOldImpl->iParent->iParent->iDllEntry->GetName().CompareF(aNewImpl->iParent->iParent->iDllEntry->GetName()) == 0)
  2062 					{
  2063 					return aNewImpl;
  2064 					}
  2065 				else
  2066 					{
  2067 					aLegitimateImpl = EFalse; //not secure
  2068 					return aOldImpl;
  2069 					}
  2070 				}
  2071 			else
  2072 				{
  2073 				return aNewImpl;
  2074 				}
  2075 			}
  2076 		else if(aNewImpl->iImplInfo->Version() == aOldImpl->iImplInfo->Version())
  2077 			{
  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()))                 
  2082 				{
  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)
  2086 					{							
  2087 					//test if a later implementation version on R/W drive has the same filename as the current one before upgrading.
  2088 					//If not,...
  2089 					if(aOldImpl->iParent->iParent->iDllEntry->GetName().CompareF(aNewImpl->iParent->iParent->iDllEntry->GetName()) == 0)
  2090 						{
  2091 						return aNewImpl;
  2092 						}
  2093 					else
  2094 						{
  2095 						aLegitimateImpl = EFalse; //not secure
  2096 						return aOldImpl;
  2097 						}		
  2098 					}
  2099 				else
  2100 					{
  2101 					return aNewImpl;
  2102 					}
  2103 				}						
  2104 			}
  2105 		}
  2106 	return aOldImpl;
  2107 	}
  2108 
  2109 /**
  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.
  2121 @post   none
  2122 */
  2123 void
  2124 CRegistryData::FilterForLatestLegitimateImplL(TImplContainerArray& aIdxArray,
  2125 											  CImplementationData* aNewImpl,
  2126 											  TInsertImplMode& aInsertMode,
  2127 											  TInt&  aPosOfImplInArray,
  2128 											  TBool& aLegitimateImpl,
  2129 											  TBool aCheckIsNeeded)
  2130 	{
  2131 	aInsertMode = EInsertImplUndefinedMode;
  2132 	aLegitimateImpl = ETrue;
  2133 	TImplStruct newImplStruct;
  2134 	newImplStruct.iCurrentImpl = aNewImpl;
  2135 
  2136 	TInt idxPos = aIdxArray.FindInOrder(newImplStruct, TLinearOrder<TImplStruct> (TImplStruct::CompareImplStructUid));
  2137 
  2138 	aPosOfImplInArray = idxPos;
  2139 
  2140 	if(idxPos != KErrNotFound)
  2141 		{
  2142 		if(aNewImpl != SelectPreferredImplL(aIdxArray[idxPos].iCurrentImpl,
  2143 											aNewImpl,
  2144 											aLegitimateImpl,
  2145 											aCheckIsNeeded))
  2146 			{
  2147 			aInsertMode = EInsertImplAsUnusedImpl;
  2148 			}
  2149 		else
  2150 			{
  2151 			aInsertMode = EInsertImplAsUpgradeOfExistingImpl;
  2152 			} 
  2153 		}
  2154 	else
  2155 		{
  2156 		aInsertMode = EInsertImplAsNewcomerOfInterface;
  2157 		}
  2158 	}
  2159 
  2160 void CRegistryData::ResetTInterfaceIndex(TAny* aObject)
  2161 	{
  2162 	TInterfaceIndex* index=reinterpret_cast<TInterfaceIndex*>(aObject);
  2163 	index->Reset();
  2164 	}
  2165 void CRegistryData::RemoveImplFromImplIndexCleanUp(TAny* aPtr)
  2166     {
  2167     TCleanupImplIndexEntry *aCleanup = (TCleanupImplIndexEntry*)aPtr;
  2168     aCleanup->iRegistryData->RemoveImplByAddrFromImplIndex(aCleanup->iImplEntry);
  2169     }
  2170 
  2171 /**
  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.
  2178 */
  2179 TBool CRegistryData::InsertIntoIndexL(CImplementationData* aImplPtr, TBool aCheckIsNeeded)
  2180 	{
  2181 	TBool legitimateImpl = ETrue;
  2182 
  2183 	TInterfaceIndex newIndexEl;
  2184 	TInterfaceIndex* newElPtr;
  2185 	//initialise
  2186 	newIndexEl.iInterfaceUid = aImplPtr->iParent->iInterfaceUid;
  2187 	CleanupStack::PushL(TCleanupItem(ResetTInterfaceIndex,&newIndexEl));
  2188 	
  2189 	TInt indexPos = iInterfaceImplIndex.FindInOrder(newIndexEl,TLinearOrder<TInterfaceIndex>(TInterfaceStruct::CompareInfUid));
  2190 
  2191 	
  2192 	if(indexPos!=KErrNotFound)
  2193 		newElPtr = &(iInterfaceImplIndex[indexPos]);
  2194 	else		
  2195 		newElPtr = &newIndexEl;	
  2196 
  2197 	// For each implementation make sure we only have the latest version
  2198 	TInt implIdxInContainerArray(KErrNotFound);
  2199 	TInsertImplMode insertMode;
  2200 	FilterForLatestLegitimateImplL(newElPtr->iImpData,
  2201 									aImplPtr,
  2202 									insertMode,
  2203 									implIdxInContainerArray,
  2204 									legitimateImpl,
  2205 									aCheckIsNeeded);
  2206 	
  2207 	InsertImplInIndexesL(insertMode, 
  2208 						 indexPos,
  2209 						 *newElPtr,
  2210 						 implIdxInContainerArray,
  2211 						 aImplPtr,
  2212 						 legitimateImpl);
  2213 
  2214 	CleanupStack::Pop();
  2215 
  2216 	if (insertMode == EInsertImplAsUpgradeOfExistingImpl)
  2217 		{
  2218 	    TUid ImplUid = aImplPtr->iImplInfo->ImplementationUid();
  2219 		iImplUpgradeCallBack.CallBack(ECallBackId_ImplUpgrade, &ImplUid);
  2220 		}
  2221 
  2222 	return legitimateImpl;
  2223 	}
  2224 
  2225 /** Handle inserting a CImplementationData in iInterfaceImplIndex and
  2226 iImplIndex.
  2227 
  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.
  2242 */
  2243 void CRegistryData::InsertImplInIndexesL(TInsertImplMode aInsertMode,
  2244 										TInt aIfPosInInterfaceImplIndex,
  2245 										TInterfaceIndex& aNewIfIndexEl,
  2246 										TInt aImplPosInContainerArray,
  2247 								   		CImplementationData* aNewImpl,
  2248 								   		TBool aLegitimateImpl)
  2249 	{
  2250 	if(aInsertMode == EInsertImplUndefinedMode)
  2251 		{
  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);
  2256 		}
  2257 
  2258 	// If not a legitimate implementation, aNewImpl will be deleted.
  2259 	// Do not add it to the two indexes.
  2260 
  2261 	if (!aLegitimateImpl)
  2262 		{
  2263 		return;
  2264 		}
  2265 
  2266 	TImplContainerArray& implContainerArray = aNewIfIndexEl.iImpData;
  2267 
  2268 	if(aInsertMode == EInsertImplAsNewcomerOfInterface)
  2269 		{
  2270 		TImplStruct newImplStruct;
  2271 		newImplStruct.iCurrentImpl = aNewImpl;
  2272 		implContainerArray.InsertInOrderL(newImplStruct, TLinearOrder<TImplStruct> (TImplStruct::CompareImplStructUid));
  2273 
  2274 #ifdef _DEBUG
  2275 		// on debug build, check for duplicated implementation UID
  2276 		// owned by different interfaces.
  2277 
  2278 		TInt ii = iImplIndex.SpecificFindInOrder(
  2279 			aNewImpl,
  2280 			CImplementationData::CompareImplUidIgnoreIfUid,
  2281 			EArrayFindMode_Any);
  2282 		if (ii != KErrNotFound)
  2283 			{
  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);
  2291 			}
  2292 #endif
  2293 		}
  2294 	else if(aInsertMode == EInsertImplAsUpgradeOfExistingImpl)
  2295 		{
  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;
  2301 
  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())
  2306 			{
  2307 			aNewImpl->iImplInfo->SetRomBased(ETrue);
  2308 			}
  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()));
  2311 		}
  2312 	else if (aInsertMode == EInsertImplAsUnusedImpl)
  2313 		{
  2314 		TImplContainer& implContainer = implContainerArray[aImplPosInContainerArray];
  2315 		implContainer.iUnusedImpls.AppendL(aNewImpl);
  2316 
  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())
  2325 			{
  2326 			currImpl->iImplInfo->SetRomBased(ETrue);
  2327 			}
  2328 
  2329 		return; // the trailing steps not applicable to downgrade situation
  2330 		}
  2331 
  2332 	User::LeaveIfError( InsertImplIntoImplIndex(aNewImpl) );
  2333 	
  2334 	if(aIfPosInInterfaceImplIndex==KErrNotFound)
  2335 		{		
  2336         TCleanupImplIndexEntry aCleanup(this, aNewImpl);
  2337         CleanupStack::PushL(TCleanupItem(RemoveImplFromImplIndexCleanUp,&aCleanup));
  2338 		iInterfaceImplIndex.InsertInOrderL(aNewIfIndexEl, TLinearOrder<TInterfaceIndex>(TInterfaceStruct::CompareInfUid));
  2339 		CleanupStack::Pop();
  2340 		}
  2341 	}
  2342 
  2343 /**
  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
  2348 */
  2349 void CRegistryData::RemoveFromIndexL(CDllData* aDllData) const
  2350 	{
  2351 	//remove 'old' implementations from iInterfaceImplIndex
  2352 	TInt counter=0;
  2353 	TImplStruct implStruct;
  2354 	TInt ifListCount=aDllData->iIfList->Count();
  2355 	while(counter < ifListCount)
  2356 		{
  2357 		CInterfaceData* interface = (*aDllData->iIfList)[counter];
  2358 		TInterfaceIndex index;
  2359 		index.iInterfaceUid = interface->iInterfaceUid;
  2360 		
  2361 		TInt indexPos = iInterfaceImplIndex.FindInOrder(index,TLinearOrder<TInterfaceIndex>(TInterfaceStruct::CompareInfUid));
  2362         TImplContainerArray* impDataArray = NULL;
  2363 		TInt implCount=0;
  2364 		
  2365 		if(indexPos!=KErrNotFound)
  2366 			{
  2367             impDataArray = &(iInterfaceImplIndex[indexPos].iImpData);
  2368             implCount=interface->iImplementations->Count();
  2369 			for(TInt i=0; i<implCount; i++)
  2370 				{
  2371 				implStruct.iCurrentImpl = (*interface->iImplementations)[i];
  2372 				TInt impPos = impDataArray->FindInOrder(implStruct, TLinearOrder<CRegistryData::TImplStruct> (TImplStruct::CompareImplStructUid));
  2373 				if(impPos!=KErrNotFound)
  2374 					{
  2375 					TImplContainer& implContainer = (*impDataArray)[impPos];
  2376 					// remove from unused list if exists
  2377 					TInt count = implContainer.iUnusedImpls.Count();
  2378 					while(count > 0)
  2379 						{
  2380 						--count;
  2381 						if(implContainer.iUnusedImpls[count]->iParent->iParent == aDllData)
  2382 							{
  2383 							implContainer.iUnusedImpls.Remove(count);
  2384 							}
  2385 						}
  2386 
  2387 					// update current entry
  2388 					if(implContainer.iCurrentImpl->iParent->iParent == aDllData)
  2389 						{
  2390 						// do not care about the return code.
  2391 						RemoveImplByAddrFromImplIndex(implContainer.iCurrentImpl);
  2392 						
  2393 						TInt implContainerUnusedImplCount=implContainer.iUnusedImpls.Count();
  2394 						// no unused impl's therefore no rollback and remove entry
  2395 						if(implContainerUnusedImplCount == 0)
  2396 							{
  2397 							implContainer.Reset();
  2398 							impDataArray->Remove(impPos);
  2399 							}
  2400 						else
  2401 							{
  2402 							// Rollback to implementation from unused array
  2403 							// copy first from unused array
  2404 							TInt selectedPos = 0;
  2405 							implContainer.iCurrentImpl = implContainer.iUnusedImpls[selectedPos];
  2406 							
  2407 							// now check if any others in list should be preferred
  2408 							for(count = 1;count < implContainerUnusedImplCount; count++)
  2409 								{
  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],
  2416 																dummLegitimateImpl,
  2417 																EFalse);
  2418 								if(implContainer.iCurrentImpl == implContainer.iUnusedImpls[count])
  2419 									{
  2420 									selectedPos = count;
  2421 									}
  2422 								}
  2423 							implContainer.iUnusedImpls.Remove(selectedPos);
  2424 							User::LeaveIfError(InsertImplIntoImplIndex(implContainer.iCurrentImpl));
  2425 #ifdef ECOM_TRACE
  2426 							{
  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()));
  2429 							}
  2430 #endif
  2431 							}
  2432 						}
  2433 					}
  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
  2439 				// impl UID.
  2440 				RemoveImplByAddrFromImplIndex((*interface->iImplementations)[i]);
  2441 				}
  2442 			if(impDataArray->Count() == 0)
  2443 				{
  2444 				iInterfaceImplIndex[indexPos].Reset();
  2445 				iInterfaceImplIndex.Remove(indexPos);
  2446 				}
  2447 			}
  2448 		counter++;
  2449 		}
  2450 	}
  2451 
  2452 /**
  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.
  2457 */
  2458 void CRegistryData::AddImplDataL(CDriveData* aDriveData)
  2459 	{
  2460  	TBool checkIsNeeded = EFalse;
  2461  	TInt dllCount=aDriveData->iDllList->Count();
  2462  	for(TInt dllCounter=0; dllCounter < dllCount;dllCounter++)
  2463  		{
  2464  		CDllData* aDllData = (*aDriveData->iDllList)[dllCounter];
  2465  		TInt interfaceCount=aDllData->iIfList->Count();
  2466  		for(TInt ifListCounter=0; ifListCounter < interfaceCount; ifListCounter++)
  2467  			{
  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)
  2472  				{
  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
  2478  				//KErrNone
  2479  				}
  2480  			}	
  2481  		}	
  2482  	}
  2483 
  2484 
  2485 /** This method does a search by address and removes the specified entry from iImplIndex.
  2486 
  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.
  2490 */
  2491 TBool CRegistryData::RemoveImplByAddrFromImplIndex(CImplementationData* aPtr) const
  2492     {
  2493     TInt aIdx = iImplIndex.Find(aPtr);
  2494     if (aIdx != KErrNotFound)
  2495         {
  2496         // This linear pointer search ensures a safe removal of the impl from iImplIndex so that it is free from a dead object.
  2497     
  2498         // The array does not own the pointer. Do not delete!
  2499         iImplIndex.Remove(aIdx);
  2500         return ETrue;
  2501         }
  2502     return EFalse;
  2503     }
  2504 
  2505 /** This method inserts the entry aNewImpl into iImplIndex.
  2506 
  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.
  2514 */
  2515 TInt CRegistryData::InsertImplIntoImplIndex(CImplementationData* aNewImpl) const
  2516 	{
  2517 	TLinearOrder<CImplementationData> ImplUidComparator(CImplementationData::CompareImplUid);
  2518 
  2519 	return iImplIndex.InsertInOrder(aNewImpl, ImplUidComparator);
  2520 	}
  2521 
  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.
  2524 
  2525 @param aLanguageChanged the returned value to indicate language changed.
  2526 */
  2527 void CRegistryData::LanguageChangedL(TBool& aLanguageChanged)
  2528 	{
  2529 	iLanguageChanged = RDowngradePath::HasChangedL(iFs);
  2530 	aLanguageChanged = iLanguageChanged;
  2531 	}
  2532 
  2533 /** setter
  2534 If need to unset the callback, use a TCallBackWithArg constructed with no
  2535 arguments.
  2536 */
  2537 void CRegistryData::SetImplUpgradeCallBack(const TCallBackWithArg& aCallBack)
  2538 	{
  2539 	iImplUpgradeCallBack = aCallBack;
  2540 	}
  2541 
  2542 #ifdef __ECOM_SERVER_PERFORMANCE__
  2543 /**
  2544 This method calculates the drive, plugins, interfaces, implementations counts for the drive
  2545 type set
  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
  2549 */
  2550 void CRegistryData::GetRegistryCountsL(TInt aType, RegistryCounts::TRegistryCounts& aCounts) const
  2551 	{
  2552 	aCounts.iDrives = 0;
  2553 	aCounts.iImplementations = 0;
  2554 	aCounts.iInterfaces = 0;
  2555 	aCounts.iDlls = 0;
  2556 	
  2557 	for(TInt driveIndex = 0; driveIndex< iRegistrations->Count(); driveIndex++)
  2558 		{
  2559 		CDriveData* drive = (*iRegistrations)[driveIndex];
  2560 		TBool isReadOnly = iCachedDriveInfo->DriveIsReadOnlyInternalL(drive->iDrive);
  2561 
  2562 		if((aType == RegistryCounts::TRegistryCounts::EAll) ||
  2563 			(aType == RegistryCounts::TRegistryCounts::ERoInternal && isReadOnly) ||
  2564 			(aType == RegistryCounts::TRegistryCounts::ENonRoInternal && !isReadOnly))
  2565 			{
  2566 			aCounts.iDrives++;
  2567 			aCounts.iDlls += drive->iDllList->Count();
  2568 			for(TInt dllIndex = 0; dllIndex < drive->iDllList->Count(); dllIndex++)
  2569 				{
  2570 				CDllData* dllList = (*drive->iDllList)[dllIndex];
  2571 				aCounts.iInterfaces += dllList->iIfList->Count();
  2572 				for(TInt ifIndex = 0; ifIndex < dllList->iIfList->Count(); ifIndex++)
  2573 					{
  2574 					CInterfaceData* ifList = (*dllList->iIfList)[ifIndex];
  2575 					aCounts.iImplementations += ifList->iImplementations->Count();
  2576 					}
  2577 				}
  2578 			}
  2579 		}
  2580 	}
  2581 
  2582 #endif // __ECOM_SERVER_PERFORMANCE__