os/ossrv/lowlevellibsandfws/pluginfw/Framework/frame/LoadManager.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 CLoadManager 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 "EComDebug.h"
sl@0
    24
#include "UnloadPolicy.h"
sl@0
    25
#include "LoadManager.h"
sl@0
    26
#include "EComUidCodes.h"
sl@0
    27
#include <ecom/ecomerrorcodes.h>
sl@0
    28
#include "e32math.h"
sl@0
    29
#include "EComInternalErrorCodes.h"
sl@0
    30
#include <ecom/ecomextendedinterfaceerrorcodes.h>
sl@0
    31
/**
sl@0
    32
Standardized safe construction which leaves nothing on the cleanup stack.
sl@0
    33
@return			A pointer to the new class
sl@0
    34
@post			CLoadManager is fully constructed, and initialized.
sl@0
    35
 */
sl@0
    36
CLoadManager* CLoadManager::NewL()
sl@0
    37
	{
sl@0
    38
	return new(ELeave) CLoadManager();
sl@0
    39
	}
sl@0
    40
sl@0
    41
CLoadManager::~CLoadManager()
sl@0
    42
	{
sl@0
    43
	iInstanceInfoList.ResetAndDestroy();
sl@0
    44
	iAllUnloadPolicies.ResetAndDestroy();
sl@0
    45
	ClearGarbage();
sl@0
    46
	}
sl@0
    47
sl@0
    48
/**
sl@0
    49
Notifies the interface implementation DLL that one of its objects has been destroyed if it exists,
sl@0
    50
otherwise returns false to indicate no exist aImplementationUid.
sl@0
    51
@param          aInstanceKey A key specifying a previously created implementation.
sl@0
    52
@pre 			CLoadManager is fully constructed,
sl@0
    53
@post			CLoadManager's interface implementation DLL references
sl@0
    54
				are decreased by one. The instance info representing the implementation
sl@0
    55
				is destroyed.
sl@0
    56
*/
sl@0
    57
TBool CLoadManager::DestroyedThis(TUid aInstanceKey)
sl@0
    58
	{
sl@0
    59
	// Clear the garbage list because we know we have finished with them
sl@0
    60
	ClearGarbage();
sl@0
    61
sl@0
    62
	__ECOM_TRACE1("ECOM: Implementation Instance destroyed %x", aInstanceKey.iUid);
sl@0
    63
sl@0
    64
	// Sanity check that the pointer is divisible by four. A binary number is divisible
sl@0
    65
	// by four if and only if the two rightmost bits are both zero.
sl@0
    66
	// This is a compromised check for checking that the pointer is an address.
sl@0
    67
    if ((aInstanceKey.iUid == 0) || ((aInstanceKey.iUid & 0x00000003) != 0))
sl@0
    68
        {
sl@0
    69
	    __ASSERT_DEBUG(EFalse, User::Panic(KEComClientDLLPanicCategory, EEComPanic_InvalidImplementationInstanceKey));
sl@0
    70
	    User::Leave(KErrNotFound);
sl@0
    71
        }
sl@0
    72
sl@0
    73
	// The instance info pointer is stored in the instance key.
sl@0
    74
	CInstanceInfo* instanceInfo = reinterpret_cast<CInstanceInfo*>(aInstanceKey.iUid);
sl@0
    75
sl@0
    76
	// Check that the pointer exists before using it
sl@0
    77
	TInt position = iInstanceInfoList.FindInAddressOrder(instanceInfo);
sl@0
    78
	if(position == KErrNotFound)
sl@0
    79
		{
sl@0
    80
		return EFalse;
sl@0
    81
		}
sl@0
    82
	
sl@0
    83
	CUnloadPolicy* policy = instanceInfo->UnloadPolicy();
sl@0
    84
	if (policy != NULL)
sl@0
    85
		{
sl@0
    86
		// If needed, will move the policy to the garbage list, and remove policy from the unload policy list.
sl@0
    87
		Cleanup(policy);
sl@0
    88
		}
sl@0
    89
	iInstanceInfoList.Remove(position);
sl@0
    90
	// Remove the instance info item, finished with it.
sl@0
    91
	delete instanceInfo;
sl@0
    92
	return ETrue;
sl@0
    93
	}
sl@0
    94
sl@0
    95
/**
sl@0
    96
Check whether the policy Arrays are empty or not.
sl@0
    97
@return            Returns True if the policy Arrays are empty, otherwise return False.
sl@0
    98
@pre             CLoadManager is fully constructed,
sl@0
    99
@post            CLoadManager remains the same.
sl@0
   100
*/
sl@0
   101
TBool CLoadManager::PolicyArraysEmpty() const 
sl@0
   102
    {
sl@0
   103
    if( (iAllUnloadPolicies.Count() == 0)&&(iInstanceInfoList.Count() == 0) )
sl@0
   104
        {
sl@0
   105
        return ETrue;
sl@0
   106
        }
sl@0
   107
    else
sl@0
   108
        {
sl@0
   109
        return EFalse;
sl@0
   110
        }
sl@0
   111
    }
sl@0
   112
sl@0
   113
