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 #include <botan/mem_ops.h>
10 #include <botan/exceptn.h>
11 #include <botan/internal/ct_utils.h>
12 
13 namespace Botan {
14 
15 namespace {
16 
hex_encode_nibble(uint8_t n,bool uppercase)17 char hex_encode_nibble(uint8_t n, bool uppercase)
18    {
19    BOTAN_DEBUG_ASSERT(n <= 15);
20 
21    const auto in_09 = CT::Mask<uint8_t>::is_lt(n, 10);
22 
23    const char c_09 = n + '0';
24    const char c_af = n + (uppercase ? 'A' : 'a') - 10;
25 
26    return in_09.select(c_09, c_af);
27    }
28 
29 }
30 
hex_encode(char output[],const uint8_t input[],size_t input_length,bool uppercase)31 void hex_encode(char output[],
32                 const uint8_t input[],
33                 size_t input_length,
34                 bool uppercase)
35    {
36    for(size_t i = 0; i != input_length; ++i)
37       {
38       const uint8_t n0 = (input[i] >> 4) & 0xF;
39       const uint8_t n1 = (input[i]     ) & 0xF;
40 
41       output[2*i  ] = hex_encode_nibble(n0, uppercase);
42       output[2*i+1] = hex_encode_nibble(n1, uppercase);
43       }
44    }
45 
hex_encode(const uint8_t input[],size_t input_length,bool uppercase)46 std::string hex_encode(const uint8_t input[],
47                        size_t input_length,
48                        bool uppercase)
49    {
50    std::string output(2 * input_length, 0);
51 
52    if(input_length)
53       hex_encode(&output.front(), input, input_length, uppercase);
54 
55    return output;
56    }
57 
58 namespace {
59 
hex_char_to_bin(char input)60 uint8_t hex_char_to_bin(char input)
61    {
62    const uint8_t c = static_cast<uint8_t>(input);
63 
64    const auto is_alpha_upper = CT::Mask<uint8_t>::is_within_range(c, uint8_t('A'), uint8_t('F'));
65    const auto is_alpha_lower = CT::Mask<uint8_t>::is_within_range(c, uint8_t('a'), uint8_t('f'));
66    const auto is_decimal     = CT::Mask<uint8_t>::is_within_range(c, uint8_t('0'), uint8_t('9'));
67 
68    const auto is_whitespace  = CT::Mask<uint8_t>::is_any_of(c, {
69          uint8_t(' '), uint8_t('\t'), uint8_t('\n'), uint8_t('\r')
70       });
71 
72    const uint8_t c_upper = c - uint8_t('A') + 10;
73    const uint8_t c_lower = c - uint8_t('a') + 10;
74    const uint8_t c_decim = c - uint8_t('0');
75 
76    uint8_t ret = 0xFF; // default value
77 
78    ret = is_alpha_upper.select(c_upper, ret);
79    ret = is_alpha_lower.select(c_lower, ret);
80    ret = is_decimal.select(c_decim, ret);
81    ret = is_whitespace.select(0x80, ret);
82 
83    return ret;
84    }
85 
86 }
87 
88 
hex_decode(uint8_t output[],const char input[],size_t input_length,size_t & input_consumed,bool ignore_ws)89 size_t hex_decode(uint8_t output[],
90                   const char input[],
91                   size_t input_length,
92                   size_t& input_consumed,
93                   bool ignore_ws)
94    {
95    uint8_t* out_ptr = output;
96    bool top_nibble = true;
97 
98    clear_mem(output, input_length / 2);
99 
100    for(size_t i = 0; i != input_length; ++i)
101       {
102       const uint8_t bin = hex_char_to_bin(input[i]);
103 
104       if(bin >= 0x10)
105          {
106          if(bin == 0x80 && ignore_ws)
107             continue;
108 
109          std::string bad_char(1, input[i]);
110          if(bad_char == "\t")
111            bad_char = "\\t";
112          else if(bad_char == "\n")
113            bad_char = "\\n";
114 
115          throw Invalid_Argument(
116            std::string("hex_decode: invalid hex character '") +
117            bad_char + "'");
118          }
119 
120       if(top_nibble)
121          *out_ptr |= bin << 4;
122       else
123          *out_ptr |= bin;
124 
125       top_nibble = !top_nibble;
126       if(top_nibble)
127          ++out_ptr;
128       }
129 
130    input_consumed = input_length;
131    size_t written = (out_ptr - output);
132 
133    /*
134    * We only got half of a uint8_t at the end; zap the half-written
135    * output and mark it as unread
136    */
137    if(!top_nibble)
138       {
139       *out_ptr = 0;
140       input_consumed -= 1;
141       }
142 
143    return written;
144    }
145 
hex_decode(uint8_t output[],const char input[],size_t input_length,bool ignore_ws)146 size_t hex_decode(uint8_t output[],
147                   const char input[],
148                   size_t input_length,
149                   bool ignore_ws)
150    {
151    size_t consumed = 0;
152    size_t written = hex_decode(output, input, input_length,
153                                consumed, ignore_ws);
154 
155    if(consumed != input_length)
156       throw Invalid_Argument("hex_decode: input did not have full bytes");
157 
158    return written;
159    }
160 
hex_decode(uint8_t output[],const std::string & input,bool ignore_ws)161 size_t hex_decode(uint8_t output[],
162                   const std::string& input,
163                   bool ignore_ws)
164    {
165    return hex_decode(output, input.data(), input.length(), ignore_ws);
166    }
167 
hex_decode_locked(const char input[],size_t input_length,bool ignore_ws)168 secure_vector<uint8_t> hex_decode_locked(const char input[],
169                                       size_t input_length,
170                                       bool ignore_ws)
171    {
172    secure_vector<uint8_t> bin(1 + input_length / 2);
173 
174    size_t written = hex_decode(bin.data(),
175                                input,
176                                input_length,
177                                ignore_ws);
178 
179    bin.resize(written);
180    return bin;
181    }
182 
hex_decode_locked(const std::string & input,bool ignore_ws)183 secure_vector<uint8_t> hex_decode_locked(const std::string& input,
184                                       bool ignore_ws)
185    {
186    return hex_decode_locked(input.data(), input.size(), ignore_ws);
187    }
188 
hex_decode(const char input[],size_t input_length,bool ignore_ws)189 std::vector<uint8_t> hex_decode(const char input[],
190                              size_t input_length,
191                              bool ignore_ws)
192    {
193    std::vector<uint8_t> bin(1 + input_length / 2);
194 
195    size_t written = hex_decode(bin.data(),
196                                input,
197                                input_length,
198                                ignore_ws);
199 
200    bin.resize(written);
201    return bin;
202    }
203 
hex_decode(const std::string & input,bool ignore_ws)204 std::vector<uint8_t> hex_decode(const std::string& input,
205                              bool ignore_ws)
206    {
207    return hex_decode(input.data(), input.size(), ignore_ws);
208    }
209 
210 }
211