os/ossrv/lowlevellibsandfws/pluginfw/Framework/frame/EComServerSession.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
// Server session object implementation.
sl@0
    15
// 
sl@0
    16
//
sl@0
    17
sl@0
    18
#include "EComServerStart.h"
sl@0
    19
#include "EComMessageIds.h"
sl@0
    20
#include <ecom/ecomerrorcodes.h>
sl@0
    21
#include <ecom/implementationinformation.h>
sl@0
    22
#include "EComServerSession.h"
sl@0
    23
#include "EComSessionAux.h"
sl@0
    24
#include "EComServer.h"
sl@0
    25
#include "TestUtilities.h"	// For __FILE__LINE__
sl@0
    26
#include "EComPerformance.h"
sl@0
    27
#include "EComDebug.h"
sl@0
    28
sl@0
    29
//
sl@0
    30
// RMessage::Panic() completes the message. This is:
sl@0
    31
// (a) important for efficient cleanup within the kernel.
sl@0
    32
// (b) a problem if the message is completed a second time.
sl@0
    33
//
sl@0
    34
void PanicClient(const TClientRequest& aMessage, TInt aPanicCode)
sl@0
    35
	{
sl@0
    36
	aMessage.Panic(KEComServerPanicCategory, aPanicCode);
sl@0
    37
	}
sl@0
    38
sl@0
    39
//
sl@0
    40
//	class CEComServerSession
sl@0
    41
//
sl@0
    42
CEComServerSession::CEComServerSession()
sl@0
    43
	: CSession2()
sl@0
    44
	{
sl@0
    45
	// Do nothing
sl@0
    46
	}
sl@0
    47
sl@0
    48
//
sl@0
    49
// 2nd phase construct for sessions - called by the CServer framework
sl@0
    50
//
sl@0
    51
void CEComServerSession::CreateL()
sl@0
    52
	{
sl@0
    53
	Server().AddSession();
sl@0
    54
	}
sl@0
    55
sl@0
    56
CEComServerSession::~CEComServerSession()
sl@0
    57
	{
sl@0
    58
	CleanupInternalList();
sl@0
    59
	CompleteNotifications(KErrCancel);
sl@0
    60
	delete iMemoryStore;
sl@0
    61
	Server().DropSession();
sl@0
    62
	}
sl@0
    63
sl@0
    64
//
sl@0
    65
// Deliver the notification message to the client
sl@0
    66
//
sl@0
    67
void CEComServerSession::CompleteNotifications(TInt aCompletionCode)
sl@0
    68
	{
sl@0
    69
	const TInt count = iNotificationRequests.Count();
sl@0
    70
	for(TInt i = 0; i < count; ++i)
sl@0
    71
		{
sl@0
    72
		  iNotificationRequests[i].iMessage.Complete(aCompletionCode);
sl@0
    73
sl@0
    74
		}
sl@0
    75
	iNotificationRequests.Reset();
sl@0
    76
	}
sl@0
    77
sl@0
    78
//
sl@0
    79
// Handle a client request.
sl@0
    80
// Leaving is handled by CEComServer::RunError() which reports the error code
sl@0
    81
// to the client.
sl@0
    82
//
sl@0
    83
void CEComServerSession::ServiceL(const RMessage2& aMessage)
sl@0
    84
{
sl@0
    85
	const TClientRequest msg(aMessage);
sl@0
    86
	ServiceL(msg);
sl@0
    87
}
sl@0
    88
sl@0
    89