/**
sl@0
   114
Returns an implementation object to satisfy the specified interface.
sl@0
   115
@param			aUniqueImplementationUid The implementation to find.
sl@0
   116
@param			aEntry Information on the dll containing the implementation
sl@0
   117
@param			aCreationParameters A pointer to the creation parameter
sl@0
   118
				structure passed to the creation method when called.
sl@0
   119
@param			aCreationParamsFlag A boolean flag to indicate the existence or non-existence
sl@0
   120
				of aCreationParameters. Will be ETrue even for if the value of
sl@0
   121
				aCreationParameters is NULL.
sl@0
   122
@param          aInstanceKey A key specifying a previously created implementation.
sl@0
   123
@return			TAny* a pointer to the fully constructed instantiation. The pointer is
sl@0
   124
				guaranteed to be valid only until DestroyedThis is called.
sl@0
   125
@pre 			CLoadManager is fully constructed,
sl@0
   126
@post			Fully constructed implementation is returned to the
sl@0
   127
				caller, and aUniqueUid contains the implementation Dll's
sl@0
   128
				unique UID.
sl@0
   129
*/
sl@0
   130
TAny* CLoadManager::ImplementationObjectL(const TUid& aUniqueImplementationUid,
sl@0
   131
										 const TEntry& aEntry,
sl@0
   132
										 TAny* aCreationParameters,
sl@0
   133
										 TBool aCreationParamsFlag,
sl@0
   134
										 TUid& aInstanceKey)
sl@0
   135
	{
sl@0
   136
	//if the implementation Uid here is Null or the entry returned by the ecomserver
sl@0
   137
	//contains nothing, we should leave with KErrNotFound
sl@0
   138
	if(aUniqueImplementationUid == KNullUid || (aEntry.iName).Length()==0)
sl@0
   139
		{
sl@0
   140
		User::Leave(KErrNotFound);
sl@0
   141
		}
sl@0
   142
sl@0
   143
	TAny* object = NULL;               // Instantiation object
sl@0
   144
	CUnloadPolicy* policy = NULL;      // Policy of implementation
sl@0
   145
	TLibraryFunction libFunctionProxy; // Library function proxy pointer
sl@0
   146
sl@0
   147
	GetUnloadPolicy(aUniqueImplementationUid,aEntry,policy);
sl@0
   148
	if (policy == NULL)
sl@0
   149
		{
sl@0
   150
		// No policy found, so create a new CUnloadPolicy and load DLL.
sl@0
   151
		policy = CUnloadPolicy::NewLC(aEntry);
sl@0
   152
		libFunctionProxy = policy->LoadDllAndReturnProxyL();
sl@0
   153
		if (libFunctionProxy==NULL)
sl@0
   154
			{
sl@0
   155
			User::Leave(KErrNotFound);
sl@0
   156
			}
sl@0
   157
		iAllUnloadPolicies.AppendL(policy);
sl@0
   158
		CleanupStack::Pop(policy);	// owned by iAllUnloadPolicies
sl@0
   159
		}
sl@0
   160
	else
sl@0
   161
		{
sl@0
   162
		// Found a policy. Load Dll if not already loaded.
sl@0
   163
		libFunctionProxy = policy->LoadDllAndReturnProxyL();
sl@0
   164
		if (libFunctionProxy==NULL)
sl@0
   165
			{
sl@0
   166
			User::Leave(KErrNotFound);
sl@0
   167
			}
sl@0
   168
		}
sl@0
   169
	// Initial count of instances. This is used for cleanup purposes, to see if an instance
sl@0
   170
	// was added to the instance info list during the object creation. If a failure occurs
sl@0
   171
	// than the instance info that was added to the list will be removed.
sl@0
   172
	TInt initialCount = iInstanceInfoList.Count();
sl@0
   173
	TInt err = KErrNone;
sl@0
   174
sl@0
   175
	if (aEntry[1] == KUidInterfaceImplementationCollection)
sl@0
   176
		{
sl@0
   177
		// PLUGIN1 dll. Create the implementation object.
sl@0
   178
		TRAP(err,object = ImplementationObject1L(aUniqueImplementationUid,aCreationParameters,
sl@0
   179
			aCreationParamsFlag,aInstanceKey,policy,libFunctionProxy));
sl@0
   180
		}
sl@0
   181
	else if (aEntry[1] == KUidInterfaceImplementationCollection3)
sl@0
   182
		{
sl@0
   183
		// PLUGIN3 dll. Create the implementation object.
sl@0
   184
		TRAP(err,object = ImplementationObject3L(aUniqueImplementationUid,aCreationParameters,
sl@0
   185
			aCreationParamsFlag, aInstanceKey,policy, libFunctionProxy));
sl@0
   186
		}
sl@0
   187
	else
sl@0
   188
		{
sl@0
   189
		err = KErrNotSupported;
sl@0
   190
		}
sl@0
   191
	
sl@0
   192
	if (err != KErrNone)
sl@0
   193
		{
sl@0
   194
		if (iInstanceInfoList.Count() > initialCount)
sl@0
   195
			{
sl@0
   196
			// If an instance was added to the instance list, remove it here.
sl@0
   197
			// The instance info pointer is stored in the instance key. We know its
sl@0
   198
			// valid because it was set by ecom in call to ImplementationObject1L or
sl@0
   199
			// ImplementationObject3L
sl@0
   200
			CInstanceInfo* instanceInfo = reinterpret_cast<CInstanceInfo*>(aInstanceKey.iUid);
sl@0
   201
			TInt pos = iInstanceInfoList.FindInAddressOrder(instanceInfo);
sl@0
   202
			if(pos != KErrNotFound)
sl@0
   203
				{
sl@0
   204
				iInstanceInfoList.Remove(pos);
sl@0
   205
				}
sl@0
   206
			}
sl@0
   207
sl@0
   208
		ClearGarbage();
sl@0
   209
		// If needed, will move the policy to the garbage list, and remove policy from the unload policy list.
sl@0
   210
		Cleanup(policy);
sl@0
   211
		User::Leave(err);
sl@0
   212
		}
sl@0
   213
		
sl@0
   214
	__ECOM_TRACE2("ECOM: Implementation created (%03d) %x", ++iDebugInstantiationCounter, aUniqueImplementationUid.iUid);
sl@0
   215
			
sl@0
   216
	return object;
sl@0
   217
	}
