os/ossrv/lowlevellibsandfws/pluginfw/Framework/frame/RegistryData.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
// Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
// This file contains the implementations of the classes
sl@0
    15
// which manage the internal structure of the registry.
sl@0
    16
// 
sl@0
    17
//
sl@0
    18
sl@0
    19
#include <e32base.h>
sl@0
    20
#include <e32uid.h>
sl@0
    21
#include <s32stor.h>
sl@0
    22
#include <s32file.h>
sl@0
    23
#include <startup.hrh> // for EStartupStateNonCritical
sl@0
    24
#include <e32ldr.h> // for hash checking
sl@0
    25
#include <e32ldr_private.h> // for RLoader
sl@0
    26
#include <bautils.h> // for BaflUtils::FileExists
sl@0
    27
sl@0
    28
#include "EComDebug.h"
sl@0
    29
#include <ecom/ecom.h>
sl@0
    30
#include <ecom/ecomerrorcodes.h>
sl@0
    31
#include <ecom/ecomresolverparams.h>
sl@0
    32
#include <ecom/implementationinformation.h>
sl@0
    33
#include "RegistryData.h"
sl@0
    34
#include "DowngradePath.h"
sl@0
    35
#include "DriveInfo.h"
sl@0
    36
#include "FileUtils.h"
sl@0
    37
#include "EComUidCodes.h"
sl@0
    38
#include "EComInternalErrorCodes.h"
sl@0
    39
#define UNUSED_VAR(a) a = a
sl@0
    40
sl@0
    41
const TInt KRegVersion = -9999;
sl@0
    42
const TInt KDllExtensionLength=4;
sl@0
    43
_LIT(KDllExtension,".dll");
sl@0
    44
//
sl@0
    45
//CRegistryData::TInterfaceStruct
sl@0
    46
sl@0
    47
/**
sl@0
    48
Constructor for TInterfaceStruct
sl@0
    49
@post			Its member variable is granulated as (1)
sl@0
    50
*/
sl@0
    51
CRegistryData::TInterfaceStruct::TInterfaceStruct():
sl@0
    52
	iImpData(1)
sl@0
    53
	{
sl@0
    54
	// Do nothing here
sl@0
    55
	}
sl@0
    56
sl@0
    57
/**
sl@0
    58
This method determins the order of two TInterfaceIndex objects
sl@0
    59
@param			indexEntry1 first TInterfaceIndex object
sl@0
    60
@param			indexEntry2 second TInterfaceIndex object
sl@0
    61
@return			integer indicating the order of these two
sl@0
    62
@pre			This object is fully constructed
sl@0
    63
*/
sl@0
    64
TInt CRegistryData::TInterfaceStruct::CompareInfUid(const TInterfaceIndex& indexEntry1,
sl@0
    65
													const TInterfaceIndex& indexEntry2)
sl@0
    66
	{
sl@0
    67
	return CompareTUidValues(indexEntry1.iInterfaceUid.iUid,indexEntry2.iInterfaceUid.iUid);
sl@0
    68
	}
sl@0
    69
sl@0
    70
//
sl@0
    71
//CRegistryData::TImplStruct
sl@0
    72
sl@0
    73
/**
sl@0
    74
Constructor for TImplStruct
sl@0
    75
@post			Its member variables are initialised
sl@0
    76
*/
sl@0
    77
CRegistryData::TImplStruct::TImplStruct():
sl@0
    78
	iCurrentImpl(NULL),
sl@0
    79
	iUnusedImpls(1)
sl@0
    80
	{
sl@0
    81
	// Do nothing here
sl@0
    82
	}
sl@0
    83
	
sl@0
    84
/**
sl@0
    85
This method determins the order of two TImplStruct objects
sl@0
    86
@param			aEntry1 first TImplStruct object
sl@0
    87
@param			aEntry2 second TImplStruct object
sl@0
    88
@return			integer indicating the order of these two
sl@0
    89
@pre			This object is fully constructed
sl@0
    90
*/
sl@0
    91
 TInt CRegistryData::TImplStruct::CompareImplStructUid(const TImplStruct& aEntry1,
sl@0
    92
													const TImplStruct& aEntry2)
sl@0
    93
	{
sl@0
    94
	return CompareTUidValues(aEntry1.iCurrentImpl->iImplInfo->ImplementationUid().iUid,
sl@0
    95
						aEntry2.iCurrentImpl->iImplInfo->ImplementationUid().iUid);
sl@0
    96
	}
sl@0
    97
	
sl@0
    98
/** Comparer to determine the order of a TUid object (implementation UID) in relation to and a TImplStruct object.
sl@0
    99
@param aUid the Implementation UID to compare with.
sl@0
   100
@param aEntry the TImplStruct object to compare aUid against.
sl@0
   101
@return integer indicating the order of the two.
sl@0
   102
*/
sl@0
   103
TInt CRegistryData::TImplStruct::CompareUidAgainstImplStruct(
sl@0
   104
	const TUid* aUid,
sl@0
   105
	const TImplStruct& aEntry)
sl@0
   106
	{
sl@0
   107
	return CompareTUidValues(aUid->iUid,aEntry.iCurrentImpl->iImplInfo->ImplementationUid().iUid);
sl@0
   108
	}
sl@0
   109
sl@0
   110
//
sl@0
   111
// CRegistryData::CImplementationData class
sl@0
   112
sl@0
   113
/**
sl@0
   114
Creates a new CImplemenatationData object. Note that CImplementationInformation needs to be constructed too,
sl@0
   115
for this object to be fully constructed.
sl@0
   116
@param			aParent A pointer to the parent instance of CInterfaceData
sl@0
   117
@return			A pointer to the newly created object.
sl@0
   118
@post			This object is partly constructed and initialized and is on 
sl@0
   119
				the CleanupStack.
sl@0
   120
*/
sl@0
   121
CRegistryData::CImplementationData* CRegistryData::CImplementationData::NewLC(CInterfaceData* aParent)
sl@0
   122
	{
sl@0
   123
	CImplementationData* self=new(ELeave) CImplementationData(aParent);
sl@0
   124
	CleanupStack::PushL(self);
sl@0
   125
	return self;
sl@0
   126
	}
sl@0
   127
sl@0
   128
/**
sl@0
   129
Creates a new CImplemenatationData object. It takes parameters to create a CImplementationInformation object,
sl@0
   130
during 2nd phase construction and initializes iImplInfo to reference to this CImplementationInformation object.
sl@0
   131
@param			aParent A pointer to the parent instance of CInterfaceData
sl@0
   132
@param			aUid The unique Id of the new implementation
sl@0
   133
@param			aVersion The version number of the new implementation
sl@0
   134
@param			aName The display name of the new implementation
sl@0
   135
@param			aDataType The data type supported by the new implementation
sl@0
   136
@param			aOpaqueData Data for the new implementation which is not used by the ECom framework
sl@0
   137
@param			aDrive The drive that the new implementation is on
sl@0
   138
@param			aRomOnly The flag recording whether the new implementation may be loaded from ROM only
sl@0
   139
@param			aRomBased The flag recording whether the new implementation is on ROM or is a later version of one on ROM
sl@0
   140
@return			A pointer to the newly created object
sl@0
   141
@post			This object is fully constructed and initialized.
sl@0
   142
*/
sl@0
   143
CRegistryData::CImplementationData* CRegistryData::CImplementationData::NewL(CInterfaceData* aParent,
sl@0
   144
																				TUid	aUid,
sl@0
   145
																				TInt	aVersion, 
sl@0
   146
																				HBufC*  aName,
sl@0
   147
																				HBufC8* aDataType,
sl@0
   148
																				HBufC8* aOpaqueData,
sl@0
   149
																				TDriveUnit aDrive,
sl@0
   150
																				TBool aRomOnly,
sl@0
   151
																				TBool aRomBased)
sl@0
   152
	{
sl@0
   153
	CImplementationData* self=new(ELeave) CImplementationData(aParent);
sl@0
   154
	CleanupStack::PushL(self);
sl@0
   155
	self->ConstructL(aUid,
sl@0
   156
					aVersion,
sl@0
   157
					aName,
sl@0
   158
					aDataType,
sl@0
   159
					aOpaqueData,
sl@0
   160
					aDrive,
sl@0
   161
					aRomOnly,
sl@0
   162
					aRomBased);
sl@0
   163
	CleanupStack::Pop(self);
sl@0
   164
	return self;
sl@0
   165
	}
sl@0
   166
sl@0
   167
/**
sl@0
   168
Creates a new CImplemenatationData object. It takes parameters to create a CImplementationInformation object,
sl@0
   169
during 2nd phase construction and initializes iImplInfo to reference to this CImplementationInformation object.
sl@0
   170
@param			aParent A pointer to the parent instance of CInterfaceData
sl@0
   171
@param			aUid The unique Id of the new implementation
sl@0
   172
@param			aVersion The version number of the new implementation
sl@0
   173
@param			aName The display name of the new implementation
sl@0
   174
@param			aDataType The data type supported by the new implementation
sl@0
   175
@param			aOpaqueData Data for the new implementation which is not used by the ECom framework
sl@0
   176
@param			aDrive The drive that the new implementation is on
sl@0
   177
@param			aRomOnly The flag recording whether the new implementation may be loaded from ROM only
sl@0
   178
@param			aRomBased The flag recording whether the new implementation is on ROM or is a later version of one on ROM
sl@0
   179
@param			aExtendedInterfaces The pointer to the array recording the extended interfaces supported by this implementation.
sl@0
   180
				NULL is available for PLUGIN without extended interfaces support.
sl@0
   181
@return			A pointer to the newly created object
sl@0
   182
@post			This object is fully constructed and initialized.
sl@0
   183
*/
sl@0
   184
CRegistryData::CImplementationData* CRegistryData::CImplementationData::NewL(CInterfaceData* aParent,
sl@0
   185
																				TUid	aUid,
sl@0
   186
																				TInt	aVersion, 
sl@0
   187
																				HBufC*  aName,
sl@0
   188
																				HBufC8* aDataType,
sl@0
   189
																				HBufC8* aOpaqueData,
sl@0
   190
																				TDriveUnit aDrive,
sl@0
   191
																				TBool aRomOnly,
sl@0
   192
																				TBool aRomBased,
sl@0
   193
																				RExtendedInterfacesArray* aExtendedInterfaces)
sl@0
   194
	{
sl@0
   195
	CImplementationData* self=new(ELeave) CImplementationData(aParent);
sl@0
   196
	CleanupStack::PushL(self);
sl@0
   197
	self->ConstructL(aUid,
sl@0
   198
					aVersion,
sl@0
   199
					aName,
sl@0
   200
					aDataType,
sl@0
   201
					aOpaqueData,
sl@0
   202
					aDrive,
sl@0
   203
					aRomOnly,
sl@0
   204
					aRomBased,
sl@0
   205
					aExtendedInterfaces);
sl@0
   206
	CleanupStack::Pop(self);
sl@0
   207
	return self;
sl@0
   208
	}
sl@0
   209
sl@0
   210
/**
sl@0
   211
Constructor for CImplementationData
sl@0
   212
@param			aParent The parent interface data of this implementation
sl@0
   213
@post			Its member variables are initialised
sl@0
   214
*/
sl@0
   215
CRegistryData::CImplementationData::CImplementationData( CInterfaceData* aParent):
sl@0
   216
	CBase(),
sl@0
   217
	iImplInfo(NULL),
sl@0
   218
	iParent(aParent)
sl@0
   219
	{
sl@0
   220
	// Do nothing here
sl@0
   221
	}
sl@0
   222
	
sl@0
   223
/**
sl@0
   224
Destructor of CImplementationData
sl@0
   225
*/	
sl@0
   226
CRegistryData::CImplementationData::~CImplementationData()
sl@0
   227
	{
sl@0
   228
	if(iImplInfo)
sl@0
   229
		{
sl@0
   230
		delete iImplInfo;
sl@0
   231
		iImplInfo = NULL;	
sl@0
   232
		}	
sl@0
   233
	iParent = NULL;		
sl@0
   234
	}
sl@0
   235
sl@0
   236
/**
sl@0
   237
The object's memory has been allocated.
sl@0
   238
@param			aUid The unique Id of the new implementation
sl@0
   239
@param			aVersion The version number of the new implementation
sl@0
   240
@param			aName The display name of the new implementation
sl@0
   241
@param			aDataType The data type supported by the new implementation
sl@0
   242
@param			aOpaqueData Data for the new implementation which is not used by the ECom framework
sl@0
   243
@param			aDrive The drive that the new implementation is on
sl@0
   244
@param			aRomOnly The flag recording whether the new implementation may be loaded from ROM only
sl@0
   245
@param			aRomBased The flag recording whether the new implementation is on ROM or is a later version of one on ROM
sl@0
   246
@pre 			This object is fully constructed.
sl@0
   247
@post			This object is fully constructed and initialized.
sl@0
   248
*/
sl@0
   249
void CRegistryData::CImplementationData::ConstructL(TUid	aUid,
sl@0
   250
													TInt	aVersion, 
sl@0
   251
													HBufC*  aName,
sl@0
   252
													HBufC8* aDataType,
sl@0
   253
													HBufC8* aOpaqueData,
sl@0
   254
													TDriveUnit aDrive,
sl@0
   255
													TBool aRomOnly,
sl@0
   256
													TBool aRomBased)
sl@0
   257
	{
sl@0
   258
	CImplementationInformation* newImpl = CImplementationInformation::NewL(aUid, 
sl@0
   259
													 						aVersion, 
sl@0
   260
													 						aName, 
sl@0
   261
															 				aDataType,
sl@0
   262
																			aOpaqueData,
sl@0
   263
														 					aDrive,
sl@0
   264
														 					aRomOnly,
sl@0
   265
													 						aRomBased);
sl@0
   266
	this->iImplInfo = newImpl;
sl@0
   267
	}
sl@0
   268
sl@0
   269
/**
sl@0
   270
The object's memory has been allocated.
sl@0
   271
@param			aUid The unique Id of the new implementation
sl@0
   272
@param			aVersion The version number of the new implementation
sl@0
   273
@param			aName The display name of the new implementation
sl@0
   274
@param			aDataType The data type supported by the new implementation
sl@0
   275
@param			aOpaqueData Data for the new implementation which is not used by the ECom framework
sl@0
   276
@param			aDrive The drive that the new implementation is on
sl@0
   277
@param			aRomOnly The flag recording whether the new implementation may be loaded from ROM only
sl@0
   278
@param			aRomBased The flag recording whether the new implementation is on ROM or is a later version of one on ROM
sl@0
   279
@param			aExtendedInterfaces The pointer to the array recording the extended interfaces supported by this implementation.
sl@0
   280
				NULL is available for PLUGIN without extended interfaces support.
sl@0
   281
@pre 			This object is fully constructed.
sl@0
   282
@post			This object is fully constructed and initialized.
sl@0
   283
*/
sl@0
   284
void CRegistryData::CImplementationData::ConstructL(TUid	aUid,
sl@0
   285
													TInt	aVersion, 
sl@0
   286
													HBufC*  aName,
sl@0
   287
													HBufC8* aDataType,
sl@0
   288
													HBufC8* aOpaqueData,
sl@0
   289
													TDriveUnit aDrive,
sl@0
   290
													TBool aRomOnly,
sl@0
   291
													TBool aRomBased,
sl@0
   292
													RExtendedInterfacesArray* aExtendedInterfaces)
sl@0
   293
	{
sl@0
   294
	CImplementationInformation* newImpl = CImplementationInformation::NewL(aUid, 
sl@0
   295
													 						aVersion, 
sl@0
   296
													 						aName, 
sl@0
   297
															 				aDataType,
sl@0
   298
																			aOpaqueData,
sl@0
   299
														 					aDrive,
sl@0
   300
														 					aRomOnly,
sl@0
   301
													 						aRomBased,
sl@0
   302
													 						aExtendedInterfaces);
sl@0
   303
	this->iImplInfo = newImpl;
sl@0
   304
	}
sl@0
   305
sl@0
   306
/**
sl@0
   307
Initialises member variable with the CImplementationInformation state specified in aStore.
sl@0
   308
@param			aStore The stream to read the data from.
sl@0
   309
@pre 			This object is full constructed.
sl@0
   310
@post			This object is set to the state specified in aStore.
sl@0
   311
*/
sl@0
   312
void CRegistryData::CImplementationData::InternalizeL(RReadStream& aStore)
sl@0
   313
	{
sl@0
   314
	if (iImplInfo)
sl@0
   315
		{
sl@0
   316
		delete iImplInfo;
sl@0
   317
		iImplInfo = NULL;
sl@0
   318
		}
sl@0
   319
	CImplementationInformation* implInfo = 0;
sl@0
   320
	implInfo=CImplementationInformation::NewLC(EFalse,aStore);
sl@0
   321
	//as we never store the drive name we need to get this from the parent
sl@0
   322
	implInfo->SetDrive(iParent->iParent->iParent->iDrive);
sl@0
   323
	CleanupStack::Pop(1);
sl@0
   324
	iImplInfo = implInfo;
sl@0
   325
	implInfo = 0;
sl@0
   326
	}
sl@0
   327
sl@0
   328
sl@0
   329
/**
sl@0
   330
Writes out the state of this member variable of type CImplementationInformation to aStore.
sl@0
   331
@param			aStore The stream to store the data in.
sl@0
   332
@pre 			This object is fully constructed.
sl@0
   333
*/
sl@0
   334
void CRegistryData::CImplementationData::ExternalizeL(RWriteStream& aStore) const
sl@0
   335
	{
sl@0
   336
	iImplInfo->ExternalizeL(EFalse,aStore);
sl@0
   337
	}
sl@0
   338
sl@0
   339
/**
sl@0
   340
This method determines the order of two CImplementationData objects.
sl@0
   341
For backward compatibility reason, ECom allows different I/Fs using
sl@0
   342
the same implementation UID for when the device has an implementation UID
sl@0
   343
that is not unique. Although this is not a valid or supported situation ECOM 
sl@0
   344
server should be robust.
sl@0
   345
@see FindImplementation
sl@0
   346
@see CompareTUidValues
sl@0
   347
@param			aImpl1 first reference to CImplementationData object 
sl@0
   348
@param			aImpl2 second reference to CImplementationData object
sl@0
   349
@return			integer indicating the order of these two
sl@0
   350
@pre			This object is fully constructed
sl@0
   351
*/	
sl@0
   352
