/* * 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; }