Blame inet/getnameinfo.c

Packit 6c4009
/* Convert socket address to string using Name Service Switch modules.
Packit 6c4009
   Copyright (C) 1997-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
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
/* The Inner Net License, Version 2.00
Packit 6c4009
Packit 6c4009
  The author(s) grant permission for redistribution and use in source and
Packit 6c4009
binary forms, with or without modification, of the software and documentation
Packit 6c4009
provided that the following conditions are met:
Packit 6c4009
Packit 6c4009
0. If you receive a version of the software that is specifically labelled
Packit 6c4009
   as not being for redistribution (check the version message and/or README),
Packit 6c4009
   you are not permitted to redistribute that version of the software in any
Packit 6c4009
   way or form.
Packit 6c4009
1. All terms of the all other applicable copyrights and licenses must be
Packit 6c4009
   followed.
Packit 6c4009
2. Redistributions of source code must retain the authors' copyright
Packit 6c4009
   notice(s), this list of conditions, and the following disclaimer.
Packit 6c4009
3. Redistributions in binary form must reproduce the authors' copyright
Packit 6c4009
   notice(s), this list of conditions, and the following disclaimer in the
Packit 6c4009
   documentation and/or other materials provided with the distribution.
Packit 6c4009
4. [The copyright holder has authorized the removal of this clause.]
Packit 6c4009
5. Neither the name(s) of the author(s) nor the names of its contributors
Packit 6c4009
   may be used to endorse or promote products derived from this software
Packit 6c4009
   without specific prior written permission.
Packit 6c4009
Packit 6c4009
THIS SOFTWARE IS PROVIDED BY ITS AUTHORS AND CONTRIBUTORS ``AS IS'' AND ANY
Packit 6c4009
EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
Packit 6c4009
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
Packit 6c4009
DISCLAIMED. IN NO EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE FOR ANY
Packit 6c4009
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
Packit 6c4009
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
Packit 6c4009
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
Packit 6c4009
ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
Packit 6c4009
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
Packit 6c4009
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
Packit 6c4009
Packit 6c4009
  If these license terms cause you a real problem, contact the author.  */
Packit 6c4009
Packit 6c4009
/* This software is Copyright 1996 by Craig Metz, All Rights Reserved.  */
Packit 6c4009
Packit 6c4009
#include <errno.h>
Packit 6c4009
#include <netdb.h>
Packit 6c4009
#include <stddef.h>
Packit 6c4009
#include <stdlib.h>
Packit 6c4009
#include <stdio.h>
Packit 6c4009
#include <string.h>
Packit 6c4009
#include <unistd.h>
Packit 6c4009
#include <stdint.h>
Packit 6c4009
#include <arpa/inet.h>
Packit 6c4009
#include <net/if.h>
Packit 6c4009
#include <netinet/in.h>
Packit 6c4009
#include <sys/param.h>
Packit 6c4009
#include <sys/socket.h>
Packit 6c4009
#include <sys/types.h>
Packit 6c4009
#include <sys/un.h>
Packit 6c4009
#include <sys/utsname.h>
Packit 6c4009
#include <libc-lock.h>
Packit 6c4009
#include <scratch_buffer.h>
Packit 6c4009
#include <net-internal.h>
Packit 6c4009
Packit 6c4009
#ifndef min
Packit 6c4009
# define min(x,y) (((x) > (y)) ? (y) : (x))
Packit 6c4009
#endif /* min */
Packit 6c4009
Packit 6c4009
libc_freeres_ptr (static char *domain);
Packit 6c4009
Packit 6c4009
/* Former NI_IDN_ALLOW_UNASSIGNED, NI_IDN_USE_STD3_ASCII_RULES flags,
Packit 6c4009
   now ignored.  */
