1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/ossrv/lowlevellibsandfws/apputils/bsul/src/clientmessage.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,1836 @@
1.4 +// Copyright (c) 2008-2009 Nokia Corporation and/or its subsidiary(-ies).
1.5 +// All rights reserved.
1.6 +// This component and the accompanying materials are made available
1.7 +// under the terms of "Eclipse Public License v1.0"
1.8 +// which accompanies this distribution, and is available
1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
1.10 +//
1.11 +// Initial Contributors:
1.12 +// Nokia Corporation - initial contribution.
1.13 +//
1.14 +// Contributors:
1.15 +//
1.16 +// Description:
1.17 +//
1.18 +
1.19 +#include "clientmessagecmn.h"
1.20 +
1.21 +using namespace BSUL;
1.22 +
1.23 +/**
1.24 +Base 64 decoding table
1.25 +*/
1.26 +const TInt8 AsciiToBase64[80]=
1.27 + {
1.28 + 62, -1, -1, -1, 63, 52, 53, 54, 55, 56,
1.29 + 57, 58, 59, 60, 61, -1, -1, -1, 64, -1,
1.30 + -1, -1, 0, 1, 2, 3, 4, 5, 6, 7,
1.31 + 8, 9, 10, 11, 12, 13, 14, 15, 16, 17,
1.32 + 18, 19, 20, 21, 22, 23, 24, 25, -1, -1,
1.33 + -1, -1, -1, -1, 26, 27, 28, 29, 30, 31,
1.34 + 32, 33, 34, 35, 36, 37, 38, 39, 40, 41,
1.35 + 42, 43, 44, 45, 46, 47, 48, 49, 50, 51
1.36 + };
1.37 +
1.38 +/**
1.39 +Base 64 encoding table
1.40 +*/
1.41 +const TInt8 Base64ToAscii[65]=
1.42 + {
1.43 + 65, 66, 67, 68, 69, 70, 71, 72, 73, 74,
1.44 + 75, 76, 77, 78, 79, 80, 81, 82, 83, 84,
1.45 + 85, 86, 87, 88, 89, 90, 97, 98, 99,100,
1.46 + 101,102,103,104,105,106,107,108,109,110,
1.47 + 111,112,113,114,115,116,117,118,119,120,
1.48 + 121,122, 48, 49, 50, 51, 52, 53, 54, 55,
1.49 + 56, 57, 43, 47, 61
1.50 + };
1.51 +
1.52 +const TInt8 KImcvLookUpStartOffset = 43;
1.53 +const TUint8 KImcvConvEquals = '=';
1.54 +/**
1.55 +The maximum number of US ASCII characters per line that are before the CRLF line
1.56 +terminator when sending emails. RFC 2822 recommends that each line SHOULD not exceed
1.57 +80 characters including the CRLF terminator, and MUST not exceed 1000.
1.58 +*/
1.59 +const TInt KMaxB64EncodedCharsPerLine = 60; // Could be increased to 75 characters for every encoded line if KDecodeLineLength = 675. 60 was chosen to maintain existing behaviour.
1.60 +
1.61 +/**
1.62 +Parameter factory function lookup table. This is used to instantiate a
1.63 +CMessageParameterBase derived object based on a TParamType enum value
1.64 +*/
1.65 +const TMessageParameterFactoryFn KParameterFactoryFunctions[] = {NULL,
1.66 + CIntParameter::NewL,
1.67 + CDes8ReadParameter::NewL,
1.68 + CDes8Parameter::NewL,
1.69 + CPckgParameter::NewL,
1.70 + CDes16ReadParameter::NewL,
1.71 + CDes16Parameter::NewL,
1.72 + CPtrParameter::NewL};
1.73 +
1.74 +/**
1.75 +Panic string for client message framework panic
1.76 +*/
1.77 +#ifdef _DEBUG
1.78 +_LIT(KPanicCategory,"BSUL::ClientMsg");
1.79 +#endif
1.80 +
1.81 +const TInt KMaxServerNameLength = 32;
1.82 +
1.83 +/**
1.84 +This static function is used to panic the server in case of
1.85 +incorrect use of CMessageParameterBase APIs or a badly formed
1.86 +message schema
1.87 +@param aPanic The Panic value
1.88 +*/
1.89 +void PanicServer(TInt aPanic)
1.90 + {
1.91 + _LIT(KUnknownServer, "Unknown");
1.92 + TBuf<KMaxServerNameLength> serverName(KUnknownServer);
1.93 +
1.94 + //Get the TLS data for this thread
1.95 + TClientMessageServerData* serverData = static_cast<TClientMessageServerData*>(Dll::Tls());
1.96 +
1.97 + if(serverData != NULL)
1.98 + {
1.99 + TPtrC8 name(serverData->iServerName);
1.100 + serverName.Copy(name);
1.101 + }
1.102 +
1.103 + User::Panic(serverName, aPanic);
1.104 + }
1.105 +
1.106 +/**
1.107 +Static initialisation function for ClientMessage Framework
1.108 +@param aServerData The initialisation data for the server using the library
1.109 +@leave Any system wide error code
1.110 +*/
1.111 +EXPORT_C void CClientMessage::InitialiseFrameworkL(const TClientMessageServerData& aServerData)
1.112 + {
1.113 + __ASSERT_DEBUG(User::StringLength(aServerData.iServerName) <= KMaxServerNameLength, User::Invariant());
1.114 + Dll::SetTls((TAny*)&aServerData);
1.115 + }
1.116 +
1.117 +/**
1.118 +Static factory function for CClientMessage class
1.119 +@param aMessage The message that this object encapsulates
1.120 +@return Pointer to a fully constructed CClientMessage object.
1.121 +@leave KErrNotInitialised if the framework has not been initialised by a
1.122 + call to InitialiseFrameworkL.
1.123 +*/
1.124 +EXPORT_C CClientMessage* CClientMessage::NewL(const RMessage2& aMessage)
1.125 + {
1.126 + const TClientMessageServerData* serverData = static_cast<const TClientMessageServerData*>(Dll::Tls());
1.127 + if(serverData == NULL)
1.128 + {
1.129 + User::Leave(KErrNotInitialised);
1.130 + }
1.131 +
1.132 + CClientMessage* self = new(ELeave) CClientMessage(aMessage,*serverData);
1.133 + CleanupStack::PushL(self);
1.134 + self->ConstructL();
1.135 + CleanupStack::Pop(self);
1.136 +
1.137 + return self;
1.138 + }
1.139 +
1.140 +
1.141 +/**
1.142 +Second phase constructor of CClientMessage Object
1.143 +Finds the schema for this message and checks the caller against the security
1.144 +policy defined for this message. Traverses the array of message parameters
1.145 +and instantiates a CMessageParameterBase object for each parameter and adds these
1.146 +to its internal array.
1.147 +@leave KErrInvalidFunction If the message function is not found in the message table
1.148 +@leave KErrBadHandle if the message handle is Null
1.149 +@leave KErrPermissionDenied if the security policy for this message is not satisfied
1.150 +*/
1.151 +void CClientMessage::ConstructL()
1.152 + {
1.153 + if(!iMessage.IsNull())
1.154 + {
1.155 +
1.156 + //Find the message schema for this message.
1.157 + const TClientMessageSchema* messageSchema = FindMessageSchema();
1.158 +
1.159 + if(!messageSchema)
1.160 + {
1.161 + LogBadMessageL(KErrInvalidFunction);
1.162 + User::Leave(KErrInvalidFunction);
1.163 + }
1.164 +
1.165 + //Check message against security policy
1.166 + CheckSecurityPolicyL(messageSchema->iPolicy);
1.167 +
1.168 + //iterate through message parameters and instantiate all parameter objects
1.169 + for(int index = 0;index < messageSchema->iParamCount;index++)
1.170 + {
1.171 + CMessageParameterBase* parameter =
1.172 + CMessageParameterBase::CreateL(messageSchema->iParams[index], index, iMessage);
1.173 +
1.174 + //Some parameter types are defined to be ignored. These
1.175 + //should not be added to the list of parameters
1.176 + if(parameter != NULL)
1.177 + {
1.178 + //AppendL can leave so use cleanupstack to ensure that memory is not
1.179 + //leaked if AppendL leaves.
1.180 + CleanupStack::PushL(parameter);
1.181 + iParameters.AppendL(parameter);
1.182 + CleanupStack::Pop(parameter);
1.183 + }
1.184 + }
1.185 + }
1.186 + else
1.187 + {
1.188 + LogBadMessageL(KErrBadHandle);
1.189 + User::Leave(KErrBadHandle);
1.190 + }
1.191 + }
1.192 +
1.193 +/**
1.194 +Constructor for CClientMessage Object
1.195 +@param aMessage The RMessage2 to be represented by this object
1.196 +@param aServerData The Initialisation data for the server creating this object
1.197 +*/
1.198 +EXPORT_C CClientMessage::CClientMessage(const RMessage2& aMessage,
1.199 + const TClientMessageServerData& aServerData)
1.200 + : iParameters(KMaxParameters), iMessage(aMessage),
1.201 + iServerData(aServerData),iFlags(aServerData.iFlags & 0xFFFF0000)
1.202 +
1.203 + {
1.204 +
1.205 + }
1.206 +
1.207 +/**
1.208 +Destructor for CClientMessageObject
1.209 +*/
1.210 +EXPORT_C CClientMessage::~CClientMessage()
1.211 + {
1.212 + for(int i = 0; i < iParameters.Count();i++)
1.213 + {
1.214 + delete iParameters[i];
1.215 + }
1.216 + iParameters.Reset();
1.217 + }
1.218 +
1.219 +EXPORT_C const RMessage2& CClientMessage::Message()
1.220 + {
1.221 + return iMessage;
1.222 + }
1.223 +
1.224 +/**
1.225 +Panics the client through the message object
1.226 +set an internal flag to indicate that the RMessage reference handle is now NULL
1.227 +due to the client thread being tidied up.
1.228 +@param aServer The Panic category
1.229 +@param aPanic The Panic value
1.230 +*/
1.231 +EXPORT_C void CClientMessage::PanicClient(const TDesC& aServer, TInt aPanic)
1.232 + {
1.233 + iMessage.Panic(aServer, aPanic);
1.234 + iFlags.Set(EFlagPanicClient);
1.235 + }
1.236 +
1.237 +/**
1.238 +Checks a message against the security policy defined for the server
1.239 +@param aPolicy The security policy to check this message against
1.240 +@leave KErrPermissionDenied if the message does not fulfil the security policy
1.241 +*/
1.242 +void CClientMessage::CheckSecurityPolicyL(const TSecurityPolicy& aPolicy)
1.243 + {
1.244 + if(!(aPolicy.CheckPolicy(iMessage,
1.245 + "Client failed security policy check for this server")))
1.246 + {
1.247 + User::Leave(KErrPermissionDenied);
1.248 + }
1.249 + return;
1.250 + }
1.251 +
1.252 +/**
1.253 +Finds a message schema in the message table for this server.
1.254 +Does a binary search on the function number to pull the correct
1.255 +message from the table. Note that this assumes that the table
1.256 +is sorted.
1.257 +@return A pointer to a TClientMessageSchema object in the message table, or Null if
1.258 + the message does not correpsond to a message in the message table
1.259 +*/
1.260 +const TClientMessageSchema* CClientMessage::FindMessageSchema()
1.261 + {
1.262 + //This should always be less than KNumClientMessages
1.263 + TInt function = iMessage.Function();
1.264 + TInt beg = 0;
1.265 + TInt end = iServerData.iMessageCount - 1;
1.266 + TInt mid = 0;
1.267 + TInt midFn = 0;
1.268 +
1.269 + while(beg <= end)
1.270 + {
1.271 + mid = (end + beg)/2;
1.272 +
1.273 + midFn = iServerData.iMessageSchema[mid].iFunction;
1.274 + if(midFn > function)
1.275 + {
1.276 + end = mid - 1;
1.277 + }
1.278 + else if(midFn < function)
1.279 + {
1.280 + beg = mid + 1;
1.281 + }
1.282 + else
1.283 + {
1.284 + return &iServerData.iMessageSchema[mid];
1.285 + }
1.286 + }
1.287 + return NULL;
1.288 + }
1.289 +
1.290 +
1.291 +
1.292 +
1.293 +/**
1.294 +Validates the message parameters against the constraints provided
1.295 +in the message table
1.296 +@leave KErrBadMessage if the message fails validation against the criteria supplied in the
1.297 + message table
1.298 +@leave Any system-wide error code
1.299 +*/
1.300 +EXPORT_C void CClientMessage::ValidateL()
1.301 + {
1.302 +
1.303 + for(int i = 0; i < iParameters.Count();i++)
1.304 + {
1.305 + iParameters[i]->ValidateL();
1.306 + iFlags.Set(i);
1.307 + }
1.308 + }
1.309 +
1.310 +/**
1.311 +Validates a single message argument against the constraints provided
1.312 +in the message table
1.313 +@param aParam The index value identifying the argument to validate
1.314 +@leave KErrBadMessage if the message fails validation against the criteria supplied in the
1.315 + message table for the requested argument
1.316 +@leave KErrArgument if aParam is negative or is greater than the number of parameters for
1.317 + this message
1.318 +@leave Any system-wide error code
1.319 +*/
1.320 +EXPORT_C void CClientMessage::ValidateL(TInt aParam)
1.321 + {
1.322 +
1.323 + if(( aParam >= 0) && (aParam < iParameters.Count()))
1.324 + {
1.325 + iParameters[aParam]->ValidateL();
1.326 + iFlags.Set(aParam);
1.327 + }
1.328 + else
1.329 + {
1.330 + User::Leave(KErrArgument);
1.331 + }
1.332 + }
1.333 +
1.334 +/**
1.335 +Checks if a given parameter has been validated
1.336 +@param aParam The index value identifying the paramater to check
1.337 +@leave KErrArgument if aParam is not a valid parameter value
1.338 +@leave KErrNotValidated if the parameter has not been validated
1.339 +*/
1.340 +void CClientMessage::CheckValidatedL(TInt aParam)
1.341 + {
1.342 + if((aParam < EFlagParam0Validated) || (aParam > EFlagParam3Validated))
1.343 + {
1.344 + User::Leave(KErrArgument);
1.345 + }
1.346 +
1.347 + if(!iFlags.IsSet(aParam))
1.348 + {
1.349 + User::Leave(KErrNotValidated);
1.350 + }
1.351 + }
1.352 +
1.353 +
1.354 +/**
1.355 +Checks if a bad messages should be logged
1.356 +@return True if bad messages should be logged
1.357 +*/
1.358 +TBool CClientMessage::LogBadMessages()
1.359 + {
1.360 + return iFlags.IsSet(EFlagLogBadMessages);
1.361 + }
1.362 +
1.363 +/**
1.364 +Checks if a bad messages should be logged
1.365 +@param aError The error code to log
1.366 +*/
1.367 +void CClientMessage::LogBadMessageL(TInt aError)
1.368 + {
1.369 + //Check if logging of bad messages is enabled
1.370 + if(LogBadMessages())
1.371 + {
1.372 +
1.373 + TUid sid = TUid(iMessage.SecureId());
1.374 + TUidName clientSid = sid.Name();
1.375 +
1.376 + TInt function = Function();
1.377 +
1.378 + TBuf<KMaxServerNameLength> serverName;
1.379 + TPtrC8 name(iServerData.iServerName);
1.380 + serverName.Copy(name);
1.381 +
1.382 + switch(aError)
1.383 + {
1.384 + #ifdef _DEBUG
1.385 +
1.386 +
1.387 + case KErrInvalidFunction:
1.388 + RDebug::Print(_L("%S - CClientMessage Error: %d, Function %d. Unknown function request from client %S.\n"),
1.389 + &serverName,aError,function, &clientSid);
1.390 + break;
1.391 +
1.392 + case KErrBadParameter:
1.393 + RDebug::Print(_L("%S - CClientMessage Error: %d, Function %d. Bad argument in IPC request from client %S.\n"),
1.394 + &serverName,aError,function, &clientSid);
1.395 + break;
1.396 +
1.397 + case KErrBadMessageSchema:
1.398 + RDebug::Print(_L("%S - CClientMessage Error: %d, Function %d. Message schema incotrectly defined for this function %S.\n"),
1.399 + &serverName,aError,function, &clientSid);
1.400 + break;
1.401 +
1.402 + case KErrBadDescriptor:
1.403 + case KErrOverflow:
1.404 + RDebug::Print(_L("%S - CClientMessage Error: %d, Function %d. Bad descriptor argument in IPC request from client %S.\n"),
1.405 + &serverName,aError,function, &clientSid);
1.406 + break;
1.407 +
1.408 + case KErrNotValidated:
1.409 + RDebug::Print(_L("%S - CClientMessage Error: %d, Function %d. Message parameter not validated before use %S.\n"),
1.410 + &serverName,aError,function, &clientSid);
1.411 + break;
1.412 +
1.413 + default:
1.414 + RDebug::Print(_L("%S - CClientMessage Error: %d, Function %d. Bad message received from client %S.\n"),
1.415 + &serverName,aError,function, &clientSid);
1.416 + break;
1.417 + #else
1.418 + default:
1.419 + break;
1.420 + #endif
1.421 + }
1.422 + }
1.423 +
1.424 +
1.425 + }
1.426 +
1.427 +/**
1.428 +Completes the message request or Panics the client if a
1.429 +message error has occured.
1.430 +@param aError The error value to complete the message with
1.431 +*/
1.432 +EXPORT_C void CClientMessage::CompleteRequestL(TInt aError)
1.433 + {
1.434 + //If server panics client
1.435 + //then iMessage will be NULL
1.436 + if(!iFlags.IsSet(EFlagPanicClient))
1.437 + {
1.438 + if(aError != KErrNone)
1.439 + {
1.440 + LogBadMessageL(aError);
1.441 + }
1.442 +
1.443 + switch(aError)
1.444 + {
1.445 + case KErrInvalidFunction:
1.446 + case KErrBadDescriptor:
1.447 + case KErrOverflow:
1.448 + {
1.449 + //Check if Panic is disabled
1.450 + if( iFlags.IsClear(EFlagDoNotPanicClientOnBadMessageErrors) )
1.451 + {
1.452 + TBuf<KMaxServerNameLength> serverName;
1.453 + TPtrC8 name(iServerData.iServerName);
1.454 + serverName.Copy(name);
1.455 + PanicClient(serverName, aError);
1.456 + break;
1.457 + }
1.458 + }
1.459 + default:
1.460 + {
1.461 + iMessage.Complete(aError);
1.462 + break;
1.463 + }
1.464 + }//switch
1.465 +
1.466 + }//if
1.467 + }
1.468 +
1.469 +/**
1.470 +Gets the function number of this message
1.471 +@return The function number of this message
1.472 +*/
1.473 +EXPORT_C TInt CClientMessage::Function()
1.474 + {
1.475 + return iMessage.Function();
1.476 + }
1.477 +
1.478 +/**
1.479 +Gets the requested message argument as an integer value
1.480 +@param aParam The parameter number to retrieve
1.481 +@return The Int value for the requested parameter
1.482 +@leave KErrNotValidated if the message parameter has not been validated
1.483 + KErrWrongParameterType in UREL is the parameter requested is not an integer type
1.484 + Any other system wide error code
1.485 +@panic ECMPanicWrongParameterType If this function is called for a parameter
1.486 +type that is not CIntParameter.
1.487 +*/
1.488 +EXPORT_C TInt CClientMessage::GetIntL(TInt aParam)
1.489 + {
1.490 + CheckValidatedL(aParam);
1.491 +
1.492 + return iParameters[aParam]->GetIntL();
1.493 + }
1.494 +
1.495 +/**
1.496 +Gets the requested message argument as an TAny*
1.497 +@param aParam The parameter number to retrieve
1.498 +@return The TAny* for the requested parameter
1.499 +@leave KErrNotValidated if the message parameter has not been validated
1.500 + KErrWrongParameterType in UREL is the parameter requested is not an Ptr type
1.501 + Any other system wide error code
1.502 +@panic ECMPanicWrongParameterType If this function is called for a parameter
1.503 +type that is not CPtrParameter.
1.504 +*/
1.505 +EXPORT_C const TAny* CClientMessage::GetPtrL(TInt aParam)
1.506 + {
1.507 + CheckValidatedL(aParam);
1.508 +
1.509 + return iParameters[aParam]->GetPtrL();
1.510 + }
1.511 +
1.512 +/**
1.513 +Gets a reference to the local copy of the descriptor read from the message
1.514 +@param aParam The parameter number of the descriptor to retrieve
1.515 +@leave KErrNotValidated if the message parameter has not been validated
1.516 + KErrWrongParameterType in UREL is the parameter requested is not a readable
1.517 + TDes8 type
1.518 + Any other system wide error code
1.519 +@panic ECMPanicWrongParameterType If this function is called for a parameter
1.520 +type that is not CDes8ReadParameter.
1.521 +*/
1.522 +EXPORT_C const TDesC8& CClientMessage::GetDes8L(TInt aParam)
1.523 + {
1.524 + CheckValidatedL(aParam);
1.525 +
1.526 + return iParameters[aParam]->GetDes8L();
1.527 + }
1.528 +
1.529 +/**
1.530 +Gets a reference to the local copy of the descriptor read from the message
1.531 +@param aParam The parameter number of the descriptor to retrieve
1.532 +@leave KErrNotValidated if the message parameter has not been validated
1.533 + KErrWrongParameterType in UREL is the parameter requested is not a readable
1.534 + TDes16 type
1.535 + Any other system wide error code
1.536 +@panic ECMPanicWrongParameterType If this function is called for a parameter
1.537 +type that is not CDes16ReadParameter.
1.538 +*/
1.539 +EXPORT_C const TDesC& CClientMessage::GetDes16L(TInt aParam)
1.540 + {
1.541 + CheckValidatedL(aParam);
1.542 +
1.543 + return iParameters[aParam]->GetDes16L();
1.544 + }
1.545 +
1.546 +
1.547 +
1.548 +/**
1.549 +Gets a descriptor value read from the message
1.550 +@param aParam The parameter number of the descriptor to retrieve
1.551 +@param aDes On exit contains the descriptor value requested
1.552 +@param aOffset The desired offset from which to read the descriptor
1.553 +@leave KErrNotValidated if the message parameter has not been validated
1.554 + KErrWrongParameterType in UREL is the parameter requested is not a readable
1.555 + TDes8 type
1.556 + Any other system wide error code
1.557 +@panic ECMPanicWrongParameterType If this function is called for a parameter
1.558 +type that is not CDes8Parameter or CDes8ReadParameter.
1.559 +*/
1.560 +EXPORT_C void CClientMessage::ReadL(TInt aParam, TDes8& aDes, TInt aOffset)
1.561 + {
1.562 + CheckValidatedL(aParam);
1.563 +
1.564 + iParameters[aParam]->ReadL(aDes, aOffset);
1.565 + }
1.566 +
1.567 +/**
1.568 +Gets a descriptor value read from the message
1.569 +@param aParam The parameter number of the descriptor to retrieve
1.570 +@param aDes On exit contains the descriptor value requested
1.571 +@param aOffset The desired offset from which to read the descriptor
1.572 +@leave KErrNotValidated if the message parameter has not been validated
1.573 + KErrWrongParameterType in UREL is the parameter requested is not a readable
1.574 + TDes16 type
1.575 + Any other system wide error code
1.576 +@panic ECMPanicWrongParameterType If this function is called for a parameter
1.577 +type that is not CDes16Parameter or CDes16ReadParameter.
1.578 +*/
1.579 +EXPORT_C void CClientMessage::ReadL(TInt aParam, TDes16& aDes, TInt aOffset)
1.580 + {
1.581 + CheckValidatedL(aParam);
1.582 +
1.583 + iParameters[aParam]->ReadL(aDes, aOffset);
1.584 + }
1.585 +
1.586 +/**
1.587 +Writes to a descriptor field in the message
1.588 +@param aParam The parameter number of the descriptor to write
1.589 +@param aDes The descriptor to write to the message
1.590 +@param aOffset The desired offset at which to write the descriptor
1.591 +@leave KErrNotValidated if the message parameter has not been validated
1.592 + KErrWrongParameterType in UREL is the parameter requested is not a writable
1.593 + TDes8 type
1.594 + Any other system wide error code
1.595 +@panic ECMPanicWrongParameterType If this function is called for a parameter
1.596 +type that is not CDes8Parameter
1.597 +*/
1.598 +EXPORT_C void CClientMessage::WriteL(TInt aParam, const TDesC8& aDes, TInt aOffset)
1.599 + {
1.600 + CheckValidatedL(aParam);
1.601 +
1.602 + iParameters[aParam]->WriteL(aDes, aOffset);
1.603 + }
1.604 +
1.605 +/**
1.606 +Writes to a descriptor field in the message
1.607 +@param aParam The parameter number of the descriptor to write
1.608 +@param aDes The descriptor to write to the message
1.609 +@param aOffset The desired offset at which to write the descriptor
1.610 +@leave KErrNotValidated if the message parameter has not been validated
1.611 + KErrWrongParameterType in UREL is the parameter requested is not a writable
1.612 + TDes16 type
1.613 + Any other system wide error code
1.614 +@panic ECMPanicWrongParameterType If this function is called for a parameter
1.615 +type that is notCDes16Parameter
1.616 +*/
1.617 +EXPORT_C void CClientMessage::WriteL(TInt aParam, const TDesC16& aDes, TInt aOffset)
1.618 + {
1.619 + CheckValidatedL(aParam);
1.620 +
1.621 + iParameters[aParam]->WriteL(aDes, aOffset);
1.622 + }
1.623 +
1.624 +/**
1.625 +Gets the length of the requested descriptor message argument
1.626 +@param aParam The parameter number to retrieve
1.627 +@return The Length of the descriptor in the client process
1.628 +@leave KErrNotValidated if the message parameter has not been validated
1.629 + KErrWrongParameterType in UREL is the parameter requested is not a descriptor type
1.630 + Any other system wide error code
1.631 +@panic ECMPanicWrongParameterType If this function is called for a parameter
1.632 +type that is not a descriptor type.
1.633 +*/
1.634 +EXPORT_C TInt CClientMessage::GetDesLengthL(TInt aParam)
1.635 + {
1.636 + CheckValidatedL(aParam);
1.637 +
1.638 + return iParameters[aParam]->GetDesLengthL();
1.639 + }
1.640 +
1.641 +/**
1.642 +Gets the max length of the requested descriptor message argument
1.643 +@param aParam The parameter number to retrieve
1.644 +@return The Max length of the descriptor in the client process
1.645 +@leave KErrNotValidated if the message parameter has not been validated
1.646 + KErrWrongParameterType in UREL is the parameter requested is not a descriptor type
1.647 + Any other system wide error code
1.648 +@panic ECMPanicWrongParameterType If this function is called for a parameter type
1.649 +that is not a descriptor type.
1.650 +*/
1.651 +EXPORT_C TInt CClientMessage::GetDesMaxLengthL(TInt aParam)
1.652 + {
1.653 + CheckValidatedL(aParam);
1.654 +
1.655 + return iParameters[aParam]->GetDesMaxLengthL();
1.656 + }
1.657 +
1.658 +/********************************************************************************
1.659 + * CMessageParameterBase and Derived Class Definitions
1.660 + *******************************************************************************/
1.661 +
1.662 +
1.663 +/**
1.664 +Factory function for instantiating derived Parameter classes.
1.665 +Uses factory lookup table to instantiate approptiate parameter
1.666 +object based on parameter details passed in
1.667 +@param aParam Parameter details object used to instantiate an appropriate
1.668 + implementation of CMessageParameterBase.
1.669 +@param aParamIndex The Index of this parameter within the RMessage2 arguments
1.670 +@param aMessage The RMessage2 object containing the parameter represented by
1.671 + this object
1.672 +@return A fully constructed CMessageParameterBase derived object deterimined by
1.673 + aParam.
1.674 +@leave KErrBadMessageSchema in UREL if if the schema for this parameter is
1.675 + incorrectly defined
1.676 +@leave Any system-wide error code.
1.677 +@panic ECMPanicBadMessageSchema in UDEB if the schema for this parameter is
1.678 + incorrectly defined
1.679 +*/
1.680 +CMessageParameterBase* CMessageParameterBase::CreateL(const TParameterDetails& aParam,
1.681 + TInt aParamIndex, const RMessage2& aMessage)
1.682 + {
1.683 +
1.684 + //The parameter type is the bottom 16 bits of the param type
1.685 + TInt paramType = (aParam.iType & KParamTypeMask);
1.686 +
1.687 + __ASSERT_DEBUG((paramType > 0), PanicServer(ECMPanicBadMessageSchema));
1.688 +
1.689 + CMessageParameterBase* newParam = NULL;
1.690 +
1.691 + switch(paramType)
1.692 + {
1.693 + case EParamInt:
1.694 + case EParamDes8Read:
1.695 + case EParamDes8:
1.696 + case EParamDes16Read:
1.697 + case EParamDes16:
1.698 + case EParamPtr:
1.699 + case EParamPckg:
1.700 + {
1.701 + //Create the new parameter object
1.702 + newParam = (KParameterFactoryFunctions[paramType])(aParam, aParamIndex,
1.703 + aMessage, GetValidationFunctionL(aParam));
1.704 + break;
1.705 + }
1.706 +
1.707 + default:
1.708 + {
1.709 +#ifdef _DEBUG
1.710 + PanicServer(ECMPanicBadMessageSchema);
1.711 +#else
1.712 + User::Leave(KErrBadMessageSchema);
1.713 +#endif
1.714 + }
1.715 + }
1.716 +
1.717 + return newParam;
1.718 + }
1.719 +
1.720 +/**
1.721 +Constructor for CMessageParameterBase object
1.722 +@param aParamIndex The Index of this parameter within the RMessage2 arguments
1.723 +@param aMessage The RMessage2 object containing the parameter represented by
1.724 + this object
1.725 +*/
1.726 +CMessageParameterBase::CMessageParameterBase(const TParameterDetails& aParam,
1.727 + TInt aParamIndex,const RMessage2& aMessage, TCustomValidationFn aValidationFn)
1.728 + : iIndex(aParamIndex), iMessage(aMessage), iParamDetails(aParam), iValidationFn(aValidationFn)
1.729 + {
1.730 +
1.731 + }
1.732 +
1.733 +/**
1.734 +Gets the validation function for this parameter from the
1.735 +TClientMessageServerData structure
1.736 +@param aParam Parameter object used to find the validation function
1.737 +@return The validation function for this parameter type
1.738 +@leave KErrBadMessageSchema in UREL if if the schema for this parameter is
1.739 + incorrectly defined
1.740 +@leave Any other system wide error code
1.741 +@panic ECMPanicBadMessageSchema in UDEB if the schema for this parameter is
1.742 + incorrectly defined
1.743 +*/
1.744 +TCustomValidationFn CMessageParameterBase::GetValidationFunctionL(const TParameterDetails& aParam)
1.745 + {
1.746 + //Get the TLS data for this thread - this will never be null at this point
1.747 + //as it is checked in CClientMessage::NewL
1.748 + TClientMessageServerData* serverData = static_cast<TClientMessageServerData*>(Dll::Tls());
1.749 +
1.750 + //The index of the validation function for this parameter is held in
1.751 + //the upper 16 bits of aParam.iType. Mask this out and shift down to
1.752 + //get the index.
1.753 + TInt fnIndex = (aParam.iType & KValidationFnIndexMask) >> KShift16Bit;
1.754 +
1.755 +
1.756 + if(fnIndex >= serverData->iValidationFnCount)
1.757 + {
1.758 +#ifdef _DEBUG
1.759 + PanicServer(ECMPanicBadMessageSchema);
1.760 +#else
1.761 + User::Leave(KErrBadMessageSchema);
1.762 +#endif
1.763 + }
1.764 +
1.765 + //Return the validation function
1.766 + return serverData->iCustomValidationFns[fnIndex];
1.767 + }
1.768 +
1.769 +/**
1.770 +Default implementation of GetIntL for CMessageParameterBase object.
1.771 +This is only called if this API is not defined for the given parameter type.
1.772 +@return KErrNone - A Dummy return value
1.773 +@leave KErrWrongParameterType in UREL if this function is not defined for the
1.774 + given parameter type
1.775 +@panic ECMPanicWrongParameterType in UDEB if this function is not defined for the
1.776 + given parameter type
1.777 +*/
1.778 +TInt CMessageParameterBase::GetIntL()
1.779 + {
1.780 +#ifdef _DEBUG
1.781 + User::Panic(KPanicCategory,ECMPanicWrongParameterType);
1.782 +#else
1.783 + User::Leave(KErrWrongParameterType);
1.784 +#endif
1.785 + return KErrNone;
1.786 + }
1.787 +
1.788 +/**
1.789 +Default implementation of GetPtrL for CMessageParameterBase object.
1.790 +This is only called if this API is not defined for the given parameter type.
1.791 +@return NULL - A Dummy return value
1.792 +@leave KErrWrongParameterType in UREL if this function is not defined for the
1.793 + given parameter type
1.794 +@panic ECMPanicWrongParameterType in UDEB if this function is not defined for the
1.795 + given parameter type
1.796 +*/
1.797 +const TAny* CMessageParameterBase::GetPtrL()
1.798 + {
1.799 +#ifdef _DEBUG
1.800 + User::Panic(KPanicCategory,ECMPanicWrongParameterType);
1.801 +#else
1.802 + User::Leave(KErrWrongParameterType);
1.803 +#endif
1.804 + return NULL;
1.805 + }
1.806 +
1.807 +/**
1.808 +Default implementation of WriteL for CMessageParameterBase object.
1.809 +This is only called if this API is not defined for the given parameter type.
1.810 +@leave KErrWrongParameterType in UREL if this function is not defined for the
1.811 + given parameter type
1.812 +@panic ECMPanicWrongParameterType in UDEB if this function is not defined for the
1.813 + given parameter type
1.814 +*/
1.815 +void CMessageParameterBase::WriteL(const TDesC8& /*aDes*/, TInt /*aOffset*/)
1.816 + {
1.817 +#ifdef _DEBUG
1.818 + User::Panic(KPanicCategory,ECMPanicWrongParameterType);
1.819 +#else
1.820 + User::Leave(KErrWrongParameterType);
1.821 +#endif
1.822 + }
1.823 +
1.824 +/**
1.825 +Default implementation of WriteL for CMessageParameterBase object.
1.826 +This is only called if this API is not defined for the given parameter type.
1.827 +@leave KErrWrongParameterType in UREL if this function is not defined for the
1.828 + given parameter type
1.829 +@panic ECMPanicWrongParameterType in UDEB if this function is not defined for the
1.830 + given parameter type
1.831 +*/
1.832 +void CMessageParameterBase::WriteL(const TDesC& /*aDes*/, TInt /*aOffset*/)
1.833 + {
1.834 +#ifdef _DEBUG
1.835 + User::Panic(KPanicCategory,ECMPanicWrongParameterType);
1.836 +#else
1.837 + User::Leave(KErrWrongParameterType);
1.838 +#endif
1.839 + }
1.840 +
1.841 +/**
1.842 +Default implementation of ReadL for CMessageParameterBase object.
1.843 +This is only called if this API is not defined for the given parameter type.
1.844 +@leave KErrWrongParameterType in UREL if this function is not defined for the
1.845 + given parameter type
1.846 +@panic ECMPanicWrongParameterType in UDEB if this function is not defined for the
1.847 + given parameter type
1.848 +*/
1.849 +void CMessageParameterBase::ReadL(TDes8& /*aDes*/,TInt /*aOffset*/)
1.850 + {
1.851 +#ifdef _DEBUG
1.852 + User::Panic(KPanicCategory,ECMPanicWrongParameterType);
1.853 +#else
1.854 + User::Leave(KErrWrongParameterType);
1.855 +#endif
1.856 + }
1.857 +
1.858 +/**
1.859 +Default implementation of ReadL for CMessageParameterBase object.
1.860 +This is only called if this API is not defined for the given parameter type.
1.861 +@leave KErrWrongParameterType in UREL if this function is not defined for the
1.862 + given parameter type
1.863 +@panic ECMPanicWrongParameterType in UDEB if this function is not defined for the
1.864 + given parameter type
1.865 +*/
1.866 +void CMessageParameterBase::ReadL(TDes& /*aDes*/, TInt /*aOffset*/)
1.867 + {
1.868 +#ifdef _DEBUG
1.869 + User::Panic(KPanicCategory,ECMPanicWrongParameterType);
1.870 +#else
1.871 + User::Leave(KErrWrongParameterType);
1.872 +#endif
1.873 + }
1.874 +
1.875 +/**
1.876 +Default implementation of GetDesLengthL for CMessageParameterBase object.
1.877 +This is only called if this API is not defined for the given parameter type.
1.878 +@return KErrNone - A Dummy return
1.879 +@leave KErrWrongParameterType in UREL if this function is not defined for the
1.880 + given parameter type
1.881 +@panic ECMPanicWrongParameterType in UDEB if this function is not defined for the
1.882 + given parameter type
1.883 +*/
1.884 +TInt CMessageParameterBase::GetDesLengthL()
1.885 + {
1.886 +#ifdef _DEBUG
1.887 + User::Panic(KPanicCategory,ECMPanicWrongParameterType);
1.888 +#else
1.889 + User::Leave(KErrWrongParameterType);
1.890 +#endif
1.891 + return KErrNone;
1.892 + }
1.893 +
1.894 +/**
1.895 +Default implementation of GetDesMaxLengthL for CMessageParameterBase object.
1.896 +This is only called if this API is not defined for the given parameter type.
1.897 +@return KErrNone - A Dummy return
1.898 +@leave KErrWrongParameterType in UREL if this function is not defined for the
1.899 + given parameter type
1.900 +@panic ECMPanicWrongParameterType in UDEB if this function is not defined for the
1.901 + given parameter type
1.902 +*/
1.903 +TInt CMessageParameterBase::GetDesMaxLengthL()
1.904 + {
1.905 +#ifdef _DEBUG
1.906 + User::Panic(KPanicCategory,ECMPanicWrongParameterType);
1.907 +#else
1.908 + User::Leave(KErrWrongParameterType);
1.909 +#endif
1.910 + return KErrNone;
1.911 + }
1.912 +
1.913 +/**
1.914 +Default implementation of GetDes8L for CMessageParameterBase object.
1.915 +This is only called if this API is not defined for the given parameter type.
1.916 +@return KErrNone - A Dummy return
1.917 +@leave KErrWrongParameterType in UREL if this function is not defined for the
1.918 + given parameter type
1.919 +@panic ECMPanicWrongParameterType in UDEB if this function is not defined for the
1.920 + given parameter type
1.921 +*/
1.922 +
1.923 +const TDesC8& CMessageParameterBase::GetDes8L()
1.924 + {
1.925 +#ifdef _DEBUG
1.926 + User::Panic(KPanicCategory,ECMPanicWrongParameterType);
1.927 +#else
1.928 + User::Leave(KErrWrongParameterType);
1.929 +#endif
1.930 + _LIT8(KDummy,"");
1.931 + return KDummy;
1.932 + }
1.933 +
1.934 +/**
1.935 +Default implementation of GetDes16L for CMessageParameterBase object.
1.936 +This is only called if this API is not defined for the given parameter type.
1.937 +@return KErrNone - A Dummy return
1.938 +@leave KErrWrongParameterType in UREL if this function is not defined for the
1.939 + given parameter type
1.940 +@panic ECMPanicWrongParameterType in UDEB if this function is not defined for the
1.941 + given parameter type
1.942 +*/
1.943 +const TDesC& CMessageParameterBase::GetDes16L()
1.944 + {
1.945 +#ifdef _DEBUG
1.946 + User::Panic(KPanicCategory,ECMPanicWrongParameterType);
1.947 +#else
1.948 + User::Leave(KErrWrongParameterType);
1.949 +#endif
1.950 +
1.951 + _LIT(KDummy,"");
1.952 + return KDummy;
1.953 + }
1.954 +
1.955 +/**
1.956 +Returns the value of iMin defined in the schema for this parameter
1.957 +@return The Min constraint for this parameter
1.958 +*/
1.959 +TInt CMessageParameterBase::Min()
1.960 + {
1.961 + return iParamDetails.iMin;
1.962 + }
1.963 +
1.964 +/**
1.965 +Returns the value of iMax defined in the schema for this parameter
1.966 +@return The max constraint for this parameter
1.967 +*/
1.968 +TInt CMessageParameterBase::Max()
1.969 + {
1.970 + return iParamDetails.iMax;
1.971 + }
1.972 +
1.973 +
1.974 +/**
1.975 +Factory function for instantiating CIntParameter objects
1.976 +@param aParam Parameter details object used to construct object.
1.977 +@param aParamIndex The Index of this parameter within the RMessage2 arguments
1.978 +@param aMessage The RMessage2 object containing the parameter represented by
1.979 +@return A fully constructed CIntParameter object.
1.980 +@leave Any system-wide error code.
1.981 +*/
1.982 +CMessageParameterBase* CIntParameter::NewL(const TParameterDetails& aParam,
1.983 + TInt aParamIndex, const RMessage2& aMessage, TCustomValidationFn aValidationFn)
1.984 + {
1.985 + CIntParameter* self = new(ELeave) CIntParameter(aParam, aParamIndex, aMessage, aValidationFn);
1.986 + return self;
1.987 + }
1.988 +
1.989 +/**
1.990 +Constructor for CIntParameter class.
1.991 +@param aParam Parameter details to be encapsulated by object
1.992 +@param aParamIndex The Index of this parameter within the RMessage2 arguments
1.993 +@param aMessage The RMessage2 object containing the parameter to be represented
1.994 +@panic ECMPanicBadMessageSchema in UDEB if the schema for this parameter is
1.995 + incorrectly defined
1.996 +*/
1.997 +CIntParameter::CIntParameter(const TParameterDetails& aParam, TInt aParamIndex,
1.998 + const RMessage2& aMessage, TCustomValidationFn aValidationFn)
1.999 + : CMessageParameterBase(aParam, aParamIndex, aMessage, aValidationFn)
1.1000 +
1.1001 + {
1.1002 + __ASSERT_DEBUG((iParamDetails.iMax >= iParamDetails.iMin),
1.1003 + PanicServer(ECMPanicBadMessageSchema));
1.1004 + }
1.1005 +
1.1006 +/**
1.1007 +Destructor for CIntParameter class.
1.1008 +*/
1.1009 +CIntParameter::~CIntParameter()
1.1010 + {
1.1011 +
1.1012 + }
1.1013 +
1.1014 +/**
1.1015 +Validates given message parameter agains constraints
1.1016 +represented by this object. Stores the Int value from the message
1.1017 + to allow for simple retrieval when required.
1.1018 +@leave KErrBadParameter if the message parameter does not conform
1.1019 + to the constraints represented by this object
1.1020 +@leave Any system-wide error code
1.1021 +*/
1.1022 +void CIntParameter::ValidateL()
1.1023 + {
1.1024 +
1.1025 + switch(iIndex)
1.1026 + {
1.1027 +
1.1028 + case 0:
1.1029 + iValue = iMessage.Int0();
1.1030 + break;
1.1031 +
1.1032 + case 1:
1.1033 + iValue = iMessage.Int1();
1.1034 + break;
1.1035 +
1.1036 + case 2:
1.1037 + iValue = iMessage.Int2();
1.1038 + break;
1.1039 +
1.1040 + case 3:
1.1041 + iValue = iMessage.Int3();
1.1042 + break;
1.1043 +
1.1044 + default:
1.1045 + User::Leave(KErrArgument);
1.1046 + break;
1.1047 + }
1.1048 +
1.1049 + if(iValidationFn != NULL)
1.1050 + {
1.1051 + iValidationFn(this);
1.1052 + }
1.1053 +
1.1054 + else
1.1055 + {
1.1056 + if((iValue < iParamDetails.iMin)||(iValue > iParamDetails.iMax))
1.1057 + {
1.1058 + User::Leave(KErrBadParameter);
1.1059 + }
1.1060 + }
1.1061 + }
1.1062 +
1.1063 +/**
1.1064 +Retrieves the TInt value read from the clients message during validation
1.1065 +@return The TInt value read from the client message
1.1066 +*/
1.1067 +TInt CIntParameter::GetIntL()
1.1068 + {
1.1069 + return iValue;
1.1070 + }
1.1071 +
1.1072 +/**
1.1073 +Factory function for instantiating CDes8ReadParameter objects
1.1074 +@param aParam Parameter details object used to construct object.
1.1075 +@param aParamIndex The Index of this parameter within the RMessage2 arguments
1.1076 +@param aMessage The RMessage2 object containing the parameter to be represented
1.1077 +@return A fully constructed CDes8ReadParameter object.
1.1078 +@leave Any system-wide error code.
1.1079 +*/
1.1080 +CMessageParameterBase* CDes8ReadParameter::NewL(const TParameterDetails& aParam,
1.1081 + TInt aParamIndex, const RMessage2& aMessage, TCustomValidationFn aValidationFn)
1.1082 + {
1.1083 + CDes8ReadParameter* self =
1.1084 + new(ELeave) CDes8ReadParameter(aParam, aParamIndex, aMessage, aValidationFn);
1.1085 +
1.1086 + return self;
1.1087 + }
1.1088 +
1.1089 +/**
1.1090 +Constructor for CDes8ReadParameter class.
1.1091 +@param aParam Parameter details to be encapsulated by object
1.1092 +@param aParamIndex The Index of this parameter within the RMessage2 arguments
1.1093 +@param aMessage The RMessage2 object containing the parameter to be represented
1.1094 +@panic ECMPanicBadMessageSchema in UDEB if the schema for this parameter is
1.1095 + incorrectly defined
1.1096 +*/
1.1097 +CDes8ReadParameter::CDes8ReadParameter(const TParameterDetails& aParam, TInt aParamIndex,
1.1098 + const RMessage2& aMessage, TCustomValidationFn aValidationFn)
1.1099 + : CMessageParameterBase(aParam, aParamIndex, aMessage, aValidationFn)
1.1100 + {
1.1101 + __ASSERT_DEBUG((iParamDetails.iMin >= 0),
1.1102 + PanicServer(ECMPanicBadMessageSchema));
1.1103 +
1.1104 + __ASSERT_DEBUG((iParamDetails.iMax > 0),
1.1105 + PanicServer(ECMPanicBadMessageSchema));
1.1106 +
1.1107 + __ASSERT_DEBUG((iParamDetails.iMax >= iParamDetails.iMin),
1.1108 + PanicServer(ECMPanicBadMessageSchema));
1.1109 + }
1.1110 +
1.1111 +/**
1.1112 +Destructor for CDes8ReadParameter class.
1.1113 +*/
1.1114 +CDes8ReadParameter::~CDes8ReadParameter()
1.1115 + {
1.1116 + delete iValue;
1.1117 + }
1.1118 +
1.1119 +/**
1.1120 +Validates given message argument against constraints
1.1121 +represented by this object. Reads in the descriptor from the
1.1122 +clients message to enable simple retrival when required.
1.1123 +@leave KErrBadDescriptor if the message parameter does not conform
1.1124 + to the constraints represented by this object
1.1125 +@leave Any system-wide error code
1.1126 +*/
1.1127 +void CDes8ReadParameter::ValidateL()
1.1128 + {
1.1129 + TInt length = iMessage.GetDesLengthL(iIndex);
1.1130 +
1.1131 + //if there is a supplied custom validation function, call that now
1.1132 + if(iValidationFn != NULL)
1.1133 + {
1.1134 + iValidationFn(this);
1.1135 + }
1.1136 +
1.1137 + else
1.1138 + {
1.1139 + if((length < iParamDetails.iMin) || (length > iParamDetails.iMax))
1.1140 + {
1.1141 + User::Leave(KErrBadDescriptor);
1.1142 + }
1.1143 + }
1.1144 +
1.1145 + iValue = HBufC8::NewL(length);
1.1146 + TPtr8 ptr = iValue->Des();
1.1147 + ReadL(ptr,0);
1.1148 + }
1.1149 +
1.1150 +/**
1.1151 +Gets the descriptor read from the clients message during validation
1.1152 +@return const reference to the local descriptor copy
1.1153 +*/
1.1154 +const TDesC8& CDes8ReadParameter::GetDes8L()
1.1155 + {
1.1156 + return *iValue;
1.1157 + }
1.1158 +
1.1159 +/**
1.1160 +Gets the length of the descriptor in the client message
1.1161 +@return The length of the descriptor
1.1162 +@leave KErrBadDescriptor if the message argument is not a descriptor type
1.1163 + Any other system wide error code
1.1164 +*/
1.1165 +TInt CDes8ReadParameter::GetDesLengthL()
1.1166 + {
1.1167 + return iMessage.GetDesLengthL(iIndex);
1.1168 + }
1.1169 +
1.1170 +/**
1.1171 +Retrieves the descriptor value read from the clients
1.1172 +message during validation
1.1173 +@param aDes The target descriptor.
1.1174 +@param aOffset The offset from the start of the clients descriptor
1.1175 +@leave KErrArgument if iIndex has a value outside the valid range, or if aOffset is negative.
1.1176 +@panic ECMPanicBadDescriptor in UDEB if the supplied descriptor is too small.
1.1177 +if the schema for this parameter is incorrectly defined
1.1178 +*/
1.1179 +void CDes8ReadParameter::ReadL(TDes8& aDes, TInt aOffset)
1.1180 + {
1.1181 + __ASSERT_DEBUG((aDes.MaxLength() >= (iMessage.GetDesLengthL(iIndex) - aOffset)),
1.1182 + PanicServer(ECMPanicBadDescriptor));
1.1183 +
1.1184 + iMessage.ReadL(iIndex,aDes,aOffset);
1.1185 + }
1.1186 +
1.1187 +/**
1.1188 +Factory function for instantiating CDes8WriteParameter objects
1.1189 +@param aParam Parameter details object used to construct object.
1.1190 +@param aParamIndex The Index of this parameter within the RMessage2 arguments
1.1191 +@param aMessage The RMessage2 object containing the parameter to be represented
1.1192 +@return A fully constructed CDes8WriteParameter object.
1.1193 +@leave Any system-wide error code.
1.1194 +*/
1.1195 +CMessageParameterBase* CDes8Parameter::NewL(const TParameterDetails& aParam,
1.1196 + TInt aParamIndex, const RMessage2& aMessage, TCustomValidationFn aValidationFn)
1.1197 + {
1.1198 + CDes8Parameter* self =
1.1199 + new(ELeave) CDes8Parameter(aParam, aParamIndex, aMessage, aValidationFn);
1.1200 +
1.1201 + return self;
1.1202 + }
1.1203 +
1.1204 +/**
1.1205 +Constructor for CDes8WriteParameter class.
1.1206 +@param aParam Parameter details to be encapsulated by object
1.1207 +@param aParamIndex The Index of this parameter within the RMessage2 arguments
1.1208 +@param aMessage The RMessage2 object containing the parameter to be represented
1.1209 +@panic ECMPanicBadMessageSchema in UDEB if the schema for this parameter is
1.1210 + incorrectly defined
1.1211 +*/
1.1212 +CDes8Parameter::CDes8Parameter(const TParameterDetails& aParam, TInt aParamIndex,
1.1213 + const RMessage2& aMessage, TCustomValidationFn aValidationFn)
1.1214 + : CMessageParameterBase(aParam, aParamIndex, aMessage, aValidationFn)
1.1215 +
1.1216 + {
1.1217 + __ASSERT_DEBUG((iParamDetails.iMin >= 0),
1.1218 + PanicServer(ECMPanicBadMessageSchema));
1.1219 +
1.1220 + __ASSERT_DEBUG((iParamDetails.iMax >= 0),
1.1221 + PanicServer(ECMPanicBadMessageSchema));
1.1222 + }
1.1223 +
1.1224 +/**
1.1225 +Destructor for CDes8WriteParameter class.
1.1226 +*/
1.1227 +CDes8Parameter::~CDes8Parameter()
1.1228 + {
1.1229 + }
1.1230 +
1.1231 +/**
1.1232 +Validates given message argument against constraints
1.1233 +represented by this object.
1.1234 +@leave KErrBadDescriptor if the message parameter does not conform
1.1235 + to the constraints represented by this object
1.1236 +@leave Any system-wide error code
1.1237 +*/
1.1238 +void CDes8Parameter::ValidateL()
1.1239 + {
1.1240 +
1.1241 + //if there is a supplied custom validation function, call that now
1.1242 + if(iValidationFn != NULL)
1.1243 + {
1.1244 + iValidationFn(this);
1.1245 + }
1.1246 +
1.1247 + else
1.1248 + {
1.1249 + TInt length = iMessage.GetDesLengthL(iIndex);
1.1250 + TInt maxLength = iMessage.GetDesMaxLengthL(iIndex);
1.1251 +
1.1252 + if((maxLength < iParamDetails.iMin)||(length > iParamDetails.iMax))
1.1253 + {
1.1254 + User::Leave(KErrBadDescriptor);
1.1255 + }
1.1256 + }
1.1257 + }
1.1258 +
1.1259 +/**
1.1260 +Gets the length of the descriptor in the client message
1.1261 +@return The length of the descriptor
1.1262 +@leave KErrBadDescriptor if the message argument is not a descriptor type
1.1263 + Any other system wide error code
1.1264 +*/
1.1265 +TInt CDes8Parameter::GetDesLengthL()
1.1266 + {
1.1267 + return iMessage.GetDesLengthL(iIndex);
1.1268 + }
1.1269 +
1.1270 +/**
1.1271 +Gets the max length of the descriptor in the client message
1.1272 +@return The max length of the descriptor
1.1273 +@leave KErrBadDescriptor if the message argument is not a descriptor type
1.1274 + Any other system wide error code
1.1275 +*/
1.1276 +TInt CDes8Parameter::GetDesMaxLengthL()
1.1277 + {
1.1278 + return iMessage.GetDesMaxLengthL(iIndex);
1.1279 + }
1.1280 +
1.1281 +
1.1282 +/**
1.1283 +Reads a descriptor from the requested message argument
1.1284 +@param aDes The target descriptor.
1.1285 +@param aOffset The offset from the start of the clients descriptor
1.1286 +@leave Any system wide error code.
1.1287 +@panic ECMPanicBadDescriptor in UDEB if the supplied descriptor is too small.
1.1288 +*/
1.1289 +void CDes8Parameter::ReadL(TDes8& aDes, TInt aOffset)
1.1290 + {
1.1291 + __ASSERT_DEBUG((aDes.MaxLength() >= (iMessage.GetDesLengthL(iIndex) - aOffset)),
1.1292 + PanicServer(ECMPanicBadDescriptor));
1.1293 +
1.1294 + iMessage.ReadL(iIndex,aDes,aOffset);
1.1295 + }
1.1296 +
1.1297 +/**
1.1298 +Validates and writes a descriptor to the requested
1.1299 +message argument
1.1300 +@param aDes The source descriptor containing the data to be written.
1.1301 +@param aOffset The offset from the start of the clients descriptor
1.1302 +@leave Any system wide error code.
1.1303 +*/
1.1304 +void CDes8Parameter::WriteL(const TDesC8& aDes, TInt aOffset)
1.1305 + {
1.1306 + iMessage.WriteL(iIndex,aDes,aOffset);
1.1307 + }
1.1308 +
1.1309 +/**
1.1310 +Factory function for instantiating CIntParameter objects
1.1311 +@param aParam Parameter details object used to construct object.
1.1312 +@param aParamIndex The Index of this parameter within the RMessage2 arguments
1.1313 +@param aMessage The RMessage2 object containing the parameter to be represented
1.1314 +@return A fully constructed CIntParameter object.
1.1315 +@leave Any system-wide error code.
1.1316 +*/
1.1317 +CMessageParameterBase* CDes16ReadParameter::NewL(const TParameterDetails& aParam,
1.1318 + TInt aParamIndex, const RMessage2& aMessage, TCustomValidationFn aValidationFn)
1.1319 + {
1.1320 + CDes16ReadParameter* self =
1.1321 + new(ELeave) CDes16ReadParameter(aParam, aParamIndex, aMessage, aValidationFn);
1.1322 +
1.1323 + return self;
1.1324 + }
1.1325 +
1.1326 +/**
1.1327 +Constructor for CDes8ReadParameter class.
1.1328 +@param aParam Parameter details to be encapsulated by object
1.1329 +@param aParamIndex The Index of this parameter within the RMessage2 arguments
1.1330 +@param aMessage The RMessage2 object containing the parameter to be represented
1.1331 +@panic ECMPanicBadMessageSchema in UDEB if the schema for this parameter is
1.1332 + incorrectly defined
1.1333 +*/
1.1334 +CDes16ReadParameter::CDes16ReadParameter(const TParameterDetails& aParam, TInt aParamIndex,
1.1335 + const RMessage2& aMessage, TCustomValidationFn aValidationFn)
1.1336 + : CMessageParameterBase(aParam, aParamIndex, aMessage, aValidationFn)
1.1337 + {
1.1338 + __ASSERT_DEBUG((iParamDetails.iMin >= 0),
1.1339 + PanicServer(ECMPanicBadMessageSchema));
1.1340 +
1.1341 + __ASSERT_DEBUG((iParamDetails.iMax > 0),
1.1342 + PanicServer(ECMPanicBadMessageSchema));
1.1343 +
1.1344 + __ASSERT_DEBUG((iParamDetails.iMax >= iParamDetails.iMin),
1.1345 + PanicServer(ECMPanicBadMessageSchema));
1.1346 + }
1.1347 +
1.1348 +/**
1.1349 +Destructor for CDes16ReadParameter class.
1.1350 +*/
1.1351 +CDes16ReadParameter::~CDes16ReadParameter()
1.1352 + {
1.1353 + delete iValue;
1.1354 + }
1.1355 +
1.1356 +/**
1.1357 +Validates given message argument against constraints
1.1358 +represented by this object. Reads in the descriptor from the
1.1359 +clients message to enable simple retrival when required.
1.1360 +@leave KErrBadDescriptor if the message parameter does not conform
1.1361 + to the constraints represented by this object
1.1362 +@leave Any system-wide error code
1.1363 +*/
1.1364 +void CDes16ReadParameter::ValidateL()
1.1365 + {
1.1366 + TInt length = iMessage.GetDesLengthL(iIndex);
1.1367 +
1.1368 + //if there is a supplied custom validation function, call that now
1.1369 + if(iValidationFn != NULL)
1.1370 + {
1.1371 + iValidationFn(this);
1.1372 + }
1.1373 +
1.1374 + else
1.1375 + {
1.1376 + if((length < iParamDetails.iMin) || (length > iParamDetails.iMax))
1.1377 + {
1.1378 + User::Leave(KErrBadDescriptor);
1.1379 + }
1.1380 + }
1.1381 +
1.1382 + iValue = HBufC::NewL(length);
1.1383 + TPtr ptr = iValue->Des();
1.1384 + ReadL(ptr,0);
1.1385 + }
1.1386 +
1.1387 +/**
1.1388 +Gets the descriptor read from the clients message during validation
1.1389 +@return const reference to the local descriptor copy
1.1390 +*/
1.1391 +const TDesC& CDes16ReadParameter::GetDes16L()
1.1392 + {
1.1393 + return *iValue;
1.1394 + }
1.1395 +
1.1396 +/**
1.1397 +Gets the length of the descriptor in the client message
1.1398 +@return The length of the descriptor
1.1399 +@leave KErrBadDescriptor if the message argument is not a descriptor type
1.1400 + Any other system wide error code
1.1401 +*/
1.1402 +TInt CDes16ReadParameter::GetDesLengthL()
1.1403 + {
1.1404 + return iMessage.GetDesLengthL(iIndex);
1.1405 + }
1.1406 +
1.1407 +/**
1.1408 +Retrieves the descriptor value read from the clients
1.1409 +message during validation
1.1410 +@param aDes The target descriptor.
1.1411 +@param aOffset The offset from the start of the clients descriptor
1.1412 +@leave KErrArgument if the suplied descriptor is too small or an invalid
1.1413 + offset is supplied
1.1414 +@panic ECMPanicBadDescriptor in UDEB if the supplied descriptor is too small.
1.1415 +*/
1.1416 +void CDes16ReadParameter::ReadL(TDes& aDes, TInt aOffset)
1.1417 + {
1.1418 + __ASSERT_DEBUG((aDes.MaxLength() >= (iMessage.GetDesLengthL(iIndex) - aOffset)),
1.1419 + PanicServer(ECMPanicBadDescriptor));
1.1420 +
1.1421 + iMessage.ReadL(iIndex,aDes,aOffset);
1.1422 + }
1.1423 +
1.1424 +/**
1.1425 +Factory function for instantiating CDes16WriteParameter objects
1.1426 +@param aParam Parameter details object used to construct object.
1.1427 +@param aParamIndex The Index of this parameter within the RMessage2 arguments
1.1428 +@param aMessage The RMessage2 object containing the parameter to be represented
1.1429 +@return A fully constructed CDes16WriteParameter object.
1.1430 +@leave Any system-wide error code.
1.1431 +*/
1.1432 +CMessageParameterBase* CDes16Parameter::NewL(const TParameterDetails& aParam,
1.1433 + TInt aParamIndex, const RMessage2& aMessage, TCustomValidationFn aValidationFn)
1.1434 + {
1.1435 + CDes16Parameter* self =
1.1436 + new(ELeave) CDes16Parameter(aParam, aParamIndex, aMessage, aValidationFn);
1.1437 +
1.1438 + return self;
1.1439 + }
1.1440 +
1.1441 +/**
1.1442 +Constructor for CDes16WriteParameter class.
1.1443 +@param aParam Parameter details to be encapsulated by object
1.1444 +@param aParamIndex The Index of this parameter within the RMessage2 arguments
1.1445 +@param aMessage The RMessage2 object containing the parameter to be represented
1.1446 +@panic ECMPanicBadMessageSchema in UDEB if the schema for this parameter is
1.1447 + incorrectly defined
1.1448 +*/
1.1449 +CDes16Parameter::CDes16Parameter(const TParameterDetails& aParam, TInt aParamIndex,
1.1450 + const RMessage2& aMessage, TCustomValidationFn aValidationFn)
1.1451 + : CMessageParameterBase(aParam, aParamIndex, aMessage, aValidationFn)
1.1452 + {
1.1453 + __ASSERT_DEBUG((iParamDetails.iMin >= 0),
1.1454 + PanicServer(ECMPanicBadMessageSchema));
1.1455 + __ASSERT_DEBUG((iParamDetails.iMax >= 0),
1.1456 + PanicServer(ECMPanicBadMessageSchema));
1.1457 + }
1.1458 +
1.1459 +/**
1.1460 +Destructor for CDes16WriteParameter class.
1.1461 +*/
1.1462 +CDes16Parameter::~CDes16Parameter()
1.1463 + {
1.1464 +
1.1465 + }
1.1466 +
1.1467 +/**
1.1468 +Validates given message argument against constraints
1.1469 +represented by this object.
1.1470 +@leave KErrBadDescriptor if the message parameter does not conform
1.1471 + to the constraints represented by this object
1.1472 +@leave Any system-wide error code
1.1473 +*/
1.1474 +void CDes16Parameter::ValidateL()
1.1475 + {
1.1476 +
1.1477 + //if there is a supplied custom validation function, call that now
1.1478 + if(iValidationFn != NULL)
1.1479 + {
1.1480 + iValidationFn(this);
1.1481 + }
1.1482 + else
1.1483 + {
1.1484 + TInt length = iMessage.GetDesLengthL(iIndex);
1.1485 + TInt maxLength = iMessage.GetDesMaxLengthL(iIndex);
1.1486 +
1.1487 + if((maxLength < iParamDetails.iMin)||(length > iParamDetails.iMax))
1.1488 + {
1.1489 + User::Leave(KErrBadDescriptor);
1.1490 + }
1.1491 + }
1.1492 + }
1.1493 +
1.1494 +/**
1.1495 +Gets the length of the descriptor in the client message
1.1496 +@return The length of the descriptor
1.1497 +@leave KErrBadDescriptor if the message argument is not a descriptor type
1.1498 + Any other system wide error code
1.1499 +*/
1.1500 +TInt CDes16Parameter::GetDesLengthL()
1.1501 + {
1.1502 + return iMessage.GetDesLengthL(iIndex);
1.1503 + }
1.1504 +
1.1505 +/**
1.1506 +Gets the max length of the descriptor in the client message
1.1507 +@return The max length of the descriptor
1.1508 +@leave KErrBadDescriptor if the message argument is not a descriptor type
1.1509 + Any other system wide error code
1.1510 +*/
1.1511 +TInt CDes16Parameter::GetDesMaxLengthL()
1.1512 + {
1.1513 + return iMessage.GetDesMaxLengthL(iIndex);
1.1514 + }
1.1515 +
1.1516 +/**
1.1517 +Reads a descriptor from the requested message argument
1.1518 +@param aDes The target descriptor.
1.1519 +@param aOffset The offset from the start of the clients descriptor
1.1520 +@leave Any system wide error code.
1.1521 +@panic ECMPanicBadDescriptor in UDEB if the supplied descriptor is too small.
1.1522 +*/
1.1523 +void CDes16Parameter::ReadL(TDes& aDes, TInt aOffset)
1.1524 + {
1.1525 + __ASSERT_DEBUG((aDes.MaxLength() >= (iMessage.GetDesLengthL(iIndex) - aOffset)),
1.1526 + PanicServer(ECMPanicBadDescriptor));
1.1527 +
1.1528 + iMessage.ReadL(iIndex,aDes,aOffset);
1.1529 + }
1.1530 +
1.1531 +/**
1.1532 +Writes a descriptor to the requested message argument
1.1533 +@param aDes The source descriptor containing the data to be written.
1.1534 +@param aOffset The offset from the start of the clients descriptor
1.1535 +@leave Any system wide error code.
1.1536 +*/
1.1537 +void CDes16Parameter::WriteL(const TDesC& aDes, TInt aOffset)
1.1538 + {
1.1539 + iMessage.WriteL(iIndex,aDes,aOffset);
1.1540 + }
1.1541 +
1.1542 +/**
1.1543 +Factory function for instantiating CPckgParameter objects
1.1544 +@param aParam Parameter details object used to construct object.
1.1545 +@param aParamIndex The Index of this parameter within the RMessage2 arguments
1.1546 +@param aMessage The RMessage2 object containing the parameter to be represented
1.1547 +@return A fully constructed CPckgParameter object.
1.1548 +@leave Any system-wide error code.
1.1549 +*/
1.1550 +CMessageParameterBase* CPckgParameter::NewL(const TParameterDetails& aParam,
1.1551 + TInt aParamIndex, const RMessage2& aMessage, TCustomValidationFn aValidationFn )
1.1552 + {
1.1553 + CPckgParameter* self =
1.1554 + new(ELeave) CPckgParameter(aParam, aParamIndex, aMessage, aValidationFn);
1.1555 +
1.1556 + return self;
1.1557 + }
1.1558 +
1.1559 +/**
1.1560 +Constructor for CPckgParameter class.
1.1561 +@param aParam Parameter details to be encapsulated by object
1.1562 +@param aParamIndex The Index of this parameter within the RMessage2 arguments
1.1563 +@param aMessage The RMessage2 object containing the parameter to be represented
1.1564 +*/
1.1565 +CPckgParameter::CPckgParameter(const TParameterDetails& aParam, TInt aParamIndex,
1.1566 + const RMessage2& aMessage, TCustomValidationFn aValidationFn)
1.1567 + : CDes8Parameter(aParam, aParamIndex, aMessage, aValidationFn)
1.1568 +
1.1569 + {
1.1570 +
1.1571 + }
1.1572 +
1.1573 +/**
1.1574 +Destructor for CPckgParameter class.
1.1575 +*/
1.1576 +CPckgParameter::~CPckgParameter()
1.1577 + {
1.1578 + }
1.1579 +
1.1580 +/**
1.1581 +Validates given message argument against constraints
1.1582 +represented by this object.
1.1583 +@leave KErrBadDescriptor if the message parameter does not conform
1.1584 + to the constraints represented by this object
1.1585 +@leave Any system-wide error code
1.1586 +*/
1.1587 +void CPckgParameter::ValidateL()
1.1588 + {
1.1589 +
1.1590 + //if there is a supplied custom validation function, call that now
1.1591 + if(iValidationFn != NULL)
1.1592 + {
1.1593 + iValidationFn(this);
1.1594 + }
1.1595 +
1.1596 + else
1.1597 + {
1.1598 + TInt length = iMessage.GetDesLengthL(iIndex);
1.1599 +
1.1600 + if((length < iParamDetails.iMin)||(length > iParamDetails.iMax))
1.1601 + {
1.1602 + User::Leave(KErrBadDescriptor);
1.1603 + }
1.1604 + }
1.1605 + }
1.1606 +
1.1607 +/**
1.1608 +Factory function for instantiating CPtrParameter objects
1.1609 +@param aParam Parameter details object used to construct object.
1.1610 +@param aParamIndex The Index of this parameter within the RMessage2 arguments
1.1611 +@param aMessage The RMessage2 object containing the parameter to be represented
1.1612 +@return A fully constructed CPtrParameter object.
1.1613 +@leave Any system-wide error code.
1.1614 +*/
1.1615 +CMessageParameterBase* CPtrParameter::NewL(const TParameterDetails& aParam,
1.1616 + TInt aParamIndex, const RMessage2& aMessage, TCustomValidationFn aValidationFn)
1.1617 + {
1.1618 + CPtrParameter* self = new(ELeave) CPtrParameter(aParam, aParamIndex, aMessage, aValidationFn);
1.1619 + return self;
1.1620 + }
1.1621 +
1.1622 +/**
1.1623 +Constructor for CPtrParameter class.
1.1624 +@param aParamIndex The Index of this parameter within the RMessage2 arguments
1.1625 +@param aMessage The RMessage2 object containing the parameter to be represented
1.1626 +*/
1.1627 +CPtrParameter::CPtrParameter(const TParameterDetails& aParam, TInt aParamIndex,
1.1628 + const RMessage2& aMessage, TCustomValidationFn aValidationFn)
1.1629 + : CMessageParameterBase(aParam, aParamIndex, aMessage, aValidationFn)
1.1630 + {
1.1631 +
1.1632 + }
1.1633 +
1.1634 +/**
1.1635 +Validates given message argument against constraints
1.1636 +represented by this object. Stores the TAny* from the
1.1637 +clients message to enable simple retrival when required.
1.1638 +@leave KErrArgument if the argument index is invalid
1.1639 +@leave Any system-wide error code
1.1640 +*/
1.1641 +void CPtrParameter::ValidateL()
1.1642 + {
1.1643 +
1.1644 + switch(iIndex)
1.1645 + {
1.1646 +
1.1647 + case 0:
1.1648 + iValue = iMessage.Ptr0();
1.1649 + break;
1.1650 +
1.1651 + case 1:
1.1652 + iValue = iMessage.Ptr1();
1.1653 + break;
1.1654 +
1.1655 + case 2:
1.1656 + iValue = iMessage.Ptr2();
1.1657 + break;
1.1658 +
1.1659 + case 3:
1.1660 + iValue = iMessage.Ptr3();
1.1661 + break;
1.1662 +
1.1663 + default:
1.1664 + User::Leave(KErrArgument);
1.1665 + break;
1.1666 + }
1.1667 +
1.1668 + //if there is a supplied custom validation function, call that now
1.1669 + if(iValidationFn != NULL)
1.1670 + {
1.1671 + iValidationFn(this);
1.1672 + }
1.1673 + }
1.1674 +
1.1675 +/**
1.1676 +Retrieves the TAny pointer read from the clients message during validation
1.1677 +@return The TAny pointer read from the client message
1.1678 +*/
1.1679 +const TAny* CPtrParameter::GetPtrL()
1.1680 + {
1.1681 + return iValue;
1.1682 + }
1.1683 +
1.1684 +/**
1.1685 +Decode the string
1.1686 +@param aSrcString Source string
1.1687 +@param rDestString Destination string
1.1688 +@return 1 if aSrcString is not long enough to decode fully, resulting in the storage of
1.1689 + the last character and requiring another aSrcString (poss 0 length) to be passed to it to
1.1690 + clear this character.
1.1691 +@return 0 if the line was decoded OK or the end of the encoded file is reached ie "="
1.1692 +*/
1.1693 +
1.1694 +EXPORT_C TInt Base64Codec::Decode(const TDesC8& aSrcString, TDes8& rDestString)
1.1695 + {
1.1696 + TInt shiftStored = 0;
1.1697 + TInt maskShiftStored = ESix;
1.1698 +
1.1699 + TInt decodedInt=0;
1.1700 + TInt8 offsetChar=0;
1.1701 + TUint8 decodedChar=0;
1.1702 +
1.1703 + // Clears the destination string
1.1704 + rDestString.Zero();
1.1705 +
1.1706 + // Initialise variables
1.1707 + const TUint8* srcStringPtr=aSrcString.Ptr();
1.1708 + const TUint8* srcStringEnd=aSrcString.Length()+srcStringPtr;
1.1709 + TUint8* destStringPtr=(TUint8*)rDestString.Ptr();
1.1710 + TUint8* destStringPtrBase=destStringPtr;
1.1711 +
1.1712 + TInt maskShift=maskShiftStored;
1.1713 + TInt shiftStorage=shiftStored;
1.1714 +
1.1715 + // Main character process loop
1.1716 + while(srcStringPtr<srcStringEnd)
1.1717 + {
1.1718 + offsetChar=(TInt8)(*srcStringPtr-KImcvLookUpStartOffset);
1.1719 + srcStringPtr++;
1.1720 +
1.1721 + // Check for valid B64 character
1.1722 + if((offsetChar>=0)&&(offsetChar<80))
1.1723 + {
1.1724 + // Read in next character and B64 decode
1.1725 + decodedInt=AsciiToBase64[offsetChar];
1.1726 +
1.1727 + // Exits when a PAD char is reached
1.1728 + if(decodedInt==EPadChar)
1.1729 + {
1.1730 + rDestString.SetLength((TInt)(destStringPtr-destStringPtrBase));
1.1731 + return EFalse;
1.1732 + }
1.1733 +
1.1734 + // Ensures the first 2 chars of 4 are received before processing
1.1735 + if(maskShift==ESix)
1.1736 + maskShift=EFour;
1.1737 + else
1.1738 + {
1.1739 + shiftStorage=shiftStorage<<ESix;
1.1740 + shiftStorage=shiftStorage|decodedInt;
1.1741 + decodedChar=(TUint8)((shiftStorage>>maskShift)&EEightBitMask);
1.1742 +
1.1743 + if((maskShift-=ETwo)<EZero)
1.1744 + maskShift=ESix;
1.1745 +
1.1746 + *destStringPtr++=decodedChar;
1.1747 + }
1.1748 + shiftStorage=decodedInt;
1.1749 + }
1.1750 + }
1.1751 + shiftStored=shiftStorage;
1.1752 + maskShiftStored=maskShift;
1.1753 +
1.1754 + rDestString.SetLength((TInt)(destStringPtr-destStringPtrBase));
1.1755 +
1.1756 + return maskShift<ESix;
1.1757 + }
1.1758 +
1.1759 +/**
1.1760 +Encode the string
1.1761 +@param aSrcString Source string
1.1762 +@param rDestString Destination string
1.1763 +@return 1 if aSrcString is not long enough to encode fully
1.1764 +@return 0 if the line was encoded OK
1.1765 +*/
1.1766 +EXPORT_C TInt Base64Codec::Encode(const TDesC8& aSrcString, TDes8& rDestString)
1.1767 + {
1.1768 + // Clears the destination string
1.1769 + rDestString.Zero();
1.1770 +
1.1771 + // Initialise variables
1.1772 + const TUint8* srcStringPtr=aSrcString.Ptr();
1.1773 + const TUint8* srcStringEnd=aSrcString.Length()+srcStringPtr;
1.1774 + TUint8* destStringPtr=(TUint8*)rDestString.Ptr();
1.1775 + TUint8* destStringPtrBase=destStringPtr;
1.1776 +
1.1777 + TInt character=0;
1.1778 + TUint8 encodedChar=0;
1.1779 + TInt charStorage=0;
1.1780 + TInt maskShift=EZero;
1.1781 + TInt destStringCharNum = 0;
1.1782 +
1.1783 + while(srcStringPtr<=srcStringEnd)
1.1784 + {
1.1785 + // maskShift is used as a char read counter
1.1786 + if(maskShift==ESix)
1.1787 + {
1.1788 + // If the 3rd char read is also the last char then the while loop
1.1789 + // is broken on the next check.
1.1790 + if(srcStringPtr==srcStringEnd)
1.1791 + srcStringPtr++;
1.1792 + maskShift=EZero;
1.1793 + character=0;
1.1794 + }
1.1795 + else
1.1796 + {
1.1797 + if(srcStringPtr==srcStringEnd)
1.1798 + character=0;
1.1799 + else
1.1800 + character=*srcStringPtr;
1.1801 +
1.1802 + srcStringPtr++;
1.1803 + // Shifts charStorage ready for the next char
1.1804 + charStorage=charStorage<<8;
1.1805 + maskShift+=ETwo;
1.1806 + }
1.1807 + charStorage=charStorage|character;
1.1808 + // Shifts the mask to the correct bit location
1.1809 + // Masks (AND's) the valid bits from charStorage
1.1810 + // Shifts the valid bits into the low order 8bits
1.1811 + // Converts to BASE64 char, Casts the result to an unsigned char (which it should be ?....I hope)
1.1812 + encodedChar=(TUint8)Base64ToAscii[((charStorage>>maskShift)&ESixBitMask)];
1.1813 +
1.1814 + *destStringPtr++=encodedChar;
1.1815 + destStringCharNum++;
1.1816 +
1.1817 + // Add a CRLF every KMaxB64EncodedCharsPerLine characters so as not to exceed the line length
1.1818 + // limitation specified in RFC 2822.
1.1819 + if (destStringCharNum == KMaxB64EncodedCharsPerLine)
1.1820 + {
1.1821 + destStringCharNum = 0;
1.1822 + *destStringPtr++ = '\r';
1.1823 + *destStringPtr++ = '\n';
1.1824 + }
1.1825 + }
1.1826 +
1.1827 + // Check for not enough chars and pad if required
1.1828 + if (maskShift==EFour)
1.1829 + {
1.1830 + *destStringPtr++=KImcvConvEquals;
1.1831 + *destStringPtr++=KImcvConvEquals;
1.1832 + }
1.1833 + else
1.1834 + if(maskShift==ESix)
1.1835 + *destStringPtr++=KImcvConvEquals;
1.1836 +
1.1837 + rDestString.SetLength((TInt)(destStringPtr-destStringPtrBase));
1.1838 + return ((TInt)(srcStringPtr-srcStringEnd));
1.1839 + }