Update contrib.
1 // Copyright (c) 2008-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 // Wrapper around RMessage2 that provides message validation and error handling
15 // to improve robustness of system servers under IPC attack
19 #if !defined(CLIENTMESSAGE_H)
20 #define CLIENTMESSAGE_H
25 #include <babitflags.h>
30 //Version number of the client server framework code
31 const TInt KClientMessageVersion = 0x1;
33 //Forward Declarations
34 class CMessageParameterBase;
39 Returned to caller to indicate that the requested message number is not
40 supported by the current server
44 const TInt KErrInvalidFunction = -1004;
47 Returned to caller if a message is passed in containing a non-descriptor
48 parameter which doesn't meet the constraints specified in the schema
52 const TInt KErrBadParameter = -1005;
55 Returned to caller in UREL mode when the server attempts to call a
56 CMessageParameterBase function which is not defined for the given
57 parameter type. E.g. calling GetIntL() on a Descriptor parameter.
61 const TInt KErrWrongParameterType = -1006;
64 Returned to caller in UREL mode when a message is found to be
65 incorrectly defined in the message schema.
69 const TInt KErrBadMessageSchema = -1007;
73 Returned to caller to indicate that the requested parameter has not yet
78 const TInt KErrNotValidated = -1008; //Server internal error
81 Returned to caller to indicate that the ClientMessage framework has
82 not yet been initialised
86 const TInt KErrNotInitialised = -1009; //Server internal error
92 This is the list of Panic codes used by the ClientMessage framework
98 /**This panic is raised in UDEB mode when the server attempts to call a
99 CMessageParameterBase function which is not defined for the given
100 parameter type. E.g. calling GetIntL() on a Descriptor parameter.
102 ECMPanicWrongParameterType = 0,
104 /**This panic is raised when the server attempts to read from a
105 descriptor in the clients process and supplies a target descriptor
106 that is smaller than the client descriptor.
108 ECMPanicBadDescriptor,
110 /**This panic is raised when a TParameterDetails structure for a
111 given parameter is incorrectly defined in the message schema
113 ECMPanicBadMessageSchema
117 This Enum lists the currently supported Parameter types.
118 These enum values are used as indices to the Parameter Factory Function
119 which instantiates the appropriate CMessageParammeterBase derived
120 object to represent a given message parameter.
137 This enum lists the currently supported flags that can be passed to the server
138 in TClientMessageServerData::iFlags. Note that internally this value is stored
139 in a TBitFlags32 type but this type cannot be statically initialised in a dll
140 so the value is passed to the framework in a TInt. The bottom 16 bit flags are
141 reserved for the framework's internal use.
148 /** Default empty flag */
150 /** Setting this flag indicates that for bad message errors, request
151 should be completed with error code rather than panicking the client */
152 ESrvFlagDoNotPanicClientOnBadMessageErrors = 0x40000000,
153 /** Indicates that bad messages should be logged in UDEB */
154 ESrvFlagLogBadMessages = 0x80000000
157 //Typedef Declarations
160 This typedef is used to simplify the declaration of custom
161 parameter validation functions
165 typedef void (*TCustomValidationFn)(CMessageParameterBase* aParameter);
169 //Struct Declarations
172 This structure defines the format for storing information for an
173 individual message parameter.
175 iType - This defines the parameter type and is used by the CMessageParameterBase
176 class to instantiate the appropriate derived class via the
177 KParameterFactoryFunctions factory lookup table.
179 iMin, iMax - These are two constraints that are used in the ValidateL
180 method of a class derived from CMessageParameter to validate the message
181 as required by that parameter.
182 The constraints that are used depend on the type as follows:
184 -iMin = Minimum int value accepted
185 -iMax = Maximum int value accepted. This must be > iMin
187 NOTE: Currently unsigned integers are not explicitly supported,
188 however 16 bit unsigned integers are implicitly supported
189 by the EParamInt type. Note that the constraint above
190 (iMax > iMin) means that the maximum supported value for an
191 unsigned int is 0x7FFFFFFF.
195 -iMin = Minimum length of descriptor in clients address space.
196 If no minimum restriction exists this should be set to 0 in
197 the schema. This must be >= 0
198 -iMax = Maximum length of descriptor in clients address space.
199 The descriptor supplied by the client must not have a
200 length that exceeds this value. This must be >= iMin
203 -iMin = Minimum MaxLength of descriptor in clients address space.
204 The descriptor supplied by the client must have a MaxLength
205 that is at least as big as this value. This must be >= 0
206 -iMax = Maximum length of descriptor in clients address space.
207 The descriptor supplied by the client must not have a
208 length that exceeds this value. This must be >= 0.
210 NOTE: As the EParamDes* types use MaxLength() as a restriction
211 They cannot be used if the client intends to pass a TDesC
212 derived descriptor. If the client intends read only use
213 then the explicit EParamDes*Read types should be used instead.
216 -iMin = Minimum length of descriptor in clients address space.
217 If no minimum restriction exists this should be set to 0 in
218 the schema. This must be >= 0
219 -iMax = Maximum length of descriptor in clients address space.
220 The descriptor supplied by the client must not have a
221 length that exceeds this value. This must be >= iMin
229 struct TParameterDetails
237 This struct is used to represent a single message from the client.
238 It defines the function number corresponding to the message, the security
239 policy for the message, the number of paramaters that are passed to the
240 function and the type and constraint information for each of the parameters
241 as described above. The server must define an array of these objects to be
242 used by CClientMessage to validate all incoming messages.
244 NOTE: It is assumed that the message parameters are contiguous. I.E. if there is one
245 parameter it is assumed to be located at index 0, if there are 2 parameters they
246 are located at index 0,1 etc.
251 struct TClientMessageSchema
254 //Version number of the framework
257 //The message number described by this schema
258 const TInt iFunction;
260 //The security policy for this message. The CClientMessage framework
261 //uses this security policy to police incoming messages.
262 const TStaticSecurityPolicy& iPolicy;
264 //The number of parameters expected by this message
265 const TInt iParamCount;
267 //The array of TParameterDetails structs describing the
268 //parameter types and constraints for this message
269 const TParameterDetails* iParams;
271 //Reserved values for future proofing
272 const TInt iReserved1;
273 const TInt iReserved2;
277 This struct is used to initialise the ClientMessage Framework.
278 This struct provides the framework with the message schema and
279 the custom validation functions for the server as well as the server
281 The struct should be populated by the server using the SERVER_DATA
286 struct TClientMessageServerData
288 //Version number of this structure
291 //The number of messages accepted by this server
292 const TInt iMessageCount;
294 //The array of TClientMessageSchema structs defining the schema for
295 //each message accpeted by this server
296 const TClientMessageSchema* iMessageSchema;
298 //The number of custom validation fucntions suplpied by this server
299 const TInt iValidationFnCount;
301 //The array of custom validation functions supplied by this server.
302 //These validation functions are called from ValidateL to provide
303 //customised validation of any parameter type.
304 const TCustomValidationFn* iCustomValidationFns;
306 //The name of the server using the framework. This string is used both
307 //in logging bad messages and in panicing the client or the server.
308 //In line with Panic category strings, this value should be 16 characters or less.
309 const TUint8* iServerName;
311 //Flags used to pass settings info into the framework. The supported values are
312 //defined in the TServerFlags enum above.
315 //Reserved values for future proofing
316 const TInt iReserved1;
317 const TInt iReserved2;
323 This macro is used to simplify declaration of TClientMessageSchema entries.
324 The function number, security policy and pointer to array of TParameterDetails
325 structures is converted into the expected format for the declaration of a
326 TClientMessageSchema structure.
330 #define MESSAGE_SCHEMA(Function,Policy,Params) {KClientMessageVersion,Function,Policy,sizeof(Params)/sizeof(TParameterDetails),Params,0,0}
334 This TParameterDetails structure is used to represent a message with no
335 parameters. This is required as the schema expects a const TParameterDetails*.
339 const TParameterDetails KNoParams[1] = {{EParamNull,0,0}};
342 This macro is used to simplify declaration of TClientMessageSchema entries
343 for functions with no parameters. Because the number of parameters is set
344 to 0, a CMessageParameterBase derived object is not instantiated for this
349 #define MESSAGE_SCHEMA_NO_PARAMS(Function,Policy) {KClientMessageVersion,Function,Policy,0,KNoParams,0,0}
353 This macro is used to simplify declaration of the TClientMessageServerData structure.
354 The pointer to the array of TClientMessageSchemas, pointer to array of
355 TCustomValidationFns, server name and server flags are converted into the
356 expected format for the declaration of a TClientMessageServerData structure.
360 #define SERVER_DATA(ClientMessages,ValidationFns,ServerName,Flags){KClientMessageVersion,sizeof(ClientMessages)/sizeof(TClientMessageSchema),ClientMessages,sizeof(ValidationFns)/sizeof(TCustomValidationFn),ValidationFns,(TUint8*)ServerName,Flags,0,0}
366 This is the abstract base class for all message parameter classes.
367 The class defines one pure virtual function, ValidateL, which all derived classes
368 must implement. This class also defines default implementation for the access
369 methods which should be properly defined in all derived classes.
370 This class is not intended for external derivation.
374 NONSHARABLE_CLASS(CMessageParameterBase) : public CBase
377 static CMessageParameterBase* CreateL(const TParameterDetails& aParam, TInt aParamIndex,
378 const RMessage2& aMessage);
379 virtual void ValidateL() = 0;
380 virtual ~CMessageParameterBase(){};
382 virtual TInt GetIntL();
383 virtual const TAny* GetPtrL();
384 virtual const TDesC8& GetDes8L();
385 virtual const TDesC& GetDes16L();
386 virtual TInt GetDesLengthL();
387 virtual TInt GetDesMaxLengthL();
388 virtual void ReadL(TDes8& aDes, TInt aOffset);
389 virtual void ReadL(TDes& aDes, TInt aOffset);
390 virtual void WriteL(const TDesC8& aDes, TInt aOffset);
391 virtual void WriteL(const TDesC& aDes, TInt aOffset);
397 CMessageParameterBase(const TParameterDetails& aParam, TInt aParamIndex,
398 const RMessage2& aMessage, TCustomValidationFn aValidationFn);
401 static TCustomValidationFn GetValidationFunctionL(const TParameterDetails& aParam);
404 //The index of this parameter within the corresponding RMessage2 object
407 //A reference to the RMessage2 object that contains this parameter
408 const RMessage2& iMessage;
410 //A reference to the structure that defines the expected type
411 //and constraints for this parameter
412 const TParameterDetails& iParamDetails;
414 //A custom validation function to allow user defined validation
415 //this function is called from ValidateL
416 TCustomValidationFn iValidationFn;
421 This class is used by a server to sanitise incoming messages.
422 The class provides a wrapper around an RMessage2 object and provides a method
423 for validation of the message against the constraints defined in the message schema.
424 This class also provides methods for safely accessing the message arguments,
425 and for error handling and logging of bad messages.
426 This class is not intended for derivation.
430 NONSHARABLE_CLASS(CClientMessage) : public CBase
433 IMPORT_C static void InitialiseFrameworkL(const TClientMessageServerData& aServerData);
434 IMPORT_C static CClientMessage* NewL(const RMessage2& aMessage);
435 IMPORT_C virtual ~CClientMessage();
437 IMPORT_C TInt Function();
438 IMPORT_C const RMessage2& Message();
439 IMPORT_C virtual void ValidateL();
440 IMPORT_C virtual void ValidateL(TInt aParam);
441 IMPORT_C virtual void CompleteRequestL(TInt aError);
442 IMPORT_C virtual void PanicClient(const TDesC& aServer, TInt aPanic);
444 IMPORT_C virtual TInt GetIntL(TInt aParam);
445 IMPORT_C virtual const TAny* GetPtrL(TInt aParam);
446 IMPORT_C virtual const TDesC8& GetDes8L(TInt aParam);
447 IMPORT_C virtual const TDesC& GetDes16L(TInt aParam);
448 IMPORT_C virtual TInt GetDesLengthL(TInt aParam);
449 IMPORT_C virtual TInt GetDesMaxLengthL(TInt aParam);
450 IMPORT_C virtual void ReadL(TInt aParam, TDes8& aDes, TInt aOffset = 0);
451 IMPORT_C virtual void ReadL(TInt aParam, TDes16& aDes, TInt aOffset = 0);
452 IMPORT_C virtual void WriteL(TInt aParam, const TDesC8& aDes, TInt aOffset = 0);
453 IMPORT_C virtual void WriteL(TInt aParam, const TDesC16& aDes, TInt aOffset = 0);
456 virtual void LogBadMessageL(TInt aError);
457 virtual void CheckSecurityPolicyL(const TSecurityPolicy& aPolicy);
458 CClientMessage(const RMessage2& aMessage,const TClientMessageServerData& aServerData);
460 const TClientMessageSchema* FindMessageSchema();
461 void CheckValidatedL(TInt aParam);
462 TBool LogBadMessages();
465 //Array of pointers to CMessageParameterBase Derived parameter objects used
466 // to validate and access the individual message arguments
467 RPointerArray <CMessageParameterBase> iParameters;
469 //Reference to the underlying RMessage2 wrapped by this object
470 const RMessage2& iMessage;
472 //Reference to TClientMessageServerData structure stored in TLS for
473 //this thread. This structure is passed in to the framework by the server
475 const TClientMessageServerData& iServerData;
477 //Bit flags for use internally within CClientMessage.
481 This class is used for Base64 based encoding and decoding .
482 This class provides a method for encoding and decoding
488 // base64 and UU coding defines.
490 This is the list of Bitmask used for encoding and decoding
501 This is the list of shift values used for encoding and decoding
505 enum EMaskShiftValues
514 This enum is used as padding charcter
523 IMPORT_C static TInt Encode(const TDesC8& aSrcString, TDes8& aDestString);
524 IMPORT_C static TInt Decode(const TDesC8& aSrcString, TDes8& aDestString);