diff options
Diffstat (limited to 'src/lib/libc/net/inet_pton.c')
-rw-r--r-- | src/lib/libc/net/inet_pton.c | 215 |
1 files changed, 215 insertions, 0 deletions
diff --git a/src/lib/libc/net/inet_pton.c b/src/lib/libc/net/inet_pton.c new file mode 100644 index 0000000000..ec158d4599 --- /dev/null +++ b/src/lib/libc/net/inet_pton.c | |||
@@ -0,0 +1,215 @@ | |||
1 | /* $OpenBSD: inet_pton.c,v 1.5 2005/03/25 13:24:12 otto Exp $ */ | ||
2 | |||
3 | /* Copyright (c) 1996 by Internet Software Consortium. | ||
4 | * | ||
5 | * Permission to use, copy, modify, and distribute this software for any | ||
6 | * purpose with or without fee is hereby granted, provided that the above | ||
7 | * copyright notice and this permission notice appear in all copies. | ||
8 | * | ||
9 | * THE SOFTWARE IS PROVIDED "AS IS" AND INTERNET SOFTWARE CONSORTIUM DISCLAIMS | ||
10 | * ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES | ||
11 | * OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL INTERNET SOFTWARE | ||
12 | * CONSORTIUM BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL | ||
13 | * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR | ||
14 | * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS | ||
15 | * ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS | ||
16 | * SOFTWARE. | ||
17 | */ | ||
18 | |||
19 | #if defined(LIBC_SCCS) && !defined(lint) | ||
20 | #if 0 | ||
21 | static char rcsid[] = "$From: inet_pton.c,v 8.7 1996/08/05 08:31:35 vixie Exp $"; | ||
22 | #else | ||
23 | static char rcsid[] = "$OpenBSD: inet_pton.c,v 1.5 2005/03/25 13:24:12 otto Exp $"; | ||
24 | #endif | ||
25 | #endif /* LIBC_SCCS and not lint */ | ||
26 | |||
27 | #include <sys/param.h> | ||
28 | #include <sys/types.h> | ||
29 | #include <sys/socket.h> | ||
30 | #include <netinet/in.h> | ||
31 | #include <arpa/inet.h> | ||
32 | #include <arpa/nameser.h> | ||
33 | #include <string.h> | ||
34 | #include <errno.h> | ||
35 | |||
36 | /* | ||
37 | * WARNING: Don't even consider trying to compile this on a system where | ||
38 | * sizeof(int) < 4. sizeof(int) > 4 is fine; all the world's not a VAX. | ||
39 | */ | ||
40 | |||
41 | static int inet_pton4(const char *src, u_char *dst); | ||
42 | static int inet_pton6(const char *src, u_char *dst); | ||
43 | |||
44 | /* int | ||
45 | * inet_pton(af, src, dst) | ||
46 | * convert from presentation format (which usually means ASCII printable) | ||
47 | * to network format (which is usually some kind of binary format). | ||
48 | * return: | ||
49 | * 1 if the address was valid for the specified address family | ||
50 | * 0 if the address wasn't valid (`dst' is untouched in this case) | ||
51 | * -1 if some other error occurred (`dst' is untouched in this case, too) | ||
52 | * author: | ||
53 | * Paul Vixie, 1996. | ||
54 | */ | ||
55 | int | ||
56 | inet_pton(int af, const char *src, void *dst) | ||
57 | { | ||
58 | switch (af) { | ||
59 | case AF_INET: | ||
60 | return (inet_pton4(src, dst)); | ||
61 | case AF_INET6: | ||
62 | return (inet_pton6(src, dst)); | ||
63 | default: | ||
64 | errno = EAFNOSUPPORT; | ||
65 | return (-1); | ||
66 | } | ||
67 | /* NOTREACHED */ | ||
68 | } | ||
69 | |||
70 | /* int | ||
71 | * inet_pton4(src, dst) | ||
72 | * like inet_aton() but without all the hexadecimal and shorthand. | ||
73 | * return: | ||
74 | * 1 if `src' is a valid dotted quad, else 0. | ||
75 | * notice: | ||
76 | * does not touch `dst' unless it's returning 1. | ||
77 | * author: | ||
78 | * Paul Vixie, 1996. | ||
79 | */ | ||
80 | static int | ||
81 | inet_pton4(const char *src, u_char *dst) | ||
82 | { | ||
83 | static const char digits[] = "0123456789"; | ||
84 | int saw_digit, octets, ch; | ||
85 | u_char tmp[INADDRSZ], *tp; | ||
86 | |||
87 | saw_digit = 0; | ||
88 | octets = 0; | ||
89 | *(tp = tmp) = 0; | ||
90 | while ((ch = *src++) != '\0') { | ||
91 | const char *pch; | ||
92 | |||
93 | if ((pch = strchr(digits, ch)) != NULL) { | ||
94 | u_int new = *tp * 10 + (pch - digits); | ||
95 | |||
96 | if (new > 255) | ||
97 | return (0); | ||
98 | if (! saw_digit) { | ||
99 | if (++octets > 4) | ||
100 | return (0); | ||
101 | saw_digit = 1; | ||
102 | } | ||
103 | *tp = new; | ||
104 | } else if (ch == '.' && saw_digit) { | ||
105 | if (octets == 4) | ||
106 | return (0); | ||
107 | *++tp = 0; | ||
108 | saw_digit = 0; | ||
109 | } else | ||
110 | return (0); | ||
111 | } | ||
112 | if (octets < 4) | ||
113 | return (0); | ||
114 | |||
115 | memcpy(dst, tmp, INADDRSZ); | ||
116 | return (1); | ||
117 | } | ||
118 | |||
119 | /* int | ||
120 | * inet_pton6(src, dst) | ||
121 | * convert presentation level address to network order binary form. | ||
122 | * return: | ||
123 | * 1 if `src' is a valid [RFC1884 2.2] address, else 0. | ||
124 | * notice: | ||
125 | * (1) does not touch `dst' unless it's returning 1. | ||
126 | * (2) :: in a full address is silently ignored. | ||
127 | * credit: | ||
128 | * inspired by Mark Andrews. | ||
129 | * author: | ||
130 | * Paul Vixie, 1996. | ||
131 | */ | ||
132 | static int | ||
133 | inet_pton6(const char *src, u_char *dst) | ||
134 | { | ||
135 | static const char xdigits_l[] = "0123456789abcdef", | ||
136 | xdigits_u[] = "0123456789ABCDEF"; | ||
137 | u_char tmp[IN6ADDRSZ], *tp, *endp, *colonp; | ||
138 | const char *xdigits, *curtok; | ||
139 | int ch, saw_xdigit; | ||
140 | u_int val; | ||
141 | |||
142 | memset((tp = tmp), '\0', IN6ADDRSZ); | ||
143 | endp = tp + IN6ADDRSZ; | ||
144 | colonp = NULL; | ||
145 | /* Leading :: requires some special handling. */ | ||
146 | if (*src == ':') | ||
147 | if (*++src != ':') | ||
148 | return (0); | ||
149 | curtok = src; | ||
150 | saw_xdigit = 0; | ||
151 | val = 0; | ||
152 | while ((ch = *src++) != '\0') { | ||
153 | const char *pch; | ||
154 | |||
155 | if ((pch = strchr((xdigits = xdigits_l), ch)) == NULL) | ||
156 | pch = strchr((xdigits = xdigits_u), ch); | ||
157 | if (pch != NULL) { | ||
158 | val <<= 4; | ||
159 | val |= (pch - xdigits); | ||
160 | if (val > 0xffff) | ||
161 | return (0); | ||
162 | saw_xdigit = 1; | ||
163 | continue; | ||
164 | } | ||
165 | if (ch == ':') { | ||
166 | curtok = src; | ||
167 | if (!saw_xdigit) { | ||
168 | if (colonp) | ||
169 | return (0); | ||
170 | colonp = tp; | ||
171 | continue; | ||
172 | } else if (*src == '\0') { | ||
173 | return (0); | ||
174 | } | ||
175 | if (tp + INT16SZ > endp) | ||
176 | return (0); | ||
177 | *tp++ = (u_char) (val >> 8) & 0xff; | ||
178 | *tp++ = (u_char) val & 0xff; | ||
179 | saw_xdigit = 0; | ||
180 | val = 0; | ||
181 | continue; | ||
182 | } | ||
183 | if (ch == '.' && ((tp + INADDRSZ) <= endp) && | ||
184 | inet_pton4(curtok, tp) > 0) { | ||
185 | tp += INADDRSZ; | ||
186 | saw_xdigit = 0; | ||
187 | break; /* '\0' was seen by inet_pton4(). */ | ||
188 | } | ||
189 | return (0); | ||
190 | } | ||
191 | if (saw_xdigit) { | ||
192 | if (tp + INT16SZ > endp) | ||
193 | return (0); | ||
194 | *tp++ = (u_char) (val >> 8) & 0xff; | ||
195 | *tp++ = (u_char) val & 0xff; | ||
196 | } | ||
197 | if (colonp != NULL) { | ||
198 | /* | ||
199 | * Since some memmove()'s erroneously fail to handle | ||
200 | * overlapping regions, we'll do the shift by hand. | ||
201 | */ | ||
202 | const int n = tp - colonp; | ||
203 | int i; | ||
204 | |||
205 | for (i = 1; i <= n; i++) { | ||
206 | endp[- i] = colonp[n - i]; | ||
207 | colonp[n - i] = 0; | ||
208 | } | ||
209 | tp = endp; | ||
210 | } | ||
211 | if (tp != endp) | ||
212 | return (0); | ||
213 | memcpy(dst, tmp, IN6ADDRSZ); | ||
214 | return (1); | ||
215 | } | ||