TInt CRegistryData::CImplementationData::CompareImplUid(const CImplementationData& aImpl1,
sl@0
   353
													const CImplementationData& aImpl2)
sl@0
   354
	{
sl@0
   355
	TInt ret = CompareTUidValues(aImpl1.iImplInfo->ImplementationUid().iUid, aImpl2.iImplInfo->ImplementationUid().iUid);
sl@0
   356
	if (ret != 0)
sl@0
   357
		{
sl@0
   358
		return ret;
sl@0
   359
		}
sl@0
   360
sl@0
   361
	// Implementation UIDs are equal, use I/F UID as tie breaker
sl@0
   362
	return CompareTUidValues(aImpl1.iParent->iInterfaceUid.iUid, aImpl2.iParent->iInterfaceUid.iUid);
sl@0
   363
	}
sl@0
   364
	
sl@0
   365
/**
sl@0
   366
Similar to CompareImplUid above. This comparer only compare the Impl UID
sl@0
   367
and ignore the I/F UID part.
sl@0
   368
@param			aImpl1 first reference to CImplementationData object 
sl@0
   369
@param			aImpl2 second reference to CImplementationData object
sl@0
   370
@return			integer indicating the order of these two
sl@0
   371
@pre			This object is fully constructed
sl@0
   372
*/	
sl@0
   373
TInt CRegistryData::CImplementationData::CompareImplUidIgnoreIfUid(const CImplementationData& aImpl1,
sl@0
   374
													const CImplementationData& aImpl2)
sl@0
   375
	{
sl@0
   376
	return CompareTUidValues(aImpl1.iImplInfo->ImplementationUid().iUid,
sl@0
   377
						aImpl2.iImplInfo->ImplementationUid().iUid);
sl@0
   378
	}
sl@0
   379
	
sl@0
   380
/**
sl@0
   381
This method determines the order of an CImplementationData object in relation
sl@0
   382
to aUid.
sl@0
   383
@param	aUid is really TUid. Cast it back before compare.
sl@0
   384
@param	aImplData reference to CImplementationData object
sl@0
   385
@return	integer indicating the order of the two.
sl@0
   386
*/	
sl@0
   387
TInt CRegistryData::CImplementationData::CompareUidAgainstImplData(
sl@0
   388
											const CImplementationData& aUid,
sl@0
   389
											const CImplementationData& aImplData)
sl@0
   390
	{
sl@0
   391
	// The first argument aUid is really TUid.
sl@0
   392
	const TUid* ImplUid = reinterpret_cast<const TUid*>(&aUid);
sl@0
   393
	return CompareTUidValues(ImplUid->iUid,
sl@0
   394
						aImplData.iImplInfo->ImplementationUid().iUid);
sl@0
   395
	}
sl@0
   396
sl@0
   397
//
sl@0
   398
// CRegistryData::CInterfaceData class
sl@0
   399
sl@0
   400
/**
sl@0
   401
Creates a new CInterfaceData object and leave it on the CleanupStack
sl@0
   402
@param			aInterface The unique Id of this interface.
sl@0
   403
@param			aParent A pointer to the parent dll data 
sl@0
   404
@return			A pointer to the newly created class.
sl@0
   405
@post			This object is fully constructed and initialized and is on 
sl@0
   406
				the CleanupStack.
sl@0
   407
*/
sl@0
   408
CRegistryData::CInterfaceData* CRegistryData::CInterfaceData::NewLC(TUid aInterfaceUid,CDllData* aParent)
sl@0
   409
	{
sl@0
   410
	CInterfaceData* self=new(ELeave) CInterfaceData(aInterfaceUid,aParent);
sl@0
   411
	CleanupStack::PushL(self);
sl@0
   412
	self->ConstructL();
sl@0
   413
	return self;
sl@0
   414
	}
sl@0
   415
sl@0
   416
sl@0
   417
/**
sl@0
   418
Creates a new CInterfaceData object using the supplied interface id 
sl@0
   419
and leave it on the CleanupStack
sl@0
   420
@param			aParent A pointer to the parent dll data
sl@0
   421
@return			A pointer to the newly created class.
sl@0
   422
@post			This object is fully constructed and initialized and is on 
sl@0
   423
				the CleanupStack.
sl@0
   424
*/
sl@0
   425
CRegistryData::CInterfaceData* CRegistryData::CInterfaceData::NewLC(CDllData* aParent)
sl@0
   426
	{
sl@0
   427
	CInterfaceData* self=new(ELeave) CInterfaceData(aParent);
sl@0
   428
	CleanupStack::PushL(self);
sl@0
   429
	self->ConstructL();
sl@0
   430
	return self;
sl@0
   431
	}
sl@0
   432
sl@0
   433
sl@0
   434
CRegistryData::CInterfaceData::~CInterfaceData()
sl@0
   435
	{
sl@0
   436
	if(iImplementations)
sl@0
   437
		{
sl@0
   438
		iImplementations->ResetAndDestroy();
sl@0
   439
		delete iImplementations;
sl@0
   440
		iImplementations = NULL;
sl@0
   441
		}
sl@0
   442
	iParent = NULL;
sl@0
   443
	}
sl@0
   444
sl@0
   445
/**
sl@0
   446
Adds the specified implementation to this interface in the registry.
sl@0
   447
@param			aImplementation The implementation to add to this interface.
sl@0
   448
@pre			This object is fully constructed.
sl@0
   449
@post			aImplementation is added to the list of implementations for this interface.
sl@0
   450
*/
sl@0
   451
void CRegistryData::CInterfaceData::AddL(const CImplementationData* aImplementation)
sl@0
   452
	{
sl@0
   453
#ifdef ECOM_TRACE
sl@0
   454
	static int counter = 0; counter++;
sl@0
   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()));
sl@0
   456
sl@0
   457
#endif
sl@0
   458
sl@0
   459
	User::LeaveIfError(iImplementations->Append(aImplementation));
sl@0
   460
	}
sl@0
   461
sl@0
   462
/**
sl@0
   463
Sets the uid of this interface to aInterfaceUid.
sl@0
   464
@param			aInterfaceUid The Uid which this object should take.
sl@0
   465
@pre			This object is fully constructed.
sl@0
   466
@post			The Uid of this interface is set to aInterfaceUid
sl@0
   467
*/
sl@0
   468
void CRegistryData::CInterfaceData::SetInterfaceUid(TUid aInterfaceUid)
sl@0
   469
	{
sl@0
   470
	iInterfaceUid = aInterfaceUid;
sl@0
   471
	}
sl@0
   472
sl@0
   473
/**
sl@0
   474
Writes out this CInterfaceData to aStore.
sl@0
   475
@param			aStore The stream to store the data in.
sl@0
   476
@pre 			The state of this object is stored in the stream aStore.
sl@0
   477
*/
sl@0
   478
void CRegistryData::CInterfaceData::ExternalizeL(RWriteStream& aStore) const
sl@0
   479
	{
sl@0
   480
	aStore.WriteInt32L(iInterfaceUid.iUid);
sl@0
   481
sl@0
   482
	if(iImplementations)
sl@0
   483
		{
sl@0
   484
		const TInt entryCount = iImplementations->Count();
sl@0
   485
		aStore.WriteInt32L(entryCount);
sl@0
   486
		for(TInt i = 0; i < entryCount; ++i)
sl@0
   487
			((*iImplementations)[i])->ExternalizeL(aStore);
sl@0
   488
		}
sl@0
   489
	else
sl@0
   490
		aStore.WriteInt32L(0);
sl@0
   491
	}
sl@0
   492
sl@0
   493
/**
sl@0
   494
Restores this CInterfaceData to the state specified in aStore.
sl@0
   495
@param			aStore The stream to read the data from.
sl@0
   496
@param			aPresent A boolean indicating whether the dll is still present
sl@0
   497
@pre 			This object is full constructed.
sl@0
   498
@post			This object is set to the state specified in aStore.
sl@0
   499
*/
sl@0
   500
void CRegistryData::CInterfaceData::InternalizeL(RReadStream& aStore)
sl@0
   501
	{
sl@0
   502
	iInterfaceUid.iUid = aStore.ReadInt32L();
sl@0
   503
sl@0
   504
	const TInt entryCount = aStore.ReadInt32L();
sl@0
   505
    if(entryCount < 0)
sl@0
   506
        {
sl@0
   507
        User::Leave(KErrCorrupt);
sl@0
   508
        }
sl@0
   509
	
sl@0
   510
	for(TInt i = 0; i < entryCount; ++i)
sl@0
   511
		{
sl@0
   512
		CImplementationData* implementation = CImplementationData::NewLC(this);
sl@0
   513
		implementation->InternalizeL(aStore);
sl@0
   514
		AddL(implementation);
sl@0
   515
		CleanupStack::Pop(implementation);		//now owned by this interface
sl@0
   516
		}
sl@0
   517
	}
sl@0
   518
sl@0
   519
/**
sl@0
   520
@param			aParent A pointer to the parent dll data
sl@0
   521
*/
sl@0
   522
CRegistryData::CInterfaceData::CInterfaceData(CDllData* aParent) :
sl@0
   523
	CBase(),
sl@0
   524
	iParent(aParent)
sl@0
   525
	{
sl@0
   526
	// Do nothing here
sl@0
   527
	}
sl@0
   528
sl@0
   529
/**
sl@0
   530
@param			aInterfaceUid The unique Id of this interface
sl@0
   531
@param			aParent A pointer to the parent dll data			
sl@0
   532
*/
sl@0
   533
CRegistryData::CInterfaceData::CInterfaceData(TUid aInterfaceUid, CDllData* aParent) :
sl@0
   534
	CBase(),
sl@0
   535
	iInterfaceUid(aInterfaceUid),
sl@0
   536
	iParent(aParent)
sl@0
   537
	{
sl@0
   538
	// Do nothing here
sl@0
   539
	}
sl@0
   540
sl@0
   541
/**
sl@0
   542
Standard second phase construction function
sl@0
   543
@pre 			This object is fully constructed.
sl@0
   544
@post			This object is fully constructed and initialized.
sl@0
   545
*/
sl@0
   546
void CRegistryData::CInterfaceData::ConstructL()
sl@0
   547
	{
sl@0
   548
	iImplementations = new(ELeave) RPointerArray<CImplementationData>;
sl@0
   549
	}
sl@0
   550
sl@0
   551
//
sl@0
   552
// CRegistryData::CDllData class
sl@0
   553
sl@0
   554
/**
sl@0
   555
Creates a new CDllData object using aEntry and leaves it on the CleanupStack
sl@0
   556
@param			aDllName the name of this dll
sl@0
   557
@param			aDllModTime the modified time of this dll
sl@0
   558
@param 			aSecondUid Identifies type of the DLL. (PLUGIN or PLUGIN3)
sl@0
   559
@param  			aThirdUid Identifies a component uniquely. 	
sl@0
   560
@param			aParent A pointer to the parent drive data
sl@0
   561
@return			A pointer to the newly created object.
sl@0
   562
@post			This object is fully constructed and initialized and on the CleanupStack.
sl@0
   563
*/
sl@0
   564
CRegistryData::CDllData* CRegistryData::CDllData::NewLC(const TDesC& aDllName,const TTime& aDllModTime,const TUid& aSecondUid,const TUid& aThirdUid,CDriveData* aParent)
sl@0
   565
	{
sl@0
   566
	CDllData* self=new(ELeave) CDllData(aParent);
sl@0
   567
	CleanupStack::PushL(self);
sl@0
   568
	self->ConstructL(aDllName,aDllModTime,aSecondUid,aThirdUid);
sl@0
   569
	return self;
sl@0
   570
	}
sl@0
   571
sl@0
   572
/**
sl@0
   573
Creates a new CDllData object using aParent and leaves it on the CleanupStack
sl@0
   574
@param			aParent A pointer to the parent drive data
sl@0
   575
@return			A pointer to the newly created object.
sl@0
   576
@post			This object is fully constructed and initialized and on the CleanupStack.
sl@0
   577
*/
sl@0
   578
CRegistryData::CDllData* CRegistryData::CDllData::NewLC( CDriveData* aParent)
sl@0
   579
	{
sl@0
   580
	CDllData* self=new(ELeave) CDllData(aParent);
sl@0
   581
	CleanupStack::PushL(self);
sl@0
   582
	self->ConstructL();
sl@0
   583
	return self;
sl@0
   584
	}
sl@0
   585
	
sl@0
   586
CRegistryData::CDllData::~CDllData()
sl@0
   587
	{
sl@0
   588
	if(iIfList)
sl@0
   589
		{
sl@0
   590
		// Clear the interface list and destroy its objects
sl@0
   591
		iIfList->ResetAndDestroy();
sl@0
   592
		delete iIfList;
sl@0
   593
		iIfList = NULL;
sl@0
   594
		}
sl@0
   595
sl@0
   596
	// Unload this implementation dll.
sl@0
   597
	delete iDllEntry;
sl@0
   598
	iDllEntry = NULL;
sl@0
   599
	iParent = NULL;
sl@0
   600
	delete iRscFileExtension;
sl@0
   601
	}
sl@0
   602
sl@0
   603
/**
sl@0
   604
Adds the specified interface to this dll in the registry.
sl@0
   605
@param			aInterface The interface to add to this dll
sl@0
   606
@pre 			This object is fully constructed.
sl@0
   607
@post			aInterface is added to the list of interfaces in this dll.
sl@0
   608
*/
sl@0
   609
void CRegistryData::CDllData::AddL(const CInterfaceData* aInterface)
sl@0
   610
	{
sl@0
   611
	User::LeaveIfError(iIfList->Append(aInterface));
sl@0
   612
	}
sl@0
   613
sl@0
   614
/**
sl@0
   615
Sets the resource extension for the plugin. Not set for read only internal drives.
sl@0
   616
@param			aExt The resource extension to set
sl@0
   617
@pre 			This object is fully constructed.
sl@0
   618
@post			aExt is added to the object.
sl@0
   619
*/
sl@0
   620
void CRegistryData::CDllData::SetResourceExtL(const TDesC& aExt)
sl@0
   621
	{
sl@0
   622
	delete iRscFileExtension;
sl@0
   623
	iRscFileExtension = NULL;
sl@0
   624
	iRscFileExtension = aExt.AllocL();
sl@0
   625
	}
sl@0
   626
	
sl@0
   627
/**
sl@0
   628
Set the capability,the VID and do the Security check for this DLL.
sl@0
   629
@pre 			This object is fully constructed.
sl@0
   630
@return			ETrue if the security check is done successfully. Otherwise EFalse is returned. 
sl@0
   631
*/
sl@0
   632
TBool CRegistryData::CDllData::SaveSecurityInfoL()
sl@0
   633
	{
sl@0
   634
	iSecurityChecked = EFalse;
sl@0
   635
sl@0
   636
	//need to construct the full filename i.e. appending with the
sl@0
   637
	//preconstructed drivepath name in CDriveData
sl@0
   638
	TFileName dllFullName;
sl@0
   639
	dllFullName.Append(iParent->iDrive.Name());
sl@0
   640
	dllFullName.Append(_L("\\sys\\bin\\"));
sl@0
   641
	dllFullName.Append(iDllEntry->GetName());
sl@0
   642
	RLibrary::TInfoBuf infoBuf;
sl@0
   643
	TInt ret = RLibrary::GetInfo(dllFullName, infoBuf);
sl@0
   644
	if(ret != KErrNone)
sl@0
   645
		{
sl@0
   646
		return EFalse;
sl@0
   647
		}
sl@0
   648
	// Set the DLL's capability
sl@0
   649
	iCapSet = infoBuf().iSecurityInfo.iCaps;
sl@0
   650
	// Set the DLL's VID
sl@0
   651
	iVid = infoBuf().iSecurityInfo.iVendorId;
sl@0
   652
	// Now verify that SID identified in the resource file matches the SID of the Dll file
sl@0
   653
	TBool match=iDllEntry->GetThirdUid() ==infoBuf().iSecurityInfo.iSecureId;
sl@0
   654
	if (!match)
sl@0
   655
		{
sl@0
   656
#ifdef __ECOM_TRACE 
sl@0
   657
		__ECOM_TRACE1("ERROR: Plugin SID Mismatch ERROR for %S.", &dllFullName);
sl@0
   658
#endif
sl@0
   659
		return EFalse;
sl@0
   660
		}
sl@0
   661
sl@0
   662
	// Now verify the two DLL's second Uid are of the same Uid type (used for distinguising between collection/collection3)
sl@0
   663
	match=iDllEntry->GetSecondUid() ==infoBuf().iUids[1];
sl@0
   664
	if (!match)
sl@0
   665
		{
sl@0
   666
#ifdef __ECOM_TRACE 
sl@0
   667
		__ECOM_TRACE1("ERROR: Plugin UID2 Mismatch ERROR for %S.", &iDllEntry->GetName());
sl@0
   668
#endif
sl@0
   669
		return EFalse;
sl@0
   670
		}
sl@0
   671
sl@0
   672
	// On the emulator RLoader::CheckLibraryHash() returns KErrNotSupported.
sl@0
   673
	// Also on the emulator RLoader does no hash checking for DLL's on removable drives.
sl@0
   674
	// Therefore to be consistent ECOM does not do any hash checking itself on the emulator.
sl@0
   675
	// Hence code is removed for emulator builds.
sl@0
   676
#if !defined(__WINSCW__)
sl@0
   677
	// Verify hash is available
sl@0
   678
	if(iParent->iParent->iCachedDriveInfo->DriveIsRemovableL(iParent->iDrive))
sl@0
   679
		{
sl@0
   680
		RLoader loader;
sl@0
   681
		TInt err = loader.Connect();
sl@0
   682
		if(err != KErrNone)
sl@0
   683
			{
sl@0
   684
			return EFalse;
sl@0
   685
			}
sl@0
   686
		err = loader.CheckLibraryHash(dllFullName, ETrue);
sl@0
   687
		loader.Close();
sl@0
   688
		if(err != KErrNone)
sl@0
   689
			{
sl@0
   690
#ifdef __ECOM_TRACE 
sl@0
   691
		__ECOM_TRACE2("INFO: Hash Check Failed for %S with error %d.", &dllFullName, err);
sl@0
   692
#endif
sl@0
   693
			return EFalse;
sl@0
   694
			}
sl@0
   695
		}
sl@0
   696
#endif
sl@0
   697
	iSecurityChecked = ETrue;
sl@0
   698
	return ETrue;
sl@0
   699
	}
