Blame src/uv-common.c

Packit b5b901
/* Copyright Joyent, Inc. and other Node contributors. All rights reserved.
Packit b5b901
 *
Packit b5b901
 * Permission is hereby granted, free of charge, to any person obtaining a copy
Packit b5b901
 * of this software and associated documentation files (the "Software"), to
Packit b5b901
 * deal in the Software without restriction, including without limitation the
Packit b5b901
 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
Packit b5b901
 * sell copies of the Software, and to permit persons to whom the Software is
Packit b5b901
 * furnished to do so, subject to the following conditions:
Packit b5b901
 *
Packit b5b901
 * The above copyright notice and this permission notice shall be included in
Packit b5b901
 * all copies or substantial portions of the Software.
Packit b5b901
 *
Packit b5b901
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
Packit b5b901
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
Packit b5b901
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
Packit b5b901
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
Packit b5b901
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
Packit b5b901
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
Packit b5b901
 * IN THE SOFTWARE.
Packit b5b901
 */
Packit b5b901
Packit b5b901
#include "uv.h"
Packit b5b901
#include "uv-common.h"
Packit b5b901
Packit b5b901
#include <assert.h>
Packit b5b901
#include <errno.h>
Packit b5b901
#include <stdarg.h>
Packit b5b901
#include <stddef.h> /* NULL */
Packit b5b901
#include <stdio.h>
Packit b5b901
#include <stdlib.h> /* malloc */
Packit b5b901
#include <string.h> /* memset */
Packit b5b901
Packit b5b901
#if defined(_WIN32)
Packit b5b901
# include <malloc.h> /* malloc */
Packit b5b901
#else
Packit b5b901
# include <net/if.h> /* if_nametoindex */
Packit Service e08953
# include <sys/un.h> /* AF_UNIX, sockaddr_un */
Packit b5b901
#endif
Packit b5b901
Packit b5b901
Packit b5b901
typedef struct {
Packit b5b901
  uv_malloc_func local_malloc;
Packit b5b901
  uv_realloc_func local_realloc;
Packit b5b901
  uv_calloc_func local_calloc;
Packit b5b901
  uv_free_func local_free;
Packit b5b901
} uv__allocator_t;
Packit b5b901
Packit b5b901
static uv__allocator_t uv__allocator = {
Packit b5b901
  malloc,
Packit b5b901
  realloc,
Packit b5b901
  calloc,
Packit b5b901
  free,
Packit b5b901
};
Packit b5b901
Packit b5b901
char* uv__strdup(const char* s) {
Packit b5b901
  size_t len = strlen(s) + 1;
Packit b5b901
  char* m = uv__malloc(len);
Packit b5b901
  if (m == NULL)
Packit b5b901
    return NULL;
Packit b5b901
  return memcpy(m, s, len);
Packit b5b901
}
Packit b5b901
Packit b5b901
char* uv__strndup(const char* s, size_t n) {
Packit b5b901
  char* m;
Packit b5b901
  size_t len = strlen(s);
Packit b5b901
  if (n < len)
Packit b5b901
    len = n;
Packit b5b901
  m = uv__malloc(len + 1);
Packit b5b901
  if (m == NULL)
Packit b5b901
    return NULL;
Packit b5b901
  m[len] = '\0';
Packit b5b901
  return memcpy(m, s, len);
Packit b5b901
}
Packit b5b901
Packit b5b901
void* uv__malloc(size_t size) {
Packit Service e08953
  if (size > 0)
Packit Service e08953
    return uv__allocator.local_malloc(size);
Packit Service e08953
  return NULL;
Packit b5b901
}
Packit b5b901
Packit b5b901
void uv__free(void* ptr) {
Packit b5b901
  int saved_errno;
Packit b5b901
Packit b5b901
  /* Libuv expects that free() does not clobber errno.  The system allocator
Packit b5b901
   * honors that assumption but custom allocators may not be so careful.
Packit b5b901
   */
Packit b5b901
  saved_errno = errno;
Packit b5b901
  uv__allocator.local_free(ptr);
Packit b5b901
  errno = saved_errno;
Packit b5b901
}
Packit b5b901
Packit b5b901
void* uv__calloc(size_t count, size_t size) {
Packit b5b901
  return uv__allocator.local_calloc(count, size);
Packit b5b901
}
Packit b5b901
Packit b5b901
void* uv__realloc(void* ptr, size_t size) {
Packit Service e08953
  if (size > 0)
Packit Service e08953
    return uv__allocator.local_realloc(ptr, size);
Packit Service e08953
  uv__free(ptr);
Packit Service e08953
  return NULL;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
void* uv__reallocf(void* ptr, size_t size) {
Packit Service e08953
  void* newptr;
Packit Service e08953
Packit Service e08953
  newptr = uv__realloc(ptr, size);
Packit Service e08953
  if (newptr == NULL)
Packit Service e08953
    if (size > 0)
Packit Service e08953
      uv__free(ptr);
Packit Service e08953
Packit Service e08953
  return newptr;
Packit b5b901
}
Packit b5b901
Packit b5b901
int uv_replace_allocator(uv_malloc_func malloc_func,
Packit b5b901
                         uv_realloc_func realloc_func,
Packit b5b901
                         uv_calloc_func calloc_func,
Packit b5b901
                         uv_free_func free_func) {
Packit b5b901
  if (malloc_func == NULL || realloc_func == NULL ||
Packit b5b901
      calloc_func == NULL || free_func == NULL) {
Packit b5b901
    return UV_EINVAL;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  uv__allocator.local_malloc = malloc_func;
Packit b5b901
  uv__allocator.local_realloc = realloc_func;
Packit b5b901
  uv__allocator.local_calloc = calloc_func;
Packit b5b901
  uv__allocator.local_free = free_func;
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
#define XX(uc, lc) case UV_##uc: return sizeof(uv_##lc##_t);
Packit b5b901
Packit b5b901
size_t uv_handle_size(uv_handle_type type) {
Packit b5b901
  switch (type) {
Packit b5b901
    UV_HANDLE_TYPE_MAP(XX)
Packit b5b901
    default:
Packit b5b901
      return -1;
Packit b5b901
  }
Packit b5b901
}
Packit b5b901
Packit b5b901
size_t uv_req_size(uv_req_type type) {
Packit b5b901
  switch(type) {
Packit b5b901
    UV_REQ_TYPE_MAP(XX)
Packit b5b901
    default:
Packit b5b901
      return -1;
Packit b5b901
  }
Packit b5b901
}
Packit b5b901
Packit b5b901
#undef XX
Packit b5b901
Packit b5b901
Packit b5b901
size_t uv_loop_size(void) {
Packit b5b901
  return sizeof(uv_loop_t);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
uv_buf_t uv_buf_init(char* base, unsigned int len) {
Packit b5b901
  uv_buf_t buf;
Packit b5b901
  buf.base = base;
Packit b5b901
  buf.len = len;
Packit b5b901
  return buf;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static const char* uv__unknown_err_code(int err) {
Packit b5b901
  char buf[32];
Packit b5b901
  char* copy;
Packit b5b901
Packit b5b901
  snprintf(buf, sizeof(buf), "Unknown system error %d", err);
Packit b5b901
  copy = uv__strdup(buf);
Packit b5b901
Packit b5b901
  return copy != NULL ? copy : "Unknown system error";
Packit b5b901
}
Packit b5b901
Packit b5b901
#define UV_ERR_NAME_GEN_R(name, _) \
Packit b5b901
case UV_## name: \
Packit Service e08953
  uv__strscpy(buf, #name, buflen); break;
Packit b5b901
char* uv_err_name_r(int err, char* buf, size_t buflen) {
Packit b5b901
  switch (err) {
Packit b5b901
    UV_ERRNO_MAP(UV_ERR_NAME_GEN_R)
Packit b5b901
    default: snprintf(buf, buflen, "Unknown system error %d", err);
Packit b5b901
  }
Packit b5b901
  return buf;
Packit b5b901
}
Packit b5b901
#undef UV_ERR_NAME_GEN_R
Packit b5b901
Packit b5b901
Packit b5b901
#define UV_ERR_NAME_GEN(name, _) case UV_ ## name: return #name;
Packit b5b901
const char* uv_err_name(int err) {
Packit b5b901
  switch (err) {
Packit b5b901
    UV_ERRNO_MAP(UV_ERR_NAME_GEN)
Packit b5b901
  }
Packit b5b901
  return uv__unknown_err_code(err);
Packit b5b901
}
Packit b5b901
#undef UV_ERR_NAME_GEN
Packit b5b901
Packit b5b901
Packit b5b901
#define UV_STRERROR_GEN_R(name, msg) \
Packit b5b901
case UV_ ## name: \
Packit b5b901
  snprintf(buf, buflen, "%s", msg); break;
Packit b5b901
char* uv_strerror_r(int err, char* buf, size_t buflen) {
Packit b5b901
  switch (err) {
Packit b5b901
    UV_ERRNO_MAP(UV_STRERROR_GEN_R)
Packit b5b901
    default: snprintf(buf, buflen, "Unknown system error %d", err);
Packit b5b901
  }
Packit b5b901
  return buf;
Packit b5b901
}
Packit b5b901
#undef UV_STRERROR_GEN_R
Packit b5b901
Packit b5b901
Packit b5b901
#define UV_STRERROR_GEN(name, msg) case UV_ ## name: return msg;
Packit b5b901
const char* uv_strerror(int err) {
Packit b5b901
  switch (err) {
Packit b5b901
    UV_ERRNO_MAP(UV_STRERROR_GEN)
Packit b5b901
  }
Packit b5b901
  return uv__unknown_err_code(err);
Packit b5b901
}
Packit b5b901
#undef UV_STRERROR_GEN
Packit b5b901
Packit b5b901
Packit b5b901
int uv_ip4_addr(const char* ip, int port, struct sockaddr_in* addr) {
Packit b5b901
  memset(addr, 0, sizeof(*addr));
Packit b5b901
  addr->sin_family = AF_INET;
Packit b5b901
  addr->sin_port = htons(port);
Packit Service e08953
#ifdef SIN6_LEN
Packit Service e08953
  addr->sin_len = sizeof(*addr);
Packit Service e08953
#endif
Packit b5b901
  return uv_inet_pton(AF_INET, ip, &(addr->sin_addr.s_addr));
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_ip6_addr(const char* ip, int port, struct sockaddr_in6* addr) {
Packit b5b901
  char address_part[40];
Packit b5b901
  size_t address_part_size;
Packit b5b901
  const char* zone_index;
Packit b5b901
Packit b5b901
  memset(addr, 0, sizeof(*addr));
Packit b5b901
  addr->sin6_family = AF_INET6;
Packit b5b901
  addr->sin6_port = htons(port);
Packit Service e08953
#ifdef SIN6_LEN
Packit Service e08953
  addr->sin6_len = sizeof(*addr);
Packit Service e08953
#endif
Packit b5b901
Packit b5b901
  zone_index = strchr(ip, '%');
Packit b5b901
  if (zone_index != NULL) {
Packit b5b901
    address_part_size = zone_index - ip;
Packit b5b901
    if (address_part_size >= sizeof(address_part))
Packit b5b901
      address_part_size = sizeof(address_part) - 1;
Packit b5b901
Packit b5b901
    memcpy(address_part, ip, address_part_size);
Packit b5b901
    address_part[address_part_size] = '\0';
Packit b5b901
    ip = address_part;
Packit b5b901
Packit b5b901
    zone_index++; /* skip '%' */
Packit b5b901
    /* NOTE: unknown interface (id=0) is silently ignored */
Packit b5b901
#ifdef _WIN32
Packit b5b901
    addr->sin6_scope_id = atoi(zone_index);
Packit b5b901
#else
Packit b5b901
    addr->sin6_scope_id = if_nametoindex(zone_index);
Packit b5b901
#endif
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return uv_inet_pton(AF_INET6, ip, &addr->sin6_addr);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_ip4_name(const struct sockaddr_in* src, char* dst, size_t size) {
Packit b5b901
  return uv_inet_ntop(AF_INET, &src->sin_addr, dst, size);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_ip6_name(const struct sockaddr_in6* src, char* dst, size_t size) {
Packit b5b901
  return uv_inet_ntop(AF_INET6, &src->sin6_addr, dst, size);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_tcp_bind(uv_tcp_t* handle,
Packit b5b901
                const struct sockaddr* addr,
Packit b5b901
                unsigned int flags) {
Packit b5b901
  unsigned int addrlen;
Packit b5b901
Packit b5b901
  if (handle->type != UV_TCP)
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit b5b901
  if (addr->sa_family == AF_INET)
Packit b5b901
    addrlen = sizeof(struct sockaddr_in);
Packit b5b901
  else if (addr->sa_family == AF_INET6)
Packit b5b901
    addrlen = sizeof(struct sockaddr_in6);
Packit b5b901
  else
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit b5b901
  return uv__tcp_bind(handle, addr, addrlen, flags);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit Service e08953
int uv_udp_init_ex(uv_loop_t* loop, uv_udp_t* handle, unsigned flags) {
Packit Service e08953
  unsigned extra_flags;
Packit Service e08953
  int domain;
Packit Service e08953
  int rc;
Packit Service e08953
Packit Service e08953
  /* Use the lower 8 bits for the domain. */
Packit Service e08953
  domain = flags & 0xFF;
Packit Service e08953
  if (domain != AF_INET && domain != AF_INET6 && domain != AF_UNSPEC)
Packit Service e08953
    return UV_EINVAL;
Packit Service e08953
Packit Service e08953
  /* Use the higher bits for extra flags. */
Packit Service e08953
  extra_flags = flags & ~0xFF;
Packit Service e08953
  if (extra_flags & ~UV_UDP_RECVMMSG)
Packit Service e08953
    return UV_EINVAL;
Packit Service e08953
Packit Service e08953
  rc = uv__udp_init_ex(loop, handle, flags, domain);
Packit Service e08953
Packit Service e08953
  if (rc == 0)
Packit Service e08953
    if (extra_flags & UV_UDP_RECVMMSG)
Packit Service e08953
      handle->flags |= UV_HANDLE_UDP_RECVMMSG;
Packit Service e08953
Packit Service e08953
  return rc;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit Service e08953
int uv_udp_init(uv_loop_t* loop, uv_udp_t* handle) {
Packit Service e08953
  return uv_udp_init_ex(loop, handle, AF_UNSPEC);
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit b5b901
int uv_udp_bind(uv_udp_t* handle,
Packit b5b901
                const struct sockaddr* addr,
Packit b5b901
                unsigned int flags) {
Packit b5b901
  unsigned int addrlen;
Packit b5b901
Packit b5b901
  if (handle->type != UV_UDP)
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit b5b901
  if (addr->sa_family == AF_INET)
Packit b5b901
    addrlen = sizeof(struct sockaddr_in);
Packit b5b901
  else if (addr->sa_family == AF_INET6)
Packit b5b901
    addrlen = sizeof(struct sockaddr_in6);
Packit b5b901
  else
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit b5b901
  return uv__udp_bind(handle, addr, addrlen, flags);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_tcp_connect(uv_connect_t* req,
Packit b5b901
                   uv_tcp_t* handle,
Packit b5b901
                   const struct sockaddr* addr,
Packit b5b901
                   uv_connect_cb cb) {
Packit b5b901
  unsigned int addrlen;
Packit b5b901
Packit b5b901
  if (handle->type != UV_TCP)
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit b5b901
  if (addr->sa_family == AF_INET)
Packit b5b901
    addrlen = sizeof(struct sockaddr_in);
Packit b5b901
  else if (addr->sa_family == AF_INET6)
Packit b5b901
    addrlen = sizeof(struct sockaddr_in6);
Packit b5b901
  else
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit b5b901
  return uv__tcp_connect(req, handle, addr, addrlen, cb);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit Service e08953
int uv_udp_connect(uv_udp_t* handle, const struct sockaddr* addr) {
Packit b5b901
  unsigned int addrlen;
Packit b5b901
Packit b5b901
  if (handle->type != UV_UDP)
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit Service e08953
  /* Disconnect the handle */
Packit Service e08953
  if (addr == NULL) {
Packit Service e08953
    if (!(handle->flags & UV_HANDLE_UDP_CONNECTED))
Packit Service e08953
      return UV_ENOTCONN;
Packit Service e08953
Packit Service e08953
    return uv__udp_disconnect(handle);
Packit Service e08953
  }
Packit Service e08953
Packit b5b901
  if (addr->sa_family == AF_INET)
Packit b5b901
    addrlen = sizeof(struct sockaddr_in);
Packit b5b901
  else if (addr->sa_family == AF_INET6)
Packit b5b901
    addrlen = sizeof(struct sockaddr_in6);
Packit b5b901
  else
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit Service e08953
  if (handle->flags & UV_HANDLE_UDP_CONNECTED)
Packit Service e08953
    return UV_EISCONN;
Packit Service e08953
Packit Service e08953
  return uv__udp_connect(handle, addr, addrlen);
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit Service e08953
int uv__udp_is_connected(uv_udp_t* handle) {
Packit Service e08953
  struct sockaddr_storage addr;
Packit Service e08953
  int addrlen;
Packit Service e08953
  if (handle->type != UV_UDP)
Packit Service e08953
    return 0;
Packit Service e08953
Packit Service e08953
  addrlen = sizeof(addr);
Packit Service e08953
  if (uv_udp_getpeername(handle, (struct sockaddr*) &addr, &addrlen) != 0)
Packit Service e08953
    return 0;
Packit Service e08953
Packit Service e08953
  return addrlen > 0;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit Service e08953
int uv__udp_check_before_send(uv_udp_t* handle, const struct sockaddr* addr) {
Packit Service e08953
  unsigned int addrlen;
Packit Service e08953
Packit Service e08953
  if (handle->type != UV_UDP)
Packit Service e08953
    return UV_EINVAL;
Packit Service e08953
Packit Service e08953
  if (addr != NULL && (handle->flags & UV_HANDLE_UDP_CONNECTED))
Packit Service e08953
    return UV_EISCONN;
Packit Service e08953
Packit Service e08953
  if (addr == NULL && !(handle->flags & UV_HANDLE_UDP_CONNECTED))
Packit Service e08953
    return UV_EDESTADDRREQ;
Packit Service e08953
Packit Service e08953
  if (addr != NULL) {
Packit Service e08953
    if (addr->sa_family == AF_INET)
Packit Service e08953
      addrlen = sizeof(struct sockaddr_in);
Packit Service e08953
    else if (addr->sa_family == AF_INET6)
Packit Service e08953
      addrlen = sizeof(struct sockaddr_in6);
Packit Service e08953
#if defined(AF_UNIX) && !defined(_WIN32)
Packit Service e08953
    else if (addr->sa_family == AF_UNIX)
Packit Service e08953
      addrlen = sizeof(struct sockaddr_un);
Packit Service e08953
#endif
Packit Service e08953
    else
Packit Service e08953
      return UV_EINVAL;
Packit Service e08953
  } else {
Packit Service e08953
    addrlen = 0;
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  return addrlen;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit Service e08953
int uv_udp_send(uv_udp_send_t* req,
Packit Service e08953
                uv_udp_t* handle,
Packit Service e08953
                const uv_buf_t bufs[],
Packit Service e08953
                unsigned int nbufs,
Packit Service e08953
                const struct sockaddr* addr,
Packit Service e08953
                uv_udp_send_cb send_cb) {
Packit Service e08953
  int addrlen;
Packit Service e08953
Packit Service e08953
  addrlen = uv__udp_check_before_send(handle, addr);
Packit Service e08953
  if (addrlen < 0)
Packit Service e08953
    return addrlen;
Packit Service e08953
Packit b5b901
  return uv__udp_send(req, handle, bufs, nbufs, addr, addrlen, send_cb);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_udp_try_send(uv_udp_t* handle,
Packit b5b901
                    const uv_buf_t bufs[],
Packit b5b901
                    unsigned int nbufs,
Packit b5b901
                    const struct sockaddr* addr) {
Packit Service e08953
  int addrlen;
Packit b5b901
Packit Service e08953
  addrlen = uv__udp_check_before_send(handle, addr);
Packit Service e08953
  if (addrlen < 0)
Packit Service e08953
    return addrlen;
Packit b5b901
Packit b5b901
  return uv__udp_try_send(handle, bufs, nbufs, addr, addrlen);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_udp_recv_start(uv_udp_t* handle,
Packit b5b901
                      uv_alloc_cb alloc_cb,
Packit b5b901
                      uv_udp_recv_cb recv_cb) {
Packit b5b901
  if (handle->type != UV_UDP || alloc_cb == NULL || recv_cb == NULL)
Packit b5b901
    return UV_EINVAL;
Packit b5b901
  else
Packit b5b901
    return uv__udp_recv_start(handle, alloc_cb, recv_cb);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_udp_recv_stop(uv_udp_t* handle) {
Packit b5b901
  if (handle->type != UV_UDP)
Packit b5b901
    return UV_EINVAL;
Packit b5b901
  else
Packit b5b901
    return uv__udp_recv_stop(handle);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
void uv_walk(uv_loop_t* loop, uv_walk_cb walk_cb, void* arg) {
Packit b5b901
  QUEUE queue;
Packit b5b901
  QUEUE* q;
Packit b5b901
  uv_handle_t* h;
Packit b5b901
Packit b5b901
  QUEUE_MOVE(&loop->handle_queue, &queue);
Packit b5b901
  while (!QUEUE_EMPTY(&queue)) {
Packit b5b901
    q = QUEUE_HEAD(&queue);
Packit b5b901
    h = QUEUE_DATA(q, uv_handle_t, handle_queue);
Packit b5b901
Packit b5b901
    QUEUE_REMOVE(q);
Packit b5b901
    QUEUE_INSERT_TAIL(&loop->handle_queue, q);
Packit b5b901
Packit b5b901
    if (h->flags & UV_HANDLE_INTERNAL) continue;
Packit b5b901
    walk_cb(h, arg);
Packit b5b901
  }
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static void uv__print_handles(uv_loop_t* loop, int only_active, FILE* stream) {
Packit b5b901
  const char* type;
Packit b5b901
  QUEUE* q;
Packit b5b901
  uv_handle_t* h;
Packit b5b901
Packit b5b901
  if (loop == NULL)
Packit b5b901
    loop = uv_default_loop();
Packit b5b901
Packit b5b901
  QUEUE_FOREACH(q, &loop->handle_queue) {
Packit b5b901
    h = QUEUE_DATA(q, uv_handle_t, handle_queue);
Packit b5b901
Packit b5b901
    if (only_active && !uv__is_active(h))
Packit b5b901
      continue;
Packit b5b901
Packit b5b901
    switch (h->type) {
Packit b5b901
#define X(uc, lc) case UV_##uc: type = #lc; break;
Packit b5b901
      UV_HANDLE_TYPE_MAP(X)
Packit b5b901
#undef X
Packit b5b901
      default: type = "<unknown>";
Packit b5b901
    }
Packit b5b901
Packit b5b901
    fprintf(stream,
Packit b5b901
            "[%c%c%c] %-8s %p\n",
Packit b5b901
            "R-"[!(h->flags & UV_HANDLE_REF)],
Packit b5b901
            "A-"[!(h->flags & UV_HANDLE_ACTIVE)],
Packit b5b901
            "I-"[!(h->flags & UV_HANDLE_INTERNAL)],
Packit b5b901
            type,
Packit b5b901
            (void*)h);
Packit b5b901
  }
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
void uv_print_all_handles(uv_loop_t* loop, FILE* stream) {
Packit b5b901
  uv__print_handles(loop, 0, stream);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
void uv_print_active_handles(uv_loop_t* loop, FILE* stream) {
Packit b5b901
  uv__print_handles(loop, 1, stream);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
void uv_ref(uv_handle_t* handle) {
Packit b5b901
  uv__handle_ref(handle);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
void uv_unref(uv_handle_t* handle) {
Packit b5b901
  uv__handle_unref(handle);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_has_ref(const uv_handle_t* handle) {
Packit b5b901
  return uv__has_ref(handle);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
void uv_stop(uv_loop_t* loop) {
Packit b5b901
  loop->stop_flag = 1;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
uint64_t uv_now(const uv_loop_t* loop) {
Packit b5b901
  return loop->time;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
Packit b5b901
size_t uv__count_bufs(const uv_buf_t bufs[], unsigned int nbufs) {
Packit b5b901
  unsigned int i;
Packit b5b901
  size_t bytes;
Packit b5b901
Packit b5b901
  bytes = 0;
Packit b5b901
  for (i = 0; i < nbufs; i++)
Packit b5b901
    bytes += (size_t) bufs[i].len;
Packit b5b901
Packit b5b901
  return bytes;
Packit b5b901
}
Packit b5b901
Packit b5b901
int uv_recv_buffer_size(uv_handle_t* handle, int* value) {
Packit b5b901
  return uv__socket_sockopt(handle, SO_RCVBUF, value);
Packit b5b901
}
Packit b5b901
Packit b5b901
int uv_send_buffer_size(uv_handle_t* handle, int *value) {
Packit b5b901
  return uv__socket_sockopt(handle, SO_SNDBUF, value);
Packit b5b901
}
Packit b5b901
Packit b5b901
int uv_fs_event_getpath(uv_fs_event_t* handle, char* buffer, size_t* size) {
Packit b5b901
  size_t required_len;
Packit b5b901
Packit b5b901
  if (!uv__is_active(handle)) {
Packit b5b901
    *size = 0;
Packit b5b901
    return UV_EINVAL;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  required_len = strlen(handle->path);
Packit b5b901
  if (required_len >= *size) {
Packit b5b901
    *size = required_len + 1;
Packit b5b901
    return UV_ENOBUFS;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  memcpy(buffer, handle->path, required_len);
Packit b5b901
  *size = required_len;
Packit b5b901
  buffer[required_len] = '\0';
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
/* The windows implementation does not have the same structure layout as
Packit b5b901
 * the unix implementation (nbufs is not directly inside req but is
Packit b5b901
 * contained in a nested union/struct) so this function locates it.
Packit b5b901
*/
Packit b5b901
static unsigned int* uv__get_nbufs(uv_fs_t* req) {
Packit b5b901
#ifdef _WIN32
Packit b5b901
  return &req->fs.info.nbufs;
Packit b5b901
#else
Packit b5b901
  return &req->nbufs;
Packit b5b901
#endif
Packit b5b901
}
Packit b5b901
Packit b5b901
/* uv_fs_scandir() uses the system allocator to allocate memory on non-Windows
Packit b5b901
 * systems. So, the memory should be released using free(). On Windows,
Packit b5b901
 * uv__malloc() is used, so use uv__free() to free memory.
Packit b5b901
*/
Packit b5b901
#ifdef _WIN32
Packit b5b901
# define uv__fs_scandir_free uv__free
Packit b5b901
#else
Packit b5b901
# define uv__fs_scandir_free free
Packit b5b901
#endif
Packit b5b901
Packit b5b901
void uv__fs_scandir_cleanup(uv_fs_t* req) {
Packit b5b901
  uv__dirent_t** dents;
Packit b5b901
Packit b5b901
  unsigned int* nbufs = uv__get_nbufs(req);
Packit b5b901
Packit b5b901
  dents = req->ptr;
Packit b5b901
  if (*nbufs > 0 && *nbufs != (unsigned int) req->result)
Packit b5b901
    (*nbufs)--;
Packit b5b901
  for (; *nbufs < (unsigned int) req->result; (*nbufs)++)
Packit b5b901
    uv__fs_scandir_free(dents[*nbufs]);
Packit b5b901
Packit b5b901
  uv__fs_scandir_free(req->ptr);
Packit b5b901
  req->ptr = NULL;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_fs_scandir_next(uv_fs_t* req, uv_dirent_t* ent) {
Packit b5b901
  uv__dirent_t** dents;
Packit b5b901
  uv__dirent_t* dent;
Packit b5b901
  unsigned int* nbufs;
Packit b5b901
Packit b5b901
  /* Check to see if req passed */
Packit b5b901
  if (req->result < 0)
Packit b5b901
    return req->result;
Packit b5b901
Packit b5b901
  /* Ptr will be null if req was canceled or no files found */
Packit b5b901
  if (!req->ptr)
Packit b5b901
    return UV_EOF;
Packit b5b901
Packit b5b901
  nbufs = uv__get_nbufs(req);
Packit b5b901
  assert(nbufs);
Packit b5b901
Packit b5b901
  dents = req->ptr;
Packit b5b901
Packit b5b901
  /* Free previous entity */
Packit b5b901
  if (*nbufs > 0)
Packit b5b901
    uv__fs_scandir_free(dents[*nbufs - 1]);
Packit b5b901
Packit b5b901
  /* End was already reached */
Packit b5b901
  if (*nbufs == (unsigned int) req->result) {
Packit b5b901
    uv__fs_scandir_free(dents);
Packit b5b901
    req->ptr = NULL;
Packit b5b901
    return UV_EOF;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  dent = dents[(*nbufs)++];
Packit b5b901
Packit b5b901
  ent->name = dent->d_name;
Packit Service e08953
  ent->type = uv__fs_get_dirent_type(dent);
Packit Service e08953
Packit Service e08953
  return 0;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
uv_dirent_type_t uv__fs_get_dirent_type(uv__dirent_t* dent) {
Packit Service e08953
  uv_dirent_type_t type;
Packit Service e08953
Packit b5b901
#ifdef HAVE_DIRENT_TYPES
Packit b5b901
  switch (dent->d_type) {
Packit b5b901
    case UV__DT_DIR:
Packit Service e08953
      type = UV_DIRENT_DIR;
Packit b5b901
      break;
Packit b5b901
    case UV__DT_FILE:
Packit Service e08953
      type = UV_DIRENT_FILE;
Packit b5b901
      break;
Packit b5b901
    case UV__DT_LINK:
Packit Service e08953
      type = UV_DIRENT_LINK;
Packit b5b901
      break;
Packit b5b901
    case UV__DT_FIFO:
Packit Service e08953
      type = UV_DIRENT_FIFO;
Packit b5b901
      break;
Packit b5b901
    case UV__DT_SOCKET:
Packit Service e08953
      type = UV_DIRENT_SOCKET;
Packit b5b901
      break;
Packit b5b901
    case UV__DT_CHAR:
Packit Service e08953
      type = UV_DIRENT_CHAR;
Packit b5b901
      break;
Packit b5b901
    case UV__DT_BLOCK:
Packit Service e08953
      type = UV_DIRENT_BLOCK;
Packit b5b901
      break;
Packit b5b901
    default:
Packit Service e08953
      type = UV_DIRENT_UNKNOWN;
Packit b5b901
  }
Packit b5b901
#else
Packit Service e08953
  type = UV_DIRENT_UNKNOWN;
Packit b5b901
#endif
Packit b5b901
Packit Service e08953
  return type;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
void uv__fs_readdir_cleanup(uv_fs_t* req) {
Packit Service e08953
  uv_dir_t* dir;
Packit Service e08953
  uv_dirent_t* dirents;
Packit Service e08953
  int i;
Packit Service e08953
Packit Service e08953
  if (req->ptr == NULL)
Packit Service e08953
    return;
Packit Service e08953
Packit Service e08953
  dir = req->ptr;
Packit Service e08953
  dirents = dir->dirents;
Packit Service e08953
  req->ptr = NULL;
Packit Service e08953
Packit Service e08953
  if (dirents == NULL)
Packit Service e08953
    return;
Packit Service e08953
Packit Service e08953
  for (i = 0; i < req->result; ++i) {
Packit Service e08953
    uv__free((char*) dirents[i].name);
Packit Service e08953
    dirents[i].name = NULL;
Packit Service e08953
  }
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_loop_configure(uv_loop_t* loop, uv_loop_option option, ...) {
Packit b5b901
  va_list ap;
Packit b5b901
  int err;
Packit b5b901
Packit b5b901
  va_start(ap, option);
Packit b5b901
  /* Any platform-agnostic options should be handled here. */
Packit b5b901
  err = uv__loop_configure(loop, option, ap);
Packit b5b901
  va_end(ap);
Packit b5b901
Packit b5b901
  return err;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static uv_loop_t default_loop_struct;
Packit b5b901
static uv_loop_t* default_loop_ptr;
Packit b5b901
Packit b5b901
Packit b5b901
uv_loop_t* uv_default_loop(void) {
Packit b5b901
  if (default_loop_ptr != NULL)
Packit b5b901
    return default_loop_ptr;
Packit b5b901
Packit b5b901
  if (uv_loop_init(&default_loop_struct))
Packit b5b901
    return NULL;
Packit b5b901
Packit b5b901
  default_loop_ptr = &default_loop_struct;
Packit b5b901
  return default_loop_ptr;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
uv_loop_t* uv_loop_new(void) {
Packit b5b901
  uv_loop_t* loop;
Packit b5b901
Packit b5b901
  loop = uv__malloc(sizeof(*loop));
Packit b5b901
  if (loop == NULL)
Packit b5b901
    return NULL;
Packit b5b901
Packit b5b901
  if (uv_loop_init(loop)) {
Packit b5b901
    uv__free(loop);
Packit b5b901
    return NULL;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return loop;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_loop_close(uv_loop_t* loop) {
Packit b5b901
  QUEUE* q;
Packit b5b901
  uv_handle_t* h;
Packit b5b901
#ifndef NDEBUG
Packit b5b901
  void* saved_data;
Packit b5b901
#endif
Packit b5b901
Packit b5b901
  if (uv__has_active_reqs(loop))
Packit b5b901
    return UV_EBUSY;
Packit b5b901
Packit b5b901
  QUEUE_FOREACH(q, &loop->handle_queue) {
Packit b5b901
    h = QUEUE_DATA(q, uv_handle_t, handle_queue);
Packit b5b901
    if (!(h->flags & UV_HANDLE_INTERNAL))
Packit b5b901
      return UV_EBUSY;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  uv__loop_close(loop);
Packit b5b901
Packit b5b901
#ifndef NDEBUG
Packit b5b901
  saved_data = loop->data;
Packit b5b901
  memset(loop, -1, sizeof(*loop));
Packit b5b901
  loop->data = saved_data;
Packit b5b901
#endif
Packit b5b901
  if (loop == default_loop_ptr)
Packit b5b901
    default_loop_ptr = NULL;
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
void uv_loop_delete(uv_loop_t* loop) {
Packit b5b901
  uv_loop_t* default_loop;
Packit b5b901
  int err;
Packit b5b901
Packit b5b901
  default_loop = default_loop_ptr;
Packit b5b901
Packit b5b901
  err = uv_loop_close(loop);
Packit b5b901
  (void) err;    /* Squelch compiler warnings. */
Packit b5b901
  assert(err == 0);
Packit b5b901
  if (loop != default_loop)
Packit b5b901
    uv__free(loop);
Packit b5b901
}
Packit Service e08953
Packit Service e08953
Packit Service e08953
void uv_os_free_environ(uv_env_item_t* envitems, int count) {
Packit Service e08953
  int i;
Packit Service e08953
Packit Service e08953
  for (i = 0; i < count; i++) {
Packit Service e08953
    uv__free(envitems[i].name);
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  uv__free(envitems);
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit Service e08953
void uv_free_cpu_info(uv_cpu_info_t* cpu_infos, int count) {
Packit Service e08953
  int i;
Packit Service e08953
Packit Service e08953
  for (i = 0; i < count; i++)
Packit Service e08953
    uv__free(cpu_infos[i].model);
Packit Service e08953
Packit Service e08953
  uv__free(cpu_infos);
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit Service e08953
#ifdef __GNUC__  /* Also covers __clang__ and __INTEL_COMPILER. */
Packit Service e08953
__attribute__((destructor))
Packit Service e08953
#endif
Packit Service e08953
void uv_library_shutdown(void) {
Packit Service e08953
  static int was_shutdown;
Packit Service e08953
Packit Service e08953
  if (was_shutdown)
Packit Service e08953
    return;
Packit Service e08953
Packit Service e08953
  uv__process_title_cleanup();
Packit Service e08953
  uv__signal_cleanup();
Packit Service e08953
  uv__threadpool_cleanup();
Packit Service e08953
  was_shutdown = 1;
Packit Service e08953
}