7#include <botan/base58.h>
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>
21uint32_t sha256_d_checksum(
const uint8_t input[],
size_t input_length) {
24 std::vector<uint8_t> checksum(32);
26 sha256->update(input, input_length);
27 sha256->final(checksum);
29 sha256->update(checksum);
30 sha256->final(checksum);
35char lookup_base58_char(uint8_t x) {
49 return static_cast<char>(x + offset);
52consteval word base58_conversion_radix() {
53 if constexpr(
sizeof(
word) == 8) {
55 return 430804206899405824U;
62consteval size_t base58_conversion_radix_digits() {
63 if constexpr(
sizeof(
word) == 8) {
70constexpr std::pair<uint8_t, word> divmod_58(
word x) {
79 if constexpr(
sizeof(
word) == 4) {
80 const uint64_t magic = 2369637129;
81 uint64_t z = magic * x;
84 const uint64_t magic = 5088756985850910791;
88 q =
static_cast<word>(hi >> 3);
91 const uint8_t r =
static_cast<uint8_t
>(x - q * 58);
92 return std::make_pair(r, q);
96 constexpr word radix = base58_conversion_radix();
97 constexpr size_t radix_digits = base58_conversion_radix_digits();
100 std::vector<uint8_t> digits;
102 while(v.is_nonzero()) {
106 for(
size_t i = 0; i != radix_digits; ++i) {
107 const auto [r58, q58] = divmod_58(r);
108 digits.push_back(r58);
115 while(!digits.empty() && digits.back() == 0) {
121 for(uint8_t d : digits) {
122 result.push_back(lookup_base58_char(d));
125 for(
size_t i = 0; i != leading_zeros; ++i) {
126 result.push_back(
'1');
129 return std::string(result.rbegin(), result.rend());
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;
136 while(leading_zeros < input_length && input[leading_zeros] == zero) {
140 return leading_zeros;
143uint8_t base58_value_of(
char input) {
146 const uint8_t c =
static_cast<uint8_t
>(input);
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;
161 const uint8_t c_ak = c - uint8_t(
'a') + 33;
162 const uint8_t c_mz = c - uint8_t(
'm') + 44;
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);
178 BigInt v(input, input_length);
179 return base58_encode(v, count_leading_zeros(input, input_length, 0));
183 BigInt v(input, input_length);
185 v += sha256_d_checksum(input, input_length);
186 return base58_encode(v, count_leading_zeros(input, input_length, 0));
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');
192 std::vector<uint8_t> digits;
194 for(
size_t i = leading_zeros; i != input_length; ++i) {
195 const char c = input[i];
197 if(c ==
' ' || c ==
'\n') {
201 const uint8_t idx = base58_value_of(c);
207 digits.push_back(idx);
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;
217 std::span<uint8_t> remaining{digits};
219 while(remaining.size() >= 4) {
220 const word accum = radix3 * remaining[0] + radix2 * remaining[1] + radix1 * remaining[2] + remaining[3];
223 remaining = remaining.subspan(4);
226 while(!remaining.empty()) {
229 remaining = remaining.subspan(1);
236 std::vector<uint8_t> dec =
base58_decode(input, input_length);
242 const uint32_t computed_checksum = sha256_d_checksum(dec.data(), dec.size() - 4);
245 if(checksum != computed_checksum) {
249 dec.resize(dec.size() - 4);
#define BOTAN_DEBUG_ASSERT(expr)
T serialize(size_t len) const
static constexpr Mask< T > is_within_range(T v, T l, T u)
static constexpr Mask< T > is_gt(T x, T y)
static std::unique_ptr< HashFunction > create_or_throw(std::string_view algo_spec, std::string_view provider="")
std::vector< uint8_t > base58_check_decode(const char input[], size_t input_length)
std::string base58_encode(const uint8_t input[], size_t input_length)
std::string base58_check_encode(const uint8_t input[], size_t input_length)
void ct_divide_word(const BigInt &x, word y, BigInt &q_out, word &r_out)
std::vector< uint8_t > base58_decode(const char input[], size_t input_length)
std::conditional_t< HasNative64BitRegisters, std::uint64_t, uint32_t > word
constexpr void mul64x64_128(uint64_t a, uint64_t b, uint64_t *lo, uint64_t *hi)
constexpr auto load_be(ParamTs &&... params)