Blame src/unix/getaddrinfo.c

Packit Service 7c31a4
/* Copyright Joyent, Inc. and other Node contributors. All rights reserved.
Packit Service 7c31a4
 * Permission is hereby granted, free of charge, to any person obtaining a copy
Packit Service 7c31a4
 * of this software and associated documentation files (the "Software"), to
Packit Service 7c31a4
 * deal in the Software without restriction, including without limitation the
Packit Service 7c31a4
 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
Packit Service 7c31a4
 * sell copies of the Software, and to permit persons to whom the Software is
Packit Service 7c31a4
 * furnished to do so, subject to the following conditions:
Packit Service 7c31a4
 *
Packit Service 7c31a4
 * The above copyright notice and this permission notice shall be included in
Packit Service 7c31a4
 * all copies or substantial portions of the Software.
Packit Service 7c31a4
 *
Packit Service 7c31a4
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
Packit Service 7c31a4
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
Packit Service 7c31a4
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
Packit Service 7c31a4
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
Packit Service 7c31a4
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
Packit Service 7c31a4
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
Packit Service 7c31a4
 * IN THE SOFTWARE.
Packit Service 7c31a4
 */
Packit Service 7c31a4
Packit Service 7c31a4
/* Expose glibc-specific EAI_* error codes. Needs to be defined before we
Packit Service 7c31a4
 * include any headers.
Packit Service 7c31a4
 */
