smsprotocols/smsstack/gsmu/src/gsmupriv.cpp
author Pat Downey <patd@symbian.org>
Wed, 01 Sep 2010 12:40:21 +0100
branchRCL_3
changeset 20 07a122eea281
parent 19 630d2f34d719
permissions -rw-r--r--
Revert incorrect RCL_3 drop: Revision: 201035 Kit: 201035

// Copyright (c) 2000-2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Nokia Corporation - initial contribution.
//
// Contributors:
//
// Description:
// Implements utility classes private to the GSMU dll
// 
//

/**
 @file
*/

#include "gsmupriv.h"
#include "Gsmumain.h"
#include <exterror.h>

//
// TSmsPacker - packs and unpacks data encoded in an SMS alphabet
//

/**
 *  Constructor
 */
TSmsAlphabetPacker::TSmsAlphabetPacker(TSmsDataCodingScheme::TSmsAlphabet aAlphabet,TBool aIsBinary,TInt aStartBit)
	: iAlphabet(aAlphabet),iIsBinary(aIsBinary),iStartBit(aStartBit)
	{
	// NOP
	} // TSmsAlphabetPacker::TSmsAlphabetPacker


/**
 *  Packs user data units from aIn and appends to aOut.
 */
TInt TSmsAlphabetPacker::PackL(TDes8& aOut,const TDesC8& aIn)
	{
	LOGGSMU1("TSmsAlphabetPacker::PackL()");

	// Ensure we've got the right length
	TInt packedOctetsRequired=PackedOctetsRequiredL(aIn.Length());
	if (packedOctetsRequired>(aOut.MaxLength()-aOut.Length()))
		User::Leave(KErrOverflow);
	// Do the conversion
	TInt elementSizeInBits=ElementSizeInBitsL();
	if (elementSizeInBits==8)
		{
		// Straight copy here
		aOut.Append(aIn);
		}
	else if (elementSizeInBits==7)
		{
		// Get raw pointers and do packing
		TUint8* outPtr=(TUint8*)aOut.Ptr()+aOut.Length();
		const TUint8* inPtr=aIn.Ptr();

		outPtr[0]=0;
		for (TInt i=0; i<aIn.Length(); i++)
			{
			TUint8 to=inPtr[i];
			*outPtr|=(to<<iStartBit);
			if (iStartBit)
				{
				outPtr++;
				*outPtr=(TUint8) (to>>(8-iStartBit));
				}
			iStartBit=(iStartBit+7)%8;
			}
		// Increment the length for the packed data
		aOut.SetLength(aOut.Length()+packedOctetsRequired);
		}
	else
		{
		__ASSERT_DEBUG(EFalse,Panic(KGsmuPanicPackAlphabetInvalid));
		}
	// Return number of bytes used
	return packedOctetsRequired;
	} // TSmsAlphabetPacker::PackL


/**
 *  Unpacks user data units from aIn and appends to aOut.
 */
TInt TSmsAlphabetPacker::UnpackL(const TDesC8& aIn,TDes8& aOut,TInt aNumUDUnits)
	{
	LOGGSMU1("TSmsAlphabetPacker::UnpackL()");

	TInt length=aNumUDUnits;
	// Ensure we've got enough input and output buffer
 	// defect fix for
 	// EDNOPMA-4YPJ34	Short message with alphanumeric address in From field is not received
 	// aIn.length is constant (KSmsAddressMaxAddressValueLength = 10)
 	// PackedOctetsRequiredL is dependant on aNumUDUnits
 	// otheriwse if a short alphanumeric address is received GSMU leaves
 	// and the smsprot returns protocol error to the service center
 	// if (PackedOctetsRequiredL(aIn.Length())>aNumUDUnits)
 	if (PackedOctetsRequiredL(aNumUDUnits)>aIn.Length())
		User::Leave(KErrCorrupt);
	if (aOut.Length()+length>aOut.MaxLength())
		User::Leave(KErrCorrupt);
	TInt elementSizeInBits=ElementSizeInBitsL();
	if (elementSizeInBits==8)
		{
		aOut.Append(aIn);
		}
	else if (elementSizeInBits==7)
		{
		// Get raw pointers and do packing
		TUint8* outPtr=(TUint8*)aOut.Ptr()+aOut.Length();
		const TUint8* inPtr=aIn.Ptr();

		for (TInt i=0; i<length; i++)
			{
			TInt from=(*inPtr>>iStartBit) & 0x7F;
			if (iStartBit)
				{
				inPtr++;
				from|=(*inPtr<<(8-iStartBit)) & 0x7F;
				}
			outPtr[i]=(TUint8) from;
			iStartBit=(iStartBit+7)%8;
			}
		aOut.SetLength(aOut.Length()+length);
		}
	else
		{
		__ASSERT_DEBUG(EFalse,Panic(KGsmuPanicPackAlphabetInvalid));
		}
	return length;
	} // TSmsAlphabetPacker::UnpackL