sl@0
   218
sl@0
   219
/**
sl@0
   220
Returns an implementation object to satisfy the specified interface.
sl@0
   221
@param			aUniqueImplementationUid The implementation to find.
sl@0
   222
@param			aCreationParameters A pointer to the creation parameter
sl@0
   223
				structure passed to the creation method when called.
sl@0
   224
@param			aCreationParamsFlag A boolean flag to indicate the existence or non-existence
sl@0
   225
				of aCreationParameters. Will be ETrue even for if the value of
sl@0
   226
				aCreationParameters is NULL.
sl@0
   227
@param          aInstanceKey A key specifying a previously created implementation.
sl@0
   228
@param          aPolicy policy of implementation
sl@0
   229
@param          aLibFunctionProxy Library function proxy pointer.
sl@0
   230
@return			TAny* a pointer to the fully constructed instantiation. The pointer is
sl@0
   231
				guaranteed to be valid only until DestroyedThis is called.
sl@0
   232
*/
sl@0
   233
TAny* CLoadManager::ImplementationObject1L(const TUid& aUniqueImplementationUid,
sl@0
   234
										 TAny* aCreationParameters,
sl@0
   235
										 TBool aCreationParamsFlag,
sl@0
   236
										 TUid& aInstanceKey,
sl@0
   237
										 CUnloadPolicy* aPolicy,
sl@0
   238
										 TLibraryFunction& aLibFunctionProxy)
sl@0
   239
	{
sl@0
   240
	TAny* object = NULL; // Instantiation object
sl@0
   241
	TInstantiationL proxy = reinterpret_cast<TInstantiationL>(aLibFunctionProxy);
sl@0
   242
sl@0
   243
	TImplementationProxy* implementationProxyRow = NULL;
sl@0
   244
	TAny* newLPointer = GetNewLPointerAndProxyTableRowL<TImplementationProxy,TInstantiationL>(aUniqueImplementationUid,implementationProxyRow,proxy);
sl@0
   245
	// Now create an instance info object to store the information about this instance derived from
sl@0
   246
	// the parameters just fetched. This must be created here since no leaves can occur after the object
sl@0
   247
	// instantiation below.
sl@0
   248
	CInstanceInfoSimple* instanceInfoSimple = CInstanceInfoSimple::NewL(aPolicy,implementationProxyRow);
sl@0
   249
	CleanupStack::PushL(instanceInfoSimple);
sl@0
   250
sl@0
   251
	// The pointer to instanceInfo will be used to identify this instance, and will
sl@0
   252
	// be returned to the caller for future identification of this instance.
sl@0
   253
	aInstanceKey.iUid = reinterpret_cast<TInt32>(instanceInfoSimple);
sl@0
   254
sl@0
   255
	// Add item to instance info list. This list will contain all of the instance objects.
sl@0
   256
	iInstanceInfoList.InsertInAddressOrderL(instanceInfoSimple);
sl@0
   257
sl@0
   258
	// Get the implementation object using the instantiation pointer fetched earlier. Note
sl@0
   259
	// that the object creation must be the last leaving operation to be performed. No leaves can occur
sl@0
   260
	// after this operation, as the instantiation object cannnot be deleted on the cleanup stack if
sl@0
   261
	// a leave occurs (cannot delete a TAny* pointer, the type is not known within ECOM).
sl@0
   262
	object = ImplementationObjectL(aCreationParameters, 
sl@0
   263
	 aCreationParamsFlag,newLPointer);
sl@0
   264
sl@0
   265
	CleanupStack::Pop(instanceInfoSimple);
sl@0
   266
sl@0
   267
	return object;
sl@0
   268
	}
sl@0
   269
sl@0
   270
/**
sl@0
   271
Returns an implementation object to satisfy the specified interface.
sl@0
   272
@param			aUniqueImplementationUid The implementation to find.
sl@0
   273
@param			aCreationParameters A pointer to the creation parameter
sl@0
   274
				structure passed to the creation method when called.
sl@0
   275
@param			aCreationParamsFlag A boolean flag to indicate the existence or non-existence
sl@0
   276
				of aCreationParameters. Will be ETrue even for if the value of
sl@0
   277
				aCreationParameters is NULL.
sl@0
   278
@param          aInstanceKey A key specifying a previously created implementation.
sl@0
   279
@param          aPolicy policy of implementation
sl@0
   280
@param          aLibFunctionProxy Library function proxy pointer.
sl@0
   281
@return			TAny* a pointer to the fully constructed instantiation. The pointer is
sl@0
   282
				guaranteed to be valid only until DestroyedThis is called.
sl@0
   283
*/
sl@0
   284
TAny* CLoadManager::ImplementationObject3L(const TUid& aUniqueImplementationUid,
sl@0
   285
										 TAny* aCreationParameters,
sl@0
   286
										 TBool aCreationParamsFlag,
sl@0
   287
										 TUid& aInstanceKey,
sl@0
   288
										 CUnloadPolicy* aPolicy,
sl@0
   289
										 TLibraryFunction& aLibFunctionProxy)
