Botan 3.11.0
Crypto and TLS for C&
asn1_obj.cpp
Go to the documentation of this file.
1/*
2* ASN.1 Internals
3* (C) 1999-2007,2018 Jack Lloyd
4*
5* Botan is released under the Simplified BSD License (see license.txt)
6*/
7
8#include <botan/asn1_obj.h>
9
10#include <botan/assert.h>
11#include <botan/data_src.h>
12#include <botan/der_enc.h>
13#include <botan/mem_ops.h>
14#include <botan/internal/fmt.h>
15#include <botan/internal/mem_utils.h>
16#include <sstream>
17
18namespace Botan {
19
20std::vector<uint8_t> ASN1_Object::BER_encode() const {
21 std::vector<uint8_t> output;
22 DER_Encoder der(output);
23 this->encode_into(der);
24 return output;
25}
26
30
31/*
32* Check a type invariant on BER data
33*/
34void BER_Object::assert_is_a(ASN1_Type expected_type_tag, ASN1_Class expected_class_tag, std::string_view descr) const {
35 if(!this->is_a(expected_type_tag, expected_class_tag)) {
36 std::stringstream msg;
37
38 msg << "Tag mismatch when decoding " << descr << " got ";
39
40 if(m_class_tag == ASN1_Class::NoObject && m_type_tag == ASN1_Type::NoObject) {
41 msg << "EOF";
42 } else {
43 if(m_class_tag == ASN1_Class::Universal || m_class_tag == ASN1_Class::Constructed) {
44 msg << asn1_tag_to_string(m_type_tag);
45 } else {
46 msg << std::to_string(static_cast<uint32_t>(m_type_tag));
47 }
48
49 msg << "/" << asn1_class_to_string(m_class_tag);
50 }
51
52 msg << " expected ";
53
54 if(expected_class_tag == ASN1_Class::Universal || expected_class_tag == ASN1_Class::Constructed) {
55 msg << asn1_tag_to_string(expected_type_tag);
56 } else {
57 msg << std::to_string(static_cast<uint32_t>(expected_type_tag));
58 }
59
60 msg << "/" << asn1_class_to_string(expected_class_tag);
61
62 throw BER_Decoding_Error(msg.str());
63 }
64}
65
66bool BER_Object::is_a(ASN1_Type expected_type_tag, ASN1_Class expected_class_tag) const {
67 return (m_type_tag == expected_type_tag && m_class_tag == expected_class_tag);
68}
69
70bool BER_Object::is_a(int expected_type_tag, ASN1_Class expected_class_tag) const {
71 return is_a(ASN1_Type(expected_type_tag), expected_class_tag);
72}
73
74void BER_Object::set_tagging(ASN1_Type type_tag, ASN1_Class class_tag) {
75 m_type_tag = type_tag;
76 m_class_tag = class_tag;
77}
78
80 switch(type) {
82 return "UNIVERSAL";
84 return "CONSTRUCTED";
86 return "CONTEXT_SPECIFIC";
88 return "APPLICATION";
90 return "PRIVATE";
92 return "NO_OBJECT";
93 default:
94 return "CLASS(" + std::to_string(static_cast<size_t>(type)) + ")";
95 }
96}
97
98std::string asn1_tag_to_string(ASN1_Type type) {
99 switch(type) {
101 return "SEQUENCE";
102
103 case ASN1_Type::Set:
104 return "SET";
105
107 return "PRINTABLE STRING";
108
110 return "NUMERIC STRING";
111
113 return "IA5 STRING";
114
116 return "T61 STRING";
117
119 return "UTF8 STRING";
120
122 return "VISIBLE STRING";
123
125 return "BMP STRING";
126
128 return "UNIVERSAL STRING";
129
131 return "UTC TIME";
132
134 return "GENERALIZED TIME";
135
137 return "OCTET STRING";
138
140 return "BIT STRING";
141
143 return "ENUMERATED";
144
146 return "INTEGER";
147
148 case ASN1_Type::Null:
149 return "NULL";
150
152 return "OBJECT";
153
155 return "BOOLEAN";
156
158 return "NO_OBJECT";
159
160 default:
161 return "TAG(" + std::to_string(static_cast<uint32_t>(type)) + ")";
162 }
163}
164
165/*
166* BER Decoding Exceptions
167*/
168BER_Decoding_Error::BER_Decoding_Error(std::string_view err) : Decoding_Error(fmt("BER: {}", err)) {}
169
170BER_Bad_Tag::BER_Bad_Tag(std::string_view str, uint32_t tagging) : BER_Decoding_Error(fmt("{}: {}", str, tagging)) {}
171
172namespace ASN1 {
173
174/*
175* Put some arbitrary bytes into a SEQUENCE
176*/
177std::vector<uint8_t> put_in_sequence(const std::vector<uint8_t>& contents) {
178 return ASN1::put_in_sequence(contents.data(), contents.size());
179}
180
181std::vector<uint8_t> put_in_sequence(const uint8_t bits[], size_t len) {
182 std::vector<uint8_t> output;
183 DER_Encoder(output).start_sequence().raw_bytes(bits, len).end_cons();
184 return output;
185}
186
187/*
188* Convert a BER object into a string object
189*/
190std::string to_string(const BER_Object& obj) {
191 return bytes_to_string(obj.data());
192}
193
194/*
195* Do heuristic tests for BER data
196*/
197bool maybe_BER(DataSource& source) {
198 uint8_t first_u8 = 0;
199 if(source.peek_byte(first_u8) == 0) {
200 BOTAN_ASSERT_EQUAL(source.read_byte(first_u8), 0, "Expected EOF");
201 throw Stream_IO_Error("ASN1::maybe_BER: Source was empty");
202 }
203
204 const auto cons_seq = static_cast<uint8_t>(ASN1_Class::Constructed) | static_cast<uint8_t>(ASN1_Type::Sequence);
205 return first_u8 == cons_seq;
206}
207
208} // namespace ASN1
209
210} // namespace Botan
#define BOTAN_ASSERT_EQUAL(expr1, expr2, assertion_made)
Definition assert.h:88
virtual void encode_into(DER_Encoder &to) const =0
std::vector< uint8_t > BER_encode() const
Definition asn1_obj.cpp:20
BER_Bad_Tag(std::string_view msg, uint32_t tagging)
Definition asn1_obj.cpp:170
BER_Decoding_Error(std::string_view err)
Definition asn1_obj.cpp:168
bool is_a(ASN1_Type type_tag, ASN1_Class class_tag) const
Definition asn1_obj.cpp:66
ASN1_Type type_tag() const
Definition asn1_obj.h:142
void assert_is_a(ASN1_Type type_tag, ASN1_Class class_tag, std::string_view descr="object") const
Definition asn1_obj.cpp:34
std::span< const uint8_t > data() const
Definition asn1_obj.h:154
ASN1_Class class_tag() const
Definition asn1_obj.h:144
DER_Encoder & start_sequence()
Definition der_enc.h:67
DER_Encoder & raw_bytes(const uint8_t val[], size_t len)
Definition der_enc.cpp:209
DER_Encoder & end_cons()
Definition der_enc.cpp:173
size_t peek_byte(uint8_t &out) const
Definition data_src.cpp:46
size_t read_byte(uint8_t &out)
Definition data_src.cpp:27
Decoding_Error(std::string_view name)
Definition exceptn.cpp:125
std::vector< uint8_t > put_in_sequence(const std::vector< uint8_t > &contents)
Definition asn1_obj.cpp:177
bool maybe_BER(DataSource &source)
Definition asn1_obj.cpp:197
std::string to_string(const BER_Object &obj)
Definition asn1_obj.cpp:190
std::string asn1_tag_to_string(ASN1_Type type)
Definition asn1_obj.cpp:98
ASN1_Class
Definition asn1_obj.h:28
std::string fmt(std::string_view format, const T &... args)
Definition fmt.h:53
ASN1_Type
Definition asn1_obj.h:43
std::string bytes_to_string(std::span< const uint8_t > bytes)
Definition mem_utils.h:76
void secure_scrub_memory(void *ptr, size_t n)
Definition mem_utils.cpp:25
std::string asn1_class_to_string(ASN1_Class type)
Definition asn1_obj.cpp:79