sl@0
   700
	
sl@0
   701
/**
sl@0
   702
Check whether security check has been performed if not go retrieve it.
sl@0
   703
@pre 			This object is fully constructed.
sl@0
   704
@return			ETrue if the security check is done successfully. Otherwise EFalse is returned. 
sl@0
   705
*/
sl@0
   706
TBool CRegistryData::CDllData::ProcessSecurityCheckL()
sl@0
   707
	{
sl@0
   708
	if(iSecurityChecked)
sl@0
   709
		{
sl@0
   710
		return ETrue;
sl@0
   711
		}
sl@0
   712
	return SaveSecurityInfoL();
sl@0
   713
	}
sl@0
   714
sl@0
   715
sl@0
   716
/**
sl@0
   717
Writes out the state of this CDllData to aStore.
sl@0
   718
@param			aStore The stream to store the data in.
sl@0
   719
@pre 			This object is fully constructed.
sl@0
   720
*/
sl@0
   721
void CRegistryData::CDllData::ExternalizeL(RWriteStream& aStore) const
sl@0
   722
   	{
sl@0
   723
 	const CEComEntry& dllEntryData = *iDllEntry;
sl@0
   724
	TInt size=dllEntryData.GetName().Length()-KDllExtensionLength;
sl@0
   725
 	aStore.WriteUint32L(size);
sl@0
   726
	aStore.WriteL(dllEntryData.GetName(),size);
sl@0
   727
sl@0
   728
	aStore.WriteInt32L(dllEntryData.GetSecondUid().iUid);	
sl@0
   729
  	aStore.WriteInt32L(dllEntryData.GetThirdUid().iUid);
sl@0
   730
    TPckg<TTime> modified(dllEntryData.GetModified());	
sl@0
   731
 	aStore.WriteL(modified);
sl@0
   732
	
sl@0
   733
	if(!iParent->iParent->iCachedDriveInfo->DriveIsReadOnlyInternalL(iParent->iDrive))
sl@0
   734
		{
sl@0
   735
		if(iRscFileExtension)
sl@0
   736
			{
sl@0
   737
			size = iRscFileExtension->Length();		
sl@0
   738
			aStore.WriteUint32L(size);
sl@0
   739
			aStore.WriteL(iRscFileExtension->Des(),size);
sl@0
   740
			}
sl@0
   741
		else
sl@0
   742
			{
sl@0
   743
			aStore.WriteUint32L(0);		
sl@0
   744
			}
sl@0
   745
		}
sl@0
   746
sl@0
   747
   	if(iIfList)
sl@0
   748
   		{
sl@0
   749
   		const TInt entryCount = iIfList->Count();
sl@0
   750
   		aStore.WriteInt32L(entryCount);
sl@0
   751
   		for(TInt i = 0; i < entryCount; ++i)
sl@0
   752
   			((*iIfList)[i])->ExternalizeL(aStore);
sl@0
   753
   		}
sl@0
   754
   	else
sl@0
   755
   		aStore.WriteInt32L(0);
sl@0
   756
	}
sl@0
   757
sl@0
   758
/**
sl@0
   759
Restores this CDllData to the state specified in aStore.
sl@0
   760
@param			aStore The stream to read the data from.
sl@0
   761
@pre 			This object is fully constructed.
sl@0
   762
@post			The state of this object is restored to that specified in aStore.
sl@0
   763
*/
sl@0
   764
void CRegistryData::CDllData::InternalizeL(RReadStream& aStore)
sl@0
   765
 	{	
sl@0
   766
	TInt size=aStore.ReadUint32L();
sl@0
   767
    //The next "if" checks if size < 0 not size <= 0 because it seems the ECOM server externalizes
sl@0
   768
    //the last file name as a string with length 0. If the fix is <= 0, then it makes it
sl@0
   769
    //incompatible with the existing applications
sl@0
   770
    if(size < 0 || size > KMaxFileName)
sl@0
   771
        {
sl@0
   772
        User::Leave(KErrCorrupt);
sl@0
   773
        }
sl@0
   774
	HBufC* name = HBufC::NewLC(size+KDllExtensionLength);
sl@0
   775
	TPtr ptr=name->Des();
sl@0
   776
	aStore.ReadL(ptr,size);
sl@0
   777
	ptr.Append(KDllExtension);
sl@0
   778
sl@0
   779
  	TUid secondUid=TUid::Uid(aStore.ReadInt32L());
sl@0
   780
  	TUid thirdUid=TUid::Uid(aStore.ReadInt32L());
sl@0
   781
  	TTime dllModifiedTime;  	
sl@0
   782
   	TPckg<TTime> modified(dllModifiedTime);	
sl@0
   783
 	aStore.ReadL(modified);
sl@0
   784
sl@0
   785
	if(!iParent->iParent->iCachedDriveInfo->DriveIsReadOnlyInternalL(iParent->iDrive))
sl@0
   786
		{
sl@0
   787
		size = aStore.ReadUint32L();
sl@0
   788
		if(size < 0 || size > KMaxFileName)
sl@0
   789
		        {
sl@0
   790
       		 User::Leave(KErrCorrupt);
sl@0
   791
       		 }
sl@0
   792
		if(size)
sl@0
   793
			{
sl@0
   794
			iRscFileExtension = HBufC::NewL(size);
sl@0
   795
			TPtr extPtr = iRscFileExtension->Des();
sl@0
   796
			aStore.ReadL(extPtr,size);
sl@0
   797
			}
sl@0
   798
		}
sl@0
   799
	
sl@0
   800
   	const TInt entryCount = aStore.ReadInt32L();
sl@0
   801
	if(entryCount < 0)
sl@0
   802
		{
sl@0
   803
		User::Leave(KErrCorrupt);
sl@0
   804
		}
sl@0
   805
	
sl@0
   806
   	// Security check is deferred until the DLL is needed.
sl@0
   807
	for(TInt i = 0; i < entryCount; ++i)
sl@0
   808
	   	{
sl@0
   809
	   	CInterfaceData* interface = CInterfaceData::NewLC(this);
sl@0
   810
		interface->InternalizeL(aStore);
sl@0
   811
		AddL(interface);
sl@0
   812
	   	CleanupStack::Pop(interface);	// now owned by dll	
sl@0
   813
		}  	
sl@0
   814
	iDllEntry = CEComEntry::NewL(*name,secondUid,thirdUid);
sl@0
   815
	iDllEntry->SetModified(dllModifiedTime);
sl@0
   816
 	CleanupStack::PopAndDestroy(name);
sl@0
   817
	}
sl@0
   818
	
sl@0
   819
/**
sl@0
   820
@param			aParent The parent drive data of this implementation
sl@0
   821
*/	
sl@0
   822
CRegistryData::CDllData::CDllData( CDriveData* aParent) :
sl@0
   823
	CBase(),
sl@0
   824
	iParent(aParent)	
sl@0
   825
	{
sl@0
   826
	// Do nothing here
sl@0
   827
	//Initialize empty capabilities here
sl@0
   828
	iCapSet.SetEmpty();
sl@0
   829
	}
sl@0
   830
sl@0
   831
/**
sl@0
   832
Standard second phase construction function.
sl@0
   833
@pre 			This object is fully constructed.
sl@0
   834
@post			This object is fully constructed and initialized.
sl@0
   835
*/
sl@0
   836
void CRegistryData::CDllData::ConstructL()
sl@0
   837
	{
sl@0
   838
	iIfList = new(ELeave) RInterfaceList(2);
sl@0
   839
	}
sl@0
   840
sl@0
   841
/**
sl@0
   842
Standard second phase construction function with parameter.
sl@0
   843
@param			aDllName the name of this dll
sl@0
   844
@param			aDllModTime the modified time of this dll
sl@0
   845
@param 			aSecondUid Distinguishes between components having the same UID1 (which distinguishes between EXEs and DLLs) 
sl@0
   846
				UID2 identifies Interface Implementation Collections (collection and collection 3)
sl@0
   847
@param			aThirdUid Identifies a component uniquely. In order to ensure that each binary that needs a distinguishing 
sl@0
   848
				UID is assigned a genuinely unique value. Symbian manages UID allocation through central database.	
sl@0
   849
@pre 			This object is fully constructed.
sl@0
   850
@post			This object is fully constructed and initialized.
sl@0
   851
*/
sl@0
   852
void CRegistryData::CDllData::ConstructL(const TDesC& aDllName,const TTime& aDllModTime,const TUid& aSecondUid,const TUid& aThirdUid)
sl@0
   853
	{
sl@0
   854
	iIfList = new(ELeave) RInterfaceList(2);
sl@0
   855
	iDllEntry=CEComEntry::NewL(aDllName,aSecondUid,aThirdUid);
sl@0
   856
	iDllEntry->SetModified(aDllModTime);
sl@0
   857
	}
sl@0
   858
	
sl@0
   859
	
sl@0
   860
/**
sl@0
   861
Populate a caller-supplied TEntry instance with the data from this dll.
sl@0
   862
@param			aEntry An entry to be populated (destination)
sl@0
   863
*/
sl@0
   864
void CRegistryData::CDllData::PopulateAnEntry(TEntry& aEntry) const
sl@0
   865
	{
sl@0
   866
	TPtr bufPtr=aEntry.iName.Des();
sl@0
   867
	bufPtr.Zero();
sl@0
   868
	bufPtr.Append(iParent->iDrive.Name());
sl@0
   869
	bufPtr.Append(_L("\\sys\\bin\\"));
sl@0
   870
	bufPtr.Append(iDllEntry->GetName());
sl@0
   871
	aEntry.iType	= TUidType(KDynamicLibraryUid,iDllEntry->GetSecondUid(),iDllEntry->GetThirdUid());
sl@0
   872
	aEntry.iModified= iDllEntry->GetModified();
sl@0
   873
	}
sl@0
   874
sl@0
   875
sl@0
   876
//
sl@0
   877
// CRegistryData::CDriveData class
sl@0
   878
sl@0
   879
sl@0
   880
/**
sl@0
   881
Creates a new CDriveData object and places it on the CleanupStack
sl@0
   882
@param			aDrive Information on this drive
sl@0
   883
@param			aParent A pointer to the parent registry data
sl@0
   884
@return			A pointer to the newly created class.
sl@0
   885
@post			This object is fully constructed and initialized and on the CleanupStack.
sl@0
   886
*/
sl@0
   887
CRegistryData::CDriveData* CRegistryData::CDriveData::NewLC(TDriveUnit aDrive, CRegistryData* aParent)
sl@0
   888
	{
sl@0
   889
	CDriveData* self=new(ELeave) CDriveData(aDrive,aParent);  // calls c'tor
sl@0
   890
	CleanupStack::PushL(self);	// Make the construction safe by using the cleanup stack
sl@0
   891
	self->ConstructL(); // Complete the 'construction'.
sl@0
   892
	return self;
sl@0
   893
	}
sl@0
   894
sl@0
   895
/**
sl@0
   896
Destructor of CDriveData
sl@0
   897
*/
sl@0
   898
CRegistryData::CDriveData::~CDriveData()
sl@0
   899
	{
sl@0
   900
	if(iDllList)
sl@0
   901
		{
sl@0
   902
		// Clear the interface list and destroy its objects
sl@0
   903
		iDllList->ResetAndDestroy();
sl@0
   904
		delete iDllList;
sl@0
   905
		iDllList = NULL;
sl@0
   906
		}
sl@0
   907
	iParent=NULL;
sl@0
   908
	}
sl@0
   909
sl@0
   910
/**
sl@0
   911
Adds the specified CDllData to this drive data in the registry.
sl@0
   912
@param			aDll The Dll to add to this drive
sl@0
   913
@pre 			This object is fully constructed.
sl@0
   914
@post			aDll is added to the list of dll on this drive.
sl@0
   915
*/
sl@0
   916
void CRegistryData::CDriveData::AddL(const CDllData* aDll)
sl@0
   917
	{
sl@0
   918
	User::LeaveIfError(iDllList->Append(aDll));
sl@0
   919
	}
sl@0
   920
sl@0
   921
/**
sl@0
   922
Returns the index of the DLL with this Uid in the Dll list.
sl@0
   923
@param			aDllUid the UID of to be found DLL.
sl@0
   924
@return  		The index of this DLL in the DLL list. KErrNotFound if not found.
sl@0
   925
@pre			This object is fully constructed.
sl@0
   926
*/
sl@0
   927
TInt CRegistryData::CDriveData::FindDllIndex(const TUid aDllUid) const
sl@0
   928
	{
sl@0
   929
	const TInt dllCount = iDllList->Count();
sl@0
   930
	for(TInt j = 0; j < dllCount; ++j)
sl@0
   931
		{
sl@0
   932
		// For each dll structure check the dll UID
sl@0
   933
		CDllData* dll = (*iDllList)[j];
sl@0
   934
		if (aDllUid == dll->iDllEntry->GetThirdUid())
sl@0
   935
			{
sl@0
   936
			// it is a match 
sl@0
   937
 			return j;	 				
sl@0
   938
			}
sl@0
   939
		}
sl@0
   940
	return KErrNotFound;
sl@0
   941
	}
sl@0
   942
	
sl@0
   943
/**
sl@0
   944
Writes out the state of this CDriveData to file.
sl@0
   945
@param			aFs A handle to an open file server session.
sl@0
   946
@param			aDatFileName the dat file to persist to
sl@0
   947
@pre 			This object is fully constructed.
sl@0
   948
*/
sl@0
   949
void CRegistryData::CDriveData::ExternalizeL(RFs& aFs,const TDesC& aDatFileName) 
sl@0
   950
	{
sl@0
   951
	// attempt to create the folders if they don't already exist
sl@0
   952
	TInt mkDirError = aFs.MkDirAll(aDatFileName);
sl@0
   953
		
sl@0
   954
	if((mkDirError == KErrNone) || (mkDirError == KErrAlreadyExists))
sl@0
   955
		{
sl@0
   956
		RFileWriteStream registryStream;
sl@0
   957
		if(registryStream.Replace(aFs,aDatFileName, EFileWrite) == KErrNone)
sl@0
   958
			{
sl@0
   959
			CleanupClosePushL(registryStream);
sl@0
   960
			// Write the version number as the first thing in the file stream.
sl@0
   961
			registryStream.WriteInt32L(KRegVersion);				
sl@0
   962
			if(iDllList)
sl@0
   963
				{
sl@0
   964
				TInt entryCount = iDllList->Count();
sl@0
   965
				registryStream.WriteInt32L(entryCount);
sl@0
   966
				// Now stream out the data
sl@0
   967
				for(TInt i = 0; i < entryCount; ++i)
sl@0
   968
					((*iDllList)[i])->ExternalizeL(registryStream);
sl@0
   969
				}
sl@0
   970
			else
sl@0
   971
				registryStream.WriteInt32L(0);
sl@0
   972
			CleanupStack::PopAndDestroy();	// registryStream
sl@0
   973
			}
sl@0
   974
		}
sl@0
   975
	else
sl@0
   976
		{
sl@0
   977
		User::Leave(mkDirError);
sl@0
   978
		}
sl@0
   979
	}
sl@0
   980
sl@0
   981
/**
sl@0
   982
Restores the state of this CDriveData from file.
sl@0
   983
@param			aFs A handle to an open file server session.
sl@0
   984
@param			aDatFileName the dat file name to internalize from
sl@0
   985
@pre 			This object is fully constructed.
sl@0
   986
@post			The state of this object is restored to that specified
sl@0
   987
				in the file on this drive.
sl@0
   988
*/
sl@0
   989
sl@0
   990
void CRegistryData::CDriveData::InternalizeL(RFs& aFs,const TDesC& aDatFileName)
sl@0
   991
	{
sl@0
   992
	DoInternalizeL(aFs, aDatFileName);  
sl@0
   993
	}
sl@0
   994
/**
sl@0
   995
The method internalizes the data from aRegistryStream.
sl@0
   996
@param aFileName The ECOM registry file name.
sl@0
   997
@leave KErrNoMemory
sl@0
   998
@leave Some other system-wide error codes as well.
sl@0
   999
*/
sl@0
  1000
void CRegistryData::CDriveData::DoInternalizeL(RFs& aFs, const TDesC& aFileName)
sl@0
  1001
	{
sl@0
  1002
	RFileReadStream registryStream;
sl@0
  1003
	User::LeaveIfError(registryStream.Open(aFs, aFileName, EFileRead));
sl@0
  1004
	CleanupClosePushL(registryStream);	
sl@0
  1005
	// Read in version number. If its not what expected - the file is treated as corrupted.
sl@0
  1006
	const TInt version = registryStream.ReadInt32L();
sl@0
  1007
	if(version != KRegVersion)
sl@0
  1008
		{
sl@0
  1009
		User::Leave(KErrCorrupt);
sl@0
  1010
		}	
sl@0
  1011
	const TInt entryCount = registryStream.ReadInt32L();
sl@0
  1012
    if(entryCount < 0)
sl@0
  1013
        {
sl@0
  1014
        User::Leave(KErrCorrupt);
sl@0
  1015
        }
sl@0
  1016
	for(TInt i = 0; i < entryCount; ++i)
sl@0
  1017
		{
sl@0
  1018
		CDllData* dll = CDllData::NewLC(this);
sl@0
  1019
		dll->InternalizeL(registryStream);
sl@0
  1020
		// always add the DLL, DLL security check will be deferred 
sl@0
  1021
		// until the DLL is needed.		
sl@0
  1022
		AddL(dll);
sl@0
  1023
		CleanupStack::Pop(dll);	// now owned by drive	
sl@0
  1024
		}
sl@0
  1025
	CleanupStack::PopAndDestroy();//registryStream
sl@0
  1026
	}
sl@0
  1027
	
sl@0
  1028
