mirror of
https://gitlab.nic.cz/labs/bird.git
synced 2024-12-22 01:31:55 +00:00
c5c3a22bcc
Either hex:01234567, or hex:01:23:45:67. No confusing formats like
hex:0123:4567🆎cdef, which looks like there is an implicit zero byte.
151 lines
2.2 KiB
C
151 lines
2.2 KiB
C
/*
|
|
* BIRD Library -- Parse numbers
|
|
*
|
|
* (c) 2019 Maria Matejka <mq@jmq.cz>
|
|
*
|
|
* Can be freely distributed and used under the terms of the GNU GPL.
|
|
*/
|
|
|
|
#include "nest/bird.h"
|
|
#include "lib/string.h"
|
|
|
|
#include <errno.h>
|
|
|
|
#define ULI_MAX_DIV10 (UINT64_MAX / 10)
|
|
#define ULI_MAX_MOD10 (UINT64_MAX % 10)
|
|
|
|
u64
|
|
bstrtoul10(const char *str, char **end)
|
|
{
|
|
u64 out = 0;
|
|
for (*end = (char *) str; (**end >= '0') && (**end <= '9'); (*end)++) {
|
|
u64 digit = **end - '0';
|
|
if ((out > ULI_MAX_DIV10) ||
|
|
(out == ULI_MAX_DIV10) && (digit > ULI_MAX_MOD10)) {
|
|
errno = ERANGE;
|
|
return UINT64_MAX;
|
|
}
|
|
|
|
out *= 10;
|
|
out += (**end) - '0';
|
|
}
|
|
return out;
|
|
}
|
|
|
|
u64
|
|
bstrtoul16(const char *str, char **end)
|
|
{
|
|
u64 out = 0;
|
|
for (int i=0; i<=(64/4); i++) {
|
|
switch (str[i]) {
|
|
case '0' ... '9':
|
|
out *= 16;
|
|
out += str[i] - '0';
|
|
break;
|
|
case 'a' ... 'f':
|
|
out *= 16;
|
|
out += str[i] + 10 - 'a';
|
|
break;
|
|
case 'A' ... 'F':
|
|
out *= 16;
|
|
out += str[i] + 10 - 'A';
|
|
break;
|
|
default:
|
|
*end = (char *) &(str[i]);
|
|
return out;
|
|
}
|
|
}
|
|
|
|
errno = ERANGE;
|
|
return UINT64_MAX;
|
|
}
|
|
|
|
static int
|
|
fromxdigit(char c)
|
|
{
|
|
switch (c)
|
|
{
|
|
case '0' ... '9':
|
|
return c - '0';
|
|
case 'a' ... 'f':
|
|
return c + 10 - 'a';
|
|
case 'A' ... 'F':
|
|
return c + 10 - 'A';
|
|
default:
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
int
|
|
bstrhextobin(const char *s, byte *b)
|
|
{
|
|
int len = 0;
|
|
int hi = 0;
|
|
|
|
for (; *s; s++)
|
|
{
|
|
int v = fromxdigit(*s);
|
|
if (v < 0)
|
|
{
|
|
if (strchr(" -.:", *s) && !hi)
|
|
continue;
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
if (len == INT32_MAX)
|
|
return -1;
|
|
|
|
if (b)
|
|
{
|
|
if (!hi)
|
|
b[len] = (v << 4);
|
|
else
|
|
b[len] |= v;
|
|
}
|
|
|
|
len += hi;
|
|
hi = !hi;
|
|
}
|
|
|
|
return !hi ? len : -1;
|
|
}
|
|
|
|
static char
|
|
toxdigit(uint b)
|
|
{
|
|
if (b < 10)
|
|
return ('0' + b);
|
|
else if (b < 16)
|
|
return ('a' + b - 10);
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
bstrbintohex(const byte *b, size_t len, char *buf, size_t size, char delim)
|
|
{
|
|
ASSERT(size >= 6);
|
|
char *bound = buf + size - 3;
|
|
|
|
size_t i;
|
|
for (i = 0; i < len; i++)
|
|
{
|
|
if (buf > bound)
|
|
{
|
|
strcpy(buf - 4, "...");
|
|
return -1;
|
|
}
|
|
|
|
uint x = b[i];
|
|
buf[0] = toxdigit(x >> 4);
|
|
buf[1] = toxdigit(x & 0xF);
|
|
buf[2] = delim;
|
|
buf += 3;
|
|
}
|
|
|
|
buf[i ? -1 : 0] = 0;
|
|
|
|
return 0;
|
|
}
|