9#include <botan/base32.h>
10#include <botan/internal/codec_base.h>
11#include <botan/internal/rounding.h>
12#include <botan/internal/ct_utils.h>
21 static inline std::string
name() noexcept
26 static inline size_t encoding_bytes_in() noexcept
28 return m_encoding_bytes_in;
30 static inline size_t encoding_bytes_out() noexcept
32 return m_encoding_bytes_out;
35 static inline size_t decoding_bytes_in() noexcept
37 return m_encoding_bytes_out;
39 static inline size_t decoding_bytes_out() noexcept
41 return m_encoding_bytes_in;
44 static inline size_t bits_consumed() noexcept
46 return m_encoding_bits;
48 static inline size_t remaining_bits_before_padding() noexcept
50 return m_remaining_bits_before_padding;
53 static inline size_t encode_max_output(
size_t input_length)
55 return (
round_up(input_length, m_encoding_bytes_in) / m_encoding_bytes_in) * m_encoding_bytes_out;
57 static inline size_t decode_max_output(
size_t input_length)
59 return (
round_up(input_length, m_encoding_bytes_out) * m_encoding_bytes_in) / m_encoding_bytes_out;
62 static void encode(
char out[8],
const uint8_t in[5])
noexcept;
64 static uint8_t lookup_binary_value(
char input)
noexcept;
66 static bool check_bad_char(uint8_t bin,
char input,
bool ignore_ws);
68 static void decode(uint8_t* out_ptr,
const uint8_t decode_buf[8])
70 out_ptr[0] = (decode_buf[0] << 3) | (decode_buf[1] >> 2);
71 out_ptr[1] = (decode_buf[1] << 6) | (decode_buf[2] << 1) | (decode_buf[3] >> 4);
72 out_ptr[2] = (decode_buf[3] << 4) | (decode_buf[4] >> 1);
73 out_ptr[3] = (decode_buf[4] << 7) | (decode_buf[5] << 2) | (decode_buf[6] >> 3);
74 out_ptr[4] = (decode_buf[6] << 5) | decode_buf[7];
77 static inline size_t bytes_to_remove(
size_t final_truncate)
79 return final_truncate ? (final_truncate / 2) + 1 : 0;
83 static const size_t m_encoding_bits = 5;
84 static const size_t m_remaining_bits_before_padding = 6;
86 static const size_t m_encoding_bytes_in = 5;
87 static const size_t m_encoding_bytes_out = 8;
92char lookup_base32_char(uint8_t x)
98 const char c_AZ =
'A' + x;
99 const char c_27 =
'2' + (x - 26);
101 return in_AZ.select(c_AZ, c_27);
109 const uint8_t b0 = (in[0] & 0xF8) >> 3;
110 const uint8_t b1 = ((in[0] & 0x07) << 2) | (in[1] >> 6);
111 const uint8_t b2 = ((in[1] & 0x3E) >> 1);
112 const uint8_t b3 = ((in[1] & 0x01) << 4) | (in[2] >> 4);
113 const uint8_t b4 = ((in[2] & 0x0F) << 1) | (in[3] >> 7);
114 const uint8_t b5 = ((in[3] & 0x7C) >> 2);
115 const uint8_t b6 = ((in[3] & 0x03) << 3) | (in[4] >> 5);
116 const uint8_t b7 = in[4] & 0x1F;
118 out[0] = lookup_base32_char(b0);
119 out[1] = lookup_base32_char(b1);
120 out[2] = lookup_base32_char(b2);
121 out[3] = lookup_base32_char(b3);
122 out[4] = lookup_base32_char(b4);
123 out[5] = lookup_base32_char(b5);
124 out[6] = lookup_base32_char(b6);
125 out[7] = lookup_base32_char(b7);
129uint8_t Base32::lookup_binary_value(
char input)
noexcept
131 const uint8_t c =
static_cast<uint8_t
>(input);
138 uint8_t(
' '), uint8_t(
'\t'), uint8_t(
'\n'), uint8_t(
'\r')
141 const uint8_t c_upper = c - uint8_t(
'A');
142 const uint8_t c_decim = c - uint8_t(
'2') + 26;
146 ret = is_alpha_upper.select(c_upper, ret);
147 ret = is_decimal.select(c_decim, ret);
148 ret = is_equal.select(0x81, ret);
149 ret = is_whitespace.select(0x80, ret);
155bool Base32::check_bad_char(uint8_t bin,
char input,
bool ignore_ws)
161 else if(!(bin == 0x81 || (bin == 0x80 && ignore_ws)))
163 std::string bad_char(1, input);
165 { bad_char =
"\\t"; }
166 else if(bad_char ==
"\n")
167 { bad_char =
"\\n"; }
168 else if(bad_char ==
"\r")
169 { bad_char =
"\\r"; }
171 throw Invalid_Argument(
172 std::string(
"base32_decode: invalid base32 character '") +
183 size_t& input_consumed,
186 return base_encode(Base32(), out, in, input_length, input_consumed, final_inputs);
198 size_t& input_consumed,
202 return base_decode(Base32(), out, in, input_length, input_consumed, final_inputs, ignore_ws);
214 const std::string& input,
217 return base32_decode(output, input.data(), input.length(), ignore_ws);
224 return base_decode_to_vec<secure_vector<uint8_t>>(Base32(), input, input_length, ignore_ws);
#define BOTAN_DEBUG_ASSERT(expr)
static Mask< T > is_equal(T x, T y)
static Mask< T > is_any_of(T v, std::initializer_list< T > accepted)
static Mask< T > is_lt(T x, T y)
static Mask< T > is_within_range(T v, T l, T u)
int(* final)(unsigned char *, CTX *)
std::string encode(const uint8_t der[], size_t length, const std::string &label, size_t width)
secure_vector< uint8_t > decode(DataSource &source, std::string &label)
size_t base_encode(Base &&base, char output[], const uint8_t input[], size_t input_length, size_t &input_consumed, bool final_inputs)
size_t base32_decode(uint8_t out[], const char in[], size_t input_length, size_t &input_consumed, bool final_inputs, bool ignore_ws)
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)
std::string base_encode_to_string(Base &&base, const uint8_t input[], size_t input_length)
size_t base32_encode(char out[], const uint8_t in[], size_t input_length, size_t &input_consumed, bool final_inputs)
std::vector< T, secure_allocator< T > > secure_vector
size_t round_up(size_t n, size_t align_to)
size_t base_decode_full(Base &&base, uint8_t output[], const char input[], size_t input_length, bool ignore_ws)