os/ossrv/lowlevellibsandfws/pluginfw/Framework/frame/Registrar.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
// Implementation of the registrar class
sl@0
    15
// 
sl@0
    16
//
sl@0
    17
sl@0
    18
/**
sl@0
    19
 @internalComponent
sl@0
    20
 @file 
sl@0
    21
*/
sl@0
    22
sl@0
    23
#include <barsc2.h> // CResourceFile
sl@0
    24
#include <barsread2.h> // RResourceReader
sl@0
    25
#include <bautils.h>
sl@0
    26
#include <e32uid.h>
sl@0
    27
#include <startup.hrh> // for EStartupStateNonCritical
sl@0
    28
sl@0
    29
#include "ecompanics.h"
sl@0
    30
#include "EComDebug.h"
sl@0
    31
#include <ecom/ecom.h>
sl@0
    32
#include <ecom/ecomerrorcodes.h>
sl@0
    33
#include <ecom/ecomresolverparams.h>
sl@0
    34
#include "TestUtilities.h"	// For __FILE__LINE__
sl@0
    35
#include "Registrar.h"
sl@0
    36
#include "Discoverer.h"
sl@0
    37
#include "BackupNotifier.h"
sl@0
    38
#include "RegistryData.h"
sl@0
    39
#include "EComUidCodes.h"
sl@0
    40
#include "RegistrarObserver.h"
sl@0
    41
#include "DriveInfo.h"
sl@0
    42
#include "ParseImplementationData.h"
sl@0
    43
#include "EComInternalErrorCodes.h"
sl@0
    44
	
sl@0
    45
const TInt KMinBuffAllocSize = 1;
sl@0
    46
sl@0
    47
sl@0
    48
#define UNUSED_VAR(a) a = a
sl@0
    49
sl@0
    50
sl@0
    51
CRegistrar* CRegistrar::NewL(CRegistryData& aRegistry, MRegistrarObserver& aRegistrarObserver, RFs& aFs)
sl@0
    52
	{
sl@0
    53
	CRegistrar* self = new(ELeave) CRegistrar(aRegistry,aRegistrarObserver,aFs);
sl@0
    54
	CleanupStack::PushL(self);
sl@0
    55
	self->ConstructL();
sl@0
    56
	CleanupStack::Pop(self);
sl@0
    57
	return self;
sl@0
    58
	}
sl@0
    59
sl@0
    60
// Default d'tor
sl@0
    61
sl@0
    62
CRegistrar::~CRegistrar()
sl@0
    63
	{
sl@0
    64
	delete iBackupNotifier;
sl@0
    65
	delete iDiscoverer;
sl@0
    66
	delete iCachedDriveInfo;
sl@0
    67
	}
sl@0
    68
sl@0
    69
// Default c'tor
sl@0
    70
sl@0
    71
CRegistrar::CRegistrar(CRegistryData& aRegistry, MRegistrarObserver& aRegistrarObserver, RFs& aFs)
sl@0
    72
: CBase(), iRegistry(aRegistry), iRegistrarObserver(aRegistrarObserver), iFs(aFs), iState(EReg_Undefined)
sl@0
    73
	{
sl@0
    74
	}
sl@0
    75
sl@0
    76
sl@0
    77
void CRegistrar::ConstructL()
sl@0
    78
	{
sl@0
    79
	iCachedDriveInfo = CEComCachedDriveInfo::NewL(iFs);
sl@0
    80
sl@0
    81
	// Instantiate the discoverer
sl@0
    82
	iDiscoverer = CDiscoverer::NewL(*this, iFs);
sl@0
    83
	InitialiseEvent();
sl@0
    84
	}
sl@0
    85
sl@0
    86
sl@0
    87
TInt CRegistrar::Disable(TUid aImplementationUid)
sl@0
    88
	{
sl@0
    89
	return iRegistry.SetEnabledState(aImplementationUid, EFalse);
sl@0
    90
	}
sl@0
    91
sl@0
    92
sl@0
    93
TInt CRegistrar::Enable(TUid aImplementationUid)
sl@0
    94
	{
sl@0
    95
	return iRegistry.SetEnabledState(aImplementationUid, ETrue);
sl@0
    96
	}
sl@0
    97
sl@0
    98
sl@0
    99
TInt CRegistrar::Resume()
sl@0
   100
	{
sl@0
   101
	return iDiscoverer->Resume();
sl@0
   102
	}
sl@0
   103
sl@0
   104
sl@0
   105
