sl@0: /* sl@0: * Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies). sl@0: * All rights reserved. sl@0: * This component and the accompanying materials are made available sl@0: * under the terms of the License "Eclipse Public License v1.0" sl@0: * which accompanies this distribution, and is available sl@0: * at the URL "http://www.eclipse.org/legal/epl-v10.html". sl@0: * sl@0: * Initial Contributors: sl@0: * Nokia Corporation - initial contribution. sl@0: * sl@0: * Contributors: sl@0: * sl@0: * Description: sl@0: * Server-side classes which are required to implement a session counting server. sl@0: * sl@0: */ sl@0: sl@0: sl@0: /** sl@0: @file sl@0: @publishedPartner sl@0: @released sl@0: */ sl@0: sl@0: #ifndef SCSSERVER_H sl@0: #define SCSSERVER_H sl@0: sl@0: #include sl@0: sl@0: #include sl@0: #include sl@0: sl@0: #ifndef SYMBIAN_ENABLE_SPLIT_HEADERS sl@0: #include "scsserverconstants.h" sl@0: #endif sl@0: sl@0: // -------- error handling -------- sl@0: sl@0: sl@0: const TInt KErrScsAsyncAlreadyQueued = -5702; ///< Tried to re-queue an outstanding request. sl@0: sl@0: sl@0: sl@0: // forward declarations sl@0: class CScsServer; sl@0: class CScsSession; sl@0: class CScsSubsession; sl@0: sl@0: class CShutdownTimer : public CTimer sl@0: /** sl@0: When an instance of this class is constructed it is given a delay period. sl@0: If the object runs to completion, i.e. if it is not cancelled, then it stops sl@0: the active scheduler, which has the effect of terminating the server. sl@0: */ sl@0: { sl@0: public: sl@0: static CShutdownTimer* NewL(TInt aDelayUs); sl@0: sl@0: void Restart(); sl@0: sl@0: void ImmediateTimeoutNextRestart(); sl@0: sl@0: private: sl@0: CShutdownTimer(TInt aDelayUs); sl@0: sl@0: // implement CTimer sl@0: virtual void RunL(); sl@0: sl@0: private: sl@0: TInt iDelayUs; ///< Timer delay in microseconds. sl@0: TBool iImmediateTimeoutNextRestart; sl@0: }; sl@0: sl@0: class CAsyncRequest : public CActive sl@0: /** sl@0: An outstanding request on the server side. A subclass should sl@0: be created for each type of outstanding request. sl@0: sl@0: This class is implemented as an active object, because it is sl@0: anticipated that some asynchronous requests can be implemented sl@0: simply by completing the client request when a TRequestStatus sl@0: completes. The implementor can override this behavior by sl@0: re-implementing RunL, e.g. to free some resources before completing sl@0: the request. If they do so, they must call CompleteAndMarkForDeletion to sl@0: complete the client request and mark this object for deletion (alternatively sl@0: they can simply call CAsyncRequest::RunL from their RunL). sl@0: sl@0: This class cannot, however, anticipate the cancellation sl@0: mechanism. The implementor must implement DoCancel for that. sl@0: DoCancel should not delete the client request or mark this object sl@0: for deletion. sl@0: sl@0: When the implementor has allocated an instance of this sl@0: class, they must add it to the session by calling CAsyncRequest::TransferToScsFrameworkL. sl@0: The second-phase constructor must not leave between that and calling sl@0: SetActive. sl@0: sl@0: These objects are cancelled and destroyed when the sl@0: client-side session / subsession sends a cancel command. sl@0: sl@0: @see CScsSession::AddAsyncRequestL sl@0: */ sl@0: { sl@0: public: sl@0: IMPORT_C CAsyncRequest(CScsSession* aSession, CScsSubsession* aSubsession, const RMessage2& aMessage); sl@0: sl@0: void CancelCompleteAndMarkForDeletion(); // explicit cancel request sl@0: void MarkForDeletion(); // session close sl@0: IMPORT_C virtual void DoCleanup(); // Cancel this request sl@0: sl@0: IMPORT_C void TransferToScsFrameworkL(); sl@0: protected: sl@0: sl@0: IMPORT_C void CompleteAndMarkForDeletion(TInt aError); sl@0: sl@0: // implement CActive sl@0: IMPORT_C virtual void RunL(); sl@0: sl@0: // override CActive sl@0: IMPORT_C virtual TInt RunError(TInt aError); sl@0: sl@0: public: sl@0: /** Session on which this request is queued. NULL if completed. */ sl@0: CScsSession* iSession; sl@0: /** Subsession on which this request is queued. NULL if session-relative. */ sl@0: CScsSubsession* iSubsession; sl@0: /** Outstanding message to complete. */ sl@0: const RMessagePtr2 iMessagePtr2; sl@0: /** Identifies outstanding request. */ sl@0: TInt iFunction; sl@0: }; sl@0: sl@0: class CScsSubsession : public CObject sl@0: /** sl@0: If the server implementation supports subsessions, they must sl@0: derive from this class. sl@0: sl@0: WARNING: Because this class derives from CObject you must not simply delete sl@0: an instance of a class derived from this class (ie. your subsession), instead you MUST sl@0: Close() it. sl@0: sl@0: In practise this probably means you need a ConstructL like this:- sl@0: CFred *self = new(ELeave) CFred(....); sl@0: CleanupClosePushL(*self); // Note use of CleanupClosePushL and of *self sl@0: self->ConstructL(...); sl@0: CleanupStack::Pop(self); // Note use of *self instead of self sl@0: */ sl@0: { sl@0: public: sl@0: /** sl@0: SCS routes subsession messages to this function; the session sl@0: object does not have to decode them itself. sl@0: sl@0: @param aFunction Implementation function, i.e. the function sl@0: identifier with the SCS field removed. sl@0: @param aMessage Standard server message object. sl@0: @return ETrue means complete client request now. sl@0: */ sl@0: virtual TBool DoServiceL(TInt aFunction, const RMessage2& aMessage) = 0; sl@0: sl@0: protected: sl@0: IMPORT_C CScsSubsession(CScsSession &aSession); sl@0: sl@0: public: sl@0: // Note this is setup by the constructor, before the derived class sl@0: // ConstructL is called. sl@0: CScsSession& iSession; ///< Owning session. sl@0: }; sl@0: sl@0: class CScsSession : public CSession2 sl@0: /** sl@0: When this session object is destroyed (because the client-server session sl@0: has closed,) this notifies the server object which decrements the session sl@0: count. Therefore, the server can be shut down after an inactivity sl@0: period during which there are no open sessions. sl@0: sl@0: This object also frees any remaining subsession objects when it is closed. sl@0: */ sl@0: { sl@0: protected: sl@0: IMPORT_C void ConstructL(); sl@0: IMPORT_C virtual ~CScsSession(); sl@0: sl@0: // implement CSession2 sl@0: IMPORT_C virtual void ServiceL(const RMessage2& aMessage); sl@0: // override CSession2 sl@0: IMPORT_C virtual void ServiceError(const RMessage2& aMessage, TInt aError); sl@0: IMPORT_C void CloseAllSubsessionsL(); sl@0: sl@0: // asynchronous requests sl@0: //IMPORT_C void CompleteAsyncRequest(CAsyncRequest* aAsyncRequest, TInt aReason); sl@0: sl@0: private: sl@0: /** sl@0: This function is called from ServiceL after it has removed the sl@0: SCS-specific field and handled any other messages. I.e., it is sl@0: called for functions which really require the session as opposed sl@0: to subsession commands or debugging commands such as heap failure. sl@0: sl@0: @param aFunction Implementation function, i.e. the function sl@0: identifier with the SCS field removed. sl@0: @param aMessage Standard server message object. sl@0: @return ETrue if client message should be completed now. sl@0: */ sl@0: virtual TBool DoServiceL(TInt aFunction, const RMessage2& aMessage) = 0; sl@0: sl@0: void PreCloseSession(); sl@0: sl@0: IMPORT_C virtual CScsSubsession* DoCreateSubsessionL(TInt aFunction, const RMessage2& aMessage); sl@0: sl@0: // subessions sl@0: CScsSubsession* GetSubsessionL(const RMessage2& aMessage); sl@0: void DeleteSubsessionContainers(); sl@0: sl@0: void CreateSubsessionL(TInt aFunction, const RMessage2& aMessage); sl@0: void CloseSubsessionL(const RMessage2& aMessage); sl@0: TBool CallSubsessionFunctionL(TInt aFunction, const RMessage2& aMessage); sl@0: void CancelAsyncSubsessionRequestL(TInt aFunction, const RMessage2& aMessage); sl@0: sl@0: // asynchronous requests (support) sl@0: CAsyncRequest* FindSessionAsyncRequest(TInt aFunction); sl@0: void CancelAsyncSessionRequestL(TInt aFunction); sl@0: sl@0: private: sl@0: /** sl@0: Number of open subsessions before a new one is created. This is used sl@0: to manage cleanup if the subsession cannot be created. sl@0: */ sl@0: TInt iPreCreateSubsessionCount; sl@0: sl@0: CObjectCon* iSsObjects; ///< Currently open subsessions. sl@0: CObjectIx* iSsHandles; ///< Maps handles to open subsessions. sl@0: sl@0: public: sl@0: IMPORT_C CScsSession(CScsServer &aServer); sl@0: // This duplicates the iServer/Server() code in the base class, sl@0: // BUT this variable IS setup before derived class ConstructL is sl@0: // called and is NOT const. This trades off 4 bytes of memory sl@0: // against making the code cleaner and more consistent.... sl@0: CScsServer& iServer; ///< Owning server. sl@0: }; sl@0: sl@0: /** sl@0: Pass to CScsServer::CancelOutstandingRequests to mean all requests sl@0: associated with a session or its subsessions. sl@0: */ sl@0: CScsSubsession*const KWildSubsession = reinterpret_cast(~0); sl@0: sl@0: class CScsServer : public CPolicyServer sl@0: /** sl@0: The main server object allocates sessions. It also uses sl@0: a shutdown timer to stop the server when there have been no sl@0: open sessions for a set period. sl@0: sl@0: The implementor must define a subclass which returns session sl@0: objects of the appropriate type. sl@0: */ sl@0: { sl@0: public: sl@0: IMPORT_C virtual ~CScsServer(); sl@0: sl@0: void IncrementSessionCount(); sl@0: void DecrementSessionCount(); sl@0: inline TInt SessionCount() const; sl@0: sl@0: IMPORT_C void ShutdownWhenIdleL(); sl@0: sl@0: // asynchronous requests sl@0: void AddAsyncRequestL(CAsyncRequest* aAsyncRequest); sl@0: void CancelAsyncRequest( sl@0: CScsSession* aSession, CScsSubsession* aSubsession, TInt aFunction); sl@0: sl@0: void CancelOutstandingRequests(CScsSession* aSession, TBool aCompleteClientRequests); sl@0: void CancelOutstandingRequests(CScsSession* aSession, CScsSubsession* aSubsession, TBool aCompleteClientRequests); sl@0: sl@0: /** sl@0: This function is called just before the SCS framework marks the heap for OOM sl@0: testing and just before checking the heap. sl@0: sl@0: Typically this function should compact any arrays and free objects which change size and can sl@0: not be compacted back to the same level. sl@0: */ sl@0: IMPORT_C virtual void DoPreHeapMarkOrCheckL(); sl@0: sl@0: /** sl@0: This function is called just after the heap has either been marked or checked (which is just sl@0: after DoPreHeapMarkOrCheck). sl@0: sl@0: Typically this function should re-create any objects which had to be freed by DoPreHeapMarkOrCheck. sl@0: */ sl@0: IMPORT_C virtual void DoPostHeapMarkOrCheckL(); sl@0: sl@0: enum TFunctionRanges sl@0: /** sl@0: Function ranges to be used when configuring the server security via sl@0: a CPolicyServer::TPolicy object. sl@0: sl@0: Session/sSubsession function codes will be ORed into the sl@0: EBaseSession/EBaseSubSession ranges. sl@0: sl@0: Values from EBaseMustAllow, upwards, are used internally and must be allowed. sl@0: sl@0: If there are multiple subsession types (with different function sl@0: code values), then codes must be be different for each subsession sl@0: (this restriction only applies if using the CPolicyServer::TPolicy sl@0: mechanism to configure server security). sl@0: sl@0: WARNNIG: These ranges MUST match the values in scscommon.h TScsFunction. sl@0: */ sl@0: { sl@0: EBaseSession = 0x01000000, sl@0: EBaseSubSession = 0x02000000, sl@0: EBaseMustAllow = 0x03000000 //< Must allow from EBaseMustAllow upwards sl@0: }; sl@0: sl@0: static TInt StripScsFunctionMask(TInt aFunctionId) { return aFunctionId & ~ScsImpl::KScsFunctionMask;} sl@0: sl@0: protected: sl@0: IMPORT_C CScsServer(const TVersion& aVersion, CActive::TPriority aPriority = CActive::EPriorityStandard); sl@0: IMPORT_C CScsServer(const TVersion& aVersion, const CPolicyServer::TPolicy& aPolicy, CActive::TPriority aPriority = CActive::EPriorityStandard); sl@0: IMPORT_C void ConstructL(TInt aShutdownPeriodUs); sl@0: sl@0: IMPORT_C void DisableShutdownTimer(); sl@0: IMPORT_C void EnableShutdownTimerL(TInt aShutdownPeriodUs); sl@0: sl@0: sl@0: // implement CServer2 sl@0: IMPORT_C virtual CSession2* NewSessionL(const TVersion& aVersion, const RMessage2& aMessage) const; sl@0: sl@0: /** sl@0: NewSessionL checks that this server supports the version which is sl@0: requested by the client. If that is so, then it calls this function sl@0: to allocate a new session object. sl@0: sl@0: @param aMessage Connection message, as passed to NewSessionL. sl@0: @return New session object. This is owned by the kernel, sl@0: which will delete it when the session is closed. sl@0: */ sl@0: virtual CScsSession* DoNewSessionL(const RMessage2& aMessage) = 0; sl@0: sl@0: private: sl@0: // asynchronous requests sl@0: CAsyncRequest* FindAsyncRequest( sl@0: CScsSession* aSession, CScsSubsession* aSubsession, TInt aFunction); sl@0: sl@0: static TInt RemoveCompletedRequests(TAny* aPtr); sl@0: void RemoveCompletedRequests(); sl@0: sl@0: private: sl@0: /** sl@0: This server's version. It is compared against client's requested version sl@0: when the client attempts to make a connection. sl@0: */ sl@0: const TVersion iVersion; sl@0: sl@0: /** Number of open sessions. Used to start and cancel the shutdown timer. */ sl@0: TInt iSessionCount; sl@0: sl@0: /** sl@0: Shutdown timer, started when there are no open sessions, cancelled when sl@0: the first session is created. sl@0: */ sl@0: CShutdownTimer* iShutdownTimer; sl@0: sl@0: sl@0: public: sl@0: /** sl@0: Generates instances of CObjectCon, for each session to host its subsessions. sl@0: Public because must be accessible to session objects. sl@0: */ sl@0: CObjectConIx* iContainerIndex; sl@0: sl@0: /** Currently outstanding requests. */ sl@0: RPointerArray iAsyncRequests; sl@0: sl@0: /** Runs to remove completed requests from iAsyncRequests. */ sl@0: CAsyncCallBack* iAsyncCleanup; sl@0: }; sl@0: sl@0: inline TInt CScsServer::SessionCount() const sl@0: { sl@0: return iSessionCount; sl@0: } sl@0: sl@0: sl@0: sl@0: // -------- startup -------- sl@0: sl@0: /** sl@0: The server executable must implement a factory function with this sl@0: signature and pass it to StartScsServer to allocate and start the server. sl@0: */ sl@0: typedef CScsServer* (*TScsServerFactory)(); sl@0: sl@0: IMPORT_C TInt StartScsServer(TScsServerFactory aServerFactoryLC); sl@0: sl@0: sl@0: #endif // #ifndef SCSSERVER_H sl@0: