|
Packit |
6c4009 |
/* Copyright (C) 1997-2018 Free Software Foundation, Inc.
|
|
Packit |
6c4009 |
This file is part of the GNU C Library.
|
|
Packit |
6c4009 |
Contributed by H.J. Lu <hjl@gnu.ai.mit.edu>, 1997.
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
The GNU C Library is free software; you can redistribute it and/or
|
|
Packit |
6c4009 |
modify it under the terms of the GNU Lesser General Public
|
|
Packit |
6c4009 |
License as published by the Free Software Foundation; either
|
|
Packit |
6c4009 |
version 2.1 of the License, or (at your option) any later version.
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
The GNU C Library is distributed in the hope that it will be useful,
|
|
Packit |
6c4009 |
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
Packit |
6c4009 |
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Packit |
6c4009 |
Lesser General Public License for more details.
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
You should have received a copy of the GNU Lesser General Public
|
|
Packit |
6c4009 |
License along with the GNU C Library; if not, see
|
|
Packit |
6c4009 |
<http://www.gnu.org/licenses/>. */
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
#include <assert.h>
|
|
Packit |
6c4009 |
#include <errno.h>
|
|
Packit |
6c4009 |
#include <string.h>
|
|
Packit |
6c4009 |
#include <stdlib.h>
|
|
Packit |
6c4009 |
#include <ctype.h>
|
|
Packit |
6c4009 |
#include <wctype.h>
|
|
Packit |
6c4009 |
#include <resolv/resolv-internal.h>
|
|
Packit |
6c4009 |
#include <resolv/resolv_context.h>
|
|
Packit |
6c4009 |
#include <netdb.h>
|
|
Packit |
6c4009 |
#include <arpa/inet.h>
|
|
Packit |
6c4009 |
#include "nsswitch.h"
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
#ifdef USE_NSCD
|
|
Packit |
6c4009 |
# include <nscd/nscd_proto.h>
|
|
Packit |
6c4009 |
#endif
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
int
|
|
Packit |
6c4009 |
__nss_hostname_digits_dots (const char *name, struct hostent *resbuf,
|
|
Packit |
6c4009 |
char **buffer, size_t *buffer_size,
|
|
Packit |
6c4009 |
size_t buflen, struct hostent **result,
|
|
Packit |
6c4009 |
enum nss_status *status, int af, int *h_errnop)
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
/* We have to test for the use of IPv6 which can only be done by
|
|
Packit |
6c4009 |
examining `_res'. */
|
|
Packit |
6c4009 |
struct resolv_context *ctx = __resolv_context_get ();
|
|
Packit |
6c4009 |
if (ctx == NULL)
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
if (h_errnop)
|
|
Packit |
6c4009 |
*h_errnop = NETDB_INTERNAL;
|
|
Packit |
6c4009 |
if (buffer_size == NULL)
|
|
Packit |
6c4009 |
*status = NSS_STATUS_TRYAGAIN;
|
|
Packit |
6c4009 |
else
|
|
Packit |
6c4009 |
*result = NULL;
|
|
Packit |
6c4009 |
return -1;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
int ret = __nss_hostname_digits_dots_context
|
|
Packit |
6c4009 |
(ctx, name, resbuf, buffer, buffer_size, buflen,
|
|
Packit |
6c4009 |
result, status, af, h_errnop);
|
|
Packit |
6c4009 |
__resolv_context_put (ctx);
|
|
Packit |
6c4009 |
return ret;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
int
|
|
Packit |
6c4009 |
__nss_hostname_digits_dots_context (struct resolv_context *ctx,
|
|
Packit |
6c4009 |
const char *name, struct hostent *resbuf,
|
|
Packit |
6c4009 |
char **buffer, size_t *buffer_size,
|
|
Packit |
6c4009 |
size_t buflen, struct hostent **result,
|
|
Packit |
6c4009 |
enum nss_status *status, int af, int *h_errnop)
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
int save;
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
/*
|
|
Packit |
6c4009 |
* disallow names consisting only of digits/dots, unless
|
|
Packit |
6c4009 |
* they end in a dot.
|
|
Packit |
6c4009 |
*/
|
|
Packit |
6c4009 |
if (isdigit (name[0]) || isxdigit (name[0]) || name[0] == ':')
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
const char *cp;
|
|
Packit |
6c4009 |
char *hostname;
|
|
Packit |
6c4009 |
typedef unsigned char host_addr_t[16];
|
|
Packit |
6c4009 |
host_addr_t *host_addr;
|
|
Packit |
6c4009 |
typedef char *host_addr_list_t[2];
|
|
Packit |
6c4009 |
host_addr_list_t *h_addr_ptrs;
|
|
Packit |
6c4009 |
char **h_alias_ptr;
|
|
Packit |
6c4009 |
size_t size_needed;
|
|
Packit |
6c4009 |
int addr_size;
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
switch (af)
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
case AF_INET:
|
|
Packit |
6c4009 |
addr_size = INADDRSZ;
|
|
Packit |
6c4009 |
break;
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
case AF_INET6:
|
|
Packit |
6c4009 |
addr_size = IN6ADDRSZ;
|
|
Packit |
6c4009 |
break;
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
default:
|
|
Packit |
6c4009 |
af = res_use_inet6 () ? AF_INET6 : AF_INET;
|
|
Packit |
6c4009 |
addr_size = af == AF_INET6 ? IN6ADDRSZ : INADDRSZ;
|
|
Packit |
6c4009 |
break;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
size_needed = (sizeof (*host_addr)
|
|
Packit |
6c4009 |
+ sizeof (*h_addr_ptrs)
|
|
Packit |
6c4009 |
+ sizeof (*h_alias_ptr) + strlen (name) + 1);
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
if (buffer_size == NULL)
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
if (buflen < size_needed)
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
*status = NSS_STATUS_TRYAGAIN;
|
|
Packit |
6c4009 |
if (h_errnop != NULL)
|
|
Packit |
6c4009 |
*h_errnop = NETDB_INTERNAL;
|
|
Packit |
6c4009 |
__set_errno (ERANGE);
|
|
Packit |
6c4009 |
goto done;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
else if (buffer_size != NULL && *buffer_size < size_needed)
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
char *new_buf;
|
|
Packit |
6c4009 |
*buffer_size = size_needed;
|
|
Packit |
6c4009 |
new_buf = (char *) realloc (*buffer, *buffer_size);
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
if (new_buf == NULL)
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
save = errno;
|
|
Packit |
6c4009 |
free (*buffer);
|
|
Packit |
6c4009 |
*buffer = NULL;
|
|
Packit |
6c4009 |
*buffer_size = 0;
|
|
Packit |
6c4009 |
__set_errno (save);
|
|
Packit |
6c4009 |
if (h_errnop != NULL)
|
|
Packit |
6c4009 |
*h_errnop = NETDB_INTERNAL;
|
|
Packit |
6c4009 |
*result = NULL;
|
|
Packit |
6c4009 |
goto done;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
*buffer = new_buf;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
memset (*buffer, '\0', size_needed);
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
host_addr = (host_addr_t *) *buffer;
|
|
Packit |
6c4009 |
h_addr_ptrs = (host_addr_list_t *)
|
|
Packit |
6c4009 |
((char *) host_addr + sizeof (*host_addr));
|
|
Packit |
6c4009 |
h_alias_ptr = (char **) ((char *) h_addr_ptrs + sizeof (*h_addr_ptrs));
|
|
Packit |
6c4009 |
hostname = (char *) h_alias_ptr + sizeof (*h_alias_ptr);
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
if (isdigit (name[0]))
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
for (cp = name;; ++cp)
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
if (*cp == '\0')
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
int ok;
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
if (*--cp == '.')
|
|
Packit |
6c4009 |
break;
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
/* All-numeric, no dot at the end. Fake up a hostent as if
|
|
Packit |
6c4009 |
we'd actually done a lookup. What if someone types
|
|
Packit |
6c4009 |
255.255.255.255? The test below will succeed
|
|
Packit |
6c4009 |
spuriously... ??? */
|
|
Packit |
6c4009 |
if (af == AF_INET)
|
|
Packit |
ac9958 |
ok = __inet_aton_exact (name, (struct in_addr *) host_addr);
|
|
Packit |
6c4009 |
else
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
assert (af == AF_INET6);
|
|
Packit |
6c4009 |
ok = inet_pton (af, name, host_addr) > 0;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
if (! ok)
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
*h_errnop = HOST_NOT_FOUND;
|
|
Packit |
6c4009 |
if (buffer_size == NULL)
|
|
Packit |
6c4009 |
*status = NSS_STATUS_NOTFOUND;
|
|
Packit |
6c4009 |
else
|
|
Packit |
6c4009 |
*result = NULL;
|
|
Packit |
6c4009 |
goto done;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
resbuf->h_name = strcpy (hostname, name);
|
|
Packit |
6c4009 |
h_alias_ptr[0] = NULL;
|
|
Packit |
6c4009 |
resbuf->h_aliases = h_alias_ptr;
|
|
Packit |
6c4009 |
(*h_addr_ptrs)[0] = (char *) host_addr;
|
|
Packit |
6c4009 |
(*h_addr_ptrs)[1] = NULL;
|
|
Packit |
6c4009 |
resbuf->h_addr_list = *h_addr_ptrs;
|
|
Packit |
6c4009 |
if (af == AF_INET && res_use_inet6 ())
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
/* We need to change the IP v4 address into the
|
|
Packit |
6c4009 |
IP v6 address. */
|
|
Packit |
6c4009 |
char tmp[INADDRSZ];
|
|
Packit |
6c4009 |
char *p = (char *) host_addr;
|
|
Packit |
6c4009 |
int i;
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
/* Save a copy of the IP v4 address. */
|
|
Packit |
6c4009 |
memcpy (tmp, host_addr, INADDRSZ);
|
|
Packit |
6c4009 |
/* Mark this ipv6 addr as a mapped ipv4. */
|
|
Packit |
6c4009 |
for (i = 0; i < 10; i++)
|
|
Packit |
6c4009 |
*p++ = 0x00;
|
|
Packit |
6c4009 |
*p++ = 0xff;
|
|
Packit |
6c4009 |
*p++ = 0xff;
|
|
Packit |
6c4009 |
/* Copy the IP v4 address. */
|
|
Packit |
6c4009 |
memcpy (p, tmp, INADDRSZ);
|
|
Packit |
6c4009 |
resbuf->h_addrtype = AF_INET6;
|
|
Packit |
6c4009 |
resbuf->h_length = IN6ADDRSZ;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
else
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
resbuf->h_addrtype = af;
|
|
Packit |
6c4009 |
resbuf->h_length = addr_size;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
if (h_errnop != NULL)
|
|
Packit |
6c4009 |
*h_errnop = NETDB_SUCCESS;
|
|
Packit |
6c4009 |
if (buffer_size == NULL)
|
|
Packit |
6c4009 |
*status = NSS_STATUS_SUCCESS;
|
|
Packit |
6c4009 |
else
|
|
Packit |
6c4009 |
*result = resbuf;
|
|
Packit |
6c4009 |
goto done;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
if (!isdigit (*cp) && *cp != '.')
|
|
Packit |
6c4009 |
break;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
if ((isxdigit (name[0]) && strchr (name, ':') != NULL) || name[0] == ':')
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
switch (af)
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
default:
|
|
Packit |
6c4009 |
af = res_use_inet6 () ? AF_INET6 : AF_INET;
|
|
Packit |
6c4009 |
if (af == AF_INET6)
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
addr_size = IN6ADDRSZ;
|
|
Packit |
6c4009 |
break;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
/* FALLTHROUGH */
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
case AF_INET:
|
|
Packit |
6c4009 |
/* This is not possible. We cannot represent an IPv6 address
|
|
Packit |
6c4009 |
in an `struct in_addr' variable. */
|
|
Packit |
6c4009 |
*h_errnop = HOST_NOT_FOUND;
|
|
Packit |
6c4009 |
if (buffer_size == NULL)
|
|
Packit |
6c4009 |
*status = NSS_STATUS_NOTFOUND;
|
|
Packit |
6c4009 |
else
|
|
Packit |
6c4009 |
*result = NULL;
|
|
Packit |
6c4009 |
goto done;
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
case AF_INET6:
|
|
Packit |
6c4009 |
addr_size = IN6ADDRSZ;
|
|
Packit |
6c4009 |
break;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
for (cp = name;; ++cp)
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
if (!*cp)
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
if (*--cp == '.')
|
|
Packit |
6c4009 |
break;
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
/* All-IPv6-legal, no dot at the end. Fake up a
|
|
Packit |
6c4009 |
hostent as if we'd actually done a lookup. */
|
|
Packit |
6c4009 |
if (inet_pton (AF_INET6, name, host_addr) <= 0)
|
|
Packit |
6c4009 |
{
|
|
Packit |
6c4009 |
*h_errnop = HOST_NOT_FOUND;
|
|
Packit |
6c4009 |
if (buffer_size == NULL)
|
|
Packit |
6c4009 |
*status = NSS_STATUS_NOTFOUND;
|
|
Packit |
6c4009 |
else
|
|
Packit |
6c4009 |
*result = NULL;
|
|
Packit |
6c4009 |
goto done;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
resbuf->h_name = strcpy (hostname, name);
|
|
Packit |
6c4009 |
h_alias_ptr[0] = NULL;
|
|
Packit |
6c4009 |
resbuf->h_aliases = h_alias_ptr;
|
|
Packit |
6c4009 |
(*h_addr_ptrs)[0] = (char *) host_addr;
|
|
Packit |
6c4009 |
(*h_addr_ptrs)[1] = (char *) 0;
|
|
Packit |
6c4009 |
resbuf->h_addr_list = *h_addr_ptrs;
|
|
Packit |
6c4009 |
resbuf->h_addrtype = AF_INET6;
|
|
Packit |
6c4009 |
resbuf->h_length = addr_size;
|
|
Packit |
6c4009 |
*h_errnop = NETDB_SUCCESS;
|
|
Packit |
6c4009 |
if (buffer_size == NULL)
|
|
Packit |
6c4009 |
*status = NSS_STATUS_SUCCESS;
|
|
Packit |
6c4009 |
else
|
|
Packit |
6c4009 |
*result = resbuf;
|
|
Packit |
6c4009 |
goto done;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
if (!isxdigit (*cp) && *cp != ':' && *cp != '.')
|
|
Packit |
6c4009 |
break;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
return 0;
|
|
Packit |
6c4009 |
|
|
Packit |
6c4009 |
done:
|
|
Packit |
6c4009 |
return 1;
|
|
Packit |
6c4009 |
}
|
|
Packit |
6c4009 |
libc_hidden_def (__nss_hostname_digits_dots)
|