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