os/ossrv/lowlevellibsandfws/pluginfw/Framework/frame/EComServer.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
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
// The Implementation of the CEComServer singleton class which
sl@0
    15
// instantiates an instance of the requested ECom Interface Implementation.
sl@0
    16
// 
sl@0
    17
//
sl@0
    18
sl@0
    19
/**
sl@0
    20
 @internalComponent
sl@0
    21
 @file
sl@0
    22
*/
sl@0
    23
#include <e32base.h>
sl@0
    24
#include <bautils.h>
sl@0
    25
#include <startupdomaindefs.h>
sl@0
    26
#include "EComDebug.h"
sl@0
    27
#include <ecom/ecom.h>
sl@0
    28
#include <ecom/ecomerrorcodes.h>
sl@0
    29
#include <ecom/ecomresolverparams.h>
sl@0
    30
#include <ecom/implementationinformation.h>
sl@0
    31
sl@0
    32
#include "ServerStartupManager.h"
sl@0
    33
#include "RegistryData.h"
sl@0
    34
#include "Registrar.h"
sl@0
    35
#include "DefaultResolver.h"
sl@0
    36
#include "RomOnlyResolver.h"
sl@0
    37
#include "TlsData.h"
sl@0
    38
#include "EComServerStart.h"
sl@0
    39
#include "EComMessageIds.h"
sl@0
    40
#include "EComServerSession.h"
sl@0
    41
#include "EComServer.h"
sl@0
    42
#include "EComUidCodes.h"
sl@0
    43
#include "EComPerformance.h"
sl@0
    44
#include "DriveInfo.h"
sl@0
    45
#include "FileUtils.h"
sl@0
    46
#include "RegistryResolveTransaction.h"
sl@0
    47
#include "resolvercache.h"
sl@0
    48
#include "EComPatchDataConstantv2.h"
sl@0
    49
sl@0
    50
#define UNUSED_VAR(a) a = a
sl@0
    51
sl@0
    52
/** enum for special system events */
sl@0
    53
enum TSpecialEvents
sl@0
    54
	{
sl@0
    55
	EBURInProgress,
sl@0
    56
	ESWIInProgress
sl@0
    57
	};
sl@0
    58
sl@0
    59
/** The location of server INI file. */
sl@0
    60
_LIT(KEComSrvrIniFile,"_:\\private\\10009D8F\\EComSrvr.ini");
sl@0
    61
/** Buffer descriptor to hold full path and name of server INI file. */
sl@0
    62
typedef TBuf<32> TEComSrvrIniFileName;
sl@0
    63
sl@0
    64
_LIT(KEComSrvrIniFileROM,"Z:\\private\\10009D8F\\EComSrvr.ini");
sl@0
    65
sl@0
    66
static void CloseAndDeleteImplInfoArray(TAny* aObject)
sl@0
    67
	{
sl@0
    68
	RImplInfoArray* array=reinterpret_cast<RImplInfoArray*>(aObject);
sl@0
    69
	if (array)
sl@0
    70
		{
sl@0
    71
		array->Close();
sl@0
    72
		}
sl@0
    73
	delete array;
sl@0
    74
	}
sl@0
    75
//
sl@0
    76
// Initiate server exit when the timer expires
sl@0
    77
// by stopping the local scheduler.
sl@0
    78
void CShutdown::RunL()
sl@0
    79
	{
sl@0
    80
	CActiveScheduler::Stop();
sl@0
    81
	}
sl@0
    82
sl@0
    83
CEComServer* CEComServer::NewLC()
sl@0
    84
	{
sl@0
    85
	// Standard 2 phase construction code
sl@0
    86
	CEComServer* instance = new(ELeave) CEComServer;
sl@0
    87
	CleanupStack::PushL(instance);
sl@0
    88
	instance->ConstructL();
sl@0
    89
	return instance;
sl@0
    90
	}
sl@0
    91
sl@0
    92
sl@0
    93
CEComServer::~CEComServer()
sl@0
    94
	{
sl@0
    95
	// Uninstall callbacks first. Do not want to receive any callback
sl@0
    96
	// in destructor. TCallBackWithArg constructed with no argument
sl@0
    97
	// is a null callback.
sl@0
    98
	TCallBackWithArg nullCallback;
sl@0
    99
	if (iRegistrar)
sl@0
   100
		{
sl@0
   101
		iRegistrar->InstallSwiEventCallBack(nullCallback);
sl@0
   102
		iRegistrar->InstallBurEventCallBack(nullCallback);
sl@0
   103
		}
sl@0
   104
	if (iRegistryData)
sl@0
   105
		{
sl@0
   106
		iRegistryData->SetImplUpgradeCallBack(nullCallback);
sl@0
   107
		}
sl@0
   108
sl@0
   109
	// Ensure this deletion order is maintained - in particular the registrydata 
sl@0
   110
	// must last longer than the others which hold a reference to it.
sl@0
   111
sl@0
   112
	//remove the CServerStartupMgr
sl@0
   113
	delete iServerStartupMgr;
sl@0
   114
	// Then destroy the registrar object
sl@0
   115
	delete iRegistrar;
sl@0
   116
	// remove the registry data
sl@0
   117
	delete iRegistryData;
sl@0
   118
	
sl@0
   119
	delete iResolverCache;
sl@0
   120
sl@0
   121
	// Finally close down the file server connection
sl@0
   122
	iFs.Close();
sl@0
   123
	// Make sure the non-default resolver library is closed
sl@0
   124
	iResolverLibrary.Close();
sl@0
   125
	}
