cryptoservices/certificateandkeymgmt/asn1/bitstrenc.cpp
author William Roberts <williamr@symbian.org>
Sat, 27 Feb 2010 23:09:01 +0000
branchRCL_3
changeset 44 ea00662efad9
parent 8 35751d3474b7
permissions -rw-r--r--
Re-merge fix for Bug 1301

/*
* Copyright (c) 2002-2009 Nokia Corporation and/or its subsidiary(-ies).
* All rights reserved.
* This component and the accompanying materials are made available
* under the terms of the License "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: 
* This file contains the implementation for class encoding bit strings in ASN1 DER.
*
*/


/**
 @file
*/

#include <asn1enc.h>

EXPORT_C CASN1EncBitString* CASN1EncBitString::NewLC(const TDesC8& aBitStr)
	{
	CASN1EncBitString* self = new (ELeave) CASN1EncBitString();
	CleanupStack::PushL(self);
	self->ConstructL(aBitStr);
	return self;
	}

EXPORT_C CASN1EncBitString* CASN1EncBitString::NewL(const TDesC8& aBitStr)
	{
	CASN1EncBitString* self = NewLC(aBitStr);
	CleanupStack::Pop(self);
	return self;
	}

EXPORT_C CASN1EncBitString* CASN1EncBitString::NewLC(const TDesC8& aBitStr, 
													 TUint aLengthBits)
	{
	CASN1EncBitString* self = new (ELeave) CASN1EncBitString();
	CleanupStack::PushL(self);
	self->ConstructL(aBitStr, aLengthBits);
	return self;
	}

EXPORT_C CASN1EncBitString* CASN1EncBitString::NewL(const TDesC8& aBitStr, 
													TUint aLengthBits)
	{
	CASN1EncBitString* self = NewLC(aBitStr, aLengthBits);
	CleanupStack::Pop(self);
	return self;
	}

EXPORT_C CASN1EncBitString* CASN1EncBitString::NewL(const CASN1EncBase& aEncObj)
	{
	CASN1EncBitString* self = NewLC(aEncObj);
	CleanupStack::Pop(self);
	return self;
	}

EXPORT_C CASN1EncBitString* CASN1EncBitString::NewLC(const CASN1EncBase& aEncObj)
	{
	CASN1EncBitString* self = new (ELeave) CASN1EncBitString();
	CleanupStack::PushL(self);
	self->ConstructL(aEncObj);
	return self;
	}

EXPORT_C CASN1EncBitString::~CASN1EncBitString()
	{
	delete iContents;
	}

CASN1EncBitString::CASN1EncBitString()
: CASN1EncPrimitive(EASN1BitString),
  iPadding(0)
	{
	}

void CASN1EncBitString::ConstructL(const TDesC8& aBitStr)
	{
	iContents = aBitStr.AllocL();
	CASN1EncPrimitive::ConstructL();
	}

void CASN1EncBitString::ConstructL(const TDesC8& aBitStr, TUint aLengthBits)
	{
	iContents = aBitStr.AllocL();
	CASN1EncPrimitive::ConstructL();
	TUint totalBits = aBitStr.Length() * 8;
	__ASSERT_ALWAYS(aLengthBits <= totalBits, User::Leave(KErrArgument));
	iPadding = (TUint8)(totalBits - aLengthBits);
	if(iPadding > 7)
		User::Leave(KErrArgument);
	}

/**
 * @internalTechnology
 * Constructs bit string from ASN.1 encoding object.
 * @param aEncObj ASN.1 encoding object to wrap in bit string.
 * @note First produces raw DER encoding from the object, then creates 
 *     a bit string using other construct function.
 */
void CASN1EncBitString::ConstructL(const CASN1EncBase& aEncObj)
	{
	// produce raw DER encoding from the created ASN.1 encoding
	TUint len = aEncObj.LengthDER();
	HBufC8* intDer = HBufC8::NewMaxLC(len);
	TPtr8 ptrDer = intDer->Des();
	TUint pos = 0;
	aEncObj.WriteDERL(ptrDer, pos);
	// wrap the produced DER encoding into a bit string
	ConstructL(*intDer);
	// cleanup
	CleanupStack::PopAndDestroy(intDer);
	}

/** 
 * @internalTechnology 
 * Calculates length of DER-encoded bit string contents. For non empty
 * bit strings this differs from octet string in the extra leading byte 
 * containing the number of unused bits in the last octet.
 */
void CASN1EncBitString::CalculateContentsLengthDER()
	{
	iContentsLengthDER = iContents->Length();
	if (iContentsLengthDER > 0)
		{
		iContentsLengthDER++;
		}
	}

/**
 * @internalTechnology 
 * Writes DER-encoded bit string contents to aBuf. Prepends 
 * the actual bit string octets with extra octet containing 
 * number of unused bits in the last octet. Before writing, 
 * converts contents of the bit string to big-endian form.
 * @param aBuf Buffer to write to; must be long enough; 
 *     see #CalculateContentsLengthDER method.
 */
void CASN1EncBitString::WriteContentsDERL(TDes8& aBuf) const
	{
	if (iContents->Length() > 0)
		{
		aBuf[0] = iPadding;
		TUint len = iContents->Length();
		// We do not need to swap bits, as it is already done.
		aBuf.Replace(1, len, *iContents);		
		}
	else
		{
		// no padding octet for the empty bit string
		aBuf.SetLength(0);
		}
	}