Botan 3.10.0
Crypto and TLS for C&
base58.cpp
Go to the documentation of this file.
1/*
2* (C) 2018,2020 Jack Lloyd
3*
4* Botan is released under the Simplified BSD License (see license.txt)
5*/
6
7#include <botan/base58.h>
8
9#include <botan/bigint.h>
10#include <botan/exceptn.h>
11#include <botan/hash.h>
12#include <botan/internal/ct_utils.h>
13#include <botan/internal/divide.h>
14#include <botan/internal/loadstor.h>
15#include <botan/internal/mul128.h>
16
17namespace Botan {
18
19namespace {
20
21uint32_t sha256_d_checksum(const uint8_t input[], size_t input_length) {
22 auto sha256 = HashFunction::create_or_throw("SHA-256");
23
24 std::vector<uint8_t> checksum(32);
25
26 sha256->update(input, input_length);
27 sha256->final(checksum);
28
29 sha256->update(checksum);
30 sha256->final(checksum);
31
32 return load_be<uint32_t>(checksum.data(), 0);
33}
34
35char lookup_base58_char(uint8_t x) {
36 // "123456789 ABCDEFGH JKLMN PQRSTUVWXYZ abcdefghijk mnopqrstuvwxyz"
37 BOTAN_DEBUG_ASSERT(x < 58);
38
39 // This works by computing offset(x) such that x + offset(x) is equal to the
40 // desired character
41
42 size_t offset = 49;
43
44 offset += CT::Mask<uint8_t>::is_gt(x, 8).if_set_return(7);
45 offset += CT::Mask<uint8_t>::is_gt(x, 16).if_set_return(1);
46 offset += CT::Mask<uint8_t>::is_gt(x, 21).if_set_return(1);
47 offset += CT::Mask<uint8_t>::is_gt(x, 32).if_set_return(6);
48 offset += CT::Mask<uint8_t>::is_gt(x, 43).if_set_return(1);
49 return static_cast<char>(x + offset);
50}
51
52consteval word base58_conversion_radix() {
53 if constexpr(sizeof(word) == 8) {
54 // 58^10 largest that fits into a 64 bit word
55 return 430804206899405824U;
56 } else {
57 // 58^5 largest that fits into a 32 bit word
58 return 656356768U;
59 }
60}
61
62consteval size_t base58_conversion_radix_digits() {
63 if constexpr(sizeof(word) == 8) {
64 return 10;
65 } else {
66 return 5;
67 }
68}
69
70constexpr std::pair<uint8_t, word> divmod_58(word x) {
71 BOTAN_DEBUG_ASSERT(x < base58_conversion_radix());
72
73 word q = 0;
74
75 // Division by constant 58
76 //
77 // Compilers will *usually* convert an expression like `x / 58` into
78 // exactly this kind of operation, but not necessarily always...
79 if constexpr(sizeof(word) == 4) {
80 const uint64_t magic = 2369637129; // ceil(2**36 / 29)
81 uint64_t z = magic * x;
82 q = z >> 37;
83 } else {
84 const uint64_t magic = 5088756985850910791; // ceil(2**67 / 29)
85 uint64_t lo = 0; // unused
86 uint64_t hi = 0;
87 mul64x64_128(magic, x >> 1, &lo, &hi);
88 q = static_cast<word>(hi >> 3);
89 }
90
91 const uint8_t r = static_cast<uint8_t>(x - q * 58);
92 return std::make_pair(r, q);
93}
94
95std::string base58_encode(BigInt v, size_t leading_zeros) {
96 constexpr word radix = base58_conversion_radix();
97 constexpr size_t radix_digits = base58_conversion_radix_digits();
98
99 BigInt q;
100 std::vector<uint8_t> digits;
101
102 while(v.is_nonzero()) {
103 word r = 0;
104 ct_divide_word(v, radix, q, r);
105
106 for(size_t i = 0; i != radix_digits; ++i) {
107 const auto [r58, q58] = divmod_58(r);
108 digits.push_back(r58);
109 r = q58;
110 }
111 v.swap(q);
112 }
113
114 // remove leading zeros
115 while(!digits.empty() && digits.back() == 0) {
116 digits.pop_back();
117 }
118
119 std::string result;
120
121 for(uint8_t d : digits) {
122 result.push_back(lookup_base58_char(d));
123 }
124
125 for(size_t i = 0; i != leading_zeros; ++i) {
126 result.push_back('1'); // 'zero' byte
127 }
128
129 return std::string(result.rbegin(), result.rend());
130}
131
132template <typename T, typename Z>
133size_t count_leading_zeros(const T input[], size_t input_length, Z zero) {
134 size_t leading_zeros = 0;
135
136 while(leading_zeros < input_length && input[leading_zeros] == zero) {
137 leading_zeros += 1;
138 }
139
140 return leading_zeros;
141}
142
143uint8_t base58_value_of(char input) {
144 // "123456789 ABCDEFGH JKLMN PQRSTUVWXYZ abcdefghijk mnopqrstuvwxyz"
145
146 const uint8_t c = static_cast<uint8_t>(input);
147
148 const auto is_dec_19 = CT::Mask<uint8_t>::is_within_range(c, uint8_t('1'), uint8_t('9'));
149 const auto is_alpha_AH = CT::Mask<uint8_t>::is_within_range(c, uint8_t('A'), uint8_t('H'));
150 const auto is_alpha_JN = CT::Mask<uint8_t>::is_within_range(c, uint8_t('J'), uint8_t('N'));
151 const auto is_alpha_PZ = CT::Mask<uint8_t>::is_within_range(c, uint8_t('P'), uint8_t('Z'));
152
153 const auto is_alpha_ak = CT::Mask<uint8_t>::is_within_range(c, uint8_t('a'), uint8_t('k'));
154 const auto is_alpha_mz = CT::Mask<uint8_t>::is_within_range(c, uint8_t('m'), uint8_t('z'));
155
156 const uint8_t c_dec_19 = c - uint8_t('1');
157 const uint8_t c_AH = c - uint8_t('A') + 9;
158 const uint8_t c_JN = c - uint8_t('J') + 17;
159 const uint8_t c_PZ = c - uint8_t('P') + 22;
160
161 const uint8_t c_ak = c - uint8_t('a') + 33;
162 const uint8_t c_mz = c - uint8_t('m') + 44;
163
164 uint8_t ret = 0xFF; // default value
165
166 ret = is_dec_19.select(c_dec_19, ret);
167 ret = is_alpha_AH.select(c_AH, ret);
168 ret = is_alpha_JN.select(c_JN, ret);
169 ret = is_alpha_PZ.select(c_PZ, ret);
170 ret = is_alpha_ak.select(c_ak, ret);
171 ret = is_alpha_mz.select(c_mz, ret);
172 return ret;
173}
174
175} // namespace
176
177std::string base58_encode(const uint8_t input[], size_t input_length) {
178 BigInt v(input, input_length);
179 return base58_encode(v, count_leading_zeros(input, input_length, 0));
180}
181
182std::string base58_check_encode(const uint8_t input[], size_t input_length) {
183 BigInt v(input, input_length);
184 v <<= 32;
185 v += sha256_d_checksum(input, input_length);
186 return base58_encode(v, count_leading_zeros(input, input_length, 0));
187}
188
189std::vector<uint8_t> base58_decode(const char input[], size_t input_length) {
190 const size_t leading_zeros = count_leading_zeros(input, input_length, '1');
191
192 std::vector<uint8_t> digits;
193
194 for(size_t i = leading_zeros; i != input_length; ++i) {
195 const char c = input[i];
196
197 if(c == ' ' || c == '\n') {
198 continue;
199 }
200
201 const uint8_t idx = base58_value_of(c);
202
203 if(idx == 0xFF) {
204 throw Decoding_Error("Invalid base58");
205 }
206
207 digits.push_back(idx);
208 }
209
210 BigInt v;
211
212 constexpr word radix1 = 58;
213 constexpr word radix2 = 58 * 58;
214 constexpr word radix3 = 58 * 58 * 58;
215 constexpr word radix4 = 58 * 58 * 58 * 58;
216
217 std::span<uint8_t> remaining{digits};
218
219 while(remaining.size() >= 4) {
220 const word accum = radix3 * remaining[0] + radix2 * remaining[1] + radix1 * remaining[2] + remaining[3];
221 v *= radix4;
222 v += accum;
223 remaining = remaining.subspan(4);
224 }
225
226 while(!remaining.empty()) {
227 v *= 58;
228 v += remaining[0];
229 remaining = remaining.subspan(1);
230 }
231
232 return v.serialize(v.bytes() + leading_zeros);
233}
234
235std::vector<uint8_t> base58_check_decode(const char input[], size_t input_length) {
236 std::vector<uint8_t> dec = base58_decode(input, input_length);
237
238 if(dec.size() < 4) {
239 throw Decoding_Error("Invalid base58 too short for checksum");
240 }
241
242 const uint32_t computed_checksum = sha256_d_checksum(dec.data(), dec.size() - 4);
243 const uint32_t checksum = load_be<uint32_t>(&dec[dec.size() - 4], 0);
244
245 if(checksum != computed_checksum) {
246 throw Decoding_Error("Invalid base58 checksum");
247 }
248
249 dec.resize(dec.size() - 4);
250
251 return dec;
252}
253
254} // namespace Botan
#define BOTAN_DEBUG_ASSERT(expr)
Definition assert.h:129
size_t bytes() const
Definition bigint.cpp:298
T serialize(size_t len) const
Definition bigint.h:711
static constexpr Mask< T > is_within_range(T v, T l, T u)
Definition ct_utils.h:498
static constexpr Mask< T > is_gt(T x, T y)
Definition ct_utils.h:486
static std::unique_ptr< HashFunction > create_or_throw(std::string_view algo_spec, std::string_view provider="")
Definition hash.cpp:308
std::vector< uint8_t > base58_check_decode(const char input[], size_t input_length)
Definition base58.cpp:235
std::string base58_encode(const uint8_t input[], size_t input_length)
Definition base58.cpp:177
std::string base58_check_encode(const uint8_t input[], size_t input_length)
Definition base58.cpp:182
void ct_divide_word(const BigInt &x, word y, BigInt &q_out, word &r_out)
Definition divide.cpp:122
std::vector< uint8_t > base58_decode(const char input[], size_t input_length)
Definition base58.cpp:189
std::conditional_t< HasNative64BitRegisters, std::uint64_t, uint32_t > word
Definition types.h:119
constexpr void mul64x64_128(uint64_t a, uint64_t b, uint64_t *lo, uint64_t *hi)
Definition mul128.h:24
constexpr auto load_be(ParamTs &&... params)
Definition loadstor.h:504