sl@0
   290
	{
sl@0
   291
	TAny* object = NULL;  // Instantiation object
sl@0
   292
	TInstantiation3L proxy = reinterpret_cast<TInstantiation3L>(aLibFunctionProxy);
sl@0
   293
sl@0
   294
	TImplementationProxy3* implementationProxyRow = NULL;
sl@0
   295
	TAny* newLPointer = GetNewLPointerAndProxyTableRowL<TImplementationProxy3,TInstantiation3L>(aUniqueImplementationUid,implementationProxyRow,proxy);
sl@0
   296
sl@0
   297
	// Now create an instance info object to store the information about this instance derived from
sl@0
   298
	// the parameters just fetched. This must be created here since no leaves can occur after the object
sl@0
   299
	// instantiation below.
sl@0
   300
	CInstanceInfoExtended* instanceInfoExtended = CInstanceInfoExtended::NewL(aPolicy,implementationProxyRow);
sl@0
   301
	CleanupStack::PushL(instanceInfoExtended);
sl@0
   302
sl@0
   303
	// The pointer to instanceInfo will be used to identify this instance, and will
sl@0
   304
	// be returned to the caller for future identification of this instance.
sl@0
   305
	aInstanceKey.iUid = reinterpret_cast<TInt32>(instanceInfoExtended);
sl@0
   306
sl@0
   307
	// Add item to instance info list. This list will contain all of the instance objects.
sl@0
   308
	iInstanceInfoList.InsertInAddressOrderL(instanceInfoExtended);
sl@0
   309
sl@0
   310
	// Get the implementation object using the instantiation pointer fetched earlier. Note
sl@0
   311
	// that the object creation must be the last leaving operation to be performed. No leaves can occur
sl@0
   312
	// after this operation, as the instantiation object cannnot be deleted on the cleanup stack if
sl@0
   313
	// a leave occurs (cannot delete a TAny* pointer, the type is not known within ECOM).
sl@0
   314
	object = ImplementationObjectL(aCreationParameters, 
sl@0
   315
	 aCreationParamsFlag,newLPointer);
sl@0
   316
sl@0
   317
	// The extended instance info requires an object set. This is done here
sl@0
   318
	// as the object is not known until now, but the instance info object was required to
sl@0
   319
	// be created earlier to avoid a leave after the object creation.
sl@0
   320
	instanceInfoExtended->SetObject(object);
sl@0
   321
	CleanupStack::Pop(instanceInfoExtended);
sl@0
   322
sl@0
   323
	return object;
sl@0
   324
	}
sl@0
   325
sl@0
   326
/**
sl@0
   327
Gets the main implementation object using the instantiation pointer provided
sl@0
   328
@param			aCreationParameters A pointer to the creation parameter
sl@0
   329
				structure passed to the creation method when called.
sl@0
   330
@param			aCreationParamsFlag A boolean flag to indicate the existence or non-existence
sl@0
   331
				of aCreationParameters. Will be ETrue even for if the value of
sl@0
   332
				aCreationParameters is NULL.
sl@0
   333
@param			aNewLpointer Instantation pointer.
sl@0
   334
				aCreationParameters is NULL.
sl@0
   335
@return			TAny* a pointer to the fully constructed instantiation.
sl@0
   336
*/
sl@0
   337
TAny* CLoadManager::ImplementationObjectL(TAny* aCreationParameters,
sl@0
   338
										 TBool aCreationParamsFlag,
sl@0
   339
										 const TAny* aNewLpointer)
sl@0
   340
	{
sl@0
   341
	TAny* object=NULL;
sl@0
   342
sl@0
   343
	// So cast to the correct type : This gives an ANSI C++ warning
sl@0
   344
	// When using a REINTERPRET_CAST so simply cast instead
sl@0
   345
	// Two different object creation one with creation parameters
sl@0
   346
	if (aCreationParamsFlag)
sl@0
   347
		{
sl@0
   348
		typedef TAny* (*TNewL)(TAny*);
sl@0
   349
		TNewL creationL = (TNewL)(aNewLpointer);
sl@0
   350
		object=creationL(aCreationParameters);
sl@0
   351
		}
sl@0
   352
	else
sl@0
   353
		{
sl@0
   354
		typedef TAny* (*TNewL)();
sl@0
   355
		TNewL creationL = (TNewL)(aNewLpointer);
sl@0
   356
		object=creationL();
sl@0
   357
		}
sl@0
   358
sl@0
   359
	return object;
sl@0
   360
	}
sl@0
   361
sl@0
   362
/**
sl@0
   363
Get the unload policy.
sl@0
   364
@param			aUniqueImplementationUid The implementation to find.
sl@0
   365
@param			aEntry Information on the dll containing the implementation
sl@0
   366
@param			aPolicy Return parameter containing the policy
sl@0
   367
@return			None
sl@0
   368
*/
sl@0
   369
void CLoadManager::GetUnloadPolicy(TUid aUniqueImplementationUid,
sl@0
   370
							  const TEntry& aEntry,
sl@0
   371
							  CUnloadPolicy*& aPolicy)
