sl@0
|
1 |
/*
|
sl@0
|
2 |
* Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies).
|
sl@0
|
3 |
* All rights reserved.
|
sl@0
|
4 |
* This component and the accompanying materials are made available
|
sl@0
|
5 |
* under the terms of the License "Eclipse Public License v1.0"
|
sl@0
|
6 |
* which accompanies this distribution, and is available
|
sl@0
|
7 |
* at the URL "http://www.eclipse.org/legal/epl-v10.html".
|
sl@0
|
8 |
*
|
sl@0
|
9 |
* Initial Contributors:
|
sl@0
|
10 |
* Nokia Corporation - initial contribution.
|
sl@0
|
11 |
*
|
sl@0
|
12 |
* Contributors:
|
sl@0
|
13 |
*
|
sl@0
|
14 |
* Description:
|
sl@0
|
15 |
* Server-side classes which are required to implement a session counting server.
|
sl@0
|
16 |
*
|
sl@0
|
17 |
*/
|
sl@0
|
18 |
|
sl@0
|
19 |
|
sl@0
|
20 |
/**
|
sl@0
|
21 |
@file
|
sl@0
|
22 |
@publishedPartner
|
sl@0
|
23 |
@released
|
sl@0
|
24 |
*/
|
sl@0
|
25 |
|
sl@0
|
26 |
#ifndef SCSSERVER_H
|
sl@0
|
27 |
#define SCSSERVER_H
|
sl@0
|
28 |
|
sl@0
|
29 |
#include <e32base.h>
|
sl@0
|
30 |
|
sl@0
|
31 |
#include <scs/scsclient.h>
|
sl@0
|
32 |
#include <scs/scscommon.h>
|
sl@0
|
33 |
|
sl@0
|
34 |
#ifndef SYMBIAN_ENABLE_SPLIT_HEADERS
|
sl@0
|
35 |
#include "scsserverconstants.h"
|
sl@0
|
36 |
#endif
|
sl@0
|
37 |
|
sl@0
|
38 |
// -------- error handling --------
|
sl@0
|
39 |
|
sl@0
|
40 |
|
sl@0
|
41 |
const TInt KErrScsAsyncAlreadyQueued = -5702; ///< Tried to re-queue an outstanding request.
|
sl@0
|
42 |
|
sl@0
|
43 |
|
sl@0
|
44 |
|
sl@0
|
45 |
// forward declarations
|
sl@0
|
46 |
class CScsServer;
|
sl@0
|
47 |
class CScsSession;
|
sl@0
|
48 |
class CScsSubsession;
|
sl@0
|
49 |
|
sl@0
|
50 |
class CShutdownTimer : public CTimer
|
sl@0
|
51 |
/**
|
sl@0
|
52 |
When an instance of this class is constructed it is given a delay period.
|
sl@0
|
53 |
If the object runs to completion, i.e. if it is not cancelled, then it stops
|
sl@0
|
54 |
the active scheduler, which has the effect of terminating the server.
|
sl@0
|
55 |
*/
|
sl@0
|
56 |
{
|
sl@0
|
57 |
public:
|
sl@0
|
58 |
static CShutdownTimer* NewL(TInt aDelayUs);
|
sl@0
|
59 |
|
sl@0
|
60 |
void Restart();
|
sl@0
|
61 |
|
sl@0
|
62 |
void ImmediateTimeoutNextRestart();
|
sl@0
|
63 |
|
sl@0
|
64 |
private:
|
sl@0
|
65 |
CShutdownTimer(TInt aDelayUs);
|
sl@0
|
66 |
|
sl@0
|
67 |
// implement CTimer
|
sl@0
|
68 |
virtual void RunL();
|
sl@0
|
69 |
|
sl@0
|
70 |
private:
|
sl@0
|
71 |
TInt iDelayUs; ///< Timer delay in microseconds.
|
sl@0
|
72 |
TBool iImmediateTimeoutNextRestart;
|
sl@0
|
73 |
};
|
sl@0
|
74 |
|
sl@0
|
75 |
class CAsyncRequest : public CActive
|
sl@0
|
76 |
/**
|
sl@0
|
77 |
An outstanding request on the server side. A subclass should
|
sl@0
|
78 |
be created for each type of outstanding request.
|
sl@0
|
79 |
|
sl@0
|
80 |
This class is implemented as an active object, because it is
|
sl@0
|
81 |
anticipated that some asynchronous requests can be implemented
|
sl@0
|
82 |
simply by completing the client request when a TRequestStatus
|
sl@0
|
83 |
completes. The implementor can override this behavior by
|
sl@0
|
84 |
re-implementing RunL, e.g. to free some resources before completing
|
sl@0
|
85 |
the request. If they do so, they must call CompleteAndMarkForDeletion to
|
sl@0
|
86 |
complete the client request and mark this object for deletion (alternatively
|
sl@0
|
87 |
they can simply call CAsyncRequest::RunL from their RunL).
|
sl@0
|
88 |
|
sl@0
|
89 |
This class cannot, however, anticipate the cancellation
|
sl@0
|
90 |
mechanism. The implementor must implement DoCancel for that.
|
sl@0
|
91 |
DoCancel should not delete the client request or mark this object
|
sl@0
|
92 |
for deletion.
|
sl@0
|
93 |
|
sl@0
|
94 |
When the implementor has allocated an instance of this
|
sl@0
|
95 |
class, they must add it to the session by calling CAsyncRequest::TransferToScsFrameworkL.
|
sl@0
|
96 |
The second-phase constructor must not leave between that and calling
|
sl@0
|
97 |
SetActive.
|
sl@0
|
98 |
|
sl@0
|
99 |
These objects are cancelled and destroyed when the
|
sl@0
|
100 |
client-side session / subsession sends a cancel command.
|
sl@0
|
101 |
|
sl@0
|
102 |
@see CScsSession::AddAsyncRequestL
|
sl@0
|
103 |
*/
|
sl@0
|
104 |
{
|
sl@0
|
105 |
public:
|
sl@0
|
106 |
IMPORT_C CAsyncRequest(CScsSession* aSession, CScsSubsession* aSubsession, const RMessage2& aMessage);
|
sl@0
|
107 |
|
sl@0
|
108 |
void CancelCompleteAndMarkForDeletion(); // explicit cancel request
|
sl@0
|
109 |
void MarkForDeletion(); // session close
|
sl@0
|
110 |
IMPORT_C virtual void DoCleanup(); // Cancel this request
|
sl@0
|
111 |
|
sl@0
|
112 |
IMPORT_C void TransferToScsFrameworkL();
|
sl@0
|
113 |
protected:
|
sl@0
|
114 |
|
sl@0
|
115 |
IMPORT_C void CompleteAndMarkForDeletion(TInt aError);
|
sl@0
|
116 |
|
sl@0
|
117 |
// implement CActive
|
sl@0
|
118 |
IMPORT_C virtual void RunL();
|
sl@0
|
119 |
|
sl@0
|
120 |
// override CActive
|
sl@0
|
121 |
IMPORT_C virtual TInt RunError(TInt aError);
|
sl@0
|
122 |
|
sl@0
|
123 |
public:
|
sl@0
|
124 |
/** Session on which this request is queued. NULL if completed. */
|
sl@0
|
125 |
CScsSession* iSession;
|
sl@0
|
126 |
/** Subsession on which this request is queued. NULL if session-relative. */
|
sl@0
|
127 |
CScsSubsession* iSubsession;
|
sl@0
|
128 |
/** Outstanding message to complete. */
|
sl@0
|
129 |
const RMessagePtr2 iMessagePtr2;
|
sl@0
|
130 |
/** Identifies outstanding request. */
|
sl@0
|
131 |
TInt iFunction;
|
sl@0
|
132 |
};
|
sl@0
|
133 |
|
sl@0
|
134 |
class CScsSubsession : public CObject
|
sl@0
|
135 |
/**
|
sl@0
|
136 |
If the server implementation supports subsessions, they must
|
sl@0
|
137 |
derive from this class.
|
sl@0
|
138 |
|
sl@0
|
139 |
WARNING: Because this class derives from CObject you must not simply delete
|
sl@0
|
140 |
an instance of a class derived from this class (ie. your subsession), instead you MUST
|
sl@0
|
141 |
Close() it.
|
sl@0
|
142 |
|
sl@0
|
143 |
In practise this probably means you need a ConstructL like this:-
|
sl@0
|
144 |
CFred *self = new(ELeave) CFred(....);
|
sl@0
|
145 |
CleanupClosePushL(*self); // Note use of CleanupClosePushL and of *self
|
sl@0
|
146 |
self->ConstructL(...);
|
sl@0
|
147 |
CleanupStack::Pop(self); // Note use of *self instead of self
|
sl@0
|
148 |
*/
|
sl@0
|
149 |
{
|
sl@0
|
150 |
public:
|
sl@0
|
151 |
/**
|
sl@0
|
152 |
SCS routes subsession messages to this function; the session
|
sl@0
|
153 |
object does not have to decode them itself.
|
sl@0
|
154 |
|
sl@0
|
155 |
@param aFunction Implementation function, i.e. the function
|
sl@0
|
156 |
identifier with the SCS field removed.
|
sl@0
|
157 |
@param aMessage Standard server message object.
|
sl@0
|
158 |
@return ETrue means complete client request now.
|
sl@0
|
159 |
*/
|
sl@0
|
160 |
virtual TBool DoServiceL(TInt aFunction, const RMessage2& aMessage) = 0;
|
sl@0
|
161 |
|
sl@0
|
162 |
protected:
|
sl@0
|
163 |
IMPORT_C CScsSubsession(CScsSession &aSession);
|
sl@0
|
164 |
|
sl@0
|
165 |
public:
|
sl@0
|
166 |
// Note this is setup by the constructor, before the derived class
|
sl@0
|
167 |
// ConstructL is called.
|
sl@0
|
168 |
CScsSession& iSession; ///< Owning session.
|
sl@0
|
169 |
};
|
sl@0
|
170 |
|
sl@0
|
171 |
class CScsSession : public CSession2
|
sl@0
|
172 |
/**
|
sl@0
|
173 |
When this session object is destroyed (because the client-server session
|
sl@0
|
174 |
has closed,) this notifies the server object which decrements the session
|
sl@0
|
175 |
count. Therefore, the server can be shut down after an inactivity
|
sl@0
|
176 |
period during which there are no open sessions.
|
sl@0
|
177 |
|
sl@0
|
178 |
This object also frees any remaining subsession objects when it is closed.
|
sl@0
|
179 |
*/
|
sl@0
|
180 |
{
|
sl@0
|
181 |
protected:
|
sl@0
|
182 |
IMPORT_C void ConstructL();
|
sl@0
|
183 |
IMPORT_C virtual ~CScsSession();
|
sl@0
|
184 |
|
sl@0
|
185 |
// implement CSession2
|
sl@0
|
186 |
IMPORT_C virtual void ServiceL(const RMessage2& aMessage);
|
sl@0
|
187 |
// override CSession2
|
sl@0
|
188 |
IMPORT_C virtual void ServiceError(const RMessage2& aMessage, TInt aError);
|
sl@0
|
189 |
IMPORT_C void CloseAllSubsessionsL();
|
sl@0
|
190 |
|
sl@0
|
191 |
// asynchronous requests
|
sl@0
|
192 |
//IMPORT_C void CompleteAsyncRequest(CAsyncRequest* aAsyncRequest, TInt aReason);
|
sl@0
|
193 |
|
sl@0
|
194 |
private:
|
sl@0
|
195 |
/**
|
sl@0
|
196 |
This function is called from ServiceL after it has removed the
|
sl@0
|
197 |
SCS-specific field and handled any other messages. I.e., it is
|
sl@0
|
198 |
called for functions which really require the session as opposed
|
sl@0
|
199 |
to subsession commands or debugging commands such as heap failure.
|
sl@0
|
200 |
|
sl@0
|
201 |
@param aFunction Implementation function, i.e. the function
|
sl@0
|
202 |
identifier with the SCS field removed.
|
sl@0
|
203 |
@param aMessage Standard server message object.
|
sl@0
|
204 |
@return ETrue if client message should be completed now.
|
sl@0
|
205 |
*/
|
sl@0
|
206 |
virtual TBool DoServiceL(TInt aFunction, const RMessage2& aMessage) = 0;
|
sl@0
|
207 |
|
sl@0
|
208 |
void PreCloseSession();
|
sl@0
|
209 |
|
sl@0
|
210 |
IMPORT_C virtual CScsSubsession* DoCreateSubsessionL(TInt aFunction, const RMessage2& aMessage);
|
sl@0
|
211 |
|
sl@0
|
212 |
// subessions
|
sl@0
|
213 |
CScsSubsession* GetSubsessionL(const RMessage2& aMessage);
|
sl@0
|
214 |
void DeleteSubsessionContainers();
|
sl@0
|
215 |
|
sl@0
|
216 |
void CreateSubsessionL(TInt aFunction, const RMessage2& aMessage);
|
sl@0
|
217 |
void CloseSubsessionL(const RMessage2& aMessage);
|
sl@0
|
218 |
TBool CallSubsessionFunctionL(TInt aFunction, const RMessage2& aMessage);
|
sl@0
|
219 |
void CancelAsyncSubsessionRequestL(TInt aFunction, const RMessage2& aMessage);
|
sl@0
|
220 |
|
sl@0
|
221 |
// asynchronous requests (support)
|
sl@0
|
222 |
CAsyncRequest* FindSessionAsyncRequest(TInt aFunction);
|
sl@0
|
223 |
void CancelAsyncSessionRequestL(TInt aFunction);
|
sl@0
|
224 |
|
sl@0
|
225 |
private:
|
sl@0
|
226 |
/**
|
sl@0
|
227 |
Number of open subsessions before a new one is created. This is used
|
sl@0
|
228 |
to manage cleanup if the subsession cannot be created.
|
sl@0
|
229 |
*/
|
sl@0
|
230 |
TInt iPreCreateSubsessionCount;
|
sl@0
|
231 |
|
sl@0
|
232 |
CObjectCon* iSsObjects; ///< Currently open subsessions.
|
sl@0
|
233 |
CObjectIx* iSsHandles; ///< Maps handles to open subsessions.
|
sl@0
|
234 |
|
sl@0
|
235 |
public:
|
sl@0
|
236 |
IMPORT_C CScsSession(CScsServer &aServer);
|
sl@0
|
237 |
// This duplicates the iServer/Server() code in the base class,
|
sl@0
|
238 |
// BUT this variable IS setup before derived class ConstructL is
|
sl@0
|
239 |
// called and is NOT const. This trades off 4 bytes of memory
|
sl@0
|
240 |
// against making the code cleaner and more consistent....
|
sl@0
|
241 |
CScsServer& iServer; ///< Owning server.
|
sl@0
|
242 |
};
|
sl@0
|
243 |
|
sl@0
|
244 |
/**
|
sl@0
|
245 |
Pass to CScsServer::CancelOutstandingRequests to mean all requests
|
sl@0
|
246 |
associated with a session or its subsessions.
|
sl@0
|
247 |
*/
|
sl@0
|
248 |
CScsSubsession*const KWildSubsession = reinterpret_cast<CScsSubsession*>(~0);
|
sl@0
|
249 |
|
sl@0
|
250 |
class CScsServer : public CPolicyServer
|
sl@0
|
251 |
/**
|
sl@0
|
252 |
The main server object allocates sessions. It also uses
|
sl@0
|
253 |
a shutdown timer to stop the server when there have been no
|
sl@0
|
254 |
open sessions for a set period.
|
sl@0
|
255 |
|
sl@0
|
256 |
The implementor must define a subclass which returns session
|
sl@0
|
257 |
objects of the appropriate type.
|
sl@0
|
258 |
*/
|
sl@0
|
259 |
{
|
sl@0
|
260 |
public:
|
sl@0
|
261 |
IMPORT_C virtual ~CScsServer();
|
sl@0
|
262 |
|
sl@0
|
263 |
void IncrementSessionCount();
|
sl@0
|
264 |
void DecrementSessionCount();
|
sl@0
|
265 |
inline TInt SessionCount() const;
|
sl@0
|
266 |
|
sl@0
|
267 |
IMPORT_C void ShutdownWhenIdleL();
|
sl@0
|
268 |
|
sl@0
|
269 |
// asynchronous requests
|
sl@0
|
270 |
void AddAsyncRequestL(CAsyncRequest* aAsyncRequest);
|
sl@0
|
271 |
void CancelAsyncRequest(
|
sl@0
|
272 |
CScsSession* aSession, CScsSubsession* aSubsession, TInt aFunction);
|
sl@0
|
273 |
|
sl@0
|
274 |
void CancelOutstandingRequests(CScsSession* aSession, TBool aCompleteClientRequests);
|
sl@0
|
275 |
void CancelOutstandingRequests(CScsSession* aSession, CScsSubsession* aSubsession, TBool aCompleteClientRequests);
|
sl@0
|
276 |
|
sl@0
|
277 |
/**
|
sl@0
|
278 |
This function is called just before the SCS framework marks the heap for OOM
|
sl@0
|
279 |
testing and just before checking the heap.
|
sl@0
|
280 |
|
sl@0
|
281 |
Typically this function should compact any arrays and free objects which change size and can
|
sl@0
|
282 |
not be compacted back to the same level.
|
sl@0
|
283 |
*/
|
sl@0
|
284 |
IMPORT_C virtual void DoPreHeapMarkOrCheckL();
|
sl@0
|
285 |
|
sl@0
|
286 |
/**
|
sl@0
|
287 |
This function is called just after the heap has either been marked or checked (which is just
|
sl@0
|
288 |
after DoPreHeapMarkOrCheck).
|
sl@0
|
289 |
|
sl@0
|
290 |
Typically this function should re-create any objects which had to be freed by DoPreHeapMarkOrCheck.
|
sl@0
|
291 |
*/
|
sl@0
|
292 |
IMPORT_C virtual void DoPostHeapMarkOrCheckL();
|
sl@0
|
293 |
|
sl@0
|
294 |
enum TFunctionRanges
|
sl@0
|
295 |
/**
|
sl@0
|
296 |
Function ranges to be used when configuring the server security via
|
sl@0
|
297 |
a CPolicyServer::TPolicy object.
|
sl@0
|
298 |
|
sl@0
|
299 |
Session/sSubsession function codes will be ORed into the
|
sl@0
|
300 |
EBaseSession/EBaseSubSession ranges.
|
sl@0
|
301 |
|
sl@0
|
302 |
Values from EBaseMustAllow, upwards, are used internally and must be allowed.
|
sl@0
|
303 |
|
sl@0
|
304 |
If there are multiple subsession types (with different function
|
sl@0
|
305 |
code values), then codes must be be different for each subsession
|
sl@0
|
306 |
(this restriction only applies if using the CPolicyServer::TPolicy
|
sl@0
|
307 |
mechanism to configure server security).
|
sl@0
|
308 |
|
sl@0
|
309 |
WARNNIG: These ranges MUST match the values in scscommon.h TScsFunction.
|
sl@0
|
310 |
*/
|
sl@0
|
311 |
{
|
sl@0
|
312 |
EBaseSession = 0x01000000,
|
sl@0
|
313 |
EBaseSubSession = 0x02000000,
|
sl@0
|
314 |
EBaseMustAllow = 0x03000000 //< Must allow from EBaseMustAllow upwards
|
sl@0
|
315 |
};
|
sl@0
|
316 |
|
sl@0
|
317 |
static TInt StripScsFunctionMask(TInt aFunctionId) { return aFunctionId & ~ScsImpl::KScsFunctionMask;}
|
sl@0
|
318 |
|
sl@0
|
319 |
protected:
|
sl@0
|
320 |
IMPORT_C CScsServer(const TVersion& aVersion, CActive::TPriority aPriority = CActive::EPriorityStandard);
|
sl@0
|
321 |
IMPORT_C CScsServer(const TVersion& aVersion, const CPolicyServer::TPolicy& aPolicy, CActive::TPriority aPriority = CActive::EPriorityStandard);
|
sl@0
|
322 |
IMPORT_C void ConstructL(TInt aShutdownPeriodUs);
|
sl@0
|
323 |
|
sl@0
|
324 |
IMPORT_C void DisableShutdownTimer();
|
sl@0
|
325 |
IMPORT_C void EnableShutdownTimerL(TInt aShutdownPeriodUs);
|
sl@0
|
326 |
|
sl@0
|
327 |
|
sl@0
|
328 |
// implement CServer2
|
sl@0
|
329 |
IMPORT_C virtual CSession2* NewSessionL(const TVersion& aVersion, const RMessage2& aMessage) const;
|
sl@0
|
330 |
|
sl@0
|
331 |
/**
|
sl@0
|
332 |
NewSessionL checks that this server supports the version which is
|
sl@0
|
333 |
requested by the client. If that is so, then it calls this function
|
sl@0
|
334 |
to allocate a new session object.
|
sl@0
|
335 |
|
sl@0
|
336 |
@param aMessage Connection message, as passed to NewSessionL.
|
sl@0
|
337 |
@return New session object. This is owned by the kernel,
|
sl@0
|
338 |
which will delete it when the session is closed.
|
sl@0
|
339 |
*/
|
sl@0
|
340 |
virtual CScsSession* DoNewSessionL(const RMessage2& aMessage) = 0;
|
sl@0
|
341 |
|
sl@0
|
342 |
private:
|
sl@0
|
343 |
// asynchronous requests
|
sl@0
|
344 |
CAsyncRequest* FindAsyncRequest(
|
sl@0
|
345 |
CScsSession* aSession, CScsSubsession* aSubsession, TInt aFunction);
|
sl@0
|
346 |
|
sl@0
|
347 |
static TInt RemoveCompletedRequests(TAny* aPtr);
|
sl@0
|
348 |
void RemoveCompletedRequests();
|
sl@0
|
349 |
|
sl@0
|
350 |
private:
|
sl@0
|
351 |
/**
|
sl@0
|
352 |
This server's version. It is compared against client's requested version
|
sl@0
|
353 |
when the client attempts to make a connection.
|
sl@0
|
354 |
*/
|
sl@0
|
355 |
const TVersion iVersion;
|
sl@0
|
356 |
|
sl@0
|
357 |
/** Number of open sessions. Used to start and cancel the shutdown timer. */
|
sl@0
|
358 |
TInt iSessionCount;
|
sl@0
|
359 |
|
sl@0
|
360 |
/**
|
sl@0
|
361 |
Shutdown timer, started when there are no open sessions, cancelled when
|
sl@0
|
362 |
the first session is created.
|
sl@0
|
363 |
*/
|
sl@0
|
364 |
CShutdownTimer* iShutdownTimer;
|
sl@0
|
365 |
|
sl@0
|
366 |
|
sl@0
|
367 |
public:
|
sl@0
|
368 |
/**
|
sl@0
|
369 |
Generates instances of CObjectCon, for each session to host its subsessions.
|
sl@0
|
370 |
Public because must be accessible to session objects.
|
sl@0
|
371 |
*/
|
sl@0
|
372 |
CObjectConIx* iContainerIndex;
|
sl@0
|
373 |
|
sl@0
|
374 |
/** Currently outstanding requests. */
|
sl@0
|
375 |
RPointerArray<CAsyncRequest> iAsyncRequests;
|
sl@0
|
376 |
|
sl@0
|
377 |
/** Runs to remove completed requests from iAsyncRequests. */
|
sl@0
|
378 |
CAsyncCallBack* iAsyncCleanup;
|
sl@0
|
379 |
};
|
sl@0
|
380 |
|
sl@0
|
381 |
inline TInt CScsServer::SessionCount() const
|
sl@0
|
382 |
{
|
sl@0
|
383 |
return iSessionCount;
|
sl@0
|
384 |
}
|
sl@0
|
385 |
|
sl@0
|
386 |
|
sl@0
|
387 |
|
sl@0
|
388 |
// -------- startup --------
|
sl@0
|
389 |
|
sl@0
|
390 |
/**
|
sl@0
|
391 |
The server executable must implement a factory function with this
|
sl@0
|
392 |
signature and pass it to StartScsServer to allocate and start the server.
|
sl@0
|
393 |
*/
|
sl@0
|
394 |
typedef CScsServer* (*TScsServerFactory)();
|
sl@0
|
395 |
|
sl@0
|
396 |
IMPORT_C TInt StartScsServer(TScsServerFactory aServerFactoryLC);
|
sl@0
|
397 |
|
sl@0
|
398 |
|
sl@0
|
399 |
#endif // #ifndef SCSSERVER_H
|
sl@0
|
400 |
|