void CEComServerSession::ServiceL(const TClientRequest& aMessage)
sl@0
    90
	{
sl@0
    91
	TInt completionCode = KErrNone;
sl@0
    92
	TBool asyncRequest = EFalse;
sl@0
    93
sl@0
    94
	switch (aMessage.Function())
sl@0
    95
		{
sl@0
    96
	case ENotifyOnChange:
sl@0
    97
		{
sl@0
    98
RECORD_CLIENT_REQUEST_START_TIMER_RESULT(EEComNotifyOnChangeRequestType, Server().GetCurrentStartupState());
sl@0
    99
		SEComNotification notif;
sl@0
   100
		notif.iMessage=aMessage;
sl@0
   101
		// the TRequestStatus as a TInt is stored for later comparisons
sl@0
   102
		notif.iRequestStatusHandle=aMessage.Int0();
sl@0
   103
		//Check that this TRequestStatus is not already being used.
sl@0
   104
		const TInt count = iNotificationRequests.Count();
sl@0
   105
		for(TInt i = 0; i < count; ++i)
sl@0
   106
		{
sl@0
   107
			if(iNotificationRequests[i].iRequestStatusHandle == notif.iRequestStatusHandle)
sl@0
   108
				User::Leave(KErrArgument);
sl@0
   109
sl@0
   110
		}
sl@0
   111
		User::LeaveIfError(iNotificationRequests.Append(notif));
sl@0
   112
		asyncRequest = ETrue;
sl@0
   113
RECORD_CLIENT_REQUEST_END_TIMER_RESULT;
sl@0
   114
		break;
sl@0
   115
		}
sl@0
   116
sl@0
   117
	case ECancelNotifyOnChange:
sl@0
   118
			{
sl@0
   119
RECORD_CLIENT_REQUEST_START_TIMER_RESULT(EEComCancelNotifyOnChangeRequestType, Server().GetCurrentStartupState());
sl@0
   120
			TInt statusHandle = aMessage.Int0();
sl@0
   121
sl@0
   122
			const TInt count = iNotificationRequests.Count();
sl@0
   123
			for(TInt i = 0; i < count; ++i)
sl@0
   124
				{
sl@0
   125
				if(iNotificationRequests[i].iRequestStatusHandle == statusHandle)
sl@0
   126
					{
sl@0
   127
						iNotificationRequests[i].iMessage.Complete(KErrCancel);
sl@0
   128
						iNotificationRequests.Remove(i);
sl@0
   129
						break;  // Terminate the loop
sl@0
   130
					}
sl@0
   131
				}
sl@0
   132
			}
sl@0
   133
RECORD_CLIENT_REQUEST_END_TIMER_RESULT;
sl@0
   134
		break;
sl@0
   135
sl@0
   136
	case EListImplementations:
sl@0
   137
	case EListResolvedImplementations:
sl@0
   138
	case EListCustomResolvedImplementations:
sl@0
   139
		if(Server().RegistryIndexValid())
sl@0
   140
			{
sl@0
   141
RECORD_CLIENT_REQUEST_START_TIMER_RESULT(EEComListRequestType, Server().GetCurrentStartupState());
sl@0
   142
			DoListImplementationsL(aMessage);
sl@0
   143
RECORD_CLIENT_REQUEST_END_TIMER_RESULT;
sl@0
   144
			}
sl@0
   145
		else
sl@0
   146
			{
sl@0
   147
			if(ReceivePending())
sl@0
   148
				User::Leave(KEComErrListInvalidAwaitNotification);
sl@0
   149
			else
sl@0
   150
				User::Leave(KEComErrListCurrentlyUnavailable);
sl@0
   151
			}
sl@0
   152
		break;
sl@0
   153
sl@0
   154
	case ECollectImplementationsList:
sl@0
   155
RECORD_CLIENT_REQUEST_START_TIMER_RESULT(EEComCollectImplementationsRequestType, Server().GetCurrentStartupState());
sl@0
   156
		if(!DoCollectListL(aMessage))
sl@0
   157
			completionCode = KErrNotReady;
sl@0
   158
RECORD_CLIENT_REQUEST_END_TIMER_RESULT;
sl@0
   159
		break;
sl@0
   160
sl@0
   161
	case EGetImplementationCreationMethod:
sl@0
   162
	case EGetResolvedCreationMethod:
sl@0
   163
	case EGetCustomResolvedCreationMethod:
sl@0
   164
RECORD_CLIENT_REQUEST_START_TIMER_RESULT(EEComCreateRequestType, Server().GetCurrentStartupState());
sl@0
   165
		DoGetResolvedImplementationL(aMessage);
sl@0
   166
RECORD_CLIENT_REQUEST_END_TIMER_RESULT;
sl@0
   167
		break;
sl@0
   168
sl@0
   169
	case EListExtendedInterfaces:
sl@0
   170
RECORD_CLIENT_REQUEST_START_TIMER_RESULT(EEComListExtendedInterfacesRequestType, Server().GetCurrentStartupState());
sl@0
   171
		DoListExtendedInterfacesL(aMessage);
sl@0
   172
RECORD_CLIENT_REQUEST_END_TIMER_RESULT;
sl@0
   173
		break;
sl@0
   174
sl@0
   175
	case EEnableImplementation:
sl@0
   176
	case EDisableImplementation:
sl@0
   177
		// Ommissions for 6.2
sl@0
   178
		completionCode = KErrNotSupported;
sl@0
   179
		break;
sl@0
   180
#if defined(__ECOM_SERVER_TESTABILITY__) || defined(__ECOM_SERVER_PERFORMANCE__)
sl@0
   181
	case ESetGetParameters:
sl@0
   182
		DoSetGetParametersL(aMessage);
sl@0
   183
		break;
sl@0
   184
#endif
sl@0
   185
	//EDestroyedImplementation obsolete due to implementation creation
sl@0
   186
	//relocation to client side from server
sl@0
   187
	case EDestroyedImplementation:
sl@0
   188
	default:
sl@0
   189
		// Something badly wrong if we get here.
sl@0
   190
		PanicClient(aMessage,KEComErrUnknownService);
sl@0
   191
		// RMessage::Panic() has completed the message
sl@0
   192
		// so treat this as an asynch request.
sl@0
   193
		asyncRequest = ETrue;
sl@0
   194
		}
sl@0
   195
	if (!asyncRequest)
sl@0
   196
		aMessage.Complete(completionCode);
sl@0
   197
	}