sl@0
   372
	{
sl@0
   373
	const TInt numImps = iInstanceInfoList.Count();
sl@0
   374
	TBool foundImp = EFalse;
sl@0
   375
	TInt matchingDllIndex=0;
sl@0
   376
	TBool matchingDllFound=EFalse;
sl@0
   377
	CUnloadPolicy* policy = NULL;
sl@0
   378
sl@0
   379
	for (TInt index = 0; (index<numImps) && !foundImp; ++index)
sl@0
   380
		{
sl@0
   381
		//Check if there is existing mapping between the supplied implUid and a policy
sl@0
   382
		if(iInstanceInfoList[index]->ImplementationUid() == aUniqueImplementationUid)
sl@0
   383
			{
sl@0
   384
			policy = iInstanceInfoList[index]->UnloadPolicy();
sl@0
   385
sl@0
   386
			foundImp = ETrue;
sl@0
   387
			}
sl@0
   388
		else
sl@0
   389
			{
sl@0
   390
			//if cannot find a mapping entry but current index has the same DLL as the one requested from
sl@0
   391
			//the client, store the index to this unloadPolicy for use later on if we have finished
sl@0
   392
			//searching the entire mapping list
sl@0
   393
			if (!matchingDllFound && (iInstanceInfoList[index]->UnloadPolicy()->DllEntryInformation().GetName()).CompareF(aEntry.iName)==0)
sl@0
   394
				{
sl@0
   395
				matchingDllIndex=index;
sl@0
   396
				matchingDllFound=ETrue;
sl@0
   397
				}
sl@0
   398
			}
sl@0
   399
		}
sl@0
   400
sl@0
   401
	//If we cannot find any mapping in the policy index array(iUnloadPolicyMapping)
sl@0
   402
	if(!foundImp)
sl@0
   403
		{
sl@0
   404
		//if not found but there is a matching DLL,we can simply create a new policy index mapping without
sl@0
   405
		//having to load the library again.
sl@0
   406
		if (matchingDllFound)
sl@0
   407
			{
sl@0
   408
			policy = iInstanceInfoList[matchingDllIndex]->UnloadPolicy();
sl@0
   409
			}
sl@0
   410
		}
sl@0
   411
sl@0
   412
	aPolicy = policy;
sl@0
   413
	}
sl@0
   414
sl@0
   415
/**
sl@0
   416
Clears the policy inside the iGarbagePolicies attribute.
sl@0
   417
@pre			CLoadManager is fully constructed
sl@0
   418
@post			CLoadManager iGarbagePolicies is zero'd
sl@0
   419
*/
sl@0
   420
void CLoadManager::ClearGarbage()
sl@0
   421
	{
sl@0
   422
	if (iGarbagePolicy != NULL)
sl@0
   423
		{
sl@0
   424
		delete iGarbagePolicy;
sl@0
   425
		iGarbagePolicy=0;
sl@0
   426
		}
sl@0
   427
	}
sl@0
   428
sl@0
   429
CLoadManager::CLoadManager() :
sl@0
   430
	CBase()
sl@0
   431
	{
sl@0
   432
	// Do nothing here
sl@0
   433
	}
sl@0
   434
sl@0
   435
/**
sl@0
   436
Returns the implementation ID for a given instance Key.
sl@0
   437
@leave			KErrNotFound
sl@0
   438
@param			aInstanceKey A key specifying a previously created implementation instance.
sl@0
   439
@return			TUid The uid of the corresponding implementation.
sl@0
   440
@pre 			CLoadManager is fully constructed,
sl@0
   441
@post			CLoadManager remains the same.
sl@0
   442
*/
sl@0
   443
TUid CLoadManager::GetImplementationUidL(TUid aInstanceKey)
sl@0
   444
	{
sl@0
   445
	// Sanity check that the pointer is divisible by four. A binary number is divisible
sl@0
   446
	// by four if and only if the two rightmost bits are both zero.
sl@0
   447
	// This is a compromised check for checking that the pointer is an address.
sl@0
   448
    if ((aInstanceKey.iUid == 0) || ((aInstanceKey.iUid & 0x00000003) != 0))
sl@0
   449
        {
sl@0
   450
	    __ASSERT_DEBUG(EFalse, User::Panic(KEComClientDLLPanicCategory, EEComPanic_InvalidImplementationInstanceKey));
sl@0
   451
	    User::Leave(KErrNotFound);
sl@0
   452
        }
sl@0
   453
sl@0
   454
	// The instance info pointer is stored in the instance key.
sl@0
   455
	CInstanceInfo* instanceInfo = reinterpret_cast<CInstanceInfo*>(aInstanceKey.iUid);
sl@0
   456
sl@0
   457
	// Check that the pointer exists before using it - leaves with KErrNotFound
sl@0
   458
	iInstanceInfoList.FindInAddressOrderL(instanceInfo);
sl@0
   459
	return instanceInfo->ImplementationUid();
sl@0
   460
	}
sl@0
   461
sl@0
   462
/**
sl@0
   463
Fetches the requested extended interface from a specified implementation instance.
sl@0
   464
@leave			KErrNotFound
sl@0
   465
@param 			aInstanceKey A key specifying a previously created implementation.
sl@0
   466
@param 			aExtendedInterfaceUid Identifies an interface to fetch from the plug-in instance.
sl@0
   467
@return			TAny* A pointer to the extended interface, will be NULL if it does not exist.
sl@0
   468
*/
sl@0
   469