TInt CRegistrar::Suspend()
sl@0
   106
	{
sl@0
   107
	return iDiscoverer->Suspend();
sl@0
   108
	}
sl@0
   109
sl@0
   110
sl@0
   111
void CRegistrar::DriveRemovedL(TDriveUnit aDrive)
sl@0
   112
	{
sl@0
   113
	iRegistry.TemporaryUninstallL(aDrive);
sl@0
   114
	}
sl@0
   115
sl@0
   116
sl@0
   117
void CRegistrar::DriveReinstatedL(TDriveUnit aDrive)
sl@0
   118
	{
sl@0
   119
	iRegistry.UndoTemporaryUninstallL(aDrive);
sl@0
   120
	}
sl@0
   121
	
sl@0
   122
TBool CRegistrar::IsAnyDllRegisteredWithDriveL(const TDriveUnit aDrive)const
sl@0
   123
	{
sl@0
   124
	return iRegistry.IsAnyDllRegisteredWithDriveL(aDrive);
sl@0
   125
	}
sl@0
   126
sl@0
   127
void CRegistrar::RegisterDiscoveryL(const TDriveName& aDrive,CPluginBase*& aDirEntry, TBool aAnyDllDiscovered)
sl@0
   128
	{
sl@0
   129
	TBool found = EFalse;
sl@0
   130
	TBool update = EFalse;
sl@0
   131
	TInt  registryDriveIndex = KErrNotFound;
sl@0
   132
sl@0
   133
	// Find the appropriate drive entry
sl@0
   134
	TChar driveLetter = User::UpperCase(aDrive[0]) - 'A';
sl@0
   135
	TDriveUnit driveUnit = EDriveA + driveLetter;
sl@0
   136
	CRegistryData::CDriveData* drive =NULL;
sl@0
   137
	
sl@0
   138
	// Get the drive data in the registry.
sl@0
   139
	registryDriveIndex = iRegistry.FindDriveL(driveUnit, drive);
sl@0
   140
	if(registryDriveIndex == KErrNotFound)
sl@0
   141
		{
sl@0
   142
		User::Leave(KEComErrDriveNotFound);
sl@0
   143
		}
sl@0
   144
		
sl@0
   145
sl@0
   146
	// Get the registry to return enough data so that the Add and Update routines
sl@0
   147
	// can be optimised to avoid rescanning the registry data.
sl@0
   148
	// This means that registryDriveIndex MUST remain valid until after
sl@0
   149
	// the ParseRegistrationDataL call below.
sl@0
   150
	
sl@0
   151
	// IsRegisteredWithDate will be called if the this drive is NOT
sl@0
   152
	// under its initial discovery. If any Dll is discovered in the drive, then
sl@0
   153
	// this drive is NOT in its initial discovery.
sl@0
   154
	
sl@0
   155
	// In read-only drive with SPI enable, aAnyDllDiscovered will be always false. 
sl@0
   156
	// If language downgrade path has been changed, IsRegisteredWithDate will be
sl@0
   157
	// called to check whether current localised file need to be updated. 
sl@0
   158
	if((aAnyDllDiscovered) || (iRegistry.HasLanguageChanged()))
sl@0
   159
		{
sl@0
   160
		found = iRegistry.IsRegisteredWithDate(aDirEntry->iDllThirdUid, aDirEntry->iDllModifiedTime, update, drive);
sl@0
   161
		}
sl@0
   162
	
sl@0
   163
	if(!found || update || iRegistry.HasLanguageChanged())
sl@0
   164
		{
sl@0
   165
		if(found && !update && iRegistry.HasLanguageChanged())
sl@0
   166
			{
sl@0
   167
			// If we got here because the language has changed
sl@0
   168
			// ensure the existing registry data is updated rather than a new entry added
sl@0
   169
			update = ETrue;
sl@0
   170
			}
sl@0
   171
		ParseRegistrationDataL(aDirEntry,driveUnit, update, registryDriveIndex, drive);
sl@0
   172
		}
sl@0
   173
	}
sl@0
   174
sl@0
   175
sl@0
   176
void CRegistrar::DiscoveriesBegin()
sl@0
   177
	{
sl@0
   178
	// We are about to receive registrations so inform the registry that its index
sl@0
   179
	// is about to be invalid
sl@0
   180
	iRegistry.DiscoveriesBeginning();
sl@0
   181
	}
sl@0
   182
sl@0
   183
sl@0
   184