/**	
sl@0
  1029
Constructor for CDriveData
sl@0
  1030
@param			aDrive Information on this drive
sl@0
  1031
@param			aParent A pointer to the parent registry data
sl@0
  1032
*/	
sl@0
  1033
CRegistryData::CDriveData::CDriveData(TDriveUnit aDrive, CRegistryData* aParent) :
sl@0
  1034
	CBase(), 
sl@0
  1035
sl@0
  1036
	iDrive(aDrive),
sl@0
  1037
	iParent(aParent),
sl@0
  1038
	iDriveChanged(ETrue),
sl@0
  1039
	iRegistryChanged(EFalse)
sl@0
  1040
	{
sl@0
  1041
	}
sl@0
  1042
sl@0
  1043
/**
sl@0
  1044
Standard second phase construction function.
sl@0
  1045
@pre 			This object is fully constructed.
sl@0
  1046
@post			This object is fully constructed and initialized.
sl@0
  1047
*/
sl@0
  1048
void CRegistryData::CDriveData::ConstructL()
sl@0
  1049
	{
sl@0
  1050
	iDllList = new(ELeave)TDll;
sl@0
  1051
	}
sl@0
  1052
	
sl@0
  1053
//
sl@0
  1054
// CRegistryData class
sl@0
  1055
sl@0
  1056
/**
sl@0
  1057
Standardized safe construction which leaves nothing on the cleanup stack.
sl@0
  1058
This overload is used by CEComImplIndexPerfTest which plots 
sl@0
  1059
discovery time vs. granularity settings.
sl@0
  1060
@leave 			KErrNoMemory.	
sl@0
  1061
@param 			aFs The open file session.
sl@0
  1062
@param			aInterfaceImplIndexGranularity granularity of main index.
sl@0
  1063
@param			aImplIndexGranularity granularity of auxiliary index.
sl@0
  1064
@return			The newly created instance of the registry data.
sl@0
  1065
@post			CRegistryData is fully constructed, and initialized.
sl@0
  1066
*/
sl@0
  1067
CRegistryData* CRegistryData::NewL(RFs& aFs, TInt aInterfaceImplIndexGranularity, TInt aImplIndexGranularity)
sl@0
  1068
	{
sl@0
  1069
	CRegistryData* self=new(ELeave) CRegistryData(aFs, aInterfaceImplIndexGranularity, aImplIndexGranularity);  // calls c'tor
sl@0
  1070
	CleanupStack::PushL(self);	// Make the construction safe by using the cleanup stack
sl@0
  1071
	self->ConstructL(); // Complete the 'construction'.
sl@0
  1072
	CleanupStack::Pop(self);
sl@0
  1073
	return self;
sl@0
  1074
	}
sl@0
  1075
sl@0
  1076
/**
sl@0
  1077
Standardized safe construction which leaves nothing on the cleanup stack.
sl@0
  1078
This overload uses default granularities for the two indexes. These
sl@0
  1079
default values were found to provide reasonably good performance at
sl@0
  1080
the time of testing.
sl@0
  1081
@leave 			KErrNoMemory.	
sl@0
  1082
@param 			aFs The open file session.
sl@0
  1083
@return			The newly created instance of the registry data.
sl@0
  1084
@post			CRegistryData is fully constructed, and initialized.
sl@0
  1085
*/
sl@0
  1086
CRegistryData* CRegistryData::NewL(RFs& aFs)
sl@0
  1087
	{
sl@0
  1088
	return NewL(aFs, KDefaultInterfaceImplIndexGranularity, KDefaultImplIndexGranularity);
sl@0
  1089
	}
sl@0
  1090
sl@0
  1091
CRegistryData::~CRegistryData()
sl@0
  1092
	{
sl@0
  1093
	for(TInt index = 0; index < iInterfaceImplIndex.Count(); ++index)
sl@0
  1094
		{
sl@0
  1095
		iInterfaceImplIndex[index].Reset();
sl@0
  1096
		}
sl@0
  1097
	iInterfaceImplIndex.Reset();
sl@0
  1098
	
sl@0
  1099
	// Reset only. It does not own the pointers!
sl@0
  1100
	iImplIndex.Reset();
sl@0
  1101
sl@0
  1102
	if(iRegistrations)
sl@0
  1103
		{
sl@0
  1104
		iRegistrations->ResetAndDestroy();
sl@0
  1105
		delete iRegistrations;
sl@0
  1106
		iRegistrations = NULL;
sl@0
  1107
		}
sl@0
  1108
	// reset the cached language settings
sl@0
  1109
	RDowngradePath::Reset();
sl@0
  1110
sl@0
  1111
	delete iCachedDriveInfo;
sl@0
  1112
	}
sl@0
  1113
sl@0
  1114
/**
sl@0
  1115
Lists the implementations given the interface UID. The filtered list is returned in the client provided
sl@0
  1116
RImplDataArray parameter.
sl@0
  1117
@param			aInterfaceUid The Uid of the interface which the implementations should provide
sl@0
  1118
@param			aImplementationData Return value. The filtered list.
sl@0
  1119
*/
sl@0
  1120
void CRegistryData::ListImplementationsL(TUid aInterfaceUid,
sl@0
  1121
										 RImplDataArray& aImplementationData) const
sl@0
  1122
	{
sl@0
  1123
	// If discoveries are taking place then the index is invalid so we cannot do
sl@0
  1124
	// this request
sl@0
  1125
	if(iCurrentlyDiscovering)
sl@0
  1126
		User::Leave(KEComErrListCurrentlyUnavailable);
sl@0
  1127
    
sl@0
  1128
	TBool securityPassed = EFalse;
sl@0
  1129
	//Do the security check
sl@0
  1130
	TInt index = 0;
sl@0
  1131
	while (!securityPassed)
sl@0
  1132
		{
sl@0
  1133
		aImplementationData.Reset();
sl@0
  1134
		index = IndexedFind(aInterfaceUid);
sl@0
  1135
		if(index == KErrNotFound)
sl@0
  1136
			{
sl@0
  1137
			User::Leave(KEComErrNoInterfaceIdentified);
sl@0
  1138
			}
sl@0
  1139
		User::LeaveIfError(index);
sl@0
  1140
	
sl@0
  1141
		TImplContainerArray& implementationList = iInterfaceImplIndex[index].iImpData;
sl@0
  1142
		for (TInt j = 0; j < implementationList.Count(); j++)
sl@0
  1143
			{
sl@0
  1144
			CImplementationData* currentImplementation = implementationList[j].iCurrentImpl;
sl@0
  1145
			CDllData* dll = currentImplementation->iParent->iParent;
sl@0
  1146
			
sl@0
  1147
			securityPassed = dll->ProcessSecurityCheckL();
sl@0
  1148
			if(!securityPassed)
sl@0
  1149
				{
sl@0
  1150
				// remove the implementations of the DLL from iInterfaceImplIndex
sl@0
  1151
				// and remove the DLL from its parent DLL list.
sl@0
  1152
				DeleteDllL(dll);
sl@0
  1153
				delete dll;
sl@0
  1154
				dll = NULL;
sl@0
  1155
				// don't continue processing implementations after we have deleted the DLL
sl@0
  1156
				// because the implementations list will have changed, so we need to 
sl@0
  1157
				// re-do the IndexedFind
sl@0
  1158
				break;	
sl@0
  1159
				}
sl@0
  1160
			else
sl@0
  1161
				{
sl@0
  1162
				User::LeaveIfError(aImplementationData.Append(currentImplementation));
sl@0
  1163
				}	
sl@0
  1164
			}
sl@0
  1165
		}
sl@0
  1166
	}
sl@0
  1167
sl@0
  1168
/**
sl@0
  1169
Enables or disables the specified interface implementation within the registry.
sl@0
  1170
@param			aImplementationUid The interface implementation to change.
sl@0
  1171
@param			aState ETrue to enable the implementation, EFalse to disable it
sl@0
  1172
@return			KErrNone, KEComErrEnableFailed or KEComErrDisableFailed
sl@0
  1173
@pre 			CRegistry is fully constructed,
sl@0
  1174
@post			The implementation is now enabled or disabled as described 
sl@0
  1175
				by aState.
sl@0
  1176
*/
sl@0
  1177
TInt CRegistryData::SetEnabledState(TUid aImplementationUid, TBool aState)
sl@0
  1178
	{
sl@0
  1179
	CImplementationData* impData = NULL;
sl@0
  1180
	TUid dummy={0x00000000};
sl@0
  1181
	
sl@0
  1182
	(void)FindImplementation(aImplementationUid, dummy, impData);
sl@0
  1183
sl@0
  1184
	if(impData != NULL)
sl@0
  1185
		{
sl@0
  1186
		impData->iImplInfo->SetDisabled((aState) ? EFalse : ETrue);
sl@0
  1187
		return KErrNone;
sl@0
  1188
		}
sl@0
  1189
sl@0
  1190
    return (aState) ? KEComErrEnableFailed : KEComErrDisableFailed;
sl@0
  1191
	}
sl@0
  1192
sl@0
  1193
sl@0
  1194
/**
sl@0
  1195
Temporarily uninstalls the interface implementation groups upon the specified drive.
sl@0
  1196
(ie this can be undone later).
sl@0
  1197
@param			aDrive	The identifier of the drive to uninstall.
sl@0
  1198
@pre 			CRegistry is fully constructed
sl@0
  1199
@post			The registered implementation groupings
sl@0
  1200
				stored upon the specified drive are no 
sl@0
  1201
				longer available for use.
sl@0
  1202
*/
sl@0
  1203
void CRegistryData::TemporaryUninstallL(const TDriveUnit& aDrive)
sl@0
  1204
	{
sl@0
  1205
	// Find the appropriate drive entry and remove it...
sl@0
  1206
	// Note : the drive may have already been taken offline, so no save is possible!
sl@0
  1207
	CDriveData* drive = NULL;
sl@0
  1208
	TInt driveIndex = FindDriveL(aDrive, drive);
sl@0
  1209
	if(driveIndex != KErrNotFound)
sl@0
  1210
		{
sl@0
  1211
    	TInt dllIndex = drive->iDllList->Count();
sl@0
  1212
		while(dllIndex > 0)
sl@0
  1213
			{
sl@0
  1214
			--dllIndex;
sl@0
  1215
			RemoveFromIndexL((*drive->iDllList)[dllIndex]);
sl@0
  1216
			}       
sl@0
  1217
		iRegistrations->Remove(driveIndex);
sl@0
  1218
		delete drive;
sl@0
  1219
sl@0
  1220
		DriveChanged(aDrive, ETrue);
sl@0
  1221
		}
sl@0
  1222
	// The flag iRegistryChanged has been moved to each drive.
sl@0
  1223
	}
sl@0
  1224
sl@0
  1225
/**
sl@0
  1226
Undoes a temporary uninstall of the interface 
sl@0
  1227
implementation groups upon the specified drive.
sl@0
  1228
(ie this can be undone later).
sl@0
  1229
@leave			KErrNoMemory, KErrReinstallFailed.
sl@0
  1230
@param			aDrive	The identifier of the drive to reinstate.
sl@0
  1231
@pre 			CRegistry is fully constructed
sl@0
  1232
@post			The registered implementation groupings
sl@0
  1233
				stored upon the specified drive are again 
sl@0
  1234
				made available for use.
sl@0
  1235
*/
sl@0
  1236
void CRegistryData::UndoTemporaryUninstallL(const TDriveUnit& aDrive)
sl@0
  1237
	{
sl@0
  1238
	CDriveData* driveData = NULL;
sl@0
  1239
	// If the drive was not found then install it, otherwise do nothing because it
sl@0
  1240
	// is already there.  NB. We could leave here but the problem with that is the 
sl@0
  1241
	// use case where the drive has been mounted during registry load but the registrar 
sl@0
  1242
	// is started afterwards, and makes calls to this method as it discovers drives 
sl@0
  1243
	// for the first time. 
sl@0
  1244
	if(FindDriveL(aDrive, driveData) == KErrNotFound)
sl@0
  1245
		{
sl@0
  1246
		// Add the drive entry
sl@0
  1247
		CDriveData* drive = CDriveData::NewLC(aDrive,this);
sl@0
  1248
		AddImplDataL(drive);
sl@0
  1249
		User::LeaveIfError(iRegistrations->Append(drive));
sl@0
  1250
		CleanupStack::Pop(drive);		// now owned by iRegistrations
sl@0
  1251
sl@0
  1252
		DriveChanged(aDrive, EFalse);
sl@0
  1253
sl@0
  1254
		// NB We DO need to set iRegistryChanged = ETrue because the idx file needs to be
sl@0
  1255
		// rewritten to include the reinstalled drive
sl@0
  1256
		}
sl@0
  1257
	}
sl@0
  1258
	
sl@0
  1259
/** 
sl@0
  1260
Update the staus of drives in the system
sl@0
  1261
@param 	aDrive The drive that has changed
sl@0
  1262
@param 	aDriveRemoved Indicates whether a drive has been removed or added
sl@0
  1263
@pre 	CRegistry is fully constructed.
sl@0
  1264
@post	The state of the drive has been stored
sl@0
  1265
*/
sl@0
  1266
void CRegistryData::DriveChanged(const TDriveUnit& aDrive, TBool aDriveRemoved)
sl@0
  1267
	{
sl@0
  1268
	if(aDriveRemoved)
sl@0
  1269
		{
sl@0
  1270
		iRemovedDrives |= (0x1 << aDrive);
sl@0
  1271
		}
sl@0
  1272
	else
sl@0
  1273
		{
sl@0
  1274
		iRemovedDrives &= ~(0x1 << aDrive);
sl@0
  1275
		}
sl@0
  1276
	}
sl@0
  1277
sl@0
  1278
/** 
sl@0
  1279
Find if any Dll is regsitered in the drive unit.
sl@0
  1280
@param aDrive The identifier of the drive to find if any Dll is registered.
sl@0
  1281
@return ETrue if any Dll is registered in the drive, otherwise EFalse.
sl@0
  1282
@pre 			CRegistry is fully constructed.
sl@0
  1283
@post			If any Dll is registered in the drive is returned if successfully.
sl@0
  1284
*/
sl@0
  1285
TBool CRegistryData::IsAnyDllRegisteredWithDriveL(const TDriveUnit& aDrive)const
sl@0
  1286
	{
sl@0
  1287
	CDriveData* driveData = NULL;
sl@0
  1288
	if(FindDriveL(aDrive, driveData) == KErrNotFound)
sl@0
  1289
		{
sl@0
  1290
		User::Leave(KEComErrDriveNotFound);	
sl@0
  1291
		}
sl@0
  1292
	return driveData->iDllList->Count() > 0;
sl@0
  1293
	}
sl@0
  1294
	
sl@0
  1295
/**
sl@0
  1296
Determine if the specified implementation grouping already registered, but needs an update.
sl@0
  1297
@param			aDllUid The interface implementation collection data to add. 
sl@0
  1298
				CRegistryData takes ownership of this object
sl@0
  1299
				so it should not be on the CleanupStack.
sl@0
  1300
@param			aModified The date of the last modification of the Interface 
sl@0
  1301
				Implementation Collection. 
sl@0
  1302
@param			aUpdate An output to indicate if the registry entry requires an update. 
sl@0
  1303
				(Only valid if the return is ETrue).
sl@0
  1304
@param			aDriveData The drive data used to find DLL.
sl@0
  1305
@return			ETrue if the Interface Implementation Collection already has a registry entry.
sl@0
  1306
				EFalse otherwise.
sl@0
  1307
@pre 			This object is fully constructed and there is also a valid drive entry
sl@0
  1308
				in the registry for aDrive
sl@0
  1309
@post			The new data is added to the registry
sl@0
  1310
*/
sl@0
  1311
TBool CRegistryData::IsRegisteredWithDate (TUid aDllUid, 
sl@0
  1312
										   	const TTime& aModified, 
sl@0
  1313
										   	TBool& aUpdate, 
sl@0
  1314
										   	CDriveData* aDriveData)
sl@0
  1315
	{
sl@0
  1316
	// Find the appropriate drive entry for this 
sl@0
  1317
	// Interface Implementation Collection
sl@0
  1318
	TBool found = EFalse;
sl@0
  1319
	TInt index = aDriveData->FindDllIndex(aDllUid);
sl@0
  1320
	if(index != KErrNotFound)
sl@0
  1321
		{
sl@0
  1322
		TDll* dllList = aDriveData->iDllList;
sl@0
  1323
		CDllData* dll = (*dllList)[index];
sl@0
  1324
		const CEComEntry& dllEntry =*(dll->iDllEntry);	
sl@0
  1325
		aUpdate = dllEntry.GetModified().Int64() < aModified.Int64(); 
sl@0
  1326
		found = ETrue;
sl@0
  1327
		}
sl@0
  1328
	return found;
sl@0
  1329
	}
sl@0
  1330
sl@0
  1331
/**
sl@0
  1332
Adds data on a specific dll to the registry. The data to be added
sl@0
  1333
is parsed by CRegistryParser.
sl@0
  1334
@leave			KErrNoMemory If the item could not be appended to the registry
sl@0
  1335
@leave			KEComErrDriveNotFound If aDrive is not a valid drive
sl@0
  1336
@param			aDrive The drive the registry data has been found on 
sl@0
  1337
@param			aFoundDriveIndex The index of the registry data for the drive 
sl@0
  1338
				the interface implementation collection data has been found on.
sl@0
  1339
@param			aDllData The dll data to add. CRegistryData takes ownership of this object
sl@0
  1340
				so it should not be on the CleanupStack.
sl@0
  1341
@pre 			This object is fully constructed and there is also a valid drive entry
sl@0
  1342
				in the registry for aDrive
sl@0
  1343
@post			The new data is added to the registry
sl@0
  1344
*/
sl@0
  1345