sl@0
   198
sl@0
   199
sl@0
   200
sl@0
   201
	
sl@0
   202
TInt const KDefaultStoreSize = 256;	// Enough space for approx 1 implementation : will grow to fit if required
sl@0
   203
sl@0
   204
/**
sl@0
   205
UnPack the match string and extended interface from the client supplied parameters.
sl@0
   206
@param			aMessage 
sl@0
   207
@param			aExtendedInterfaces Return value consisting of an array containing the extended interfaces.
sl@0
   208
@param			aMatchStr Return value consisting of the matching string.
sl@0
   209
*/
sl@0
   210
void CEComServerSession::UnpackMatchStrAndExtendedInterfacesFromClientL(const TClientRequest& aMessage,
sl@0
   211
													RExtendedInterfacesArray& aExtendedInterfaces,
sl@0
   212
													RBuf8& aMatchStr)
sl@0
   213
	{
sl@0
   214
	
sl@0
   215
	//now get the matchString and extendedInterfaces
sl@0
   216
	TInt sizeOfMatchStrExtInfBuf = aMessage.GetDesLength(KIPCParameterMatchStrExtInf);
sl@0
   217
	User::LeaveIfError(sizeOfMatchStrExtInfBuf);
sl@0
   218
	RBuf8 matchStrExtInfBuf;
sl@0
   219
	matchStrExtInfBuf.CreateMaxL(sizeOfMatchStrExtInfBuf);
sl@0
   220
	matchStrExtInfBuf.CleanupClosePushL();
sl@0
   221
	
sl@0
   222
	aMessage.ReadL(KIPCParameterMatchStrExtInf,matchStrExtInfBuf);
sl@0
   223
	RDesReadStream readStream;
sl@0
   224
	CleanupClosePushL(readStream);
sl@0
   225
	readStream.Open(matchStrExtInfBuf);
sl@0
   226
	TInt lenOfMatchStr = readStream.ReadInt32L();
sl@0
   227
	aMatchStr.CreateMaxL(lenOfMatchStr);
sl@0
   228
	aMatchStr.CleanupClosePushL();
sl@0
   229
	if (lenOfMatchStr>0)
sl@0
   230
		{
sl@0
   231
		readStream.ReadL(aMatchStr,lenOfMatchStr);		
sl@0
   232
		}
sl@0
   233
	TInt numOfExtendedInterfaces = readStream.ReadInt32L();
sl@0
   234
	CleanupClosePushL(aExtendedInterfaces);
sl@0
   235
	for(TInt i = 0; i < numOfExtendedInterfaces; i++)
sl@0
   236
		{
sl@0
   237
		aExtendedInterfaces.AppendL(TUid::Uid(readStream.ReadInt32L()));
sl@0
   238
		}
sl@0
   239
	
sl@0
   240
	CleanupStack::Pop(&aExtendedInterfaces);
sl@0
   241
	CleanupStack::Pop(&aMatchStr);
sl@0
   242
	CleanupStack::PopAndDestroy(&readStream);
sl@0
   243
	CleanupStack::PopAndDestroy(&matchStrExtInfBuf); 
sl@0
   244
	}
sl@0
   245
sl@0
   246
sl@0
   247
// Note that this method for returning the arbitrary sized data set 
sl@0
   248
// will not work IF the session is shared so...
sl@0
   249
// DO NOT SHARE SERVER SIDE SESSIONS BETWEEN CLIENTS
sl@0
   250
