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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // Server session object implementation.
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"
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.
34 void PanicClient(const TClientRequest& aMessage, TInt aPanicCode)
36 aMessage.Panic(KEComServerPanicCategory, aPanicCode);
40 // class CEComServerSession
42 CEComServerSession::CEComServerSession()
49 // 2nd phase construct for sessions - called by the CServer framework
51 void CEComServerSession::CreateL()
53 Server().AddSession();
56 CEComServerSession::~CEComServerSession()
58 CleanupInternalList();
59 CompleteNotifications(KErrCancel);
61 Server().DropSession();
65 // Deliver the notification message to the client
67 void CEComServerSession::CompleteNotifications(TInt aCompletionCode)
69 const TInt count = iNotificationRequests.Count();
70 for(TInt i = 0; i < count; ++i)
72 iNotificationRequests[i].iMessage.Complete(aCompletionCode);
75 iNotificationRequests.Reset();
79 // Handle a client request.
80 // Leaving is handled by CEComServer::RunError() which reports the error code
83 void CEComServerSession::ServiceL(const RMessage2& aMessage)
85 const TClientRequest msg(aMessage);
89 void CEComServerSession::ServiceL(const TClientRequest& aMessage)
91 TInt completionCode = KErrNone;
92 TBool asyncRequest = EFalse;
94 switch (aMessage.Function())
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)
107 if(iNotificationRequests[i].iRequestStatusHandle == notif.iRequestStatusHandle)
108 User::Leave(KErrArgument);
111 User::LeaveIfError(iNotificationRequests.Append(notif));
112 asyncRequest = ETrue;
113 RECORD_CLIENT_REQUEST_END_TIMER_RESULT;
117 case ECancelNotifyOnChange:
119 RECORD_CLIENT_REQUEST_START_TIMER_RESULT(EEComCancelNotifyOnChangeRequestType, Server().GetCurrentStartupState());
120 TInt statusHandle = aMessage.Int0();
122 const TInt count = iNotificationRequests.Count();
123 for(TInt i = 0; i < count; ++i)
125 if(iNotificationRequests[i].iRequestStatusHandle == statusHandle)
127 iNotificationRequests[i].iMessage.Complete(KErrCancel);
128 iNotificationRequests.Remove(i);
129 break; // Terminate the loop
133 RECORD_CLIENT_REQUEST_END_TIMER_RESULT;
136 case EListImplementations:
137 case EListResolvedImplementations:
138 case EListCustomResolvedImplementations:
139 if(Server().RegistryIndexValid())
141 RECORD_CLIENT_REQUEST_START_TIMER_RESULT(EEComListRequestType, Server().GetCurrentStartupState());
142 DoListImplementationsL(aMessage);
143 RECORD_CLIENT_REQUEST_END_TIMER_RESULT;
148 User::Leave(KEComErrListInvalidAwaitNotification);
150 User::Leave(KEComErrListCurrentlyUnavailable);
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;
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;
169 case EListExtendedInterfaces:
170 RECORD_CLIENT_REQUEST_START_TIMER_RESULT(EEComListExtendedInterfacesRequestType, Server().GetCurrentStartupState());
171 DoListExtendedInterfacesL(aMessage);
172 RECORD_CLIENT_REQUEST_END_TIMER_RESULT;
175 case EEnableImplementation:
176 case EDisableImplementation:
177 // Ommissions for 6.2
178 completionCode = KErrNotSupported;
180 #if defined(__ECOM_SERVER_TESTABILITY__) || defined(__ECOM_SERVER_PERFORMANCE__)
181 case ESetGetParameters:
182 DoSetGetParametersL(aMessage);
185 //EDestroyedImplementation obsolete due to implementation creation
186 //relocation to client side from server
187 case EDestroyedImplementation:
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;
196 aMessage.Complete(completionCode);
202 TInt const KDefaultStoreSize = 256; // Enough space for approx 1 implementation : will grow to fit if required
205 UnPack the match string and extended interface from the client supplied parameters.
207 @param aExtendedInterfaces Return value consisting of an array containing the extended interfaces.
208 @param aMatchStr Return value consisting of the matching string.
210 void CEComServerSession::UnpackMatchStrAndExtendedInterfacesFromClientL(const TClientRequest& aMessage,
211 RExtendedInterfacesArray& aExtendedInterfaces,
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();
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();
231 readStream.ReadL(aMatchStr,lenOfMatchStr);
233 TInt numOfExtendedInterfaces = readStream.ReadInt32L();
234 CleanupClosePushL(aExtendedInterfaces);
235 for(TInt i = 0; i < numOfExtendedInterfaces; i++)
237 aExtendedInterfaces.AppendL(TUid::Uid(readStream.ReadInt32L()));
240 CleanupStack::Pop(&aExtendedInterfaces);
241 CleanupStack::Pop(&aMatchStr);
242 CleanupStack::PopAndDestroy(&readStream);
243 CleanupStack::PopAndDestroy(&matchStrExtInfBuf);
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)
252 // Unpack the client supplied parameters
253 // Firstly get the uids
255 TPckg<TUidType> uidsPkg(uids);
256 aMessage.ReadL(KIPCParameterUids, uidsPkg);
258 if(uids[KInterfaceUidIndex] == KNullUid)
260 User::Leave(KEComErrMissingParameter);
263 //now get the TListImplParam parameters
264 TListImplParam listParam;
265 TPckg<TListImplParam> listParamPkg(listParam);
266 aMessage.ReadL(2,listParamPkg);
268 // Now rebuild the TEComResolverParams
269 TEComResolverParams resolverParameters;
270 resolverParameters.SetGenericMatch(listParam.iMatchType);
272 //now get the matchString and extendedInterfaces
274 RExtendedInterfacesArray extendedInterfaces;
275 UnpackMatchStrAndExtendedInterfacesFromClientL(aMessage,extendedInterfaces,matchStr);
276 matchStr.CleanupClosePushL();
277 CleanupClosePushL(extendedInterfaces);
279 if(matchStr.Length()>0)
281 resolverParameters.SetDataType(matchStr);
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.
287 // Pass to the server
288 iListContext = aMessage.Function();
291 case EListImplementations:
292 iList = Server().ListImplementationsL( uids[KInterfaceUidIndex], extendedInterfaces, aMessage );
294 case EListResolvedImplementations:
295 if(matchStr.Length() == 0)
297 User::Leave(KEComErrMissingParameter);
299 iList = Server().ListImplementationsL( uids[KInterfaceUidIndex], resolverParameters, extendedInterfaces, aMessage);
301 case EListCustomResolvedImplementations:
302 if(uids[KResolverUidIndex] == KNullUid)
304 User::Leave(KEComErrMissingParameter);
306 iList = Server().ListImplementationsL( uids[KInterfaceUidIndex], resolverParameters, uids[KResolverUidIndex], extendedInterfaces, aMessage);
311 //Now cleanup the extended interface
312 CleanupStack::PopAndDestroy(&extendedInterfaces);
313 CleanupStack::PopAndDestroy(&matchStr);
315 TInt bufferSizeRequired=0;
316 // Package the array for return
321 // Allocate a new store and replace the old one first
322 CBufFlat* memoryStore = CBufFlat::NewL(KDefaultStoreSize);
324 iMemoryStore = memoryStore;
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);
332 // Build the store data then calculate the end size;
333 const TInt entryCount = iList->Count();
334 writeStream.WriteInt32L(entryCount);
336 for(TInt i = 0; i < entryCount; ++i)
337 (*iList)[i]->ExternalizeL(ETrue,writeStream);
339 writeStream.CommitL();
341 // Set to actual size
342 bufferSizeRequired=iMemoryStore->Size();
343 __ECOM_TRACE1("ECOM ListImplementations request buffer size required=%d",bufferSizeRequired);
345 CleanupInternalList();
348 //if nothing is returned we should still indicate this to the client side
349 if (bufferSizeRequired==0)
351 //write back the bufferSize
352 listParam.iBufferSize=0;
353 aMessage.WriteL(2,listParamPkg);
357 //if the preallocated size is big enough to hold our entry
358 //copy it to the client
359 if (listParam.iBufferSize >= bufferSizeRequired)
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);
372 //if not rewrite back to the client the size that is required
373 //and signal with KErrOverFlow to the client
376 //write back the bufferSize
377 listParam.iBufferSize=bufferSizeRequired;
378 aMessage.WriteL(2,listParamPkg);
379 User::Leave(KErrOverflow);
383 TBool CEComServerSession::DoCollectListL(const TClientRequest& aMessage)
385 TBool success = EFalse;
388 TPtr8 data=iMemoryStore->Ptr(0);
389 aMessage.WriteL(0, data);
397 void CEComServerSession::DoGetResolvedImplementationL(const TClientRequest& aMessage)
399 // Unpack the client supplied parameters
400 // Firstly get the uids
402 TPckg<TUidType> uidsPkg(uids);
403 aMessage.ReadL(KIPCParameterUids, uidsPkg);
405 // Now rebuild the TEComResolverParams
406 TEComResolverParams resolverParameters;
407 resolverParameters.SetGenericMatch(KIPCParameterResolverParamsTypePtr);
409 //now get the matchString and extendedInterfaces
411 matchStr.CleanupClosePushL();
412 RExtendedInterfacesArray extendedInterfaces;
413 CleanupClosePushL(extendedInterfaces);
414 UnpackMatchStrAndExtendedInterfacesFromClientL(aMessage,extendedInterfaces,matchStr);
415 if(matchStr.Length()>0)
417 resolverParameters.SetDataType(matchStr);
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);
425 switch(aMessage.Function())
427 case EGetImplementationCreationMethod:
428 Server().GetResolvedDllInfoL(uids[KInterfaceUidIndex],
433 case EGetResolvedCreationMethod:
434 Server().GetResolvedDllInfoL(uids[KInterfaceUidIndex],
441 case EGetCustomResolvedCreationMethod:
442 Server().GetResolvedDllInfoL(uids[KInterfaceUidIndex],
444 uids[KResolverUidIndex],
453 CleanupStack::PopAndDestroy(&extendedInterfaces);
454 CleanupStack::PopAndDestroy(&matchStr);
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...
461 // Then re-package the results for return
462 // Firstly the Interface Implementation creation method pointer
463 TPckg<TEntry> result(loadInfo);
465 aMessage.WriteL(KIPCReturnParameterCreationMethodPtr, result);
466 // Next the destruction identification uid
467 TUidType type(KNullUid,dtorIdKey,KNullUid);
468 TPckg<TUidType> dtorIdKeyPkg(type);
470 aMessage.WriteL(KIPCReturnParameterUidsPtr, dtorIdKeyPkg);
473 // Return the list of interfaces to the client. If not enough space
474 // has been allocated by the client, KErrOverflow will be returned.
476 void CEComServerSession::DoListExtendedInterfacesL(const TClientRequest& aMessage)
478 // Unpack the client supplied parameters
480 TUid implementationUid(KNullUid);
481 TPckg<TUid> implementationUidDes(implementationUid);
482 TPckg<TInt> bufferSizeDes(bufferSize);
483 aMessage.ReadL(KIPCParameterImplementationUid,implementationUidDes);
484 aMessage.ReadL(KIPCParameterBufferSize,bufferSizeDes);
486 // Get the implementation information for this implementation UID.
487 CImplementationInformation* implInfo = NULL;
488 Server().GetImplementationInformationL(implementationUid,implInfo,aMessage);
490 TInt numExtendedInterfaces = 0; // Number of extended interfaces to return to client.
494 TInt bufferSizeRequired = 0; // Buffer required to send extended interface data back to client
496 // Fetch the list of extended interfaces
497 RExtendedInterfacesArray* extendedInterfaceList = implInfo->GetExtendedInterfaceList();
498 if (extendedInterfaceList != NULL)
500 numExtendedInterfaces = extendedInterfaceList->Count();
502 if (numExtendedInterfaces > 0)
504 bufferSizeRequired = numExtendedInterfaces * sizeof(TUid);
505 __ECOM_TRACE1("ECOM ListInterfaces request buffer size required=%d",bufferSizeRequired);
507 //if the preallocated size is big enough to hold our entry
508 //copy it to the client
509 if (bufferSize >= bufferSizeRequired)
512 CleanupClosePushL(buf);
513 buf.CreateL(bufferSizeRequired); // Create the RBuf.
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);
520 // Build the data of extendedInterfaces;
521 for(TInt i = 0; i < numExtendedInterfaces; ++i)
523 writeStream.WriteInt32L((*extendedInterfaceList)[i].iUid);
526 writeStream.CommitL();
528 // Copy the data to the client.
529 bufferSize=bufferSizeRequired;
530 aMessage.WriteL(KIPCParameterBufferSize,bufferSizeDes);
531 aMessage.WriteL(KIPCParameterInterfaceData,buf);
533 CleanupStack::PopAndDestroy(&buf);
535 //if not rewrite back to the client the size that is required
536 //and signal with KErrOverFlow to the client
539 bufferSize=bufferSizeRequired;
540 aMessage.WriteL(KIPCParameterBufferSize,bufferSizeDes);
541 User::Leave(KErrOverflow);
546 //if nothing is returned we should still indicate this to the client side
547 if (numExtendedInterfaces == 0)
550 aMessage.WriteL(KIPCParameterBufferSize,bufferSizeDes);
554 #if defined(__ECOM_SERVER_TESTABILITY__) || defined(__ECOM_SERVER_PERFORMANCE__)
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
560 void CEComServerSession::DoSetGetParametersL(const TClientRequest& aMessage)
562 TInt parameterType = aMessage.Int0();
564 switch(parameterType)
566 #ifdef __ECOM_SERVER_TESTABILITY__
567 case EChangeStartupState:
568 Server().ChangeStartupStateL(aMessage.Int1());
570 case EProcessStartupState:
571 Server().ProcessCurrentStartupStateL();
573 case EGetStartupState:
575 TInt state = Server().GetCurrentStartupState();
576 TPckg<TInt> pckgState(state);
577 aMessage.Write(1, pckgState);
581 #ifdef __ECOM_SERVER_PERFORMANCE__
582 case EGetStartupStateTimerResult:
584 TStartupStateTimerEntry timerEntry;
586 TInt ret = EComPerformance::GetStartupStateTimerResult(aMessage.Int1(), timerEntry.iTimerResult, timerEntry.iState);
588 TPckg<TInt> pckgRetValue(ret);
589 aMessage.Write(2, pckgRetValue);
590 TPckg<TStartupStateTimerEntry> pckgTimerEntry(timerEntry);
591 aMessage.Write(3, pckgTimerEntry);
595 case EGetAccumulatedClientRequestsTimerResult:
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);
605 case EGetRegistryCounts:
607 RegistryCounts::TRegistryCounts counts;
608 Server().GetRegistryCountsL(aMessage.Int1(), counts);
609 TPckg<RegistryCounts::TRegistryCounts> pckgRegistryCounts(counts);
610 aMessage.Write(2, pckgRegistryCounts);
613 case EResetStartupStateTimerCounts:
615 EComPerformance::ResetStartupStateTimerResult();
618 case EGetEComPerfTimeRecord:
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);
628 case EResetEComPerfTimeRecords:
630 EComPerformance::ResetEComPerfTimeRecords();
632 case EGetEComServerHeapResult:
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);
649 void CEComServerSession::CleanupInternalList()