sl@0
   126
sl@0
   127
sl@0
   128
CEComServer::CEComServer()
sl@0
   129
: CServer2(CActive::EPriorityStandard)
sl@0
   130
	{
sl@0
   131
	// Do nothing
sl@0
   132
	}
sl@0
   133
sl@0
   134
sl@0
   135
void CEComServer::ConstructL()
sl@0
   136
	{
sl@0
   137
	START_TIMER
sl@0
   138
	START_HEAP
sl@0
   139
	StartL(KEComServerName);
sl@0
   140
	// Connect to the file server
sl@0
   141
	User::LeaveIfError(iFs.Connect());
sl@0
   142
	__ECOM_TRACE("ECOM: Server INIT - File Server session initialised");
sl@0
   143
	// construct the registry data handling object here
sl@0
   144
	iRegistryData = CRegistryData::NewL(iFs);
sl@0
   145
	__ECOM_TRACE("ECOM: Server INIT - Registry Data initialised");
sl@0
   146
	// Then the registrar
sl@0
   147
	iRegistrar = CRegistrar::NewL(*iRegistryData, *this, iFs);
sl@0
   148
	__ECOM_TRACE("ECOM: Server INIT - Registrar initialised");
sl@0
   149
	
sl@0
   150
	//Then the CServerStartupMgr
sl@0
   151
sl@0
   152
#ifdef SYMBIAN_SYSTEM_STATE_MANAGEMENT
sl@0
   153
	iServerStartupMgr = new(ELeave) CServerStartupMgr(KDmHierarchyIdStartup, KSM2OSServicesDomain3, iFs);
sl@0
   154
#else
sl@0
   155
	iServerStartupMgr = new(ELeave) CServerStartupMgr(KDmHierarchyIdStartup, KBaseServicesDomain3, iFs);
sl@0
   156
#endif //SYMBIAN_SYSTEM_STATE_MANAGEMENT
sl@0
   157
sl@0
   158
	__ECOM_TRACE("ECOM: Server INIT - ServerStartupMgr initialised");
sl@0
   159
	iServerStartupMgr->RegisterObserverL(iRegistrar);
sl@0
   160
	
sl@0
   161
	iServerStartupMgr->InitialiseL(IsSSA(iFs));
sl@0
   162
	
sl@0
   163
	iResolverCache = CCustomResolverCache::NewL(KCustomResolverCacheSize,
sl@0
   164
		KCustomResolverCacheTimeout);
sl@0
   165
sl@0
   166
	TCallBackWithArg eventCallback(&CEComServer::NotifyEvents, this);
sl@0
   167
	iRegistrar->InstallSwiEventCallBack(eventCallback);
sl@0
   168
	iRegistrar->InstallBurEventCallBack(eventCallback);
sl@0
   169
	iRegistryData->SetImplUpgradeCallBack(eventCallback);
sl@0
   170
sl@0
   171
	// The server is about to start so construct the transient shutdown guy
sl@0
   172
	iShutdown.ConstructL();
sl@0
   173
	// ensure that the server still exits even
sl@0
   174
	// if the server start fails or the
sl@0
   175
	// 1st client fails to connect
sl@0
   176
	iShutdown.Start();
sl@0
   177
	RECORD_INITIALISE_HEAP
sl@0
   178
	RECORD_INITIALISE_RESULT
sl@0
   179
	}
sl@0
   180
	
sl@0
   181
TBool CEComServer::IsSSA(RFs& aFs)
sl@0
   182
	{
sl@0
   183
	// Check Z drive first.
sl@0
   184
	TBool isFileExisting = BaflUtils::FileExists(aFs, KEComSrvrIniFileROM);
sl@0
   185
	// Check system drive next
sl@0
   186
	if(!isFileExisting)
sl@0
   187
		{
sl@0
   188
		TEComSrvrIniFileName iniFile(KEComSrvrIniFile);
sl@0
   189
		TEComFileUtils::SetToDrive(iniFile,iRegistryData->iSystemDrive);
sl@0
   190
		isFileExisting = BaflUtils::FileExists(aFs, iniFile);
sl@0
   191
		}
sl@0
   192
	return !isFileExisting;		
sl@0
   193
	}
sl@0
   194
sl@0
   195
// Get implementation info for one implementation, and return to client
sl@0
   196
