1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
|
/* ISC license. */
#include <sys/types.h>
#include <stdint.h>
#include <errno.h>
#include <skalibs/uint16.h>
#include <skalibs/fmtscan.h>
size_t ip6_scan (char const *s, char *ip6)
{
static const unsigned char class[256] = "2222222222222222222222222222222222222222222222220000000000122222200000022222222222222222222222222000000222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222" ;
static const unsigned char table[5][3] = { "\024#\005", "\024\"\005", "\024\005\006", "\005\002\005", "\024\t\016" } ;
uint16_t tmp[8] = { 0, 0, 0, 0, 0, 0, 0, 0 } ;
size_t i = 0 ;
unsigned int pos = 8, j = 0, state = 0 ;
while (state < 5)
{
unsigned char c = table[state][class[(unsigned char)s[i]] - '0'] ;
state = c & 7 ;
if (c & 0x20) { if (pos < 8) state = 5 ; else pos = j ; }
if (c & 0x10)
{
if (tmp[j] & 0xf000) state = 5 ;
else tmp[j] = (tmp[j] << 4) + fmtscan_num(s[i], 16) ;
}
if (c & 0x08) if ((++j > 7) && (state < 5)) state = 5 ;
i++ ;
}
if (((pos < 8) && (j > 6)) || ((pos == 8) && (j < 8))) state = 5 ;
if (state == 5) return (errno = EINVAL, 0) ;
for (state = j ; state > pos ; state--) tmp[state - j + 7] = tmp[state - 1] ;
for (; state < pos + 8 - j ; state++) tmp[state] = 0 ;
for (j = 0 ; j < 8 ; j++) uint16_pack_big(ip6 + (j<<1), tmp[j]) ;
return i - 1 ;
}
|