rippled
Loading...
Searching...
No Matches
libxrpl/basics/base64.cpp
1// Distributed under the Boost Software License, Version 1.0. (See accompanying
2// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
3//
4
5/*
6 Portions from http://www.adp-gmbh.ch/cpp/common/base64.html
7 Copyright notice:
8
9 base64.cpp and base64.h
10
11 Copyright (C) 2004-2008 René Nyffenegger
12
13 This source code is provided 'as-is', without any express or implied
14 warranty. In no event will the author be held liable for any damages
15 arising from the use of this software.
16
17 Permission is granted to anyone to use this software for any purpose,
18 including commercial applications, and to alter it and redistribute it
19 freely, subject to the following restrictions:
20
21 1. The origin of this source code must not be misrepresented; you must not
22 claim that you wrote the original source code. If you use this source code
23 in a product, an acknowledgment in the product documentation would be
24 appreciated but is not required.
25
26 2. Altered source versions must be plainly marked as such, and must not be
27 misrepresented as being the original source code.
28
29 3. This notice may not be removed or altered from any source distribution.
30
31 René Nyffenegger rene.nyffenegger@adp-gmbh.ch
32
33*/
34
35#include <xrpl/basics/base64.h>
36
37#include <cstddef>
38#include <cstdint>
39#include <string>
40#include <string_view>
41#include <utility>
42
43namespace xrpl {
44
45namespace base64 {
46
47inline char const*
49{
50 static char constexpr tab[] = {
51 "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"};
52 return &tab[0];
53}
54
55inline signed char const*
57{
58 static signed char constexpr tab[] = {
59 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, // 0-15
60 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, // 16-31
61 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 62, -1, -1, -1, 63, // 32-47
62 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, -1, -1, -1, -1, -1, -1, // 48-63
63 -1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, // 64-79
64 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, -1, -1, -1, -1, -1, // 80-95
65 -1, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, // 96-111
66 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, -1, -1, -1, -1, -1, // 112-127
67 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, // 128-143
68 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, // 144-159
69 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, // 160-175
70 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, // 176-191
71 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, // 192-207
72 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, // 208-223
73 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, // 224-239
74 -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1 // 240-255
75 };
76 return &tab[0];
77}
78
81{
82 return 4 * ((n + 2) / 3);
83}
84
87{
88 return ((n / 4) * 3) + 2;
89}
90
104encode(void* dest, void const* src, std::size_t len)
105{
106 char* out = static_cast<char*>(dest); // NOLINT(misc-const-correctness)
107 char const* in = static_cast<char const*>(src);
108 auto const tab = base64::get_alphabet();
109
110 for (auto n = len / 3; n != 0u; --n)
111 {
112 *out++ = tab[(in[0] & 0xfc) >> 2];
113 *out++ = tab[((in[0] & 0x03) << 4) + ((in[1] & 0xf0) >> 4)];
114 *out++ = tab[((in[2] & 0xc0) >> 6) + ((in[1] & 0x0f) << 2)];
115 *out++ = tab[in[2] & 0x3f];
116 in += 3;
117 }
118
119 // NOLINTNEXTLINE(bugprone-switch-missing-default-case)
120 switch (len % 3)
121 {
122 case 2:
123 *out++ = tab[(in[0] & 0xfc) >> 2];
124 *out++ = tab[((in[0] & 0x03) << 4) + ((in[1] & 0xf0) >> 4)];
125 *out++ = tab[(in[1] & 0x0f) << 2];
126 *out++ = '=';
127 break;
128
129 case 1:
130 *out++ = tab[(in[0] & 0xfc) >> 2];
131 *out++ = tab[((in[0] & 0x03) << 4)];
132 *out++ = '=';
133 *out++ = '=';
134 break;
135
136 case 0:
137 break;
138 }
139
140 return out - static_cast<char*>(dest);
141}
142
155decode(void* dest, char const* src, std::size_t len)
156{
157 char* out = static_cast<char*>(dest); // NOLINT(misc-const-correctness)
158 auto in = reinterpret_cast<unsigned char const*>(src);
159 unsigned char c3[3]{}, c4[4]{};
160 int i = 0;
161 int j = 0;
162
163 auto const inverse = base64::get_inverse();
164
165 while (((len--) != 0u) && *in != '=')
166 {
167 auto const v = inverse[*in];
168 if (v == -1)
169 break;
170 ++in;
171 c4[i] = v;
172 if (++i; i == 4)
173 {
174 c3[0] = (c4[0] << 2) + ((c4[1] & 0x30) >> 4);
175 c3[1] = ((c4[1] & 0xf) << 4) + ((c4[2] & 0x3c) >> 2);
176 c3[2] = ((c4[2] & 0x3) << 6) + c4[3];
177
178 for (i = 0; i < 3; i++)
179 *out++ = c3[i];
180 i = 0;
181 }
182 }
183
184 if (i != 0)
185 {
186 c3[0] = (c4[0] << 2) + ((c4[1] & 0x30) >> 4);
187 c3[1] = ((c4[1] & 0xf) << 4) + ((c4[2] & 0x3c) >> 2);
188 c3[2] = ((c4[2] & 0x3) << 6) + c4[3];
189
190 for (j = 0; j < i - 1; j++)
191 *out++ = c3[j];
192 }
193
194 return {out - static_cast<char*>(dest), in - reinterpret_cast<unsigned char const*>(src)};
195}
196
197} // namespace base64
198
201{
202 std::string dest;
203 dest.resize(base64::encoded_size(len));
204 dest.resize(base64::encode(&dest[0], data, len));
205 return dest;
206}
207
210{
211 std::string dest;
212 dest.resize(base64::decoded_size(data.size()));
213 auto const result = base64::decode(&dest[0], data.data(), data.size());
214 dest.resize(result.first);
215 return dest;
216}
217
218} // namespace xrpl
signed char const * get_inverse()
std::size_t constexpr encoded_size(std::size_t n)
Returns max chars needed to encode a base64 string.
char const * get_alphabet()
std::pair< std::size_t, std::size_t > decode(void *dest, char const *src, std::size_t len)
Decode a padded base64 string into a series of octets.
std::size_t encode(void *dest, void const *src, std::size_t len)
Encode a series of octets as a padded, base64 string.
std::size_t constexpr decoded_size(std::size_t n)
Returns max bytes needed to decode a base64 string.
Use hash_* containers for keys that do not need a cryptographically secure hashing algorithm.
Definition algorithm.h:5
std::string base64_decode(std::string_view data)
std::string base64_encode(std::uint8_t const *data, std::size_t len)
T resize(T... args)