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 ripple {
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,
60 -1, -1, -1, -1, -1, -1, -1, -1, // 0-15
61 -1, -1, -1, -1, -1, -1, -1, -1,
62 -1, -1, -1, -1, -1, -1, -1, -1, // 16-31
63 -1, -1, -1, -1, -1, -1, -1, -1,
64 -1, -1, -1, 62, -1, -1, -1, 63, // 32-47
65 52, 53, 54, 55, 56, 57, 58, 59,
66 60, 61, -1, -1, -1, -1, -1, -1, // 48-63
67 -1, 0, 1, 2, 3, 4, 5, 6,
68 7, 8, 9, 10, 11, 12, 13, 14, // 64-79
69 15, 16, 17, 18, 19, 20, 21, 22,
70 23, 24, 25, -1, -1, -1, -1, -1, // 80-95
71 -1, 26, 27, 28, 29, 30, 31, 32,
72 33, 34, 35, 36, 37, 38, 39, 40, // 96-111
73 41, 42, 43, 44, 45, 46, 47, 48,
74 49, 50, 51, -1, -1, -1, -1, -1, // 112-127
75 -1, -1, -1, -1, -1, -1, -1, -1,
76 -1, -1, -1, -1, -1, -1, -1, -1, // 128-143
77 -1, -1, -1, -1, -1, -1, -1, -1,
78 -1, -1, -1, -1, -1, -1, -1, -1, // 144-159
79 -1, -1, -1, -1, -1, -1, -1, -1,
80 -1, -1, -1, -1, -1, -1, -1, -1, // 160-175
81 -1, -1, -1, -1, -1, -1, -1, -1,
82 -1, -1, -1, -1, -1, -1, -1, -1, // 176-191
83 -1, -1, -1, -1, -1, -1, -1, -1,
84 -1, -1, -1, -1, -1, -1, -1, -1, // 192-207
85 -1, -1, -1, -1, -1, -1, -1, -1,
86 -1, -1, -1, -1, -1, -1, -1, -1, // 208-223
87 -1, -1, -1, -1, -1, -1, -1, -1,
88 -1, -1, -1, -1, -1, -1, -1, -1, // 224-239
89 -1, -1, -1, -1, -1, -1, -1, -1,
90 -1, -1, -1, -1, -1, -1, -1, -1 // 240-255
91 };
92 return &tab[0];
93}
94
97{
98 return 4 * ((n + 2) / 3);
99}
100
103{
104 return ((n / 4) * 3) + 2;
105}
106
120encode(void* dest, void const* src, std::size_t len)
121{
122 char* out = static_cast<char*>(dest);
123 char const* in = static_cast<char const*>(src);
124 auto const tab = base64::get_alphabet();
125
126 for (auto n = len / 3; n--;)
127 {
128 *out++ = tab[(in[0] & 0xfc) >> 2];
129 *out++ = tab[((in[0] & 0x03) << 4) + ((in[1] & 0xf0) >> 4)];
130 *out++ = tab[((in[2] & 0xc0) >> 6) + ((in[1] & 0x0f) << 2)];
131 *out++ = tab[in[2] & 0x3f];
132 in += 3;
133 }
134
135 switch (len % 3)
136 {
137 case 2:
138 *out++ = tab[(in[0] & 0xfc) >> 2];
139 *out++ = tab[((in[0] & 0x03) << 4) + ((in[1] & 0xf0) >> 4)];
140 *out++ = tab[(in[1] & 0x0f) << 2];
141 *out++ = '=';
142 break;
143
144 case 1:
145 *out++ = tab[(in[0] & 0xfc) >> 2];
146 *out++ = tab[((in[0] & 0x03) << 4)];
147 *out++ = '=';
148 *out++ = '=';
149 break;
150
151 case 0:
152 break;
153 }
154
155 return out - static_cast<char*>(dest);
156}
157
170decode(void* dest, char const* src, std::size_t len)
171{
172 char* out = static_cast<char*>(dest);
173 auto in = reinterpret_cast<unsigned char const*>(src);
174 unsigned char c3[3]{}, c4[4]{};
175 int i = 0;
176 int j = 0;
177
178 auto const inverse = base64::get_inverse();
179
180 while (len-- && *in != '=')
181 {
182 auto const v = inverse[*in];
183 if (v == -1)
184 break;
185 ++in;
186 c4[i] = v;
187 if (++i == 4)
188 {
189 c3[0] = (c4[0] << 2) + ((c4[1] & 0x30) >> 4);
190 c3[1] = ((c4[1] & 0xf) << 4) + ((c4[2] & 0x3c) >> 2);
191 c3[2] = ((c4[2] & 0x3) << 6) + c4[3];
192
193 for (i = 0; i < 3; i++)
194 *out++ = c3[i];
195 i = 0;
196 }
197 }
198
199 if (i)
200 {
201 c3[0] = (c4[0] << 2) + ((c4[1] & 0x30) >> 4);
202 c3[1] = ((c4[1] & 0xf) << 4) + ((c4[2] & 0x3c) >> 2);
203 c3[2] = ((c4[2] & 0x3) << 6) + c4[3];
204
205 for (j = 0; j < i - 1; j++)
206 *out++ = c3[j];
207 }
208
209 return {
210 out - static_cast<char*>(dest),
211 in - reinterpret_cast<unsigned char const*>(src)};
212}
213
214} // namespace base64
215
218{
219 std::string dest;
220 dest.resize(base64::encoded_size(len));
221 dest.resize(base64::encode(&dest[0], data, len));
222 return dest;
223}
224
227{
228 std::string dest;
229 dest.resize(base64::decoded_size(data.size()));
230 auto const result = base64::decode(&dest[0], data.data(), data.size());
231 dest.resize(result.first);
232 return dest;
233}
234
235} // namespace ripple
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.
signed char const * get_inverse()
char const * get_alphabet()
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.
std::size_t constexpr encoded_size(std::size_t n)
Returns max chars needed to encode a base64 string.
Use hash_* containers for keys that do not need a cryptographically secure hashing algorithm.
Definition algorithm.h:6
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)