ext-cryptopp/asn.h

576 lines
20 KiB
C
Raw Normal View History

// asn.h - originally written and placed in the public domain by Wei Dai
2015-11-05 06:59:46 +00:00
2016-04-08 05:51:21 +00:00
//! \file asn.h
2015-11-05 06:59:46 +00:00
//! \brief Classes and functions for working with ANS.1 objects
#ifndef CRYPTOPP_ASN_H
#define CRYPTOPP_ASN_H
#include "cryptlib.h"
#include "filters.h"
#include "smartptr.h"
#include "stdcpp.h"
#include "queue.h"
#include "misc.h"
// Issue 340
#if CRYPTOPP_GCC_DIAGNOSTIC_AVAILABLE
# pragma GCC diagnostic push
# pragma GCC diagnostic ignored "-Wconversion"
# pragma GCC diagnostic ignored "-Wsign-conversion"
#endif
2015-11-05 06:59:46 +00:00
NAMESPACE_BEGIN(CryptoPP)
2016-04-08 05:51:21 +00:00
//! \brief ASN.1 types
//! \note These tags and flags are not complete
2015-11-05 06:59:46 +00:00
enum ASNTag
{
BOOLEAN = 0x01,
INTEGER = 0x02,
BIT_STRING = 0x03,
OCTET_STRING = 0x04,
TAG_NULL = 0x05,
OBJECT_IDENTIFIER = 0x06,
OBJECT_DESCRIPTOR = 0x07,
EXTERNAL = 0x08,
REAL = 0x09,
ENUMERATED = 0x0a,
UTF8_STRING = 0x0c,
SEQUENCE = 0x10,
SET = 0x11,
NUMERIC_STRING = 0x12,
PRINTABLE_STRING = 0x13,
T61_STRING = 0x14,
VIDEOTEXT_STRING = 0x15,
IA5_STRING = 0x16,
UTC_TIME = 0x17,
GENERALIZED_TIME = 0x18,
GRAPHIC_STRING = 0x19,
VISIBLE_STRING = 0x1a,
GENERAL_STRING = 0x1b
};
2016-04-08 05:51:21 +00:00
//! \brief ASN.1 flags
//! \note These tags and flags are not complete
2015-11-05 06:59:46 +00:00
enum ASNIdFlag
{
UNIVERSAL = 0x00,
// DATA = 0x01,
// HEADER = 0x02,
PRIMITIVE = 0x00,
CONSTRUCTED = 0x20,
APPLICATION = 0x40,
CONTEXT_SPECIFIC = 0x80,
PRIVATE = 0xc0
2015-11-05 06:59:46 +00:00
};
2016-04-08 05:51:21 +00:00
//! \brief Raises a BERDecodeErr
2015-11-05 06:59:46 +00:00
inline void BERDecodeError() {throw BERDecodeErr();}
2016-04-08 05:51:21 +00:00
//! \brief Exception thrown when an unknown object identifier is encountered
2015-11-05 06:59:46 +00:00
class CRYPTOPP_DLL UnknownOID : public BERDecodeErr
{
public:
2016-04-08 05:51:21 +00:00
//! \brief Construct an UnknownOID
2015-11-05 06:59:46 +00:00
UnknownOID() : BERDecodeErr("BER decode error: unknown object identifier") {}
2016-04-08 05:51:21 +00:00
//! \brief Construct an UnknownOID
//! \param err error message to use for the execption
2015-11-05 06:59:46 +00:00
UnknownOID(const char *err) : BERDecodeErr(err) {}
};
// unsigned int DERLengthEncode(unsigned int length, byte *output=0);
2016-04-08 05:51:21 +00:00
//! \brief DER encode a length
2016-04-20 23:03:46 +00:00
//! \param bt BufferedTransformation object for writing
2016-04-08 05:51:21 +00:00
//! \param length the size to encode
//! \returns the number of octets used for the encoding
2016-04-20 23:03:46 +00:00
CRYPTOPP_DLL size_t CRYPTOPP_API DERLengthEncode(BufferedTransformation &bt, lword length);
2016-04-08 05:51:21 +00:00
//! \brief BER decode a length
2016-04-20 23:03:46 +00:00
//! \param bt BufferedTransformation object for reading
2016-04-08 05:51:21 +00:00
//! \param length the decoded size
2016-09-10 08:57:48 +00:00
//! \returns true if the value was decoded
2016-04-08 05:51:21 +00:00
//! \throws BERDecodeError if the value fails to decode or is too large for size_t
//! \details BERLengthDecode() returns false if the encoding is indefinite length.
2016-04-20 23:03:46 +00:00
CRYPTOPP_DLL bool CRYPTOPP_API BERLengthDecode(BufferedTransformation &bt, size_t &length);
2015-11-05 06:59:46 +00:00
2016-04-08 05:51:21 +00:00
//! \brief DER encode NULL
2016-04-20 23:03:46 +00:00
//! \param bt BufferedTransformation object for writing
CRYPTOPP_DLL void CRYPTOPP_API DEREncodeNull(BufferedTransformation &bt);
2016-04-08 05:51:21 +00:00
//! \brief BER decode NULL
2016-04-20 23:03:46 +00:00
//! \param bt BufferedTransformation object for reading
CRYPTOPP_DLL void CRYPTOPP_API BERDecodeNull(BufferedTransformation &bt);
2015-11-05 06:59:46 +00:00
2016-04-08 05:51:21 +00:00
//! \brief DER encode octet string
2016-04-20 23:03:46 +00:00
//! \param bt BufferedTransformation object for writing
2016-04-08 05:51:21 +00:00
//! \param str the string to encode
//! \param strLen the length of the string
//! \returns the number of octets used for the encoding
2016-04-20 23:03:46 +00:00
CRYPTOPP_DLL size_t CRYPTOPP_API DEREncodeOctetString(BufferedTransformation &bt, const byte *str, size_t strLen);
2016-04-08 05:51:21 +00:00
//! \brief DER encode octet string
2016-04-20 23:03:46 +00:00
//! \param bt BufferedTransformation object for reading
2016-04-08 05:51:21 +00:00
//! \param str the string to encode
//! \returns the number of octets used for the encoding
2016-04-20 23:03:46 +00:00
CRYPTOPP_DLL size_t CRYPTOPP_API DEREncodeOctetString(BufferedTransformation &bt, const SecByteBlock &str);
2016-04-08 05:51:21 +00:00
//! \brief BER decode octet string
2016-04-20 23:03:46 +00:00
//! \param bt BufferedTransformation object for reading
2016-04-08 05:51:21 +00:00
//! \param str the decoded string
//! \returns the number of octets used for the encoding
2016-04-20 23:03:46 +00:00
CRYPTOPP_DLL size_t CRYPTOPP_API BERDecodeOctetString(BufferedTransformation &bt, SecByteBlock &str);
2016-04-08 05:51:21 +00:00
//! \brief BER decode octet string
2016-04-20 23:03:46 +00:00
//! \param bt BufferedTransformation object for reading
2016-04-08 05:51:21 +00:00
//! \param str the decoded string
//! \returns the number of octets used for the encoding
2016-04-20 23:03:46 +00:00
CRYPTOPP_DLL size_t CRYPTOPP_API BERDecodeOctetString(BufferedTransformation &bt, BufferedTransformation &str);
2015-11-05 06:59:46 +00:00
2016-04-08 05:51:21 +00:00
//! \brief DER encode text string
2016-04-20 23:03:46 +00:00
//! \param bt BufferedTransformation object for writing
2016-04-08 05:51:21 +00:00
//! \param str the string to encode
//! \param asnTag the ASN.1 type
//! \returns the number of octets used for the encoding
//! \details DEREncodeTextString() can be used for UTF8_STRING, PRINTABLE_STRING, and IA5_STRING
2016-04-20 23:03:46 +00:00
CRYPTOPP_DLL size_t CRYPTOPP_API DEREncodeTextString(BufferedTransformation &bt, const std::string &str, byte asnTag);
2016-04-08 05:51:21 +00:00
//! \brief BER decode text string
2016-04-20 23:03:46 +00:00
//! \param bt BufferedTransformation object for reading
2016-04-08 05:51:21 +00:00
//! \param str the string to encode
//! \param asnTag the ASN.1 type
//! \details DEREncodeTextString() can be used for UTF8_STRING, PRINTABLE_STRING, and IA5_STRING
2016-04-20 23:03:46 +00:00
CRYPTOPP_DLL size_t CRYPTOPP_API BERDecodeTextString(BufferedTransformation &bt, std::string &str, byte asnTag);
2015-11-05 06:59:46 +00:00
2016-04-08 05:51:21 +00:00
//! \brief DER encode bit string
2016-04-20 23:03:46 +00:00
//! \param bt BufferedTransformation object for writing
2016-04-08 05:51:21 +00:00
//! \param str the string to encode
//! \param strLen the length of the string
//! \param unusedBits the number of unused bits
//! \returns the number of octets used for the encoding
2016-04-20 23:03:46 +00:00
CRYPTOPP_DLL size_t CRYPTOPP_API DEREncodeBitString(BufferedTransformation &bt, const byte *str, size_t strLen, unsigned int unusedBits=0);
2016-04-08 05:51:21 +00:00
//! \brief DER decode bit string
2016-04-20 23:03:46 +00:00
//! \param bt BufferedTransformation object for reading
2016-04-08 05:51:21 +00:00
//! \param str the decoded string
//! \param unusedBits the number of unused bits
2016-04-20 23:03:46 +00:00
CRYPTOPP_DLL size_t CRYPTOPP_API BERDecodeBitString(BufferedTransformation &bt, SecByteBlock &str, unsigned int &unusedBits);
2015-11-05 06:59:46 +00:00
2016-04-08 05:51:21 +00:00
//! \brief BER decode and DER re-encode
2016-04-20 23:03:46 +00:00
//! \param bt BufferedTransformation object for writing
2016-04-08 05:51:21 +00:00
//! \param dest BufferedTransformation object
2016-04-20 23:03:46 +00:00
CRYPTOPP_DLL void CRYPTOPP_API DERReencode(BufferedTransformation &bt, BufferedTransformation &dest);
2015-11-05 06:59:46 +00:00
2016-04-08 05:51:21 +00:00
//! \brief Object Identifier
2015-11-05 06:59:46 +00:00
class CRYPTOPP_DLL OID
{
public:
virtual ~OID() {}
2016-04-08 05:51:21 +00:00
//! \brief Construct an OID
2015-11-05 06:59:46 +00:00
OID() {}
2016-04-08 05:51:21 +00:00
//! \brief Construct an OID
//! \param v value to initialize the OID
2015-11-05 06:59:46 +00:00
OID(word32 v) : m_values(1, v) {}
2016-04-08 05:51:21 +00:00
//! \brief Construct an OID
//! \param bt BufferedTransformation object
2015-11-05 06:59:46 +00:00
OID(BufferedTransformation &bt) {BERDecode(bt);}
2016-04-08 05:51:21 +00:00
//! \brief Append a value to an OID
//! \param rhs the value to append
2015-11-05 06:59:46 +00:00
inline OID & operator+=(word32 rhs) {m_values.push_back(rhs); return *this;}
2016-04-08 05:51:21 +00:00
//! \brief DER encode this OID
//! \param bt BufferedTransformation object
2015-11-05 06:59:46 +00:00
void DEREncode(BufferedTransformation &bt) const;
2016-09-10 08:57:48 +00:00
2016-04-08 05:51:21 +00:00
//! \brief BER decode an OID
//! \param bt BufferedTransformation object
2015-11-05 06:59:46 +00:00
void BERDecode(BufferedTransformation &bt);
2016-04-08 05:51:21 +00:00
//! \brief BER decode an OID
//! \param bt BufferedTransformation object
2016-04-08 08:23:42 +00:00
//! \throws BERDecodeErr() if decoded value doesn't match an expected OID
//! \details BERDecodeAndCheck() can be used to parse an OID and verify it matches an expected.
//! <pre>
//! BERSequenceDecoder key(bt);
//! ...
//! BERSequenceDecoder algorithm(key);
//! GetAlgorithmID().BERDecodeAndCheck(algorithm);
//! </pre>
2015-11-05 06:59:46 +00:00
void BERDecodeAndCheck(BufferedTransformation &bt) const;
std::vector<word32> m_values;
private:
static void EncodeValue(BufferedTransformation &bt, word32 v);
static size_t DecodeValue(BufferedTransformation &bt, word32 &v);
};
2016-04-08 05:51:21 +00:00
//! \brief ASN.1 encoded object filter
2015-11-05 06:59:46 +00:00
class EncodedObjectFilter : public Filter
{
public:
enum Flag {PUT_OBJECTS=1, PUT_MESSANGE_END_AFTER_EACH_OBJECT=2, PUT_MESSANGE_END_AFTER_ALL_OBJECTS=4, PUT_MESSANGE_SERIES_END_AFTER_ALL_OBJECTS=8};
2016-04-08 05:51:21 +00:00
virtual ~EncodedObjectFilter() {}
2016-04-08 05:51:21 +00:00
//! \brief Construct an EncodedObjectFilter
//! \param attachment a BufferedTrasformation to attach to this object
//! \param nObjects the number of objects
2016-04-08 05:51:21 +00:00
//! \param flags bitwise OR of EncodedObjectFilter::Flag
2017-03-01 11:10:06 +00:00
EncodedObjectFilter(BufferedTransformation *attachment = NULLPTR, unsigned int nObjects = 1, word32 flags = 0);
2015-11-05 06:59:46 +00:00
2016-04-08 08:23:42 +00:00
//! \brief Input a byte buffer for processing
//! \param inString the byte buffer to process
//! \param length the size of the string, in bytes
2015-11-05 06:59:46 +00:00
void Put(const byte *inString, size_t length);
unsigned int GetNumberOfCompletedObjects() const {return m_nCurrentObject;}
unsigned long GetPositionOfObject(unsigned int i) const {return m_positions[i];}
private:
BufferedTransformation & CurrentTarget();
word32 m_flags;
unsigned int m_nObjects, m_nCurrentObject, m_level;
std::vector<unsigned int> m_positions;
ByteQueue m_queue;
enum State {IDENTIFIER, LENGTH, BODY, TAIL, ALL_DONE} m_state;
byte m_id;
lword m_lengthRemaining;
};
2016-04-08 05:51:21 +00:00
//! \brief BER General Decoder
2015-11-05 06:59:46 +00:00
class CRYPTOPP_DLL BERGeneralDecoder : public Store
{
public:
virtual ~BERGeneralDecoder();
2015-11-05 06:59:46 +00:00
explicit BERGeneralDecoder(BufferedTransformation &inQueue, byte asnTag);
explicit BERGeneralDecoder(BERGeneralDecoder &inQueue, byte asnTag);
bool IsDefiniteLength() const {return m_definiteLength;}
lword RemainingLength() const {CRYPTOPP_ASSERT(m_definiteLength); return m_length;}
2015-11-05 06:59:46 +00:00
bool EndReached() const;
byte PeekByte() const;
void CheckByte(byte b);
size_t TransferTo2(BufferedTransformation &target, lword &transferBytes, const std::string &channel=DEFAULT_CHANNEL, bool blocking=true);
size_t CopyRangeTo2(BufferedTransformation &target, lword &begin, lword end=LWORD_MAX, const std::string &channel=DEFAULT_CHANNEL, bool blocking=true) const;
// call this to denote end of sequence
void MessageEnd();
protected:
BufferedTransformation &m_inQueue;
bool m_finished, m_definiteLength;
lword m_length;
private:
void Init(byte asnTag);
void StoreInitialize(const NameValuePairs &parameters)
{CRYPTOPP_UNUSED(parameters); CRYPTOPP_ASSERT(false);}
2015-11-05 06:59:46 +00:00
lword ReduceLength(lword delta);
};
// GCC (and likely other compilers) identify the explicit DERGeneralEncoder as a copy constructor;
// and not a constructor. We had to remove the default asnTag value to point the compiler in the
// proper direction. We did not break the library or versioning based on the output of
// `nm --demangle libcryptopp.a | grep DERGeneralEncoder::DERGeneralEncoder | grep -v " U "`.
2016-04-08 05:51:21 +00:00
//! \brief DER General Encoder
2015-11-05 06:59:46 +00:00
class CRYPTOPP_DLL DERGeneralEncoder : public ByteQueue
{
public:
virtual ~DERGeneralEncoder();
2015-11-05 06:59:46 +00:00
explicit DERGeneralEncoder(BufferedTransformation &outQueue, byte asnTag = SEQUENCE | CONSTRUCTED);
explicit DERGeneralEncoder(DERGeneralEncoder &outQueue, byte asnTag = SEQUENCE | CONSTRUCTED);
// call this to denote end of sequence
void MessageEnd();
private:
BufferedTransformation &m_outQueue;
bool m_finished;
byte m_asnTag;
};
2016-04-08 05:51:21 +00:00
//! \brief BER Sequence Decoder
2015-11-05 06:59:46 +00:00
class CRYPTOPP_DLL BERSequenceDecoder : public BERGeneralDecoder
{
public:
explicit BERSequenceDecoder(BufferedTransformation &inQueue, byte asnTag = SEQUENCE | CONSTRUCTED)
: BERGeneralDecoder(inQueue, asnTag) {}
explicit BERSequenceDecoder(BERSequenceDecoder &inQueue, byte asnTag = SEQUENCE | CONSTRUCTED)
: BERGeneralDecoder(inQueue, asnTag) {}
};
2016-04-08 05:51:21 +00:00
//! \brief DER Sequence Encoder
2015-11-05 06:59:46 +00:00
class CRYPTOPP_DLL DERSequenceEncoder : public DERGeneralEncoder
{
public:
explicit DERSequenceEncoder(BufferedTransformation &outQueue, byte asnTag = SEQUENCE | CONSTRUCTED)
: DERGeneralEncoder(outQueue, asnTag) {}
explicit DERSequenceEncoder(DERSequenceEncoder &outQueue, byte asnTag = SEQUENCE | CONSTRUCTED)
: DERGeneralEncoder(outQueue, asnTag) {}
};
2016-04-08 05:51:21 +00:00
//! \brief BER Set Decoder
2015-11-05 06:59:46 +00:00
class CRYPTOPP_DLL BERSetDecoder : public BERGeneralDecoder
{
public:
explicit BERSetDecoder(BufferedTransformation &inQueue, byte asnTag = SET | CONSTRUCTED)
: BERGeneralDecoder(inQueue, asnTag) {}
explicit BERSetDecoder(BERSetDecoder &inQueue, byte asnTag = SET | CONSTRUCTED)
: BERGeneralDecoder(inQueue, asnTag) {}
};
2016-04-08 05:51:21 +00:00
//! \brief DER Set Encoder
2015-11-05 06:59:46 +00:00
class CRYPTOPP_DLL DERSetEncoder : public DERGeneralEncoder
{
public:
explicit DERSetEncoder(BufferedTransformation &outQueue, byte asnTag = SET | CONSTRUCTED)
: DERGeneralEncoder(outQueue, asnTag) {}
explicit DERSetEncoder(DERSetEncoder &outQueue, byte asnTag = SET | CONSTRUCTED)
: DERGeneralEncoder(outQueue, asnTag) {}
};
2016-04-08 08:23:42 +00:00
//! \brief Optional data encoder and decoder
//! \tparam T class or type
2015-11-05 06:59:46 +00:00
template <class T>
class ASNOptional : public member_ptr<T>
{
public:
2016-04-08 08:23:42 +00:00
//! \brief BER decode optional data
//! \param seqDecoder sequence with the optional ASN.1 data
//! \param tag ASN.1 tag to match as optional data
//! \param mask the mask to apply when matching the tag
//! \sa ASNTag and ASNIdFlag
2015-11-05 06:59:46 +00:00
void BERDecode(BERSequenceDecoder &seqDecoder, byte tag, byte mask = ~CONSTRUCTED)
{
byte b;
if (seqDecoder.Peek(b) && (b & mask) == tag)
reset(new T(seqDecoder));
}
2016-04-08 08:23:42 +00:00
//! \brief DER encode optional data
//! \param out BufferedTransformation object
2015-11-05 06:59:46 +00:00
void DEREncode(BufferedTransformation &out)
{
2017-03-01 11:10:06 +00:00
if (this->get() != NULLPTR)
2015-11-05 06:59:46 +00:00
this->get()->DEREncode(out);
}
};
2016-04-08 08:23:42 +00:00
//! \brief Encode and decode ASN.1 objects with additional information
//! \tparam BASE base class or type
//! \details Encodes and decodes public keys, private keys and group
//! parameters with OID identifying the algorithm or scheme.
2015-11-05 06:59:46 +00:00
template <class BASE>
class CRYPTOPP_DLL CRYPTOPP_NO_VTABLE ASN1CryptoMaterial : public ASN1Object, public BASE
{
public:
2016-04-08 08:23:42 +00:00
//! \brief DER encode ASN.1 object
//! \param bt BufferedTransformation object
//! \details Save() will write the OID associated with algorithm or scheme.
//! In the case of public and private keys, this function writes the
//! subjectPubicKeyInfo and privateKeyInfo parts.
2015-11-05 06:59:46 +00:00
void Save(BufferedTransformation &bt) const
{BEREncode(bt);}
2016-04-08 08:23:42 +00:00
//! \brief BER decode ASN.1 object
//! \param bt BufferedTransformation object
2015-11-05 06:59:46 +00:00
void Load(BufferedTransformation &bt)
{BERDecode(bt);}
};
2016-04-08 21:34:15 +00:00
//! \brief Encodes and decodes subjectPublicKeyInfo
2015-11-05 06:59:46 +00:00
class CRYPTOPP_DLL X509PublicKey : public ASN1CryptoMaterial<PublicKey>
{
public:
virtual ~X509PublicKey() {}
2015-11-05 06:59:46 +00:00
void BERDecode(BufferedTransformation &bt);
void DEREncode(BufferedTransformation &bt) const;
2016-04-08 21:34:15 +00:00
//! \brief Retrieves the OID of the algorithm
//! \returns OID of the algorithm
2015-11-05 06:59:46 +00:00
virtual OID GetAlgorithmID() const =0;
virtual bool BERDecodeAlgorithmParameters(BufferedTransformation &bt)
{BERDecodeNull(bt); return false;}
virtual bool DEREncodeAlgorithmParameters(BufferedTransformation &bt) const
{DEREncodeNull(bt); return false;} // see RFC 2459, section 7.3.1
//! decode subjectPublicKey part of subjectPublicKeyInfo, without the BIT STRING header
virtual void BERDecodePublicKey(BufferedTransformation &bt, bool parametersPresent, size_t size) =0;
//! encode subjectPublicKey part of subjectPublicKeyInfo, without the BIT STRING header
virtual void DEREncodePublicKey(BufferedTransformation &bt) const =0;
};
2016-04-08 21:34:15 +00:00
//! \brief Encodes and decodesprivateKeyInfo
2015-11-05 06:59:46 +00:00
class CRYPTOPP_DLL PKCS8PrivateKey : public ASN1CryptoMaterial<PrivateKey>
{
public:
virtual ~PKCS8PrivateKey() {}
2015-11-05 06:59:46 +00:00
void BERDecode(BufferedTransformation &bt);
void DEREncode(BufferedTransformation &bt) const;
2016-04-08 21:34:15 +00:00
//! \brief Retrieves the OID of the algorithm
//! \returns OID of the algorithm
2015-11-05 06:59:46 +00:00
virtual OID GetAlgorithmID() const =0;
virtual bool BERDecodeAlgorithmParameters(BufferedTransformation &bt)
{BERDecodeNull(bt); return false;}
virtual bool DEREncodeAlgorithmParameters(BufferedTransformation &bt) const
{DEREncodeNull(bt); return false;} // see RFC 2459, section 7.3.1
//! decode privateKey part of privateKeyInfo, without the OCTET STRING header
virtual void BERDecodePrivateKey(BufferedTransformation &bt, bool parametersPresent, size_t size) =0;
//! encode privateKey part of privateKeyInfo, without the OCTET STRING header
virtual void DEREncodePrivateKey(BufferedTransformation &bt) const =0;
//! decode optional attributes including context-specific tag
/*! /note default implementation stores attributes to be output in DEREncodeOptionalAttributes */
virtual void BERDecodeOptionalAttributes(BufferedTransformation &bt);
//! encode optional attributes including context-specific tag
virtual void DEREncodeOptionalAttributes(BufferedTransformation &bt) const;
protected:
ByteQueue m_optionalAttributes;
};
// ********************************************************
2016-04-08 08:23:42 +00:00
//! \brief DER Encode unsigned value
//! \tparam T class or type
//! \param out BufferedTransformation object
//! \param w unsigned value to encode
//! \param asnTag the ASN.1 type
//! \details DEREncodeUnsigned() can be used with INTEGER, BOOLEAN, and ENUM
2015-11-05 06:59:46 +00:00
template <class T>
size_t DEREncodeUnsigned(BufferedTransformation &out, T w, byte asnTag = INTEGER)
{
byte buf[sizeof(w)+1];
unsigned int bc;
if (asnTag == BOOLEAN)
{
buf[sizeof(w)] = w ? 0xff : 0;
bc = 1;
}
else
{
buf[0] = 0;
for (unsigned int i=0; i<sizeof(w); i++)
buf[i+1] = byte(w >> (sizeof(w)-1-i)*8);
bc = sizeof(w);
while (bc > 1 && buf[sizeof(w)+1-bc] == 0)
--bc;
if (buf[sizeof(w)+1-bc] & 0x80)
++bc;
}
out.Put(asnTag);
size_t lengthBytes = DERLengthEncode(out, bc);
out.Put(buf+sizeof(w)+1-bc, bc);
return 1+lengthBytes+bc;
}
2016-04-08 08:23:42 +00:00
//! \brief BER Decode unsigned value
//! \tparam T fundamental C++ type
2016-04-08 08:23:42 +00:00
//! \param in BufferedTransformation object
//! \param w the decoded value
2016-04-08 08:23:42 +00:00
//! \param asnTag the ASN.1 type
//! \param minValue the minimum expected value
//! \param maxValue the maximum expected value
//! \throws BERDecodeErr() if the value cannot be parsed or the decoded value is not within range.
//! \details DEREncodeUnsigned() can be used with INTEGER, BOOLEAN, and ENUM
2015-11-05 06:59:46 +00:00
template <class T>
void BERDecodeUnsigned(BufferedTransformation &in, T &w, byte asnTag = INTEGER,
T minValue = 0, T maxValue = T(0xffffffff))
2015-11-05 06:59:46 +00:00
{
byte b;
if (!in.Get(b) || b != asnTag)
BERDecodeError();
size_t bc;
bool definite = BERLengthDecode(in, bc);
if (!definite)
BERDecodeError();
if (bc > in.MaxRetrievable()) // Issue 346
BERDecodeError();
if (asnTag == BOOLEAN && bc != 1) // X.690, 8.2.1
BERDecodeError();
if ((asnTag == INTEGER || asnTag == ENUMERATED) && bc == 0) // X.690, 8.3.1 and 8.4
BERDecodeError();
2015-11-05 06:59:46 +00:00
SecByteBlock buf(bc);
if (bc != in.Get(buf, bc))
BERDecodeError();
// This consumes leading 0 octets. According to X.690, 8.3.2, it could be non-conforming behavior.
// X.690, 8.3.2 says "the bits of the first octet and bit 8 of the second octet ... (a) shall
// not all be ones and (b) shall not all be zeros ... These rules ensure that an integer value
// is always encoded in the smallest possible number of octet".
// We invented AER (Alternate Encoding Rules), which is more relaxed than BER, CER, and DER.
2015-11-05 06:59:46 +00:00
const byte *ptr = buf;
while (bc > sizeof(w) && *ptr == 0)
{
bc--;
ptr++;
}
if (bc > sizeof(w))
BERDecodeError();
w = 0;
for (unsigned int i=0; i<bc; i++)
w = (w << 8) | ptr[i];
if (w < minValue || w > maxValue)
BERDecodeError();
}
2016-04-08 08:23:42 +00:00
#ifdef CRYPTOPP_DOXYGEN_PROCESSING
//! \brief Compare two OIDs for equality
//! \param lhs the first OID
//! \param rhs the second OID
//! \returns true if the OIDs are equal, false otherwise
inline bool operator==(const OID &lhs, const OID &rhs);
//! \brief Compare two OIDs for inequality
//! \param lhs the first OID
//! \param rhs the second OID
//! \returns true if the OIDs are not equal, false otherwise
inline bool operator!=(const OID &lhs, const OID &rhs);
//! \brief Compare two OIDs for ordering
//! \param lhs the first OID
//! \param rhs the second OID
//! \returns true if the first OID is less than the second OID, false otherwise
//! \details operator<() calls std::lexicographical_compare() on each element in the array of values.
inline bool operator<(const OID &lhs, const OID &rhs);
//! \brief Append a value to an OID
//! \param lhs the OID
//! \param rhs the value to append
inline OID operator+(const OID &lhs, unsigned long rhs);
#else
2015-11-05 06:59:46 +00:00
inline bool operator==(const ::CryptoPP::OID &lhs, const ::CryptoPP::OID &rhs)
{return lhs.m_values == rhs.m_values;}
inline bool operator!=(const ::CryptoPP::OID &lhs, const ::CryptoPP::OID &rhs)
{return lhs.m_values != rhs.m_values;}
inline bool operator<(const ::CryptoPP::OID &lhs, const ::CryptoPP::OID &rhs)
{return std::lexicographical_compare(lhs.m_values.begin(), lhs.m_values.end(), rhs.m_values.begin(), rhs.m_values.end());}
inline ::CryptoPP::OID operator+(const ::CryptoPP::OID &lhs, unsigned long rhs)
{return ::CryptoPP::OID(lhs)+=rhs;}
2016-04-08 08:23:42 +00:00
#endif
2015-11-05 06:59:46 +00:00
NAMESPACE_END
// Issue 340
#if CRYPTOPP_GCC_DIAGNOSTIC_AVAILABLE
# pragma GCC diagnostic pop
#endif
2015-11-05 06:59:46 +00:00
#endif