void CEComServer::GetImplementationInformationL(const TUid& aImplementationUid,
sl@0
   197
												CImplementationInformation*& aImplInfo,
sl@0
   198
												const TClientRequest& aClientRequest)
sl@0
   199
	{
sl@0
   200
	TEntry dllInfo;
sl@0
   201
	User::LeaveIfError(iRegistryData->GetImplementationDllInfoForClientL(
sl@0
   202
		aClientRequest, aImplementationUid, KNullUid, dllInfo, aImplInfo, ETrue));
sl@0
   203
	}
sl@0
   204
sl@0
   205
RImplInfoArray* CEComServer::ListImplementationsL(TUid aInterfaceUid, 
sl@0
   206
												   const TEComResolverParams& aAdditionalParameters, 
sl@0
   207
												   TUid aResolverUid,
sl@0
   208
												   const RExtendedInterfacesArray& aExtendedInterfaces,
sl@0
   209
												   const TClientRequest& aMessage
sl@0
   210
												   ) 
sl@0
   211
	{
sl@0
   212
	RImplInfoArray* result = NULL;
sl@0
   213
	CResolver* resolver = NULL;
sl@0
   214
	//create registry resolver transaction
sl@0
   215
	//get the TListImplParam parameters
sl@0
   216
	TBool capability= ETrue;
sl@0
   217
	if(!(aMessage.IsNull()))
sl@0
   218
		{
sl@0
   219
		TListImplParam listParam;
sl@0
   220
  		TPckg<TListImplParam> listParamPkg(listParam);
sl@0
   221
  		aMessage.ReadL(2,listParamPkg);
sl@0
   222
  		capability=listParam.iCapabilityCheck;
sl@0
   223
		}
sl@0
   224
  	
sl@0
   225
  	
sl@0
   226
	CRegistryResolveTransaction* registryResolveTransaction = CRegistryResolveTransaction::NewL(*iRegistryData,
sl@0
   227
																								aExtendedInterfaces,
sl@0
   228
																								aMessage,capability);
sl@0
   229
	CleanupStack::PushL(registryResolveTransaction);
sl@0
   230
	//create resolver
sl@0
   231
	resolver = CreateResolverLC(aResolverUid,registryResolveTransaction);
sl@0
   232
	result = ListImplementationsL(aInterfaceUid, aAdditionalParameters, resolver);
sl@0
   233
	//clean up
sl@0
   234
	CleanupStack::PopAndDestroy(resolver);
sl@0
   235
	CleanupStack::PopAndDestroy(registryResolveTransaction);
sl@0
   236
	if ((aResolverUid != KDefaultResolverUid) && (aResolverUid != KRomOnlyResolverUid))
sl@0
   237
		{
sl@0
   238
		iResolverLibrary.Close();
sl@0
   239
		}
sl@0
   240
	return result;
sl@0
   241
	}
sl@0
   242
sl@0
   243
RImplInfoArray* CEComServer::ListImplementationsL(TUid aInterfaceUid, 
sl@0
   244
												   TUid aResolverUid,
sl@0
   245
												   const RExtendedInterfacesArray& aExtendedInterfaces,
sl@0
   246
												   const TClientRequest& aMessage
sl@0
   247
												   ) 
sl@0
   248
	{
sl@0
   249
	RImplInfoArray* result = NULL;
sl@0
   250
	CResolver* resolver = NULL;
sl@0
   251
	//create registry resolver transaction
sl@0
   252
	//get the TListImplParam parameters
sl@0
   253
  		//get the TListImplParam parameters
sl@0
   254
	TBool capability= ETrue;
sl@0
   255
	if(!(aMessage.IsNull()))
sl@0
   256
		{
sl@0
   257
		TListImplParam listParam;
sl@0
   258
  		TPckg<TListImplParam> listParamPkg(listParam);
sl@0
   259
  		aMessage.ReadL(2,listParamPkg);
sl@0
   260
  		capability=listParam.iCapabilityCheck;
sl@0
   261
		}
sl@0
   262
	CRegistryResolveTransaction* registryResolveTransaction = CRegistryResolveTransaction::NewL(*iRegistryData,
sl@0
   263
																								aExtendedInterfaces,
sl@0
   264
																								aMessage,capability);
sl@0
   265
	CleanupStack::PushL(registryResolveTransaction);
sl@0
   266
	//create resolver
sl@0
   267
	resolver = CreateResolverLC(aResolverUid,registryResolveTransaction);
sl@0
   268
	result = ListImplementationsL(aInterfaceUid, resolver);
sl@0
   269
	//clean up
sl@0
   270
	CleanupStack::PopAndDestroy(resolver);
sl@0
   271
	CleanupStack::PopAndDestroy(registryResolveTransaction);
sl@0
   272
	if ((aResolverUid != KDefaultResolverUid) && (aResolverUid != KRomOnlyResolverUid))
sl@0
   273
		{
sl@0
   274
		iResolverLibrary.Close();
sl@0
   275
		}
sl@0
   276
	return result;
sl@0
   277
	}