void CRegistrar::DiscoveriesComplete(TBool aSuccessful, TPluginProcessingTypeIdentifier aProcessingType)
sl@0
   185
	{
sl@0
   186
	
sl@0
   187
	TBool doesRegChanged = EFalse;
sl@0
   188
sl@0
   189
	// Successfully completed, if registry data has been changed, update 
sl@0
   190
	TRAPD(error,iRegistry.DiscoveriesCompleteL(aSuccessful, aProcessingType,doesRegChanged));
sl@0
   191
    UNUSED_VAR(error);
sl@0
   192
	// Notify if the registry data has been changed.
sl@0
   193
	if(doesRegChanged)
sl@0
   194
		{
sl@0
   195
	// Notify that we have updated the registry
sl@0
   196
		iRegistrarObserver.Notification(KErrNone);
sl@0
   197
		}
sl@0
   198
	}
sl@0
   199
sl@0
   200
void CRegistrar::SetDiscoveryFlagL(const TDriveUnit& aDriveUnit)
sl@0
   201
	{
sl@0
   202
	//set the drive flag to indicate it has changes.
sl@0
   203
	iRegistry.SetDiscoveryFlagL(aDriveUnit);
sl@0
   204
	}
sl@0
   205
	
sl@0
   206
TBool CRegistrar::NotifiedWithErrorCode(TInt aError)
sl@0
   207
	{
sl@0
   208
	// Test the safe error codes
sl@0
   209
	return (aError == KErrNotReady ||		// Drive removed?
sl@0
   210
			aError == KErrPathNotFound);	// Directory deleted?
sl@0
   211
	}
sl@0
   212
sl@0
   213
void CRegistrar::ParseRegistrationDataL(CPluginBase*& aEntry,
sl@0
   214
										const TDriveUnit& aDrive, 
sl@0
   215
										TBool aUpdate, 
sl@0
   216
										TInt aRegistryDriveIndex,
sl@0
   217
										CRegistryData::CDriveData* aDriveData)
sl@0
   218
	{
sl@0
   219
	// Ok we intend installing this item so 
sl@0
   220
	// create the Interface Implementation Collection entry first
sl@0
   221
	CRegistryData::CDllData* dll = CRegistryData::CDllData::NewLC(*(aEntry->iDllName),aEntry->iDllModifiedTime,aEntry->iDllSecondUid,aEntry->iDllThirdUid,aDriveData);
sl@0
   222
sl@0
   223
#ifdef ECOM_TRACE
sl@0
   224
	static int counter = 0; counter++;
sl@0
   225
	TFileName fullFileName(aDrive.Name());
sl@0
   226
	fullFileName.Append(_L("\\sys\\bin\\"));
sl@0
   227
	fullFileName.Append(*(aEntry->iDllName));
sl@0
   228
	__ECOM_TRACE2("ECOM: Plugin discovered (%04d) %S", counter, &fullFileName);
sl@0
   229
#endif
sl@0
   230
sl@0
   231
	// update resource ext info
sl@0
   232
	HBufC* resourceExt = aEntry->RscFileExt();
sl@0
   233
	if(resourceExt)
sl@0
   234
		dll->SetResourceExtL(*resourceExt);
sl@0
   235
	
sl@0
   236
	// Don't automatically leave during the parse of the implementation
sl@0
   237
	// collection's registration data.
sl@0
   238
	// If we did then there would be a cascade of leaves which stop the 
sl@0
   239
	// entire registration process for all discoveries remaining on the list.
sl@0
   240
	// Check to see if there is a problem specifically with the registration data
sl@0
   241
	// OR if its a general problem. In the latter case leave.
sl@0
   242
	TInt error=KErrNone;
sl@0
   243
			
sl@0
   244
	TRAP(error, ParseL(aEntry,*dll));
sl@0
   245
	if (error==KErrNoMemory)
sl@0
   246
		User::LeaveNoMemory();
sl@0
   247
	if(error == KErrNone)
sl@0
   248
		{
sl@0
   249
		if(aUpdate)
sl@0
   250
			{	
sl@0
   251
			iRegistry.UpdateDllDataL(aDrive, aRegistryDriveIndex, dll);
sl@0
   252
			}
sl@0
   253
		else
sl@0
   254
			{	
sl@0
   255
			iRegistry.AddDllDataL(aDrive, aRegistryDriveIndex, dll);
sl@0
   256
			}
sl@0
   257
		// Remove dll from CleanupStack as ownership has been passed in one of
sl@0
   258
		// the two functions above
sl@0
   259
		CleanupStack::Pop(dll);
sl@0
   260
		}
sl@0
   261
	else
sl@0
   262
		{
sl@0
   263
		// This interface implementation collection 
sl@0
   264
		// cannot be registered correctly.
sl@0
   265
		CleanupStack::PopAndDestroy(dll);
sl@0
   266
		}
sl@0
   267
	}
