This source file includes following definitions.
- rpl_inet_pton
- inet_pton
- inet_pton4
- inet_pton6
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 #include <config.h>
36
37
38 #include <arpa/inet.h>
39
40 #if HAVE_DECL_INET_PTON
41
42 # undef inet_pton
43
44 int
45 rpl_inet_pton (int af, const char *restrict src, void *restrict dst)
46 {
47 return inet_pton (af, src, dst);
48 }
49
50 #else
51
52 # include <c-ctype.h>
53 # include <string.h>
54 # include <errno.h>
55
56 # define NS_INADDRSZ 4
57 # define NS_IN6ADDRSZ 16
58 # define NS_INT16SZ 2
59
60
61
62
63
64
65 static int inet_pton4 (const char *src, unsigned char *dst);
66 # if HAVE_IPV6
67 static int inet_pton6 (const char *src, unsigned char *dst);
68 # endif
69
70
71
72
73
74
75
76
77
78
79
80
81 int
82 inet_pton (int af, const char *restrict src, void *restrict dst)
83 {
84 switch (af)
85 {
86 case AF_INET:
87 return (inet_pton4 (src, dst));
88
89 # if HAVE_IPV6
90 case AF_INET6:
91 return (inet_pton6 (src, dst));
92 # endif
93
94 default:
95 errno = EAFNOSUPPORT;
96 return (-1);
97 }
98
99 }
100
101
102
103
104
105
106
107
108
109
110
111
112 static int
113 inet_pton4 (const char *restrict src, unsigned char *restrict dst)
114 {
115 int saw_digit, octets, ch;
116 unsigned char tmp[NS_INADDRSZ], *tp;
117
118 saw_digit = 0;
119 octets = 0;
120 *(tp = tmp) = 0;
121 while ((ch = *src++) != '\0')
122 {
123
124 if (ch >= '0' && ch <= '9')
125 {
126 unsigned new = *tp * 10 + (ch - '0');
127
128 if (saw_digit && *tp == 0)
129 return (0);
130 if (new > 255)
131 return (0);
132 *tp = new;
133 if (!saw_digit)
134 {
135 if (++octets > 4)
136 return (0);
137 saw_digit = 1;
138 }
139 }
140 else if (ch == '.' && saw_digit)
141 {
142 if (octets == 4)
143 return (0);
144 *++tp = 0;
145 saw_digit = 0;
146 }
147 else
148 return (0);
149 }
150 if (octets < 4)
151 return (0);
152 memcpy (dst, tmp, NS_INADDRSZ);
153 return (1);
154 }
155
156 # if HAVE_IPV6
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171 static int
172 inet_pton6 (const char *restrict src, unsigned char *restrict dst)
173 {
174 static const char xdigits[] = "0123456789abcdef";
175 unsigned char tmp[NS_IN6ADDRSZ], *tp, *endp, *colonp;
176 const char *curtok;
177 int ch, saw_xdigit;
178 unsigned val;
179
180 tp = memset (tmp, '\0', NS_IN6ADDRSZ);
181 endp = tp + NS_IN6ADDRSZ;
182 colonp = NULL;
183
184 if (*src == ':')
185 if (*++src != ':')
186 return (0);
187 curtok = src;
188 saw_xdigit = 0;
189 val = 0;
190 while ((ch = c_tolower (*src++)) != '\0')
191 {
192 const char *pch;
193
194 pch = strchr (xdigits, ch);
195 if (pch != NULL)
196 {
197 val <<= 4;
198 val |= (pch - xdigits);
199 if (val > 0xffff)
200 return (0);
201 saw_xdigit = 1;
202 continue;
203 }
204 if (ch == ':')
205 {
206 curtok = src;
207 if (!saw_xdigit)
208 {
209 if (colonp)
210 return (0);
211 colonp = tp;
212 continue;
213 }
214 else if (*src == '\0')
215 {
216 return (0);
217 }
218 if (tp + NS_INT16SZ > endp)
219 return (0);
220 *tp++ = (u_char) (val >> 8) & 0xff;
221 *tp++ = (u_char) val & 0xff;
222 saw_xdigit = 0;
223 val = 0;
224 continue;
225 }
226 if (ch == '.' && ((tp + NS_INADDRSZ) <= endp) &&
227 inet_pton4 (curtok, tp) > 0)
228 {
229 tp += NS_INADDRSZ;
230 saw_xdigit = 0;
231 break;
232 }
233 return (0);
234 }
235 if (saw_xdigit)
236 {
237 if (tp + NS_INT16SZ > endp)
238 return (0);
239 *tp++ = (u_char) (val >> 8) & 0xff;
240 *tp++ = (u_char) val & 0xff;
241 }
242 if (colonp != NULL)
243 {
244
245
246
247
248 const int n = tp - colonp;
249 int i;
250
251 if (tp == endp)
252 return (0);
253 for (i = 1; i <= n; i++)
254 {
255 endp[-i] = colonp[n - i];
256 colonp[n - i] = 0;
257 }
258 tp = endp;
259 }
260 if (tp != endp)
261 return (0);
262 memcpy (dst, tmp, NS_IN6ADDRSZ);
263 return (1);
264 }
265
266 # endif
267
268 #endif