TAny* CLoadManager::GetExtendedInterfaceL(const TUid& aInstanceKey, const TUid& aExtendedInterfaceUid)
sl@0
   470
	{
sl@0
   471
	// Sanity check that the pointer is divisible by four. A binary number is divisible
sl@0
   472
	// by four if and only if the two rightmost bits are both zero.
sl@0
   473
	// This is a compromised check for checking that the pointer is an address.
sl@0
   474
    if ((aInstanceKey.iUid == 0) || ((aInstanceKey.iUid & 0x00000003) != 0))
sl@0
   475
        {
sl@0
   476
	    __ASSERT_DEBUG(EFalse, User::Panic(KEComClientDLLPanicCategory, EEComPanic_InvalidImplementationInstanceKey));
sl@0
   477
	    User::Leave(KErrNotFound);
sl@0
   478
        }
sl@0
   479
sl@0
   480
	// The instance info pointer is stored in the instance key.
sl@0
   481
	CInstanceInfo* instanceInfo = reinterpret_cast<CInstanceInfo*>(aInstanceKey.iUid);
sl@0
   482
sl@0
   483
	// Check that the pointer exists before using it - leaves with KErrNotFound
sl@0
   484
	iInstanceInfoList.FindInAddressOrderL(instanceInfo);
sl@0
   485
	
sl@0
   486
	// Create the extension object. The instance info object will be populated with
sl@0
   487
	// the extension info during creation.
sl@0
   488
	TAny* object = instanceInfo->CreateExtObjectL(aExtendedInterfaceUid);
sl@0
   489
	return object;
sl@0
   490
	}
sl@0
   491
sl@0
   492
/**
sl@0
   493
Manually releases the requested interface. Does nothing if it does not exist.
sl@0
   494
This interface is optional, normally the interfaces are cleaned up automatically.
sl@0
   495
@leave			KErrNotFound
sl@0
   496
@param			aInstanceKey A key specifying a previously created implementation.
sl@0
   497
@param			aExtendedInterfaceUid Identifies the interface to release
sl@0
   498
@return None.
sl@0
   499
*/
sl@0
   500
void CLoadManager::ManuallyReleaseExtendedInterfaceL(const TUid& aInstanceKey, const TUid& aExtendedInterfaceUid)
sl@0
   501
	{
sl@0
   502
	// Sanity check that the pointer is divisible by four. A binary number is divisible
sl@0
   503
	// by four if and only if the two rightmost bits are both zero.
sl@0
   504
	// This is a compromised check for checking that the pointer is an address.
sl@0
   505
    if ((aInstanceKey.iUid == 0) || ((aInstanceKey.iUid & 0x00000003) != 0))
sl@0
   506
        {
sl@0
   507
	    __ASSERT_DEBUG(EFalse, User::Panic(KEComClientDLLPanicCategory, EEComPanic_InvalidImplementationInstanceKey));
sl@0
   508
	    User::Leave(KErrNotFound);
sl@0
   509
        }
sl@0
   510
        
sl@0
   511
	// The instance info pointer is stored in the instance key.
sl@0
   512
	CInstanceInfo* instanceInfo = reinterpret_cast<CInstanceInfo*>(aInstanceKey.iUid);
sl@0
   513
sl@0
   514
	// Check that the pointer exists before using it - leaves with KErrNotFound
sl@0
   515
	iInstanceInfoList.FindInAddressOrderL(instanceInfo);
sl@0
   516
	
sl@0
   517
	instanceInfo->DestroyExtObject(aExtendedInterfaceUid);
sl@0
   518
	}
sl@0
   519
sl@0
   520
/**
sl@0
   521
Utility method to move the policy to the garbage list, remove
sl@0
   522
policy from the unload policy list.
sl@0
   523
@param aPolicy Unload policy to clean up
sl@0
   524
@return None.
sl@0
   525
*/
sl@0
   526
void CLoadManager::Cleanup(CUnloadPolicy* aPolicy)
sl@0
   527
	{
sl@0
   528
	if (aPolicy->DecreaseReference() == EDeleteMe)
sl@0
   529
		{
sl@0
   530
		// Move the policy to the garbage list
sl@0
   531
		iGarbagePolicy=aPolicy;
sl@0
   532
sl@0
   533
		TInt index = iAllUnloadPolicies.Find(aPolicy);
sl@0
   534
		if (index != KErrNotFound)
sl@0
   535
			{
sl@0
   536
			iAllUnloadPolicies.Remove(index);
sl@0
   537
			}
sl@0
   538
		}
sl@0
   539
	}
sl@0
   540
sl@0
   541
//
sl@0
   542
// CInstanceInfo
sl@0
   543
/**
sl@0
   544
Default constructor of CInstanceInfo
sl@0
   545
@param			aUnloadPolicy The CUnloadPolicy of the dll
sl@0
   546
*/
sl@0
   547
CInstanceInfo::CInstanceInfo(CUnloadPolicy* aUnloadPolicy):
sl@0
   548
	iUnloadPolicy(aUnloadPolicy)
sl@0
   549
	{
sl@0
   550
	// Do nothing here
sl@0
   551
	}
sl@0
   552
sl@0
   553
/** 
sl@0
   554
Destructor of CInstanceInfo
sl@0
   555
*/
sl@0
   556
CInstanceInfo::~CInstanceInfo()
sl@0
   557
	{
sl@0
   558
	// Do nothing here
sl@0
   559
	}
sl@0
   560
sl@0
   561
sl@0
   562
//
sl@0
   563
// CInstanceInfoExtended
sl@0
   564
/**
sl@0
   565
Default constructor of CInstanceInfoExtended
sl@0
   566
@param			aUnloadPolicy The CUnloadPolicy of the dll
sl@0
   567
@param			aImplementationProxyRow The interface implementation proxy row entry
sl@0
   568
*/
sl@0
   569
CInstanceInfoExtended::CInstanceInfoExtended(CUnloadPolicy* aUnloadPolicy,const TImplementationProxy3* aImplementationProxyRow):
sl@0
   570
	CInstanceInfo(aUnloadPolicy),iImplementationProxyRow(aImplementationProxyRow)
sl@0
   571
	{
sl@0
   572
	// Do nothing here
sl@0
   573
	}
sl@0
   574
sl@0
   575
