os/ossrv/lowlevellibsandfws/pluginfw/Framework/frame/LoadManager.cpp
changeset 0 bde4ae8d615e
     1.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     1.2 +++ b/os/ossrv/lowlevellibsandfws/pluginfw/Framework/frame/LoadManager.cpp	Fri Jun 15 03:10:57 2012 +0200
     1.3 @@ -0,0 +1,754 @@
     1.4 +// Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
     1.5 +// All rights reserved.
     1.6 +// This component and the accompanying materials are made available
     1.7 +// under the terms of "Eclipse Public License v1.0"
     1.8 +// which accompanies this distribution, and is available
     1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
    1.10 +//
    1.11 +// Initial Contributors:
    1.12 +// Nokia Corporation - initial contribution.
    1.13 +//
    1.14 +// Contributors:
    1.15 +//
    1.16 +// Description:
    1.17 +// Implementation of the CLoadManager class
    1.18 +// 
    1.19 +//
    1.20 +
    1.21 +/**
    1.22 + @internalComponent
    1.23 + @file
    1.24 +*/
    1.25 +
    1.26 +#include "EComDebug.h"
    1.27 +#include "UnloadPolicy.h"
    1.28 +#include "LoadManager.h"
    1.29 +#include "EComUidCodes.h"
    1.30 +#include <ecom/ecomerrorcodes.h>
    1.31 +#include "e32math.h"
    1.32 +#include "EComInternalErrorCodes.h"
    1.33 +#include <ecom/ecomextendedinterfaceerrorcodes.h>
    1.34 +/**
    1.35 +Standardized safe construction which leaves nothing on the cleanup stack.
    1.36 +@return			A pointer to the new class
    1.37 +@post			CLoadManager is fully constructed, and initialized.
    1.38 + */
    1.39 +CLoadManager* CLoadManager::NewL()
    1.40 +	{
    1.41 +	return new(ELeave) CLoadManager();
    1.42 +	}
    1.43 +
    1.44 +CLoadManager::~CLoadManager()
    1.45 +	{
    1.46 +	iInstanceInfoList.ResetAndDestroy();
    1.47 +	iAllUnloadPolicies.ResetAndDestroy();
    1.48 +	ClearGarbage();
    1.49 +	}
    1.50 +
    1.51 +/**
    1.52 +Notifies the interface implementation DLL that one of its objects has been destroyed if it exists,
    1.53 +otherwise returns false to indicate no exist aImplementationUid.
    1.54 +@param          aInstanceKey A key specifying a previously created implementation.
    1.55 +@pre 			CLoadManager is fully constructed,
    1.56 +@post			CLoadManager's interface implementation DLL references
    1.57 +				are decreased by one. The instance info representing the implementation
    1.58 +				is destroyed.
    1.59 +*/
    1.60 +TBool CLoadManager::DestroyedThis(TUid aInstanceKey)
    1.61 +	{
    1.62 +	// Clear the garbage list because we know we have finished with them
    1.63 +	ClearGarbage();
    1.64 +
    1.65 +	__ECOM_TRACE1("ECOM: Implementation Instance destroyed %x", aInstanceKey.iUid);
    1.66 +
    1.67 +	// Sanity check that the pointer is divisible by four. A binary number is divisible
    1.68 +	// by four if and only if the two rightmost bits are both zero.
    1.69 +	// This is a compromised check for checking that the pointer is an address.
    1.70 +    if ((aInstanceKey.iUid == 0) || ((aInstanceKey.iUid & 0x00000003) != 0))
    1.71 +        {
    1.72 +	    __ASSERT_DEBUG(EFalse, User::Panic(KEComClientDLLPanicCategory, EEComPanic_InvalidImplementationInstanceKey));
    1.73 +	    User::Leave(KErrNotFound);
    1.74 +        }
    1.75 +
    1.76 +	// The instance info pointer is stored in the instance key.
    1.77 +	CInstanceInfo* instanceInfo = reinterpret_cast<CInstanceInfo*>(aInstanceKey.iUid);
    1.78 +
    1.79 +	// Check that the pointer exists before using it
    1.80 +	TInt position = iInstanceInfoList.FindInAddressOrder(instanceInfo);
    1.81 +	if(position == KErrNotFound)
    1.82 +		{
    1.83 +		return EFalse;
    1.84 +		}
    1.85 +	
    1.86 +	CUnloadPolicy* policy = instanceInfo->UnloadPolicy();
    1.87 +	if (policy != NULL)
    1.88 +		{
    1.89 +		// If needed, will move the policy to the garbage list, and remove policy from the unload policy list.
    1.90 +		Cleanup(policy);
    1.91 +		}
    1.92 +	iInstanceInfoList.Remove(position);
    1.93 +	// Remove the instance info item, finished with it.
    1.94 +	delete instanceInfo;
    1.95 +	return ETrue;
    1.96 +	}
    1.97 +
    1.98 +/**
    1.99 +Check whether the policy Arrays are empty or not.
   1.100 +@return            Returns True if the policy Arrays are empty, otherwise return False.
   1.101 +@pre             CLoadManager is fully constructed,
   1.102 +@post            CLoadManager remains the same.
   1.103 +*/
   1.104 +TBool CLoadManager::PolicyArraysEmpty() const 
   1.105 +    {
   1.106 +    if( (iAllUnloadPolicies.Count() == 0)&&(iInstanceInfoList.Count() == 0) )
   1.107 +        {
   1.108 +        return ETrue;
   1.109 +        }
   1.110 +    else
   1.111 +        {
   1.112 +        return EFalse;
   1.113 +        }
   1.114 +    }
   1.115 +
   1.116 +/**
   1.117 +Returns an implementation object to satisfy the specified interface.
   1.118 +@param			aUniqueImplementationUid The implementation to find.
   1.119 +@param			aEntry Information on the dll containing the implementation
   1.120 +@param			aCreationParameters A pointer to the creation parameter
   1.121 +				structure passed to the creation method when called.
   1.122 +@param			aCreationParamsFlag A boolean flag to indicate the existence or non-existence
   1.123 +				of aCreationParameters. Will be ETrue even for if the value of
   1.124 +				aCreationParameters is NULL.
   1.125 +@param          aInstanceKey A key specifying a previously created implementation.
   1.126 +@return			TAny* a pointer to the fully constructed instantiation. The pointer is
   1.127 +				guaranteed to be valid only until DestroyedThis is called.
   1.128 +@pre 			CLoadManager is fully constructed,
   1.129 +@post			Fully constructed implementation is returned to the
   1.130 +				caller, and aUniqueUid contains the implementation Dll's
   1.131 +				unique UID.
   1.132 +*/
   1.133 +TAny* CLoadManager::ImplementationObjectL(const TUid& aUniqueImplementationUid,
   1.134 +										 const TEntry& aEntry,
   1.135 +										 TAny* aCreationParameters,
   1.136 +										 TBool aCreationParamsFlag,
   1.137 +										 TUid& aInstanceKey)
   1.138 +	{
   1.139 +	//if the implementation Uid here is Null or the entry returned by the ecomserver
   1.140 +	//contains nothing, we should leave with KErrNotFound
   1.141 +	if(aUniqueImplementationUid == KNullUid || (aEntry.iName).Length()==0)
   1.142 +		{
   1.143 +		User::Leave(KErrNotFound);
   1.144 +		}
   1.145 +
   1.146 +	TAny* object = NULL;               // Instantiation object
   1.147 +	CUnloadPolicy* policy = NULL;      // Policy of implementation
   1.148 +	TLibraryFunction libFunctionProxy; // Library function proxy pointer
   1.149 +
   1.150 +	GetUnloadPolicy(aUniqueImplementationUid,aEntry,policy);
   1.151 +	if (policy == NULL)
   1.152 +		{
   1.153 +		// No policy found, so create a new CUnloadPolicy and load DLL.
   1.154 +		policy = CUnloadPolicy::NewLC(aEntry);
   1.155 +		libFunctionProxy = policy->LoadDllAndReturnProxyL();
   1.156 +		if (libFunctionProxy==NULL)
   1.157 +			{
   1.158 +			User::Leave(KErrNotFound);
   1.159 +			}
   1.160 +		iAllUnloadPolicies.AppendL(policy);
   1.161 +		CleanupStack::Pop(policy);	// owned by iAllUnloadPolicies
   1.162 +		}
   1.163 +	else
   1.164 +		{
   1.165 +		// Found a policy. Load Dll if not already loaded.
   1.166 +		libFunctionProxy = policy->LoadDllAndReturnProxyL();
   1.167 +		if (libFunctionProxy==NULL)
   1.168 +			{
   1.169 +			User::Leave(KErrNotFound);
   1.170 +			}
   1.171 +		}
   1.172 +	// Initial count of instances. This is used for cleanup purposes, to see if an instance
   1.173 +	// was added to the instance info list during the object creation. If a failure occurs
   1.174 +	// than the instance info that was added to the list will be removed.
   1.175 +	TInt initialCount = iInstanceInfoList.Count();
   1.176 +	TInt err = KErrNone;
   1.177 +
   1.178 +	if (aEntry[1] == KUidInterfaceImplementationCollection)
   1.179 +		{
   1.180 +		// PLUGIN1 dll. Create the implementation object.
   1.181 +		TRAP(err,object = ImplementationObject1L(aUniqueImplementationUid,aCreationParameters,
   1.182 +			aCreationParamsFlag,aInstanceKey,policy,libFunctionProxy));
   1.183 +		}
   1.184 +	else if (aEntry[1] == KUidInterfaceImplementationCollection3)
   1.185 +		{
   1.186 +		// PLUGIN3 dll. Create the implementation object.
   1.187 +		TRAP(err,object = ImplementationObject3L(aUniqueImplementationUid,aCreationParameters,
   1.188 +			aCreationParamsFlag, aInstanceKey,policy, libFunctionProxy));
   1.189 +		}
   1.190 +	else
   1.191 +		{
   1.192 +		err = KErrNotSupported;
   1.193 +		}
   1.194 +	
   1.195 +	if (err != KErrNone)
   1.196 +		{
   1.197 +		if (iInstanceInfoList.Count() > initialCount)
   1.198 +			{
   1.199 +			// If an instance was added to the instance list, remove it here.
   1.200 +			// The instance info pointer is stored in the instance key. We know its
   1.201 +			// valid because it was set by ecom in call to ImplementationObject1L or
   1.202 +			// ImplementationObject3L
   1.203 +			CInstanceInfo* instanceInfo = reinterpret_cast<CInstanceInfo*>(aInstanceKey.iUid);
   1.204 +			TInt pos = iInstanceInfoList.FindInAddressOrder(instanceInfo);
   1.205 +			if(pos != KErrNotFound)
   1.206 +				{
   1.207 +				iInstanceInfoList.Remove(pos);
   1.208 +				}
   1.209 +			}
   1.210 +
   1.211 +		ClearGarbage();
   1.212 +		// If needed, will move the policy to the garbage list, and remove policy from the unload policy list.
   1.213 +		Cleanup(policy);
   1.214 +		User::Leave(err);
   1.215 +		}
   1.216 +		
   1.217 +	__ECOM_TRACE2("ECOM: Implementation created (%03d) %x", ++iDebugInstantiationCounter, aUniqueImplementationUid.iUid);
   1.218 +			
   1.219 +	return object;
   1.220 +	}
   1.221 +
   1.222 +/**
   1.223 +Returns an implementation object to satisfy the specified interface.
   1.224 +@param			aUniqueImplementationUid The implementation to find.
   1.225 +@param			aCreationParameters A pointer to the creation parameter
   1.226 +				structure passed to the creation method when called.
   1.227 +@param			aCreationParamsFlag A boolean flag to indicate the existence or non-existence
   1.228 +				of aCreationParameters. Will be ETrue even for if the value of
   1.229 +				aCreationParameters is NULL.
   1.230 +@param          aInstanceKey A key specifying a previously created implementation.
   1.231 +@param          aPolicy policy of implementation
   1.232 +@param          aLibFunctionProxy Library function proxy pointer.
   1.233 +@return			TAny* a pointer to the fully constructed instantiation. The pointer is
   1.234 +				guaranteed to be valid only until DestroyedThis is called.
   1.235 +*/
   1.236 +TAny* CLoadManager::ImplementationObject1L(const TUid& aUniqueImplementationUid,
   1.237 +										 TAny* aCreationParameters,
   1.238 +										 TBool aCreationParamsFlag,
   1.239 +										 TUid& aInstanceKey,
   1.240 +										 CUnloadPolicy* aPolicy,
   1.241 +										 TLibraryFunction& aLibFunctionProxy)
   1.242 +	{
   1.243 +	TAny* object = NULL; // Instantiation object
   1.244 +	TInstantiationL proxy = reinterpret_cast<TInstantiationL>(aLibFunctionProxy);
   1.245 +
   1.246 +	TImplementationProxy* implementationProxyRow = NULL;
   1.247 +	TAny* newLPointer = GetNewLPointerAndProxyTableRowL<TImplementationProxy,TInstantiationL>(aUniqueImplementationUid,implementationProxyRow,proxy);
   1.248 +	// Now create an instance info object to store the information about this instance derived from
   1.249 +	// the parameters just fetched. This must be created here since no leaves can occur after the object
   1.250 +	// instantiation below.
   1.251 +	CInstanceInfoSimple* instanceInfoSimple = CInstanceInfoSimple::NewL(aPolicy,implementationProxyRow);
   1.252 +	CleanupStack::PushL(instanceInfoSimple);
   1.253 +
   1.254 +	// The pointer to instanceInfo will be used to identify this instance, and will
   1.255 +	// be returned to the caller for future identification of this instance.
   1.256 +	aInstanceKey.iUid = reinterpret_cast<TInt32>(instanceInfoSimple);
   1.257 +
   1.258 +	// Add item to instance info list. This list will contain all of the instance objects.
   1.259 +	iInstanceInfoList.InsertInAddressOrderL(instanceInfoSimple);
   1.260 +
   1.261 +	// Get the implementation object using the instantiation pointer fetched earlier. Note
   1.262 +	// that the object creation must be the last leaving operation to be performed. No leaves can occur
   1.263 +	// after this operation, as the instantiation object cannnot be deleted on the cleanup stack if
   1.264 +	// a leave occurs (cannot delete a TAny* pointer, the type is not known within ECOM).
   1.265 +	object = ImplementationObjectL(aCreationParameters, 
   1.266 +	 aCreationParamsFlag,newLPointer);
   1.267 +
   1.268 +	CleanupStack::Pop(instanceInfoSimple);
   1.269 +
   1.270 +	return object;
   1.271 +	}
   1.272 +
   1.273 +/**
   1.274 +Returns an implementation object to satisfy the specified interface.
   1.275 +@param			aUniqueImplementationUid The implementation to find.
   1.276 +@param			aCreationParameters A pointer to the creation parameter
   1.277 +				structure passed to the creation method when called.
   1.278 +@param			aCreationParamsFlag A boolean flag to indicate the existence or non-existence
   1.279 +				of aCreationParameters. Will be ETrue even for if the value of
   1.280 +				aCreationParameters is NULL.
   1.281 +@param          aInstanceKey A key specifying a previously created implementation.
   1.282 +@param          aPolicy policy of implementation
   1.283 +@param          aLibFunctionProxy Library function proxy pointer.
   1.284 +@return			TAny* a pointer to the fully constructed instantiation. The pointer is
   1.285 +				guaranteed to be valid only until DestroyedThis is called.
   1.286 +*/
   1.287 +TAny* CLoadManager::ImplementationObject3L(const TUid& aUniqueImplementationUid,
   1.288 +										 TAny* aCreationParameters,
   1.289 +										 TBool aCreationParamsFlag,
   1.290 +										 TUid& aInstanceKey,
   1.291 +										 CUnloadPolicy* aPolicy,
   1.292 +										 TLibraryFunction& aLibFunctionProxy)
   1.293 +	{
   1.294 +	TAny* object = NULL;  // Instantiation object
   1.295 +	TInstantiation3L proxy = reinterpret_cast<TInstantiation3L>(aLibFunctionProxy);
   1.296 +
   1.297 +	TImplementationProxy3* implementationProxyRow = NULL;
   1.298 +	TAny* newLPointer = GetNewLPointerAndProxyTableRowL<TImplementationProxy3,TInstantiation3L>(aUniqueImplementationUid,implementationProxyRow,proxy);
   1.299 +
   1.300 +	// Now create an instance info object to store the information about this instance derived from
   1.301 +	// the parameters just fetched. This must be created here since no leaves can occur after the object
   1.302 +	// instantiation below.
   1.303 +	CInstanceInfoExtended* instanceInfoExtended = CInstanceInfoExtended::NewL(aPolicy,implementationProxyRow);
   1.304 +	CleanupStack::PushL(instanceInfoExtended);
   1.305 +
   1.306 +	// The pointer to instanceInfo will be used to identify this instance, and will
   1.307 +	// be returned to the caller for future identification of this instance.
   1.308 +	aInstanceKey.iUid = reinterpret_cast<TInt32>(instanceInfoExtended);
   1.309 +
   1.310 +	// Add item to instance info list. This list will contain all of the instance objects.
   1.311 +	iInstanceInfoList.InsertInAddressOrderL(instanceInfoExtended);
   1.312 +
   1.313 +	// Get the implementation object using the instantiation pointer fetched earlier. Note
   1.314 +	// that the object creation must be the last leaving operation to be performed. No leaves can occur
   1.315 +	// after this operation, as the instantiation object cannnot be deleted on the cleanup stack if
   1.316 +	// a leave occurs (cannot delete a TAny* pointer, the type is not known within ECOM).
   1.317 +	object = ImplementationObjectL(aCreationParameters, 
   1.318 +	 aCreationParamsFlag,newLPointer);
   1.319 +
   1.320 +	// The extended instance info requires an object set. This is done here
   1.321 +	// as the object is not known until now, but the instance info object was required to
   1.322 +	// be created earlier to avoid a leave after the object creation.
   1.323 +	instanceInfoExtended->SetObject(object);
   1.324 +	CleanupStack::Pop(instanceInfoExtended);
   1.325 +
   1.326 +	return object;
   1.327 +	}
   1.328 +
   1.329 +/**
   1.330 +Gets the main implementation object using the instantiation pointer provided
   1.331 +@param			aCreationParameters A pointer to the creation parameter
   1.332 +				structure passed to the creation method when called.
   1.333 +@param			aCreationParamsFlag A boolean flag to indicate the existence or non-existence
   1.334 +				of aCreationParameters. Will be ETrue even for if the value of
   1.335 +				aCreationParameters is NULL.
   1.336 +@param			aNewLpointer Instantation pointer.
   1.337 +				aCreationParameters is NULL.
   1.338 +@return			TAny* a pointer to the fully constructed instantiation.
   1.339 +*/
   1.340 +TAny* CLoadManager::ImplementationObjectL(TAny* aCreationParameters,
   1.341 +										 TBool aCreationParamsFlag,
   1.342 +										 const TAny* aNewLpointer)
   1.343 +	{
   1.344 +	TAny* object=NULL;
   1.345 +
   1.346 +	// So cast to the correct type : This gives an ANSI C++ warning
   1.347 +	// When using a REINTERPRET_CAST so simply cast instead
   1.348 +	// Two different object creation one with creation parameters
   1.349 +	if (aCreationParamsFlag)
   1.350 +		{
   1.351 +		typedef TAny* (*TNewL)(TAny*);
   1.352 +		TNewL creationL = (TNewL)(aNewLpointer);
   1.353 +		object=creationL(aCreationParameters);
   1.354 +		}
   1.355 +	else
   1.356 +		{
   1.357 +		typedef TAny* (*TNewL)();
   1.358 +		TNewL creationL = (TNewL)(aNewLpointer);
   1.359 +		object=creationL();
   1.360 +		}
   1.361 +
   1.362 +	return object;
   1.363 +	}
   1.364 +
   1.365 +/**
   1.366 +Get the unload policy.
   1.367 +@param			aUniqueImplementationUid The implementation to find.
   1.368 +@param			aEntry Information on the dll containing the implementation
   1.369 +@param			aPolicy Return parameter containing the policy
   1.370 +@return			None
   1.371 +*/
   1.372 +void CLoadManager::GetUnloadPolicy(TUid aUniqueImplementationUid,
   1.373 +							  const TEntry& aEntry,
   1.374 +							  CUnloadPolicy*& aPolicy)
   1.375 +	{
   1.376 +	const TInt numImps = iInstanceInfoList.Count();
   1.377 +	TBool foundImp = EFalse;
   1.378 +	TInt matchingDllIndex=0;
   1.379 +	TBool matchingDllFound=EFalse;
   1.380 +	CUnloadPolicy* policy = NULL;
   1.381 +
   1.382 +	for (TInt index = 0; (index<numImps) && !foundImp; ++index)
   1.383 +		{
   1.384 +		//Check if there is existing mapping between the supplied implUid and a policy
   1.385 +		if(iInstanceInfoList[index]->ImplementationUid() == aUniqueImplementationUid)
   1.386 +			{
   1.387 +			policy = iInstanceInfoList[index]->UnloadPolicy();
   1.388 +
   1.389 +			foundImp = ETrue;
   1.390 +			}
   1.391 +		else
   1.392 +			{
   1.393 +			//if cannot find a mapping entry but current index has the same DLL as the one requested from
   1.394 +			//the client, store the index to this unloadPolicy for use later on if we have finished
   1.395 +			//searching the entire mapping list
   1.396 +			if (!matchingDllFound && (iInstanceInfoList[index]->UnloadPolicy()->DllEntryInformation().GetName()).CompareF(aEntry.iName)==0)
   1.397 +				{
   1.398 +				matchingDllIndex=index;
   1.399 +				matchingDllFound=ETrue;
   1.400 +				}
   1.401 +			}
   1.402 +		}
   1.403 +
   1.404 +	//If we cannot find any mapping in the policy index array(iUnloadPolicyMapping)
   1.405 +	if(!foundImp)
   1.406 +		{
   1.407 +		//if not found but there is a matching DLL,we can simply create a new policy index mapping without
   1.408 +		//having to load the library again.
   1.409 +		if (matchingDllFound)
   1.410 +			{
   1.411 +			policy = iInstanceInfoList[matchingDllIndex]->UnloadPolicy();
   1.412 +			}
   1.413 +		}
   1.414 +
   1.415 +	aPolicy = policy;
   1.416 +	}
   1.417 +
   1.418 +/**
   1.419 +Clears the policy inside the iGarbagePolicies attribute.
   1.420 +@pre			CLoadManager is fully constructed
   1.421 +@post			CLoadManager iGarbagePolicies is zero'd
   1.422 +*/
   1.423 +void CLoadManager::ClearGarbage()
   1.424 +	{
   1.425 +	if (iGarbagePolicy != NULL)
   1.426 +		{
   1.427 +		delete iGarbagePolicy;
   1.428 +		iGarbagePolicy=0;
   1.429 +		}
   1.430 +	}
   1.431 +
   1.432 +CLoadManager::CLoadManager() :
   1.433 +	CBase()
   1.434 +	{
   1.435 +	// Do nothing here
   1.436 +	}
   1.437 +
   1.438 +/**
   1.439 +Returns the implementation ID for a given instance Key.
   1.440 +@leave			KErrNotFound
   1.441 +@param			aInstanceKey A key specifying a previously created implementation instance.
   1.442 +@return			TUid The uid of the corresponding implementation.
   1.443 +@pre 			CLoadManager is fully constructed,
   1.444 +@post			CLoadManager remains the same.
   1.445 +*/
   1.446 +TUid CLoadManager::GetImplementationUidL(TUid aInstanceKey)
   1.447 +	{
   1.448 +	// Sanity check that the pointer is divisible by four. A binary number is divisible
   1.449 +	// by four if and only if the two rightmost bits are both zero.
   1.450 +	// This is a compromised check for checking that the pointer is an address.
   1.451 +    if ((aInstanceKey.iUid == 0) || ((aInstanceKey.iUid & 0x00000003) != 0))
   1.452 +        {
   1.453 +	    __ASSERT_DEBUG(EFalse, User::Panic(KEComClientDLLPanicCategory, EEComPanic_InvalidImplementationInstanceKey));
   1.454 +	    User::Leave(KErrNotFound);
   1.455 +        }
   1.456 +
   1.457 +	// The instance info pointer is stored in the instance key.
   1.458 +	CInstanceInfo* instanceInfo = reinterpret_cast<CInstanceInfo*>(aInstanceKey.iUid);
   1.459 +
   1.460 +	// Check that the pointer exists before using it - leaves with KErrNotFound
   1.461 +	iInstanceInfoList.FindInAddressOrderL(instanceInfo);
   1.462 +	return instanceInfo->ImplementationUid();
   1.463 +	}
   1.464 +
   1.465 +/**
   1.466 +Fetches the requested extended interface from a specified implementation instance.
   1.467 +@leave			KErrNotFound
   1.468 +@param 			aInstanceKey A key specifying a previously created implementation.
   1.469 +@param 			aExtendedInterfaceUid Identifies an interface to fetch from the plug-in instance.
   1.470 +@return			TAny* A pointer to the extended interface, will be NULL if it does not exist.
   1.471 +*/
   1.472 +TAny* CLoadManager::GetExtendedInterfaceL(const TUid& aInstanceKey, const TUid& aExtendedInterfaceUid)
   1.473 +	{
   1.474 +	// Sanity check that the pointer is divisible by four. A binary number is divisible
   1.475 +	// by four if and only if the two rightmost bits are both zero.
   1.476 +	// This is a compromised check for checking that the pointer is an address.
   1.477 +    if ((aInstanceKey.iUid == 0) || ((aInstanceKey.iUid & 0x00000003) != 0))
   1.478 +        {
   1.479 +	    __ASSERT_DEBUG(EFalse, User::Panic(KEComClientDLLPanicCategory, EEComPanic_InvalidImplementationInstanceKey));
   1.480 +	    User::Leave(KErrNotFound);
   1.481 +        }
   1.482 +
   1.483 +	// The instance info pointer is stored in the instance key.
   1.484 +	CInstanceInfo* instanceInfo = reinterpret_cast<CInstanceInfo*>(aInstanceKey.iUid);
   1.485 +
   1.486 +	// Check that the pointer exists before using it - leaves with KErrNotFound
   1.487 +	iInstanceInfoList.FindInAddressOrderL(instanceInfo);
   1.488 +	
   1.489 +	// Create the extension object. The instance info object will be populated with
   1.490 +	// the extension info during creation.
   1.491 +	TAny* object = instanceInfo->CreateExtObjectL(aExtendedInterfaceUid);
   1.492 +	return object;
   1.493 +	}
   1.494 +
   1.495 +/**
   1.496 +Manually releases the requested interface. Does nothing if it does not exist.
   1.497 +This interface is optional, normally the interfaces are cleaned up automatically.
   1.498 +@leave			KErrNotFound
   1.499 +@param			aInstanceKey A key specifying a previously created implementation.
   1.500 +@param			aExtendedInterfaceUid Identifies the interface to release
   1.501 +@return None.
   1.502 +*/
   1.503 +void CLoadManager::ManuallyReleaseExtendedInterfaceL(const TUid& aInstanceKey, const TUid& aExtendedInterfaceUid)
   1.504 +	{
   1.505 +	// Sanity check that the pointer is divisible by four. A binary number is divisible
   1.506 +	// by four if and only if the two rightmost bits are both zero.
   1.507 +	// This is a compromised check for checking that the pointer is an address.
   1.508 +    if ((aInstanceKey.iUid == 0) || ((aInstanceKey.iUid & 0x00000003) != 0))
   1.509 +        {
   1.510 +	    __ASSERT_DEBUG(EFalse, User::Panic(KEComClientDLLPanicCategory, EEComPanic_InvalidImplementationInstanceKey));
   1.511 +	    User::Leave(KErrNotFound);
   1.512 +        }
   1.513 +        
   1.514 +	// The instance info pointer is stored in the instance key.
   1.515 +	CInstanceInfo* instanceInfo = reinterpret_cast<CInstanceInfo*>(aInstanceKey.iUid);
   1.516 +
   1.517 +	// Check that the pointer exists before using it - leaves with KErrNotFound
   1.518 +	iInstanceInfoList.FindInAddressOrderL(instanceInfo);
   1.519 +	
   1.520 +	instanceInfo->DestroyExtObject(aExtendedInterfaceUid);
   1.521 +	}
   1.522 +
   1.523 +/**
   1.524 +Utility method to move the policy to the garbage list, remove
   1.525 +policy from the unload policy list.
   1.526 +@param aPolicy Unload policy to clean up
   1.527 +@return None.
   1.528 +*/
   1.529 +void CLoadManager::Cleanup(CUnloadPolicy* aPolicy)
   1.530 +	{
   1.531 +	if (aPolicy->DecreaseReference() == EDeleteMe)
   1.532 +		{
   1.533 +		// Move the policy to the garbage list
   1.534 +		iGarbagePolicy=aPolicy;
   1.535 +
   1.536 +		TInt index = iAllUnloadPolicies.Find(aPolicy);
   1.537 +		if (index != KErrNotFound)
   1.538 +			{
   1.539 +			iAllUnloadPolicies.Remove(index);
   1.540 +			}
   1.541 +		}
   1.542 +	}
   1.543 +
   1.544 +//
   1.545 +// CInstanceInfo
   1.546 +/**
   1.547 +Default constructor of CInstanceInfo
   1.548 +@param			aUnloadPolicy The CUnloadPolicy of the dll
   1.549 +*/
   1.550 +CInstanceInfo::CInstanceInfo(CUnloadPolicy* aUnloadPolicy):
   1.551 +	iUnloadPolicy(aUnloadPolicy)
   1.552 +	{
   1.553 +	// Do nothing here
   1.554 +	}
   1.555 +
   1.556 +/** 
   1.557 +Destructor of CInstanceInfo
   1.558 +*/
   1.559 +CInstanceInfo::~CInstanceInfo()
   1.560 +	{
   1.561 +	// Do nothing here
   1.562 +	}
   1.563 +
   1.564 +
   1.565 +//
   1.566 +// CInstanceInfoExtended
   1.567 +/**
   1.568 +Default constructor of CInstanceInfoExtended
   1.569 +@param			aUnloadPolicy The CUnloadPolicy of the dll
   1.570 +@param			aImplementationProxyRow The interface implementation proxy row entry
   1.571 +*/
   1.572 +CInstanceInfoExtended::CInstanceInfoExtended(CUnloadPolicy* aUnloadPolicy,const TImplementationProxy3* aImplementationProxyRow):
   1.573 +	CInstanceInfo(aUnloadPolicy),iImplementationProxyRow(aImplementationProxyRow)
   1.574 +	{
   1.575 +	// Do nothing here
   1.576 +	}
   1.577 +
   1.578 +/**
   1.579 +create an instance of CInstanceInfoExtended
   1.580 +@param			aUnloadPolicy The CUnloadPolicy of the dll
   1.581 +@param			aImplementationProxyRow The interface implementation proxy row entry
   1.582 +@return			A pointer to the newly created object.
   1.583 +*/
   1.584 +CInstanceInfoExtended* CInstanceInfoExtended::NewL(CUnloadPolicy* aUnloadPolicy,const TImplementationProxy3* aImplementationProxyRow)
   1.585 +	{
   1.586 +	return new(ELeave) CInstanceInfoExtended(aUnloadPolicy,aImplementationProxyRow);
   1.587 +	}
   1.588 +
   1.589 +/**
   1.590 +Sets the implementation object.
   1.591 +@param			aImplementationObject The object instance of this instances' implementation
   1.592 +@return			None
   1.593 +*/
   1.594 +void CInstanceInfoExtended::SetObject(TAny* aImplementationObject)
   1.595 +	{
   1.596 +	iImplementationObject = aImplementationObject;
   1.597 +	}
   1.598 +
   1.599 +/**
   1.600 +Creates the extension interface object. This will use the get extended interface
   1.601 +function pointer from the proxy table to fetch the extended interface from the
   1.602 +plug-in implementation.
   1.603 +@param			aExtendedInterfaceUID The extended interface UID
   1.604 +@return			TAny* A pointer to an instance of an extended interface created
   1.605 +*/
   1.606 +TAny* CInstanceInfoExtended::CreateExtObjectL(const TUid& aExtendedInterfaceUID)
   1.607 +	{
   1.608 +	// Fetch the function pointer to create the extended interface
   1.609 +	TProxyExtendedInterfaceGetPtrL createFunctionPtrL = iImplementationProxyRow->iFuncPtrInterfaceGetL;
   1.610 +	if (createFunctionPtrL == NULL)
   1.611 +		{
   1.612 +		// No extension interface object can be created. Return NULL indicating that
   1.613 +		// no extended interface object is available.
   1.614 +		return NULL;
   1.615 +		}
   1.616 +
   1.617 +	// Valid function pointer exists in proxy table.
   1.618 +	TAny* object = NULL;         // Extended interface object (this points to the interface
   1.619 +							     // within the object)
   1.620 +	TAny* releaseObject = NULL;  // Eextended interface object (this points to the extended
   1.621 +								 // object itself). Used to delete the extended interface
   1.622 +								 // object later.
   1.623 +	TUint32 flags = 0;           // Flags to allow the plug-in and ECOM to communicate
   1.624 +	// Create the extension object.
   1.625 +	object = createFunctionPtrL(iImplementationObject,aExtendedInterfaceUID,flags,releaseObject);
   1.626 +
   1.627 +	if (flags & KReleaseRequiredMask)
   1.628 +		{
   1.629 +		// If release of the extended interface is required then save the release object pointer.
   1.630 +		// The interface object (returned by the function pointer call) and the release object
   1.631 +		// are not necessarily the same pointer. This is because the  interface pointer is not
   1.632 +		// guaranteed to be the same as the pointer to the extended interface object. That
   1.633 +		// is why the release object is required to be fetched from the plug-in.
   1.634 +
   1.635 +        // First perform some checks to ensure that the plugin is consistent 
   1.636 +        TProxyExtendedInterfaceReleasePtr release = iImplementationProxyRow->iFuncPtrInterfaceRelease;
   1.637 +                
   1.638 +        if (release == NULL)
   1.639 +            {
   1.640 +            // ...the release object pointer must not be non null
   1.641 +            __ECOM_TRACE("ECOM: PANIC in CInstanceInfoExtended::CreateExtObjectL, release required but release function missing");
   1.642 +	        __ASSERT_DEBUG(EFalse, User::Panic (KEComClientDLLPanicCategory, EEComPanic_CInstanceInfoExtended_CreateExtObjectL_NoReleaseFunc));
   1.643 +            User::Leave(KEComErrNoExtendedInterfaceReleaseFunction);
   1.644 +            }
   1.645 +            
   1.646 +        if (releaseObject == NULL)
   1.647 +            {
   1.648 +            // ... the releaseObject must be non null
   1.649 +            __ECOM_TRACE("ECOM: PANIC in CInstanceInfoExtended::CreateExtObjectL, release required but release object missing");
   1.650 +	        __ASSERT_DEBUG(EFalse, User::Panic (KEComClientDLLPanicCategory, EEComPanic_CInstanceInfoExtended_CreateExtObjectL_NoReleaseObj));
   1.651 +           User::Leave(KEComErrNoExtendedInterfaceReleaseObject);
   1.652 + 		    }
   1.653 +            
   1.654 +		//Create the extended object info type and add it to the extended object info array.
   1.655 +		TExtendedObjectInfo extendedObjectInfo;
   1.656 +		extendedObjectInfo.iExtendedInterfaceObject = releaseObject;
   1.657 +		extendedObjectInfo.iExtendedInterfaceUID = aExtendedInterfaceUID;
   1.658 +
   1.659 +		TInt err = iExtendedObjectInfo.Append(extendedObjectInfo);
   1.660 +
   1.661 +		if (err != KErrNone)
   1.662 +			{
   1.663 +			if (release != NULL)
   1.664 +				{
   1.665 +				// Release the extended interface. Release must not leave.
   1.666 +				release(extendedObjectInfo.iExtendedInterfaceObject,extendedObjectInfo.iExtendedInterfaceUID);
   1.667 +				}
   1.668 +			User::Leave(err);
   1.669 +			}
   1.670 +		}
   1.671 +
   1.672 +	return object;
   1.673 +	}
   1.674 +
   1.675 +
   1.676 +/**
   1.677 +Destroys the extension interface object.
   1.678 +@param			aExtendedInterfaceUID The extended interface UID
   1.679 +@return			None
   1.680 +*/
   1.681 +void CInstanceInfoExtended::DestroyExtObject(const TUid& aExtendedInterfaceUID)
   1.682 +	{
   1.683 +	// Get release interface
   1.684 +	TProxyExtendedInterfaceReleasePtr release = iImplementationProxyRow->iFuncPtrInterfaceRelease;
   1.685 +
   1.686 +	if (release != NULL)
   1.687 +		{
   1.688 +		// Release extended interface. Find the extended object info.
   1.689 +		for(TInt i = 0; i < iExtendedObjectInfo.Count(); i++)
   1.690 +			{
   1.691 +			if (iExtendedObjectInfo[i].iExtendedInterfaceUID == aExtendedInterfaceUID)
   1.692 +				{
   1.693 +				TAny* releaseObject = iExtendedObjectInfo[i].iExtendedInterfaceObject;
   1.694 +                if (releaseObject == NULL)
   1.695 +                    {
   1.696 +                    // ... the releaseObject must be non null
   1.697 +                    __ECOM_TRACE("ECOM: PANIC in CInstanceInfoExtended::DestroyExtObject, release required but release object missing");
   1.698 +	                __ASSERT_DEBUG(EFalse, User::Panic (KEComClientDLLPanicCategory, EEComPanic_CInstanceInfoExtended_DestroyExtObject_NoReleaseObj));	              
   1.699 +		            }
   1.700 +
   1.701 +				// Release the extended interface. Release should not be leaving.
   1.702 +				release(releaseObject, iExtendedObjectInfo[i].iExtendedInterfaceUID);
   1.703 +				// Remove the extended object info element from the array.
   1.704 +				iExtendedObjectInfo.Remove(i);
   1.705 +				break;
   1.706 +				}
   1.707 +			}
   1.708 +		}
   1.709 +	}
   1.710 +
   1.711 +/**
   1.712 +Destructor of CInstanceInfoExtended
   1.713 +*/
   1.714 +CInstanceInfoExtended::~CInstanceInfoExtended()
   1.715 +	{
   1.716 +	// Get release interface
   1.717 +	if (iImplementationProxyRow != NULL)
   1.718 +		{
   1.719 +		TProxyExtendedInterfaceReleasePtr release = iImplementationProxyRow->iFuncPtrInterfaceRelease;
   1.720 +
   1.721 +		if (release != NULL)
   1.722 +			{
   1.723 +			// Release all extended interfaces (if any still to be released)
   1.724 +			for(TInt i = 0; i < iExtendedObjectInfo.Count(); i++)
   1.725 +				{
   1.726 +				// Release is supposed to be non-leavable.
   1.727 +				release(iExtendedObjectInfo[i].iExtendedInterfaceObject,iExtendedObjectInfo[i].iExtendedInterfaceUID);
   1.728 +				}
   1.729 +			}
   1.730 +		}
   1.731 +
   1.732 +	iExtendedObjectInfo.Close();
   1.733 +	}
   1.734 +
   1.735 +//
   1.736 +// CInstanceInfoSimple
   1.737 +/**
   1.738 +Default constructor of CInstanceInfoSimple
   1.739 +@param			aUnloadPolicy The CUnloadPolicy of the dll
   1.740 +@param			aImplementationProxyRow The interface implementation proxy row entry
   1.741 +*/
   1.742 +CInstanceInfoSimple::CInstanceInfoSimple(CUnloadPolicy* aUnloadPolicy,const TImplementationProxy* aImplementationProxyRow):
   1.743 +	CInstanceInfo(aUnloadPolicy),iImplementationProxyRow(aImplementationProxyRow)
   1.744 +	{
   1.745 +	// Do nothing here
   1.746 +	}
   1.747 +
   1.748 +/**
   1.749 +Create an instance of CInstanceInfoSimple
   1.750 +@param			aUnloadPolicy The CUnloadPolicy of the dll
   1.751 +@param			aImplementationProxyRow The interface implementation proxy row entry
   1.752 +@return			A pointer to the newly created object.
   1.753 +*/
   1.754 +CInstanceInfoSimple* CInstanceInfoSimple::NewL(CUnloadPolicy* aUnloadPolicy,const TImplementationProxy* aImplementationProxyRow)
   1.755 +	{
   1.756 +	return new(ELeave) CInstanceInfoSimple(aUnloadPolicy,aImplementationProxyRow);
   1.757 +	}