void CEComServerSession::DoListImplementationsL(const TClientRequest& aMessage)
sl@0
   251
	{
sl@0
   252
	// Unpack the client supplied parameters
sl@0
   253
	// Firstly get the uids
sl@0
   254
	TUidType uids;
sl@0
   255
	TPckg<TUidType> uidsPkg(uids);	
sl@0
   256
	aMessage.ReadL(KIPCParameterUids, uidsPkg);
sl@0
   257
	
sl@0
   258
	if(uids[KInterfaceUidIndex] == KNullUid)
sl@0
   259
		{
sl@0
   260
		User::Leave(KEComErrMissingParameter);
sl@0
   261
		}
sl@0
   262
sl@0
   263
	//now get the TListImplParam parameters
sl@0
   264
  	TListImplParam listParam;
sl@0
   265
  	TPckg<TListImplParam> listParamPkg(listParam);
sl@0
   266
  	aMessage.ReadL(2,listParamPkg);
sl@0
   267
	
sl@0
   268
	// Now rebuild the TEComResolverParams
sl@0
   269
	TEComResolverParams resolverParameters;
sl@0
   270
	resolverParameters.SetGenericMatch(listParam.iMatchType);
sl@0
   271
	
sl@0
   272
	//now get the matchString and extendedInterfaces
sl@0
   273
	RBuf8 matchStr;
sl@0
   274
	RExtendedInterfacesArray extendedInterfaces;
sl@0
   275
	UnpackMatchStrAndExtendedInterfacesFromClientL(aMessage,extendedInterfaces,matchStr);
sl@0
   276
	matchStr.CleanupClosePushL();
sl@0
   277
	CleanupClosePushL(extendedInterfaces);
sl@0
   278
	
sl@0
   279
	if(matchStr.Length()>0)
sl@0
   280
		{	
sl@0
   281
		resolverParameters.SetDataType(matchStr);
sl@0
   282
		}
sl@0
   283
	// Else the client's resolver params are default constructed i.e. invalid
sl@0
   284
	// data type descriptor or its length is zero, so use empty RBuf8 above.
sl@0
   285
	
sl@0
   286
		
sl@0
   287
	// Pass to the server
sl@0
   288
	iListContext = aMessage.Function();
sl@0
   289
	switch(iListContext)
sl@0
   290
		{
sl@0
   291
		case EListImplementations:
sl@0
   292
			iList = Server().ListImplementationsL( uids[KInterfaceUidIndex], extendedInterfaces, aMessage );
sl@0
   293
			break;
sl@0
   294
		case EListResolvedImplementations:
sl@0
   295
			if(matchStr.Length() == 0)
sl@0
   296
				{
sl@0
   297
				User::Leave(KEComErrMissingParameter);
sl@0
   298
				}
sl@0
   299
			iList = Server().ListImplementationsL( uids[KInterfaceUidIndex], resolverParameters, extendedInterfaces, aMessage);
sl@0
   300
			break;
sl@0
   301
		case EListCustomResolvedImplementations:
sl@0
   302
			if(uids[KResolverUidIndex] == KNullUid)
sl@0
   303
				{
sl@0
   304
				User::Leave(KEComErrMissingParameter);
sl@0
   305
				}
sl@0
   306
			iList = Server().ListImplementationsL( uids[KInterfaceUidIndex], resolverParameters, uids[KResolverUidIndex], extendedInterfaces, aMessage);
sl@0
   307
			break;
sl@0
   308
		default:
sl@0
   309
			break;
sl@0
   310
		}
sl@0
   311
	//Now cleanup the extended interface
sl@0
   312
	CleanupStack::PopAndDestroy(&extendedInterfaces);
sl@0
   313
	CleanupStack::PopAndDestroy(&matchStr);
sl@0
   314
sl@0
   315
	TInt bufferSizeRequired=0;
sl@0
   316
	// Package the array for return
sl@0
   317
	if(iList)
sl@0
   318
		{
sl@0
   319
		if(iList->Count()>0)
sl@0
   320
			{
sl@0
   321
			// Allocate a new store and replace the old one first
sl@0
   322
			CBufFlat* memoryStore = CBufFlat::NewL(KDefaultStoreSize);
sl@0
   323
			delete iMemoryStore;
sl@0
   324
			iMemoryStore = memoryStore;
sl@0
   325
sl@0
   326
			// Note : There is no need to push
sl@0
   327
			// the write stream onto the cleanup stack
sl@0
   328
			// because it has no internal resources.
sl@0
   329
			RBufWriteStream writeStream;
sl@0
   330
			writeStream.Open(*iMemoryStore);
sl@0
   331
sl@0
   332
			// Build the store data then calculate the end size;
sl@0
   333
			const TInt entryCount = iList->Count();		
sl@0
   334
			writeStream.WriteInt32L(entryCount);
sl@0
   335
			
sl@0
   336
			for(TInt i = 0; i < entryCount; ++i)
sl@0
   337
				(*iList)[i]->ExternalizeL(ETrue,writeStream);
sl@0
   338
	
sl@0
   339
			writeStream.CommitL();
sl@0
   340
sl@0
   341
			// Set to actual size
sl@0
   342
			bufferSizeRequired=iMemoryStore->Size();
sl@0
   343
			__ECOM_TRACE1("ECOM ListImplementations request buffer size required=%d",bufferSizeRequired);
sl@0
   344
			}
sl@0
   345
			CleanupInternalList();	
sl@0
   346
		}
sl@0
   347
	
sl@0
   348
	//if nothing is returned we should still indicate this to the client side
sl@0
   349
	if (bufferSizeRequired==0)
sl@0
   350
		{
sl@0
   351
		//write back the bufferSize
sl@0
   352
		listParam.iBufferSize=0;
sl@0
   353
  		aMessage.WriteL(2,listParamPkg);
sl@0
   354
		return;
sl@0
   355
		}
sl@0
   356
		
sl@0
   357
	//if the preallocated size is big enough to hold our entry
sl@0
   358
	//copy it to the client
sl@0
   359
	if (listParam.iBufferSize >= bufferSizeRequired)
sl@0
   360
		{
sl@0
   361
		if (iMemoryStore)
sl@0
   362
			{
sl@0
   363
			//write back the bufferSize
sl@0
   364
			listParam.iBufferSize=bufferSizeRequired;
sl@0
   365
  			aMessage.WriteL(2,listParamPkg);
sl@0
   366
			TPtr8 data=iMemoryStore->Ptr(0);
sl@0
   367
			aMessage.WriteL(3,data);
sl@0
   368
			delete iMemoryStore;
sl@0
   369
			iMemoryStore=NULL;
sl@0
   370
			}
sl@0
   371
		}
sl@0
   372
	//if not rewrite back to the client the size that is required
sl@0
   373
	//and signal with KErrOverFlow to the client
sl@0
   374
	else
sl@0
   375
		{
sl@0
   376
		//write back the bufferSize
sl@0
   377
		listParam.iBufferSize=bufferSizeRequired;
sl@0
   378
  		aMessage.WriteL(2,listParamPkg);
sl@0
   379
		User::Leave(KErrOverflow);
sl@0
   380
		}
sl@0
   381
	}