sl@0
   278
sl@0
   279
sl@0
   280
RImplInfoArray* CEComServer::ListImplementationsL(TUid aInterfaceUid, 
sl@0
   281
												   const TEComResolverParams& aAdditionalParameters,
sl@0
   282
												   const RExtendedInterfacesArray& aExtendedInterfaces,
sl@0
   283
												   const TClientRequest& aMessage) 
sl@0
   284
	{
sl@0
   285
	// Use the default resolver in the overloaded method.
sl@0
   286
	return ListImplementationsL(aInterfaceUid, aAdditionalParameters, KDefaultResolverUid,aExtendedInterfaces, aMessage);
sl@0
   287
	}
sl@0
   288
sl@0
   289
RImplInfoArray* CEComServer::ListImplementationsL(TUid aInterfaceUid,
sl@0
   290
												  const RExtendedInterfacesArray& aExtendedInterfaces,	
sl@0
   291
												  const TClientRequest& aMessage) 
sl@0
   292
	{
sl@0
   293
	// Use the default resolver in the overloaded method.
sl@0
   294
	return ListImplementationsL(aInterfaceUid, KDefaultResolverUid,aExtendedInterfaces, aMessage);
sl@0
   295
	}
sl@0
   296
sl@0
   297
// The private helper
sl@0
   298
sl@0
   299
RImplInfoArray* CEComServer::ListImplementationsL(TUid aInterfaceUid, 
sl@0
   300
												   const TEComResolverParams& aAdditionalParameters, 
sl@0
   301
												   CResolver* aResolver) const
sl@0
   302
	{
sl@0
   303
	if(!aResolver)
sl@0
   304
		User::Leave(KEComErrNoResolver);
sl@0
   305
	// Use the client provided resolver to build up the list.
sl@0
   306
	RImplInfoArray* infoArray = aResolver->ListAllL(aInterfaceUid, aAdditionalParameters);
sl@0
   307
	return infoArray;
sl@0
   308
	}
sl@0
   309
sl@0
   310
RImplInfoArray* CEComServer::ListImplementationsL(TUid aInterfaceUid, 
sl@0
   311
												  CResolver* aResolver) const
sl@0
   312
	{
sl@0
   313
	if(!aResolver)
sl@0
   314
		User::Leave(KEComErrNoResolver);
sl@0
   315
	// Use the provided resolver to build up the list.
sl@0
   316
	RImplInfoArray* infoArray = &aResolver->ListAllL(aInterfaceUid);
sl@0
   317
	// infoArray points to iImplementationInfo, which is owned by CRegistryResolveTransaction. 
sl@0
   318
	// CRegistryResolveTransaction object is transient and will be destroyed before return the implementation
sl@0
   319
	// info list to the CEComServerSession. Therefore, we need to have a copy to return
sl@0
   320
	RImplInfoArray* retList = new (ELeave) RImplInfoArray;
sl@0
   321
	CleanupStack::PushL(TCleanupItem(CloseAndDeleteImplInfoArray,retList));
sl@0
   322
	const TInt numImps = infoArray->Count();
sl@0
   323
	for(TInt index = 0; index < numImps; ++index)
sl@0
   324
		{	
sl@0
   325
		retList->AppendL((*infoArray)[index]);
sl@0
   326
		}
sl@0
   327
	// Reset the member variable because we are passing ownership back
sl@0
   328
	CleanupStack::Pop();
sl@0
   329
	return retList;
sl@0
   330
	}
sl@0
   331
sl@0
   332
void CEComServer::GetResolvedDllInfoL(	const TUid aImplementationUid,
sl@0
   333
										TEntry& aDllInfo,
sl@0
   334
										TUid& aDtor_Key,
sl@0
   335
										const TClientRequest& aClientRequest)
sl@0
   336
	{
sl@0
   337
	// No resolution to do create directly.
sl@0
   338
	aDtor_Key = aImplementationUid;
sl@0
   339
	CImplementationInformation* implInfo = NULL;
sl@0
   340
	TUid dummyUid={0x00000000};
sl@0
   341
	//We need to do the security check for the case that implementationUid is known.
sl@0
   342
	//if implementationUid is unknown, the security check will be done in ListImplementationsL.
sl@0
   343
	User::LeaveIfError(iRegistryData->GetImplementationDllInfoForClientL(
sl@0
   344
		aClientRequest, aImplementationUid, dummyUid, aDllInfo, implInfo, ETrue));
sl@0
   345
	}
sl@0
   346
sl@0
   347
sl@0
   348
void CEComServer::GetResolvedDllInfoL(  const TUid aInterfaceUid,
sl@0
   349
										const TEComResolverParams& aAdditionalParameters,
sl@0
   350
										const RExtendedInterfacesArray& aExtendedInterfaces,
sl@0
   351
										TEntry& aDllInfo, 
sl@0
   352
										TUid& aDtor_Key,
sl@0
   353
										const TClientRequest& aClientRequest)
