Botan 3.0.0
Crypto and TLS for C&
base64.cpp
Go to the documentation of this file.
1/*
2* Base64 Encoding and Decoding
3* (C) 2010,2015,2020 Jack Lloyd
4*
5* Botan is released under the Simplified BSD License (see license.txt)
6*/
7
8#include <botan/base64.h>
9#include <botan/internal/codec_base.h>
10#include <botan/exceptn.h>
11#include <botan/internal/rounding.h>
12#include <botan/internal/ct_utils.h>
13#include <botan/internal/charset.h>
14#include <botan/internal/fmt.h>
15
16namespace Botan {
17
18namespace {
19
20class Base64 final
21 {
22 public:
23 static inline std::string name() noexcept
24 {
25 return "base64";
26 }
27
28 static inline size_t encoding_bytes_in() noexcept
29 {
30 return m_encoding_bytes_in;
31 }
32 static inline size_t encoding_bytes_out() noexcept
33 {
34 return m_encoding_bytes_out;
35 }
36
37 static inline size_t decoding_bytes_in() noexcept
38 {
39 return m_encoding_bytes_out;
40 }
41 static inline size_t decoding_bytes_out() noexcept
42 {
43 return m_encoding_bytes_in;
44 }
45
46 static inline size_t bits_consumed() noexcept
47 {
48 return m_encoding_bits;
49 }
50 static inline size_t remaining_bits_before_padding() noexcept
51 {
52 return m_remaining_bits_before_padding;
53 }
54
55 static inline size_t encode_max_output(size_t input_length)
56 {
57 return (round_up(input_length, m_encoding_bytes_in) / m_encoding_bytes_in) * m_encoding_bytes_out;
58 }
59 static inline size_t decode_max_output(size_t input_length)
60 {
61 return (round_up(input_length, m_encoding_bytes_out) * m_encoding_bytes_in) / m_encoding_bytes_out;
62 }
63
64 static void encode(char out[8], const uint8_t in[5]) noexcept;
65
66 static uint8_t lookup_binary_value(char input) noexcept;
67
68 static bool check_bad_char(uint8_t bin, char input, bool ignore_ws);
69
70 static void decode(uint8_t* out_ptr, const uint8_t decode_buf[4])
71 {
72 out_ptr[0] = (decode_buf[0] << 2) | (decode_buf[1] >> 4);
73 out_ptr[1] = (decode_buf[1] << 4) | (decode_buf[2] >> 2);
74 out_ptr[2] = (decode_buf[2] << 6) | decode_buf[3];
75 }
76
77 static inline size_t bytes_to_remove(size_t final_truncate)
78 {
79 return final_truncate;
80 }
81
82 private:
83 static const size_t m_encoding_bits = 6;
84 static const size_t m_remaining_bits_before_padding = 8;
85
86 static const size_t m_encoding_bytes_in = 3;
87 static const size_t m_encoding_bytes_out = 4;
88 };
89
90char lookup_base64_char(uint8_t x)
91 {
92 BOTAN_DEBUG_ASSERT(x < 64);
93
94 const auto in_az = CT::Mask<uint8_t>::is_within_range(x, 26, 51);
95 const auto in_09 = CT::Mask<uint8_t>::is_within_range(x, 52, 61);
96 const auto eq_plus = CT::Mask<uint8_t>::is_equal(x, 62);
97 const auto eq_slash = CT::Mask<uint8_t>::is_equal(x, 63);
98
99 const char c_AZ = 'A' + x;
100 const char c_az = 'a' + (x - 26);
101 const char c_09 = '0' + (x - 2*26);
102 const char c_plus = '+';
103 const char c_slash = '/';
104
105 char ret = c_AZ;
106 ret = in_az.select(c_az, ret);
107 ret = in_09.select(c_09, ret);
108 ret = eq_plus.select(c_plus, ret);
109 ret = eq_slash.select(c_slash, ret);
110
111 return ret;
112 }
113
114//static
115void Base64::encode(char out[8], const uint8_t in[5]) noexcept
116 {
117 const uint8_t b0 = (in[0] & 0xFC) >> 2;
118 const uint8_t b1 = ((in[0] & 0x03) << 4) | (in[1] >> 4);
119 const uint8_t b2 = ((in[1] & 0x0F) << 2) | (in[2] >> 6);
120 const uint8_t b3 = in[2] & 0x3F;
121 out[0] = lookup_base64_char(b0);
122 out[1] = lookup_base64_char(b1);
123 out[2] = lookup_base64_char(b2);
124 out[3] = lookup_base64_char(b3);
125 }
126
127//static
128uint8_t Base64::lookup_binary_value(char input) noexcept
129 {
130 const uint8_t c = static_cast<uint8_t>(input);
131
132 const auto is_alpha_upper = CT::Mask<uint8_t>::is_within_range(c, uint8_t('A'), uint8_t('Z'));
133 const auto is_alpha_lower = CT::Mask<uint8_t>::is_within_range(c, uint8_t('a'), uint8_t('z'));
134 const auto is_decimal = CT::Mask<uint8_t>::is_within_range(c, uint8_t('0'), uint8_t('9'));
135
136 const auto is_plus = CT::Mask<uint8_t>::is_equal(c, uint8_t('+'));
137 const auto is_slash = CT::Mask<uint8_t>::is_equal(c, uint8_t('/'));
138 const auto is_equal = CT::Mask<uint8_t>::is_equal(c, uint8_t('='));
139
140 const auto is_whitespace = CT::Mask<uint8_t>::is_any_of(c, {
141 uint8_t(' '), uint8_t('\t'), uint8_t('\n'), uint8_t('\r')
142 });
143
144 const uint8_t c_upper = c - uint8_t('A');
145 const uint8_t c_lower = c - uint8_t('a') + 26;
146 const uint8_t c_decim = c - uint8_t('0') + 2*26;
147
148 uint8_t ret = 0xFF; // default value
149
150 ret = is_alpha_upper.select(c_upper, ret);
151 ret = is_alpha_lower.select(c_lower, ret);
152 ret = is_decimal.select(c_decim, ret);
153 ret = is_plus.select(62, ret);
154 ret = is_slash.select(63, ret);
155 ret = is_equal.select(0x81, ret);
156 ret = is_whitespace.select(0x80, ret);
157
158 return ret;
159 }
160
161//static
162bool Base64::check_bad_char(uint8_t bin, char input, bool ignore_ws)
163 {
164 if(bin <= 0x3F)
165 {
166 return true;
167 }
168 else if(!(bin == 0x81 || (bin == 0x80 && ignore_ws)))
169 {
170 throw Invalid_Argument(fmt("base64_decode: invalid character '{}'",
172 }
173 return false;
174 }
175
176}
177
178size_t base64_encode(char out[],
179 const uint8_t in[],
180 size_t input_length,
181 size_t& input_consumed,
182 bool final_inputs)
183 {
184 return base_encode(Base64(), out, in, input_length, input_consumed, final_inputs);
185 }
186
187std::string base64_encode(const uint8_t input[],
188 size_t input_length)
189 {
190 return base_encode_to_string(Base64(), input, input_length);
191 }
192
193size_t base64_decode(uint8_t out[],
194 const char in[],
195 size_t input_length,
196 size_t& input_consumed,
197 bool final_inputs,
198 bool ignore_ws)
199 {
200 return base_decode(Base64(), out, in, input_length, input_consumed, final_inputs, ignore_ws);
201 }
202
203size_t base64_decode(uint8_t output[],
204 const char input[],
205 size_t input_length,
206 bool ignore_ws)
207 {
208 return base_decode_full(Base64(), output, input, input_length, ignore_ws);
209 }
210
211size_t base64_decode(uint8_t output[],
212 std::string_view input,
213 bool ignore_ws)
214 {
215 return base64_decode(output, input.data(), input.length(), ignore_ws);
216 }
217
218
219size_t base64_decode(std::span<uint8_t> output,
220 std::string_view input,
221 bool ignore_ws)
222 {
223 if(output.size() < base64_decode_max_output(input.size()))
224 { throw Invalid_Argument("base64_decode: output buffer is too short"); }
225 return base64_decode(output.data(), input.data(), input.length(), ignore_ws);
226 }
227
229 size_t input_length,
230 bool ignore_ws)
231 {
232 return base_decode_to_vec<secure_vector<uint8_t>>(Base64(), input, input_length, ignore_ws);
233 }
234
236 bool ignore_ws)
237 {
238 return base64_decode(input.data(), input.size(), ignore_ws);
239 }
240
241size_t base64_encode_max_output(size_t input_length)
242 {
243 return Base64::encode_max_output(input_length);
244 }
245
246size_t base64_decode_max_output(size_t input_length)
247 {
248 return Base64::decode_max_output(input_length);
249 }
250
251}
#define BOTAN_DEBUG_ASSERT(expr)
Definition: assert.h:122
static Mask< T > is_equal(T x, T y)
Definition: ct_utils.h:147
static Mask< T > is_any_of(T v, std::initializer_list< T > accepted)
Definition: ct_utils.h:194
static Mask< T > is_within_range(T v, T l, T u)
Definition: ct_utils.h:184
std::string name
int(* final)(unsigned char *, CTX *)
std::string encode(const uint8_t der[], size_t length, std::string_view label, size_t width)
Definition: pem.cpp:42
secure_vector< uint8_t > decode(DataSource &source, std::string &label)
Definition: pem.cpp:70
Definition: alg_id.cpp:12
std::string format_char_for_display(char c)
Definition: charset.cpp:109
std::string fmt(std::string_view format, const T &... args)
Definition: fmt.h:60
size_t base64_encode(char out[], const uint8_t in[], size_t input_length, size_t &input_consumed, bool final_inputs)
Definition: base64.cpp:178
size_t base64_decode(uint8_t out[], const char in[], size_t input_length, size_t &input_consumed, bool final_inputs, bool ignore_ws)
Definition: base64.cpp:193
size_t base_encode(Base &&base, char output[], const uint8_t input[], size_t input_length, size_t &input_consumed, bool final_inputs)
Definition: codec_base.h:34
size_t base64_encode_max_output(size_t input_length)
Definition: base64.cpp:241
size_t base64_decode_max_output(size_t input_length)
Definition: base64.cpp:246
size_t base_decode(Base &&base, uint8_t output[], const char input[], size_t input_length, size_t &input_consumed, bool final_inputs, bool ignore_ws=true)
Definition: codec_base.h:124
std::string base_encode_to_string(Base &&base, const uint8_t input[], size_t input_length)
Definition: codec_base.h:86
std::vector< T, secure_allocator< T > > secure_vector
Definition: secmem.h:64
size_t round_up(size_t n, size_t align_to)
Definition: rounding.h:21
size_t base_decode_full(Base &&base, uint8_t output[], const char input[], size_t input_length, bool ignore_ws)
Definition: codec_base.h:189