mirror of
				https://gitlab.labs.nic.cz/labs/bird.git
				synced 2024-05-11 16:54:54 +00:00 
			
		
		
		
	Either hex:01234567, or hex:01:23:45:67. No confusing formats like hex:0123:4567:ab: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;
 | |
| }
 |