sl@0
   354
	{
sl@0
   355
	GetResolvedDllInfoL(aInterfaceUid, aAdditionalParameters, KDefaultResolverUid, aExtendedInterfaces, aDllInfo, aDtor_Key, aClientRequest);
sl@0
   356
	}
sl@0
   357
sl@0
   358
sl@0
   359
void CEComServer::GetResolvedDllInfoL(  const TUid aInterfaceUid,
sl@0
   360
										const TEComResolverParams& aAdditionalParameters,
sl@0
   361
										const TUid aResolverUid,
sl@0
   362
										const RExtendedInterfacesArray& aExtendedInterfaces,
sl@0
   363
										TEntry& aDllInfo,
sl@0
   364
										TUid& aDtor_Key,
sl@0
   365
										const TClientRequest& aClientRequest)
sl@0
   366
	{
sl@0
   367
	CResolver* resolver = NULL;
sl@0
   368
	TBool capability= ETrue;
sl@0
   369
	//create registry resolver transaction
sl@0
   370
	CRegistryResolveTransaction* registryResolveTransaction = CRegistryResolveTransaction::NewL(*iRegistryData,
sl@0
   371
																								aExtendedInterfaces,
sl@0
   372
																								aClientRequest,capability);
sl@0
   373
	CleanupStack::PushL(registryResolveTransaction);
sl@0
   374
	//create resolver
sl@0
   375
	resolver = CreateResolverLC(aResolverUid,registryResolveTransaction);
sl@0
   376
	aDtor_Key = resolver->IdentifyImplementationL(aInterfaceUid, aAdditionalParameters);
sl@0
   377
sl@0
   378
	//clean up
sl@0
   379
	CleanupStack::PopAndDestroy(resolver);
sl@0
   380
	CleanupStack::PopAndDestroy(registryResolveTransaction);
sl@0
   381
	if ((aResolverUid != KDefaultResolverUid) && (aResolverUid != KRomOnlyResolverUid))
sl@0
   382
		{
sl@0
   383
		iResolverLibrary.Close();
sl@0
   384
		}
sl@0
   385
	CImplementationInformation* implInfo = NULL;
sl@0
   386
	//Don't need to do the security check because it has been done in IdentifyImplementationL.
sl@0
   387
	User::LeaveIfError(iRegistryData->GetImplementationDllInfoForClientL(
sl@0
   388
		aClientRequest, aDtor_Key, aInterfaceUid, aDllInfo, implInfo, EFalse));
sl@0
   389
	}
sl@0
   390
sl@0
   391
// Server Session management
sl@0
   392
CSession2* CEComServer::NewSessionL(const TVersion& aVersion,const RMessage2& /* aMessage*/) const
sl@0
   393
	{
sl@0
   394
	const TVersionName version = CONST_CAST(TVersion&,aVersion).Name();
sl@0
   395
	const TVersionName thisVersion = TVersion(KEComServerMajorVN,KEComServerMinorVN,KEComServerBuildVN).Name();
sl@0
   396
	if(thisVersion != version)
sl@0
   397
		User::Leave(KErrNotSupported);
sl@0
   398
	return new(ELeave) CEComServerSession();
sl@0
   399
	}
sl@0
   400
sl@0
   401
//
sl@0
   402
// A new session is being created
sl@0
   403
// Cancel the shutdown timer if it was running
sl@0
   404
//
sl@0
   405
void CEComServer::AddSession()
sl@0
   406
	{
sl@0
   407
	++iSessionCount;
sl@0
   408
	iShutdown.Cancel();
sl@0
   409
	}
sl@0
   410
sl@0
   411
//
sl@0
   412
// A session is being destroyed
sl@0
   413
// Start the shutdown timer if it is the last session.
sl@0
   414
//
sl@0
   415
void CEComServer::DropSession()
sl@0
   416
	{
sl@0
   417
	if (--iSessionCount==0)
sl@0
   418
		iShutdown.Start();
sl@0
   419
	}
sl@0
   420
sl@0
   421
void CEComServer::Notification(TInt aCompletionCode)
sl@0
   422
//
sl@0
   423
// Pass on the signal to all clients
sl@0
   424
//
sl@0
   425
	{
sl@0
   426
	iSessionIter.SetToFirst();
sl@0
   427
	CSession2* s;
sl@0
   428
	while ((s = iSessionIter++)!=0)
sl@0
   429
		STATIC_CAST(CEComServerSession*,s)->CompleteNotifications(aCompletionCode);
sl@0
   430
	}
sl@0
   431
sl@0
   432
TInt CEComServer::RunError(TInt aError)
sl@0
   433
//
sl@0
   434
// Handle an error from CMySession::ServiceL()
sl@0
   435
