-
Notifications
You must be signed in to change notification settings - Fork 4k
/
Copy pathconverter.cc
304 lines (246 loc) · 9.59 KB
/
converter.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
/*
Copyright (c) 2018, 2024, Oracle and/or its affiliates.
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License, version 2.0,
as published by the Free Software Foundation.
This program is designed to work with certain software (including
but not limited to OpenSSL) that is licensed under separate terms,
as designated in a particular file or component or in included license
documentation. The authors of MySQL hereby grant you an additional
permission to link the program and your derivative works with the
separately licensed software that they have either included with
the program or referenced in the documentation.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License, version 2.0, for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include "converter.h"
#include <string.h>
#include <iostream>
namespace keyring {
const size_t WORD_32 = 4; //!< number of bytes in 32 bit word
const size_t WORD_64 = 8; //!< number of bytes in 64 bit word
const size_t LENGTHS_COUNT = 5; //!< number of length values in serialization
// determine native architecture at startup
const Converter::Arch Converter::native_arch{Converter::detect_native_arch()};
/**
fetches native machine architecture of server binary
@return - native machine architecture
*/
Converter::Arch Converter::get_native_arch() { return native_arch; }
/**
(static) determines width of architecture word in bytes
@param arch - architecture to return word width of
@retval > 0 - word width of the architecture
@retval = 0 - architecture has no known word width
*/
size_t Converter::get_width(Arch arch) {
if (arch == Arch::LE_64 || arch == Arch::BE_64) return WORD_64;
if (arch == Arch::LE_32 || arch == Arch::BE_32) return WORD_32;
return 0;
}
/**
(static) calculates native unsigned integer (size_t) value of binary buffer
- buffer has to provide for at least sizeof(size_t) bytes
@param binary_value - pointer to native binary representation of unsigned
value
@return - unsigned integer (size_t) value of binary buffer value
*/
size_t Converter::native_value(char const *binary_value) {
// overwrite variable with bytes from binary buffer
size_t real_value = 0;
memcpy(&real_value, binary_value, sizeof(size_t));
return real_value;
}
/**
(static) converts serialized key data from one architecture to another
- key lengths word widths are expanded/contracted, key data just copied
@param[in] data - buffer holding key data to be converted
@param[in] data_size - size of the key data within buffer
@param[in] src - machine architecture of the input key data
@param[in] dst - machine architecture of the output key data
@param[out] out - output key data
@retval false - data was successfully converted
@retval true - error occurred during conversion
*/
bool Converter::convert_data(char const *data, size_t data_size, Arch src,
Arch dst, std::string &out) {
// at least source or destination has to be native
if (src != native_arch && dst != native_arch) return true;
// no data - we can convert that :)
if (data_size == 0) {
out = std::string();
return false;
}
// no need to convert between identical architectures
if (src == dst) {
out = std::string(data, data_size);
return false;
}
// get word width of source and destination architecture
auto src_width = get_width(src);
auto dst_width = get_width(dst);
size_t loc = 0;
std::string output;
char number[WORD_64] = {0};
size_t lengths[LENGTHS_COUNT] = {0};
std::string key_content;
// load remaining keys and convert them
while (data_size >= loc + LENGTHS_COUNT * src_width) {
key_content.clear();
// load file length values, convert them and append
for (size_t i = 0; i < LENGTHS_COUNT; i++) {
// convert value to different architecture
const size_t converted_width = convert(data + loc, number, src, dst);
if (i > 0) key_content.append(number, converted_width);
// determine length integer value
if (src == get_native_arch())
lengths[i] = native_value(data + loc);
else
lengths[i] = native_value(number);
// move to next length
loc += src_width;
}
// real size without padding has to be smaller that total size
const size_t real_size = lengths[1] + lengths[2] + lengths[3] + lengths[4];
if (lengths[0] < real_size) return true;
// we also have to have at least remaining key data
if (loc + lengths[0] - LENGTHS_COUNT * src_width > data_size) return true;
// append only key data without padding
key_content.append(data + loc, real_size);
loc += lengths[0] - LENGTHS_COUNT * src_width;
// append required padding to destination
auto total = LENGTHS_COUNT * dst_width + real_size;
const size_t padding = (dst_width - total % dst_width) % dst_width;
key_content.append(padding, '\0');
// new key package size is waranted
lengths[0] = total + padding;
char tmp_buffer[WORD_64];
memcpy(tmp_buffer, lengths, sizeof(size_t));
// conversion may be necessary
if (dst != get_native_arch()) {
auto converted_width = convert(tmp_buffer, number, src, dst);
output += std::string(number, converted_width);
output += key_content;
} else {
output += std::string(tmp_buffer, sizeof(size_t));
output += key_content;
}
}
// we must've taken all keys
if (loc != data_size) return true;
// return converted data buffer
out = output;
return false;
}
/**
(static) converts binary length representation between architecture types
@param[in] src - input binary representation
@param[out] dst - output binary representation
@param[in] src_t - architecture type of input representation
@param[in] dst_t - architecture type of output representation
@retval > 0 - size of destination representation
@retval = 0 - conversion was not possible
*/
size_t Converter::convert(char const *src, char *dst, Arch src_t, Arch dst_t) {
// source and destination type should be known
if (src_t == Arch::UNKNOWN || dst_t == Arch::UNKNOWN) return 0;
// find out source and destination characteristics
const size_t src_width = get_width(src_t);
const size_t dst_width = get_width(dst_t);
const bool src_is_le = get_endian(src_t) == Endian::LITTLE;
const bool dst_is_le = get_endian(dst_t) == Endian::LITTLE;
// determine required operations
const bool swap = (src_is_le != dst_is_le);
const bool grow = (src_width < dst_width);
const bool crop = (src_width > dst_width);
// crop security - we mustn't lose precision
if (crop) {
if (src_is_le) // for little endian ending bytes must be zero
{
if (src[4] | src[5] | src[6] | src[7]) return 0;
} else // for big endian leading bytes must be zero
{
if (src[0] | src[1] | src[2] | src[3]) return 0;
}
}
// if needed, prepare reversed number
char swapped_src[WORD_64] = {0};
if (swap) {
// copy reversed byte order
for (size_t i = 0; i < src_width; i++)
swapped_src[i] = src[src_width - i - 1];
// use reversed representation instead of original
src = swapped_src;
}
// ================== CASE 1 - no change in bit width ================== //
if (!grow && !crop) {
memcpy(dst, src, dst_width);
}
// ================== CASE 2 - grow bit width ========================== //
else if (grow) {
if (dst_is_le) {
// copy to starting bytes, clear upper bytes
memcpy(dst, src, src_width);
memset(dst + src_width, 0, dst_width - src_width);
} else {
// clear starting bytes, copy to upper bytes
memset(dst, 0, dst_width - src_width);
memcpy(dst + dst_width - src_width, src, src_width);
}
}
// ================== CASE 3 - crop bit width ========================== //
else if (crop) {
if (dst_is_le)
memcpy(dst, src, dst_width);
else
memcpy(dst, src + 4, dst_width);
}
// we're returning size of output binary representation
return dst_width;
}
// =========================== PRIVATE UTILITIES ============================ //
/**
(static) detects native architecture type of machine server is running on
- to be called only at server startup
@return - native architecture of machine
*/
Converter::Arch Converter::detect_native_arch() {
auto type = Arch::UNKNOWN;
// determine bit width
const size_t bit_width = 8 * sizeof(size_t);
// determine endianness
size_t number = 1;
const bool isLittleEndian = *(char *)(&number);
// assign architecture type based on findings
switch (bit_width) {
case 32:
if (isLittleEndian)
type = Arch::LE_32;
else
type = Arch::BE_32;
break;
case 64:
if (isLittleEndian)
type = Arch::LE_64;
else
type = Arch::BE_64;
break;
}
return type;
}
/**
(static) determines endianness of architecture
@param arch - architecture to return endianness of
@return - endianness of the architecture
*/
Converter::Endian Converter::get_endian(Arch arch) {
if (arch == Arch::LE_64 || arch == Arch::LE_32) return Endian::LITTLE;
if (arch == Arch::BE_64 || arch == Arch::BE_32) return Endian::BIG;
return Endian::UNKNOWN;
}
} /* namespace keyring */