void CRegistryData::AddDllDataL(const TDriveUnit& aDrive, TInt aFoundDriveIndex, CRegistryData::CDllData* aDllData)
sl@0
  1346
	{
sl@0
  1347
	const TInt regCount = iRegistrations->Count();
sl@0
  1348
	if(regCount == 0)
sl@0
  1349
		User::Leave(KEComErrDriveNotFound);
sl@0
  1350
sl@0
  1351
	// Find the appropriate drive entry
sl@0
  1352
	CDriveData* drive = NULL;
sl@0
  1353
	if(aFoundDriveIndex == KErrNotFound)
sl@0
  1354
		{
sl@0
  1355
		if(FindDriveL(aDrive, drive) == KErrNotFound)
sl@0
  1356
			User::Leave(KEComErrDriveNotFound);
sl@0
  1357
		}
sl@0
  1358
	else
sl@0
  1359
		drive = (*iRegistrations)[aFoundDriveIndex];
sl@0
  1360
	
sl@0
  1361
	// Append the new data to the list
sl@0
  1362
	TDll* dllList = drive->iDllList;
sl@0
  1363
	
sl@0
  1364
	// NOTE : This function MUST NOT leave after the following line because 
sl@0
  1365
	// aDllData will potentially be 'owned' in two places (dllList and the caller) 
sl@0
  1366
	// until we return and the caller can Pop.
sl@0
  1367
	TBool checkNeeded = ETrue;
sl@0
  1368
	TInt ifListCount=aDllData->iIfList->Count();
sl@0
  1369
	for(TInt ifListCounter=0; ifListCounter < ifListCount; ifListCounter++)
sl@0
  1370
		{
sl@0
  1371
		// For each interface structure
sl@0
  1372
		CInterfaceData* interface = (*aDllData->iIfList)[ifListCounter];
sl@0
  1373
		for(TInt impNum = 0; impNum < interface->iImplementations->Count(); ++impNum)
sl@0
  1374
			{
sl@0
  1375
			CImplementationData* implData = (*interface->iImplementations)[impNum];
sl@0
  1376
			if(!InsertIntoIndexL(implData,checkNeeded))
sl@0
  1377
				{
sl@0
  1378
				interface->iImplementations->Remove(impNum); //remove from implementation array
sl@0
  1379
				delete implData; //delete object
sl@0
  1380
				}	
sl@0
  1381
			}
sl@0
  1382
		}
sl@0
  1383
	//add filtered list for legitimate implementations into registry
sl@0
  1384
	User::LeaveIfError(dllList->Append(aDllData)); 
sl@0
  1385
		
sl@0
  1386
	drive->iRegistryChanged = ETrue;	
sl@0
  1387
	}
sl@0
  1388
sl@0
  1389
/**
sl@0
  1390
Updates the data for a dll in the registry.
sl@0
  1391
@leave			KErrNoMemory If the indexes cannot be rebuilt
sl@0
  1392
@param			aFoundDriveIndex The index of the registry data for the drive 
sl@0
  1393
				the interface implementation collection data has been found on.
sl@0
  1394
@param			aDllData The updated interface implementation collection data. 
sl@0
  1395
				CRegistryData takes ownership of this object
sl@0
  1396
				so it should not be on the CleanupStack.
sl@0
  1397
@pre 			This object is fully constructed
sl@0
  1398
@post			The dll data is updated in the registry
sl@0
  1399
*/
sl@0
  1400
void CRegistryData::UpdateDllDataL(const TDriveUnit& aDrive,TInt aFoundDriveIndex, CDllData* aDllData)
sl@0
  1401
	{
sl@0
  1402
	const TInt regCount = iRegistrations->Count();
sl@0
  1403
	if(regCount == 0)
sl@0
  1404
		User::Leave(KEComErrDriveNotFound);
sl@0
  1405
sl@0
  1406
	// Find the appropriate drive entry
sl@0
  1407
	CDriveData* drive = (*iRegistrations)[aFoundDriveIndex];
sl@0
  1408
	
sl@0
  1409
	TInt index = drive->FindDllIndex(aDllData->iDllEntry->GetThirdUid());
sl@0
  1410
	
sl@0
  1411
	// find the dll entry and clear that too
sl@0
  1412
	if(index != KErrNotFound)
sl@0
  1413
		{
sl@0
  1414
		// Ok its a match, so remove this registry entry and call AddDllData()
sl@0
  1415
		//to add the new dll
sl@0
  1416
		TDll* dllList = drive->iDllList;
sl@0
  1417
		CDllData* olddll = (*dllList)[index];
sl@0
  1418
		//remove 'old'implementations from iInterfaceImplIndex
sl@0
  1419
 		RemoveFromIndexL(olddll);
sl@0
  1420
 		//now remove old dll from the array and registry
sl@0
  1421
 		dllList->Remove(index);
sl@0
  1422
   		delete olddll;
sl@0
  1423
   		AddDllDataL(aDrive,aFoundDriveIndex,aDllData);
sl@0
  1424
   		drive->iRegistryChanged = ETrue;
sl@0
  1425
sl@0
  1426
		}
sl@0
  1427
	if (index == KErrNotFound)
sl@0
  1428
	    {
sl@0
  1429
        __ECOM_TRACE("ECOM: PANIC in CRegistryData::UpdateDllDataL(), expected DLL data missing from ECOM registry");
sl@0
  1430
	    __ASSERT_DEBUG(EFalse, User::Panic (KEComServerPanicCategory, EEComPanic_CRegistryData_UpdateDllDataL_DllRegistrationMissing));
sl@0
  1431
	    }
sl@0
  1432
	}
sl@0
  1433
sl@0
  1434
/**
sl@0
  1435
The cue that newly discovered Dlls are about to be registered, 
sl@0
  1436
and, therefore, the registry index will be out of date until 
sl@0
  1437
DiscoveriesCompleteL is called.
sl@0
  1438
@pre 			CRegistrar is fully constructed
sl@0
  1439
@post			iCurrentlyDiscovering is set and the index list is cleared.
sl@0
  1440
*/
sl@0
  1441
void CRegistryData::DiscoveriesBeginning() const
sl@0
  1442
	{
sl@0
  1443
	iCurrentlyDiscovering = ETrue;
sl@0
  1444
	// The registry is about to start changing so the index is now invalid
sl@0
  1445
	}
sl@0
  1446
sl@0
  1447
/**
sl@0
  1448
The cue to rebuild the registry indexes
sl@0
  1449
@leave			KErrNoMemory
sl@0
  1450
@see 			CStore
sl@0
  1451
@param			aSuccessful Indicates whether discovery completed fully or not
sl@0
  1452
@param			aProcessingType indicates the type of processing for plugins
sl@0
  1453
				for ensuring that plugins are not processed multiple times
sl@0
  1454
				during start-up phase
sl@0
  1455
@param			aHasRegistryChanged The flag to indicate whether registry data has been changed 
sl@0
  1456
@pre 			CRegistrar is fully constructed
sl@0
  1457
@post			The internal access indexes have been rebuilt
sl@0
  1458
*/
sl@0
  1459
void CRegistryData::DiscoveriesCompleteL(TBool aSuccessful, TPluginProcessingTypeIdentifier aProcessingType, TBool& aHasRegistryChanged)
sl@0
  1460
	{
sl@0
  1461
	iCurrentlyDiscovering = EFalse;
sl@0
  1462
	if (!aSuccessful)
sl@0
  1463
		{
sl@0
  1464
		return;
sl@0
  1465
		}
sl@0
  1466
	//if the rediscovery is a result of iLanguageChanged, we should reset it to false
sl@0
  1467
	iLanguageChanged=EFalse;		
sl@0
  1468
	// Warning: this method may be called from the CEComServer destructor.
sl@0
  1469
	// E.g. ecomservertest running OOM test on CEComServer::NewLC.
sl@0
  1470
	// The call stack: 
sl@0
  1471
	// ~CEComServer delete iRegistrar, ~CRegistrar delete iDiscoverer,
sl@0
  1472
	// ~CDiscoverer calls CRegistrar::DiscoveriesComplete,
sl@0
  1473
	// then here.
sl@0
  1474
    // Hence if ! aSuccessful do not validate.
sl@0
  1475
sl@0
  1476
	if(aProcessingType != EPluginProcessingTypeCriticalOnly)
sl@0
  1477
		{
sl@0
  1478
		ValidateRegistryL();
sl@0
  1479
		}
sl@0
  1480
	if(aProcessingType==EPluginProcessingTypeNonCriticalOnly || aProcessingType==EPluginProcessingTypeAll)
sl@0
  1481
		{
sl@0
  1482
		//first find whether there is any registry data changed
sl@0
  1483
		TInt regCount=iRegistrations->Count();
sl@0
  1484
		for(TInt i = 0; i <regCount; i++)
sl@0
  1485
			{
sl@0
  1486
			if ((*iRegistrations)[i]->iRegistryChanged)
sl@0
  1487
				{
sl@0
  1488
	 			aHasRegistryChanged = ETrue;
sl@0
  1489
	 			//now having cached that the registry has changed we need
sl@0
  1490
	 			//to reset this flag in ALL the CDriveData, note that it is
sl@0
  1491
	 			//possible that more than one drive will have the flag set
sl@0
  1492
	 			(*iRegistrations)[i]->iRegistryChanged=EFalse;
sl@0
  1493
				}
sl@0
  1494
			}
sl@0
  1495
			//Also indicate registry change if any drives have been removed
sl@0
  1496
			aHasRegistryChanged |= (iRemovedDrives != 0);
sl@0
  1497
			iRemovedDrives = 0;
sl@0
  1498
		}
sl@0
  1499
	}
sl@0
  1500
sl@0
  1501
/**
sl@0
  1502
	@fn 			SetDiscoveryFlag(const TDriveUnit aDrive)
sl@0
  1503
	Intended Usage	: Set the flag to indicate the drive has change(s)
sl@0
  1504
	Error Condition	: None.
sl@0
  1505
	@param 			aDrive The identifier of the drive changed.
sl@0
  1506
	@pre 			The CRegistrar must be fully constructed
sl@0
  1507
	@post			The flag is set.				 
sl@0
  1508
*/
sl@0
  1509
sl@0
  1510
void CRegistryData::SetDiscoveryFlagL(const TDriveUnit& aDriveUnit)
sl@0
  1511
	{
sl@0
  1512
	CDriveData* drive = NULL;
sl@0
  1513
	TInt driveIndex = FindDriveL(aDriveUnit, drive);
sl@0
  1514
	if(driveIndex != KErrNotFound)
sl@0
  1515
		{
sl@0
  1516
		drive->iDriveChanged = ETrue;
sl@0
  1517
		}
sl@0
  1518
	
sl@0
  1519
	}
sl@0
  1520
/**
sl@0
  1521
Indicates whether the registry index is currently valid.  The 
sl@0
  1522
index will not be valid if discoveries are currently taking place.
sl@0
  1523
If the index is not currently valid then calls to 
sl@0
  1524
ListImplementationsL() cannot be serviced.
sl@0
  1525
@return			ETrue if the index is currently valid, EFalse otherwise.
sl@0
  1526
@pre 			CRegistrar is fully constructed
sl@0
  1527
*/
sl@0
  1528
TBool CRegistryData::IndexValid() const
sl@0
  1529
	{
sl@0
  1530
	return !iCurrentlyDiscovering;
sl@0
  1531
	}
sl@0
  1532
sl@0
  1533
/**
sl@0
  1534
@param			aCapabilitySet A capability set
sl@0
  1535
@param			aImplUid The Uid of the implementation for which info is required
sl@0
  1536
@param			aInterfaceUid The uid of the interface associated with aImplUid to find or less than
sl@0
  1537
				0 if uid is unknown.
sl@0
  1538
@param			aEntry Output parameter which will contain the dll information
sl@0
  1539
@param			aImplInfo An output parameter. If the call succeeds it will point to the found implementation information,
sl@0
  1540
				NULL otherwise.
sl@0
  1541
@param			aIsOnRWDrive an output parameter. If the call is successful, this will
sl@0
  1542
				be set to ETrue if the implementation is on RW drive. EFalse if the
sl@0
  1543
				implementation is on ReadOnly drive.
sl@0
  1544
@return			KErrNone if the call succeeds, KErrNotFound - no implementation found, 
sl@0
  1545
				KErrPermissionDenied - the caller has not enough capabilities to load the plugin.
sl@0
  1546
@pre			CRegistrar is fully constructed
sl@0
  1547
*/
sl@0
  1548
TInt CRegistryData::GetImplementationDllInfoForServer(
sl@0
  1549
	const TCapabilitySet& /*aCapabilitySet*/,
sl@0
  1550
	const TUid aImplUid,
sl@0
  1551
	const TUid aInterfaceUid,	
sl@0
  1552
	TEntry& aEntry,
sl@0
  1553
	CImplementationInformation*& aImplInfo,
sl@0
  1554
	TBool& aIsOnRWDrive) const
sl@0
  1555
	{
sl@0
  1556
	aImplInfo = NULL;
sl@0
  1557
	CImplementationData* implData;
sl@0
  1558
	
sl@0
  1559
	TInt res = FindImplementation(aImplUid, aInterfaceUid, implData);
sl@0
  1560
	if (KErrNone != res)
sl@0
  1561
		{
sl@0
  1562
		return res;
sl@0
  1563
		}
sl@0
  1564
	
sl@0
  1565
	aImplInfo = implData->iImplInfo;
sl@0
  1566
	const CDllData* dll = implData->iParent->iParent;
sl@0
  1567
	dll->PopulateAnEntry(aEntry);
sl@0
  1568
sl@0
  1569
	TEComCachedDriveInfoIterator iter(*iCachedDriveInfo);
sl@0
  1570
	if (! iter.SetPos(dll->iParent->iDrive))
sl@0
  1571
		{
sl@0
  1572
		res = KErrNotFound;
sl@0
  1573
		}
sl@0
  1574
	else
sl@0
  1575
		{
sl@0
  1576
		aIsOnRWDrive = iter.DriveIsWritable();
sl@0
  1577
		}
sl@0
  1578
	return res;
sl@0
  1579
	}
sl@0
  1580
sl@0
  1581
/**
sl@0
  1582
@param			aClientRequest A client request
sl@0
  1583
@param			aImplUid The Uid of the implementation for which info is required
sl@0
  1584
@param			aInterfaceUid The uid of the interface associated with aImplUid to find or less than
sl@0
  1585
				0 if uid is unknown.
sl@0
  1586
@param			aEntry Output parameter which will contain the dll information
sl@0
  1587
@param			aImplInfo An output parameter. If the call succeeds it will point to the found implementation information,
sl@0
  1588
				NULL otherwise.
sl@0
  1589
@param			aSecurityCheckNeeded The bool value to identify whether the security check is needed here. The default value is false.				
sl@0
  1590
@return			KErrNone if the call succeeds, KErrNotFound - no implementation found,
sl@0
  1591
				KErrPermissionDenied - the caller has not enough capabilities to load the plugin.
sl@0
  1592
@pre			CRegistrar is fully constructed
sl@0
  1593
*/
sl@0
  1594
TInt CRegistryData::GetImplementationDllInfoForClientL(
sl@0
  1595
	const TClientRequest& aClientRequest,
sl@0
  1596
	const TUid aImplUid,
sl@0
  1597
	const TUid aInterfaceUid,	
sl@0
  1598
	TEntry& aEntry,
sl@0
  1599
	CImplementationInformation*& aImplInfo,
sl@0
  1600
	TBool aSecurityCheckNeeded)const
sl@0
  1601
	{
sl@0
  1602
	aImplInfo = NULL;
sl@0
  1603
	TInt res = KErrNotFound;
sl@0
  1604
	CImplementationData* implData;
sl@0
  1605
	if (!aSecurityCheckNeeded)
sl@0
  1606
		{
sl@0
  1607
		//The security check has been done already. All the invalid dll have been removed.
sl@0
  1608
		res = FindImplementation(aImplUid, aInterfaceUid, implData);
sl@0
  1609
		if (KErrNone != res)
sl@0
  1610
			{
sl@0
  1611
			return res;
sl@0
  1612
			}
sl@0
  1613
		CDllData* dll = implData->iParent->iParent;
sl@0
  1614
		dll->PopulateAnEntry(aEntry);
sl@0
  1615
		aImplInfo = implData->iImplInfo;
sl@0
  1616
		}
sl@0
  1617
	else
sl@0
  1618
		{
sl@0
  1619
		TBool securityPassed = EFalse;
sl@0
  1620
		while(!securityPassed) // go out of this loop either Security check is passed or no DLL found.
sl@0
  1621
			{
sl@0
  1622
			res = FindImplementation(aImplUid, aInterfaceUid, implData);
sl@0
  1623
			if (KErrNone != res)
sl@0
  1624
				{
sl@0
  1625
				return res;
sl@0
  1626
				}
sl@0
  1627
			CDllData* dll = implData->iParent->iParent;
sl@0
  1628
			// security check is deferred to here.
sl@0
  1629
			securityPassed = dll->ProcessSecurityCheckL();
sl@0
  1630
			if(securityPassed)
sl@0
  1631
				{
sl@0
  1632
				if (!aClientRequest.CheckCapability(dll->iCapSet, *(implData->iImplInfo)))
sl@0
  1633
					{
sl@0
  1634
					return KErrPermissionDenied;
sl@0
  1635
					}
sl@0
  1636
				dll->PopulateAnEntry(aEntry);
sl@0
  1637
				aImplInfo = implData->iImplInfo;
sl@0
  1638
				}
sl@0
  1639
			else
sl@0
  1640
				{
sl@0
  1641
				// remove the implementations of the DLL from iInterfaceImplIndex
sl@0
  1642
				// and remove the DLL from its parent DLL list.
sl@0
  1643
				DeleteDllL(dll);
sl@0
  1644
				delete dll;
sl@0
  1645
				dll = NULL;
sl@0
  1646
				}
sl@0
  1647
			}
sl@0
  1648
		}
sl@0
  1649
	return res;
sl@0
  1650
	}
sl@0
  1651
sl@0
  1652
/**
sl@0
  1653
This method removes all implementations of This Dll from the iInterfaceImplIndex,
sl@0
  1654
then delete this Dll from its parent Dll list.
sl@0
  1655
@param	aDllData the Dll to be cleaned.
sl@0
  1656
@pre	CRegistrar is fully constructed	
sl@0
  1657
*/
sl@0
  1658