// A bad descriptor error implies a badly programmed client, so panic it;
sl@0
   436
// otherwise report the error to the client
sl@0
   437
//
sl@0
   438
	{
sl@0
   439
	if (aError == KErrBadDescriptor)
sl@0
   440
		{
sl@0
   441
		PanicClient(Message(), aError);
sl@0
   442
		}
sl@0
   443
	else
sl@0
   444
		{
sl@0
   445
		Message().Complete(aError);
sl@0
   446
		}
sl@0
   447
	//
sl@0
   448
	// The leave will result in an early return from CServer::RunL(), skipping
sl@0
   449
	// the call to request another message. So do that now in order to keep the
sl@0
   450
	// server running.
sl@0
   451
	ReStart();
sl@0
   452
	return KErrNone;	// handled the error fully
sl@0
   453
	}
sl@0
   454
sl@0
   455
/**
sl@0
   456
Creates resolver object with aResolverUid UID.
sl@0
   457
The method leaves resolver onto the cleanup stack.
sl@0
   458
@param aResolverUid Resolver UID.
sl@0
   459
@param			aRegistryResolveTransaction A pointer to Registry resolve transaction object
sl@0
   460
@return A pointer to the created CResolver object.
sl@0
   461
@leave System-wide error codes, including KErrNoMemory.
sl@0
   462
*/
sl@0
   463
CResolver* CEComServer::CreateResolverLC(const TUid& aResolverUid,CRegistryResolveTransaction* aRegistryResolveTransaction)
sl@0
   464
	{
sl@0
   465
	CResolver* resolver = NULL;
sl@0
   466
	if(aResolverUid == KDefaultResolverUid)
sl@0
   467
		{
sl@0
   468
		// Create default resolver
sl@0
   469
		resolver = static_cast<CResolver*>(CDefaultResolver::NewL(*aRegistryResolveTransaction));
sl@0
   470
		CleanupStack::PushL(resolver);
sl@0
   471
		}
sl@0
   472
	else if(aResolverUid == KRomOnlyResolverUid)
sl@0
   473
		{
sl@0
   474
		// Create Rom Only resolver
sl@0
   475
		resolver = static_cast<CResolver*>(CRomOnlyResolver::NewL(*aRegistryResolveTransaction));
sl@0
   476
		CleanupStack::PushL(resolver);
sl@0
   477
		}
sl@0
   478
	else
sl@0
   479
		{
sl@0
   480
		// Create Custom Resolver
sl@0
   481
		resolver = CreateCustomResolverLC(aResolverUid, aRegistryResolveTransaction);
sl@0
   482
		}
sl@0
   483
	return resolver;
sl@0
   484
	}
sl@0
   485
sl@0
   486
/**
sl@0
   487
Creates custom resolver object with aResolverUid UID.
sl@0
   488
The method leaves custom resolver onto the cleanup stack.
sl@0
   489
@param aResolverUid Custom resolver UID.
sl@0
   490
@param			aRegistryResolveTransaction A pointer to Registry resolve transaction object
sl@0
   491
@return A pointer to the created CResolver object.
sl@0
   492
@leave System-wide error codes, including KErrNoMemory.
sl@0
   493
*/
sl@0
   494
