os/security/cryptomgmtlibs/securitycommonutils/inc/scsserver.h
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
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