void CRegistryData::DeleteDllL(CDllData* aDllData) const
sl@0
  1659
	{
sl@0
  1660
	// find the index of the passed aDllData in its parent's Dll list.
sl@0
  1661
	TInt index = aDllData->iParent->FindDllIndex(aDllData->iDllEntry->GetThirdUid());
sl@0
  1662
	
sl@0
  1663
	//removes all implementations of This Dll from the iInterfaceImplIndex
sl@0
  1664
	RemoveFromIndexL(aDllData);
sl@0
  1665
	if(index != KErrNotFound)
sl@0
  1666
		{
sl@0
  1667
		//now remove the dll from the array and registry
sl@0
  1668
		TDll* dllList = aDllData->iParent->iDllList;
sl@0
  1669
		dllList->Remove(index);
sl@0
  1670
		}
sl@0
  1671
	}
sl@0
  1672
sl@0
  1673
sl@0
  1674
/**
sl@0
  1675
Indicates whether the language downgrade path has changed.
sl@0
  1676
This means we need to call NearestLanguageFile again
sl@0
  1677
@return			ETrue if the language downgrade path has changed, EFalse otherwise.
sl@0
  1678
@pre 			CRegistrar is fully constructed
sl@0
  1679
*/
sl@0
  1680
TBool CRegistryData::HasLanguageChanged() const
sl@0
  1681
	{
sl@0
  1682
	return iLanguageChanged;
sl@0
  1683
	}
sl@0
  1684
sl@0
  1685
	
sl@0
  1686
/**
sl@0
  1687
@param 			aFs A handle to a connected file server.
sl@0
  1688
*/
sl@0
  1689
CRegistryData::CRegistryData(RFs& aFs,
sl@0
  1690
							 TInt aInterfaceImplIndexGranularity,
sl@0
  1691
							 TInt aImplIndexGranularity) :
sl@0
  1692
	CBase(), 
sl@0
  1693
	iFs(aFs),
sl@0
  1694
	iInterfaceImplIndex(aInterfaceImplIndexGranularity),
sl@0
  1695
	iImplIndex(aImplIndexGranularity)
sl@0
  1696
	{
sl@0
  1697
	}
sl@0
  1698
sl@0
  1699
sl@0
  1700
/**
sl@0
  1701
Completes the safe construction of the CRegistryData object.
sl@0
  1702
@leave 			KErrNoMemory.
sl@0
  1703
@pre 			This object is constructed
sl@0
  1704
@post			This object is fully initialized
sl@0
  1705
*/
sl@0
  1706
void CRegistryData::ConstructL()
sl@0
  1707
	{
sl@0
  1708
	iCachedDriveInfo = CEComCachedDriveInfo::NewL(iFs);
sl@0
  1709
sl@0
  1710
	// Construction of the empty registration data structure here
sl@0
  1711
	iRegistrations = new(ELeave)TRegistration;
sl@0
  1712
	iSystemDrive=iFs.GetSystemDrive();	
sl@0
  1713
	// during construction we always need to call this function in
sl@0
  1714
	// order to initialise and store the language settings at boot time
sl@0
  1715
	TRAPD(err,iLanguageChanged=RDowngradePath::HasChangedL(iFs))
sl@0
  1716
	if (err==KErrNoMemory)
sl@0
  1717
		{
sl@0
  1718
		User::LeaveNoMemory();
sl@0
  1719
		}
sl@0
  1720
	}
sl@0
  1721
sl@0
  1722
/**
sl@0
  1723
To find the index entry for aImplementationUid.
sl@0
  1724
@return			The index of the item or KErrIndexEntryNotFound. 
sl@0
  1725
@pre 			This object is fully constructed
sl@0
  1726
*/
sl@0
  1727
TInt CRegistryData::IndexedFind(TUid aInterfaceUid) const
sl@0
  1728
	{
sl@0
  1729
	// Find the correct implementation
sl@0
  1730
	TInterfaceIndex key;
sl@0
  1731
	key.iInterfaceUid = aInterfaceUid;
sl@0
  1732
	return iInterfaceImplIndex.FindInOrder(key,TLinearOrder<TInterfaceIndex>(TInterfaceStruct::CompareInfUid));
sl@0
  1733
	}
sl@0
  1734
sl@0
  1735
/**
sl@0
  1736
Used by a TIdentityRelation to decide if two CDriveData match.
sl@0
  1737
@return 		ETrue if the TDriveUnit inside the indexes match.
sl@0
  1738
@param			aIndexOne The first CDriveData to compare
sl@0
  1739
@param			aIndexTwo The second CDriveData to compare
sl@0
  1740
*/
sl@0
  1741
TBool CRegistryData::MatchOnDrive(const CRegistryData::CDriveData& aIndexOne, 
sl@0
  1742
								  const CRegistryData::CDriveData& aIndexTwo)
sl@0
  1743
	{
sl@0
  1744
	return aIndexOne.iDrive == aIndexTwo.iDrive;
sl@0
  1745
	}
sl@0
  1746
sl@0
  1747
TInt CRegistryData::FindDriveL(const TDriveUnit& aDrive,
sl@0
  1748
							   CRegistryData::CDriveData*& aDriveData)const 
sl@0
  1749
	{
sl@0
  1750
	TInt index = KErrNotFound;
sl@0
  1751
sl@0
  1752
	// Set up the find parameters
sl@0
  1753
	TIdentityRelation<CRegistryData::CDriveData> identity(MatchOnDrive);
sl@0
  1754
    CDriveData* driveToMatch = CDriveData::NewLC(aDrive,const_cast<CRegistryData*>(this));
sl@0
  1755
sl@0
  1756
	index = iRegistrations->Find(driveToMatch, identity);
sl@0
  1757
	
sl@0
  1758
	CleanupStack::PopAndDestroy(driveToMatch);
sl@0
  1759
sl@0
  1760
	if(index != KErrNotFound)
sl@0
  1761
		aDriveData = (*iRegistrations)[index];
sl@0
  1762
	else
sl@0
  1763
		aDriveData = NULL;
sl@0
  1764
sl@0
  1765
	return index;
sl@0
  1766
	}
sl@0
  1767
sl@0
  1768
/**
sl@0
  1769
Finds the instance of CImplementationData for given impl uid and interface uid if known.
sl@0
  1770
@return			KErrNone if impl found, otherwise KErrNotFound.
sl@0
  1771
@param			aImplUid The uid of the impl to find
sl@0
  1772
@param			aInterfaceUid The uid of the interface associated with the impl to find or less than
sl@0
  1773
				0 if uid is unknown
sl@0
  1774
@param			aImplData Set to the instance CImplementationData found in the index or if
sl@0
  1775
				not found set to NULL. This argument is always overwritten.
sl@0
  1776
*/
sl@0
  1777
TInt CRegistryData::FindImplementation(const TUid aImplUid, const TUid aInterfaceUid, 
sl@0
  1778
												CImplementationData*& aImplData) const
sl@0
  1779
	{
sl@0
  1780
	aImplData = NULL;
sl@0
  1781
	TInt i;
sl@0
  1782
	// Index is kept up to date even when discoveries are occurring
sl@0
  1783
	// therefore always search index for implementation
sl@0
  1784
sl@0
  1785
	// if aInterfaceUid is non zero use it to find position in index list
sl@0
  1786
	if(aInterfaceUid.iUid != 0)
sl@0
  1787
		{
sl@0
  1788
		i = IndexedFind(aInterfaceUid);
sl@0
  1789
		if(i == KErrNotFound)
sl@0
  1790
			{
sl@0
  1791
			return KErrNotFound;
sl@0
  1792
			}
sl@0
  1793
		
sl@0
  1794
		TInterfaceIndex listItem = iInterfaceImplIndex[i];
sl@0
  1795
		i = listItem.iImpData.FindInOrder(aImplUid,
sl@0
  1796
			TImplStruct::CompareUidAgainstImplStruct);
sl@0
  1797
		if (i >= 0)
sl@0
  1798
			{
sl@0
  1799
			aImplData = listItem.iImpData[i].iCurrentImpl;
sl@0
  1800
			return KErrNone;
sl@0
  1801
			}
sl@0
  1802
		}
sl@0
  1803
	else // not given I/F UID. Use the iImplIndex.
sl@0
  1804
		{
sl@0
  1805
		// NB: 1. Impl UID should be globally unique. It is an error if impls
sl@0
  1806
		// have same Impl UID but different i/f UIDs. It means one of the 
sl@0
  1807
		// plug-in supplier made an error in the .RSS file or it could be a
sl@0
  1808
		// deliberate attach. But to maintain backward compatibility, ECOM
sl@0
  1809
		// allows this error.
sl@0
  1810
		// (Multiple impls can have same i/f UID and same Impl UID. That is
sl@0
  1811
		//  the upgrade situation and only one of them get stored in iImplIndex.)
sl@0
  1812
sl@0
  1813
		// 2. Entries in iImplIndex are ordered by Impl UID, and if duplicated,
sl@0
  1814
		// ordered by i/f UID. Here i/f UID is wild card. The situation is 
sl@0
  1815
		// analogous to the array was built with InsertInOrderAllowRepeats.
sl@0
  1816
		// RPointerArray::SpecificFindInOrder is for ordered search in array
sl@0
  1817
		// with duplicates. Note: it is very expensive to instantiate
sl@0
  1818
		// a CImplementationData* just for binary search. Hence reinterpret
sl@0
  1819
		// cast the TUid* as CImplementationData*. CompareUidAgainstImplData
sl@0
  1820
		// knows to cast the first argument back to TUid.
sl@0
  1821
		i = iImplIndex.SpecificFindInOrder(
sl@0
  1822
			reinterpret_cast<const CImplementationData*>(&aImplUid), 
sl@0
  1823
			TLinearOrder<CImplementationData>(CImplementationData::CompareUidAgainstImplData),
sl@0
  1824
			EArrayFindMode_First);
sl@0
  1825
		if (i == KErrNotFound)
sl@0
  1826
			{
sl@0
  1827
			return KErrNotFound;
sl@0
  1828
			}
sl@0
  1829
sl@0
  1830
		aImplData = iImplIndex[i];
sl@0
  1831
sl@0
  1832
		// If duplicates exist, they are sorted according to i/f UID
sl@0
  1833
		// but we cannot take advantage of this order. To miniize risk
sl@0
  1834
		// of malware using duplicate Impl UID as DOS attack, ECOM applies
sl@0
  1835
		// the rule that ROMBased plug-in preferred over non-ROMBased,
sl@0
  1836
		// higher drive letter preferred over lower drive letter, and 
sl@0
  1837
		// lastly lower i/f UID preferred over higher i/f UID. Must visit
sl@0
  1838
		// every duplicate to compare their ROMBasedness and driver letter
sl@0
  1839
sl@0
  1840
		TInt count = iImplIndex.Count();
sl@0
  1841
		for (TInt j = i + 1; j < count; j++)
sl@0
  1842
			{
sl@0
  1843
			if ( iImplIndex[j]->iImplInfo->ImplementationUid().iUid != aImplUid.iUid )
sl@0
  1844
				{
sl@0
  1845
				break;
sl@0
  1846
				}
sl@0
  1847
sl@0
  1848
			aImplData = SelectDuplicatedImpl(aImplData, iImplIndex[j]);
sl@0
  1849
			}
sl@0
  1850
sl@0
  1851
		return KErrNone;
sl@0
  1852
		}
sl@0
  1853
		
sl@0
  1854
	return KErrNotFound;
sl@0
  1855
	}
sl@0
  1856
sl@0
  1857
/**
sl@0
  1858
This function helps FindImplementation() to decide which of two
sl@0
  1859
implementations with duplicated implementation UID to choose.
sl@0
  1860
The selection rules are:
sl@0
  1861
1. ROM based plugins > non-ROM based.
sl@0
  1862
2. Higher drive letter > lower drive letter
sl@0
  1863
3. Lower I/F UID > Higher I/F UID (for backward compatibility reason)
sl@0
  1864
sl@0
  1865
@param aImpl1 - one of the two implementations to compare.
sl@0
  1866
@param aImpl2 - the other implementation to compare.
sl@0
  1867
@return the preferred implementation.
sl@0
  1868
*/
sl@0
  1869
CRegistryData::CImplementationData* CRegistryData::SelectDuplicatedImpl(
sl@0
  1870
	const CImplementationData* aImpl1, 
sl@0
  1871
	const CImplementationData* aImpl2) const
sl@0
  1872
	{
sl@0
  1873
sl@0
  1874
#ifdef ECOM_TRACE
sl@0
  1875
	TPtrC oldName = aImpl1->iImplInfo->DisplayName().Left(60);
sl@0
  1876
	TPtrC newName = aImpl2->iImplInfo->DisplayName().Left(60);
sl@0
  1877
	TPtrC oldDll = aImpl1->iParent->iParent->iDllEntry->GetName();
sl@0
  1878
	TPtrC newDll = aImpl2->iParent->iParent->iDllEntry->GetName();
sl@0
  1879
#endif
sl@0
  1880
sl@0
  1881
	const TInt KRomBasedFactor = 0x100;
sl@0
  1882
sl@0
  1883
	TInt drive1 = aImpl1->iImplInfo->Drive();
sl@0
  1884
	if (aImpl1->iImplInfo->RomBased())
sl@0
  1885
		{
sl@0
  1886
		drive1 |= KRomBasedFactor;
sl@0
  1887
		}
sl@0
  1888
sl@0
  1889
	TInt drive2 = aImpl2->iImplInfo->Drive();
sl@0
  1890
	if (aImpl2->iImplInfo->RomBased())
sl@0
  1891
		{
sl@0
  1892
		drive2 |= KRomBasedFactor;
sl@0
  1893
		}
sl@0
  1894
sl@0
  1895
	if (drive1 > drive2)
sl@0
  1896
		{
sl@0
  1897
#ifdef ECOM_TRACE
sl@0
  1898
		if ((drive1 & KRomBasedFactor) && !(drive2 & KRomBasedFactor))
sl@0
  1899
			{
sl@0
  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);
sl@0
  1901
			}
sl@0
  1902
		else
sl@0
  1903
			{
sl@0
  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);
sl@0
  1905
			}
sl@0
  1906
#endif
sl@0
  1907
sl@0
  1908
		return const_cast<CImplementationData*>(aImpl1);
sl@0
  1909
		}
sl@0
  1910
	else if (drive1 < drive2)
sl@0
  1911
		{
sl@0
  1912
#ifdef ECOM_TRACE
sl@0
  1913
		if ((drive2 & KRomBasedFactor) && !(drive1 & KRomBasedFactor))
sl@0
  1914
			{
sl@0
  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);
sl@0
  1916
			}
sl@0
  1917
		else
sl@0
  1918
			{
sl@0
  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);
sl@0
  1920
			}
sl@0
  1921
#endif
sl@0
  1922
sl@0
  1923
		return const_cast<CImplementationData*>(aImpl2);
sl@0
  1924
		}
sl@0
  1925
	// They are on the same drive. Choose the one with lower I/F UID.
sl@0
  1926
	else if (aImpl2->iParent->iInterfaceUid.iUid < aImpl1->iParent->iInterfaceUid.iUid) 
sl@0
  1927
		{
sl@0
  1928
#ifdef ECOM_TRACE
sl@0
  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);
sl@0
  1930
#endif
sl@0
  1931
sl@0
  1932
		return const_cast<CImplementationData*>(aImpl2);
sl@0
  1933
		}
sl@0
  1934
sl@0
  1935
#ifdef ECOM_TRACE
sl@0
  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);
sl@0
  1937
#endif
sl@0
  1938
sl@0
  1939
	return const_cast<CImplementationData*>(aImpl1);
sl@0
  1940
	}
sl@0
  1941
sl@0
  1942
/**
sl@0
  1943
Checks each entry in the registry to ensure that both the RSC file and the 
sl@0
  1944
corresponding dll exist.  If not the dll branch is removed
sl@0
  1945
from the tree.  If a drive branch is found which contains no dlls
sl@0
  1946
it is also removed.
sl@0
  1947
@pre 			This object is fully constructed.
sl@0
  1948
@post			Any out of date registry entries are removed from the tree.
sl@0
  1949
*/
sl@0
  1950
void CRegistryData::ValidateRegistryL()
sl@0
  1951
	{
sl@0
  1952
	TInt driveIndex = iRegistrations->Count();
sl@0
  1953
	TFileName rscFileName;
sl@0
  1954
	while(driveIndex > 0)
sl@0
  1955
		{
sl@0
  1956
		--driveIndex;
sl@0
  1957
		CDriveData* drive = (*iRegistrations)[driveIndex];
sl@0
  1958
		
sl@0
  1959
		if( !iCachedDriveInfo->DriveIsReadOnlyInternalL(drive->iDrive) &&
sl@0
  1960
			drive->iDriveChanged)
sl@0
  1961
			{
sl@0
  1962
			TInt dllIndex = drive->iDllList->Count();	
sl@0
  1963
			while(dllIndex > 0)
sl@0
  1964
				{
sl@0
  1965
				--dllIndex;
sl@0
  1966
				CDllData* dll = (*drive->iDllList)[dllIndex];
sl@0
  1967
				//reset the buffer first			
sl@0
  1968
				rscFileName.Zero();
sl@0
  1969
				rscFileName.Append(dll->iParent->iDrive.Name());
sl@0
  1970
				rscFileName.Append(KEComResourceFilePath);
sl@0
  1971
				rscFileName.Append(dll->iDllEntry->GetName().Left(dll->iDllEntry->GetName().Length()-4));
sl@0
  1972
				if(dll->iRscFileExtension)
sl@0
  1973
					{
sl@0
  1974
					rscFileName.Append(dll->iRscFileExtension->Des());
sl@0
  1975
					}
sl@0
  1976
			
sl@0
  1977
					// check the existence of RSC file in resource\plugins\ directory.
sl@0
  1978
					// RSC filename is already a full name here.
sl@0
  1979
				TBool rscFileExistence = BaflUtils::FileExists(iFs, rscFileName);
sl@0
  1980
sl@0
  1981
  				//save the security info, this will both check existence of the dll
sl@0
  1982
  				//and cache the information for future use
sl@0
  1983
  				TBool success=dll->SaveSecurityInfoL();
sl@0
  1984
				// If this dll is not found or the corresponding RSC file is not found, then remove it from the registry
sl@0
  1985
				if(!success || !rscFileExistence)
sl@0
  1986
					{
sl@0
  1987
					//remove all implementations of this dll from iInterfaceImplIndex
sl@0
  1988
					RemoveFromIndexL(dll);
sl@0
  1989
					drive->iDllList->Remove(dllIndex);
sl@0
  1990
					//set flag to indicate registry data has been changed
sl@0
  1991
					drive->iRegistryChanged = ETrue;
sl@0
  1992
					delete dll;
sl@0
  1993
					}
sl@0
  1994
				}
sl@0
  1995
			}
sl@0
  1996
		}
sl@0
  1997
	}