CResolver* CEComServer::CreateCustomResolverLC(TUid aResolverUid,CRegistryResolveTransaction* aRegistryResolveTransaction)
sl@0
   495
	{
sl@0
   496
	typedef CResolver* (*TNewL)(MPublicRegistry&);
sl@0
   497
	TNewL newL = NULL;
sl@0
   498
	CResolver* resolver=NULL;
sl@0
   499
sl@0
   500
	TProxyNewLPtr tmpPtr;
sl@0
   501
	if (iResolverCache->CacheLookup(aResolverUid, tmpPtr)) // cache hit
sl@0
   502
		{
sl@0
   503
		newL = reinterpret_cast<TNewL>(tmpPtr);
sl@0
   504
		resolver = newL(*aRegistryResolveTransaction);
sl@0
   505
		CleanupStack::PushL(resolver);
sl@0
   506
		return resolver;
sl@0
   507
		}
sl@0
   508
sl@0
   509
	TEntry resolverDllInfo;
sl@0
   510
	// We should only load custom resolvers that are in the ROM
sl@0
   511
	//Initialize the server cap to ProtServ
sl@0
   512
	TCapabilitySet servercap(ECapabilityProtServ);
sl@0
   513
	CImplementationInformation* implInfo = NULL;
sl@0
   514
	TBool onWritableDrv = EFalse;  
sl@0
   515
	User::LeaveIfError(iRegistryData->GetImplementationDllInfoForServer(
sl@0
   516
		servercap, aResolverUid, KEComResolverInterfaceUid, resolverDllInfo,
sl@0
   517
		implInfo, onWritableDrv));
sl@0
   518
sl@0
   519
	// Type of the function pointer which is the proxy into the interface implementation collection
sl@0
   520
	typedef TImplementationProxy* (*TInstantiationL)(TInt&);
sl@0
   521
	// Function at ordinal 1 is InstantiationMethodL()
sl@0
   522
	const TInt KImplementationGroupProxy = 1;
sl@0
   523
	// So cast to the correct type : This gives an ANSI C++ warning
sl@0
   524
	// When using a REINTERPRET_CAST so simply cast instead
sl@0
   525
sl@0
   526
	const TDesC& libraryPath = resolverDllInfo.iName;
sl@0
   527
	// Make sure the non-default resolver library is closed
sl@0
   528
	iResolverLibrary.Close();
sl@0
   529
	User::LeaveIfError(iResolverLibrary.Load(libraryPath, resolverDllInfo.iType));
sl@0
   530
	__ECOM_TRACE2("ECOM: Resolver Loaded UID:0x%X - %S", aResolverUid.iUid, &resolverDllInfo.iName);
sl@0
   531
	TInstantiationL proxy= REINTERPRET_CAST(TInstantiationL, iResolverLibrary.Lookup(KImplementationGroupProxy));
sl@0
   532
sl@0
   533
	// Scan the returned table for a UID match, and return the associated
sl@0
   534
	// creation method if found.
sl@0
   535
	TInt count = 0;
sl@0
   536
	TImplementationProxy* implementationTable = proxy(count);
sl@0
   537
	for(TInt i = 0; i < count; ++i)
sl@0
   538
		{
sl@0
   539
		if(aResolverUid == implementationTable[i].iImplementationUid)
sl@0
   540
			{
sl@0
   541
			newL = (TNewL)(implementationTable[i].iNewLFuncPtr);
sl@0
   542
			}
sl@0
   543
		}
sl@0
   544
sl@0
   545
	if(newL)
sl@0
   546
		{
sl@0
   547
		if (IsCachable(onWritableDrv))
sl@0
   548
			{
sl@0
   549
			TUint32 flags = (onWritableDrv) ? EEntryIsOnReadWriteDrive : EEntryFlagsNone;
sl@0
   550
			User::LeaveIfError(iResolverCache->CacheResolver(aResolverUid,
sl@0
   551
				iResolverLibrary, (TProxyNewLPtr)newL, flags));
sl@0
   552
			// Handle is now owned by iResolverCache.
sl@0
   553
			iResolverLibrary.SetHandle(KNullHandle);
sl@0
   554
			}
sl@0
   555
sl@0
   556
		// Create the non-default resolver
sl@0
   557
		resolver = newL(*aRegistryResolveTransaction);
sl@0
   558
		CleanupStack::PushL(resolver);
sl@0
   559
		}
sl@0
   560
	else
sl@0
   561
		{
sl@0
   562
		User::Leave(KEComErrNoResolver);
sl@0
   563
		}
sl@0
   564
	return resolver;
sl@0
   565
	}
sl@0
   566
sl@0
   567
sl@0
   568
TBool CEComServer::RegistryIndexValid() const
sl@0
   569
	{
sl@0
   570
	return iRegistryData->IndexValid();
sl@0
   571
	}
sl@0
   572
	
sl@0
   573
/** Callback function. CRegistryData uses this to notify of implementation
sl@0
   574
upgrade. CDiscoverer uses this to notify state changes in SWI/BUR.
sl@0
   575
@param aObj Pointer to CEComServer object.
sl@0
   576
@param aEvent Identify the event.
sl@0
   577
@param aData Data associated with the callback.
sl@0
   578
@return none, not-used, ignored.
sl@0
   579
*/
sl@0
   580
TInt CEComServer::NotifyEvents(TAny* aObj, TInt aEvent, TAny* aData)
sl@0
   581
	{
sl@0
   582
	CEComServer* self = static_cast<CEComServer*>(aObj);
sl@0
   583
	switch (aEvent)
sl@0
   584
		{
sl@0
   585
		case ECallBackId_ImplUpgrade:
sl@0
   586
			{
sl@0
   587
			TUid* uid = static_cast<TUid*>(aData);
sl@0
   588
			self->NotifyUpgrade(*uid);
sl@0
   589
			}
sl@0
   590
			break;
sl@0
   591
		case ECallBackId_SwiEvent:
sl@0
   592
			self->NotifySWIEvent(aData);
sl@0
   593
			break;
sl@0
   594
		case ECallBackId_BurEvent:
sl@0
   595
			self->NotifyBUREvent(aData);
sl@0
   596
			break;
sl@0
   597
		default:
sl@0
   598
			__ECOM_TRACE1("ECOM: CEComServer::NotifyEvents received unknown event %d", aEvent);
sl@0
   599
		}
sl@0
   600
sl@0
   601
	return 0;
sl@0
   602
	}
sl@0
   603
sl@0
   604
/** This method is called when an implementation is upgraded.
sl@0
   605
@param aImplementationUid identify the implementation being upgraded.
sl@0
   606
*/
sl@0
   607