Packit 6c4009
#define DEPRECATED_NI_IDN 192
Packit 6c4009
Packit 6c4009
static char *
Packit 6c4009
nrl_domainname (void)
Packit 6c4009
{
Packit 6c4009
  static int not_first;
Packit 6c4009
Packit 6c4009
  if (! not_first)
Packit 6c4009
    {
Packit 6c4009
      __libc_lock_define_initialized (static, lock);
Packit 6c4009
      __libc_lock_lock (lock);
Packit 6c4009
Packit 6c4009
      if (! not_first)
Packit 6c4009
	{
Packit 6c4009
	  char *c;
Packit 6c4009
	  struct hostent *h, th;
Packit 6c4009
	  int herror;
Packit 6c4009
	  struct scratch_buffer tmpbuf;
Packit 6c4009
Packit 6c4009
	  scratch_buffer_init (&tmpbuf);
Packit 6c4009
	  not_first = 1;
Packit 6c4009
Packit 6c4009
	  while (__gethostbyname_r ("localhost", &th,
Packit 6c4009
				    tmpbuf.data, tmpbuf.length,
Packit 6c4009
				    &h, &herror))
Packit 6c4009
	    {
Packit 6c4009
	      if (herror == NETDB_INTERNAL && errno == ERANGE)
Packit 6c4009
		{
Packit 6c4009
		  if (!scratch_buffer_grow (&tmpbuf))
Packit 6c4009
		    goto done;
Packit 6c4009
		}
Packit 6c4009
	      else
Packit 6c4009
		break;
Packit 6c4009
	    }
Packit 6c4009
Packit 6c4009
	  if (h && (c = strchr (h->h_name, '.')))
Packit 6c4009
	    domain = __strdup (++c);
Packit 6c4009
	  else
Packit 6c4009
	    {
Packit 6c4009
	      /* The name contains no domain information.  Use the name
Packit 6c4009
		 now to get more information.  */
Packit 6c4009
	      while (__gethostname (tmpbuf.data, tmpbuf.length))
Packit 6c4009
		if (!scratch_buffer_grow (&tmpbuf))
Packit 6c4009
		  goto done;
Packit 6c4009
Packit 6c4009
	      if ((c = strchr (tmpbuf.data, '.')))
Packit 6c4009
		domain = __strdup (++c);
Packit 6c4009
	      else
Packit 6c4009
		{
Packit 6c4009
		  /* We need to preserve the hostname.  */
Packit 6c4009
		  const char *hstname = strdupa (tmpbuf.data);
Packit 6c4009
Packit 6c4009
		  while (__gethostbyname_r (hstname, &th,
Packit 6c4009
					    tmpbuf.data, tmpbuf.length,
Packit 6c4009
					    &h, &herror))
Packit 6c4009
		    {
Packit 6c4009
		      if (herror == NETDB_INTERNAL && errno == ERANGE)
Packit 6c4009
			{
Packit 6c4009
			  if (!scratch_buffer_grow (&tmpbuf))
Packit 6c4009
			    goto done;
Packit 6c4009
			}
Packit 6c4009
		      else
Packit 6c4009
			break;
Packit 6c4009
		    }
Packit 6c4009
Packit 6c4009
		  if (h && (c = strchr(h->h_name, '.')))
Packit 6c4009
		    domain = __strdup (++c);
Packit 6c4009
		  else
Packit 6c4009
		    {
Packit 6c4009
		      struct in_addr in_addr;
Packit 6c4009
Packit 6c4009
		      in_addr.s_addr = htonl (INADDR_LOOPBACK);
Packit 6c4009
Packit 6c4009
		      while (__gethostbyaddr_r ((const char *) &in_addr,
Packit 6c4009
						sizeof (struct in_addr),
Packit 6c4009
						AF_INET, &th,
Packit 6c4009
						tmpbuf.data, tmpbuf.length,
Packit 6c4009
						&h, &herror))
Packit 6c4009
			{
Packit 6c4009
			  if (herror == NETDB_INTERNAL && errno == ERANGE)
Packit 6c4009
			    {
Packit 6c4009
			      if (!scratch_buffer_grow (&tmpbuf))
Packit 6c4009
				goto done;
Packit 6c4009
			    }
Packit 6c4009
			  else
Packit 6c4009
			    break;
Packit 6c4009
			}
Packit 6c4009
Packit 6c4009
		      if (h && (c = strchr (h->h_name, '.')))
Packit 6c4009
			domain = __strdup (++c);
Packit 6c4009
		    }
Packit 6c4009
		}
Packit 6c4009
	    }
Packit 6c4009
	done:
Packit 6c4009
	  scratch_buffer_free (&tmpbuf);
Packit 6c4009
	}
Packit 6c4009
Packit 6c4009
      __libc_lock_unlock (lock);
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  return domain;
Packit 6c4009
};
Packit 6c4009
Packit 6c4009
/* Copy a string to a destination buffer with length checking.  Return
Packit 6c4009
   EAI_OVERFLOW if the buffer is not large enough, and 0 on
Packit 6c4009
   success.  */
Packit 6c4009
static int
Packit 6c4009
checked_copy (char *dest, size_t destlen, const char *source)
Packit 6c4009
{
Packit 6c4009
  size_t source_length = strlen (source);
Packit 6c4009
  if (source_length + 1 > destlen)
Packit 6c4009
    return EAI_OVERFLOW;
Packit 6c4009
  memcpy (dest, source, source_length + 1);
Packit 6c4009
  return 0;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Helper function for CHECKED_SNPRINTF below.  */
Packit 6c4009
static int
Packit 6c4009
check_sprintf_result (int result, size_t destlen)
Packit 6c4009
{
Packit 6c4009
  if (result < 0)
Packit 6c4009
    return EAI_SYSTEM;
Packit 6c4009
  if ((size_t) result >= destlen)
Packit 6c4009
    /* If ret == destlen, there was no room for the terminating NUL
Packit 6c4009
       character.  */
Packit 6c4009
    return EAI_OVERFLOW;
Packit 6c4009
  return 0;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Format a string in the destination buffer.  Return 0 on success,
Packit 6c4009
   EAI_OVERFLOW in case the buffer is too small, or EAI_SYSTEM on any
Packit 6c4009
   other error.  */
Packit 6c4009
#define CHECKED_SNPRINTF(dest, destlen, format, ...)			\
Packit 6c4009
  check_sprintf_result							\
Packit 6c4009
    (__snprintf (dest, destlen, format, __VA_ARGS__), destlen)
Packit 6c4009
Packit 6c4009
/* Convert host name, AF_INET/AF_INET6 case, name only.  */
Packit 6c4009
static int
Packit 6c4009
gni_host_inet_name (struct scratch_buffer *tmpbuf,
Packit 6c4009
		    const struct sockaddr *sa, socklen_t addrlen,
Packit 6c4009
		    char *host, socklen_t hostlen, int flags)
Packit 6c4009
{
Packit 6c4009
  int herrno;
Packit 6c4009
  struct hostent th;
Packit 6c4009
  struct hostent *h = NULL;
Packit 6c4009
  if (sa->sa_family == AF_INET6)
Packit 6c4009
    {
Packit 6c4009
      const struct sockaddr_in6 *sin6p = (const struct sockaddr_in6 *) sa;
Packit 6c4009
      while (__gethostbyaddr_r (&sin6p->sin6_addr, sizeof(struct in6_addr),
Packit 6c4009
				AF_INET6, &th, tmpbuf->data, tmpbuf->length,
Packit 6c4009
				&h, &herrno))
Packit 6c4009
	if (herrno == NETDB_INTERNAL && errno == ERANGE)
Packit 6c4009
	  {
Packit 6c4009
	    if (!scratch_buffer_grow (tmpbuf))
Packit 6c4009
	      {
Packit 6c4009
		__set_h_errno (herrno);
Packit 6c4009
		return EAI_MEMORY;
Packit 6c4009
	      }
Packit 6c4009
	  }
Packit 6c4009
	else
Packit 6c4009
	  break;
Packit 6c4009
    }
Packit 6c4009
  else
Packit 6c4009
    {
Packit 6c4009
      const struct sockaddr_in *sinp = (const struct sockaddr_in *) sa;
Packit 6c4009
      while (__gethostbyaddr_r (&sinp->sin_addr, sizeof(struct in_addr),
Packit 6c4009
				AF_INET, &th, tmpbuf->data, tmpbuf->length,
Packit 6c4009
				&h, &herrno))
Packit 6c4009
	if (herrno == NETDB_INTERNAL && errno == ERANGE)
Packit 6c4009
	    {
Packit 6c4009
	      if (!scratch_buffer_grow (tmpbuf))
Packit 6c4009
		{
Packit 6c4009
		  __set_h_errno (herrno);
Packit 6c4009
		  return EAI_MEMORY;
Packit 6c4009
		}
Packit 6c4009
	    }
Packit 6c4009
	else
Packit 6c4009
	  break;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  if (h == NULL)
Packit 6c4009
    {
Packit 6c4009
      if (herrno == NETDB_INTERNAL)
Packit 6c4009
	{
Packit 6c4009
	  __set_h_errno (herrno);
Packit 6c4009
	  return EAI_SYSTEM;
Packit 6c4009
	}
Packit 6c4009
      if (herrno == TRY_AGAIN)
Packit 6c4009
	{
Packit 6c4009
	  __set_h_errno (herrno);
Packit 6c4009
	  return EAI_AGAIN;
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  if (h)
Packit 6c4009
    {
Packit 6c4009
      char *c;
Packit 6c4009
      if ((flags & NI_NOFQDN)
Packit 6c4009
	  && (c = nrl_domainname ())
Packit 6c4009
	  && (c = strstr (h->h_name, c))
Packit 6c4009
	  && (c != h->h_name) && (*(--c) == '.'))
Packit 6c4009
	/* Terminate the string after the prefix.  */
Packit 6c4009
	*c = '\0';
Packit 6c4009
Packit 6c4009
      /* If requested, convert from the IDN format.  */
Packit 6c4009
      bool do_idn = flags & NI_IDN;
Packit 6c4009
      char *h_name;
Packit 6c4009
      if (do_idn)
Packit 6c4009
	{
Packit 6c4009
	  int rc = __idna_from_dns_encoding (h->h_name, &h_name);
Packit 6c4009
	  if (rc == EAI_IDN_ENCODE)
Packit 6c4009
	    /* Use the punycode name as a fallback.  */
Packit 6c4009
	    do_idn = false;
Packit 6c4009
	  else if (rc != 0)
Packit 6c4009
	    return rc;
Packit 6c4009
	}
Packit 6c4009
      if (!do_idn)
Packit 6c4009
	h_name = h->h_name;
Packit 6c4009
Packit 6c4009
      size_t len = strlen (h_name) + 1;
Packit 6c4009
      if (len > hostlen)
Packit 6c4009
	return EAI_OVERFLOW;
Packit 6c4009
      memcpy (host, h_name, len);
Packit 6c4009
Packit 6c4009
      if (do_idn)
Packit 6c4009
	free (h_name);
Packit 6c4009
Packit 6c4009
      return 0;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  return EAI_NONAME;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Convert host name, AF_INET/AF_INET6 case, numeric conversion.  */
Packit 6c4009
static int
Packit 6c4009
gni_host_inet_numeric (struct scratch_buffer *tmpbuf,
Packit 6c4009
		       const struct sockaddr *sa, socklen_t addrlen,
Packit 6c4009
		       char *host, socklen_t hostlen, int flags)
Packit 6c4009
{
Packit 6c4009
  if (sa->sa_family == AF_INET6)
Packit 6c4009
    {
Packit 6c4009
      const struct sockaddr_in6 *sin6p = (const struct sockaddr_in6 *) sa;
Packit 6c4009
      if (inet_ntop (AF_INET6, &sin6p->sin6_addr, host, hostlen) == NULL)
Packit 6c4009
	return EAI_OVERFLOW;
Packit 6c4009
Packit 6c4009
      uint32_t scopeid = sin6p->sin6_scope_id;
Packit 6c4009
      if (scopeid != 0)
Packit 6c4009
	{
Packit 6c4009
	  size_t used_hostlen = __strnlen (host, hostlen);
Packit 6c4009
	  /* Location of the scope string in the host buffer.  */
Packit 6c4009
	  char *scope_start = host + used_hostlen;
Packit 6c4009
	  size_t scope_length = hostlen - used_hostlen;
Packit 6c4009
Packit 6c4009
	  if (IN6_IS_ADDR_LINKLOCAL (&sin6p->sin6_addr)
Packit 6c4009
	      || IN6_IS_ADDR_MC_LINKLOCAL (&sin6p->sin6_addr))
Packit 6c4009
	    {
Packit 6c4009
	      char scopebuf[IFNAMSIZ];
Packit 6c4009
	      if (if_indextoname (scopeid, scopebuf) != NULL)
Packit 6c4009
		return CHECKED_SNPRINTF
Packit 6c4009
		  (scope_start, scope_length,
Packit 6c4009
		   "%c%s", SCOPE_DELIMITER, scopebuf);
Packit 6c4009
	    }
Packit 6c4009
	  return CHECKED_SNPRINTF
Packit 6c4009
	    (scope_start, scope_length, "%c%u", SCOPE_DELIMITER, scopeid);
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
  else
Packit 6c4009
    {
Packit 6c4009
      const struct sockaddr_in *sinp = (const struct sockaddr_in *) sa;
Packit 6c4009
      if (inet_ntop (AF_INET, &sinp->sin_addr, host, hostlen) == NULL)
Packit 6c4009
	return EAI_OVERFLOW;
Packit 6c4009
    }
Packit 6c4009
  return 0;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Convert AF_INET or AF_INET6 socket address, host part.  */
Packit 6c4009
static int
Packit 6c4009
gni_host_inet (struct scratch_buffer *tmpbuf,
Packit 6c4009
	       const struct sockaddr *sa, socklen_t addrlen,
Packit 6c4009
	       char *host, socklen_t hostlen, int flags)
Packit 6c4009
{
Packit 6c4009
  if (!(flags & NI_NUMERICHOST))
Packit 6c4009
    {
Packit 6c4009
      int result = gni_host_inet_name
Packit 6c4009
	(tmpbuf, sa, addrlen, host, hostlen, flags);
Packit 6c4009
      if (result != EAI_NONAME)
Packit 6c4009
	return result;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  if (flags & NI_NAMEREQD)
Packit 6c4009
    return EAI_NONAME;
Packit 6c4009
  else
Packit 6c4009
    return gni_host_inet_numeric
Packit 6c4009
      (tmpbuf, sa, addrlen, host, hostlen, flags);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Convert AF_LOCAL socket address, host part.   */
Packit 6c4009
static int
Packit 6c4009
gni_host_local (struct scratch_buffer *tmpbuf,
Packit 6c4009
		const struct sockaddr *sa, socklen_t addrlen,
Packit 6c4009
		char *host, socklen_t hostlen, int flags)
Packit 6c4009
{
Packit 6c4009
  if (!(flags & NI_NUMERICHOST))
Packit 6c4009
    {
Packit 6c4009
      struct utsname utsname;
Packit 6c4009
      if (uname (&utsname) == 0)
Packit 6c4009
	return checked_copy (host, hostlen, utsname.nodename);
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  if (flags & NI_NAMEREQD)
Packit 6c4009
    return EAI_NONAME;
Packit 6c4009
Packit 6c4009
  return checked_copy (host, hostlen, "localhost");
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Convert the host part of an AF_LOCAK socket address.   */
Packit 6c4009
static int
Packit 6c4009
gni_host (struct scratch_buffer *tmpbuf,
Packit 6c4009
	  const struct sockaddr *sa, socklen_t addrlen,
Packit 6c4009
	  char *host, socklen_t hostlen, int flags)
Packit 6c4009
{
Packit 6c4009
  switch (sa->sa_family)
Packit 6c4009
    {
Packit 6c4009
    case AF_INET:
Packit 6c4009
    case AF_INET6:
Packit 6c4009
      return gni_host_inet (tmpbuf, sa, addrlen, host, hostlen, flags);
Packit 6c4009
Packit 6c4009
    case AF_LOCAL:
Packit 6c4009
      return gni_host_local (tmpbuf, sa, addrlen, host, hostlen, flags);
Packit 6c4009
Packit 6c4009
    default:
Packit 6c4009
      return EAI_FAMILY;
Packit 6c4009
    }
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Convert service to string, AF_INET and AF_INET6 variant.  */
Packit 6c4009
static int
Packit 6c4009
gni_serv_inet (struct scratch_buffer *tmpbuf,
Packit 6c4009
	       const struct sockaddr *sa, socklen_t addrlen,
Packit 6c4009
	       char *serv, socklen_t servlen, int flags)
Packit 6c4009
{
Packit 6c4009
  _Static_assert
Packit 6c4009
    (offsetof (struct sockaddr_in, sin_port)
Packit 6c4009
     == offsetof (struct sockaddr_in6, sin6_port)
Packit 6c4009
     && sizeof (((struct sockaddr_in) {}).sin_port) == sizeof (in_port_t)
Packit 6c4009
     && sizeof (((struct sockaddr_in6) {}).sin6_port) == sizeof (in_port_t),
Packit 6c4009
     "AF_INET and AF_INET6 port consistency");
Packit 6c4009
  const struct sockaddr_in *sinp = (const struct sockaddr_in *) sa;
Packit 6c4009
  if (!(flags & NI_NUMERICSERV))
Packit 6c4009
    {
Packit 6c4009
      struct servent *s, ts;
Packit 6c4009
      int e;
Packit 6c4009
      while ((e = __getservbyport_r (sinp->sin_port,
Packit 6c4009
				     ((flags & NI_DGRAM)
Packit 6c4009
				      ? "udp" : "tcp"), &ts,
Packit 6c4009
				     tmpbuf->data, tmpbuf->length, &s)))
Packit 6c4009
	{
Packit 6c4009
	  if (e == ERANGE)
Packit 6c4009
	    {
Packit 6c4009
	      if (!scratch_buffer_grow (tmpbuf))
Packit 6c4009
		return EAI_MEMORY;
Packit 6c4009
	    }
Packit 6c4009
	  else
Packit 6c4009
	    break;
Packit 6c4009
	}
Packit 6c4009
      if (s)
Packit 6c4009
	return checked_copy (serv, servlen, s->s_name);
Packit 6c4009
      /* Fall through to numeric conversion.  */
Packit 6c4009
    }
Packit 6c4009
  return CHECKED_SNPRINTF (serv, servlen, "%d", ntohs (sinp->sin_port));
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Convert service to string, AF_LOCAL variant.  */
Packit 6c4009
static int
Packit 6c4009
gni_serv_local (struct scratch_buffer *tmpbuf,
Packit 6c4009
	       const struct sockaddr *sa, socklen_t addrlen,
Packit 6c4009
	       char *serv, socklen_t servlen, int flags)
Packit 6c4009
{
Packit 6c4009
  return checked_copy
Packit 6c4009
    (serv, servlen, ((const struct sockaddr_un *) sa)->sun_path);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
/* Convert service to string, dispatching to the implementations
Packit 6c4009
   above.  */
Packit 6c4009
static int
Packit 6c4009
gni_serv (struct scratch_buffer *tmpbuf,
Packit 6c4009
	  const struct sockaddr *sa, socklen_t addrlen,
Packit 6c4009
	  char *serv, socklen_t servlen, int flags)
Packit 6c4009
{
Packit 6c4009
  switch (sa->sa_family)
Packit 6c4009
    {
Packit 6c4009
    case AF_INET:
Packit 6c4009
    case AF_INET6:
Packit 6c4009
      return gni_serv_inet (tmpbuf, sa, addrlen, serv, servlen, flags);
Packit 6c4009
    case AF_LOCAL:
Packit 6c4009
      return gni_serv_local (tmpbuf, sa, addrlen, serv, servlen, flags);
Packit 6c4009
    default:
Packit 6c4009
      return EAI_FAMILY;
Packit 6c4009
    }
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
int
Packit 6c4009
getnameinfo (const struct sockaddr *sa, socklen_t addrlen, char *host,
Packit 6c4009
	     socklen_t hostlen, char *serv, socklen_t servlen,
Packit 6c4009
	     int flags)
Packit 6c4009
{
Packit 6c4009
  if (flags & ~(NI_NUMERICHOST|NI_NUMERICSERV|NI_NOFQDN|NI_NAMEREQD|NI_DGRAM
Packit 6c4009
		|NI_IDN|DEPRECATED_NI_IDN))
Packit 6c4009
    return EAI_BADFLAGS;
Packit 6c4009
Packit 6c4009
  if (sa == NULL || addrlen < sizeof (sa_family_t))
Packit 6c4009
    return EAI_FAMILY;
Packit 6c4009
Packit 6c4009
  if ((flags & NI_NAMEREQD) && host == NULL && serv == NULL)
Packit 6c4009
    return EAI_NONAME;
Packit 6c4009
Packit 6c4009
  switch (sa->sa_family)
Packit 6c4009
    {
Packit 6c4009
    case AF_LOCAL:
Packit 6c4009
      if (addrlen < (socklen_t) offsetof (struct sockaddr_un, sun_path))
Packit 6c4009
	return EAI_FAMILY;
Packit 6c4009
      break;
Packit 6c4009
    case AF_INET:
Packit 6c4009
      if (addrlen < sizeof (struct sockaddr_in))
Packit 6c4009
	return EAI_FAMILY;
Packit 6c4009
      break;
Packit 6c4009
    case AF_INET6:
Packit 6c4009
      if (addrlen < sizeof (struct sockaddr_in6))
Packit 6c4009
	return EAI_FAMILY;
Packit 6c4009
      break;
Packit 6c4009
    default:
Packit 6c4009
      return EAI_FAMILY;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  struct scratch_buffer tmpbuf;
Packit 6c4009
  scratch_buffer_init (&tmpbuf);
Packit 6c4009
Packit 6c4009
  if (host != NULL && hostlen > 0)
Packit 6c4009
    {
Packit 6c4009
      int result = gni_host (&tmpbuf, sa, addrlen, host, hostlen, flags);
Packit 6c4009
      if (result != 0)
Packit 6c4009
	{
Packit 6c4009
	  scratch_buffer_free (&tmpbuf);
Packit 6c4009
	  return result;
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  if (serv && (servlen > 0))
Packit 6c4009
    {
Packit 6c4009
      int result = gni_serv (&tmpbuf, sa, addrlen, serv, servlen, flags);
Packit 6c4009
      if (result != 0)
Packit 6c4009
	{
Packit 6c4009
	  scratch_buffer_free (&tmpbuf);
Packit 6c4009
	  return result;
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  scratch_buffer_free (&tmpbuf);
Packit 6c4009
  return 0;
Packit 6c4009
}
Packit 6c4009
libc_hidden_def (getnameinfo)