sl@0
   268
sl@0
   269
void CRegistrar::GetResourceFormatVersionL(RResourceReader& aReader, TInt& aResourceFormatVersion)
sl@0
   270
	{
sl@0
   271
	aResourceFormatVersion = 1;
sl@0
   272
	TUid dllUid(KNullUid);
sl@0
   273
	TUid uid = {aReader.ReadInt32L()};
sl@0
   274
	if(uid==KUidEComResourceFormatV2)
sl@0
   275
		{
sl@0
   276
		aResourceFormatVersion = 2;
sl@0
   277
		dllUid.iUid = aReader.ReadInt32L();
sl@0
   278
		}
sl@0
   279
	else if(uid==KUidEComResourceFormatV3)
sl@0
   280
		{
sl@0
   281
		aResourceFormatVersion = 3;
sl@0
   282
		dllUid.iUid = aReader.ReadInt32L();
sl@0
   283
		}		
sl@0
   284
	else
sl@0
   285
		dllUid = uid;
sl@0
   286
	}
sl@0
   287
sl@0
   288
	
sl@0
   289
void CRegistrar::ParseL(CPluginBase*& aEntry,CRegistryData::CDllData& aDll)
sl@0
   290
	{
sl@0
   291
// Read the resource file starting at offset:0 section size:0
sl@0
   292
	CResourceFile* regFile =aEntry->RscFile();
sl@0
   293
//	Note : There may be an issue here with resource reading when the 
sl@0
   294
//  file is not in the ROM. The solution would be to call
sl@0
   295
//	regFile.ConfirmSignatureL(regFile.SignatureL());
sl@0
   296
//	However for 6.2 interface implementation collections will ALL
sl@0
   297
//	be within the ROM. and future development will move to an
sl@0
   298
//	XML parsed solution @ build time.
sl@0
   299
//	IF this situation changes, place the fix here.
sl@0
   300
 	RResourceReader theReader;
sl@0
   301
 	theReader.OpenLC(regFile, KMinBuffAllocSize);
sl@0
   302
	TInt resourceFormatVersion = 0;
sl@0
   303
	GetResourceFormatVersionL(theReader,resourceFormatVersion);
sl@0
   304
	
sl@0
   305
	TDriveUnit drive(aDll.iParent->iDrive);
sl@0
   306
	TBool romBased = iCachedDriveInfo->DriveIsReadOnlyInternalL(drive);
sl@0
   307
	const TInt numInterfaces = theReader.ReadInt16L();
sl@0
   308
	if (resourceFormatVersion == 3 && numInterfaces > KMaxInterfacesInResV3)
sl@0
   309
		{
sl@0
   310
		User::Leave(KErrNotSupported);
sl@0
   311
		}
sl@0
   312
	for(TInt iface = 0; iface < numInterfaces; ++iface)
sl@0
   313
		{
sl@0
   314
		// Read the interface uid
sl@0
   315
		const TUid interfaceUid = {theReader.ReadInt32L()};
sl@0
   316
		CRegistryData::CInterfaceData* interfaceList = CRegistryData::CInterfaceData::NewLC(interfaceUid,&aDll);
sl@0
   317
		aDll.AddL(interfaceList);
sl@0
   318
		CleanupStack::Pop(interfaceList);		// Now owned by aDll
sl@0
   319
		const TInt numImplementations = theReader.ReadInt16L();
sl@0
   320
		if (resourceFormatVersion == 3 && numImplementations > KMaxImplementationsForEachInterfacesInResV3)
sl@0
   321
			{
sl@0
   322
			User::Leave(KErrNotSupported);
sl@0
   323
			}
sl@0
   324
		for(TInt index = 0; index < numImplementations; ++index)
sl@0
   325
			{
sl@0
   326
			TUid implementationUid;
sl@0
   327
			TInt versionNo;
sl@0
   328
			TInt infoFormat;
sl@0
   329
			TBool romOnly;
sl@0
   330
			RExtendedInterfacesArray* extendedInterfaces = NULL;
sl@0
   331
			HBufC* name = NULL;
sl@0
   332
			HBufC8* defaultData = NULL;
sl@0
   333
			HBufC8* opaqueData = NULL;
sl@0
   334
			CParseImplementationData* parseImplementationData = CParseImplementationData::NewLC(resourceFormatVersion);
sl@0
   335
			parseImplementationData->ParseL(theReader,
sl@0
   336
											infoFormat,
sl@0
   337
											implementationUid,
sl@0
   338
											versionNo,
sl@0
   339
									   	  	name,
sl@0
   340
									      	defaultData,
sl@0
   341
									      	opaqueData,
sl@0
   342
									      	extendedInterfaces,
sl@0
   343
									      	romOnly);
sl@0
   344
			CleanupStack::PopAndDestroy(parseImplementationData);									      	
sl@0
   345
			
sl@0
   346
			CleanupStack::PushL(TCleanupItem(CloseAndDeleteArray, extendedInterfaces));
sl@0
   347
			CleanupStack::PushL(name);
sl@0
   348
			CleanupStack::PushL(defaultData);
sl@0
   349
			CleanupStack::PushL(opaqueData);
sl@0
   350
			if (romOnly && !(romBased))
sl@0
   351
				{
sl@0
   352
				// pop and destroy opaquedata, defaultdata,name and extendedInterfaces
sl@0
   353
				CleanupStack::PopAndDestroy(4,extendedInterfaces); 
sl@0
   354
				}
sl@0
   355
			else
sl@0
   356
				{
sl@0
   357
				CRegistryData::CImplementationData* implData = CRegistryData::CImplementationData::NewL(interfaceList,
sl@0
   358
																										implementationUid, 
sl@0
   359
																				 						versionNo, 
sl@0
   360
																				 						name, 
sl@0
   361
																						 				defaultData,
sl@0
   362
																										opaqueData,
sl@0
   363
																					 					drive,
sl@0
   364
																					 					romOnly,
sl@0
   365
																				 						romBased,
sl@0
   366
																				 						extendedInterfaces);
sl@0
   367
				CleanupStack::Pop(4,extendedInterfaces); // opaqueData,defaultData,name,extendedInterfaces owned by implData
sl@0
   368
				CleanupStack::PushL(implData);
sl@0
   369
				interfaceList->AddL(implData);	// Now owned by interfaceList	
sl@0
   370
				CleanupStack::Pop(implData);
sl@0
   371
				}
sl@0
   372
			}
sl@0
   373
		}
sl@0
   374
	CleanupStack::PopAndDestroy(&theReader);		
sl@0
   375
	}