sl@0
   382
sl@0
   383
TBool CEComServerSession::DoCollectListL(const TClientRequest& aMessage)
sl@0
   384
	{
sl@0
   385
	TBool success = EFalse;
sl@0
   386
	if(iMemoryStore)
sl@0
   387
		{
sl@0
   388
		TPtr8 data=iMemoryStore->Ptr(0);
sl@0
   389
		aMessage.WriteL(0, data);
sl@0
   390
		delete iMemoryStore;
sl@0
   391
		iMemoryStore = NULL;
sl@0
   392
		success = ETrue;
sl@0
   393
		}
sl@0
   394
	return success;
sl@0
   395
	}
sl@0
   396
sl@0
   397
void CEComServerSession::DoGetResolvedImplementationL(const TClientRequest& aMessage)
sl@0
   398
	{
sl@0
   399
	// Unpack the client supplied parameters
sl@0
   400
	// Firstly get the uids
sl@0
   401
	TUidType uids;
sl@0
   402
	TPckg<TUidType> uidsPkg(uids);
sl@0
   403
	aMessage.ReadL(KIPCParameterUids, uidsPkg);
sl@0
   404
sl@0
   405
	// Now rebuild the TEComResolverParams
sl@0
   406
	TEComResolverParams resolverParameters;
sl@0
   407
	resolverParameters.SetGenericMatch(KIPCParameterResolverParamsTypePtr);	
sl@0
   408
	
sl@0
   409
	//now get the matchString and extendedInterfaces
sl@0
   410
	RBuf8 matchStr;
sl@0
   411
	matchStr.CleanupClosePushL();
sl@0
   412
	RExtendedInterfacesArray extendedInterfaces;
sl@0
   413
	CleanupClosePushL(extendedInterfaces);
sl@0
   414
	UnpackMatchStrAndExtendedInterfacesFromClientL(aMessage,extendedInterfaces,matchStr);
sl@0
   415
    if(matchStr.Length()>0)
sl@0
   416
		{	
sl@0
   417
		resolverParameters.SetDataType(matchStr);
sl@0
   418
		}
sl@0
   419
	// Else the client's resolver params are default constructed i.e. invalid
sl@0
   420
	// data type descriptor or its length is zero, so use empty HBufC8 above.
sl@0
   421
	// Set up for the return value
sl@0
   422
	TUid dtorIdKey(KNullUid);
sl@0
   423
	TEntry loadInfo;
sl@0
   424
	
sl@0
   425
	switch(aMessage.Function())
sl@0
   426
		{
sl@0
   427
		case EGetImplementationCreationMethod:
sl@0
   428
			Server().GetResolvedDllInfoL(uids[KInterfaceUidIndex], 
sl@0
   429
										 loadInfo, 
sl@0
   430
										 dtorIdKey,
sl@0
   431
										 aMessage);
sl@0
   432
			break;
sl@0
   433
		case EGetResolvedCreationMethod:
sl@0
   434
			Server().GetResolvedDllInfoL(uids[KInterfaceUidIndex], 
sl@0
   435
										 resolverParameters, 
sl@0
   436
										 extendedInterfaces,
sl@0
   437
										 loadInfo, 
sl@0
   438
										 dtorIdKey,
sl@0
   439
										 aMessage);
sl@0
   440
			break;
sl@0
   441
		case EGetCustomResolvedCreationMethod:
sl@0
   442
			Server().GetResolvedDllInfoL(uids[KInterfaceUidIndex], 
sl@0
   443
										 resolverParameters, 
sl@0
   444
										 uids[KResolverUidIndex], 
sl@0
   445
										 extendedInterfaces,
sl@0
   446
										 loadInfo, 
sl@0
   447
										 dtorIdKey,
sl@0
   448
										 aMessage);
sl@0
   449
			break;
sl@0
   450
		default:
sl@0
   451
			break;
sl@0
   452
		}
sl@0
   453
	CleanupStack::PopAndDestroy(&extendedInterfaces);
sl@0
   454
	CleanupStack::PopAndDestroy(&matchStr);
sl@0
   455
	
sl@0
   456
// ??? Compile time assert that sizeof(TProxyNewLPtr) == sizeof(TAny*)?
sl@0
   457
// Currently I'm not arranging for the client-side of the session to
sl@0
   458
// convert from TAny* to TProxyNewLPtr, and using this to avoid the
sl@0
   459
// full agony of following through with conversion...
sl@0
   460
sl@0
   461
	// Then re-package the results for return
sl@0
   462
	// Firstly the Interface Implementation creation method pointer
sl@0
   463
	TPckg<TEntry> result(loadInfo);
sl@0
   464
	
sl@0
   465
	aMessage.WriteL(KIPCReturnParameterCreationMethodPtr, result);
sl@0
   466
	// Next the destruction identification uid
sl@0
   467
	TUidType type(KNullUid,dtorIdKey,KNullUid);
sl@0
   468
	TPckg<TUidType> dtorIdKeyPkg(type);
sl@0
   469
	
sl@0
   470
	aMessage.WriteL(KIPCReturnParameterUidsPtr, dtorIdKeyPkg);
sl@0
   471
	}
