Blame src/win/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 <assert.h>
Packit b5b901
#include <stdlib.h>
Packit b5b901
Packit b5b901
#include "uv.h"
Packit b5b901
#include "internal.h"
Packit b5b901
#include "handle-inl.h"
Packit b5b901
#include "stream-inl.h"
Packit b5b901
#include "req-inl.h"
Packit b5b901
Packit b5b901
Packit b5b901
/*
Packit b5b901
 * Threshold of active udp streams for which to preallocate udp read buffers.
Packit b5b901
 */
Packit b5b901
const unsigned int uv_active_udp_streams_threshold = 0;
Packit b5b901
Packit b5b901
/* A zero-size buffer for use by uv_udp_read */
Packit b5b901
static char uv_zero_[] = "";
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
                             0);
Packit Service e08953
}
Packit b5b901
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 Service e08953
Packit Service e08953
  return uv__getsockpeername((const uv_handle_t*) handle,
Packit Service e08953
                             getsockname,
Packit Service e08953
                             name,
Packit Service e08953
                             namelen,
Packit Service e08953
                             0);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static int uv_udp_set_socket(uv_loop_t* loop, uv_udp_t* handle, SOCKET socket,
Packit b5b901
    int family) {
Packit b5b901
  DWORD yes = 1;
Packit b5b901
  WSAPROTOCOL_INFOW info;
Packit b5b901
  int opt_len;
Packit b5b901
Packit b5b901
  if (handle->socket != INVALID_SOCKET)
Packit b5b901
    return UV_EBUSY;
Packit b5b901
Packit b5b901
  /* Set the socket to nonblocking mode */
Packit b5b901
  if (ioctlsocket(socket, FIONBIO, &yes) == SOCKET_ERROR) {
Packit b5b901
    return WSAGetLastError();
Packit b5b901
  }
Packit b5b901
Packit b5b901
  /* Make the socket non-inheritable */
Packit b5b901
  if (!SetHandleInformation((HANDLE)socket, HANDLE_FLAG_INHERIT, 0)) {
Packit b5b901
    return GetLastError();
Packit b5b901
  }
Packit b5b901
Packit b5b901
  /* Associate it with the I/O completion port. Use uv_handle_t pointer as
Packit b5b901
   * completion key. */
Packit b5b901
  if (CreateIoCompletionPort((HANDLE)socket,
Packit b5b901
                             loop->iocp,
Packit b5b901
                             (ULONG_PTR)socket,
Packit b5b901
                             0) == NULL) {
Packit b5b901
    return GetLastError();
Packit b5b901
  }
Packit b5b901
Packit b5b901
  /* All known Windows that support SetFileCompletionNotificationModes have a
Packit b5b901
   * bug that makes it impossible to use this function in conjunction with
Packit b5b901
   * datagram sockets. We can work around that but only if the user is using
Packit b5b901
   * the default UDP driver (AFD) and has no other. LSPs stacked on top. Here
Packit b5b901
   * we check whether that is the case. */
Packit b5b901
  opt_len = (int) sizeof info;
Packit b5b901
  if (getsockopt(
Packit b5b901
          socket, SOL_SOCKET, SO_PROTOCOL_INFOW, (char*) &info, &opt_len) ==
Packit b5b901
      SOCKET_ERROR) {
Packit b5b901
    return GetLastError();
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (info.ProtocolChain.ChainLen == 1) {
Packit b5b901
    if (SetFileCompletionNotificationModes(
Packit b5b901
            (HANDLE) socket,
Packit b5b901
            FILE_SKIP_SET_EVENT_ON_HANDLE |
Packit b5b901
                FILE_SKIP_COMPLETION_PORT_ON_SUCCESS)) {
Packit b5b901
      handle->flags |= UV_HANDLE_SYNC_BYPASS_IOCP;
Packit b5b901
      handle->func_wsarecv = uv_wsarecv_workaround;
Packit b5b901
      handle->func_wsarecvfrom = uv_wsarecvfrom_workaround;
Packit b5b901
    } else if (GetLastError() != ERROR_INVALID_FUNCTION) {
Packit b5b901
      return GetLastError();
Packit b5b901
    }
Packit b5b901
  }
Packit b5b901
Packit b5b901
  handle->socket = socket;
Packit b5b901
Packit b5b901
  if (family == AF_INET6) {
Packit b5b901
    handle->flags |= UV_HANDLE_IPV6;
Packit b5b901
  } else {
Packit b5b901
    assert(!(handle->flags & UV_HANDLE_IPV6));
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
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 b5b901
  uv__handle_init(loop, (uv_handle_t*) handle, UV_UDP);
Packit b5b901
  handle->socket = INVALID_SOCKET;
Packit b5b901
  handle->reqs_pending = 0;
Packit b5b901
  handle->activecnt = 0;
Packit b5b901
  handle->func_wsarecv = WSARecv;
Packit b5b901
  handle->func_wsarecvfrom = WSARecvFrom;
Packit b5b901
  handle->send_queue_size = 0;
Packit b5b901
  handle->send_queue_count = 0;
Packit b5b901
  UV_REQ_INIT(&handle->recv_req, UV_UDP_RECV);
Packit b5b901
  handle->recv_req.data = handle;
Packit b5b901
Packit b5b901
  /* If anything fails beyond this point we need to remove the handle from
Packit b5b901
   * the handle queue, since it was added by uv__handle_init.
Packit b5b901
   */
Packit b5b901
Packit b5b901
  if (domain != AF_UNSPEC) {
Packit b5b901
    SOCKET sock;
Packit b5b901
    DWORD err;
Packit b5b901
Packit b5b901
    sock = socket(domain, SOCK_DGRAM, 0);
Packit b5b901
    if (sock == INVALID_SOCKET) {
Packit b5b901
      err = WSAGetLastError();
Packit b5b901
      QUEUE_REMOVE(&handle->handle_queue);
Packit b5b901
      return uv_translate_sys_error(err);
Packit b5b901
    }
Packit b5b901
Packit b5b901
    err = uv_udp_set_socket(handle->loop, handle, sock, domain);
Packit b5b901
    if (err) {
Packit b5b901
      closesocket(sock);
Packit b5b901
      QUEUE_REMOVE(&handle->handle_queue);
Packit b5b901
      return uv_translate_sys_error(err);
Packit b5b901
    }
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
void uv_udp_close(uv_loop_t* loop, uv_udp_t* handle) {
Packit b5b901
  uv_udp_recv_stop(handle);
Packit b5b901
  closesocket(handle->socket);
Packit b5b901
  handle->socket = INVALID_SOCKET;
Packit b5b901
Packit b5b901
  uv__handle_closing(handle);
Packit b5b901
Packit b5b901
  if (handle->reqs_pending == 0) {
Packit b5b901
    uv_want_endgame(loop, (uv_handle_t*) handle);
Packit b5b901
  }
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
void uv_udp_endgame(uv_loop_t* loop, uv_udp_t* handle) {
Packit b5b901
  if (handle->flags & UV_HANDLE_CLOSING &&
Packit b5b901
      handle->reqs_pending == 0) {
Packit b5b901
    assert(!(handle->flags & UV_HANDLE_CLOSED));
Packit b5b901
    uv__handle_close(handle);
Packit b5b901
  }
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static int uv_udp_maybe_bind(uv_udp_t* handle,
Packit b5b901
                             const struct sockaddr* addr,
Packit b5b901
                             unsigned int addrlen,
Packit b5b901
                             unsigned int flags) {
Packit b5b901
  int r;
Packit b5b901
  int err;
Packit b5b901
  DWORD no = 0;
Packit b5b901
Packit b5b901
  if (handle->flags & UV_HANDLE_BOUND)
Packit b5b901
    return 0;
Packit b5b901
Packit b5b901
  if ((flags & UV_UDP_IPV6ONLY) && addr->sa_family != AF_INET6) {
Packit b5b901
    /* UV_UDP_IPV6ONLY is supported only for IPV6 sockets */
Packit b5b901
    return ERROR_INVALID_PARAMETER;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (handle->socket == INVALID_SOCKET) {
Packit b5b901
    SOCKET sock = socket(addr->sa_family, SOCK_DGRAM, 0);
Packit b5b901
    if (sock == INVALID_SOCKET) {
Packit b5b901
      return WSAGetLastError();
Packit b5b901
    }
Packit b5b901
Packit b5b901
    err = uv_udp_set_socket(handle->loop, handle, sock, addr->sa_family);
Packit b5b901
    if (err) {
Packit b5b901
      closesocket(sock);
Packit b5b901
      return err;
Packit b5b901
    }
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (flags & UV_UDP_REUSEADDR) {
Packit b5b901
    DWORD yes = 1;
Packit b5b901
    /* Set SO_REUSEADDR on the socket. */
Packit b5b901
    if (setsockopt(handle->socket,
Packit b5b901
                   SOL_SOCKET,
Packit b5b901
                   SO_REUSEADDR,
Packit b5b901
                   (char*) &yes,
Packit b5b901
                   sizeof yes) == SOCKET_ERROR) {
Packit b5b901
      err = WSAGetLastError();
Packit b5b901
      return err;
Packit b5b901
    }
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (addr->sa_family == AF_INET6)
Packit b5b901
    handle->flags |= UV_HANDLE_IPV6;
Packit b5b901
Packit b5b901
  if (addr->sa_family == AF_INET6 && !(flags & UV_UDP_IPV6ONLY)) {
Packit b5b901
    /* On windows IPV6ONLY is on by default. If the user doesn't specify it
Packit b5b901
     * libuv turns it off. */
Packit b5b901
Packit b5b901
    /* TODO: how to handle errors? This may fail if there is no ipv4 stack
Packit b5b901
     * available, or when run on XP/2003 which have no support for dualstack
Packit b5b901
     * sockets. For now we're silently ignoring the error. */
Packit b5b901
    setsockopt(handle->socket,
Packit b5b901
               IPPROTO_IPV6,
Packit b5b901
               IPV6_V6ONLY,
Packit b5b901
               (char*) &no,
Packit b5b901
               sizeof no);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  r = bind(handle->socket, addr, addrlen);
Packit b5b901
  if (r == SOCKET_ERROR) {
Packit b5b901
    return WSAGetLastError();
Packit b5b901
  }
Packit b5b901
Packit b5b901
  handle->flags |= UV_HANDLE_BOUND;
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static void uv_udp_queue_recv(uv_loop_t* loop, uv_udp_t* handle) {
Packit b5b901
  uv_req_t* req;
Packit b5b901
  uv_buf_t buf;
Packit b5b901
  DWORD bytes, flags;
Packit b5b901
  int result;
Packit b5b901
Packit b5b901
  assert(handle->flags & UV_HANDLE_READING);
Packit b5b901
  assert(!(handle->flags & UV_HANDLE_READ_PENDING));
Packit b5b901
Packit b5b901
  req = &handle->recv_req;
Packit b5b901
  memset(&req->u.io.overlapped, 0, sizeof(req->u.io.overlapped));
Packit b5b901
Packit b5b901
  /*
Packit b5b901
   * Preallocate a read buffer if the number of active streams is below
Packit b5b901
   * the threshold.
Packit b5b901
  */
Packit b5b901
  if (loop->active_udp_streams < uv_active_udp_streams_threshold) {
Packit b5b901
    handle->flags &= ~UV_HANDLE_ZERO_READ;
Packit b5b901
Packit b5b901
    handle->recv_buffer = uv_buf_init(NULL, 0);
Packit b5b901
    handle->alloc_cb((uv_handle_t*) handle, 65536, &handle->recv_buffer);
Packit b5b901
    if (handle->recv_buffer.base == NULL || handle->recv_buffer.len == 0) {
Packit b5b901
      handle->recv_cb(handle, UV_ENOBUFS, &handle->recv_buffer, NULL, 0);
Packit b5b901
      return;
Packit b5b901
    }
Packit b5b901
    assert(handle->recv_buffer.base != NULL);
Packit b5b901
Packit b5b901
    buf = handle->recv_buffer;
Packit b5b901
    memset(&handle->recv_from, 0, sizeof handle->recv_from);
Packit b5b901
    handle->recv_from_len = sizeof handle->recv_from;
Packit b5b901
    flags = 0;
Packit b5b901
Packit b5b901
    result = handle->func_wsarecvfrom(handle->socket,
Packit b5b901
                                      (WSABUF*) &buf,
Packit b5b901
                                      1,
Packit b5b901
                                      &bytes,
Packit b5b901
                                      &flags,
Packit b5b901
                                      (struct sockaddr*) &handle->recv_from,
Packit b5b901
                                      &handle->recv_from_len,
Packit b5b901
                                      &req->u.io.overlapped,
Packit b5b901
                                      NULL);
Packit b5b901
Packit b5b901
    if (UV_SUCCEEDED_WITHOUT_IOCP(result == 0)) {
Packit b5b901
      /* Process the req without IOCP. */
Packit b5b901
      handle->flags |= UV_HANDLE_READ_PENDING;
Packit b5b901
      req->u.io.overlapped.InternalHigh = bytes;
Packit b5b901
      handle->reqs_pending++;
Packit b5b901
      uv_insert_pending_req(loop, req);
Packit b5b901
    } else if (UV_SUCCEEDED_WITH_IOCP(result == 0)) {
Packit b5b901
      /* The req will be processed with IOCP. */
Packit b5b901
      handle->flags |= UV_HANDLE_READ_PENDING;
Packit b5b901
      handle->reqs_pending++;
Packit b5b901
    } else {
Packit b5b901
      /* Make this req pending reporting an error. */
Packit b5b901
      SET_REQ_ERROR(req, WSAGetLastError());
Packit b5b901
      uv_insert_pending_req(loop, req);
Packit b5b901
      handle->reqs_pending++;
Packit b5b901
    }
Packit b5b901
Packit b5b901
  } else {
Packit b5b901
    handle->flags |= UV_HANDLE_ZERO_READ;
Packit b5b901
Packit b5b901
    buf.base = (char*) uv_zero_;
Packit b5b901
    buf.len = 0;
Packit b5b901
    flags = MSG_PEEK;
Packit b5b901
Packit b5b901
    result = handle->func_wsarecv(handle->socket,
Packit b5b901
                                  (WSABUF*) &buf,
Packit b5b901
                                  1,
Packit b5b901
                                  &bytes,
Packit b5b901
                                  &flags,
Packit b5b901
                                  &req->u.io.overlapped,
Packit b5b901
                                  NULL);
Packit b5b901
Packit b5b901
    if (UV_SUCCEEDED_WITHOUT_IOCP(result == 0)) {
Packit b5b901
      /* Process the req without IOCP. */
Packit b5b901
      handle->flags |= UV_HANDLE_READ_PENDING;
Packit b5b901
      req->u.io.overlapped.InternalHigh = bytes;
Packit b5b901
      handle->reqs_pending++;
Packit b5b901
      uv_insert_pending_req(loop, req);
Packit b5b901
    } else if (UV_SUCCEEDED_WITH_IOCP(result == 0)) {
Packit b5b901
      /* The req will be processed with IOCP. */
Packit b5b901
      handle->flags |= UV_HANDLE_READ_PENDING;
Packit b5b901
      handle->reqs_pending++;
Packit b5b901
    } else {
Packit b5b901
      /* Make this req pending reporting an error. */
Packit b5b901
      SET_REQ_ERROR(req, WSAGetLastError());
Packit b5b901
      uv_insert_pending_req(loop, req);
Packit b5b901
      handle->reqs_pending++;
Packit b5b901
    }
Packit b5b901
  }
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv__udp_recv_start(uv_udp_t* handle, uv_alloc_cb alloc_cb,
Packit b5b901
    uv_udp_recv_cb recv_cb) {
Packit b5b901
  uv_loop_t* loop = handle->loop;
Packit b5b901
  int err;
Packit b5b901
Packit b5b901
  if (handle->flags & UV_HANDLE_READING) {
Packit b5b901
    return UV_EALREADY;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  err = uv_udp_maybe_bind(handle,
Packit b5b901
                          (const struct sockaddr*) &uv_addr_ip4_any_,
Packit b5b901
                          sizeof(uv_addr_ip4_any_),
Packit b5b901
                          0);
Packit b5b901
  if (err)
Packit b5b901
    return uv_translate_sys_error(err);
Packit b5b901
Packit b5b901
  handle->flags |= UV_HANDLE_READING;
Packit b5b901
  INCREASE_ACTIVE_COUNT(loop, handle);
Packit b5b901
  loop->active_udp_streams++;
Packit b5b901
Packit b5b901
  handle->recv_cb = recv_cb;
Packit b5b901
  handle->alloc_cb = alloc_cb;
Packit b5b901
Packit b5b901
  /* If reading was stopped and then started again, there could still be a recv
Packit b5b901
   * request pending. */
Packit b5b901
  if (!(handle->flags & UV_HANDLE_READ_PENDING))
Packit b5b901
    uv_udp_queue_recv(loop, 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
  if (handle->flags & UV_HANDLE_READING) {
Packit b5b901
    handle->flags &= ~UV_HANDLE_READING;
Packit b5b901
    handle->loop->active_udp_streams--;
Packit b5b901
    DECREASE_ACTIVE_COUNT(loop, handle);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static int uv__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 cb) {
Packit b5b901
  uv_loop_t* loop = handle->loop;
Packit b5b901
  DWORD result, bytes;
Packit b5b901
Packit b5b901
  UV_REQ_INIT(req, UV_UDP_SEND);
Packit b5b901
  req->handle = handle;
Packit b5b901
  req->cb = cb;
Packit b5b901
  memset(&req->u.io.overlapped, 0, sizeof(req->u.io.overlapped));
Packit b5b901
Packit b5b901
  result = WSASendTo(handle->socket,
Packit b5b901
                     (WSABUF*)bufs,
Packit b5b901
                     nbufs,
Packit b5b901
                     &bytes,
Packit b5b901
                     0,
Packit b5b901
                     addr,
Packit b5b901
                     addrlen,
Packit b5b901
                     &req->u.io.overlapped,
Packit b5b901
                     NULL);
Packit b5b901
Packit b5b901
  if (UV_SUCCEEDED_WITHOUT_IOCP(result == 0)) {
Packit b5b901
    /* Request completed immediately. */
Packit b5b901
    req->u.io.queued_bytes = 0;
Packit b5b901
    handle->reqs_pending++;
Packit b5b901
    handle->send_queue_size += req->u.io.queued_bytes;
Packit b5b901
    handle->send_queue_count++;
Packit b5b901
    REGISTER_HANDLE_REQ(loop, handle, req);
Packit b5b901
    uv_insert_pending_req(loop, (uv_req_t*)req);
Packit b5b901
  } else if (UV_SUCCEEDED_WITH_IOCP(result == 0)) {
Packit b5b901
    /* Request queued by the kernel. */
Packit b5b901
    req->u.io.queued_bytes = uv__count_bufs(bufs, nbufs);
Packit b5b901
    handle->reqs_pending++;
Packit b5b901
    handle->send_queue_size += req->u.io.queued_bytes;
Packit b5b901
    handle->send_queue_count++;
Packit b5b901
    REGISTER_HANDLE_REQ(loop, handle, req);
Packit b5b901
  } else {
Packit b5b901
    /* Send failed due to an error. */
Packit b5b901
    return WSAGetLastError();
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
void uv_process_udp_recv_req(uv_loop_t* loop, uv_udp_t* handle,
Packit b5b901
    uv_req_t* req) {
Packit b5b901
  uv_buf_t buf;
Packit b5b901
  int partial;
Packit b5b901
Packit b5b901
  assert(handle->type == UV_UDP);
Packit b5b901
Packit b5b901
  handle->flags &= ~UV_HANDLE_READ_PENDING;
Packit b5b901
Packit b5b901
  if (!REQ_SUCCESS(req)) {
Packit b5b901
    DWORD err = GET_REQ_SOCK_ERROR(req);
Packit b5b901
    if (err == WSAEMSGSIZE) {
Packit b5b901
      /* Not a real error, it just indicates that the received packet was
Packit b5b901
       * bigger than the receive buffer. */
Packit b5b901
    } else if (err == WSAECONNRESET || err == WSAENETRESET) {
Packit b5b901
      /* A previous sendto operation failed; ignore this error. If zero-reading
Packit b5b901
       * we need to call WSARecv/WSARecvFrom _without_ the. MSG_PEEK flag to
Packit b5b901
       * clear out the error queue. For nonzero reads, immediately queue a new
Packit b5b901
       * receive. */
Packit b5b901
      if (!(handle->flags & UV_HANDLE_ZERO_READ)) {
Packit b5b901
        goto done;
Packit b5b901
      }
Packit b5b901
    } else {
Packit b5b901
      /* A real error occurred. Report the error to the user only if we're
Packit b5b901
       * currently reading. */
Packit b5b901
      if (handle->flags & UV_HANDLE_READING) {
Packit b5b901
        uv_udp_recv_stop(handle);
Packit b5b901
        buf = (handle->flags & UV_HANDLE_ZERO_READ) ?
Packit b5b901
              uv_buf_init(NULL, 0) : handle->recv_buffer;
Packit b5b901
        handle->recv_cb(handle, uv_translate_sys_error(err), &buf, NULL, 0);
Packit b5b901
      }
Packit b5b901
      goto done;
Packit b5b901
    }
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (!(handle->flags & UV_HANDLE_ZERO_READ)) {
Packit b5b901
    /* Successful read */
Packit b5b901
    partial = !REQ_SUCCESS(req);
Packit b5b901
    handle->recv_cb(handle,
Packit b5b901
                    req->u.io.overlapped.InternalHigh,
Packit b5b901
                    &handle->recv_buffer,
Packit b5b901
                    (const struct sockaddr*) &handle->recv_from,
Packit b5b901
                    partial ? UV_UDP_PARTIAL : 0);
Packit b5b901
  } else if (handle->flags & UV_HANDLE_READING) {
Packit b5b901
    DWORD bytes, err, flags;
Packit b5b901
    struct sockaddr_storage from;
Packit b5b901
    int from_len;
Packit b5b901
Packit b5b901
    /* Do a nonblocking receive.
Packit b5b901
     * TODO: try to read multiple datagrams at once. FIONREAD maybe? */
Packit b5b901
    buf = uv_buf_init(NULL, 0);
Packit b5b901
    handle->alloc_cb((uv_handle_t*) handle, 65536, &buf;;
Packit b5b901
    if (buf.base == NULL || buf.len == 0) {
Packit b5b901
      handle->recv_cb(handle, UV_ENOBUFS, &buf, NULL, 0);
Packit b5b901
      goto done;
Packit b5b901
    }
Packit b5b901
    assert(buf.base != NULL);
Packit b5b901
Packit b5b901
    memset(&from, 0, sizeof from);
Packit b5b901
    from_len = sizeof from;
Packit b5b901
Packit b5b901
    flags = 0;
Packit b5b901
Packit b5b901
    if (WSARecvFrom(handle->socket,
Packit b5b901
                    (WSABUF*)&buf,
Packit b5b901
                    1,
Packit b5b901
                    &bytes,
Packit b5b901
                    &flags,
Packit b5b901
                    (struct sockaddr*) &from,
Packit b5b901
                    &from_len,
Packit b5b901
                    NULL,
Packit b5b901
                    NULL) != SOCKET_ERROR) {
Packit b5b901
Packit b5b901
      /* Message received */
Packit b5b901
      handle->recv_cb(handle, bytes, &buf, (const struct sockaddr*) &from, 0);
Packit b5b901
    } else {
Packit b5b901
      err = WSAGetLastError();
Packit b5b901
      if (err == WSAEMSGSIZE) {
Packit b5b901
        /* Message truncated */
Packit b5b901
        handle->recv_cb(handle,
Packit b5b901
                        bytes,
Packit b5b901
                        &buf,
Packit b5b901
                        (const struct sockaddr*) &from,
Packit b5b901
                        UV_UDP_PARTIAL);
Packit b5b901
      } else if (err == WSAEWOULDBLOCK) {
Packit b5b901
        /* Kernel buffer empty */
Packit b5b901
        handle->recv_cb(handle, 0, &buf, NULL, 0);
Packit b5b901
      } else if (err == WSAECONNRESET || err == WSAENETRESET) {
Packit b5b901
        /* WSAECONNRESET/WSANETRESET is ignored because this just indicates
Packit b5b901
         * that a previous sendto operation failed.
Packit b5b901
         */
Packit b5b901
        handle->recv_cb(handle, 0, &buf, NULL, 0);
Packit b5b901
      } else {
Packit b5b901
        /* Any other error that we want to report back to the user. */
Packit b5b901
        uv_udp_recv_stop(handle);
Packit b5b901
        handle->recv_cb(handle, uv_translate_sys_error(err), &buf, NULL, 0);
Packit b5b901
      }
Packit b5b901
    }
Packit b5b901
  }
Packit b5b901
Packit b5b901
done:
Packit b5b901
  /* Post another read if still reading and not closing. */
Packit b5b901
  if ((handle->flags & UV_HANDLE_READING) &&
Packit b5b901
      !(handle->flags & UV_HANDLE_READ_PENDING)) {
Packit b5b901
    uv_udp_queue_recv(loop, handle);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  DECREASE_PENDING_REQ_COUNT(handle);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
void uv_process_udp_send_req(uv_loop_t* loop, uv_udp_t* handle,
Packit b5b901
    uv_udp_send_t* req) {
Packit b5b901
  int err;
Packit b5b901
Packit b5b901
  assert(handle->type == UV_UDP);
Packit b5b901
Packit b5b901
  assert(handle->send_queue_size >= req->u.io.queued_bytes);
Packit b5b901
  assert(handle->send_queue_count >= 1);
Packit b5b901
  handle->send_queue_size -= req->u.io.queued_bytes;
Packit b5b901
  handle->send_queue_count--;
Packit b5b901
Packit b5b901
  UNREGISTER_HANDLE_REQ(loop, handle, req);
Packit b5b901
Packit b5b901
  if (req->cb) {
Packit b5b901
    err = 0;
Packit b5b901
    if (!REQ_SUCCESS(req)) {
Packit b5b901
      err = GET_REQ_SOCK_ERROR(req);
Packit b5b901
    }
Packit b5b901
    req->cb(req, uv_translate_sys_error(err));
Packit b5b901
  }
Packit b5b901
Packit b5b901
  DECREASE_PENDING_REQ_COUNT(handle);
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
  int err;
Packit b5b901
  int optname;
Packit b5b901
  struct ip_mreq mreq;
Packit b5b901
Packit b5b901
  if (handle->flags & UV_HANDLE_IPV6)
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit b5b901
  /* If the socket is unbound, bind to inaddr_any. */
Packit b5b901
  err = uv_udp_maybe_bind(handle,
Packit b5b901
                          (const struct sockaddr*) &uv_addr_ip4_any_,
Packit b5b901
                          sizeof(uv_addr_ip4_any_),
Packit b5b901
                          UV_UDP_REUSEADDR);
Packit b5b901
  if (err)
Packit b5b901
    return uv_translate_sys_error(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->socket,
Packit b5b901
                 IPPROTO_IP,
Packit b5b901
                 optname,
Packit b5b901
                 (char*) &mreq,
Packit b5b901
                 sizeof mreq) == SOCKET_ERROR) {
Packit b5b901
    return uv_translate_sys_error(WSAGetLastError());
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
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
  int err;
Packit b5b901
  struct ipv6_mreq mreq;
Packit b5b901
  struct sockaddr_in6 addr6;
Packit b5b901
Packit b5b901
  if ((handle->flags & UV_HANDLE_BOUND) && !(handle->flags & UV_HANDLE_IPV6))
Packit b5b901
    return UV_EINVAL;
Packit b5b901
Packit b5b901
  err = uv_udp_maybe_bind(handle,
Packit b5b901
                          (const struct sockaddr*) &uv_addr_ip6_any_,
Packit b5b901
                          sizeof(uv_addr_ip6_any_),
Packit b5b901
                          UV_UDP_REUSEADDR);
Packit b5b901
Packit b5b901
  if (err)
Packit b5b901
    return uv_translate_sys_error(err);
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->socket,
Packit b5b901
                 IPPROTO_IPV6,
Packit b5b901
                 optname,
Packit b5b901
                 (char*) &mreq,
Packit b5b901
                 sizeof mreq) == SOCKET_ERROR) {
Packit b5b901
    return uv_translate_sys_error(WSAGetLastError());
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
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 Service e08953
  int err;
Packit Service e08953
Packit Service e08953
  if (handle->flags & UV_HANDLE_IPV6)
Packit Service e08953
    return UV_EINVAL;
Packit Service e08953
Packit Service e08953
  /* If the socket is unbound, bind to inaddr_any. */
Packit Service e08953
  err = uv_udp_maybe_bind(handle,
Packit Service e08953
                          (const struct sockaddr*) &uv_addr_ip4_any_,
Packit Service e08953
                          sizeof(uv_addr_ip4_any_),
Packit Service e08953
                          UV_UDP_REUSEADDR);
Packit Service e08953
  if (err)
Packit Service e08953
    return uv_translate_sys_error(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_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 Service e08953
    return UV_EINVAL;
Packit Service e08953
Packit Service e08953
  if (setsockopt(handle->socket,
Packit Service e08953
                 IPPROTO_IP,
Packit Service e08953
                 optname,
Packit Service e08953
                 (char*) &mreq,
Packit Service e08953
                 sizeof(mreq)) == SOCKET_ERROR) {
Packit Service e08953
    return uv_translate_sys_error(WSAGetLastError());
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  return 0;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
Packit Service e08953
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
  if ((handle->flags & UV_HANDLE_BOUND) && !(handle->flags & UV_HANDLE_IPV6))
Packit Service e08953
    return UV_EINVAL;
Packit Service e08953
Packit Service e08953
  err = uv_udp_maybe_bind(handle,
Packit Service e08953
                          (const struct sockaddr*) &uv_addr_ip6_any_,
Packit Service e08953
                          sizeof(uv_addr_ip6_any_),
Packit Service e08953
                          UV_UDP_REUSEADDR);
Packit Service e08953
Packit Service e08953
  if (err)
Packit Service e08953
    return uv_translate_sys_error(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 Service e08953
      return err;
Packit Service e08953
    mreq.gsr_interface = addr6.sin6_scope_id;
Packit Service e08953
  } 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->socket,
Packit Service e08953
                 IPPROTO_IPV6,
Packit Service e08953
                 optname,
Packit Service e08953
                 (char*) &mreq,
Packit Service e08953
                 sizeof(mreq)) == SOCKET_ERROR) {
Packit Service e08953
    return uv_translate_sys_error(WSAGetLastError());
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  return 0;
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
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
  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
    return uv__udp_set_membership4(handle, &addr4, interface_addr, membership);
Packit b5b901
  else if (uv_ip6_addr(multicast_addr, 0, &addr6) == 0)
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
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
  int err;
Packit Service e08953
  struct sockaddr_storage mcast_addr;
Packit Service e08953
  struct sockaddr_in* mcast_addr4;
Packit Service e08953
  struct sockaddr_in6* mcast_addr6;
Packit Service e08953
  struct sockaddr_storage src_addr;
Packit Service e08953
  struct sockaddr_in* src_addr4;
Packit Service e08953
  struct sockaddr_in6* src_addr6;
Packit Service e08953
Packit Service e08953
  mcast_addr4 = (struct sockaddr_in*)&mcast_addr;
Packit Service e08953
  mcast_addr6 = (struct sockaddr_in6*)&mcast_addr;
Packit Service e08953
  src_addr4 = (struct sockaddr_in*)&src_addr;
Packit Service e08953
  src_addr6 = (struct sockaddr_in6*)&src_addr;
Packit Service e08953
Packit Service e08953
  err = uv_ip4_addr(multicast_addr, 0, mcast_addr4);
Packit Service e08953
  if (err) {
Packit Service e08953
    err = uv_ip6_addr(multicast_addr, 0, mcast_addr6);
Packit Service e08953
    if (err)
Packit Service e08953
      return err;
Packit Service e08953
    err = uv_ip6_addr(source_addr, 0, src_addr6);
Packit Service e08953
    if (err)
Packit Service e08953
      return err;
Packit Service e08953
    return uv__udp_set_source_membership6(handle,
Packit Service e08953
                                          mcast_addr6,
Packit Service e08953
                                          interface_addr,
Packit Service e08953
                                          src_addr6,
Packit Service e08953
                                          membership);
Packit Service e08953
  }
Packit Service e08953
  
Packit Service e08953
  err = uv_ip4_addr(source_addr, 0, src_addr4);
Packit Service e08953
  if (err)
Packit Service e08953
    return err;
Packit Service e08953
  return uv__udp_set_source_membership4(handle,
Packit Service e08953
                                        mcast_addr4,
Packit Service e08953
                                        interface_addr,
Packit Service e08953
                                        src_addr4,
Packit Service e08953
                                        membership);
Packit Service e08953
}
Packit Service e08953
Packit Service e08953
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 (handle->socket == INVALID_SOCKET)
Packit b5b901
    return UV_EBADF;
Packit b5b901
Packit b5b901
  if (addr_st.ss_family == AF_INET) {
Packit b5b901
    if (setsockopt(handle->socket,
Packit b5b901
                   IPPROTO_IP,
Packit b5b901
                   IP_MULTICAST_IF,
Packit b5b901
                   (char*) &addr4->sin_addr,
Packit b5b901
                   sizeof(addr4->sin_addr)) == SOCKET_ERROR) {
Packit b5b901
      return uv_translate_sys_error(WSAGetLastError());
Packit b5b901
    }
Packit b5b901
  } else if (addr_st.ss_family == AF_INET6) {
Packit b5b901
    if (setsockopt(handle->socket,
Packit b5b901
                   IPPROTO_IPV6,
Packit b5b901
                   IPV6_MULTICAST_IF,
Packit b5b901
                   (char*) &addr6->sin6_scope_id,
Packit b5b901
                   sizeof(addr6->sin6_scope_id)) == SOCKET_ERROR) {
Packit b5b901
      return uv_translate_sys_error(WSAGetLastError());
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 b5b901
Packit b5b901
int uv_udp_set_broadcast(uv_udp_t* handle, int value) {
Packit b5b901
  BOOL optval = (BOOL) value;
Packit b5b901
Packit b5b901
  if (handle->socket == INVALID_SOCKET)
Packit b5b901
    return UV_EBADF;
Packit b5b901
Packit b5b901
  if (setsockopt(handle->socket,
Packit b5b901
                 SOL_SOCKET,
Packit b5b901
                 SO_BROADCAST,
Packit b5b901
                 (char*) &optval,
Packit b5b901
                 sizeof optval)) {
Packit b5b901
    return uv_translate_sys_error(WSAGetLastError());
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit Service e08953
int uv__udp_is_bound(uv_udp_t* handle) {
Packit Service e08953
  struct sockaddr_storage addr;
Packit Service e08953
  int addrlen;
Packit Service e08953
Packit Service e08953
  addrlen = sizeof(addr);
Packit Service e08953
  if (uv_udp_getsockname(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 b5b901
int uv_udp_open(uv_udp_t* handle, uv_os_sock_t sock) {
Packit b5b901
  WSAPROTOCOL_INFOW protocol_info;
Packit b5b901
  int opt_len;
Packit b5b901
  int err;
Packit b5b901
Packit b5b901
  /* Detect the address family of the socket. */
Packit b5b901
  opt_len = (int) sizeof protocol_info;
Packit b5b901
  if (getsockopt(sock,
Packit b5b901
                 SOL_SOCKET,
Packit b5b901
                 SO_PROTOCOL_INFOW,
Packit b5b901
                 (char*) &protocol_info,
Packit b5b901
                 &opt_len) == SOCKET_ERROR) {
Packit b5b901
    return uv_translate_sys_error(GetLastError());
Packit b5b901
  }
Packit b5b901
Packit b5b901
  err = uv_udp_set_socket(handle->loop,
Packit b5b901
                          handle,
Packit b5b901
                          sock,
Packit b5b901
                          protocol_info.iAddressFamily);
Packit Service e08953
  if (err)
Packit Service e08953
    return uv_translate_sys_error(err);
Packit Service e08953
Packit Service e08953
  if (uv__udp_is_bound(handle))
Packit Service e08953
    handle->flags |= UV_HANDLE_BOUND;
Packit Service e08953
Packit Service e08953
  if (uv__udp_is_connected(handle))
Packit Service e08953
    handle->flags |= UV_HANDLE_UDP_CONNECTED;
Packit Service e08953
Packit Service e08953
  return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
#define SOCKOPT_SETTER(name, option4, option6, validate)                      \
Packit b5b901
  int uv_udp_set_##name(uv_udp_t* handle, int value) {                        \
Packit b5b901
    DWORD optval = (DWORD) value;                                             \
Packit b5b901
                                                                              \
Packit b5b901
    if (!(validate(value))) {                                                 \
Packit b5b901
      return UV_EINVAL;                                                       \
Packit b5b901
    }                                                                         \
Packit b5b901
                                                                              \
Packit b5b901
    if (handle->socket == INVALID_SOCKET)                                     \
Packit b5b901
      return UV_EBADF;                                                        \
Packit b5b901
                                                                              \
Packit b5b901
    if (!(handle->flags & UV_HANDLE_IPV6)) {                                  \
Packit b5b901
      /* Set IPv4 socket option */                                            \
Packit b5b901
      if (setsockopt(handle->socket,                                          \
Packit b5b901
                     IPPROTO_IP,                                              \
Packit b5b901
                     option4,                                                 \
Packit b5b901
                     (char*) &optval,                                         \
Packit b5b901
                     sizeof optval)) {                                        \
Packit b5b901
        return uv_translate_sys_error(WSAGetLastError());                     \
Packit b5b901
      }                                                                       \
Packit b5b901
    } else {                                                                  \
Packit b5b901
      /* Set IPv6 socket option */                                            \
Packit b5b901
      if (setsockopt(handle->socket,                                          \
Packit b5b901
                     IPPROTO_IPV6,                                            \
Packit b5b901
                     option6,                                                 \
Packit b5b901
                     (char*) &optval,                                         \
Packit b5b901
                     sizeof optval)) {                                        \
Packit b5b901
        return uv_translate_sys_error(WSAGetLastError());                     \
Packit b5b901
      }                                                                       \
Packit b5b901
    }                                                                         \
Packit b5b901
    return 0;                                                                 \
Packit b5b901
  }
Packit b5b901
Packit b5b901
#define VALIDATE_TTL(value) ((value) >= 1 && (value) <= 255)
Packit b5b901
#define VALIDATE_MULTICAST_TTL(value) ((value) >= -1 && (value) <= 255)
Packit b5b901
#define VALIDATE_MULTICAST_LOOP(value) (1)
Packit b5b901
Packit b5b901
SOCKOPT_SETTER(ttl,
Packit b5b901
               IP_TTL,
Packit b5b901
               IPV6_HOPLIMIT,
Packit b5b901
               VALIDATE_TTL)
Packit b5b901
SOCKOPT_SETTER(multicast_ttl,
Packit b5b901
               IP_MULTICAST_TTL,
Packit b5b901
               IPV6_MULTICAST_HOPS,
Packit b5b901
               VALIDATE_MULTICAST_TTL)
Packit b5b901
SOCKOPT_SETTER(multicast_loop,
Packit b5b901
               IP_MULTICAST_LOOP,
Packit b5b901
               IPV6_MULTICAST_LOOP,
Packit b5b901
               VALIDATE_MULTICAST_LOOP)
Packit b5b901
Packit b5b901
#undef SOCKOPT_SETTER
Packit b5b901
#undef VALIDATE_TTL
Packit b5b901
#undef VALIDATE_MULTICAST_TTL
Packit b5b901
#undef VALIDATE_MULTICAST_LOOP
Packit b5b901
Packit b5b901
Packit b5b901
/* This function is an egress point, i.e. it returns libuv errors rather than
Packit b5b901
 * system errors.
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
Packit b5b901
  err = uv_udp_maybe_bind(handle, addr, addrlen, flags);
Packit b5b901
  if (err)
Packit b5b901
    return uv_translate_sys_error(err);
Packit b5b901
Packit b5b901
  return 0;
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
  const struct sockaddr* bind_addr;
Packit Service e08953
  int err;
Packit Service e08953
Packit Service e08953
  if (!(handle->flags & UV_HANDLE_BOUND)) {
Packit Service e08953
    if (addrlen == sizeof(uv_addr_ip4_any_))
Packit Service e08953
      bind_addr = (const struct sockaddr*) &uv_addr_ip4_any_;
Packit Service e08953
    else if (addrlen == sizeof(uv_addr_ip6_any_))
Packit Service e08953
      bind_addr = (const struct sockaddr*) &uv_addr_ip6_any_;
Packit Service e08953
    else
Packit Service e08953
      return UV_EINVAL;
Packit Service e08953
Packit Service e08953
    err = uv_udp_maybe_bind(handle, bind_addr, addrlen, 0);
Packit Service e08953
    if (err)
Packit Service e08953
      return uv_translate_sys_error(err);
Packit Service e08953
  }
Packit Service e08953
Packit Service e08953
  err = connect(handle->socket, addr, addrlen);
Packit Service e08953
  if (err)
Packit Service e08953
    return uv_translate_sys_error(err);
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 err;
Packit Service e08953
    struct sockaddr addr;
Packit Service e08953
Packit Service e08953
    memset(&addr, 0, sizeof(addr));
Packit Service e08953
Packit Service e08953
    err = connect(handle->socket, &addr, sizeof(addr));
Packit Service e08953
    if (err)
Packit Service e08953
      return uv_translate_sys_error(err);
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
/* This function is an egress point, i.e. it returns libuv errors rather than
Packit b5b901
 * system errors.
Packit b5b901
 */
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
  const struct sockaddr* bind_addr;
Packit b5b901
  int err;
Packit b5b901
Packit b5b901
  if (!(handle->flags & UV_HANDLE_BOUND)) {
Packit b5b901
    if (addrlen == sizeof(uv_addr_ip4_any_))
Packit b5b901
      bind_addr = (const struct sockaddr*) &uv_addr_ip4_any_;
Packit b5b901
    else if (addrlen == sizeof(uv_addr_ip6_any_))
Packit b5b901
      bind_addr = (const struct sockaddr*) &uv_addr_ip6_any_;
Packit b5b901
    else
Packit b5b901
      return UV_EINVAL;
Packit Service e08953
Packit b5b901
    err = uv_udp_maybe_bind(handle, bind_addr, addrlen, 0);
Packit b5b901
    if (err)
Packit b5b901
      return uv_translate_sys_error(err);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  err = uv__send(req, handle, bufs, nbufs, addr, addrlen, send_cb);
Packit b5b901
  if (err)
Packit b5b901
    return uv_translate_sys_error(err);
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
  DWORD bytes;
Packit b5b901
  const struct sockaddr* bind_addr;
Packit b5b901
  struct sockaddr_storage converted;
Packit b5b901
  int err;
Packit b5b901
Packit b5b901
  assert(nbufs > 0);
Packit b5b901
Packit Service e08953
  if (addr != NULL) {
Packit Service e08953
    err = uv__convert_to_localhost_if_unspecified(addr, &converted);
Packit Service e08953
    if (err)
Packit Service e08953
      return err;
Packit Service e08953
  }
Packit b5b901
Packit b5b901
  /* Already sending a message.*/
Packit b5b901
  if (handle->send_queue_count != 0)
Packit b5b901
    return UV_EAGAIN;
Packit b5b901
Packit b5b901
  if (!(handle->flags & UV_HANDLE_BOUND)) {
Packit b5b901
    if (addrlen == sizeof(uv_addr_ip4_any_))
Packit b5b901
      bind_addr = (const struct sockaddr*) &uv_addr_ip4_any_;
Packit b5b901
    else if (addrlen == sizeof(uv_addr_ip6_any_))
Packit b5b901
      bind_addr = (const struct sockaddr*) &uv_addr_ip6_any_;
Packit b5b901
    else
Packit b5b901
      return UV_EINVAL;
Packit b5b901
    err = uv_udp_maybe_bind(handle, bind_addr, addrlen, 0);
Packit b5b901
    if (err)
Packit b5b901
      return uv_translate_sys_error(err);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  err = WSASendTo(handle->socket,
Packit b5b901
                  (WSABUF*)bufs,
Packit b5b901
                  nbufs,
Packit b5b901
                  &bytes,
Packit b5b901
                  0,
Packit b5b901
                  (const struct sockaddr*) &converted,
Packit b5b901
                  addrlen,
Packit b5b901
                  NULL,
Packit b5b901
                  NULL);
Packit b5b901
Packit b5b901
  if (err)
Packit b5b901
    return uv_translate_sys_error(WSAGetLastError());
Packit b5b901
Packit b5b901
  return bytes;
Packit b5b901
}