Converts a hex string to a number. Out buffer can be same as in buffer.
Note that this routine is about 43% slower than to_hex_string(), half of which is due to input validation.
146 {
147 if(inlen % 2)
148#ifdef __cpp_exceptions
149 throw std::invalid_argument("Input buffer not multiple of two.");
150#else
151 abort();
152#endif
153 if(outlen < inlen / 2)
154#ifdef __cpp_exceptions
155 throw std::invalid_argument("Output buffer too small.");
156#else
157 abort();
158#endif
159 bool is_invalid = false;
160 auto fromhex = [&is_invalid](CharType c) -> unsigned char
161 {
162#if 1
163
164
165 static constexpr unsigned char table[] = {
166 0, 1, 2, 3, 4, 5, 6, 7, 8, 9,
167 255, 255, 255, 255, 255, 255, 255,
168 10, 11, 12, 13, 14, 15,
169 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
170 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
171 10, 11, 12, 13, 14, 15};
172 unsigned char r = 255;
173 if(c >= 48 && c <= 102)
174 r = table[c - 48];
175 if(r == 255)
176 is_invalid = true;
177 return r;
178#else
179 if(c >= '0' && c <= '9')
180 return c - '0';
181 if(c >= 'a' && c <= 'f')
182 return c - 'a' + 10;
183 if(c >= 'A' && c <= 'F')
184 return c - 'A' + 10;
185#ifdef __cpp_exceptions
186 throw std::invalid_argument("Input is not hexadecimal.");
187#else
188 abort();
189#endif
190#endif
191 };
192 const auto bulklen = inlen / 2 - (inlen / 2) % 4;
193 if(bulklen >= 4)
194 {
195 for(size_t n = 0; n < bulklen; n += 4)
196 {
197 unsigned char c[8];
198 c[0] = fromhex(in[n * 2]);
199 c[1] = fromhex(in[n * 2 + 1]);
200 c[2] = fromhex(in[n * 2 + 2]);
201 c[3] = fromhex(in[n * 2 + 3]);
202 out[n] = (T) ((c[0] << 4) | c[1]);
203 c[4] = fromhex(in[n * 2 + 4]);
204 c[5] = fromhex(in[n * 2 + 5]);
205 out[n + 1] = (T) ((c[2] << 4) | c[3]);
206 c[6] = fromhex(in[n * 2 + 6]);
207 c[7] = fromhex(in[n * 2 + 7]);
208 out[n + 2] = (T) ((c[4] << 4) | c[5]);
209 out[n + 3] = (T) ((c[6] << 4) | c[7]);
210 }
211 }
212 for(size_t n = bulklen; n < inlen / 2; n++)
213 {
214 auto c1 = fromhex(in[n * 2]), c2 = fromhex(in[n * 2 + 1]);
215 out[n] = (T) ((c1 << 4) | c2);
216 }
217 if(is_invalid)
218#ifdef __cpp_exceptions
219 throw std::invalid_argument("Input is not hexadecimal.");
220#else
221 abort();
222#endif
223 return inlen / 2;
224 }