Update contrib.
2 * Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies).
4 * This component and the accompanying materials are made available
5 * under the terms of the License "Eclipse Public License v1.0"
6 * which accompanies this distribution, and is available
7 * at the URL "http://www.eclipse.org/legal/epl-v10.html".
9 * Initial Contributors:
10 * Nokia Corporation - initial contribution.
15 * Server-side classes which are required to implement a session counting server.
31 #include <scs/scsclient.h>
32 #include <scs/scscommon.h>
34 #ifndef SYMBIAN_ENABLE_SPLIT_HEADERS
35 #include "scsserverconstants.h"
38 // -------- error handling --------
41 const TInt KErrScsAsyncAlreadyQueued = -5702; ///< Tried to re-queue an outstanding request.
45 // forward declarations
50 class CShutdownTimer : public CTimer
52 When an instance of this class is constructed it is given a delay period.
53 If the object runs to completion, i.e. if it is not cancelled, then it stops
54 the active scheduler, which has the effect of terminating the server.
58 static CShutdownTimer* NewL(TInt aDelayUs);
62 void ImmediateTimeoutNextRestart();
65 CShutdownTimer(TInt aDelayUs);
71 TInt iDelayUs; ///< Timer delay in microseconds.
72 TBool iImmediateTimeoutNextRestart;
75 class CAsyncRequest : public CActive
77 An outstanding request on the server side. A subclass should
78 be created for each type of outstanding request.
80 This class is implemented as an active object, because it is
81 anticipated that some asynchronous requests can be implemented
82 simply by completing the client request when a TRequestStatus
83 completes. The implementor can override this behavior by
84 re-implementing RunL, e.g. to free some resources before completing
85 the request. If they do so, they must call CompleteAndMarkForDeletion to
86 complete the client request and mark this object for deletion (alternatively
87 they can simply call CAsyncRequest::RunL from their RunL).
89 This class cannot, however, anticipate the cancellation
90 mechanism. The implementor must implement DoCancel for that.
91 DoCancel should not delete the client request or mark this object
94 When the implementor has allocated an instance of this
95 class, they must add it to the session by calling CAsyncRequest::TransferToScsFrameworkL.
96 The second-phase constructor must not leave between that and calling
99 These objects are cancelled and destroyed when the
100 client-side session / subsession sends a cancel command.
102 @see CScsSession::AddAsyncRequestL
106 IMPORT_C CAsyncRequest(CScsSession* aSession, CScsSubsession* aSubsession, const RMessage2& aMessage);
108 void CancelCompleteAndMarkForDeletion(); // explicit cancel request
109 void MarkForDeletion(); // session close
110 IMPORT_C virtual void DoCleanup(); // Cancel this request
112 IMPORT_C void TransferToScsFrameworkL();
115 IMPORT_C void CompleteAndMarkForDeletion(TInt aError);
118 IMPORT_C virtual void RunL();
121 IMPORT_C virtual TInt RunError(TInt aError);
124 /** Session on which this request is queued. NULL if completed. */
125 CScsSession* iSession;
126 /** Subsession on which this request is queued. NULL if session-relative. */
127 CScsSubsession* iSubsession;
128 /** Outstanding message to complete. */
129 const RMessagePtr2 iMessagePtr2;
130 /** Identifies outstanding request. */
134 class CScsSubsession : public CObject
136 If the server implementation supports subsessions, they must
137 derive from this class.
139 WARNING: Because this class derives from CObject you must not simply delete
140 an instance of a class derived from this class (ie. your subsession), instead you MUST
143 In practise this probably means you need a ConstructL like this:-
144 CFred *self = new(ELeave) CFred(....);
145 CleanupClosePushL(*self); // Note use of CleanupClosePushL and of *self
146 self->ConstructL(...);
147 CleanupStack::Pop(self); // Note use of *self instead of self
152 SCS routes subsession messages to this function; the session
153 object does not have to decode them itself.
155 @param aFunction Implementation function, i.e. the function
156 identifier with the SCS field removed.
157 @param aMessage Standard server message object.
158 @return ETrue means complete client request now.
160 virtual TBool DoServiceL(TInt aFunction, const RMessage2& aMessage) = 0;
163 IMPORT_C CScsSubsession(CScsSession &aSession);
166 // Note this is setup by the constructor, before the derived class
167 // ConstructL is called.
168 CScsSession& iSession; ///< Owning session.
171 class CScsSession : public CSession2
173 When this session object is destroyed (because the client-server session
174 has closed,) this notifies the server object which decrements the session
175 count. Therefore, the server can be shut down after an inactivity
176 period during which there are no open sessions.
178 This object also frees any remaining subsession objects when it is closed.
182 IMPORT_C void ConstructL();
183 IMPORT_C virtual ~CScsSession();
185 // implement CSession2
186 IMPORT_C virtual void ServiceL(const RMessage2& aMessage);
187 // override CSession2
188 IMPORT_C virtual void ServiceError(const RMessage2& aMessage, TInt aError);
189 IMPORT_C void CloseAllSubsessionsL();
191 // asynchronous requests
192 //IMPORT_C void CompleteAsyncRequest(CAsyncRequest* aAsyncRequest, TInt aReason);
196 This function is called from ServiceL after it has removed the
197 SCS-specific field and handled any other messages. I.e., it is
198 called for functions which really require the session as opposed
199 to subsession commands or debugging commands such as heap failure.
201 @param aFunction Implementation function, i.e. the function
202 identifier with the SCS field removed.
203 @param aMessage Standard server message object.
204 @return ETrue if client message should be completed now.
206 virtual TBool DoServiceL(TInt aFunction, const RMessage2& aMessage) = 0;
208 void PreCloseSession();
210 IMPORT_C virtual CScsSubsession* DoCreateSubsessionL(TInt aFunction, const RMessage2& aMessage);
213 CScsSubsession* GetSubsessionL(const RMessage2& aMessage);
214 void DeleteSubsessionContainers();
216 void CreateSubsessionL(TInt aFunction, const RMessage2& aMessage);
217 void CloseSubsessionL(const RMessage2& aMessage);
218 TBool CallSubsessionFunctionL(TInt aFunction, const RMessage2& aMessage);
219 void CancelAsyncSubsessionRequestL(TInt aFunction, const RMessage2& aMessage);
221 // asynchronous requests (support)
222 CAsyncRequest* FindSessionAsyncRequest(TInt aFunction);
223 void CancelAsyncSessionRequestL(TInt aFunction);
227 Number of open subsessions before a new one is created. This is used
228 to manage cleanup if the subsession cannot be created.
230 TInt iPreCreateSubsessionCount;
232 CObjectCon* iSsObjects; ///< Currently open subsessions.
233 CObjectIx* iSsHandles; ///< Maps handles to open subsessions.
236 IMPORT_C CScsSession(CScsServer &aServer);
237 // This duplicates the iServer/Server() code in the base class,
238 // BUT this variable IS setup before derived class ConstructL is
239 // called and is NOT const. This trades off 4 bytes of memory
240 // against making the code cleaner and more consistent....
241 CScsServer& iServer; ///< Owning server.
245 Pass to CScsServer::CancelOutstandingRequests to mean all requests
246 associated with a session or its subsessions.
248 CScsSubsession*const KWildSubsession = reinterpret_cast<CScsSubsession*>(~0);
250 class CScsServer : public CPolicyServer
252 The main server object allocates sessions. It also uses
253 a shutdown timer to stop the server when there have been no
254 open sessions for a set period.
256 The implementor must define a subclass which returns session
257 objects of the appropriate type.
261 IMPORT_C virtual ~CScsServer();
263 void IncrementSessionCount();
264 void DecrementSessionCount();
265 inline TInt SessionCount() const;
267 IMPORT_C void ShutdownWhenIdleL();
269 // asynchronous requests
270 void AddAsyncRequestL(CAsyncRequest* aAsyncRequest);
271 void CancelAsyncRequest(
272 CScsSession* aSession, CScsSubsession* aSubsession, TInt aFunction);
274 void CancelOutstandingRequests(CScsSession* aSession, TBool aCompleteClientRequests);
275 void CancelOutstandingRequests(CScsSession* aSession, CScsSubsession* aSubsession, TBool aCompleteClientRequests);
278 This function is called just before the SCS framework marks the heap for OOM
279 testing and just before checking the heap.
281 Typically this function should compact any arrays and free objects which change size and can
282 not be compacted back to the same level.
284 IMPORT_C virtual void DoPreHeapMarkOrCheckL();
287 This function is called just after the heap has either been marked or checked (which is just
288 after DoPreHeapMarkOrCheck).
290 Typically this function should re-create any objects which had to be freed by DoPreHeapMarkOrCheck.
292 IMPORT_C virtual void DoPostHeapMarkOrCheckL();
296 Function ranges to be used when configuring the server security via
297 a CPolicyServer::TPolicy object.
299 Session/sSubsession function codes will be ORed into the
300 EBaseSession/EBaseSubSession ranges.
302 Values from EBaseMustAllow, upwards, are used internally and must be allowed.
304 If there are multiple subsession types (with different function
305 code values), then codes must be be different for each subsession
306 (this restriction only applies if using the CPolicyServer::TPolicy
307 mechanism to configure server security).
309 WARNNIG: These ranges MUST match the values in scscommon.h TScsFunction.
312 EBaseSession = 0x01000000,
313 EBaseSubSession = 0x02000000,
314 EBaseMustAllow = 0x03000000 //< Must allow from EBaseMustAllow upwards
317 static TInt StripScsFunctionMask(TInt aFunctionId) { return aFunctionId & ~ScsImpl::KScsFunctionMask;}
320 IMPORT_C CScsServer(const TVersion& aVersion, CActive::TPriority aPriority = CActive::EPriorityStandard);
321 IMPORT_C CScsServer(const TVersion& aVersion, const CPolicyServer::TPolicy& aPolicy, CActive::TPriority aPriority = CActive::EPriorityStandard);
322 IMPORT_C void ConstructL(TInt aShutdownPeriodUs);
324 IMPORT_C void DisableShutdownTimer();
325 IMPORT_C void EnableShutdownTimerL(TInt aShutdownPeriodUs);
328 // implement CServer2
329 IMPORT_C virtual CSession2* NewSessionL(const TVersion& aVersion, const RMessage2& aMessage) const;
332 NewSessionL checks that this server supports the version which is
333 requested by the client. If that is so, then it calls this function
334 to allocate a new session object.
336 @param aMessage Connection message, as passed to NewSessionL.
337 @return New session object. This is owned by the kernel,
338 which will delete it when the session is closed.
340 virtual CScsSession* DoNewSessionL(const RMessage2& aMessage) = 0;
343 // asynchronous requests
344 CAsyncRequest* FindAsyncRequest(
345 CScsSession* aSession, CScsSubsession* aSubsession, TInt aFunction);
347 static TInt RemoveCompletedRequests(TAny* aPtr);
348 void RemoveCompletedRequests();
352 This server's version. It is compared against client's requested version
353 when the client attempts to make a connection.
355 const TVersion iVersion;
357 /** Number of open sessions. Used to start and cancel the shutdown timer. */
361 Shutdown timer, started when there are no open sessions, cancelled when
362 the first session is created.
364 CShutdownTimer* iShutdownTimer;
369 Generates instances of CObjectCon, for each session to host its subsessions.
370 Public because must be accessible to session objects.
372 CObjectConIx* iContainerIndex;
374 /** Currently outstanding requests. */
375 RPointerArray<CAsyncRequest> iAsyncRequests;
377 /** Runs to remove completed requests from iAsyncRequests. */
378 CAsyncCallBack* iAsyncCleanup;
381 inline TInt CScsServer::SessionCount() const
383 return iSessionCount;
388 // -------- startup --------
391 The server executable must implement a factory function with this
392 signature and pass it to StartScsServer to allocate and start the server.
394 typedef CScsServer* (*TScsServerFactory)();
396 IMPORT_C TInt StartScsServer(TScsServerFactory aServerFactoryLC);
399 #endif // #ifndef SCSSERVER_H