Packit Service 7c31a4
#ifndef _GNU_SOURCE
Packit Service 7c31a4
# define _GNU_SOURCE
Packit Service 7c31a4
#endif
Packit Service 7c31a4
Packit Service 7c31a4
#include "uv.h"
Packit Service 7c31a4
#include "internal.h"
Packit Service 7c31a4
#include "idna.h"
Packit Service 7c31a4
Packit Service 7c31a4
#include <errno.h>
Packit Service 7c31a4
#include <stddef.h> /* NULL */
Packit Service 7c31a4
#include <stdlib.h>
Packit Service 7c31a4
#include <string.h>
Packit Service 7c31a4
#include <net/if.h> /* if_indextoname() */
Packit Service 7c31a4
Packit Service 7c31a4
/* EAI_* constants. */
Packit Service 7c31a4
#include <netdb.h>
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
int uv__getaddrinfo_translate_error(int sys_err) {
Packit Service 7c31a4
  switch (sys_err) {
Packit Service 7c31a4
  case 0: return 0;
Packit Service 7c31a4
#if defined(EAI_ADDRFAMILY)
Packit Service 7c31a4
  case EAI_ADDRFAMILY: return UV_EAI_ADDRFAMILY;
Packit Service 7c31a4
#endif
Packit Service 7c31a4
#if defined(EAI_AGAIN)
Packit Service 7c31a4
  case EAI_AGAIN: return UV_EAI_AGAIN;
Packit Service 7c31a4
#endif
Packit Service 7c31a4
#if defined(EAI_BADFLAGS)
Packit Service 7c31a4
  case EAI_BADFLAGS: return UV_EAI_BADFLAGS;
Packit Service 7c31a4
#endif
Packit Service 7c31a4
#if defined(EAI_BADHINTS)
Packit Service 7c31a4
  case EAI_BADHINTS: return UV_EAI_BADHINTS;
Packit Service 7c31a4
#endif
Packit Service 7c31a4
#if defined(EAI_CANCELED)
Packit Service 7c31a4
  case EAI_CANCELED: return UV_EAI_CANCELED;
Packit Service 7c31a4
#endif
Packit Service 7c31a4
#if defined(EAI_FAIL)
Packit Service 7c31a4
  case EAI_FAIL: return UV_EAI_FAIL;
Packit Service 7c31a4
#endif
Packit Service 7c31a4
#if defined(EAI_FAMILY)
Packit Service 7c31a4
  case EAI_FAMILY: return UV_EAI_FAMILY;
Packit Service 7c31a4
#endif
Packit Service 7c31a4
#if defined(EAI_MEMORY)
Packit Service 7c31a4
  case EAI_MEMORY: return UV_EAI_MEMORY;
Packit Service 7c31a4
#endif
Packit Service 7c31a4
#if defined(EAI_NODATA)
Packit Service 7c31a4
  case EAI_NODATA: return UV_EAI_NODATA;
Packit Service 7c31a4
#endif
Packit Service 7c31a4
#if defined(EAI_NONAME)
Packit Service 7c31a4
# if !defined(EAI_NODATA) || EAI_NODATA != EAI_NONAME
Packit Service 7c31a4
  case EAI_NONAME: return UV_EAI_NONAME;
Packit Service 7c31a4
# endif
Packit Service 7c31a4
#endif
Packit Service 7c31a4
#if defined(EAI_OVERFLOW)
Packit Service 7c31a4
  case EAI_OVERFLOW: return UV_EAI_OVERFLOW;
Packit Service 7c31a4
#endif
Packit Service 7c31a4
#if defined(EAI_PROTOCOL)
Packit Service 7c31a4
  case EAI_PROTOCOL: return UV_EAI_PROTOCOL;
Packit Service 7c31a4
#endif
Packit Service 7c31a4
#if defined(EAI_SERVICE)
Packit Service 7c31a4
  case EAI_SERVICE: return UV_EAI_SERVICE;
Packit Service 7c31a4
#endif
Packit Service 7c31a4
#if defined(EAI_SOCKTYPE)
Packit Service 7c31a4
  case EAI_SOCKTYPE: return UV_EAI_SOCKTYPE;
Packit Service 7c31a4
#endif
Packit Service 7c31a4
#if defined(EAI_SYSTEM)
Packit Service 7c31a4
  case EAI_SYSTEM: return UV__ERR(errno);
Packit Service 7c31a4
#endif
Packit Service 7c31a4
  }
Packit Service 7c31a4
  assert(!"unknown EAI_* error code");
Packit Service 7c31a4
  abort();
Packit Service 7c31a4
#ifndef __SUNPRO_C
Packit Service 7c31a4
  return 0;  /* Pacify compiler. */
Packit Service 7c31a4
#endif
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void uv__getaddrinfo_work(struct uv__work* w) {
Packit Service 7c31a4
  uv_getaddrinfo_t* req;
Packit Service 7c31a4
  int err;
Packit Service 7c31a4
Packit Service 7c31a4
  req = container_of(w, uv_getaddrinfo_t, work_req);
Packit Service 7c31a4
  err = getaddrinfo(req->hostname, req->service, req->hints, &req->addrinfo);
Packit Service 7c31a4
  req->retcode = uv__getaddrinfo_translate_error(err);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void uv__getaddrinfo_done(struct uv__work* w, int status) {
Packit Service 7c31a4
  uv_getaddrinfo_t* req;
Packit Service 7c31a4
Packit Service 7c31a4
  req = container_of(w, uv_getaddrinfo_t, work_req);
Packit Service 7c31a4
  uv__req_unregister(req->loop, req);
Packit Service 7c31a4
Packit Service 7c31a4
  /* See initialization in uv_getaddrinfo(). */
Packit Service 7c31a4
  if (req->hints)
Packit Service 7c31a4
    uv__free(req->hints);
Packit Service 7c31a4
  else if (req->service)
Packit Service 7c31a4
    uv__free(req->service);
Packit Service 7c31a4
  else if (req->hostname)
Packit Service 7c31a4
    uv__free(req->hostname);
Packit Service 7c31a4
  else
Packit Service 7c31a4
    assert(0);
Packit Service 7c31a4
Packit Service 7c31a4
  req->hints = NULL;
Packit Service 7c31a4
  req->service = NULL;
Packit Service 7c31a4
  req->hostname = NULL;
Packit Service 7c31a4
Packit Service 7c31a4
  if (status == UV_ECANCELED) {
Packit Service 7c31a4
    assert(req->retcode == 0);
Packit Service 7c31a4
    req->retcode = UV_EAI_CANCELED;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (req->cb)
Packit Service 7c31a4
    req->cb(req, req->retcode, req->addrinfo);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
int uv_getaddrinfo(uv_loop_t* loop,
Packit Service 7c31a4
                   uv_getaddrinfo_t* req,
Packit Service 7c31a4
                   uv_getaddrinfo_cb cb,
Packit Service 7c31a4
                   const char* hostname,
Packit Service 7c31a4
                   const char* service,
Packit Service 7c31a4
                   const struct addrinfo* hints) {
Packit Service 7c31a4
  char hostname_ascii[256];
Packit Service 7c31a4
  size_t hostname_len;
Packit Service 7c31a4
  size_t service_len;
Packit Service 7c31a4
  size_t hints_len;
Packit Service 7c31a4
  size_t len;
Packit Service 7c31a4
  char* buf;
Packit Service 7c31a4
  long rc;
Packit Service 7c31a4
Packit Service 7c31a4
  if (req == NULL || (hostname == NULL && service == NULL))
Packit Service 7c31a4
    return UV_EINVAL;
Packit Service 7c31a4
Packit Service 7c31a4
  /* FIXME(bnoordhuis) IDNA does not seem to work z/OS,
Packit Service 7c31a4
   * probably because it uses EBCDIC rather than ASCII.
Packit Service 7c31a4
   */
Packit Service 7c31a4
#ifdef __MVS__
Packit Service 7c31a4
  (void) &hostname_ascii;
Packit Service 7c31a4
#else
Packit Service 7c31a4
  if (hostname != NULL) {
Packit Service 7c31a4
    rc = uv__idna_toascii(hostname,
Packit Service 7c31a4
                          hostname + strlen(hostname),
Packit Service 7c31a4
                          hostname_ascii,
Packit Service 7c31a4
                          hostname_ascii + sizeof(hostname_ascii));
Packit Service 7c31a4
    if (rc < 0)
Packit Service 7c31a4
      return rc;
Packit Service 7c31a4
    hostname = hostname_ascii;
Packit Service 7c31a4
  }
Packit Service 7c31a4
#endif
Packit Service 7c31a4
Packit Service 7c31a4
  hostname_len = hostname ? strlen(hostname) + 1 : 0;
Packit Service 7c31a4
  service_len = service ? strlen(service) + 1 : 0;
Packit Service 7c31a4
  hints_len = hints ? sizeof(*hints) : 0;
Packit Service 7c31a4
  buf = uv__malloc(hostname_len + service_len + hints_len);
Packit Service 7c31a4
Packit Service 7c31a4
  if (buf == NULL)
Packit Service 7c31a4
    return UV_ENOMEM;
Packit Service 7c31a4
Packit Service 7c31a4
  uv__req_init(loop, req, UV_GETADDRINFO);
Packit Service 7c31a4
  req->loop = loop;
Packit Service 7c31a4
  req->cb = cb;
Packit Service 7c31a4
  req->addrinfo = NULL;
Packit Service 7c31a4
  req->hints = NULL;
Packit Service 7c31a4
  req->service = NULL;
Packit Service 7c31a4
  req->hostname = NULL;
Packit Service 7c31a4
  req->retcode = 0;
Packit Service 7c31a4
Packit Service 7c31a4
  /* order matters, see uv_getaddrinfo_done() */
Packit Service 7c31a4
  len = 0;
Packit Service 7c31a4
Packit Service 7c31a4
  if (hints) {
Packit Service 7c31a4
    req->hints = memcpy(buf + len, hints, sizeof(*hints));
Packit Service 7c31a4
    len += sizeof(*hints);
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (service) {
Packit Service 7c31a4
    req->service = memcpy(buf + len, service, service_len);
Packit Service 7c31a4
    len += service_len;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (hostname)
Packit Service 7c31a4
    req->hostname = memcpy(buf + len, hostname, hostname_len);
Packit Service 7c31a4
Packit Service 7c31a4
  if (cb) {
Packit Service 7c31a4
    uv__work_submit(loop,
Packit Service 7c31a4
                    &req->work_req,
Packit Service 7c31a4
                    UV__WORK_SLOW_IO,
Packit Service 7c31a4
                    uv__getaddrinfo_work,
Packit Service 7c31a4
                    uv__getaddrinfo_done);
Packit Service 7c31a4
    return 0;
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    uv__getaddrinfo_work(&req->work_req);
Packit Service 7c31a4
    uv__getaddrinfo_done(&req->work_req, 0);
Packit Service 7c31a4
    return req->retcode;
Packit Service 7c31a4
  }
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
void uv_freeaddrinfo(struct addrinfo* ai) {
Packit Service 7c31a4
  if (ai)
Packit Service 7c31a4
    freeaddrinfo(ai);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
int uv_if_indextoname(unsigned int ifindex, char* buffer, size_t* size) {
Packit Service 7c31a4
  char ifname_buf[UV_IF_NAMESIZE];
Packit Service 7c31a4
  size_t len;
Packit Service 7c31a4
Packit Service 7c31a4
  if (buffer == NULL || size == NULL || *size == 0)
Packit Service 7c31a4
    return UV_EINVAL;
Packit Service 7c31a4
Packit Service 7c31a4
  if (if_indextoname(ifindex, ifname_buf) == NULL)
Packit Service 7c31a4
    return UV__ERR(errno);
Packit Service 7c31a4
Packit Service 7c31a4
  len = strnlen(ifname_buf, sizeof(ifname_buf));
Packit Service 7c31a4
Packit Service 7c31a4
  if (*size <= len) {
Packit Service 7c31a4
    *size = len + 1;
Packit Service 7c31a4
    return UV_ENOBUFS;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  memcpy(buffer, ifname_buf, len);
Packit Service 7c31a4
  buffer[len] = '\0';
Packit Service 7c31a4
  *size = len;
Packit Service 7c31a4
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
int uv_if_indextoiid(unsigned int ifindex, char* buffer, size_t* size) {
Packit Service 7c31a4
  return uv_if_indextoname(ifindex, buffer, size);
Packit Service 7c31a4
}