Botan 3.9.0
Crypto and TLS for C&
base32.cpp
Go to the documentation of this file.
1/*
2* Base32 Encoding and Decoding
3* (C) 2018 Erwan Chaussy
4* (C) 2018,2020,2025 Jack Lloyd
5*
6* Botan is released under the Simplified BSD License (see license.txt)
7*/
8
9#include <botan/base32.h>
10
11#include <botan/internal/charset.h>
12#include <botan/internal/codec_base.h>
13#include <botan/internal/ct_utils.h>
14#include <botan/internal/fmt.h>
15#include <botan/internal/int_utils.h>
16#include <botan/internal/loadstor.h>
17#include <botan/internal/rounding.h>
18
19namespace Botan {
20
21namespace {
22
23class Base32 final {
24 public:
25 static std::string name() noexcept { return "base32"; }
26
27 static constexpr size_t encoding_bytes_in() noexcept { return m_encoding_bytes_in; }
28
29 static constexpr size_t encoding_bytes_out() noexcept { return m_encoding_bytes_out; }
30
31 static constexpr size_t decoding_bytes_in() noexcept { return m_encoding_bytes_out; }
32
33 static constexpr size_t decoding_bytes_out() noexcept { return m_encoding_bytes_in; }
34
35 static constexpr size_t bits_consumed() noexcept { return m_encoding_bits; }
36
37 static constexpr size_t remaining_bits_before_padding() noexcept { return m_remaining_bits_before_padding; }
38
39 static constexpr size_t encode_max_output(size_t input_length) {
40 return (round_up(input_length, m_encoding_bytes_in) / m_encoding_bytes_in) * m_encoding_bytes_out;
41 }
42
43 static constexpr size_t decode_max_output(size_t input_length) {
44 return (round_up(input_length, m_encoding_bytes_out) * m_encoding_bytes_in) / m_encoding_bytes_out;
45 }
46
47 static void encode(char out[8], const uint8_t in[5]) noexcept;
48
49 static uint8_t lookup_binary_value(char input) noexcept;
50
51 static bool check_bad_char(uint8_t bin, char input, bool ignore_ws);
52
53 static void decode(uint8_t* out_ptr, const uint8_t decode_buf[8]) {
54 out_ptr[0] = (decode_buf[0] << 3) | (decode_buf[1] >> 2);
55 out_ptr[1] = (decode_buf[1] << 6) | (decode_buf[2] << 1) | (decode_buf[3] >> 4);
56 out_ptr[2] = (decode_buf[3] << 4) | (decode_buf[4] >> 1);
57 out_ptr[3] = (decode_buf[4] << 7) | (decode_buf[5] << 2) | (decode_buf[6] >> 3);
58 out_ptr[4] = (decode_buf[6] << 5) | decode_buf[7];
59 }
60
61 static size_t bytes_to_remove(size_t final_truncate) {
62 return (final_truncate > 0) ? (final_truncate / 2) + 1 : 0;
63 }
64
65 private:
66 static constexpr size_t m_encoding_bits = 5;
67 static constexpr size_t m_remaining_bits_before_padding = 6;
68
69 static constexpr size_t m_encoding_bytes_in = 5;
70 static constexpr size_t m_encoding_bytes_out = 8;
71};
72
73namespace {
74
75uint64_t lookup_base32_char(uint64_t x) {
76 uint64_t r = x;
77 r += swar_lt<uint64_t>(x, 0x1a1a1a1a1a1a1a1a) & 0x2929292929292929;
78 r += 0x1818181818181818;
79
80 return r;
81}
82
83} // namespace
84
85//static
86void Base32::encode(char out[8], const uint8_t in[5]) noexcept {
87 const uint8_t b0 = (in[0] & 0xF8) >> 3;
88 const uint8_t b1 = ((in[0] & 0x07) << 2) | (in[1] >> 6);
89 const uint8_t b2 = ((in[1] & 0x3E) >> 1);
90 const uint8_t b3 = ((in[1] & 0x01) << 4) | (in[2] >> 4);
91 const uint8_t b4 = ((in[2] & 0x0F) << 1) | (in[3] >> 7);
92 const uint8_t b5 = ((in[3] & 0x7C) >> 2);
93 const uint8_t b6 = ((in[3] & 0x03) << 3) | (in[4] >> 5);
94 const uint8_t b7 = in[4] & 0x1F;
95
96 auto b = lookup_base32_char(make_uint64(b0, b1, b2, b3, b4, b5, b6, b7));
97
98 out[0] = static_cast<char>(get_byte<0>(b));
99 out[1] = static_cast<char>(get_byte<1>(b));
100 out[2] = static_cast<char>(get_byte<2>(b));
101 out[3] = static_cast<char>(get_byte<3>(b));
102 out[4] = static_cast<char>(get_byte<4>(b));
103 out[5] = static_cast<char>(get_byte<5>(b));
104 out[6] = static_cast<char>(get_byte<6>(b));
105 out[7] = static_cast<char>(get_byte<7>(b));
106}
107
108//static
109uint8_t Base32::lookup_binary_value(char input) noexcept {
110 const uint8_t c = static_cast<uint8_t>(input);
111
112 const auto is_alpha_upper = CT::Mask<uint8_t>::is_within_range(c, uint8_t('A'), uint8_t('Z'));
113 const auto is_decimal = CT::Mask<uint8_t>::is_within_range(c, uint8_t('2'), uint8_t('7'));
114
115 const auto is_equal = CT::Mask<uint8_t>::is_equal(c, uint8_t('='));
116 const auto is_whitespace =
117 CT::Mask<uint8_t>::is_any_of(c, {uint8_t(' '), uint8_t('\t'), uint8_t('\n'), uint8_t('\r')});
118
119 const uint8_t c_upper = c - uint8_t('A');
120 const uint8_t c_decim = c - uint8_t('2') + 26;
121
122 uint8_t ret = 0xFF; // default value
123
124 ret = is_alpha_upper.select(c_upper, ret);
125 ret = is_decimal.select(c_decim, ret);
126 ret = is_equal.select(0x81, ret);
127 ret = is_whitespace.select(0x80, ret);
128
129 return ret;
130}
131
132//static
133bool Base32::check_bad_char(uint8_t bin, char input, bool ignore_ws) {
134 if(bin <= 0x1F) {
135 return true;
136 } else if(!(bin == 0x81 || (bin == 0x80 && ignore_ws))) {
137 throw Invalid_Argument(fmt("base32_decode: invalid character '{}'", format_char_for_display(input)));
138 }
139 return false;
140}
141
142} // namespace
143
144size_t base32_encode(char out[], const uint8_t in[], size_t input_length, size_t& input_consumed, bool final_inputs) {
145 return base_encode(Base32(), out, in, input_length, input_consumed, final_inputs);
146}
147
148std::string base32_encode(const uint8_t input[], size_t input_length) {
149 return base_encode_to_string(Base32(), input, input_length);
150}
151
153 uint8_t out[], const char in[], size_t input_length, size_t& input_consumed, bool final_inputs, bool ignore_ws) {
154 return base_decode(Base32(), out, in, input_length, input_consumed, final_inputs, ignore_ws);
155}
156
157size_t base32_decode(uint8_t output[], const char input[], size_t input_length, bool ignore_ws) {
158 return base_decode_full(Base32(), output, input, input_length, ignore_ws);
159}
160
161size_t base32_decode(uint8_t output[], std::string_view input, bool ignore_ws) {
162 return base32_decode(output, input.data(), input.length(), ignore_ws);
163}
164
165secure_vector<uint8_t> base32_decode(const char input[], size_t input_length, bool ignore_ws) {
166 return base_decode_to_vec<secure_vector<uint8_t>>(Base32(), input, input_length, ignore_ws);
167}
168
169secure_vector<uint8_t> base32_decode(std::string_view input, bool ignore_ws) {
170 return base32_decode(input.data(), input.size(), ignore_ws);
171}
172
173size_t base32_encode_max_output(size_t input_length) {
174 return Base32::encode_max_output(input_length);
175}
176
177size_t base32_decode_max_output(size_t input_length) {
178 return Base32::decode_max_output(input_length);
179}
180
181} // namespace Botan
static constexpr Mask< T > is_within_range(T v, T l, T u)
Definition ct_utils.h:498
static constexpr Mask< T > is_equal(T x, T y)
Definition ct_utils.h:470
static constexpr Mask< T > is_any_of(T v, std::initializer_list< T > accepted)
Definition ct_utils.h:507
constexpr CT::Mask< T > is_equal(const T x[], const T y[], size_t len)
Definition ct_utils.h:826
std::string encode(const uint8_t der[], size_t length, std::string_view label, size_t width)
Definition pem.cpp:39
secure_vector< uint8_t > decode(DataSource &source, std::string &label)
Definition pem.cpp:62
constexpr uint8_t get_byte(T input)
Definition loadstor.h:79
size_t base32_encode_max_output(size_t input_length)
Definition base32.cpp:173
std::string format_char_for_display(char c)
Definition charset.cpp:98
constexpr T swar_lt(T a, T b)
Definition int_utils.h:91
std::string base_encode_to_string(const Base &base, const uint8_t input[], size_t input_length)
Definition codec_base.h:84
constexpr auto out_ptr(T &outptr) noexcept
Definition stl_util.h:415
constexpr uint64_t make_uint64(uint8_t i0, uint8_t i1, uint8_t i2, uint8_t i3, uint8_t i4, uint8_t i5, uint8_t i6, uint8_t i7)
Definition loadstor.h:121
std::string fmt(std::string_view format, const T &... args)
Definition fmt.h:53
constexpr size_t round_up(size_t n, size_t align_to)
Definition rounding.h:26
size_t base_encode(const Base &base, char output[], const uint8_t input[], size_t input_length, size_t &input_consumed, bool final_inputs)
Definition codec_base.h:35
size_t base32_decode(uint8_t out[], const char in[], size_t input_length, size_t &input_consumed, bool final_inputs, bool ignore_ws)
Definition base32.cpp:152
size_t base_decode(const 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:118
size_t base32_decode_max_output(size_t input_length)
Definition base32.cpp:177
std::vector< T, secure_allocator< T > > secure_vector
Definition secmem.h:69
Vector base_decode_to_vec(const Base &base, const char input[], size_t input_length, bool ignore_ws)
Definition codec_base.h:190
size_t base32_encode(char out[], const uint8_t in[], size_t input_length, size_t &input_consumed, bool final_inputs)
Definition base32.cpp:144
size_t base_decode_full(const Base &base, uint8_t output[], const char input[], size_t input_length, bool ignore_ws)
Definition codec_base.h:178