void CEComServer::NotifyUpgrade(const TUid aImplementationUid)
sl@0
   608
	{
sl@0
   609
	// Ignore return code which indicates if the UID is actually in cache.
sl@0
   610
	(void)iResolverCache->Remove(aImplementationUid);
sl@0
   611
	}
sl@0
   612
sl@0
   613
/** Called when there is SWI status change.
sl@0
   614
@param aData is TCallBackState* indicating start or end of SWI. 
sl@0
   615
*/
sl@0
   616
void CEComServer::NotifySWIEvent(TAny* aData)
sl@0
   617
	{
sl@0
   618
	TCallBackState* state = static_cast<TCallBackState*>(aData);
sl@0
   619
	UpdateSpecialEvents(ESWIInProgress, *state);
sl@0
   620
	}
sl@0
   621
sl@0
   622
/** Called when there is BUR status change.
sl@0
   623
@param aData is TCallBackState* indicating start or end of BUR. 
sl@0
   624
*/
sl@0
   625
void CEComServer::NotifyBUREvent(TAny* aData)
sl@0
   626
	{
sl@0
   627
	TCallBackState* state = static_cast<TCallBackState*>(aData);
sl@0
   628
	UpdateSpecialEvents(EBURInProgress, *state);
sl@0
   629
	}
sl@0
   630
sl@0
   631
/** Updates the BUR/SWI status.
sl@0
   632
@param aBit Indicate which bit to update.
sl@0
   633
@param aState Indicate start or end of event.
sl@0
   634
*/
sl@0
   635
void CEComServer::UpdateSpecialEvents(TUint32 aBit, TCallBackState aState)
sl@0
   636
	{
sl@0
   637
	TBitFlags32 oldstate = iSpecialEvents;
sl@0
   638
sl@0
   639
	if (aState == ECallBackState_EventStart)
sl@0
   640
		{
sl@0
   641
		iSpecialEvents.Set( aBit );
sl@0
   642
		}
sl@0
   643
	else
sl@0
   644
		{
sl@0
   645
		iSpecialEvents.Clear( aBit );
sl@0
   646
		}
sl@0
   647
sl@0
   648
	if (oldstate.Value() == 0 && iSpecialEvents.Value() != 0)
sl@0
   649
		{
sl@0
   650
		// BUR or SWI start. Need to evict cached resolvers on RW drives.
sl@0
   651
		iResolverCache->RemoveItemsWithFlags(EEntryIsOnReadWriteDrive);
sl@0
   652
		}
sl@0
   653
	}
sl@0
   654
sl@0
   655
/** Determine if a resolver entry is cachable.
sl@0
   656
@param aResolverEntry the resolver to check.
sl@0
   657
@return ETrue if the resolver should be added to cache. EFalse otherwise.
sl@0
   658
*/
sl@0
   659
TBool CEComServer::IsCachable(TBool aEntryIsOnRWDrive)
sl@0
   660
	{
sl@0
   661
	// Check the following conditions:
sl@0
   662
	// 1. DLL is on RW drive with BUR or SWI in progress.
sl@0
   663
	// 2. Cache size and cache timeout non-zero.
sl@0
   664
	return	iResolverCache->CachingEnabled() &&
sl@0
   665
			!(iSpecialEvents.Value() && aEntryIsOnRWDrive);
sl@0
   666
	}
sl@0
   667
sl@0
   668
#ifdef __ECOM_SERVER_TESTABILITY__
sl@0
   669
void CEComServer::ChangeStartupStateL(TInt aState) const
sl@0
   670
	{
sl@0
   671
	iServerStartupMgr->ChangeStartupStateL(aState);
sl@0
   672
	}
sl@0
   673
sl@0
   674
void CEComServer::ProcessCurrentStartupStateL() const
sl@0
   675
	{
sl@0
   676
	iServerStartupMgr->ResetRequestTransitionNotificationL();
sl@0
   677
	iServerStartupMgr->ResetLastStateAcknowledgedL();
sl@0
   678
	iServerStartupMgr->RunL();
sl@0
   679
	}
sl@0
   680
	
sl@0
   681
TInt CEComServer::GetCurrentStartupState() const
sl@0
   682
	{
sl@0
   683
	return iServerStartupMgr->CurrentStartupState();
sl@0
   684
	}
sl@0
   685
#endif //__ECOM_SERVER_TESTABILITY__
sl@0
   686
sl@0
   687
#ifdef __ECOM_SERVER_PERFORMANCE__
sl@0
   688
void CEComServer::GetRegistryCountsL(TInt aType, RegistryCounts::TRegistryCounts& aCounts) const
sl@0
   689
	{
sl@0
   690
	iRegistryData->GetRegistryCountsL(aType, aCounts);
sl@0
   691
	}	
sl@0
   692
#endif //__ECOM_SERVER_PERFORMANCE__