Botan 3.6.1
Crypto and TLS for C&
hex.cpp
Go to the documentation of this file.
1/*
2* Hex Encoding and Decoding
3* (C) 2010,2020 Jack Lloyd
4*
5* Botan is released under the Simplified BSD License (see license.txt)
6*/
7
8#include <botan/hex.h>
9
10#include <botan/exceptn.h>
11#include <botan/mem_ops.h>
12#include <botan/internal/charset.h>
13#include <botan/internal/ct_utils.h>
14#include <botan/internal/fmt.h>
15#include <botan/internal/int_utils.h>
16#include <botan/internal/loadstor.h>
17
18namespace Botan {
19
20namespace {
21
22uint16_t hex_encode_2nibble(uint8_t n8, bool uppercase) {
23 // Offset for upper or lower case 'a' resp
24 const uint16_t a_mask = uppercase ? 0x0707 : 0x2727;
25
26 const uint16_t n = (static_cast<uint16_t>(n8 & 0xF0) << 4) | (n8 & 0x0F);
27 // n >= 10? If so add offset
28 const uint16_t diff = swar_lt<uint16_t>(0x0909, n) & a_mask;
29 // Can't overflow between bytes, so don't need explicit SWAR addition:
30 return n + 0x3030 + diff;
31}
32
33} // namespace
34
35void hex_encode(char output[], const uint8_t input[], size_t input_length, bool uppercase) {
36 for(size_t i = 0; i != input_length; ++i) {
37 const uint16_t h = hex_encode_2nibble(input[i], uppercase);
38 output[2 * i] = get_byte<0>(h);
39 output[2 * i + 1] = get_byte<1>(h);
40 }
41}
42
43std::string hex_encode(const uint8_t input[], size_t input_length, bool uppercase) {
44 std::string output(2 * input_length, 0);
45
46 if(input_length) {
47 hex_encode(&output.front(), input, input_length, uppercase);
48 }
49
50 return output;
51}
52
53namespace {
54
55uint8_t hex_char_to_bin(char input) {
56 // Starts of valid value ranges (v_lo) and their lengths (v_range)
57 constexpr uint64_t v_lo = make_uint64(0, '0', 'a', 'A', ' ', '\n', '\t', '\r');
58 constexpr uint64_t v_range = make_uint64(0, 10, 6, 6, 1, 1, 1, 1);
59
60 const uint8_t x = static_cast<uint8_t>(input);
61 const uint64_t x8 = x * 0x0101010101010101;
62
63 const uint64_t v_mask = swar_in_range<uint64_t>(x8, v_lo, v_range) ^ 0x8000000000000000;
64
65 // This is the offset added to x to get the value we need
66 const uint64_t val_v = 0xd0a9c960767773 ^ static_cast<uint64_t>(0xFF - x) << 56;
67
68 return x + static_cast<uint8_t>(val_v >> (8 * index_of_first_set_byte(v_mask)));
69}
70
71} // namespace
72
73size_t hex_decode(uint8_t output[], const char input[], size_t input_length, size_t& input_consumed, bool ignore_ws) {
74 uint8_t* out_ptr = output;
75 bool top_nibble = true;
76
77 clear_mem(output, input_length / 2);
78
79 for(size_t i = 0; i != input_length; ++i) {
80 const uint8_t bin = hex_char_to_bin(input[i]);
81
82 if(bin >= 0x10) {
83 if(bin == 0x80 && ignore_ws) {
84 continue;
85 }
86
87 throw Invalid_Argument(fmt("hex_decode: invalid character '{}'", format_char_for_display(input[i])));
88 }
89
90 if(top_nibble) {
91 *out_ptr |= bin << 4;
92 } else {
93 *out_ptr |= bin;
94 }
95
96 top_nibble = !top_nibble;
97 if(top_nibble) {
98 ++out_ptr;
99 }
100 }
101
102 input_consumed = input_length;
103 size_t written = (out_ptr - output);
104
105 /*
106 * We only got half of a uint8_t at the end; zap the half-written
107 * output and mark it as unread
108 */
109 if(!top_nibble) {
110 *out_ptr = 0;
111 input_consumed -= 1;
112 }
113
114 return written;
115}
116
117size_t hex_decode(uint8_t output[], const char input[], size_t input_length, bool ignore_ws) {
118 size_t consumed = 0;
119 size_t written = hex_decode(output, input, input_length, consumed, ignore_ws);
120
121 if(consumed != input_length) {
122 throw Invalid_Argument("hex_decode: input did not have full bytes");
123 }
124
125 return written;
126}
127
128size_t hex_decode(uint8_t output[], std::string_view input, bool ignore_ws) {
129 return hex_decode(output, input.data(), input.length(), ignore_ws);
130}
131
132size_t hex_decode(std::span<uint8_t> output, std::string_view input, bool ignore_ws) {
133 return hex_decode(output.data(), input.data(), input.length(), ignore_ws);
134}
135
136secure_vector<uint8_t> hex_decode_locked(const char input[], size_t input_length, bool ignore_ws) {
137 secure_vector<uint8_t> bin(1 + input_length / 2);
138
139 size_t written = hex_decode(bin.data(), input, input_length, ignore_ws);
140
141 bin.resize(written);
142 return bin;
143}
144
145secure_vector<uint8_t> hex_decode_locked(std::string_view input, bool ignore_ws) {
146 return hex_decode_locked(input.data(), input.size(), ignore_ws);
147}
148
149std::vector<uint8_t> hex_decode(const char input[], size_t input_length, bool ignore_ws) {
150 std::vector<uint8_t> bin(1 + input_length / 2);
151
152 size_t written = hex_decode(bin.data(), input, input_length, ignore_ws);
153
154 bin.resize(written);
155 return bin;
156}
157
158std::vector<uint8_t> hex_decode(std::string_view input, bool ignore_ws) {
159 return hex_decode(input.data(), input.size(), ignore_ws);
160}
161
162} // namespace Botan
constexpr uint8_t get_byte(T input)
Definition loadstor.h:75
std::string format_char_for_display(char c)
Definition charset.cpp:98
constexpr T swar_lt(T a, T b)
Definition int_utils.h:91
constexpr auto out_ptr(T &outptr) noexcept
Definition stl_util.h:420
constexpr uint64_t make_uint64(uint8_t i0, uint8_t i1, uint8_t i2, uint8_t i3, uint8_t i4, uint8_t i5, uint8_t i6, uint8_t i7)
Definition loadstor.h:117
std::string fmt(std::string_view format, const T &... args)
Definition fmt.h:53
constexpr T swar_in_range(T v, T lower, T upper)
Definition int_utils.h:114
secure_vector< uint8_t > hex_decode_locked(const char input[], size_t input_length, bool ignore_ws)
Definition hex.cpp:136
constexpr size_t index_of_first_set_byte(T v)
Definition int_utils.h:130
void hex_encode(char output[], const uint8_t input[], size_t input_length, bool uppercase)
Definition hex.cpp:35
size_t hex_decode(uint8_t output[], const char input[], size_t input_length, size_t &input_consumed, bool ignore_ws)
Definition hex.cpp:73
std::vector< T, secure_allocator< T > > secure_vector
Definition secmem.h:61
constexpr void clear_mem(T *ptr, size_t n)
Definition mem_ops.h:120