sl@0
  1998
sl@0
  1999
/**
sl@0
  2000
Determines whether the new implmentation should be preferred over the existing implementation.
sl@0
  2001
Validates later version implementations from R/W drives.
sl@0
  2002
@param		aOldImpl The array to append to
sl@0
  2003
@param		aNewImpl The item to append
sl@0
  2004
@param		aLegitimateImpl Flag, indicating whether current implementation is secure
sl@0
  2005
@return		The preferred implementation
sl@0
  2006
@pre		This object is fully constructed.
sl@0
  2007
*/
sl@0
  2008
CRegistryData::CImplementationData* CRegistryData::SelectPreferredImplL(CImplementationData* aOldImpl,
sl@0
  2009
																CImplementationData* aNewImpl,
sl@0
  2010
																TBool& aLegitimateImpl,
sl@0
  2011
																TBool aCheckIsNeeded) const
sl@0
  2012
	{
sl@0
  2013
	aLegitimateImpl = ETrue;
sl@0
  2014
sl@0
  2015
	TBool newIsRomOnly = aNewImpl->iImplInfo->RomOnly();
sl@0
  2016
	TBool oldIsRomOnly = aOldImpl->iImplInfo->RomOnly();
sl@0
  2017
		
sl@0
  2018
	/* 	In addition to selecting the highest version of an 
sl@0
  2019
		implementation this check takes care of the following special 
sl@0
  2020
		cases:
sl@0
  2021
		1. Ensure that a higher-versioned RAM-based implementation 
sl@0
  2022
		   cannot override a ROM-based version
sl@0
  2023
		2. Allows for the case where there are two versions of the 
sl@0
  2024
		   same ROM-only implementation on ROM,	thus ensuring the 
sl@0
  2025
		   higher of the two versions is used.
sl@0
  2026
	*/
sl@0
  2027
	if(newIsRomOnly && !oldIsRomOnly)
sl@0
  2028
		{
sl@0
  2029
		return aNewImpl;
sl@0
  2030
		}
sl@0
  2031
	else if((newIsRomOnly && oldIsRomOnly))
sl@0
  2032
		{
sl@0
  2033
		if(aNewImpl->iImplInfo->Version() > aOldImpl->iImplInfo->Version())
sl@0
  2034
			{
sl@0
  2035
			return aNewImpl;
sl@0
  2036
			}
sl@0
  2037
		else if(aNewImpl->iImplInfo->Version() == aOldImpl->iImplInfo->Version())
sl@0
  2038
			{
sl@0
  2039
			//any drive from Y-A has higher priority than Z drive
sl@0
  2040
            //any drive with a letter alphabetically greater has higher priority
sl@0
  2041
            if((aNewImpl->iImplInfo->Drive() != EDriveZ) &&
sl@0
  2042
                (aOldImpl->iImplInfo->Drive() == EDriveZ || aNewImpl->iImplInfo->Drive() > aOldImpl->iImplInfo->Drive()))
sl@0
  2043
                {
sl@0
  2044
				return aNewImpl;
sl@0
  2045
				}
sl@0
  2046
			}
sl@0
  2047
		}
sl@0
  2048
	//all new implementations which are not flagged'RomOnly'
sl@0
  2049
	else if(!newIsRomOnly && !oldIsRomOnly)
sl@0
  2050
		{
sl@0
  2051
		TBool newIsReadOnly = iCachedDriveInfo->DriveIsReadOnlyInternalL(aNewImpl->iParent->iParent->iParent->iDrive);
sl@0
  2052
		TBool oldIsReadOnly = iCachedDriveInfo->DriveIsReadOnlyInternalL(aOldImpl->iParent->iParent->iParent->iDrive);
sl@0
  2053
			
sl@0
  2054
		if(aNewImpl->iImplInfo->Version() > aOldImpl->iImplInfo->Version())
sl@0
  2055
			{
sl@0
  2056
			//filter for implementations on R/W drive trying to upgrade implementations on R/O drive
sl@0
  2057
			if((!newIsReadOnly && oldIsReadOnly) && aCheckIsNeeded)
sl@0
  2058
				{						
sl@0
  2059
				//test if a later implementation version on R/W drive has the same filename as the current one before upgrading.
sl@0
  2060
				//If not,...
sl@0
  2061
				if(aOldImpl->iParent->iParent->iDllEntry->GetName().CompareF(aNewImpl->iParent->iParent->iDllEntry->GetName()) == 0)
sl@0
  2062
					{
sl@0
  2063
					return aNewImpl;
sl@0
  2064
					}
sl@0
  2065
				else
sl@0
  2066
					{
sl@0
  2067
					aLegitimateImpl = EFalse; //not secure
sl@0
  2068
					return aOldImpl;
sl@0
  2069
					}
sl@0
  2070
				}
sl@0
  2071
			else
sl@0
  2072
				{
sl@0
  2073
				return aNewImpl;
sl@0
  2074
				}
sl@0
  2075
			}
sl@0
  2076
		else if(aNewImpl->iImplInfo->Version() == aOldImpl->iImplInfo->Version())
sl@0
  2077
			{
sl@0
  2078
			//any drive from Y-A has higher priority than Z drive OR
sl@0
  2079
			//any drive with a letter alphabetically greater has higher priority
sl@0
  2080
			if((aNewImpl->iImplInfo->Drive() != EDriveZ) &&
sl@0
  2081
                (aOldImpl->iImplInfo->Drive() == EDriveZ || aNewImpl->iImplInfo->Drive() > aOldImpl->iImplInfo->Drive()))                 
sl@0
  2082
				{
sl@0
  2083
				//filename check needs to performed on implementation from R/W drives, trying to 
sl@0
  2084
				//upgrade implementations on R/O drive    
sl@0
  2085
				if((!newIsReadOnly && oldIsReadOnly) && aCheckIsNeeded)
sl@0
  2086
					{							
sl@0
  2087
					//test if a later implementation version on R/W drive has the same filename as the current one before upgrading.
sl@0
  2088
					//If not,...
sl@0
  2089
					if(aOldImpl->iParent->iParent->iDllEntry->GetName().CompareF(aNewImpl->iParent->iParent->iDllEntry->GetName()) == 0)
sl@0
  2090
						{
sl@0
  2091
						return aNewImpl;
sl@0
  2092
						}
sl@0
  2093
					else
sl@0
  2094
						{
sl@0
  2095
						aLegitimateImpl = EFalse; //not secure
sl@0
  2096
						return aOldImpl;
sl@0
  2097
						}		
sl@0
  2098
					}
sl@0
  2099
				else
sl@0
  2100
					{
sl@0
  2101
					return aNewImpl;
sl@0
  2102
					}
sl@0
  2103
				}						
sl@0
  2104
			}
sl@0
  2105
		}
sl@0
  2106
	return aOldImpl;
sl@0
  2107
	}
sl@0
  2108
sl@0
  2109
/**
sl@0
  2110
This functions checks if a given implementation already exists in
sl@0
  2111
the indexes. If it does exist, determine if the given implementation
sl@0
  2112
should replace the existing one or not.
sl@0
  2113
@param  aIdxArray The container array of the interface to hold the implementation.
sl@0
  2114
@param  aNewImpl The implementation to filter.
sl@0
  2115
@param  aInsertMode whether aNewIMpl is a newcomer of the interface, or an 
sl@0
  2116
		update of an existing implementation, or a older version of an
sl@0
  2117
		existing implementation.
sl@0
  2118
@param  aPosOfImplInArray return the index of aNewImpl in aIdxArray,
sl@0
  2119
@param  aLegitimateImpl Flag,indicating whether current implementation is secure
sl@0
  2120
@pre    This object is fully constructed.
sl@0
  2121
@post   none
sl@0
  2122
*/
sl@0
  2123
void
sl@0
  2124
CRegistryData::FilterForLatestLegitimateImplL(TImplContainerArray& aIdxArray,
sl@0
  2125
											  CImplementationData* aNewImpl,
sl@0
  2126
											  TInsertImplMode& aInsertMode,
sl@0
  2127
											  TInt&  aPosOfImplInArray,
sl@0
  2128
											  TBool& aLegitimateImpl,
sl@0
  2129
											  TBool aCheckIsNeeded)
sl@0
  2130
	{
sl@0
  2131
	aInsertMode = EInsertImplUndefinedMode;
sl@0
  2132
	aLegitimateImpl = ETrue;
sl@0
  2133
	TImplStruct newImplStruct;
sl@0
  2134
	newImplStruct.iCurrentImpl = aNewImpl;
sl@0
  2135
sl@0
  2136
	TInt idxPos = aIdxArray.FindInOrder(newImplStruct, TLinearOrder<TImplStruct> (TImplStruct::CompareImplStructUid));
sl@0
  2137
sl@0
  2138
	aPosOfImplInArray = idxPos;
sl@0
  2139
sl@0
  2140
	if(idxPos != KErrNotFound)
sl@0
  2141
		{
sl@0
  2142
		if(aNewImpl != SelectPreferredImplL(aIdxArray[idxPos].iCurrentImpl,
sl@0
  2143
											aNewImpl,
sl@0
  2144
											aLegitimateImpl,
sl@0
  2145
											aCheckIsNeeded))
sl@0
  2146
			{
sl@0
  2147
			aInsertMode = EInsertImplAsUnusedImpl;
sl@0
  2148
			}
sl@0
  2149
		else
sl@0
  2150
			{
sl@0
  2151
			aInsertMode = EInsertImplAsUpgradeOfExistingImpl;
sl@0
  2152
			} 
sl@0
  2153
		}
sl@0
  2154
	else
sl@0
  2155
		{
sl@0
  2156
		aInsertMode = EInsertImplAsNewcomerOfInterface;
sl@0
  2157
		}
sl@0
  2158
	}
sl@0
  2159
sl@0
  2160
void CRegistryData::ResetTInterfaceIndex(TAny* aObject)
sl@0
  2161
	{
sl@0
  2162
	TInterfaceIndex* index=reinterpret_cast<TInterfaceIndex*>(aObject);
sl@0
  2163
	index->Reset();
sl@0
  2164
	}
sl@0
  2165
void CRegistryData::RemoveImplFromImplIndexCleanUp(TAny* aPtr)
sl@0
  2166
    {
sl@0
  2167
    TCleanupImplIndexEntry *aCleanup = (TCleanupImplIndexEntry*)aPtr;
sl@0
  2168
    aCleanup->iRegistryData->RemoveImplByAddrFromImplIndex(aCleanup->iImplEntry);
sl@0
  2169
    }
sl@0
  2170
sl@0
  2171
/**
sl@0
  2172
This method retrieves the data for security checks from the iInterfaceImplIndex
sl@0
  2173
@param          aImplPtr The new item to be checked
sl@0
  2174
@param			aCheckIsNeeded	Boolean indicating, whether a filename check is needed
sl@0
  2175
@pre            This object is fully constructed.
sl@0
  2176
@post           aImplPtr has been checked and added to the index, or not, depending on the 
sl@0
  2177
				outcome of the checks.
sl@0
  2178
*/
sl@0
  2179
TBool CRegistryData::InsertIntoIndexL(CImplementationData* aImplPtr, TBool aCheckIsNeeded)
sl@0
  2180
	{
sl@0
  2181
	TBool legitimateImpl = ETrue;
sl@0
  2182
sl@0
  2183
	TInterfaceIndex newIndexEl;
sl@0
  2184
	TInterfaceIndex* newElPtr;
sl@0
  2185
	//initialise
sl@0
  2186
	newIndexEl.iInterfaceUid = aImplPtr->iParent->iInterfaceUid;
sl@0
  2187
	CleanupStack::PushL(TCleanupItem(ResetTInterfaceIndex,&newIndexEl));
sl@0
  2188
	
sl@0
  2189
	TInt indexPos = iInterfaceImplIndex.FindInOrder(newIndexEl,TLinearOrder<TInterfaceIndex>(TInterfaceStruct::CompareInfUid));
sl@0
  2190
sl@0
  2191
	
sl@0
  2192
	if(indexPos!=KErrNotFound)
sl@0
  2193
		newElPtr = &(iInterfaceImplIndex[indexPos]);
sl@0
  2194
	else		
sl@0
  2195
		newElPtr = &newIndexEl;	
sl@0
  2196
sl@0
  2197
	// For each implementation make sure we only have the latest version
sl@0
  2198
	TInt implIdxInContainerArray(KErrNotFound);
sl@0
  2199
	TInsertImplMode insertMode;
sl@0
  2200
	FilterForLatestLegitimateImplL(newElPtr->iImpData,
sl@0
  2201
									aImplPtr,
sl@0
  2202
									insertMode,
sl@0
  2203
									implIdxInContainerArray,
sl@0
  2204
									legitimateImpl,
sl@0
  2205
									aCheckIsNeeded);
sl@0
  2206
	
sl@0
  2207
	InsertImplInIndexesL(insertMode, 
sl@0
  2208
						 indexPos,
sl@0
  2209
						 *newElPtr,
sl@0
  2210
						 implIdxInContainerArray,
sl@0
  2211
						 aImplPtr,
sl@0
  2212
						 legitimateImpl);
sl@0
  2213
sl@0
  2214
	CleanupStack::Pop();
sl@0
  2215
sl@0
  2216
	if (insertMode == EInsertImplAsUpgradeOfExistingImpl)
sl@0
  2217
		{
sl@0
  2218
	    TUid ImplUid = aImplPtr->iImplInfo->ImplementationUid();
sl@0
  2219
		iImplUpgradeCallBack.CallBack(ECallBackId_ImplUpgrade, &ImplUid);
sl@0
  2220
		}
sl@0
  2221
sl@0
  2222
	return legitimateImpl;
sl@0
  2223
	}
sl@0
  2224
sl@0
  2225
/** Handle inserting a CImplementationData in iInterfaceImplIndex and
sl@0
  2226
iImplIndex.
sl@0
  2227
sl@0
  2228
@param aNewImpl the implementation to add to the indexes.
sl@0
  2229
@param aInsertMode indicates whether the implementation is a newcover
sl@0
  2230
			of the interface, or is an update of existing implementation
sl@0
  2231
			or is an older version of an existing implementation.
sl@0
  2232
@param aIfPosInInterfaceImplIndex is the index of the interface in
sl@0
  2233
		iInterfaceImplIndex.
sl@0
  2234
@param aNewIfIndexEl the TInterfaceIndex object containing the implementation.
sl@0
  2235
@param aImplPosInContainerArray is the index of the implementation in
sl@0
  2236
		the iImpData member of aNewIfIndexEl.
sl@0
  2237
@param aLegitimateImpl indicate if the implementation passed security check or not.
sl@0
  2238
@leave KErrNoMemory operation fails because the system is out of memory.
sl@0
  2239
@leave KErrGeneral any non-specific programming error.
sl@0
  2240
@leave KErrAlreadyExists the indexes already have an entry with the same
sl@0
  2241
		Impl. UID and Interface UID.
sl@0
  2242
*/
sl@0
  2243
void CRegistryData::InsertImplInIndexesL(TInsertImplMode aInsertMode,
sl@0
  2244
										TInt aIfPosInInterfaceImplIndex,
sl@0
  2245
										TInterfaceIndex& aNewIfIndexEl,
sl@0
  2246
										TInt aImplPosInContainerArray,
sl@0
  2247
								   		CImplementationData* aNewImpl,
sl@0
  2248
								   		TBool aLegitimateImpl)
sl@0
  2249
	{
sl@0
  2250
	if(aInsertMode == EInsertImplUndefinedMode)
sl@0
  2251
		{
sl@0
  2252
		// Will not happen because if FilterForLatestLegitimateImplL
sl@0
  2253
		// does not leave then insertMode is set to one of the valid
sl@0
  2254
		// values. If FilterForLatestLegitimateImplL leaves ecomserver exits.
sl@0
  2255
		User::Leave(KErrGeneral);
sl@0
  2256
		}
sl@0
  2257
sl@0
  2258
	// If not a legitimate implementation, aNewImpl will be deleted.
sl@0
  2259
	// Do not add it to the two indexes.
sl@0
  2260
sl@0
  2261
	if (!aLegitimateImpl)
sl@0
  2262
		{
sl@0
  2263
		return;
sl@0
  2264
		}
sl@0
  2265
sl@0
  2266
	TImplContainerArray& implContainerArray = aNewIfIndexEl.iImpData;
sl@0
  2267
sl@0
  2268
	if(aInsertMode == EInsertImplAsNewcomerOfInterface)
sl@0
  2269
		{
sl@0
  2270
		TImplStruct newImplStruct;
sl@0
  2271
		newImplStruct.iCurrentImpl = aNewImpl;
sl@0
  2272
		implContainerArray.InsertInOrderL(newImplStruct, TLinearOrder<TImplStruct> (TImplStruct::CompareImplStructUid));
sl@0
  2273
sl@0
  2274
#ifdef _DEBUG
sl@0
  2275
		// on debug build, check for duplicated implementation UID
sl@0
  2276
		// owned by different interfaces.
sl@0
  2277
sl@0
  2278
		TInt ii = iImplIndex.SpecificFindInOrder(
sl@0
  2279
			aNewImpl,
sl@0
  2280
			CImplementationData::CompareImplUidIgnoreIfUid,
sl@0
  2281
			EArrayFindMode_Any);
sl@0
  2282
		if (ii != KErrNotFound)
sl@0
  2283
			{
sl@0
  2284
			CImplementationData* otherImpl = iImplIndex[ii];
sl@0
  2285
			TPtrC oldName = otherImpl->iImplInfo->DisplayName().Left(60);
sl@0
  2286
			TPtrC newName = aNewImpl->iImplInfo->DisplayName().Left(60);
sl@0
  2287
			__ECOM_TRACE5("** ECOM: error Impl UID %X, I/F UID %X DLL %S duplicating I/F %X DLL %S",\\ 
sl@0
  2288
				aNewImpl->iImplInfo->ImplementationUid().iUid, \\
sl@0
  2289
				aNewImpl->iParent->iInterfaceUid.iUid, &newName,\\
sl@0
  2290
				otherImpl->iParent->iInterfaceUid.iUid, &oldName);
sl@0
  2291
			}
sl@0
  2292
#endif
sl@0
  2293
		}
sl@0
  2294
	else if(aInsertMode == EInsertImplAsUpgradeOfExistingImpl)
sl@0
  2295
		{
sl@0
  2296
		TImplContainer& implContainer = implContainerArray[aImplPosInContainerArray];
sl@0
  2297
		CImplementationData* oldImpl =  implContainer.iCurrentImpl;
sl@0
  2298
		implContainer.iUnusedImpls.AppendL(oldImpl);
sl@0
  2299
		RemoveImplByAddrFromImplIndex(oldImpl); // ignore return code
sl@0
  2300
		implContainer.iCurrentImpl = aNewImpl;
sl@0
  2301
sl@0
  2302
		// We are replacing existing impl with aNewImpl.  If existing  
sl@0
  2303
		// corresponds to a ROM-based plug-in then aNewImpl is an 
sl@0
  2304
		// update and qualifies as ROM-based under current policy.
sl@0
  2305
		if(oldImpl->iImplInfo->RomBased())
sl@0
  2306
			{
sl@0
  2307
			aNewImpl->iImplInfo->SetRomBased(ETrue);
sl@0
  2308
			}
sl@0
  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()));
