Blame src/unix/udp.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 "internal.h"
Packit b5b901
Packit b5b901
#include <assert.h>
Packit b5b901
#include <string.h>
Packit b5b901
#include <errno.h>
Packit b5b901
#include <stdlib.h>
Packit b5b901
#include <unistd.h>
Packit b5b901
#if defined(__MVS__)
Packit b5b901
#include <xti.h>
Packit b5b901
#endif
Packit Service e08953
#include <sys/un.h>
Packit Service e08953
Packit Service e08953
#define UV__UDP_DGRAM_MAXSIZE (64 * 1024)
Packit b5b901
Packit b5b901
#if defined(IPV6_JOIN_GROUP) && !defined(IPV6_ADD_MEMBERSHIP)
Packit b5b901
# define IPV6_ADD_MEMBERSHIP IPV6_JOIN_GROUP
Packit b5b901
#endif
Packit b5b901
Packit b5b901
#if defined(IPV6_LEAVE_GROUP) && !defined(IPV6_DROP_MEMBERSHIP)
Packit b5b901
# define IPV6_DROP_MEMBERSHIP IPV6_LEAVE_GROUP
Packit b5b901
#endif
Packit b5b901
Packit Service e08953
union uv__sockaddr {
Packit Service e08953
  struct sockaddr_in6 in6;
Packit Service e08953
  struct sockaddr_in in;
Packit Service e08953
  struct sockaddr addr;
Packit Service e08953
};
Packit b5b901
Packit b5b901
static void uv__udp_run_completed(uv_udp_t* handle);
Packit b5b901
static void uv__udp_io(uv_loop_t* loop, uv__io_t* w, unsigned int revents);
Packit b5b901
static void uv__udp_recvmsg(uv_udp_t* handle);
Packit b5b901
static void uv__udp_sendmsg(uv_udp_t* handle);
Packit b5b901
static int uv__udp_maybe_deferred_bind(uv_udp_t* handle,
Packit b5b901
                                       int domain,
Packit b5b901
                                       unsigned int flags);