/**
sl@0
   576
create an instance of CInstanceInfoExtended
sl@0
   577
@param			aUnloadPolicy The CUnloadPolicy of the dll
sl@0
   578
@param			aImplementationProxyRow The interface implementation proxy row entry
sl@0
   579
@return			A pointer to the newly created object.
sl@0
   580
*/
sl@0
   581
CInstanceInfoExtended* CInstanceInfoExtended::NewL(CUnloadPolicy* aUnloadPolicy,const TImplementationProxy3* aImplementationProxyRow)
sl@0
   582
	{
sl@0
   583
	return new(ELeave) CInstanceInfoExtended(aUnloadPolicy,aImplementationProxyRow);
sl@0
   584
	}
sl@0
   585
sl@0
   586
/**
sl@0
   587
Sets the implementation object.
sl@0
   588
@param			aImplementationObject The object instance of this instances' implementation
sl@0
   589
@return			None
sl@0
   590
*/
sl@0
   591
void CInstanceInfoExtended::SetObject(TAny* aImplementationObject)
sl@0
   592
	{
sl@0
   593
	iImplementationObject = aImplementationObject;
sl@0
   594
	}
sl@0
   595
sl@0
   596
/**
sl@0
   597
Creates the extension interface object. This will use the get extended interface
sl@0
   598
function pointer from the proxy table to fetch the extended interface from the
sl@0
   599
plug-in implementation.
sl@0
   600
@param			aExtendedInterfaceUID The extended interface UID
sl@0
   601
@return			TAny* A pointer to an instance of an extended interface created
sl@0
   602
*/
sl@0
   603
TAny* CInstanceInfoExtended::CreateExtObjectL(const TUid& aExtendedInterfaceUID)
sl@0
   604
	{
sl@0
   605
	// Fetch the function pointer to create the extended interface
sl@0
   606
	TProxyExtendedInterfaceGetPtrL createFunctionPtrL = iImplementationProxyRow->iFuncPtrInterfaceGetL;
sl@0
   607
	if (createFunctionPtrL == NULL)
sl@0
   608
		{
sl@0
   609
		// No extension interface object can be created. Return NULL indicating that
sl@0
   610
		// no extended interface object is available.
sl@0
   611
		return NULL;
sl@0
   612
		}
sl@0
   613
sl@0
   614
	// Valid function pointer exists in proxy table.
sl@0
   615
	TAny* object = NULL;         // Extended interface object (this points to the interface
sl@0
   616
							     // within the object)
sl@0
   617
	TAny* releaseObject = NULL;  // Eextended interface object (this points to the extended
sl@0
   618
								 // object itself). Used to delete the extended interface
sl@0
   619
								 // object later.
sl@0
   620
	TUint32 flags = 0;           // Flags to allow the plug-in and ECOM to communicate
sl@0
   621
	// Create the extension object.
sl@0
   622
	object = createFunctionPtrL(iImplementationObject,aExtendedInterfaceUID,flags,releaseObject);
sl@0
   623
sl@0
   624
	if (flags & KReleaseRequiredMask)
sl@0
   625
		{
sl@0
   626
		// If release of the extended interface is required then save the release object pointer.
sl@0
   627
		// The interface object (returned by the function pointer call) and the release object
sl@0
   628
		// are not necessarily the same pointer. This is because the  interface pointer is not
sl@0
   629
		// guaranteed to be the same as the pointer to the extended interface object. That
sl@0
   630
		// is why the release object is required to be fetched from the plug-in.
sl@0
   631
sl@0
   632
        // First perform some checks to ensure that the plugin is consistent 
sl@0
   633
        TProxyExtendedInterfaceReleasePtr release = iImplementationProxyRow->iFuncPtrInterfaceRelease;
sl@0
   634
                
sl@0
   635
        if (release == NULL)
sl@0
   636
            {
sl@0
   637
            // ...the release object pointer must not be non null
sl@0
   638
            __ECOM_TRACE("ECOM: PANIC in CInstanceInfoExtended::CreateExtObjectL, release required but release function missing");
sl@0
   639
	        __ASSERT_DEBUG(EFalse, User::Panic (KEComClientDLLPanicCategory, EEComPanic_CInstanceInfoExtended_CreateExtObjectL_NoReleaseFunc));
sl@0
   640
            User::Leave(KEComErrNoExtendedInterfaceReleaseFunction);
sl@0
   641
            }
sl@0
   642
            
sl@0
   643
        if (releaseObject == NULL)
sl@0
   644
            {
sl@0
   645
            // ... the releaseObject must be non null
sl@0
   646
            __ECOM_TRACE("ECOM: PANIC in CInstanceInfoExtended::CreateExtObjectL, release required but release object missing");
sl@0
   647
	        __ASSERT_DEBUG(EFalse, User::Panic (KEComClientDLLPanicCategory, EEComPanic_CInstanceInfoExtended_CreateExtObjectL_NoReleaseObj));
sl@0
   648
           User::Leave(KEComErrNoExtendedInterfaceReleaseObject);
sl@0
   649
 		    }
sl@0
   650
            
sl@0
   651
		//Create the extended object info type and add it to the extended object info array.
sl@0
   652
		TExtendedObjectInfo extendedObjectInfo;
sl@0
   653
		extendedObjectInfo.iExtendedInterfaceObject = releaseObject;
sl@0
   654
		extendedObjectInfo.iExtendedInterfaceUID = aExtendedInterfaceUID;
sl@0
   655
sl@0
   656
		TInt err = iExtendedObjectInfo.Append(extendedObjectInfo);
sl@0
   657
sl@0
   658
		if (err != KErrNone)
sl@0
   659
			{
sl@0
   660
			if (release != NULL)
sl@0
   661
				{
sl@0
   662
				// Release the extended interface. Release must not leave.
sl@0
   663
				release(extendedObjectInfo.iExtendedInterfaceObject,extendedObjectInfo.iExtendedInterfaceUID);
sl@0
   664
				}
sl@0
   665
			User::Leave(err);
sl@0
   666
			}
sl@0
   667
		}
sl@0
   668
sl@0
   669
	return object;
sl@0
   670
	}