sl@0
  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()));
sl@0
  2311
		}
sl@0
  2312
	else if (aInsertMode == EInsertImplAsUnusedImpl)
sl@0
  2313
		{
sl@0
  2314
		TImplContainer& implContainer = implContainerArray[aImplPosInContainerArray];
sl@0
  2315
		implContainer.iUnusedImpls.AppendL(aNewImpl);
sl@0
  2316
sl@0
  2317
		CImplementationData* currImpl =  implContainer.iCurrentImpl;
sl@0
  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()));
sl@0
  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()));
sl@0
  2320
		// We are not replacing existing with aNewImpl.  However, 
sl@0
  2321
		// if aNewImpl corresponds to a ROM-based plug-in then the 
sl@0
  2322
		// existing impl is an update and qualifies as ROM-based 
sl@0
  2323
		// under current policy.
sl@0
  2324
		if(aNewImpl->iImplInfo->RomBased())
sl@0
  2325
			{
sl@0
  2326
			currImpl->iImplInfo->SetRomBased(ETrue);
sl@0
  2327
			}
sl@0
  2328
sl@0
  2329
		return; // the trailing steps not applicable to downgrade situation
sl@0
  2330
		}
sl@0
  2331
sl@0
  2332
	User::LeaveIfError( InsertImplIntoImplIndex(aNewImpl) );
sl@0
  2333
	
sl@0
  2334
	if(aIfPosInInterfaceImplIndex==KErrNotFound)
sl@0
  2335
		{		
sl@0
  2336
        TCleanupImplIndexEntry aCleanup(this, aNewImpl);
sl@0
  2337
        CleanupStack::PushL(TCleanupItem(RemoveImplFromImplIndexCleanUp,&aCleanup));
sl@0
  2338
		iInterfaceImplIndex.InsertInOrderL(aNewIfIndexEl, TLinearOrder<TInterfaceIndex>(TInterfaceStruct::CompareInfUid));
sl@0
  2339
		CleanupStack::Pop();
sl@0
  2340
		}
sl@0
  2341
	}
sl@0
  2342
sl@0
  2343
/**
sl@0
  2344
This method takes removes all implementations of a plugin from the iInterfaceImplIndex
sl@0
  2345
@param			aDllData Reference to a fully constructed CDllData object 
sl@0
  2346
@pre			This object is constructed
sl@0
  2347
@post			All implementations of the CDllData object, have been removed from the index
sl@0
  2348
*/
sl@0
  2349
void CRegistryData::RemoveFromIndexL(CDllData* aDllData) const
sl@0
  2350
	{
sl@0
  2351
	//remove 'old' implementations from iInterfaceImplIndex
sl@0
  2352
	TInt counter=0;
sl@0
  2353
	TImplStruct implStruct;
sl@0
  2354
	TInt ifListCount=aDllData->iIfList->Count();
sl@0
  2355
	while(counter < ifListCount)
sl@0
  2356
		{
sl@0
  2357
		CInterfaceData* interface = (*aDllData->iIfList)[counter];
sl@0
  2358
		TInterfaceIndex index;
sl@0
  2359
		index.iInterfaceUid = interface->iInterfaceUid;
sl@0
  2360
		
sl@0
  2361
		TInt indexPos = iInterfaceImplIndex.FindInOrder(index,TLinearOrder<TInterfaceIndex>(TInterfaceStruct::CompareInfUid));
sl@0
  2362
        TImplContainerArray* impDataArray = NULL;
sl@0
  2363
		TInt implCount=0;
sl@0
  2364
		
sl@0
  2365
		if(indexPos!=KErrNotFound)
sl@0
  2366
			{
sl@0
  2367
            impDataArray = &(iInterfaceImplIndex[indexPos].iImpData);
sl@0
  2368
            implCount=interface->iImplementations->Count();
sl@0
  2369
			for(TInt i=0; i<implCount; i++)
sl@0
  2370
				{
sl@0
  2371
				implStruct.iCurrentImpl = (*interface->iImplementations)[i];
sl@0
  2372
				TInt impPos = impDataArray->FindInOrder(implStruct, TLinearOrder<CRegistryData::TImplStruct> (TImplStruct::CompareImplStructUid));
sl@0
  2373
				if(impPos!=KErrNotFound)
sl@0
  2374
					{
sl@0
  2375
					TImplContainer& implContainer = (*impDataArray)[impPos];
sl@0
  2376
					// remove from unused list if exists
sl@0
  2377
					TInt count = implContainer.iUnusedImpls.Count();
sl@0
  2378
					while(count > 0)
sl@0
  2379
						{
sl@0
  2380
						--count;
sl@0
  2381
						if(implContainer.iUnusedImpls[count]->iParent->iParent == aDllData)
sl@0
  2382
							{
sl@0
  2383
							implContainer.iUnusedImpls.Remove(count);
sl@0
  2384
							}
sl@0
  2385
						}
sl@0
  2386
sl@0
  2387
					// update current entry
sl@0
  2388
					if(implContainer.iCurrentImpl->iParent->iParent == aDllData)
sl@0
  2389
						{
sl@0
  2390
						// do not care about the return code.
sl@0
  2391
						RemoveImplByAddrFromImplIndex(implContainer.iCurrentImpl);
sl@0
  2392
						
sl@0
  2393
						TInt implContainerUnusedImplCount=implContainer.iUnusedImpls.Count();
sl@0
  2394
						// no unused impl's therefore no rollback and remove entry
sl@0
  2395
						if(implContainerUnusedImplCount == 0)
sl@0
  2396
							{
sl@0
  2397
							implContainer.Reset();
sl@0
  2398
							impDataArray->Remove(impPos);
sl@0
  2399
							}
sl@0
  2400
						else
sl@0
  2401
							{
sl@0
  2402
							// Rollback to implementation from unused array
sl@0
  2403
							// copy first from unused array
sl@0
  2404
							TInt selectedPos = 0;
sl@0
  2405
							implContainer.iCurrentImpl = implContainer.iUnusedImpls[selectedPos];
sl@0
  2406
							
sl@0
  2407
							// now check if any others in list should be preferred
sl@0
  2408
							for(count = 1;count < implContainerUnusedImplCount; count++)
sl@0
  2409
								{
sl@0
  2410
								// determine which implementation should be used.
sl@0
  2411
								// no security check required as this will have already been done when impl was
sl@0
  2412
								// first added to unused list.
sl@0
  2413
								TBool dummLegitimateImpl;
sl@0
  2414
								implContainer.iCurrentImpl = SelectPreferredImplL(implContainer.iCurrentImpl,
sl@0
  2415
																implContainer.iUnusedImpls[count],
sl@0
  2416
																dummLegitimateImpl,
sl@0
  2417
																EFalse);
sl@0
  2418
								if(implContainer.iCurrentImpl == implContainer.iUnusedImpls[count])
sl@0
  2419
									{
sl@0
  2420
									selectedPos = count;
sl@0
  2421
									}
sl@0
  2422
								}
sl@0
  2423
							implContainer.iUnusedImpls.Remove(selectedPos);
sl@0
  2424
							User::LeaveIfError(InsertImplIntoImplIndex(implContainer.iCurrentImpl));
sl@0
  2425
#ifdef ECOM_TRACE
sl@0
  2426
							{
sl@0
  2427
							CImplementationData* p = implContainer.iCurrentImpl;
sl@0
  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()));
sl@0
  2429
							}
sl@0
  2430
#endif
sl@0
  2431
							}
sl@0
  2432
						}
sl@0
  2433
					}
sl@0
  2434
				//To make sure it is removed from the implIndex no matter what and that the return code is ignored.
sl@0
  2435
				//The previous removal is still required so that a subsequent InsertImplIntoImplIndex is possible
sl@0
  2436
				//for an implementation. e.g. an implementaion is on different drives and only one of htem was removed.
sl@0
  2437
				// The other one should now make it to the iImplIndex through InsertImplIntoImplIndex. If it wasn't 
sl@0
  2438
				// removed, before the insert operation, it will fail since both the implementations hav the same
sl@0
  2439
				// impl UID.
sl@0
  2440
				RemoveImplByAddrFromImplIndex((*interface->iImplementations)[i]);
sl@0
  2441
				}
sl@0
  2442
			if(impDataArray->Count() == 0)
sl@0
  2443
				{
sl@0
  2444
				iInterfaceImplIndex[indexPos].Reset();
sl@0
  2445
				iInterfaceImplIndex.Remove(indexPos);
sl@0
  2446
				}
sl@0
  2447
			}
sl@0
  2448
		counter++;
sl@0
  2449
		}
sl@0
  2450
	}
sl@0
  2451
sl@0
  2452
/**
sl@0
  2453
This method inserts implementations into the index
sl@0
  2454
@param          aDriveData Drive data in which we are searching the implementations
sl@0
  2455
@pre            This object is fully constructed.
sl@0
  2456
@post           all implemementations found, have been inserted into the index.
sl@0
  2457
*/
sl@0
  2458
void CRegistryData::AddImplDataL(CDriveData* aDriveData)
sl@0
  2459
	{
sl@0
  2460
 	TBool checkIsNeeded = EFalse;
sl@0
  2461
 	TInt dllCount=aDriveData->iDllList->Count();
sl@0
  2462
 	for(TInt dllCounter=0; dllCounter < dllCount;dllCounter++)
sl@0
  2463
 		{
sl@0
  2464
 		CDllData* aDllData = (*aDriveData->iDllList)[dllCounter];
sl@0
  2465
 		TInt interfaceCount=aDllData->iIfList->Count();
sl@0
  2466
 		for(TInt ifListCounter=0; ifListCounter < interfaceCount; ifListCounter++)
sl@0
  2467
 			{
sl@0
  2468
 			// For each interface structure
sl@0
  2469
 			CInterfaceData* interface = (*aDllData->iIfList)[ifListCounter];
sl@0
  2470
 			TInt implementationCount=interface->iImplementations->Count();
sl@0
  2471
 			for(TInt impNum = 0; impNum < implementationCount; ++impNum)
sl@0
  2472
 				{
sl@0
  2473
 				CImplementationData* implData = (*interface->iImplementations)[impNum];
sl@0
  2474
 				TInt retValue = InsertIntoIndexL(implData, checkIsNeeded);
sl@0
  2475
 				//we know that during internalize, all implementations are legitimate,
sl@0
  2476
 				//as the check was already performed during discovery. No check is 
sl@0
  2477
 				//performed at this stage, therefore ignore return value, as it will always
sl@0
  2478
 				//KErrNone
sl@0
  2479
 				}
sl@0
  2480
 			}	
sl@0
  2481
 		}	
sl@0
  2482
 	}
sl@0
  2483
sl@0
  2484
sl@0
  2485
/** This method does a search by address and removes the specified entry from iImplIndex.
sl@0
  2486
sl@0
  2487
@param  aPtr is the entry to remove
sl@0
  2488
@return True if aPtr is removed from iImplIndex. False if aPtr is not
sl@0
  2489
        in iImplIndex, i.e. nothing is removed.
sl@0
  2490
*/
sl@0
  2491
TBool CRegistryData::RemoveImplByAddrFromImplIndex(CImplementationData* aPtr) const
sl@0
  2492
    {
sl@0
  2493
    TInt aIdx = iImplIndex.Find(aPtr);
sl@0
  2494
    if (aIdx != KErrNotFound)
sl@0
  2495
        {
sl@0
  2496
        // This linear pointer search ensures a safe removal of the impl from iImplIndex so that it is free from a dead object.
sl@0
  2497
    
sl@0
  2498
        // The array does not own the pointer. Do not delete!
sl@0
  2499
        iImplIndex.Remove(aIdx);
sl@0
  2500
        return ETrue;
sl@0
  2501
        }
sl@0
  2502
    return EFalse;
sl@0
  2503
    }
sl@0
  2504
sl@0
  2505
/** This method inserts the entry aNewImpl into iImplIndex.
sl@0
  2506
sl@0
  2507
@param aNewImpl the item to add to iImplIndex.
sl@0
  2508
@return KErrNone aNewImpl is successfully added to the index.
sl@0
  2509
		KErrAlreadyExists iImplIndex has an entry with the same 
sl@0
  2510
		implementation UID and same parent Interface UID.
sl@0
  2511
		KErrNoMemory fail to insert due to out of memory.
sl@0
  2512
		Other system wide errors.
sl@0
  2513
@pre    aNewImpl is fully constructed.
sl@0
  2514
*/
sl@0
  2515
TInt CRegistryData::InsertImplIntoImplIndex(CImplementationData* aNewImpl) const
sl@0
  2516
	{
sl@0
  2517
	TLinearOrder<CImplementationData> ImplUidComparator(CImplementationData::CompareImplUid);
sl@0
  2518
sl@0
  2519
	return iImplIndex.InsertInOrder(aNewImpl, ImplUidComparator);
sl@0
  2520
	}
sl@0
  2521
sl@0
  2522
/** This method checks whether the language downgrade path has been changed.
sl@0
  2523
If it is, save language downgrade path information and set related flag true.
sl@0
  2524
sl@0
  2525
@param aLanguageChanged the returned value to indicate language changed.
sl@0
  2526
*/
sl@0
  2527
void CRegistryData::LanguageChangedL(TBool& aLanguageChanged)
sl@0
  2528
	{
sl@0
  2529
	iLanguageChanged = RDowngradePath::HasChangedL(iFs);
sl@0
  2530
	aLanguageChanged = iLanguageChanged;
sl@0
  2531
	}
sl@0
  2532
sl@0
  2533
/** setter
sl@0
  2534
If need to unset the callback, use a TCallBackWithArg constructed with no
sl@0
  2535
arguments.
sl@0
  2536
*/
sl@0
  2537
void CRegistryData::SetImplUpgradeCallBack(const TCallBackWithArg& aCallBack)
sl@0
  2538
	{
sl@0
  2539
	iImplUpgradeCallBack = aCallBack;
sl@0
  2540
	}
sl@0
  2541
sl@0
  2542
#ifdef __ECOM_SERVER_PERFORMANCE__
sl@0
  2543
/**
sl@0
  2544
This method calculates the drive, plugins, interfaces, implementations counts for the drive
sl@0
  2545
type set
sl@0
  2546
@param			aType The drive type for which the counts should be calculated 
sl@0
  2547
@param			aCounts Holds the calculated counts
sl@0
  2548
@pre			This object is constructed
sl@0
  2549
*/
sl@0
  2550
void CRegistryData::GetRegistryCountsL(TInt aType, RegistryCounts::TRegistryCounts& aCounts) const
sl@0
  2551
	{
sl@0
  2552
	aCounts.iDrives = 0;
sl@0
  2553
	aCounts.iImplementations = 0;
sl@0
  2554
	aCounts.iInterfaces = 0;
sl@0
  2555
	aCounts.iDlls = 0;
sl@0
  2556
	
sl@0
  2557
	for(TInt driveIndex = 0; driveIndex< iRegistrations->Count(); driveIndex++)
sl@0
  2558
		{
sl@0
  2559
		CDriveData* drive = (*iRegistrations)[driveIndex];
sl@0
  2560
		TBool isReadOnly = iCachedDriveInfo->DriveIsReadOnlyInternalL(drive->iDrive);
sl@0
  2561
sl@0
  2562
		if((aType == RegistryCounts::TRegistryCounts::EAll) ||
sl@0
  2563
			(aType == RegistryCounts::TRegistryCounts::ERoInternal && isReadOnly) ||
sl@0
  2564
			(aType == RegistryCounts::TRegistryCounts::ENonRoInternal && !isReadOnly))
sl@0
  2565
			{
sl@0
  2566
			aCounts.iDrives++;
sl@0
  2567
			aCounts.iDlls += drive->iDllList->Count();
sl@0
  2568
			for(TInt dllIndex = 0; dllIndex < drive->iDllList->Count(); dllIndex++)
sl@0
  2569
				{
sl@0
  2570
				CDllData* dllList = (*drive->iDllList)[dllIndex];
sl@0
  2571
				aCounts.iInterfaces += dllList->iIfList->Count();
sl@0
  2572
				for(TInt ifIndex = 0; ifIndex < dllList->iIfList->Count(); ifIndex++)
sl@0
  2573
					{
sl@0
  2574
					CInterfaceData* ifList = (*dllList->iIfList)[ifIndex];
sl@0
  2575
					aCounts.iImplementations += ifList->iImplementations->Count();
sl@0
  2576
					}
sl@0
  2577
				}
sl@0
  2578
			}
sl@0
  2579
		}
sl@0
  2580
	}
sl@0
  2581
sl@0
  2582
#endif // __ECOM_SERVER_PERFORMANCE__