sl@0
   376
sl@0
   377
	
sl@0
   378
CRegistrar::TRegistrarState CRegistrar::State()const
sl@0
   379
	{
sl@0
   380
	return iState;
sl@0
   381
	}
sl@0
   382
sl@0
   383
void CRegistrar::ProcessSSAEventL(TStartupStateIdentifier aKnownState)
sl@0
   384
	{
sl@0
   385
	iDiscoverer->ProcessSSAEventL(aKnownState); 
sl@0
   386
	if(aKnownState == EStartupStateNonCritical && iState == EReg_StartupInProgess)
sl@0
   387
		{
sl@0
   388
		__ECOM_TRACE("ECOM: CRegistrar::ProcessSSAEventL(): EStartupStateNonCritical is reached.");
sl@0
   389
sl@0
   390
		iState = EReg_StartupComplete;
sl@0
   391
		iBackupNotifier = CBackupNotifier::NewL(*this);
sl@0
   392
		}
sl@0
   393
	}
sl@0
   394
void CRegistrar::InitialiseEvent()
sl@0
   395
	{
sl@0
   396
	iState = EReg_StartupInProgess;
sl@0
   397
	}
sl@0
   398
sl@0
   399
void CRegistrar::LanguageChangedL(TBool& aLanguageChanged)
sl@0
   400
	{
sl@0
   401
	iRegistry.LanguageChangedL(aLanguageChanged);
sl@0
   402
	}
sl@0
   403
sl@0
   404
void CRegistrar::InstallSwiEventCallBack(const TCallBackWithArg& aCallBack)
sl@0
   405
	{
sl@0
   406
	iDiscoverer->SetSwiChangeCallBack(aCallBack);
sl@0
   407
	}
sl@0
   408
sl@0
   409
void CRegistrar::InstallBurEventCallBack(const TCallBackWithArg& aCallBack)
sl@0
   410
	{
sl@0
   411
	iDiscoverer->SetBurChangeCallBack(aCallBack);
sl@0
   412
	}