sl@0
   671
sl@0
   672
sl@0
   673
/**
sl@0
   674
Destroys the extension interface object.
sl@0
   675
@param			aExtendedInterfaceUID The extended interface UID
sl@0
   676
@return			None
sl@0
   677
*/
sl@0
   678
void CInstanceInfoExtended::DestroyExtObject(const TUid& aExtendedInterfaceUID)
sl@0
   679
	{
sl@0
   680
	// Get release interface
sl@0
   681
	TProxyExtendedInterfaceReleasePtr release = iImplementationProxyRow->iFuncPtrInterfaceRelease;
sl@0
   682
sl@0
   683
	if (release != NULL)
sl@0
   684
		{
sl@0
   685
		// Release extended interface. Find the extended object info.
sl@0
   686
		for(TInt i = 0; i < iExtendedObjectInfo.Count(); i++)
sl@0
   687
			{
sl@0
   688
			if (iExtendedObjectInfo[i].iExtendedInterfaceUID == aExtendedInterfaceUID)
sl@0
   689
				{
sl@0
   690
				TAny* releaseObject = iExtendedObjectInfo[i].iExtendedInterfaceObject;
sl@0
   691
                if (releaseObject == NULL)
sl@0
   692
                    {
sl@0
   693
                    // ... the releaseObject must be non null
sl@0
   694
                    __ECOM_TRACE("ECOM: PANIC in CInstanceInfoExtended::DestroyExtObject, release required but release object missing");
sl@0
   695
	                __ASSERT_DEBUG(EFalse, User::Panic (KEComClientDLLPanicCategory, EEComPanic_CInstanceInfoExtended_DestroyExtObject_NoReleaseObj));	              
sl@0
   696
		            }
sl@0
   697
sl@0
   698
				// Release the extended interface. Release should not be leaving.
sl@0
   699
				release(releaseObject, iExtendedObjectInfo[i].iExtendedInterfaceUID);
sl@0
   700
				// Remove the extended object info element from the array.
sl@0
   701
				iExtendedObjectInfo.Remove(i);
sl@0
   702
				break;
sl@0
   703
				}
sl@0
   704
			}
sl@0
   705
		}
sl@0
   706
	}
sl@0
   707
sl@0
   708
/**
sl@0
   709
Destructor of CInstanceInfoExtended
sl@0
   710
*/
sl@0
   711
CInstanceInfoExtended::~CInstanceInfoExtended()
sl@0
   712
	{
sl@0
   713
	// Get release interface
sl@0
   714
	if (iImplementationProxyRow != NULL)
sl@0
   715
		{
sl@0
   716
		TProxyExtendedInterfaceReleasePtr release = iImplementationProxyRow->iFuncPtrInterfaceRelease;
sl@0
   717
sl@0
   718
		if (release != NULL)
sl@0
   719
			{
sl@0
   720
			// Release all extended interfaces (if any still to be released)
sl@0
   721
			for(TInt i = 0; i < iExtendedObjectInfo.Count(); i++)
sl@0
   722
				{
sl@0
   723
				// Release is supposed to be non-leavable.
sl@0
   724
				release(iExtendedObjectInfo[i].iExtendedInterfaceObject,iExtendedObjectInfo[i].iExtendedInterfaceUID);
sl@0
   725
				}
sl@0
   726
			}
sl@0
   727
		}
sl@0
   728
sl@0
   729
	iExtendedObjectInfo.Close();
sl@0
   730
	}
sl@0
   731
sl@0
   732
//
sl@0
   733
// CInstanceInfoSimple
sl@0
   734
/**
sl@0
   735
Default constructor of CInstanceInfoSimple
sl@0
   736
@param			aUnloadPolicy The CUnloadPolicy of the dll
sl@0
   737
@param			aImplementationProxyRow The interface implementation proxy row entry
sl@0
   738
*/
sl@0
   739
CInstanceInfoSimple::CInstanceInfoSimple(CUnloadPolicy* aUnloadPolicy,const TImplementationProxy* aImplementationProxyRow):
sl@0
   740
	CInstanceInfo(aUnloadPolicy),iImplementationProxyRow(aImplementationProxyRow)
sl@0
   741
	{
sl@0
   742
	// Do nothing here
sl@0
   743
	}
sl@0
   744
sl@0
   745
/**
sl@0
   746
Create an instance of CInstanceInfoSimple
sl@0
   747
@param			aUnloadPolicy The CUnloadPolicy of the dll
sl@0
   748
@param			aImplementationProxyRow The interface implementation proxy row entry
sl@0
   749
@return			A pointer to the newly created object.
sl@0
   750
*/
sl@0
   751
CInstanceInfoSimple* CInstanceInfoSimple::NewL(CUnloadPolicy* aUnloadPolicy,const TImplementationProxy* aImplementationProxyRow)
sl@0
   752
	{
sl@0
   753
	return new(ELeave) CInstanceInfoSimple(aUnloadPolicy,aImplementationProxyRow);
sl@0
   754
	}