/**
 *  Converts then packs the input data, aIn, and appends to aOut
 */
TInt TSmsAlphabetPacker::ConvertAndPackL(CCnvCharacterSetConverter& aCharacterSetConverter,RFs& aFs,TDes8& aOut,const TDesC& aIn,TInt& aConvertedNumUDUnits)
    {
	LOGGSMU1("TSmsAlphabetPacker::ConvertAndPackL()");

	// Do the conversion
	// VEP Fix for defect EXT-568BMW, when length of alphanumeric destination address
	// was set wrong and also special characters defined in 7-bit default aplhabet (ä,ö...)
	// were converted incorrectly
	CSmsAlphabetConverter* converter=CSmsAlphabetConverter::NewLC(aCharacterSetConverter,aFs,iAlphabet,iIsBinary);
	TPtrC8 convertedPtr=converter->ConvertFromNativeL(aIn);
	aConvertedNumUDUnits=convertedPtr.Length();
	// Do the packing
	TInt octetsUsed=PackL(aOut,convertedPtr);
	// Cleanup and return
	CleanupStack::PopAndDestroy(converter);
	return octetsUsed;
    } // TSmsAlphabetPacker::ConvertAndPackL


/**
 *  Unpacks the converts the input data, aIn, and appends to aOut
 */
TInt TSmsAlphabetPacker::UnpackAndConvertL(CCnvCharacterSetConverter& aCharacterSetConverter,RFs& aFs,const TDesC8& aIn,TDes& aOut,TInt aNumUDUnits)
	{
	LOGGSMU1("TSmsAlphabetPacker::UnpackAndConvertL()");

	// Unpack first
	HBufC8* unpackedBuffer=HBufC8::NewLC(aNumUDUnits);
	TPtr8 unpackedBufferPtr(unpackedBuffer->Des());
	UnpackL(aIn,unpackedBufferPtr,aNumUDUnits);
	// Convert
	CSmsAlphabetConverter* converter=CSmsAlphabetConverter::NewLC(aCharacterSetConverter,aFs,iAlphabet,iIsBinary);
	TPtrC convertedPtr=converter->ConvertToNativeL(*unpackedBuffer);
	if (convertedPtr.Length()>(aOut.MaxLength()-aOut.Length()))
		User::Leave(KErrCorrupt);
	// Cleanup and return
	aOut.Append(convertedPtr);
	CleanupStack::PopAndDestroy(2);	// unpackedBuffer,converter
	return aNumUDUnits;
	} // TSmsAlphabetPacker::UnpackAndConvertL


/**
 *  Returns the number of octets needed to pack the specified number of
 */
TInt TSmsAlphabetPacker::PackedOctetsRequiredL(TInt aNumUDUnits) const
	{
	LOGGSMU1("TSmsAlphabetPacker::PackedOctetsRequiredL()");

	TInt octetsRequired=0;
	TInt elementSizeInBits=ElementSizeInBitsL();
	if (elementSizeInBits==8)
		octetsRequired=aNumUDUnits;
	else
		octetsRequired=(iStartBit+aNumUDUnits*elementSizeInBits + 7)/8;	// Rounds up
	return octetsRequired;
	} // TSmsAlphabetPacker::PackedOctetsRequiredL

/**
 * Returns the number of UD units that are packed in the specified number of octets
 */
TInt TSmsAlphabetPacker::NumUDUnitsL(TInt aOctets) const
	{
	TInt numUD=0;
	TInt elementSizeInBits=ElementSizeInBitsL();
	if (elementSizeInBits==8)
		numUD=aOctets;
	else
		numUD=(8*aOctets - iStartBit)/elementSizeInBits;
	return numUD;
	}

/**
 *  Returns the size in bits of a UDL element for the alphabet.  Leaves if
 *  invalid data coding scheme.
 */
TInt TSmsAlphabetPacker::ElementSizeInBitsL() const
	{
	LOGGSMU1("TSmsAlphabetPacker::ElementSizeInBitsL()");

    TInt ret = 8;

	if (iIsBinary)
		return ret;
	switch (iAlphabet)
		{
		case TSmsDataCodingScheme::ESmsAlphabet7Bit:
			{
			ret = 7;
            break;
			}
		case TSmsDataCodingScheme::ESmsAlphabet8Bit:
		case TSmsDataCodingScheme::ESmsAlphabetUCS2:
			{
			ret = 8;
            break;
			}
		default:
			{
            User::Leave(KErrGsmSMSDataCodingSchemeNotSupported);
			}
		}
    return ret;
	} // TSmsAlphabetPacker::ElementSizeInBitsL