Packit b5b901
Packit Service e08953
#if HAVE_MMSG
Packit Service e08953
Packit Service e08953
#define UV__MMSG_MAXWIDTH 20
Packit Service e08953
Packit Service e08953
static int uv__udp_recvmmsg(uv_udp_t* handle, uv_buf_t* buf);
Packit Service e08953
static void uv__udp_sendmmsg(uv_udp_t* handle);
Packit Service e08953
Packit Service e08953
static int uv__recvmmsg_avail;
Packit Service e08953
static int uv__sendmmsg_avail;
Packit Service e08953
static uv_once_t once = UV_ONCE_INIT;
Packit Service e08953
Packit Service e08953
static void uv__udp_mmsg_init(void) {
Packit Service e08953
  int ret;
Packit Service e08953
  int s;
Packit Service e08953
  s = uv__socket(AF_INET, SOCK_DGRAM, 0);
Packit Service e08953
  if (s < 0)
Packit Service e08953
    return;
Packit Service e08953
  ret = uv__sendmmsg(s, NULL, 0, 0);
Packit Service e08953
  if (ret == 0 || errno != ENOSYS) {
Packit Service e08953
    uv__sendmmsg_avail = 1;
Packit Service e08953
    uv__recvmmsg_avail = 1;
Packit Service e08953
  } else {
Packit Service e08953
    ret = uv__recvmmsg(s, NULL, 0, 0, NULL);
Packit Service e08953
    if (ret == 0 || errno != ENOSYS)
Packit Service e08953
      uv__recvmmsg_avail = 1;
Packit Service e08953
  }
Packit Service e08953
  uv__close(s);
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
#endif
Packit b5b901
Packit b5b901
void uv__udp_close(uv_udp_t* handle) {
Packit b5b901
  uv__io_close(handle->loop, &handle->io_watcher);
Packit b5b901
  uv__handle_stop(handle);
Packit b5b901
Packit b5b901
  if (handle->io_watcher.fd != -1) {
Packit b5b901
    uv__close(handle->io_watcher.fd);
Packit b5b901
    handle->io_watcher.fd = -1;
Packit b5b901
  }
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
void uv__udp_finish_close(uv_udp_t* handle) {
Packit b5b901
  uv_udp_send_t* req;
Packit b5b901
  QUEUE* q;
Packit b5b901
Packit b5b901
  assert(!uv__io_active(&handle->io_watcher, POLLIN | POLLOUT));
Packit b5b901
  assert(handle->io_watcher.fd == -1);
Packit b5b901
Packit b5b901
  while (!QUEUE_EMPTY(&handle->write_queue)) {
Packit b5b901
    q = QUEUE_HEAD(&handle->write_queue);
Packit b5b901
    QUEUE_REMOVE(q);
Packit b5b901
Packit b5b901
    req = QUEUE_DATA(q, uv_udp_send_t, queue);
Packit b5b901
    req->status = UV_ECANCELED;
Packit b5b901
    QUEUE_INSERT_TAIL(&handle->write_completed_queue, &req->queue);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  uv__udp_run_completed(handle);
Packit b5b901
Packit b5b901
  assert(handle->send_queue_size == 0);
Packit b5b901
  assert(handle->send_queue_count == 0);
Packit b5b901
Packit b5b901
  /* Now tear down the handle. */
Packit b5b901
  handle->recv_cb = NULL;
Packit b5b901
  handle->alloc_cb = NULL;
Packit b5b901
  /* but _do not_ touch close_cb */
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static void uv__udp_run_completed(uv_udp_t* handle) {
Packit b5b901
  uv_udp_send_t* req;
Packit b5b901
  QUEUE* q;
Packit b5b901
Packit b5b901
  assert(!(handle->flags & UV_HANDLE_UDP_PROCESSING));
Packit b5b901
  handle->flags |= UV_HANDLE_UDP_PROCESSING;
Packit b5b901
Packit b5b901
  while (!QUEUE_EMPTY(&handle->write_completed_queue)) {
Packit b5b901
    q = QUEUE_HEAD(&handle->write_completed_queue);
Packit b5b901
    QUEUE_REMOVE(q);
Packit b5b901
Packit b5b901
    req = QUEUE_DATA(q, uv_udp_send_t, queue);
Packit b5b901
    uv__req_unregister(handle->loop, req);
Packit b5b901
Packit b5b901
    handle->send_queue_size -= uv__count_bufs(req->bufs, req->nbufs);
Packit b5b901
    handle->send_queue_count--;
Packit b5b901
Packit b5b901
    if (req->bufs != req->bufsml)
Packit b5b901
      uv__free(req->bufs);
Packit b5b901
    req->bufs = NULL;
Packit b5b901
Packit b5b901
    if (req->send_cb == NULL)
Packit b5b901
      continue;
Packit b5b901
Packit b5b901
    /* req->status >= 0 == bytes written
Packit b5b901
     * req->status <  0 == errno
Packit b5b901
     */
Packit b5b901
    if (req->status >= 0)
Packit b5b901
      req->send_cb(req, 0);
Packit b5b901
    else
Packit b5b901
      req->send_cb(req, req->status);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (QUEUE_EMPTY(&handle->write_queue)) {
Packit b5b901
    /* Pending queue and completion queue empty, stop watcher. */
Packit b5b901
    uv__io_stop(handle->loop, &handle->io_watcher, POLLOUT);
Packit b5b901
    if (!uv__io_active(&handle->io_watcher, POLLIN))
Packit b5b901
      uv__handle_stop(handle);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  handle->flags &= ~UV_HANDLE_UDP_PROCESSING;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static void uv__udp_io(uv_loop_t* loop, uv__io_t* w, unsigned int revents) {
Packit b5b901
  uv_udp_t* handle;
Packit b5b901
Packit b5b901
  handle = container_of(w, uv_udp_t, io_watcher);
Packit b5b901
  assert(handle->type == UV_UDP);
Packit b5b901
Packit b5b901
  if (revents & POLLIN)
Packit b5b901
    uv__udp_recvmsg(handle);
Packit b5b901
Packit b5b901
  if (revents & POLLOUT) {
Packit b5b901
    uv__udp_sendmsg(handle);
Packit b5b901
    uv__udp_run_completed(handle);
Packit b5b901
  }
Packit b5b901
}
Packit b5b901
Packit Service e08953
#if HAVE_MMSG
Packit Service e08953
static int uv__udp_recvmmsg(uv_udp_t* handle, uv_buf_t* buf) {
Packit Service e08953
  struct sockaddr_in6 peers[UV__MMSG_MAXWIDTH];
Packit Service e08953
  struct iovec iov[UV__MMSG_MAXWIDTH];
Packit Service e08953
  struct uv__mmsghdr msgs[UV__MMSG_MAXWIDTH];
Packit Service e08953
  ssize_t nread;
Packit Service e08953
  uv_buf_t chunk_buf;
Packit Service e08953
  size_t chunks;
Packit Service e08953
  int flags;
Packit Service e08953
  size_t k;
Packit Service e08953
Packit Service e08953
  /* prepare structures for recvmmsg */
Packit Service e08953
  chunks = buf->len / UV__UDP_DGRAM_MAXSIZE;
Packit Service e08953
  if (chunks > ARRAY_SIZE(iov))
Packit Service e08953
    chunks = ARRAY_SIZE(iov);
Packit Service e08953
  for (k = 0; k < chunks; ++k) {
Packit Service e08953
    iov[k].iov_base = buf->base + k * UV__UDP_DGRAM_MAXSIZE;
Packit Service e08953
    iov[k].iov_len = UV__UDP_DGRAM_MAXSIZE;
Packit Service e08953
    msgs[k].msg_hdr.msg_iov = iov + k;
Packit Service e08953
    msgs[k].msg_hdr.msg_iovlen = 1;
Packit Service e08953
    msgs[k].msg_hdr.msg_name = peers + k;
Packit Service e08953
    msgs[k].msg_hdr.msg_namelen = sizeof(peers[0]);
Packit Service e08953
    msgs[k].msg_hdr.msg_control = NULL;
Packit Service e08953
    msgs[k].msg_hdr.msg_controllen = 0;
Packit Service e08953
    msgs[k].msg_hdr.msg_flags = 0;
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  do
Packit Service e08953
    nread = uv__recvmmsg(handle->io_watcher.fd, msgs, chunks, 0, NULL);
Packit Service e08953
  while (nread == -1 && errno == EINTR);
Packit Service e08953
Packit Service e08953
  if (nread < 1) {
Packit Service e08953
    if (nread == 0 || errno == EAGAIN || errno == EWOULDBLOCK)
Packit Service e08953
      handle->recv_cb(handle, 0, buf, NULL, 0);
Packit Service e08953
    else
Packit Service e08953
      handle->recv_cb(handle, UV__ERR(errno), buf, NULL, 0);
Packit Service e08953
  } else {
Packit Service e08953
    /* pass each chunk to the application */
Packit Service e08953
    for (k = 0; k < (size_t) nread && handle->recv_cb != NULL; k++) {
Packit Service e08953
      flags = UV_UDP_MMSG_CHUNK;
Packit Service e08953
      if (msgs[k].msg_hdr.msg_flags & MSG_TRUNC)
Packit Service e08953
        flags |= UV_UDP_PARTIAL;
Packit Service e08953
Packit Service e08953
      chunk_buf = uv_buf_init(iov[k].iov_base, iov[k].iov_len);
Packit Service e08953
      handle->recv_cb(handle,
Packit Service e08953
                      msgs[k].msg_len,
Packit Service e08953
                      &chunk_buf,
Packit Service e08953
                      msgs[k].msg_hdr.msg_name,
Packit Service e08953
                      flags);
Packit Service e08953
    }
Packit Service e08953
Packit Service e08953
    /* one last callback so the original buffer is freed */
Packit Service e08953
    if (handle->recv_cb != NULL)
Packit Service e08953
      handle->recv_cb(handle, 0, buf, NULL, 0);
Packit Service e08953
  }
Packit Service e08953
  return nread;
Packit Service e08953
}
Packit Service e08953
#endif
Packit b5b901
Packit b5b901
static void uv__udp_recvmsg(uv_udp_t* handle) {
Packit b5b901
  struct sockaddr_storage peer;
Packit b5b901
  struct msghdr h;
Packit b5b901
  ssize_t nread;
Packit b5b901
  uv_buf_t buf;
Packit b5b901
  int flags;
Packit b5b901
  int count;
Packit b5b901
Packit b5b901
  assert(handle->recv_cb != NULL);
Packit b5b901
  assert(handle->alloc_cb != NULL);
Packit b5b901
Packit b5b901
  /* Prevent loop starvation when the data comes in as fast as (or faster than)
Packit b5b901
   * we can read it. XXX Need to rearm fd if we switch to edge-triggered I/O.
Packit b5b901
   */
Packit b5b901
  count = 32;
Packit b5b901
Packit b5b901
  do {
Packit b5b901
    buf = uv_buf_init(NULL, 0);
Packit Service e08953
    handle->alloc_cb((uv_handle_t*) handle, UV__UDP_DGRAM_MAXSIZE, &buf;;
Packit b5b901
    if (buf.base == NULL || buf.len == 0) {
Packit b5b901
      handle->recv_cb(handle, UV_ENOBUFS, &buf, NULL, 0);
Packit b5b901
      return;
Packit b5b901
    }
Packit b5b901
    assert(buf.base != NULL);
Packit b5b901
Packit Service e08953
#if HAVE_MMSG
Packit Service e08953
    if (handle->flags & UV_HANDLE_UDP_RECVMMSG) {
Packit Service e08953
      uv_once(&once, uv__udp_mmsg_init);
Packit Service e08953
      if (uv__recvmmsg_avail) {
Packit Service e08953
        nread = uv__udp_recvmmsg(handle, &buf;;
Packit Service e08953
        if (nread > 0)
Packit Service e08953
          count -= nread;
Packit Service e08953
        continue;
Packit Service e08953
      }
Packit Service e08953
    }
Packit Service e08953
#endif
Packit Service e08953
Packit Service e08953
    memset(&h, 0, sizeof(h));
Packit Service e08953
    memset(&peer, 0, sizeof(peer));
Packit Service e08953
    h.msg_name = &pee;;
Packit b5b901
    h.msg_namelen = sizeof(peer);
Packit b5b901
    h.msg_iov = (void*) &buf;
Packit b5b901
    h.msg_iovlen = 1;
Packit b5b901
Packit b5b901
    do {
Packit b5b901
      nread = recvmsg(handle->io_watcher.fd, &h, 0);
Packit b5b901
    }
Packit b5b901
    while (nread == -1 && errno == EINTR);
Packit b5b901
Packit b5b901
    if (nread == -1) {
Packit b5b901
      if (errno == EAGAIN || errno == EWOULDBLOCK)
Packit b5b901
        handle->recv_cb(handle, 0, &buf, NULL, 0);
Packit b5b901
      else
Packit b5b901
        handle->recv_cb(handle, UV__ERR(errno), &buf, NULL, 0);
Packit b5b901
    }
Packit b5b901
    else {
Packit b5b901
      flags = 0;
Packit b5b901
      if (h.msg_flags & MSG_TRUNC)
Packit b5b901
        flags |= UV_UDP_PARTIAL;
Packit b5b901
Packit Service e08953
      handle->recv_cb(handle, nread, &buf, (const struct sockaddr*) &peer, flags);
Packit b5b901
    }
Packit Service e08953
    count--;
Packit b5b901
  }
Packit b5b901
  /* recv_cb callback may decide to pause or close the handle */
Packit b5b901
  while (nread != -1
Packit Service e08953
      && count > 0
Packit b5b901
      && handle->io_watcher.fd != -1
Packit b5b901
      && handle->recv_cb != NULL);
Packit b5b901
}
Packit b5b901
Packit Service e08953
#if HAVE_MMSG
Packit Service e08953
static void uv__udp_sendmmsg(uv_udp_t* handle) {
Packit Service e08953
  uv_udp_send_t* req;
Packit Service e08953
  struct uv__mmsghdr h[UV__MMSG_MAXWIDTH];
Packit Service e08953
  struct uv__mmsghdr *p;
Packit Service e08953
  QUEUE* q;
Packit Service e08953
  ssize_t npkts;
Packit Service e08953
  size_t pkts;
Packit Service e08953
  size_t i;
Packit Service e08953
Packit Service e08953
  if (QUEUE_EMPTY(&handle->write_queue))
Packit Service e08953
    return;
Packit Service e08953
Packit Service e08953
write_queue_drain:
Packit Service e08953
  for (pkts = 0, q = QUEUE_HEAD(&handle->write_queue);
Packit Service e08953
       pkts < UV__MMSG_MAXWIDTH && q != &handle->write_queue;
Packit Service e08953
       ++pkts, q = QUEUE_HEAD(q)) {
Packit Service e08953
    assert(q != NULL);
Packit Service e08953
    req = QUEUE_DATA(q, uv_udp_send_t, queue);
Packit Service e08953
    assert(req != NULL);
Packit Service e08953
Packit Service e08953
    p = &h[pkts];
Packit Service e08953
    memset(p, 0, sizeof(*p));
Packit Service e08953
    if (req->addr.ss_family == AF_UNSPEC) {
Packit Service e08953
      p->msg_hdr.msg_name = NULL;
Packit Service e08953
      p->msg_hdr.msg_namelen = 0;
Packit Service e08953
    } else {
Packit Service e08953
      p->msg_hdr.msg_name = &req->addr;
Packit Service e08953
      if (req->addr.ss_family == AF_INET6)
Packit Service e08953
        p->msg_hdr.msg_namelen = sizeof(struct sockaddr_in6);
Packit Service e08953
      else if (req->addr.ss_family == AF_INET)
Packit Service e08953
        p->msg_hdr.msg_namelen = sizeof(struct sockaddr_in);
Packit Service e08953
      else if (req->addr.ss_family == AF_UNIX)
Packit Service e08953
        p->msg_hdr.msg_namelen = sizeof(struct sockaddr_un);
Packit Service e08953
      else {
Packit Service e08953
        assert(0 && "unsupported address family");
Packit Service e08953
        abort();
Packit Service e08953
      }
Packit Service e08953
    }
Packit Service e08953
    h[pkts].msg_hdr.msg_iov = (struct iovec*) req->bufs;
Packit Service e08953
    h[pkts].msg_hdr.msg_iovlen = req->nbufs;
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  do
Packit Service e08953
    npkts = uv__sendmmsg(handle->io_watcher.fd, h, pkts, 0);
Packit Service e08953
  while (npkts == -1 && errno == EINTR);
Packit Service e08953
Packit Service e08953
  if (npkts < 1) {
Packit Service e08953
    if (errno == EAGAIN || errno == EWOULDBLOCK || errno == ENOBUFS)
Packit Service e08953
      return;
Packit Service e08953
    for (i = 0, q = QUEUE_HEAD(&handle->write_queue);
Packit Service e08953
         i < pkts && q != &handle->write_queue;
Packit Service e08953
         ++i, q = QUEUE_HEAD(q)) {
Packit Service e08953
      assert(q != NULL);
Packit Service e08953
      req = QUEUE_DATA(q, uv_udp_send_t, queue);
Packit Service e08953
      assert(req != NULL);
Packit Service e08953
Packit Service e08953
      req->status = UV__ERR(errno);
Packit Service e08953
      QUEUE_REMOVE(&req->queue);
Packit Service e08953
      QUEUE_INSERT_TAIL(&handle->write_completed_queue, &req->queue);
Packit Service e08953
    }
Packit Service e08953
    uv__io_feed(handle->loop, &handle->io_watcher);
Packit Service e08953
    return;
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  for (i = 0, q = QUEUE_HEAD(&handle->write_queue);
Packit Service e08953
       i < pkts && q != &handle->write_queue;
Packit Service e08953
       ++i, q = QUEUE_HEAD(&handle->write_queue)) {
Packit Service e08953
    assert(q != NULL);
Packit Service e08953
    req = QUEUE_DATA(q, uv_udp_send_t, queue);
Packit Service e08953
    assert(req != NULL);
Packit Service e08953
Packit Service e08953
    req->status = req->bufs[0].len;
Packit Service e08953
Packit Service e08953
    /* Sending a datagram is an atomic operation: either all data
Packit Service e08953
     * is written or nothing is (and EMSGSIZE is raised). That is
Packit Service e08953
     * why we don't handle partial writes. Just pop the request
Packit Service e08953
     * off the write queue and onto the completed queue, done.
Packit Service e08953
     */
Packit Service e08953
    QUEUE_REMOVE(&req->queue);
Packit Service e08953
    QUEUE_INSERT_TAIL(&handle->write_completed_queue, &req->queue);
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  /* couldn't batch everything, continue sending (jump to avoid stack growth) */
Packit Service e08953
  if (!QUEUE_EMPTY(&handle->write_queue))
Packit Service e08953
    goto write_queue_drain;
Packit Service e08953
  uv__io_feed(handle->loop, &handle->io_watcher);
Packit Service e08953
  return;
Packit Service e08953
}
Packit Service e08953
#endif
Packit b5b901
Packit b5b901
static void uv__udp_sendmsg(uv_udp_t* handle) {
Packit b5b901
  uv_udp_send_t* req;
Packit b5b901
  struct msghdr h;
Packit Service e08953
  QUEUE* q;
Packit b5b901
  ssize_t size;
Packit b5b901
Packit Service e08953
#if HAVE_MMSG
Packit Service e08953
  uv_once(&once, uv__udp_mmsg_init);
Packit Service e08953
  if (uv__sendmmsg_avail) {
Packit Service e08953
    uv__udp_sendmmsg(handle);
Packit Service e08953
    return;
Packit Service e08953
  }
Packit Service e08953
#endif
Packit Service e08953
Packit b5b901
  while (!QUEUE_EMPTY(&handle->write_queue)) {
Packit b5b901
    q = QUEUE_HEAD(&handle->write_queue);
Packit b5b901
    assert(q != NULL);
Packit b5b901
Packit b5b901
    req = QUEUE_DATA(q, uv_udp_send_t, queue);
Packit b5b901
    assert(req != NULL);
Packit b5b901
Packit b5b901
    memset(&h, 0, sizeof h);
Packit Service e08953
    if (req->addr.ss_family == AF_UNSPEC) {
Packit Service e08953
      h.msg_name = NULL;
Packit Service e08953
      h.msg_namelen = 0;
Packit Service e08953
    } else {
Packit Service e08953
      h.msg_name = &req->addr;
Packit Service e08953
      if (req->addr.ss_family == AF_INET6)
Packit Service e08953
        h.msg_namelen = sizeof(struct sockaddr_in6);
Packit Service e08953
      else if (req->addr.ss_family == AF_INET)
Packit Service e08953
        h.msg_namelen = sizeof(struct sockaddr_in);
Packit Service e08953
      else if (req->addr.ss_family == AF_UNIX)
Packit Service e08953
        h.msg_namelen = sizeof(struct sockaddr_un);
Packit Service e08953
      else {
Packit Service e08953
        assert(0 && "unsupported address family");
Packit Service e08953
        abort();
Packit Service e08953
      }
Packit Service e08953
    }
Packit b5b901
    h.msg_iov = (struct iovec*) req->bufs;
Packit b5b901
    h.msg_iovlen = req->nbufs;
Packit b5b901
Packit b5b901
    do {
Packit b5b901
      size = sendmsg(handle->io_watcher.fd, &h, 0);
Packit b5b901
    } while (size == -1 && errno == EINTR);
Packit b5b901
Packit b5b901
    if (size == -1) {
Packit b5b901
      if (errno == EAGAIN || errno == EWOULDBLOCK || errno == ENOBUFS)
Packit b5b901
        break;
Packit b5b901
    }
Packit b5b901
Packit b5b901
    req->status = (size == -1 ? UV__ERR(errno) : size);
Packit b5b901
Packit b5b901
    /* Sending a datagram is an atomic operation: either all data
Packit b5b901
     * is written or nothing is (and EMSGSIZE is raised). That is
Packit b5b901
     * why we don't handle partial writes. Just pop the request
Packit b5b901
     * off the write queue and onto the completed queue, done.
Packit b5b901
     */
Packit b5b901
    QUEUE_REMOVE(&req->queue);
Packit b5b901
    QUEUE_INSERT_TAIL(&handle->write_completed_queue, &req->queue);
Packit b5b901
    uv__io_feed(handle->loop, &handle->io_watcher);
Packit b5b901
  }
Packit b5b901
}
Packit b5b901
Packit b5b901
/* On the BSDs, SO_REUSEPORT implies SO_REUSEADDR but with some additional
Packit b5b901
 * refinements for programs that use multicast.
Packit b5b901
 *
Packit b5b901
 * Linux as of 3.9 has a SO_REUSEPORT socket option but with semantics that
Packit b5b901
 * are different from the BSDs: it _shares_ the port rather than steal it
Packit b5b901
 * from the current listener.  While useful, it's not something we can emulate
Packit b5b901
 * on other platforms so we don't enable it.
Packit Service e08953
 *
Packit Service e08953
 * zOS does not support getsockname with SO_REUSEPORT option when using
Packit Service e08953
 * AF_UNIX.
Packit b5b901
 */
Packit b5b901
static int uv__set_reuse(int fd) {
Packit b5b901
  int yes;
Packit b5b901
  yes = 1;
Packit Service e08953
Packit Service e08953
#if defined(SO_REUSEPORT) && defined(__MVS__)
Packit Service e08953
  struct sockaddr_in sockfd;
Packit Service e08953
  unsigned int sockfd_len = sizeof(sockfd);
Packit Service e08953
  if (getsockname(fd, (struct sockaddr*) &sockfd, &sockfd_len) == -1)
Packit Service e08953
      return UV__ERR(errno);
Packit Service e08953
  if (sockfd.sin_family == AF_UNIX) {
Packit Service e08953
    if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(yes)))
Packit Service e08953
      return UV__ERR(errno);
Packit Service e08953
  } else {
Packit Service e08953
    if (setsockopt(fd, SOL_SOCKET, SO_REUSEPORT, &yes, sizeof(yes)))
Packit Service e08953
       return UV__ERR(errno);
Packit Service e08953
  }
Packit Service e08953
#elif defined(SO_REUSEPORT) && !defined(__linux__)
Packit b5b901
  if (setsockopt(fd, SOL_SOCKET, SO_REUSEPORT, &yes, sizeof(yes)))
Packit b5b901
    return UV__ERR(errno);
Packit b5b901
#else
Packit b5b901
  if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(yes)))
Packit b5b901
    return UV__ERR(errno);
Packit b5b901
#endif
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv__udp_bind(uv_udp_t* handle,
Packit b5b901
                 const struct sockaddr* addr,
Packit b5b901
                 unsigned int addrlen,
Packit b5b901
                 unsigned int flags) {
Packit b5b901
  int err;
Packit b5b901
  int yes;
Packit b5b901
  int fd;
Packit b5b901
Packit b5b901
  /* Check for bad flags. */
Packit b5b901
  if (flags & ~(UV_UDP_IPV6ONLY | UV_UDP_REUSEADDR))
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit b5b901
  /* Cannot set IPv6-only mode on non-IPv6 socket. */
Packit b5b901
  if ((flags & UV_UDP_IPV6ONLY) && addr->sa_family != AF_INET6)
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit b5b901
  fd = handle->io_watcher.fd;
Packit b5b901
  if (fd == -1) {
Packit b5b901
    err = uv__socket(addr->sa_family, SOCK_DGRAM, 0);
Packit b5b901
    if (err < 0)
Packit b5b901
      return err;
Packit b5b901
    fd = err;
Packit b5b901
    handle->io_watcher.fd = fd;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (flags & UV_UDP_REUSEADDR) {
Packit b5b901
    err = uv__set_reuse(fd);
Packit b5b901
    if (err)
Packit b5b901
      return err;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (flags & UV_UDP_IPV6ONLY) {
Packit b5b901
#ifdef IPV6_V6ONLY
Packit b5b901
    yes = 1;
Packit b5b901
    if (setsockopt(fd, IPPROTO_IPV6, IPV6_V6ONLY, &yes, sizeof yes) == -1) {
Packit b5b901
      err = UV__ERR(errno);
Packit b5b901
      return err;
Packit b5b901
    }
Packit b5b901
#else
Packit b5b901
    err = UV_ENOTSUP;
Packit b5b901
    return err;
Packit b5b901
#endif
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (bind(fd, addr, addrlen)) {
Packit b5b901
    err = UV__ERR(errno);
Packit b5b901
    if (errno == EAFNOSUPPORT)
Packit b5b901
      /* OSX, other BSDs and SunoS fail with EAFNOSUPPORT when binding a
Packit b5b901
       * socket created with AF_INET to an AF_INET6 address or vice versa. */
Packit b5b901
      err = UV_EINVAL;
Packit b5b901
    return err;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (addr->sa_family == AF_INET6)
Packit b5b901
    handle->flags |= UV_HANDLE_IPV6;
Packit b5b901
Packit b5b901
  handle->flags |= UV_HANDLE_BOUND;
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static int uv__udp_maybe_deferred_bind(uv_udp_t* handle,
Packit b5b901
                                       int domain,
Packit b5b901
                                       unsigned int flags) {
Packit Service e08953
  union uv__sockaddr taddr;
Packit b5b901
  socklen_t addrlen;
Packit b5b901
Packit b5b901
  if (handle->io_watcher.fd != -1)
Packit b5b901
    return 0;
Packit b5b901
Packit b5b901
  switch (domain) {
Packit b5b901
  case AF_INET:
Packit b5b901
  {
Packit b5b901
    struct sockaddr_in* addr = &taddr.in;
Packit b5b901
    memset(addr, 0, sizeof *addr);
Packit b5b901
    addr->sin_family = AF_INET;
Packit b5b901
    addr->sin_addr.s_addr = INADDR_ANY;
Packit b5b901
    addrlen = sizeof *addr;
Packit b5b901
    break;
Packit b5b901
  }
Packit b5b901
  case AF_INET6:
Packit b5b901
  {
Packit b5b901
    struct sockaddr_in6* addr = &taddr.in6;
Packit b5b901
    memset(addr, 0, sizeof *addr);
Packit b5b901
    addr->sin6_family = AF_INET6;
Packit b5b901
    addr->sin6_addr = in6addr_any;
Packit b5b901
    addrlen = sizeof *addr;
Packit b5b901
    break;
Packit b5b901
  }
Packit b5b901
  default:
Packit b5b901
    assert(0 && "unsupported address family");
Packit b5b901
    abort();
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return uv__udp_bind(handle, &taddr.addr, addrlen, flags);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit Service e08953
int uv__udp_connect(uv_udp_t* handle,
Packit Service e08953
                    const struct sockaddr* addr,
Packit Service e08953
                    unsigned int addrlen) {
Packit Service e08953
  int err;
Packit Service e08953
Packit Service e08953
  err = uv__udp_maybe_deferred_bind(handle, addr->sa_family, 0);
Packit Service e08953
  if (err)
Packit Service e08953
    return err;
Packit Service e08953
Packit Service e08953
  do {
Packit Service e08953
    errno = 0;
Packit Service e08953
    err = connect(handle->io_watcher.fd, addr, addrlen);
Packit Service e08953
  } while (err == -1 && errno == EINTR);
Packit Service e08953
Packit Service e08953
  if (err)
Packit Service e08953
    return UV__ERR(errno);
Packit Service e08953
Packit Service e08953
  handle->flags |= UV_HANDLE_UDP_CONNECTED;
Packit Service e08953
Packit Service e08953
  return 0;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit Service e08953
int uv__udp_disconnect(uv_udp_t* handle) {
Packit Service e08953
    int r;
Packit Service e08953
    struct sockaddr addr;
Packit Service e08953
Packit Service e08953
    memset(&addr, 0, sizeof(addr));
Packit Service e08953
Packit Service e08953
    addr.sa_family = AF_UNSPEC;
Packit Service e08953
Packit Service e08953
    do {
Packit Service e08953
      errno = 0;
Packit Service e08953
      r = connect(handle->io_watcher.fd, &addr, sizeof(addr));
Packit Service e08953
    } while (r == -1 && errno == EINTR);
Packit Service e08953
Packit Service e08953
    if (r == -1 && errno != EAFNOSUPPORT)
Packit Service e08953
      return UV__ERR(errno);
Packit Service e08953
Packit Service e08953
    handle->flags &= ~UV_HANDLE_UDP_CONNECTED;
Packit Service e08953
    return 0;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit b5b901
int uv__udp_send(uv_udp_send_t* req,
Packit b5b901
                 uv_udp_t* handle,
Packit b5b901
                 const uv_buf_t bufs[],
Packit b5b901
                 unsigned int nbufs,
Packit b5b901
                 const struct sockaddr* addr,
Packit b5b901
                 unsigned int addrlen,
Packit b5b901
                 uv_udp_send_cb send_cb) {
Packit b5b901
  int err;
Packit b5b901
  int empty_queue;
Packit b5b901
Packit b5b901
  assert(nbufs > 0);
Packit b5b901
Packit Service e08953
  if (addr) {
Packit Service e08953
    err = uv__udp_maybe_deferred_bind(handle, addr->sa_family, 0);
Packit Service e08953
    if (err)
Packit Service e08953
      return err;
Packit Service e08953
  }
Packit b5b901
Packit b5b901
  /* It's legal for send_queue_count > 0 even when the write_queue is empty;
Packit b5b901
   * it means there are error-state requests in the write_completed_queue that
Packit b5b901
   * will touch up send_queue_size/count later.
Packit b5b901
   */
Packit b5b901
  empty_queue = (handle->send_queue_count == 0);
Packit b5b901
Packit b5b901
  uv__req_init(handle->loop, req, UV_UDP_SEND);
Packit b5b901
  assert(addrlen <= sizeof(req->addr));
Packit Service e08953
  if (addr == NULL)
Packit Service e08953
    req->addr.ss_family = AF_UNSPEC;
Packit Service e08953
  else
Packit Service e08953
    memcpy(&req->addr, addr, addrlen);
Packit b5b901
  req->send_cb = send_cb;
Packit b5b901
  req->handle = handle;
Packit b5b901
  req->nbufs = nbufs;
Packit b5b901
Packit b5b901
  req->bufs = req->bufsml;
Packit b5b901
  if (nbufs > ARRAY_SIZE(req->bufsml))
Packit b5b901
    req->bufs = uv__malloc(nbufs * sizeof(bufs[0]));
Packit b5b901
Packit b5b901
  if (req->bufs == NULL) {
Packit b5b901
    uv__req_unregister(handle->loop, req);
Packit b5b901
    return UV_ENOMEM;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  memcpy(req->bufs, bufs, nbufs * sizeof(bufs[0]));
Packit b5b901
  handle->send_queue_size += uv__count_bufs(req->bufs, req->nbufs);
Packit b5b901
  handle->send_queue_count++;
Packit b5b901
  QUEUE_INSERT_TAIL(&handle->write_queue, &req->queue);
Packit b5b901
  uv__handle_start(handle);
Packit b5b901
Packit b5b901
  if (empty_queue && !(handle->flags & UV_HANDLE_UDP_PROCESSING)) {
Packit b5b901
    uv__udp_sendmsg(handle);
Packit b5b901
Packit b5b901
    /* `uv__udp_sendmsg` may not be able to do non-blocking write straight
Packit b5b901
     * away. In such cases the `io_watcher` has to be queued for asynchronous
Packit b5b901
     * write.
Packit b5b901
     */
Packit b5b901
    if (!QUEUE_EMPTY(&handle->write_queue))
Packit b5b901
      uv__io_start(handle->loop, &handle->io_watcher, POLLOUT);
Packit b5b901
  } else {
Packit b5b901
    uv__io_start(handle->loop, &handle->io_watcher, POLLOUT);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return 0;
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 b5b901
                     unsigned int addrlen) {
Packit b5b901
  int err;
Packit b5b901
  struct msghdr h;
Packit b5b901
  ssize_t size;
Packit b5b901
Packit b5b901
  assert(nbufs > 0);
Packit b5b901
Packit b5b901
  /* already sending a message */
Packit b5b901
  if (handle->send_queue_count != 0)
Packit b5b901
    return UV_EAGAIN;
Packit b5b901
Packit Service e08953
  if (addr) {
Packit Service e08953
    err = uv__udp_maybe_deferred_bind(handle, addr->sa_family, 0);
Packit Service e08953
    if (err)
Packit Service e08953
      return err;
Packit Service e08953
  } else {
Packit Service e08953
    assert(handle->flags & UV_HANDLE_UDP_CONNECTED);
Packit Service e08953
  }
Packit b5b901
Packit b5b901
  memset(&h, 0, sizeof h);
Packit b5b901
  h.msg_name = (struct sockaddr*) addr;
Packit b5b901
  h.msg_namelen = addrlen;
Packit b5b901
  h.msg_iov = (struct iovec*) bufs;
Packit b5b901
  h.msg_iovlen = nbufs;
Packit b5b901
Packit b5b901
  do {
Packit b5b901
    size = sendmsg(handle->io_watcher.fd, &h, 0);
Packit b5b901
  } while (size == -1 && errno == EINTR);
Packit b5b901
Packit b5b901
  if (size == -1) {
Packit b5b901
    if (errno == EAGAIN || errno == EWOULDBLOCK || errno == ENOBUFS)
Packit b5b901
      return UV_EAGAIN;
Packit b5b901
    else
Packit b5b901
      return UV__ERR(errno);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return size;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static int uv__udp_set_membership4(uv_udp_t* handle,
Packit b5b901
                                   const struct sockaddr_in* multicast_addr,
Packit b5b901
                                   const char* interface_addr,
Packit b5b901
                                   uv_membership membership) {
Packit b5b901
  struct ip_mreq mreq;
Packit b5b901
  int optname;
Packit b5b901
  int err;
Packit b5b901
Packit b5b901
  memset(&mreq, 0, sizeof mreq);
Packit b5b901
Packit b5b901
  if (interface_addr) {
Packit b5b901
    err = uv_inet_pton(AF_INET, interface_addr, &mreq.imr_interface.s_addr);
Packit b5b901
    if (err)
Packit b5b901
      return err;
Packit b5b901
  } else {
Packit b5b901
    mreq.imr_interface.s_addr = htonl(INADDR_ANY);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  mreq.imr_multiaddr.s_addr = multicast_addr->sin_addr.s_addr;
Packit b5b901
Packit b5b901
  switch (membership) {
Packit b5b901
  case UV_JOIN_GROUP:
Packit b5b901
    optname = IP_ADD_MEMBERSHIP;
Packit b5b901
    break;
Packit b5b901
  case UV_LEAVE_GROUP:
Packit b5b901
    optname = IP_DROP_MEMBERSHIP;
Packit b5b901
    break;
Packit b5b901
  default:
Packit b5b901
    return UV_EINVAL;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (setsockopt(handle->io_watcher.fd,
Packit b5b901
                 IPPROTO_IP,
Packit b5b901
                 optname,
Packit b5b901
                 &mreq,
Packit b5b901
                 sizeof(mreq))) {
Packit b5b901
#if defined(__MVS__)
Packit b5b901
  if (errno == ENXIO)
Packit b5b901
    return UV_ENODEV;
Packit b5b901
#endif
Packit b5b901
    return UV__ERR(errno);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static int uv__udp_set_membership6(uv_udp_t* handle,
Packit b5b901
                                   const struct sockaddr_in6* multicast_addr,
Packit b5b901
                                   const char* interface_addr,
Packit b5b901
                                   uv_membership membership) {
Packit b5b901
  int optname;
Packit b5b901
  struct ipv6_mreq mreq;
Packit b5b901
  struct sockaddr_in6 addr6;
Packit b5b901
Packit b5b901
  memset(&mreq, 0, sizeof mreq);
Packit b5b901
Packit b5b901
  if (interface_addr) {
Packit b5b901
    if (uv_ip6_addr(interface_addr, 0, &addr6))
Packit b5b901
      return UV_EINVAL;
Packit b5b901
    mreq.ipv6mr_interface = addr6.sin6_scope_id;
Packit b5b901
  } else {
Packit b5b901
    mreq.ipv6mr_interface = 0;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  mreq.ipv6mr_multiaddr = multicast_addr->sin6_addr;
Packit b5b901
Packit b5b901
  switch (membership) {
Packit b5b901
  case UV_JOIN_GROUP:
Packit b5b901
    optname = IPV6_ADD_MEMBERSHIP;
Packit b5b901
    break;
Packit b5b901
  case UV_LEAVE_GROUP:
Packit b5b901
    optname = IPV6_DROP_MEMBERSHIP;
Packit b5b901
    break;
Packit b5b901
  default:
Packit b5b901
    return UV_EINVAL;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (setsockopt(handle->io_watcher.fd,
Packit b5b901
                 IPPROTO_IPV6,
Packit b5b901
                 optname,
Packit b5b901
                 &mreq,
Packit b5b901
                 sizeof(mreq))) {
Packit b5b901
#if defined(__MVS__)
Packit b5b901
  if (errno == ENXIO)
Packit b5b901
    return UV_ENODEV;
Packit b5b901
#endif
Packit b5b901
    return UV__ERR(errno);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit Service e08953
#if !defined(__OpenBSD__) && !defined(__NetBSD__) && !defined(__ANDROID__)
Packit Service e08953
static int uv__udp_set_source_membership4(uv_udp_t* handle,
Packit Service e08953
                                          const struct sockaddr_in* multicast_addr,
Packit Service e08953
                                          const char* interface_addr,
Packit Service e08953
                                          const struct sockaddr_in* source_addr,
Packit Service e08953
                                          uv_membership membership) {
Packit Service e08953
  struct ip_mreq_source mreq;
Packit Service e08953
  int optname;
Packit b5b901
  int err;
Packit b5b901
Packit Service e08953
  err = uv__udp_maybe_deferred_bind(handle, AF_INET, UV_UDP_REUSEADDR);
Packit Service e08953
  if (err)
Packit Service e08953
    return err;
Packit b5b901
Packit Service e08953
  memset(&mreq, 0, sizeof(mreq));
Packit Service e08953
Packit Service e08953
  if (interface_addr != NULL) {
Packit Service e08953
    err = uv_inet_pton(AF_INET, interface_addr, &mreq.imr_interface.s_addr);
Packit Service e08953
    if (err)
Packit Service e08953
      return err;
Packit Service e08953
  } else {
Packit Service e08953
    mreq.imr_interface.s_addr = htonl(INADDR_ANY);
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  mreq.imr_multiaddr.s_addr = multicast_addr->sin_addr.s_addr;
Packit Service e08953
  mreq.imr_sourceaddr.s_addr = source_addr->sin_addr.s_addr;
Packit Service e08953
Packit Service e08953
  if (membership == UV_JOIN_GROUP)
Packit Service e08953
    optname = IP_ADD_SOURCE_MEMBERSHIP;
Packit Service e08953
  else if (membership == UV_LEAVE_GROUP)
Packit Service e08953
    optname = IP_DROP_SOURCE_MEMBERSHIP;
Packit Service e08953
  else
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit Service e08953
  if (setsockopt(handle->io_watcher.fd,
Packit Service e08953
                 IPPROTO_IP,
Packit Service e08953
                 optname,
Packit Service e08953
                 &mreq,
Packit Service e08953
                 sizeof(mreq))) {
Packit Service e08953
    return UV__ERR(errno);
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  return 0;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit Service e08953
static int uv__udp_set_source_membership6(uv_udp_t* handle,
Packit Service e08953
                                          const struct sockaddr_in6* multicast_addr,
Packit Service e08953
                                          const char* interface_addr,
Packit Service e08953
                                          const struct sockaddr_in6* source_addr,
Packit Service e08953
                                          uv_membership membership) {
Packit Service e08953
  struct group_source_req mreq;
Packit Service e08953
  struct sockaddr_in6 addr6;
Packit Service e08953
  int optname;
Packit Service e08953
  int err;
Packit Service e08953
Packit Service e08953
  err = uv__udp_maybe_deferred_bind(handle, AF_INET6, UV_UDP_REUSEADDR);
Packit Service e08953
  if (err)
Packit Service e08953
    return err;
Packit Service e08953
Packit Service e08953
  memset(&mreq, 0, sizeof(mreq));
Packit Service e08953
Packit Service e08953
  if (interface_addr != NULL) {
Packit Service e08953
    err = uv_ip6_addr(interface_addr, 0, &addr6);
Packit Service e08953
    if (err)
Packit b5b901
      return err;
Packit Service e08953
    mreq.gsr_interface = addr6.sin6_scope_id;
Packit b5b901
  } else {
Packit Service e08953
    mreq.gsr_interface = 0;
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  STATIC_ASSERT(sizeof(mreq.gsr_group) >= sizeof(*multicast_addr));
Packit Service e08953
  STATIC_ASSERT(sizeof(mreq.gsr_source) >= sizeof(*source_addr));
Packit Service e08953
  memcpy(&mreq.gsr_group, multicast_addr, sizeof(*multicast_addr));
Packit Service e08953
  memcpy(&mreq.gsr_source, source_addr, sizeof(*source_addr));
Packit Service e08953
Packit Service e08953
  if (membership == UV_JOIN_GROUP)
Packit Service e08953
    optname = MCAST_JOIN_SOURCE_GROUP;
Packit Service e08953
  else if (membership == UV_LEAVE_GROUP)
Packit Service e08953
    optname = MCAST_LEAVE_SOURCE_GROUP;
Packit Service e08953
  else
Packit Service e08953
    return UV_EINVAL;
Packit Service e08953
Packit Service e08953
  if (setsockopt(handle->io_watcher.fd,
Packit Service e08953
                 IPPROTO_IPV6,
Packit Service e08953
                 optname,
Packit Service e08953
                 &mreq,
Packit Service e08953
                 sizeof(mreq))) {
Packit Service e08953
    return UV__ERR(errno);
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  return 0;
Packit Service e08953
}
Packit Service e08953
#endif
Packit Service e08953
Packit Service e08953
Packit Service e08953
int uv__udp_init_ex(uv_loop_t* loop,
Packit Service e08953
                    uv_udp_t* handle,
Packit Service e08953
                    unsigned flags,
Packit Service e08953
                    int domain) {
Packit Service e08953
  int fd;
Packit Service e08953
Packit Service e08953
  fd = -1;
Packit Service e08953
  if (domain != AF_UNSPEC) {
Packit Service e08953
    fd = uv__socket(domain, SOCK_DGRAM, 0);
Packit Service e08953
    if (fd < 0)
Packit Service e08953
      return fd;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  uv__handle_init(loop, (uv_handle_t*)handle, UV_UDP);
Packit b5b901
  handle->alloc_cb = NULL;
Packit b5b901
  handle->recv_cb = NULL;
Packit b5b901
  handle->send_queue_size = 0;
Packit b5b901
  handle->send_queue_count = 0;
Packit b5b901
  uv__io_init(&handle->io_watcher, uv__udp_io, fd);
Packit b5b901
  QUEUE_INIT(&handle->write_queue);
Packit b5b901
  QUEUE_INIT(&handle->write_completed_queue);
Packit b5b901
Packit Service e08953
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_udp_open(uv_udp_t* handle, uv_os_sock_t sock) {
Packit b5b901
  int err;
Packit b5b901
Packit b5b901
  /* Check for already active socket. */
Packit b5b901
  if (handle->io_watcher.fd != -1)
Packit b5b901
    return UV_EBUSY;
Packit b5b901
Packit b5b901
  if (uv__fd_exists(handle->loop, sock))
Packit b5b901
    return UV_EEXIST;
Packit b5b901
Packit b5b901
  err = uv__nonblock(sock, 1);
Packit b5b901
  if (err)
Packit b5b901
    return err;
Packit b5b901
Packit b5b901
  err = uv__set_reuse(sock);
Packit b5b901
  if (err)
Packit b5b901
    return err;
Packit b5b901
Packit b5b901
  handle->io_watcher.fd = sock;
Packit Service e08953
  if (uv__udp_is_connected(handle))
Packit Service e08953
    handle->flags |= UV_HANDLE_UDP_CONNECTED;
Packit Service e08953
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_udp_set_membership(uv_udp_t* handle,
Packit b5b901
                          const char* multicast_addr,
Packit b5b901
                          const char* interface_addr,
Packit b5b901
                          uv_membership membership) {
Packit b5b901
  int err;
Packit b5b901
  struct sockaddr_in addr4;
Packit b5b901
  struct sockaddr_in6 addr6;
Packit b5b901
Packit b5b901
  if (uv_ip4_addr(multicast_addr, 0, &addr4) == 0) {
Packit b5b901
    err = uv__udp_maybe_deferred_bind(handle, AF_INET, UV_UDP_REUSEADDR);
Packit b5b901
    if (err)
Packit b5b901
      return err;
Packit b5b901
    return uv__udp_set_membership4(handle, &addr4, interface_addr, membership);
Packit b5b901
  } else if (uv_ip6_addr(multicast_addr, 0, &addr6) == 0) {
Packit b5b901
    err = uv__udp_maybe_deferred_bind(handle, AF_INET6, UV_UDP_REUSEADDR);
Packit b5b901
    if (err)
Packit b5b901
      return err;
Packit b5b901
    return uv__udp_set_membership6(handle, &addr6, interface_addr, membership);
Packit b5b901
  } else {
Packit b5b901
    return UV_EINVAL;
Packit b5b901
  }
Packit b5b901
}
Packit b5b901
Packit Service e08953
Packit Service e08953
int uv_udp_set_source_membership(uv_udp_t* handle,
Packit Service e08953
                                 const char* multicast_addr,
Packit Service e08953
                                 const char* interface_addr,
Packit Service e08953
                                 const char* source_addr,
Packit Service e08953
                                 uv_membership membership) {
Packit Service e08953
#if !defined(__OpenBSD__) && !defined(__NetBSD__) && !defined(__ANDROID__)
Packit Service e08953
  int err;
Packit Service e08953
  union uv__sockaddr mcast_addr;
Packit Service e08953
  union uv__sockaddr src_addr;
Packit Service e08953
Packit Service e08953
  err = uv_ip4_addr(multicast_addr, 0, &mcast_addr.in);
Packit Service e08953
  if (err) {
Packit Service e08953
    err = uv_ip6_addr(multicast_addr, 0, &mcast_addr.in6);
Packit Service e08953
    if (err)
Packit Service e08953
      return err;
Packit Service e08953
    err = uv_ip6_addr(source_addr, 0, &src_addr.in6);
Packit Service e08953
    if (err)
Packit Service e08953
      return err;
Packit Service e08953
    return uv__udp_set_source_membership6(handle,
Packit Service e08953
                                          &mcast_addr.in6,
Packit Service e08953
                                          interface_addr,
Packit Service e08953
                                          &src_addr.in6,
Packit Service e08953
                                          membership);
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  err = uv_ip4_addr(source_addr, 0, &src_addr.in);
Packit Service e08953
  if (err)
Packit Service e08953
    return err;
Packit Service e08953
  return uv__udp_set_source_membership4(handle,
Packit Service e08953
                                        &mcast_addr.in,
Packit Service e08953
                                        interface_addr,
Packit Service e08953
                                        &src_addr.in,
Packit Service e08953
                                        membership);
Packit Service e08953
#else
Packit Service e08953
  return UV_ENOSYS;
Packit Service e08953
#endif
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit b5b901
static int uv__setsockopt(uv_udp_t* handle,
Packit b5b901
                         int option4,
Packit b5b901
                         int option6,
Packit b5b901
                         const void* val,
Packit Service e08953
                         socklen_t size) {
Packit b5b901
  int r;
Packit b5b901
Packit b5b901
  if (handle->flags & UV_HANDLE_IPV6)
Packit b5b901
    r = setsockopt(handle->io_watcher.fd,
Packit b5b901
                   IPPROTO_IPV6,
Packit b5b901
                   option6,
Packit b5b901
                   val,
Packit b5b901
                   size);
Packit b5b901
  else
Packit b5b901
    r = setsockopt(handle->io_watcher.fd,
Packit b5b901
                   IPPROTO_IP,
Packit b5b901
                   option4,
Packit b5b901
                   val,
Packit b5b901
                   size);
Packit b5b901
  if (r)
Packit b5b901
    return UV__ERR(errno);
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
static int uv__setsockopt_maybe_char(uv_udp_t* handle,
Packit b5b901
                                     int option4,
Packit b5b901
                                     int option6,
Packit b5b901
                                     int val) {
Packit b5b901
#if defined(__sun) || defined(_AIX) || defined(__MVS__)
Packit b5b901
  char arg = val;
Packit b5b901
#elif defined(__OpenBSD__)
Packit b5b901
  unsigned char arg = val;
Packit b5b901
#else
Packit b5b901
  int arg = val;
Packit b5b901
#endif
Packit b5b901
Packit b5b901
  if (val < 0 || val > 255)
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit b5b901
  return uv__setsockopt(handle, option4, option6, &arg, sizeof(arg));
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_udp_set_broadcast(uv_udp_t* handle, int on) {
Packit b5b901
  if (setsockopt(handle->io_watcher.fd,
Packit b5b901
                 SOL_SOCKET,
Packit b5b901
                 SO_BROADCAST,
Packit b5b901
                 &on,
Packit b5b901
                 sizeof(on))) {
Packit b5b901
    return UV__ERR(errno);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_udp_set_ttl(uv_udp_t* handle, int ttl) {
Packit b5b901
  if (ttl < 1 || ttl > 255)
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit b5b901
#if defined(__MVS__)
Packit b5b901
  if (!(handle->flags & UV_HANDLE_IPV6))
Packit b5b901
    return UV_ENOTSUP;  /* zOS does not support setting ttl for IPv4 */
Packit b5b901
#endif
Packit b5b901
Packit b5b901
/*
Packit b5b901
 * On Solaris and derivatives such as SmartOS, the length of socket options
Packit b5b901
 * is sizeof(int) for IP_TTL and IPV6_UNICAST_HOPS,
Packit b5b901
 * so hardcode the size of these options on this platform,
Packit b5b901
 * and use the general uv__setsockopt_maybe_char call on other platforms.
Packit b5b901
 */
Packit b5b901
#if defined(__sun) || defined(_AIX) || defined(__OpenBSD__) || \
Packit b5b901
    defined(__MVS__)
Packit b5b901
Packit b5b901
  return uv__setsockopt(handle,
Packit b5b901
                        IP_TTL,
Packit b5b901
                        IPV6_UNICAST_HOPS,
Packit b5b901
                        &ttl,
Packit b5b901
                        sizeof(ttl));
Packit Service e08953
Packit Service e08953
#else /* !(defined(__sun) || defined(_AIX) || defined (__OpenBSD__) ||
Packit Service e08953
           defined(__MVS__)) */
Packit b5b901
Packit b5b901
  return uv__setsockopt_maybe_char(handle,
Packit b5b901
                                   IP_TTL,
Packit b5b901
                                   IPV6_UNICAST_HOPS,
Packit b5b901
                                   ttl);
Packit Service e08953
Packit Service e08953
#endif /* defined(__sun) || defined(_AIX) || defined (__OpenBSD__) ||
Packit Service e08953
          defined(__MVS__) */
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_udp_set_multicast_ttl(uv_udp_t* handle, int ttl) {
Packit b5b901
/*
Packit b5b901
 * On Solaris and derivatives such as SmartOS, the length of socket options
Packit b5b901
 * is sizeof(int) for IPV6_MULTICAST_HOPS and sizeof(char) for
Packit b5b901
 * IP_MULTICAST_TTL, so hardcode the size of the option in the IPv6 case,
Packit b5b901
 * and use the general uv__setsockopt_maybe_char call otherwise.
Packit b5b901
 */
Packit Service e08953
#if defined(__sun) || defined(_AIX) || defined(__OpenBSD__) || \
Packit Service e08953
    defined(__MVS__)
Packit b5b901
  if (handle->flags & UV_HANDLE_IPV6)
Packit b5b901
    return uv__setsockopt(handle,
Packit b5b901
                          IP_MULTICAST_TTL,
Packit b5b901
                          IPV6_MULTICAST_HOPS,
Packit b5b901
                          &ttl,
Packit b5b901
                          sizeof(ttl));
Packit Service e08953
#endif /* defined(__sun) || defined(_AIX) || defined(__OpenBSD__) || \
Packit Service e08953
    defined(__MVS__) */
Packit b5b901
Packit b5b901
  return uv__setsockopt_maybe_char(handle,
Packit b5b901
                                   IP_MULTICAST_TTL,
Packit b5b901
                                   IPV6_MULTICAST_HOPS,
Packit b5b901
                                   ttl);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_udp_set_multicast_loop(uv_udp_t* handle, int on) {
Packit b5b901
/*
Packit b5b901
 * On Solaris and derivatives such as SmartOS, the length of socket options
Packit b5b901
 * is sizeof(int) for IPV6_MULTICAST_LOOP and sizeof(char) for
Packit b5b901
 * IP_MULTICAST_LOOP, so hardcode the size of the option in the IPv6 case,
Packit b5b901
 * and use the general uv__setsockopt_maybe_char call otherwise.
Packit b5b901
 */
Packit Service e08953
#if defined(__sun) || defined(_AIX) || defined(__OpenBSD__) || \
Packit Service e08953
    defined(__MVS__)
Packit b5b901
  if (handle->flags & UV_HANDLE_IPV6)
Packit b5b901
    return uv__setsockopt(handle,
Packit b5b901
                          IP_MULTICAST_LOOP,
Packit b5b901
                          IPV6_MULTICAST_LOOP,
Packit b5b901
                          &on,
Packit b5b901
                          sizeof(on));
Packit Service e08953
#endif /* defined(__sun) || defined(_AIX) ||defined(__OpenBSD__) ||
Packit Service e08953
    defined(__MVS__) */
Packit b5b901
Packit b5b901
  return uv__setsockopt_maybe_char(handle,
Packit b5b901
                                   IP_MULTICAST_LOOP,
Packit b5b901
                                   IPV6_MULTICAST_LOOP,
Packit b5b901
                                   on);
Packit b5b901
}
Packit b5b901
Packit b5b901
int uv_udp_set_multicast_interface(uv_udp_t* handle, const char* interface_addr) {
Packit b5b901
  struct sockaddr_storage addr_st;
Packit b5b901
  struct sockaddr_in* addr4;
Packit b5b901
  struct sockaddr_in6* addr6;
Packit b5b901
Packit b5b901
  addr4 = (struct sockaddr_in*) &addr_st;
Packit b5b901
  addr6 = (struct sockaddr_in6*) &addr_st;
Packit b5b901
Packit b5b901
  if (!interface_addr) {
Packit b5b901
    memset(&addr_st, 0, sizeof addr_st);
Packit b5b901
    if (handle->flags & UV_HANDLE_IPV6) {
Packit b5b901
      addr_st.ss_family = AF_INET6;
Packit b5b901
      addr6->sin6_scope_id = 0;
Packit b5b901
    } else {
Packit b5b901
      addr_st.ss_family = AF_INET;
Packit b5b901
      addr4->sin_addr.s_addr = htonl(INADDR_ANY);
Packit b5b901
    }
Packit b5b901
  } else if (uv_ip4_addr(interface_addr, 0, addr4) == 0) {
Packit b5b901
    /* nothing, address was parsed */
Packit b5b901
  } else if (uv_ip6_addr(interface_addr, 0, addr6) == 0) {
Packit b5b901
    /* nothing, address was parsed */
Packit b5b901
  } else {
Packit b5b901
    return UV_EINVAL;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (addr_st.ss_family == AF_INET) {
Packit b5b901
    if (setsockopt(handle->io_watcher.fd,
Packit b5b901
                   IPPROTO_IP,
Packit b5b901
                   IP_MULTICAST_IF,
Packit b5b901
                   (void*) &addr4->sin_addr,
Packit b5b901
                   sizeof(addr4->sin_addr)) == -1) {
Packit b5b901
      return UV__ERR(errno);
Packit b5b901
    }
Packit b5b901
  } else if (addr_st.ss_family == AF_INET6) {
Packit b5b901
    if (setsockopt(handle->io_watcher.fd,
Packit b5b901
                   IPPROTO_IPV6,
Packit b5b901
                   IPV6_MULTICAST_IF,
Packit b5b901
                   &addr6->sin6_scope_id,
Packit b5b901
                   sizeof(addr6->sin6_scope_id)) == -1) {
Packit b5b901
      return UV__ERR(errno);
Packit b5b901
    }
Packit b5b901
  } else {
Packit b5b901
    assert(0 && "unexpected address family");
Packit b5b901
    abort();
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit Service e08953
int uv_udp_getpeername(const uv_udp_t* handle,
Packit b5b901
                       struct sockaddr* name,
Packit b5b901
                       int* namelen) {
Packit b5b901
Packit Service e08953
  return uv__getsockpeername((const uv_handle_t*) handle,
Packit Service e08953
                             getpeername,
Packit Service e08953
                             name,
Packit Service e08953
                             namelen);
Packit Service e08953
}
Packit b5b901
Packit Service e08953
int uv_udp_getsockname(const uv_udp_t* handle,
Packit Service e08953
                       struct sockaddr* name,
Packit Service e08953
                       int* namelen) {
Packit b5b901
Packit Service e08953
  return uv__getsockpeername((const uv_handle_t*) handle,
Packit Service e08953
                             getsockname,
Packit Service e08953
                             name,
Packit Service e08953
                             namelen);
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
  int err;
Packit b5b901
Packit b5b901
  if (alloc_cb == NULL || recv_cb == NULL)
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit b5b901
  if (uv__io_active(&handle->io_watcher, POLLIN))
Packit b5b901
    return UV_EALREADY;  /* FIXME(bnoordhuis) Should be UV_EBUSY. */
Packit b5b901
Packit b5b901
  err = uv__udp_maybe_deferred_bind(handle, AF_INET, 0);
Packit b5b901
  if (err)
Packit b5b901
    return err;
Packit b5b901
Packit b5b901
  handle->alloc_cb = alloc_cb;
Packit b5b901
  handle->recv_cb = recv_cb;
Packit b5b901
Packit b5b901
  uv__io_start(handle->loop, &handle->io_watcher, POLLIN);
Packit b5b901
  uv__handle_start(handle);
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv__udp_recv_stop(uv_udp_t* handle) {
Packit b5b901
  uv__io_stop(handle->loop, &handle->io_watcher, POLLIN);
Packit b5b901
Packit b5b901
  if (!uv__io_active(&handle->io_watcher, POLLOUT))
Packit b5b901
    uv__handle_stop(handle);
Packit b5b901
Packit b5b901
  handle->alloc_cb = NULL;
Packit b5b901
  handle->recv_cb = NULL;
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}