sl@0
   472
sl@0
   473
// Return the list of interfaces to the client. If not enough space
sl@0
   474
// has been allocated by the client, KErrOverflow will be returned.
sl@0
   475
//
sl@0
   476
void CEComServerSession::DoListExtendedInterfacesL(const TClientRequest& aMessage)
sl@0
   477
	{
sl@0
   478
	// Unpack the client supplied parameters
sl@0
   479
	TInt bufferSize = 0;
sl@0
   480
	TUid implementationUid(KNullUid);
sl@0
   481
	TPckg<TUid> implementationUidDes(implementationUid);
sl@0
   482
	TPckg<TInt> bufferSizeDes(bufferSize);
sl@0
   483
	aMessage.ReadL(KIPCParameterImplementationUid,implementationUidDes);
sl@0
   484
	aMessage.ReadL(KIPCParameterBufferSize,bufferSizeDes);
sl@0
   485
		
sl@0
   486
	// Get the implementation information for this implementation UID.
sl@0
   487
	CImplementationInformation* implInfo = NULL;
sl@0
   488
	Server().GetImplementationInformationL(implementationUid,implInfo,aMessage);
sl@0
   489
	
sl@0
   490
	TInt numExtendedInterfaces = 0; // Number of extended interfaces to return to client.
sl@0
   491
sl@0
   492
	if(implInfo)
sl@0
   493
		{
sl@0
   494
		TInt bufferSizeRequired = 0; // Buffer required to send extended interface data back to client
sl@0
   495
		
sl@0
   496
		// Fetch the list of extended interfaces
sl@0
   497
		RExtendedInterfacesArray* extendedInterfaceList = implInfo->GetExtendedInterfaceList();
sl@0
   498
		if (extendedInterfaceList != NULL)
sl@0
   499
			{
sl@0
   500
			numExtendedInterfaces = extendedInterfaceList->Count();
sl@0
   501
			}
sl@0
   502
		if (numExtendedInterfaces > 0)
sl@0
   503
			{
sl@0
   504
			bufferSizeRequired = numExtendedInterfaces * sizeof(TUid);
sl@0
   505
			__ECOM_TRACE1("ECOM ListInterfaces request buffer size required=%d",bufferSizeRequired);
sl@0
   506
			
sl@0
   507
			//if the preallocated size is big enough to hold our entry
sl@0
   508
			//copy it to the client
sl@0
   509
			if (bufferSize >= bufferSizeRequired)
sl@0
   510
				{
sl@0
   511
				RBuf8 buf;
sl@0
   512
				CleanupClosePushL(buf);
sl@0
   513
				buf.CreateL(bufferSizeRequired);   // Create the RBuf.
sl@0
   514
				
sl@0
   515
				// Note : There is no need to push the write stream onto the cleanup stack
sl@0
   516
				// because it has no internal resources.
sl@0
   517
				RDesWriteStream writeStream;
sl@0
   518
				writeStream.Open(buf);
sl@0
   519
sl@0
   520
				// Build the data of extendedInterfaces;
sl@0
   521
				for(TInt i = 0; i < numExtendedInterfaces; ++i)
sl@0
   522
					{
sl@0
   523
					writeStream.WriteInt32L((*extendedInterfaceList)[i].iUid);
sl@0
   524
					}
sl@0
   525
sl@0
   526
				writeStream.CommitL();
sl@0
   527
				
sl@0
   528
				// Copy the data to the client.
sl@0
   529
				bufferSize=bufferSizeRequired;
sl@0
   530
				aMessage.WriteL(KIPCParameterBufferSize,bufferSizeDes);
sl@0
   531
				aMessage.WriteL(KIPCParameterInterfaceData,buf);
sl@0
   532
				
sl@0
   533
				CleanupStack::PopAndDestroy(&buf);
sl@0
   534
				}
sl@0
   535
			//if not rewrite back to the client the size that is required
sl@0
   536
			//and signal with KErrOverFlow to the client
sl@0
   537
			else
sl@0
   538
				{
sl@0
   539
				bufferSize=bufferSizeRequired;
sl@0
   540
				aMessage.WriteL(KIPCParameterBufferSize,bufferSizeDes);
sl@0
   541
				User::Leave(KErrOverflow);
sl@0
   542
				}
sl@0
   543
			}
sl@0
   544
		}
sl@0
   545
		
sl@0
   546
	//if nothing is returned we should still indicate this to the client side
sl@0
   547
	if (numExtendedInterfaces == 0)
sl@0
   548
		{
sl@0
   549
		bufferSize=0;
sl@0
   550
		aMessage.WriteL(KIPCParameterBufferSize,bufferSizeDes);
sl@0
   551
		}
sl@0
   552
	}
