2006-03-01 23:54:52 +00:00
|
|
|
/* $Id: inet_pton.c,v 1.1.1.1 2002/10/26 04:25:59 lukem Exp $ */
|
|
|
|
/* from NetBSD: inet_pton.c,v 1.16 2000/02/07 18:51:02 itojun Exp */
|
|
|
|
|
|
|
|
/* Copyright (c) 1996 by Internet Software Consortium.
|
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
|
|
* copyright notice and this permission notice appear in all copies.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND INTERNET SOFTWARE CONSORTIUM DISCLAIMS
|
|
|
|
* ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES
|
|
|
|
* OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL INTERNET SOFTWARE
|
|
|
|
* CONSORTIUM BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
|
|
|
|
* PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS
|
|
|
|
* ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
|
|
|
|
* SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
2006-03-13 16:06:34 +00:00
|
|
|
#include "config.h"
|
|
|
|
|
2006-03-02 00:14:34 +00:00
|
|
|
#include <ctype.h>
|
2013-10-30 03:51:32 +00:00
|
|
|
|
|
|
|
#if defined(HAVE_INTTYPES_H)
|
|
|
|
#include <inttypes.h>
|
|
|
|
#elif defined(HAVE_STDINT_H)
|
2010-03-16 06:19:02 +00:00
|
|
|
#include <stdint.h>
|
2010-05-21 11:05:33 +00:00
|
|
|
#endif
|
|
|
|
|
2006-03-02 00:14:34 +00:00
|
|
|
#include <sys/types.h>
|
2021-01-18 13:20:14 +00:00
|
|
|
#ifdef HAVE_SYS_PARAM_H
|
2006-03-02 00:14:34 +00:00
|
|
|
#include <sys/param.h>
|
2021-01-18 13:20:14 +00:00
|
|
|
#endif
|
2006-03-13 16:06:34 +00:00
|
|
|
|
2011-10-12 16:23:52 +00:00
|
|
|
#import "GSPrivate.h"
|
|
|
|
#import "GSNetwork.h"
|
2006-03-13 16:06:34 +00:00
|
|
|
|
2008-01-28 20:33:01 +00:00
|
|
|
#if HAVE_SYS_SOCKET_H
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#endif
|
2006-03-01 23:54:52 +00:00
|
|
|
|
|
|
|
#if HAVE_ARPA_NAMESER_H
|
|
|
|
#include <arpa/nameser.h>
|
|
|
|
#endif
|
|
|
|
|
2020-06-19 07:03:06 +00:00
|
|
|
#ifndef WSAAPI
|
|
|
|
#define WSAAPI
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2006-03-01 23:54:52 +00:00
|
|
|
/*
|
|
|
|
* WARNING: Don't even consider trying to compile this on a system where
|
|
|
|
* sizeof(int) < 4. sizeof(int) > 4 is fine; all the world's not a VAX.
|
|
|
|
*/
|
|
|
|
|
2006-03-13 16:06:34 +00:00
|
|
|
static int inet_pton4(const char *src, uint8_t *dst, int pton);
|
2006-03-01 23:54:52 +00:00
|
|
|
#ifdef INET6
|
2006-03-13 16:06:34 +00:00
|
|
|
static int inet_pton6(const char *src, uint8_t *dst);
|
2006-03-01 23:54:52 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* int
|
|
|
|
* inet_pton(af, src, dst)
|
|
|
|
* convert from presentation format (which usually means ASCII printable)
|
|
|
|
* to network format (which is usually some kind of binary format).
|
|
|
|
* return:
|
|
|
|
* 1 if the address was valid for the specified address family
|
|
|
|
* 0 if the address wasn't valid (`dst' is untouched in this case)
|
|
|
|
* -1 if some other error occurred (`dst' is untouched in this case, too)
|
|
|
|
* author:
|
|
|
|
* Paul Vixie, 1996.
|
|
|
|
*/
|
|
|
|
int
|
2020-06-15 21:52:35 +00:00
|
|
|
WSAAPI
|
2008-01-10 16:31:22 +00:00
|
|
|
inet_pton(int af, const char *src, void *dst)
|
2006-03-01 23:54:52 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
switch (af) {
|
|
|
|
case AF_INET:
|
|
|
|
return (inet_pton4(src, dst, 1));
|
|
|
|
#ifdef INET6
|
|
|
|
case AF_INET6:
|
|
|
|
return (inet_pton6(src, dst));
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
errno = EAFNOSUPPORT;
|
|
|
|
return (-1);
|
|
|
|
}
|
|
|
|
/* NOTREACHED */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* int
|
|
|
|
* inet_pton4(src, dst, pton)
|
|
|
|
* when last arg is 0: inet_aton(). with hexadecimal, octal and shorthand.
|
|
|
|
* when last arg is 1: inet_pton(). decimal dotted-quad only.
|
|
|
|
* return:
|
|
|
|
* 1 if `src' is a valid input, else 0.
|
|
|
|
* notice:
|
|
|
|
* does not touch `dst' unless it's returning 1.
|
|
|
|
* author:
|
|
|
|
* Paul Vixie, 1996.
|
|
|
|
*/
|
|
|
|
static int
|
2006-03-13 16:06:34 +00:00
|
|
|
inet_pton4(const char *src, uint8_t *dst, int pton)
|
2006-03-01 23:54:52 +00:00
|
|
|
{
|
|
|
|
u_int val;
|
|
|
|
u_int digit;
|
|
|
|
int base, n;
|
|
|
|
unsigned char c;
|
|
|
|
u_int parts[4];
|
|
|
|
register u_int *pp = parts;
|
|
|
|
|
|
|
|
c = *src;
|
|
|
|
for (;;) {
|
|
|
|
/*
|
|
|
|
* Collect number up to ``.''.
|
|
|
|
* Values are specified as for C:
|
|
|
|
* 0x=hex, 0=octal, isdigit=decimal.
|
|
|
|
*/
|
|
|
|
if (!isdigit(c))
|
|
|
|
return (0);
|
|
|
|
val = 0; base = 10;
|
|
|
|
if (c == '0') {
|
|
|
|
c = *++src;
|
|
|
|
if (c == 'x' || c == 'X')
|
|
|
|
base = 16, c = *++src;
|
|
|
|
else if (isdigit(c) && c != '9')
|
|
|
|
base = 8;
|
|
|
|
}
|
|
|
|
/* inet_pton() takes decimal only */
|
|
|
|
if (pton && base != 10)
|
|
|
|
return (0);
|
|
|
|
for (;;) {
|
|
|
|
if (isdigit(c)) {
|
|
|
|
digit = c - '0';
|
|
|
|
if (digit >= base)
|
|
|
|
break;
|
|
|
|
val = (val * base) + digit;
|
|
|
|
c = *++src;
|
|
|
|
} else if (base == 16 && isxdigit(c)) {
|
|
|
|
digit = c + 10 - (islower(c) ? 'a' : 'A');
|
|
|
|
if (digit >= 16)
|
|
|
|
break;
|
|
|
|
val = (val << 4) | digit;
|
|
|
|
c = *++src;
|
|
|
|
} else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (c == '.') {
|
|
|
|
/*
|
|
|
|
* Internet format:
|
|
|
|
* a.b.c.d
|
|
|
|
* a.b.c (with c treated as 16 bits)
|
|
|
|
* a.b (with b treated as 24 bits)
|
|
|
|
* a (with a treated as 32 bits)
|
|
|
|
*/
|
|
|
|
if (pp >= parts + 3)
|
|
|
|
return (0);
|
|
|
|
*pp++ = val;
|
|
|
|
c = *++src;
|
|
|
|
} else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Check for trailing characters.
|
|
|
|
*/
|
|
|
|
if (c != '\0' && !isspace(c))
|
|
|
|
return (0);
|
|
|
|
/*
|
|
|
|
* Concoct the address according to
|
|
|
|
* the number of parts specified.
|
|
|
|
*/
|
|
|
|
n = pp - parts + 1;
|
|
|
|
/* inet_pton() takes dotted-quad only. it does not take shorthand. */
|
|
|
|
if (pton && n != 4)
|
|
|
|
return (0);
|
|
|
|
switch (n) {
|
|
|
|
|
|
|
|
case 0:
|
|
|
|
return (0); /* initial nondigit */
|
|
|
|
|
|
|
|
case 1: /* a -- 32 bits */
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 2: /* a.b -- 8.24 bits */
|
|
|
|
if (parts[0] > 0xff || val > 0xffffff)
|
|
|
|
return (0);
|
|
|
|
val |= parts[0] << 24;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 3: /* a.b.c -- 8.8.16 bits */
|
|
|
|
if ((parts[0] | parts[1]) > 0xff || val > 0xffff)
|
|
|
|
return (0);
|
|
|
|
val |= (parts[0] << 24) | (parts[1] << 16);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 4: /* a.b.c.d -- 8.8.8.8 bits */
|
|
|
|
if ((parts[0] | parts[1] | parts[2] | val) > 0xff)
|
|
|
|
return (0);
|
|
|
|
val |= (parts[0] << 24) | (parts[1] << 16) | (parts[2] << 8);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (dst) {
|
|
|
|
val = htonl(val);
|
|
|
|
memcpy(dst, &val, INADDRSZ);
|
|
|
|
}
|
|
|
|
return (1);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef INET6
|
|
|
|
/* int
|
|
|
|
* inet_pton6(src, dst)
|
|
|
|
* convert presentation level address to network order binary form.
|
|
|
|
* return:
|
|
|
|
* 1 if `src' is a valid [RFC1884 2.2] address, else 0.
|
|
|
|
* notice:
|
|
|
|
* (1) does not touch `dst' unless it's returning 1.
|
|
|
|
* (2) :: in a full address is silently ignored.
|
|
|
|
* credit:
|
|
|
|
* inspired by Mark Andrews.
|
|
|
|
* author:
|
|
|
|
* Paul Vixie, 1996.
|
|
|
|
*/
|
|
|
|
static int
|
2006-03-13 16:06:34 +00:00
|
|
|
inet_pton6(const char *src, uint8_t *dst)
|
2006-03-01 23:54:52 +00:00
|
|
|
{
|
|
|
|
static const char xdigits_l[] = "0123456789abcdef",
|
|
|
|
xdigits_u[] = "0123456789ABCDEF";
|
2006-03-13 16:06:34 +00:00
|
|
|
uint8_t tmp[IN6ADDRSZ], *tp, *endp, *colonp;
|
2006-03-01 23:54:52 +00:00
|
|
|
const char *xdigits, *curtok;
|
|
|
|
int ch, saw_xdigit;
|
|
|
|
u_int val;
|
|
|
|
|
|
|
|
memset((tp = tmp), '\0', IN6ADDRSZ);
|
|
|
|
endp = tp + IN6ADDRSZ;
|
|
|
|
colonp = NULL;
|
|
|
|
/* Leading :: requires some special handling. */
|
|
|
|
if (*src == ':')
|
|
|
|
if (*++src != ':')
|
|
|
|
return (0);
|
|
|
|
curtok = src;
|
|
|
|
saw_xdigit = 0;
|
|
|
|
val = 0;
|
|
|
|
while ((ch = *src++) != '\0') {
|
|
|
|
const char *pch;
|
|
|
|
|
|
|
|
if ((pch = strchr((xdigits = xdigits_l), ch)) == NULL)
|
|
|
|
pch = strchr((xdigits = xdigits_u), ch);
|
|
|
|
if (pch != NULL) {
|
|
|
|
val <<= 4;
|
|
|
|
val |= (pch - xdigits);
|
|
|
|
if (val > 0xffff)
|
|
|
|
return (0);
|
|
|
|
saw_xdigit = 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (ch == ':') {
|
|
|
|
curtok = src;
|
|
|
|
if (!saw_xdigit) {
|
|
|
|
if (colonp)
|
|
|
|
return (0);
|
|
|
|
colonp = tp;
|
|
|
|
continue;
|
|
|
|
} else if (*src == '\0')
|
|
|
|
return (0);
|
|
|
|
if (tp + INT16SZ > endp)
|
|
|
|
return (0);
|
2006-03-13 16:06:34 +00:00
|
|
|
*tp++ = (uint8_t) (val >> 8) & 0xff;
|
|
|
|
*tp++ = (uint8_t) val & 0xff;
|
2006-03-01 23:54:52 +00:00
|
|
|
saw_xdigit = 0;
|
|
|
|
val = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (ch == '.' && ((tp + INADDRSZ) <= endp) &&
|
|
|
|
inet_pton4(curtok, tp, 1) > 0) {
|
|
|
|
tp += INADDRSZ;
|
|
|
|
saw_xdigit = 0;
|
|
|
|
break; /* '\0' was seen by inet_pton4(). */
|
|
|
|
}
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
if (saw_xdigit) {
|
|
|
|
if (tp + INT16SZ > endp)
|
|
|
|
return (0);
|
2006-03-13 16:06:34 +00:00
|
|
|
*tp++ = (uint8_t) (val >> 8) & 0xff;
|
|
|
|
*tp++ = (uint8_t) val & 0xff;
|
2006-03-01 23:54:52 +00:00
|
|
|
}
|
|
|
|
if (colonp != NULL) {
|
|
|
|
/*
|
|
|
|
* Since some memmove()'s erroneously fail to handle
|
|
|
|
* overlapping regions, we'll do the shift by hand.
|
|
|
|
*/
|
|
|
|
const int n = tp - colonp;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (tp == endp)
|
|
|
|
return (0);
|
|
|
|
for (i = 1; i <= n; i++) {
|
|
|
|
endp[- i] = colonp[n - i];
|
|
|
|
colonp[n - i] = 0;
|
|
|
|
}
|
|
|
|
tp = endp;
|
|
|
|
}
|
|
|
|
if (tp != endp)
|
|
|
|
return (0);
|
|
|
|
memcpy(dst, tmp, IN6ADDRSZ);
|
|
|
|
return (1);
|
|
|
|
}
|
|
|
|
#endif
|
2008-01-03 12:53:11 +00:00
|
|
|
|
|
|
|
|