sl@0
   553
sl@0
   554
#if defined(__ECOM_SERVER_TESTABILITY__) || defined(__ECOM_SERVER_PERFORMANCE__)
sl@0
   555
/**
sl@0
   556
This method is provided for testability. It allows the user to 
sl@0
   557
send and receive any parameters. 
sl@0
   558
@param aMessage IPC message between server and client
sl@0
   559
*/
sl@0
   560
void CEComServerSession::DoSetGetParametersL(const TClientRequest& aMessage)
sl@0
   561
	{
sl@0
   562
	TInt parameterType = aMessage.Int0();
sl@0
   563
	
sl@0
   564
	switch(parameterType)
sl@0
   565
		{
sl@0
   566
#ifdef __ECOM_SERVER_TESTABILITY__
sl@0
   567
		case EChangeStartupState:
sl@0
   568
			Server().ChangeStartupStateL(aMessage.Int1());
sl@0
   569
			break;
sl@0
   570
		case EProcessStartupState:
sl@0
   571
			Server().ProcessCurrentStartupStateL();
sl@0
   572
			break;
sl@0
   573
		case EGetStartupState:
sl@0
   574
			{
sl@0
   575
			TInt state = Server().GetCurrentStartupState();
sl@0
   576
			TPckg<TInt> pckgState(state);
sl@0
   577
		    aMessage.Write(1, pckgState);
sl@0
   578
			break;
sl@0
   579
			}
sl@0
   580
#endif
sl@0
   581
#ifdef __ECOM_SERVER_PERFORMANCE__
sl@0
   582
		case EGetStartupStateTimerResult:
sl@0
   583
			{
sl@0
   584
			TStartupStateTimerEntry timerEntry;
sl@0
   585
			
sl@0
   586
			TInt ret = EComPerformance::GetStartupStateTimerResult(aMessage.Int1(), timerEntry.iTimerResult, timerEntry.iState);
sl@0
   587
			
sl@0
   588
			TPckg<TInt> pckgRetValue(ret);
sl@0
   589
			aMessage.Write(2, pckgRetValue);
sl@0
   590
			TPckg<TStartupStateTimerEntry> pckgTimerEntry(timerEntry);
sl@0
   591
			aMessage.Write(3, pckgTimerEntry);
sl@0
   592
			break;
sl@0
   593
			}
sl@0
   594
sl@0
   595
		case EGetAccumulatedClientRequestsTimerResult:
sl@0
   596
			{
sl@0
   597
			TClientRequestTimerEntry timerEntry;
sl@0
   598
			TInt ret = EComPerformance::GetAccumulatedClientRequestTimerResult(aMessage.Int1(), timerEntry);
sl@0
   599
			TPckg<TInt> pckgRetValue(ret);
sl@0
   600
			aMessage.Write(2, pckgRetValue);
sl@0
   601
			TPckg<TClientRequestTimerEntry> pckgTimerEntry(timerEntry);
sl@0
   602
			aMessage.Write(3, pckgTimerEntry);
sl@0
   603
			break;
sl@0
   604
			}
sl@0
   605
		case EGetRegistryCounts:
sl@0
   606
			{
sl@0
   607
			RegistryCounts::TRegistryCounts counts;
sl@0
   608
			Server().GetRegistryCountsL(aMessage.Int1(), counts);
sl@0
   609
			TPckg<RegistryCounts::TRegistryCounts> pckgRegistryCounts(counts);
sl@0
   610
			aMessage.Write(2, pckgRegistryCounts);
sl@0
   611
			break;
sl@0
   612
			}
sl@0
   613
		case EResetStartupStateTimerCounts:
sl@0
   614
			{
sl@0
   615
			EComPerformance::ResetStartupStateTimerResult();
sl@0
   616
			break;
sl@0
   617
			}
sl@0
   618
		case EGetEComPerfTimeRecord:
sl@0
   619
			{
sl@0
   620
			TEComPerfTimeRecordEntry timerEntry;
sl@0
   621
			TInt ret = EComPerformance::GetEComPerfTimeRecord(aMessage.Int1(), timerEntry);
sl@0
   622
			TPckg<TInt> pckgRetValue(ret);
sl@0
   623
			aMessage.Write(2, pckgRetValue);
sl@0
   624
			TPckg<TEComPerfTimeRecordEntry> pckgTimerEntry(timerEntry);
sl@0
   625
			aMessage.Write(3, pckgTimerEntry);
sl@0
   626
			break;
sl@0
   627
			}
sl@0
   628
		case EResetEComPerfTimeRecords:
sl@0
   629
			{
sl@0
   630
			EComPerformance::ResetEComPerfTimeRecords();
sl@0
   631
			}
sl@0
   632
		case EGetEComServerHeapResult:
sl@0
   633
			{
sl@0
   634
			TEComPerfHeapUsage heapEntry;
sl@0
   635
			TInt ret= EComPerformance::GetEComHeapSize(aMessage.Int1(),heapEntry);
sl@0
   636
			TPckg<TInt> pckgRetValue(ret);
sl@0
   637
			aMessage.Write(2, pckgRetValue);
sl@0
   638
			TPckg<TEComPerfHeapUsage> pckgHeapEntry(heapEntry);
sl@0
   639
			aMessage.Write(3, pckgHeapEntry);
sl@0
   640
			break;			
sl@0
   641
			}
sl@0
   642
#endif
sl@0
   643
		default:
sl@0
   644
			break;
sl@0
   645
		}
sl@0
   646
	}
sl@0
   647
#endif
sl@0
   648
sl@0
   649
void CEComServerSession::CleanupInternalList()
sl@0
   650
	{
sl@0
   651
	if (iList != NULL)
sl@0
   652
		{
sl@0
   653
		iList->Reset();
sl@0
   654
		delete iList;
sl@0
   655
		iList = NULL;
sl@0
   656
		}
sl@0
   657
	}
sl@0
   658