Blame src/win/pipe.c

Packit Service 7c31a4
/* Copyright Joyent, Inc. and other Node contributors. All rights reserved.
Packit Service 7c31a4
 *
Packit Service 7c31a4
 * Permission is hereby granted, free of charge, to any person obtaining a copy
Packit Service 7c31a4
 * of this software and associated documentation files (the "Software"), to
Packit Service 7c31a4
 * deal in the Software without restriction, including without limitation the
Packit Service 7c31a4
 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
Packit Service 7c31a4
 * sell copies of the Software, and to permit persons to whom the Software is
Packit Service 7c31a4
 * furnished to do so, subject to the following conditions:
Packit Service 7c31a4
 *
Packit Service 7c31a4
 * The above copyright notice and this permission notice shall be included in
Packit Service 7c31a4
 * all copies or substantial portions of the Software.
Packit Service 7c31a4
 *
Packit Service 7c31a4
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
Packit Service 7c31a4
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
Packit Service 7c31a4
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
Packit Service 7c31a4
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
Packit Service 7c31a4
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
Packit Service 7c31a4
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
Packit Service 7c31a4
 * IN THE SOFTWARE.
Packit Service 7c31a4
 */
Packit Service 7c31a4
Packit Service 7c31a4
#include <assert.h>
Packit Service 7c31a4
#include <io.h>
Packit Service 7c31a4
#include <stdio.h>
Packit Service 7c31a4
#include <stdlib.h>
Packit Service 7c31a4
#include <string.h>
Packit Service 7c31a4
Packit Service 7c31a4
#include "handle-inl.h"
Packit Service 7c31a4
#include "internal.h"
Packit Service 7c31a4
#include "req-inl.h"
Packit Service 7c31a4
#include "stream-inl.h"
Packit Service 7c31a4
#include "uv-common.h"
Packit Service 7c31a4
#include "uv.h"
Packit Service 7c31a4
Packit Service 7c31a4
#include <aclapi.h>
Packit Service 7c31a4
#include <accctrl.h>
Packit Service 7c31a4
Packit Service 7c31a4
/* A zero-size buffer for use by uv_pipe_read */
Packit Service 7c31a4
static char uv_zero_[] = "";
Packit Service 7c31a4
Packit Service 7c31a4
/* Null uv_buf_t */
Packit Service 7c31a4
static const uv_buf_t uv_null_buf_ = { 0, NULL };
Packit Service 7c31a4
Packit Service 7c31a4
/* The timeout that the pipe will wait for the remote end to write data when
Packit Service 7c31a4
 * the local ends wants to shut it down. */
Packit Service 7c31a4
static const int64_t eof_timeout = 50; /* ms */
Packit Service 7c31a4
Packit Service 7c31a4
static const int default_pending_pipe_instances = 4;
Packit Service 7c31a4
Packit Service 7c31a4
/* Pipe prefix */
Packit Service 7c31a4
static char pipe_prefix[] = "\\\\?\\pipe";
Packit Service 7c31a4
static const int pipe_prefix_len = sizeof(pipe_prefix) - 1;
Packit Service 7c31a4
Packit Service 7c31a4
/* IPC incoming xfer queue item. */
Packit Service 7c31a4
typedef struct {
Packit Service 7c31a4
  uv__ipc_socket_xfer_type_t xfer_type;
Packit Service 7c31a4
  uv__ipc_socket_xfer_info_t xfer_info;
Packit Service 7c31a4
  QUEUE member;
Packit Service 7c31a4
} uv__ipc_xfer_queue_item_t;
Packit Service 7c31a4
Packit Service 7c31a4
/* IPC frame header flags. */
Packit Service 7c31a4
/* clang-format off */
Packit Service 7c31a4
enum {
Packit Service 7c31a4
  UV__IPC_FRAME_HAS_DATA                = 0x01,
Packit Service 7c31a4
  UV__IPC_FRAME_HAS_SOCKET_XFER         = 0x02,
Packit Service 7c31a4
  UV__IPC_FRAME_XFER_IS_TCP_CONNECTION  = 0x04,
Packit Service 7c31a4
  /* These are combinations of the flags above. */
Packit Service 7c31a4
  UV__IPC_FRAME_XFER_FLAGS              = 0x06,
Packit Service 7c31a4
  UV__IPC_FRAME_VALID_FLAGS             = 0x07
Packit Service 7c31a4
};
Packit Service 7c31a4
/* clang-format on */
Packit Service 7c31a4
Packit Service 7c31a4
/* IPC frame header. */
Packit Service 7c31a4
typedef struct {
Packit Service 7c31a4
  uint32_t flags;
Packit Service 7c31a4
  uint32_t reserved1;   /* Ignored. */
Packit Service 7c31a4
  uint32_t data_length; /* Must be zero if there is no data. */
Packit Service 7c31a4
  uint32_t reserved2;   /* Must be zero. */
Packit Service 7c31a4
} uv__ipc_frame_header_t;
Packit Service 7c31a4
Packit Service 7c31a4
/* To implement the IPC protocol correctly, these structures must have exactly
Packit Service 7c31a4
 * the right size. */
Packit Service 7c31a4
STATIC_ASSERT(sizeof(uv__ipc_frame_header_t) == 16);
Packit Service 7c31a4
STATIC_ASSERT(sizeof(uv__ipc_socket_xfer_info_t) == 632);
Packit Service 7c31a4
Packit Service 7c31a4
/* Coalesced write request. */
Packit Service 7c31a4
typedef struct {
Packit Service 7c31a4
  uv_write_t req;       /* Internal heap-allocated write request. */
Packit Service 7c31a4
  uv_write_t* user_req; /* Pointer to user-specified uv_write_t. */
Packit Service 7c31a4
} uv__coalesced_write_t;
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void eof_timer_init(uv_pipe_t* pipe);
Packit Service 7c31a4
static void eof_timer_start(uv_pipe_t* pipe);
Packit Service 7c31a4
static void eof_timer_stop(uv_pipe_t* pipe);
Packit Service 7c31a4
static void eof_timer_cb(uv_timer_t* timer);
Packit Service 7c31a4
static void eof_timer_destroy(uv_pipe_t* pipe);
Packit Service 7c31a4
static void eof_timer_close_cb(uv_handle_t* handle);
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void uv_unique_pipe_name(char* ptr, char* name, size_t size) {
Packit Service 7c31a4
  snprintf(name, size, "\\\\?\\pipe\\uv\\%p-%lu", ptr, GetCurrentProcessId());
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
int uv_pipe_init(uv_loop_t* loop, uv_pipe_t* handle, int ipc) {
Packit Service 7c31a4
  uv_stream_init(loop, (uv_stream_t*)handle, UV_NAMED_PIPE);
Packit Service 7c31a4
Packit Service 7c31a4
  handle->reqs_pending = 0;
Packit Service 7c31a4
  handle->handle = INVALID_HANDLE_VALUE;
Packit Service 7c31a4
  handle->name = NULL;
Packit Service 7c31a4
  handle->pipe.conn.ipc_remote_pid = 0;
Packit Service 7c31a4
  handle->pipe.conn.ipc_data_frame.payload_remaining = 0;
Packit Service 7c31a4
  QUEUE_INIT(&handle->pipe.conn.ipc_xfer_queue);
Packit Service 7c31a4
  handle->pipe.conn.ipc_xfer_queue_length = 0;
Packit Service 7c31a4
  handle->ipc = ipc;
Packit Service 7c31a4
  handle->pipe.conn.non_overlapped_writes_tail = NULL;
Packit Service 7c31a4
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void uv_pipe_connection_init(uv_pipe_t* handle) {
Packit Service 7c31a4
  uv_connection_init((uv_stream_t*) handle);
Packit Service 7c31a4
  handle->read_req.data = handle;
Packit Service 7c31a4
  handle->pipe.conn.eof_timer = NULL;
Packit Service 7c31a4
  assert(!(handle->flags & UV_HANDLE_PIPESERVER));
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_NON_OVERLAPPED_PIPE) {
Packit Service 7c31a4
    handle->pipe.conn.readfile_thread_handle = NULL;
Packit Service 7c31a4
    InitializeCriticalSection(&handle->pipe.conn.readfile_thread_lock);
Packit Service 7c31a4
  }
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static HANDLE open_named_pipe(const WCHAR* name, DWORD* duplex_flags) {
Packit Service 7c31a4
  HANDLE pipeHandle;
Packit Service 7c31a4
Packit Service 7c31a4
  /*
Packit Service 7c31a4
   * Assume that we have a duplex pipe first, so attempt to
Packit Service 7c31a4
   * connect with GENERIC_READ | GENERIC_WRITE.
Packit Service 7c31a4
   */
Packit Service 7c31a4
  pipeHandle = CreateFileW(name,
Packit Service 7c31a4
                           GENERIC_READ | GENERIC_WRITE,
Packit Service 7c31a4
                           0,
Packit Service 7c31a4
                           NULL,
Packit Service 7c31a4
                           OPEN_EXISTING,
Packit Service 7c31a4
                           FILE_FLAG_OVERLAPPED,
Packit Service 7c31a4
                           NULL);
Packit Service 7c31a4
  if (pipeHandle != INVALID_HANDLE_VALUE) {
Packit Service 7c31a4
    *duplex_flags = UV_HANDLE_READABLE | UV_HANDLE_WRITABLE;
Packit Service 7c31a4
    return pipeHandle;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /*
Packit Service 7c31a4
   * If the pipe is not duplex CreateFileW fails with
Packit Service 7c31a4
   * ERROR_ACCESS_DENIED.  In that case try to connect
Packit Service 7c31a4
   * as a read-only or write-only.
Packit Service 7c31a4
   */
Packit Service 7c31a4
  if (GetLastError() == ERROR_ACCESS_DENIED) {
Packit Service 7c31a4
    pipeHandle = CreateFileW(name,
Packit Service 7c31a4
                             GENERIC_READ | FILE_WRITE_ATTRIBUTES,
Packit Service 7c31a4
                             0,
Packit Service 7c31a4
                             NULL,
Packit Service 7c31a4
                             OPEN_EXISTING,
Packit Service 7c31a4
                             FILE_FLAG_OVERLAPPED,
Packit Service 7c31a4
                             NULL);
Packit Service 7c31a4
Packit Service 7c31a4
    if (pipeHandle != INVALID_HANDLE_VALUE) {
Packit Service 7c31a4
      *duplex_flags = UV_HANDLE_READABLE;
Packit Service 7c31a4
      return pipeHandle;
Packit Service 7c31a4
    }
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (GetLastError() == ERROR_ACCESS_DENIED) {
Packit Service 7c31a4
    pipeHandle = CreateFileW(name,
Packit Service 7c31a4
                             GENERIC_WRITE | FILE_READ_ATTRIBUTES,
Packit Service 7c31a4
                             0,
Packit Service 7c31a4
                             NULL,
Packit Service 7c31a4
                             OPEN_EXISTING,
Packit Service 7c31a4
                             FILE_FLAG_OVERLAPPED,
Packit Service 7c31a4
                             NULL);
Packit Service 7c31a4
Packit Service 7c31a4
    if (pipeHandle != INVALID_HANDLE_VALUE) {
Packit Service 7c31a4
      *duplex_flags = UV_HANDLE_WRITABLE;
Packit Service 7c31a4
      return pipeHandle;
Packit Service 7c31a4
    }
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  return INVALID_HANDLE_VALUE;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void close_pipe(uv_pipe_t* pipe) {
Packit Service 7c31a4
  assert(pipe->u.fd == -1 || pipe->u.fd > 2);
Packit Service 7c31a4
  if (pipe->u.fd == -1)
Packit Service 7c31a4
    CloseHandle(pipe->handle);
Packit Service 7c31a4
  else
Packit Service 7c31a4
    close(pipe->u.fd);
Packit Service 7c31a4
Packit Service 7c31a4
  pipe->u.fd = -1;
Packit Service 7c31a4
  pipe->handle = INVALID_HANDLE_VALUE;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
int uv_stdio_pipe_server(uv_loop_t* loop, uv_pipe_t* handle, DWORD access,
Packit Service 7c31a4
    char* name, size_t nameSize) {
Packit Service 7c31a4
  HANDLE pipeHandle;
Packit Service 7c31a4
  int err;
Packit Service 7c31a4
  char* ptr = (char*)handle;
Packit Service 7c31a4
Packit Service 7c31a4
  for (;;) {
Packit Service 7c31a4
    uv_unique_pipe_name(ptr, name, nameSize);
Packit Service 7c31a4
Packit Service 7c31a4
    pipeHandle = CreateNamedPipeA(name,
Packit Service 7c31a4
      access | FILE_FLAG_OVERLAPPED | FILE_FLAG_FIRST_PIPE_INSTANCE | WRITE_DAC,
Packit Service 7c31a4
      PIPE_TYPE_BYTE | PIPE_READMODE_BYTE | PIPE_WAIT, 1, 65536, 65536, 0,
Packit Service 7c31a4
      NULL);
Packit Service 7c31a4
Packit Service 7c31a4
    if (pipeHandle != INVALID_HANDLE_VALUE) {
Packit Service 7c31a4
      /* No name collisions.  We're done. */
Packit Service 7c31a4
      break;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    err = GetLastError();
Packit Service 7c31a4
    if (err != ERROR_PIPE_BUSY && err != ERROR_ACCESS_DENIED) {
Packit Service 7c31a4
      goto error;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    /* Pipe name collision.  Increment the pointer and try again. */
Packit Service 7c31a4
    ptr++;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (CreateIoCompletionPort(pipeHandle,
Packit Service 7c31a4
                             loop->iocp,
Packit Service 7c31a4
                             (ULONG_PTR)handle,
Packit Service 7c31a4
                             0) == NULL) {
Packit Service 7c31a4
    err = GetLastError();
Packit Service 7c31a4
    goto error;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  uv_pipe_connection_init(handle);
Packit Service 7c31a4
  handle->handle = pipeHandle;
Packit Service 7c31a4
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
Packit Service 7c31a4
 error:
Packit Service e2ebee
  if (pipeHandle != INVALID_HANDLE_VALUE)
Packit Service 7c31a4
    CloseHandle(pipeHandle);
Packit Service 7c31a4
Packit Service 7c31a4
  return err;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static int uv_set_pipe_handle(uv_loop_t* loop,
Packit Service 7c31a4
                              uv_pipe_t* handle,
Packit Service 7c31a4
                              HANDLE pipeHandle,
Packit Service 7c31a4
                              int fd,
Packit Service 7c31a4
                              DWORD duplex_flags) {
Packit Service 7c31a4
  NTSTATUS nt_status;
Packit Service 7c31a4
  IO_STATUS_BLOCK io_status;
Packit Service 7c31a4
  FILE_MODE_INFORMATION mode_info;
Packit Service 7c31a4
  DWORD mode = PIPE_READMODE_BYTE | PIPE_WAIT;
Packit Service 7c31a4
  DWORD current_mode = 0;
Packit Service 7c31a4
  DWORD err = 0;
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_PIPESERVER)
Packit Service 7c31a4
    return UV_EINVAL;
Packit Service 7c31a4
  if (handle->handle != INVALID_HANDLE_VALUE)
Packit Service 7c31a4
    return UV_EBUSY;
Packit Service 7c31a4
Packit Service 7c31a4
  if (!SetNamedPipeHandleState(pipeHandle, &mode, NULL, NULL)) {
Packit Service 7c31a4
    err = GetLastError();
Packit Service 7c31a4
    if (err == ERROR_ACCESS_DENIED) {
Packit Service 7c31a4
      /*
Packit Service 7c31a4
       * SetNamedPipeHandleState can fail if the handle doesn't have either
Packit Service 7c31a4
       * GENERIC_WRITE  or FILE_WRITE_ATTRIBUTES.
Packit Service 7c31a4
       * But if the handle already has the desired wait and blocking modes
Packit Service 7c31a4
       * we can continue.
Packit Service 7c31a4
       */
Packit Service 7c31a4
      if (!GetNamedPipeHandleState(pipeHandle, &current_mode, NULL, NULL,
Packit Service 7c31a4
                                   NULL, NULL, 0)) {
Packit Service 7c31a4
        return -1;
Packit Service 7c31a4
      } else if (current_mode & PIPE_NOWAIT) {
Packit Service 7c31a4
        SetLastError(ERROR_ACCESS_DENIED);
Packit Service 7c31a4
        return -1;
Packit Service 7c31a4
      }
Packit Service 7c31a4
    } else {
Packit Service 7c31a4
      /* If this returns ERROR_INVALID_PARAMETER we probably opened
Packit Service 7c31a4
       * something that is not a pipe. */
Packit Service 7c31a4
      if (err == ERROR_INVALID_PARAMETER) {
Packit Service 7c31a4
        SetLastError(WSAENOTSOCK);
Packit Service 7c31a4
      }
Packit Service 7c31a4
      return -1;
Packit Service 7c31a4
    }
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* Check if the pipe was created with FILE_FLAG_OVERLAPPED. */
Packit Service 7c31a4
  nt_status = pNtQueryInformationFile(pipeHandle,
Packit Service 7c31a4
                                      &io_status,
Packit Service 7c31a4
                                      &mode_info,
Packit Service 7c31a4
                                      sizeof(mode_info),
Packit Service 7c31a4
                                      FileModeInformation);
Packit Service 7c31a4
  if (nt_status != STATUS_SUCCESS) {
Packit Service 7c31a4
    return -1;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (mode_info.Mode & FILE_SYNCHRONOUS_IO_ALERT ||
Packit Service 7c31a4
      mode_info.Mode & FILE_SYNCHRONOUS_IO_NONALERT) {
Packit Service 7c31a4
    /* Non-overlapped pipe. */
Packit Service 7c31a4
    handle->flags |= UV_HANDLE_NON_OVERLAPPED_PIPE;
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    /* Overlapped pipe.  Try to associate with IOCP. */
Packit Service 7c31a4
    if (CreateIoCompletionPort(pipeHandle,
Packit Service 7c31a4
                               loop->iocp,
Packit Service 7c31a4
                               (ULONG_PTR) handle,
Packit Service 7c31a4
                               0) == NULL) {
Packit Service 7c31a4
      handle->flags |= UV_HANDLE_EMULATE_IOCP;
Packit Service 7c31a4
    }
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  handle->handle = pipeHandle;
Packit Service 7c31a4
  handle->u.fd = fd;
Packit Service 7c31a4
  handle->flags |= duplex_flags;
Packit Service 7c31a4
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static int pipe_alloc_accept(uv_loop_t* loop, uv_pipe_t* handle,
Packit Service 7c31a4
                             uv_pipe_accept_t* req, BOOL firstInstance) {
Packit Service 7c31a4
  assert(req->pipeHandle == INVALID_HANDLE_VALUE);
Packit Service 7c31a4
Packit Service 7c31a4
  req->pipeHandle =
Packit Service 7c31a4
      CreateNamedPipeW(handle->name,
Packit Service 7c31a4
                       PIPE_ACCESS_DUPLEX | FILE_FLAG_OVERLAPPED | WRITE_DAC |
Packit Service 7c31a4
                         (firstInstance ? FILE_FLAG_FIRST_PIPE_INSTANCE : 0),
Packit Service 7c31a4
                       PIPE_TYPE_BYTE | PIPE_READMODE_BYTE | PIPE_WAIT,
Packit Service 7c31a4
                       PIPE_UNLIMITED_INSTANCES, 65536, 65536, 0, NULL);
Packit Service 7c31a4
Packit Service 7c31a4
  if (req->pipeHandle == INVALID_HANDLE_VALUE) {
Packit Service 7c31a4
    return 0;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* Associate it with IOCP so we can get events. */
Packit Service 7c31a4
  if (CreateIoCompletionPort(req->pipeHandle,
Packit Service 7c31a4
                             loop->iocp,
Packit Service 7c31a4
                             (ULONG_PTR) handle,
Packit Service 7c31a4
                             0) == NULL) {
Packit Service 7c31a4
    uv_fatal_error(GetLastError(), "CreateIoCompletionPort");
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* Stash a handle in the server object for use from places such as
Packit Service 7c31a4
   * getsockname and chmod. As we transfer ownership of these to client
Packit Service 7c31a4
   * objects, we'll allocate new ones here. */
Packit Service 7c31a4
  handle->handle = req->pipeHandle;
Packit Service 7c31a4
Packit Service 7c31a4
  return 1;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static DWORD WINAPI pipe_shutdown_thread_proc(void* parameter) {
Packit Service 7c31a4
  uv_loop_t* loop;
Packit Service 7c31a4
  uv_pipe_t* handle;
Packit Service 7c31a4
  uv_shutdown_t* req;
Packit Service 7c31a4
Packit Service 7c31a4
  req = (uv_shutdown_t*) parameter;
Packit Service 7c31a4
  assert(req);
Packit Service 7c31a4
  handle = (uv_pipe_t*) req->handle;
Packit Service 7c31a4
  assert(handle);
Packit Service 7c31a4
  loop = handle->loop;
Packit Service 7c31a4
  assert(loop);
Packit Service 7c31a4
Packit Service 7c31a4
  FlushFileBuffers(handle->handle);
Packit Service 7c31a4
Packit Service 7c31a4
  /* Post completed */
Packit Service 7c31a4
  POST_COMPLETION_FOR_REQ(loop, req);
Packit Service 7c31a4
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
void uv_pipe_endgame(uv_loop_t* loop, uv_pipe_t* handle) {
Packit Service 7c31a4
  int err;
Packit Service 7c31a4
  DWORD result;
Packit Service 7c31a4
  uv_shutdown_t* req;
Packit Service 7c31a4
  NTSTATUS nt_status;
Packit Service 7c31a4
  IO_STATUS_BLOCK io_status;
Packit Service 7c31a4
  FILE_PIPE_LOCAL_INFORMATION pipe_info;
Packit Service 7c31a4
  uv__ipc_xfer_queue_item_t* xfer_queue_item;
Packit Service 7c31a4
Packit Service 7c31a4
  if ((handle->flags & UV_HANDLE_CONNECTION) &&
Packit Service 7c31a4
      handle->stream.conn.shutdown_req != NULL &&
Packit Service 7c31a4
      handle->stream.conn.write_reqs_pending == 0) {
Packit Service 7c31a4
    req = handle->stream.conn.shutdown_req;
Packit Service 7c31a4
Packit Service 7c31a4
    /* Clear the shutdown_req field so we don't go here again. */
Packit Service 7c31a4
    handle->stream.conn.shutdown_req = NULL;
Packit Service 7c31a4
Packit Service 7c31a4
    if (handle->flags & UV_HANDLE_CLOSING) {
Packit Service 7c31a4
      UNREGISTER_HANDLE_REQ(loop, handle, req);
Packit Service 7c31a4
Packit Service 7c31a4
      /* Already closing. Cancel the shutdown. */
Packit Service 7c31a4
      if (req->cb) {
Packit Service 7c31a4
        req->cb(req, UV_ECANCELED);
Packit Service 7c31a4
      }
Packit Service 7c31a4
Packit Service 7c31a4
      DECREASE_PENDING_REQ_COUNT(handle);
Packit Service 7c31a4
      return;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    /* Try to avoid flushing the pipe buffer in the thread pool. */
Packit Service 7c31a4
    nt_status = pNtQueryInformationFile(handle->handle,
Packit Service 7c31a4
                                        &io_status,
Packit Service 7c31a4
                                        &pipe_info,
Packit Service 7c31a4
                                        sizeof pipe_info,
Packit Service 7c31a4
                                        FilePipeLocalInformation);
Packit Service 7c31a4
Packit Service 7c31a4
    if (nt_status != STATUS_SUCCESS) {
Packit Service 7c31a4
      /* Failure */
Packit Service 7c31a4
      UNREGISTER_HANDLE_REQ(loop, handle, req);
Packit Service 7c31a4
Packit Service 7c31a4
      handle->flags |= UV_HANDLE_WRITABLE; /* Questionable */
Packit Service 7c31a4
      if (req->cb) {
Packit Service 7c31a4
        err = pRtlNtStatusToDosError(nt_status);
Packit Service 7c31a4
        req->cb(req, uv_translate_sys_error(err));
Packit Service 7c31a4
      }
Packit Service 7c31a4
Packit Service 7c31a4
      DECREASE_PENDING_REQ_COUNT(handle);
Packit Service 7c31a4
      return;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    if (pipe_info.OutboundQuota == pipe_info.WriteQuotaAvailable) {
Packit Service 7c31a4
      /* Short-circuit, no need to call FlushFileBuffers. */
Packit Service 7c31a4
      uv_insert_pending_req(loop, (uv_req_t*) req);
Packit Service 7c31a4
      return;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    /* Run FlushFileBuffers in the thread pool. */
Packit Service 7c31a4
    result = QueueUserWorkItem(pipe_shutdown_thread_proc,
Packit Service 7c31a4
                               req,
Packit Service 7c31a4
                               WT_EXECUTELONGFUNCTION);
Packit Service 7c31a4
    if (result) {
Packit Service 7c31a4
      return;
Packit Service 7c31a4
Packit Service 7c31a4
    } else {
Packit Service 7c31a4
      /* Failure. */
Packit Service 7c31a4
      UNREGISTER_HANDLE_REQ(loop, handle, req);
Packit Service 7c31a4
Packit Service 7c31a4
      handle->flags |= UV_HANDLE_WRITABLE; /* Questionable */
Packit Service 7c31a4
      if (req->cb) {
Packit Service 7c31a4
        err = GetLastError();
Packit Service 7c31a4
        req->cb(req, uv_translate_sys_error(err));
Packit Service 7c31a4
      }
Packit Service 7c31a4
Packit Service 7c31a4
      DECREASE_PENDING_REQ_COUNT(handle);
Packit Service 7c31a4
      return;
Packit Service 7c31a4
    }
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_CLOSING &&
Packit Service 7c31a4
      handle->reqs_pending == 0) {
Packit Service 7c31a4
    assert(!(handle->flags & UV_HANDLE_CLOSED));
Packit Service 7c31a4
Packit Service 7c31a4
    if (handle->flags & UV_HANDLE_CONNECTION) {
Packit Service 7c31a4
      /* Free pending sockets */
Packit Service 7c31a4
      while (!QUEUE_EMPTY(&handle->pipe.conn.ipc_xfer_queue)) {
Packit Service 7c31a4
        QUEUE* q;
Packit Service 7c31a4
        SOCKET socket;
Packit Service 7c31a4
Packit Service 7c31a4
        q = QUEUE_HEAD(&handle->pipe.conn.ipc_xfer_queue);
Packit Service 7c31a4
        QUEUE_REMOVE(q);
Packit Service 7c31a4
        xfer_queue_item = QUEUE_DATA(q, uv__ipc_xfer_queue_item_t, member);
Packit Service 7c31a4
Packit Service 7c31a4
        /* Materialize socket and close it */
Packit Service 7c31a4
        socket = WSASocketW(FROM_PROTOCOL_INFO,
Packit Service 7c31a4
                            FROM_PROTOCOL_INFO,
Packit Service 7c31a4
                            FROM_PROTOCOL_INFO,
Packit Service 7c31a4
                            &xfer_queue_item->xfer_info.socket_info,
Packit Service 7c31a4
                            0,
Packit Service 7c31a4
                            WSA_FLAG_OVERLAPPED);
Packit Service 7c31a4
        uv__free(xfer_queue_item);
Packit Service 7c31a4
Packit Service 7c31a4
        if (socket != INVALID_SOCKET)
Packit Service 7c31a4
          closesocket(socket);
Packit Service 7c31a4
      }
Packit Service 7c31a4
      handle->pipe.conn.ipc_xfer_queue_length = 0;
Packit Service 7c31a4
Packit Service 7c31a4
      if (handle->flags & UV_HANDLE_EMULATE_IOCP) {
Packit Service 7c31a4
        if (handle->read_req.wait_handle != INVALID_HANDLE_VALUE) {
Packit Service 7c31a4
          UnregisterWait(handle->read_req.wait_handle);
Packit Service 7c31a4
          handle->read_req.wait_handle = INVALID_HANDLE_VALUE;
Packit Service 7c31a4
        }
Packit Service 7c31a4
        if (handle->read_req.event_handle != NULL) {
Packit Service 7c31a4
          CloseHandle(handle->read_req.event_handle);
Packit Service 7c31a4
          handle->read_req.event_handle = NULL;
Packit Service 7c31a4
        }
Packit Service 7c31a4
      }
Packit Service 7c31a4
Packit Service 7c31a4
      if (handle->flags & UV_HANDLE_NON_OVERLAPPED_PIPE)
Packit Service 7c31a4
        DeleteCriticalSection(&handle->pipe.conn.readfile_thread_lock);
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    if (handle->flags & UV_HANDLE_PIPESERVER) {
Packit Service 7c31a4
      assert(handle->pipe.serv.accept_reqs);
Packit Service 7c31a4
      uv__free(handle->pipe.serv.accept_reqs);
Packit Service 7c31a4
      handle->pipe.serv.accept_reqs = NULL;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    uv__handle_close(handle);
Packit Service 7c31a4
  }
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
void uv_pipe_pending_instances(uv_pipe_t* handle, int count) {
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_BOUND)
Packit Service 7c31a4
    return;
Packit Service 7c31a4
  handle->pipe.serv.pending_instances = count;
Packit Service 7c31a4
  handle->flags |= UV_HANDLE_PIPESERVER;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
/* Creates a pipe server. */
Packit Service 7c31a4
int uv_pipe_bind(uv_pipe_t* handle, const char* name) {
Packit Service 7c31a4
  uv_loop_t* loop = handle->loop;
Packit Service 7c31a4
  int i, err, nameSize;
Packit Service 7c31a4
  uv_pipe_accept_t* req;
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_BOUND) {
Packit Service 7c31a4
    return UV_EINVAL;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (!name) {
Packit Service 7c31a4
    return UV_EINVAL;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (!(handle->flags & UV_HANDLE_PIPESERVER)) {
Packit Service 7c31a4
    handle->pipe.serv.pending_instances = default_pending_pipe_instances;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  handle->pipe.serv.accept_reqs = (uv_pipe_accept_t*)
Packit Service 7c31a4
    uv__malloc(sizeof(uv_pipe_accept_t) * handle->pipe.serv.pending_instances);
Packit Service 7c31a4
  if (!handle->pipe.serv.accept_reqs) {
Packit Service 7c31a4
    uv_fatal_error(ERROR_OUTOFMEMORY, "uv__malloc");
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  for (i = 0; i < handle->pipe.serv.pending_instances; i++) {
Packit Service 7c31a4
    req = &handle->pipe.serv.accept_reqs[i];
Packit Service 7c31a4
    UV_REQ_INIT(req, UV_ACCEPT);
Packit Service 7c31a4
    req->data = handle;
Packit Service 7c31a4
    req->pipeHandle = INVALID_HANDLE_VALUE;
Packit Service 7c31a4
    req->next_pending = NULL;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* Convert name to UTF16. */
Packit Service 7c31a4
  nameSize = MultiByteToWideChar(CP_UTF8, 0, name, -1, NULL, 0) * sizeof(WCHAR);
Packit Service e2ebee
  handle->name = uv__malloc(nameSize);
Packit Service 7c31a4
  if (!handle->name) {
Packit Service 7c31a4
    uv_fatal_error(ERROR_OUTOFMEMORY, "uv__malloc");
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (!MultiByteToWideChar(CP_UTF8,
Packit Service 7c31a4
                           0,
Packit Service 7c31a4
                           name,
Packit Service 7c31a4
                           -1,
Packit Service 7c31a4
                           handle->name,
Packit Service 7c31a4
                           nameSize / sizeof(WCHAR))) {
Packit Service 7c31a4
    err = GetLastError();
Packit Service 7c31a4
    goto error;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /*
Packit Service 7c31a4
   * Attempt to create the first pipe with FILE_FLAG_FIRST_PIPE_INSTANCE.
Packit Service 7c31a4
   * If this fails then there's already a pipe server for the given pipe name.
Packit Service 7c31a4
   */
Packit Service 7c31a4
  if (!pipe_alloc_accept(loop,
Packit Service 7c31a4
                         handle,
Packit Service 7c31a4
                         &handle->pipe.serv.accept_reqs[0],
Packit Service 7c31a4
                         TRUE)) {
Packit Service 7c31a4
    err = GetLastError();
Packit Service 7c31a4
    if (err == ERROR_ACCESS_DENIED) {
Packit Service 7c31a4
      err = WSAEADDRINUSE;  /* Translates to UV_EADDRINUSE. */
Packit Service 7c31a4
    } else if (err == ERROR_PATH_NOT_FOUND || err == ERROR_INVALID_NAME) {
Packit Service 7c31a4
      err = WSAEACCES;  /* Translates to UV_EACCES. */
Packit Service 7c31a4
    }
Packit Service 7c31a4
    goto error;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  handle->pipe.serv.pending_accepts = NULL;
Packit Service 7c31a4
  handle->flags |= UV_HANDLE_PIPESERVER;
Packit Service 7c31a4
  handle->flags |= UV_HANDLE_BOUND;
Packit Service 7c31a4
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
Packit Service 7c31a4
error:
Packit Service 7c31a4
  if (handle->name) {
Packit Service 7c31a4
    uv__free(handle->name);
Packit Service 7c31a4
    handle->name = NULL;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  return uv_translate_sys_error(err);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static DWORD WINAPI pipe_connect_thread_proc(void* parameter) {
Packit Service 7c31a4
  uv_loop_t* loop;
Packit Service 7c31a4
  uv_pipe_t* handle;
Packit Service 7c31a4
  uv_connect_t* req;
Packit Service 7c31a4
  HANDLE pipeHandle = INVALID_HANDLE_VALUE;
Packit Service 7c31a4
  DWORD duplex_flags;
Packit Service 7c31a4
Packit Service 7c31a4
  req = (uv_connect_t*) parameter;
Packit Service 7c31a4
  assert(req);
Packit Service 7c31a4
  handle = (uv_pipe_t*) req->handle;
Packit Service 7c31a4
  assert(handle);
Packit Service 7c31a4
  loop = handle->loop;
Packit Service 7c31a4
  assert(loop);
Packit Service 7c31a4
Packit Service 7c31a4
  /* We're here because CreateFile on a pipe returned ERROR_PIPE_BUSY. We wait
Packit Service 7c31a4
   * for the pipe to become available with WaitNamedPipe. */
Packit Service 7c31a4
  while (WaitNamedPipeW(handle->name, 30000)) {
Packit Service 7c31a4
    /* The pipe is now available, try to connect. */
Packit Service 7c31a4
    pipeHandle = open_named_pipe(handle->name, &duplex_flags);
Packit Service e2ebee
    if (pipeHandle != INVALID_HANDLE_VALUE)
Packit Service 7c31a4
      break;
Packit Service 7c31a4
Packit Service 7c31a4
    SwitchToThread();
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (pipeHandle != INVALID_HANDLE_VALUE &&
Packit Service 7c31a4
      !uv_set_pipe_handle(loop, handle, pipeHandle, -1, duplex_flags)) {
Packit Service 7c31a4
    SET_REQ_SUCCESS(req);
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    SET_REQ_ERROR(req, GetLastError());
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* Post completed */
Packit Service 7c31a4
  POST_COMPLETION_FOR_REQ(loop, req);
Packit Service 7c31a4
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
void uv_pipe_connect(uv_connect_t* req, uv_pipe_t* handle,
Packit Service 7c31a4
    const char* name, uv_connect_cb cb) {
Packit Service 7c31a4
  uv_loop_t* loop = handle->loop;
Packit Service 7c31a4
  int err, nameSize;
Packit Service 7c31a4
  HANDLE pipeHandle = INVALID_HANDLE_VALUE;
Packit Service 7c31a4
  DWORD duplex_flags;
Packit Service 7c31a4
Packit Service 7c31a4
  UV_REQ_INIT(req, UV_CONNECT);
Packit Service 7c31a4
  req->handle = (uv_stream_t*) handle;
Packit Service 7c31a4
  req->cb = cb;
Packit Service 7c31a4
Packit Service 7c31a4
  /* Convert name to UTF16. */
Packit Service 7c31a4
  nameSize = MultiByteToWideChar(CP_UTF8, 0, name, -1, NULL, 0) * sizeof(WCHAR);
Packit Service e2ebee
  handle->name = uv__malloc(nameSize);
Packit Service 7c31a4
  if (!handle->name) {
Packit Service 7c31a4
    uv_fatal_error(ERROR_OUTOFMEMORY, "uv__malloc");
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (!MultiByteToWideChar(CP_UTF8,
Packit Service 7c31a4
                           0,
Packit Service 7c31a4
                           name,
Packit Service 7c31a4
                           -1,
Packit Service 7c31a4
                           handle->name,
Packit Service 7c31a4
                           nameSize / sizeof(WCHAR))) {
Packit Service 7c31a4
    err = GetLastError();
Packit Service 7c31a4
    goto error;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  pipeHandle = open_named_pipe(handle->name, &duplex_flags);
Packit Service 7c31a4
  if (pipeHandle == INVALID_HANDLE_VALUE) {
Packit Service 7c31a4
    if (GetLastError() == ERROR_PIPE_BUSY) {
Packit Service 7c31a4
      /* Wait for the server to make a pipe instance available. */
Packit Service 7c31a4
      if (!QueueUserWorkItem(&pipe_connect_thread_proc,
Packit Service 7c31a4
                             req,
Packit Service 7c31a4
                             WT_EXECUTELONGFUNCTION)) {
Packit Service 7c31a4
        err = GetLastError();
Packit Service 7c31a4
        goto error;
Packit Service 7c31a4
      }
Packit Service 7c31a4
Packit Service 7c31a4
      REGISTER_HANDLE_REQ(loop, handle, req);
Packit Service 7c31a4
      handle->reqs_pending++;
Packit Service 7c31a4
Packit Service 7c31a4
      return;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    err = GetLastError();
Packit Service 7c31a4
    goto error;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  assert(pipeHandle != INVALID_HANDLE_VALUE);
Packit Service 7c31a4
Packit Service 7c31a4
  if (uv_set_pipe_handle(loop,
Packit Service 7c31a4
                         (uv_pipe_t*) req->handle,
Packit Service 7c31a4
                         pipeHandle,
Packit Service 7c31a4
                         -1,
Packit Service 7c31a4
                         duplex_flags)) {
Packit Service 7c31a4
    err = GetLastError();
Packit Service 7c31a4
    goto error;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  SET_REQ_SUCCESS(req);
Packit Service 7c31a4
  uv_insert_pending_req(loop, (uv_req_t*) req);
Packit Service 7c31a4
  handle->reqs_pending++;
Packit Service 7c31a4
  REGISTER_HANDLE_REQ(loop, handle, req);
Packit Service 7c31a4
  return;
Packit Service 7c31a4
Packit Service 7c31a4
error:
Packit Service 7c31a4
  if (handle->name) {
Packit Service 7c31a4
    uv__free(handle->name);
Packit Service 7c31a4
    handle->name = NULL;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (pipeHandle != INVALID_HANDLE_VALUE) {
Packit Service 7c31a4
    CloseHandle(pipeHandle);
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* Make this req pending reporting an error. */
Packit Service 7c31a4
  SET_REQ_ERROR(req, err);
Packit Service 7c31a4
  uv_insert_pending_req(loop, (uv_req_t*) req);
Packit Service 7c31a4
  handle->reqs_pending++;
Packit Service 7c31a4
  REGISTER_HANDLE_REQ(loop, handle, req);
Packit Service 7c31a4
  return;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
void uv__pipe_interrupt_read(uv_pipe_t* handle) {
Packit Service 7c31a4
  BOOL r;
Packit Service 7c31a4
Packit Service 7c31a4
  if (!(handle->flags & UV_HANDLE_READ_PENDING))
Packit Service 7c31a4
    return; /* No pending reads. */
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_CANCELLATION_PENDING)
Packit Service 7c31a4
    return; /* Already cancelled. */
Packit Service 7c31a4
  if (handle->handle == INVALID_HANDLE_VALUE)
Packit Service 7c31a4
    return; /* Pipe handle closed. */
Packit Service 7c31a4
Packit Service 7c31a4
  if (!(handle->flags & UV_HANDLE_NON_OVERLAPPED_PIPE)) {
Packit Service 7c31a4
    /* Cancel asynchronous read. */
Packit Service 7c31a4
    r = CancelIoEx(handle->handle, &handle->read_req.u.io.overlapped);
Packit Service 7c31a4
    assert(r || GetLastError() == ERROR_NOT_FOUND);
Packit Service 7c31a4
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    /* Cancel synchronous read (which is happening in the thread pool). */
Packit Service 7c31a4
    HANDLE thread;
Packit Service 7c31a4
    volatile HANDLE* thread_ptr = &handle->pipe.conn.readfile_thread_handle;
Packit Service 7c31a4
Packit Service 7c31a4
    EnterCriticalSection(&handle->pipe.conn.readfile_thread_lock);
Packit Service 7c31a4
Packit Service 7c31a4
    thread = *thread_ptr;
Packit Service 7c31a4
    if (thread == NULL) {
Packit Service 7c31a4
      /* The thread pool thread has not yet reached the point of blocking, we
Packit Service 7c31a4
       * can pre-empt it by setting thread_handle to INVALID_HANDLE_VALUE. */
Packit Service 7c31a4
      *thread_ptr = INVALID_HANDLE_VALUE;
Packit Service 7c31a4
Packit Service 7c31a4
    } else {
Packit Service 7c31a4
      /* Spin until the thread has acknowledged (by setting the thread to
Packit Service 7c31a4
       * INVALID_HANDLE_VALUE) that it is past the point of blocking. */
Packit Service 7c31a4
      while (thread != INVALID_HANDLE_VALUE) {
Packit Service 7c31a4
        r = CancelSynchronousIo(thread);
Packit Service 7c31a4
        assert(r || GetLastError() == ERROR_NOT_FOUND);
Packit Service 7c31a4
        SwitchToThread(); /* Yield thread. */
Packit Service 7c31a4
        thread = *thread_ptr;
Packit Service 7c31a4
      }
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    LeaveCriticalSection(&handle->pipe.conn.readfile_thread_lock);
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* Set flag to indicate that read has been cancelled. */
Packit Service 7c31a4
  handle->flags |= UV_HANDLE_CANCELLATION_PENDING;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
void uv__pipe_read_stop(uv_pipe_t* handle) {
Packit Service 7c31a4
  handle->flags &= ~UV_HANDLE_READING;
Packit Service 7c31a4
  DECREASE_ACTIVE_COUNT(handle->loop, handle);
Packit Service 7c31a4
Packit Service 7c31a4
  uv__pipe_interrupt_read(handle);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
/* Cleans up uv_pipe_t (server or connection) and all resources associated with
Packit Service 7c31a4
 * it. */
Packit Service 7c31a4
void uv_pipe_cleanup(uv_loop_t* loop, uv_pipe_t* handle) {
Packit Service 7c31a4
  int i;
Packit Service 7c31a4
  HANDLE pipeHandle;
Packit Service 7c31a4
Packit Service 7c31a4
  uv__pipe_interrupt_read(handle);
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->name) {
Packit Service 7c31a4
    uv__free(handle->name);
Packit Service 7c31a4
    handle->name = NULL;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_PIPESERVER) {
Packit Service 7c31a4
    for (i = 0; i < handle->pipe.serv.pending_instances; i++) {
Packit Service 7c31a4
      pipeHandle = handle->pipe.serv.accept_reqs[i].pipeHandle;
Packit Service 7c31a4
      if (pipeHandle != INVALID_HANDLE_VALUE) {
Packit Service 7c31a4
        CloseHandle(pipeHandle);
Packit Service 7c31a4
        handle->pipe.serv.accept_reqs[i].pipeHandle = INVALID_HANDLE_VALUE;
Packit Service 7c31a4
      }
Packit Service 7c31a4
    }
Packit Service 7c31a4
    handle->handle = INVALID_HANDLE_VALUE;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_CONNECTION) {
Packit Service 7c31a4
    handle->flags &= ~UV_HANDLE_WRITABLE;
Packit Service 7c31a4
    eof_timer_destroy(handle);
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if ((handle->flags & UV_HANDLE_CONNECTION)
Packit Service 7c31a4
      && handle->handle != INVALID_HANDLE_VALUE)
Packit Service 7c31a4
    close_pipe(handle);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
void uv_pipe_close(uv_loop_t* loop, uv_pipe_t* handle) {
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_READING) {
Packit Service 7c31a4
    handle->flags &= ~UV_HANDLE_READING;
Packit Service 7c31a4
    DECREASE_ACTIVE_COUNT(loop, handle);
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_LISTENING) {
Packit Service 7c31a4
    handle->flags &= ~UV_HANDLE_LISTENING;
Packit Service 7c31a4
    DECREASE_ACTIVE_COUNT(loop, handle);
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  uv_pipe_cleanup(loop, handle);
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->reqs_pending == 0) {
Packit Service 7c31a4
    uv_want_endgame(loop, (uv_handle_t*) handle);
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  handle->flags &= ~(UV_HANDLE_READABLE | UV_HANDLE_WRITABLE);
Packit Service 7c31a4
  uv__handle_closing(handle);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void uv_pipe_queue_accept(uv_loop_t* loop, uv_pipe_t* handle,
Packit Service 7c31a4
    uv_pipe_accept_t* req, BOOL firstInstance) {
Packit Service 7c31a4
  assert(handle->flags & UV_HANDLE_LISTENING);
Packit Service 7c31a4
Packit Service 7c31a4
  if (!firstInstance && !pipe_alloc_accept(loop, handle, req, FALSE)) {
Packit Service 7c31a4
    SET_REQ_ERROR(req, GetLastError());
Packit Service 7c31a4
    uv_insert_pending_req(loop, (uv_req_t*) req);
Packit Service 7c31a4
    handle->reqs_pending++;
Packit Service 7c31a4
    return;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  assert(req->pipeHandle != INVALID_HANDLE_VALUE);
Packit Service 7c31a4
Packit Service 7c31a4
  /* Prepare the overlapped structure. */
Packit Service 7c31a4
  memset(&(req->u.io.overlapped), 0, sizeof(req->u.io.overlapped));
Packit Service 7c31a4
Packit Service 7c31a4
  if (!ConnectNamedPipe(req->pipeHandle, &req->u.io.overlapped) &&
Packit Service 7c31a4
      GetLastError() != ERROR_IO_PENDING) {
Packit Service 7c31a4
    if (GetLastError() == ERROR_PIPE_CONNECTED) {
Packit Service 7c31a4
      SET_REQ_SUCCESS(req);
Packit Service 7c31a4
    } else {
Packit Service 7c31a4
      CloseHandle(req->pipeHandle);
Packit Service 7c31a4
      req->pipeHandle = INVALID_HANDLE_VALUE;
Packit Service 7c31a4
      /* Make this req pending reporting an error. */
Packit Service 7c31a4
      SET_REQ_ERROR(req, GetLastError());
Packit Service 7c31a4
    }
Packit Service 7c31a4
    uv_insert_pending_req(loop, (uv_req_t*) req);
Packit Service 7c31a4
    handle->reqs_pending++;
Packit Service 7c31a4
    return;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* Wait for completion via IOCP */
Packit Service 7c31a4
  handle->reqs_pending++;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
int uv_pipe_accept(uv_pipe_t* server, uv_stream_t* client) {
Packit Service 7c31a4
  uv_loop_t* loop = server->loop;
Packit Service 7c31a4
  uv_pipe_t* pipe_client;
Packit Service 7c31a4
  uv_pipe_accept_t* req;
Packit Service 7c31a4
  QUEUE* q;
Packit Service 7c31a4
  uv__ipc_xfer_queue_item_t* item;
Packit Service 7c31a4
  int err;
Packit Service 7c31a4
Packit Service 7c31a4
  if (server->ipc) {
Packit Service 7c31a4
    if (QUEUE_EMPTY(&server->pipe.conn.ipc_xfer_queue)) {
Packit Service 7c31a4
      /* No valid pending sockets. */
Packit Service 7c31a4
      return WSAEWOULDBLOCK;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    q = QUEUE_HEAD(&server->pipe.conn.ipc_xfer_queue);
Packit Service 7c31a4
    QUEUE_REMOVE(q);
Packit Service 7c31a4
    server->pipe.conn.ipc_xfer_queue_length--;
Packit Service 7c31a4
    item = QUEUE_DATA(q, uv__ipc_xfer_queue_item_t, member);
Packit Service 7c31a4
Packit Service 7c31a4
    err = uv__tcp_xfer_import(
Packit Service 7c31a4
        (uv_tcp_t*) client, item->xfer_type, &item->xfer_info);
Packit Service 7c31a4
    if (err != 0)
Packit Service 7c31a4
      return err;
Packit Service 7c31a4
Packit Service 7c31a4
    uv__free(item);
Packit Service 7c31a4
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    pipe_client = (uv_pipe_t*) client;
Packit Service 7c31a4
Packit Service 7c31a4
    /* Find a connection instance that has been connected, but not yet
Packit Service 7c31a4
     * accepted. */
Packit Service 7c31a4
    req = server->pipe.serv.pending_accepts;
Packit Service 7c31a4
Packit Service 7c31a4
    if (!req) {
Packit Service 7c31a4
      /* No valid connections found, so we error out. */
Packit Service 7c31a4
      return WSAEWOULDBLOCK;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    /* Initialize the client handle and copy the pipeHandle to the client */
Packit Service 7c31a4
    uv_pipe_connection_init(pipe_client);
Packit Service 7c31a4
    pipe_client->handle = req->pipeHandle;
Packit Service 7c31a4
    pipe_client->flags |= UV_HANDLE_READABLE | UV_HANDLE_WRITABLE;
Packit Service 7c31a4
Packit Service 7c31a4
    /* Prepare the req to pick up a new connection */
Packit Service 7c31a4
    server->pipe.serv.pending_accepts = req->next_pending;
Packit Service 7c31a4
    req->next_pending = NULL;
Packit Service 7c31a4
    req->pipeHandle = INVALID_HANDLE_VALUE;
Packit Service 7c31a4
Packit Service 7c31a4
    server->handle = INVALID_HANDLE_VALUE;
Packit Service 7c31a4
    if (!(server->flags & UV_HANDLE_CLOSING)) {
Packit Service 7c31a4
      uv_pipe_queue_accept(loop, server, req, FALSE);
Packit Service 7c31a4
    }
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
/* Starts listening for connections for the given pipe. */
Packit Service 7c31a4
int uv_pipe_listen(uv_pipe_t* handle, int backlog, uv_connection_cb cb) {
Packit Service 7c31a4
  uv_loop_t* loop = handle->loop;
Packit Service 7c31a4
  int i;
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_LISTENING) {
Packit Service 7c31a4
    handle->stream.serv.connection_cb = cb;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (!(handle->flags & UV_HANDLE_BOUND)) {
Packit Service 7c31a4
    return WSAEINVAL;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_READING) {
Packit Service 7c31a4
    return WSAEISCONN;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (!(handle->flags & UV_HANDLE_PIPESERVER)) {
Packit Service 7c31a4
    return ERROR_NOT_SUPPORTED;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->ipc) {
Packit Service 7c31a4
    return WSAEINVAL;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  handle->flags |= UV_HANDLE_LISTENING;
Packit Service 7c31a4
  INCREASE_ACTIVE_COUNT(loop, handle);
Packit Service 7c31a4
  handle->stream.serv.connection_cb = cb;
Packit Service 7c31a4
Packit Service 7c31a4
  /* First pipe handle should have already been created in uv_pipe_bind */
Packit Service 7c31a4
  assert(handle->pipe.serv.accept_reqs[0].pipeHandle != INVALID_HANDLE_VALUE);
Packit Service 7c31a4
Packit Service 7c31a4
  for (i = 0; i < handle->pipe.serv.pending_instances; i++) {
Packit Service 7c31a4
    uv_pipe_queue_accept(loop, handle, &handle->pipe.serv.accept_reqs[i], i == 0);
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static DWORD WINAPI uv_pipe_zero_readfile_thread_proc(void* arg) {
Packit Service 7c31a4
  uv_read_t* req = (uv_read_t*) arg;
Packit Service 7c31a4
  uv_pipe_t* handle = (uv_pipe_t*) req->data;
Packit Service 7c31a4
  uv_loop_t* loop = handle->loop;
Packit Service 7c31a4
  volatile HANDLE* thread_ptr = &handle->pipe.conn.readfile_thread_handle;
Packit Service 7c31a4
  CRITICAL_SECTION* lock = &handle->pipe.conn.readfile_thread_lock;
Packit Service 7c31a4
  HANDLE thread;
Packit Service 7c31a4
  DWORD bytes;
Packit Service 7c31a4
  DWORD err;
Packit Service 7c31a4
Packit Service 7c31a4
  assert(req->type == UV_READ);
Packit Service 7c31a4
  assert(handle->type == UV_NAMED_PIPE);
Packit Service 7c31a4
Packit Service 7c31a4
  err = 0;
Packit Service 7c31a4
Packit Service 7c31a4
  /* Create a handle to the current thread. */
Packit Service 7c31a4
  if (!DuplicateHandle(GetCurrentProcess(),
Packit Service 7c31a4
                       GetCurrentThread(),
Packit Service 7c31a4
                       GetCurrentProcess(),
Packit Service 7c31a4
                       &thread,
Packit Service 7c31a4
                       0,
Packit Service 7c31a4
                       FALSE,
Packit Service 7c31a4
                       DUPLICATE_SAME_ACCESS)) {
Packit Service 7c31a4
    err = GetLastError();
Packit Service 7c31a4
    goto out1;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* The lock needs to be held when thread handle is modified. */
Packit Service 7c31a4
  EnterCriticalSection(lock);
Packit Service 7c31a4
  if (*thread_ptr == INVALID_HANDLE_VALUE) {
Packit Service 7c31a4
    /* uv__pipe_interrupt_read() cancelled reading before we got here. */
Packit Service 7c31a4
    err = ERROR_OPERATION_ABORTED;
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    /* Let main thread know which worker thread is doing the blocking read. */
Packit Service 7c31a4
    assert(*thread_ptr == NULL);
Packit Service 7c31a4
    *thread_ptr = thread;
Packit Service 7c31a4
  }
Packit Service 7c31a4
  LeaveCriticalSection(lock);
Packit Service 7c31a4
Packit Service 7c31a4
  if (err)
Packit Service 7c31a4
    goto out2;
Packit Service 7c31a4
Packit Service 7c31a4
  /* Block the thread until data is available on the pipe, or the read is
Packit Service 7c31a4
   * cancelled. */
Packit Service 7c31a4
  if (!ReadFile(handle->handle, &uv_zero_, 0, &bytes, NULL))
Packit Service 7c31a4
    err = GetLastError();
Packit Service 7c31a4
Packit Service 7c31a4
  /* Let the main thread know the worker is past the point of blocking. */
Packit Service 7c31a4
  assert(thread == *thread_ptr);
Packit Service 7c31a4
  *thread_ptr = INVALID_HANDLE_VALUE;
Packit Service 7c31a4
Packit Service 7c31a4
  /* Briefly acquire the mutex. Since the main thread holds the lock while it
Packit Service 7c31a4
   * is spinning trying to cancel this thread's I/O, we will block here until
Packit Service 7c31a4
   * it stops doing that. */
Packit Service 7c31a4
  EnterCriticalSection(lock);
Packit Service 7c31a4
  LeaveCriticalSection(lock);
Packit Service 7c31a4
Packit Service 7c31a4
out2:
Packit Service 7c31a4
  /* Close the handle to the current thread. */
Packit Service 7c31a4
  CloseHandle(thread);
Packit Service 7c31a4
Packit Service 7c31a4
out1:
Packit Service 7c31a4
  /* Set request status and post a completion record to the IOCP. */
Packit Service 7c31a4
  if (err)
Packit Service 7c31a4
    SET_REQ_ERROR(req, err);
Packit Service 7c31a4
  else
Packit Service 7c31a4
    SET_REQ_SUCCESS(req);
Packit Service 7c31a4
  POST_COMPLETION_FOR_REQ(loop, req);
Packit Service 7c31a4
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static DWORD WINAPI uv_pipe_writefile_thread_proc(void* parameter) {
Packit Service 7c31a4
  int result;
Packit Service 7c31a4
  DWORD bytes;
Packit Service 7c31a4
  uv_write_t* req = (uv_write_t*) parameter;
Packit Service 7c31a4
  uv_pipe_t* handle = (uv_pipe_t*) req->handle;
Packit Service 7c31a4
  uv_loop_t* loop = handle->loop;
Packit Service 7c31a4
Packit Service 7c31a4
  assert(req != NULL);
Packit Service 7c31a4
  assert(req->type == UV_WRITE);
Packit Service 7c31a4
  assert(handle->type == UV_NAMED_PIPE);
Packit Service 7c31a4
  assert(req->write_buffer.base);
Packit Service 7c31a4
Packit Service 7c31a4
  result = WriteFile(handle->handle,
Packit Service 7c31a4
                     req->write_buffer.base,
Packit Service 7c31a4
                     req->write_buffer.len,
Packit Service 7c31a4
                     &bytes,
Packit Service 7c31a4
                     NULL);
Packit Service 7c31a4
Packit Service 7c31a4
  if (!result) {
Packit Service 7c31a4
    SET_REQ_ERROR(req, GetLastError());
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  POST_COMPLETION_FOR_REQ(loop, req);
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void CALLBACK post_completion_read_wait(void* context, BOOLEAN timed_out) {
Packit Service 7c31a4
  uv_read_t* req;
Packit Service 7c31a4
  uv_tcp_t* handle;
Packit Service 7c31a4
Packit Service 7c31a4
  req = (uv_read_t*) context;
Packit Service 7c31a4
  assert(req != NULL);
Packit Service 7c31a4
  handle = (uv_tcp_t*)req->data;
Packit Service 7c31a4
  assert(handle != NULL);
Packit Service 7c31a4
  assert(!timed_out);
Packit Service 7c31a4
Packit Service 7c31a4
  if (!PostQueuedCompletionStatus(handle->loop->iocp,
Packit Service 7c31a4
                                  req->u.io.overlapped.InternalHigh,
Packit Service 7c31a4
                                  0,
Packit Service 7c31a4
                                  &req->u.io.overlapped)) {
Packit Service 7c31a4
    uv_fatal_error(GetLastError(), "PostQueuedCompletionStatus");
Packit Service 7c31a4
  }
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void CALLBACK post_completion_write_wait(void* context, BOOLEAN timed_out) {
Packit Service 7c31a4
  uv_write_t* req;
Packit Service 7c31a4
  uv_tcp_t* handle;
Packit Service 7c31a4
Packit Service 7c31a4
  req = (uv_write_t*) context;
Packit Service 7c31a4
  assert(req != NULL);
Packit Service 7c31a4
  handle = (uv_tcp_t*)req->handle;
Packit Service 7c31a4
  assert(handle != NULL);
Packit Service 7c31a4
  assert(!timed_out);
Packit Service 7c31a4
Packit Service 7c31a4
  if (!PostQueuedCompletionStatus(handle->loop->iocp,
Packit Service 7c31a4
                                  req->u.io.overlapped.InternalHigh,
Packit Service 7c31a4
                                  0,
Packit Service 7c31a4
                                  &req->u.io.overlapped)) {
Packit Service 7c31a4
    uv_fatal_error(GetLastError(), "PostQueuedCompletionStatus");
Packit Service 7c31a4
  }
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void uv_pipe_queue_read(uv_loop_t* loop, uv_pipe_t* handle) {
Packit Service 7c31a4
  uv_read_t* req;
Packit Service 7c31a4
  int result;
Packit Service 7c31a4
Packit Service 7c31a4
  assert(handle->flags & UV_HANDLE_READING);
Packit Service 7c31a4
  assert(!(handle->flags & UV_HANDLE_READ_PENDING));
Packit Service 7c31a4
Packit Service 7c31a4
  assert(handle->handle != INVALID_HANDLE_VALUE);
Packit Service 7c31a4
Packit Service 7c31a4
  req = &handle->read_req;
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_NON_OVERLAPPED_PIPE) {
Packit Service 7c31a4
    handle->pipe.conn.readfile_thread_handle = NULL; /* Reset cancellation. */
Packit Service 7c31a4
    if (!QueueUserWorkItem(&uv_pipe_zero_readfile_thread_proc,
Packit Service 7c31a4
                           req,
Packit Service 7c31a4
                           WT_EXECUTELONGFUNCTION)) {
Packit Service 7c31a4
      /* Make this req pending reporting an error. */
Packit Service 7c31a4
      SET_REQ_ERROR(req, GetLastError());
Packit Service 7c31a4
      goto error;
Packit Service 7c31a4
    }
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    memset(&req->u.io.overlapped, 0, sizeof(req->u.io.overlapped));
Packit Service 7c31a4
    if (handle->flags & UV_HANDLE_EMULATE_IOCP) {
Packit Service 7c31a4
      assert(req->event_handle != NULL);
Packit Service 7c31a4
      req->u.io.overlapped.hEvent = (HANDLE) ((uintptr_t) req->event_handle | 1);
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    /* Do 0-read */
Packit Service 7c31a4
    result = ReadFile(handle->handle,
Packit Service 7c31a4
                      &uv_zero_,
Packit Service 7c31a4
                      0,
Packit Service 7c31a4
                      NULL,
Packit Service 7c31a4
                      &req->u.io.overlapped);
Packit Service 7c31a4
Packit Service 7c31a4
    if (!result && GetLastError() != ERROR_IO_PENDING) {
Packit Service 7c31a4
      /* Make this req pending reporting an error. */
Packit Service 7c31a4
      SET_REQ_ERROR(req, GetLastError());
Packit Service 7c31a4
      goto error;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    if (handle->flags & UV_HANDLE_EMULATE_IOCP) {
Packit Service 7c31a4
      if (req->wait_handle == INVALID_HANDLE_VALUE) {
Packit Service 7c31a4
        if (!RegisterWaitForSingleObject(&req->wait_handle,
Packit Service 7c31a4
            req->event_handle, post_completion_read_wait, (void*) req,
Packit Service 7c31a4
            INFINITE, WT_EXECUTEINWAITTHREAD)) {
Packit Service 7c31a4
          SET_REQ_ERROR(req, GetLastError());
Packit Service 7c31a4
          goto error;
Packit Service 7c31a4
        }
Packit Service 7c31a4
      }
Packit Service 7c31a4
    }
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* Start the eof timer if there is one */
Packit Service 7c31a4
  eof_timer_start(handle);
Packit Service 7c31a4
  handle->flags |= UV_HANDLE_READ_PENDING;
Packit Service 7c31a4
  handle->reqs_pending++;
Packit Service 7c31a4
  return;
Packit Service 7c31a4
Packit Service 7c31a4
error:
Packit Service 7c31a4
  uv_insert_pending_req(loop, (uv_req_t*)req);
Packit Service 7c31a4
  handle->flags |= UV_HANDLE_READ_PENDING;
Packit Service 7c31a4
  handle->reqs_pending++;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
int uv_pipe_read_start(uv_pipe_t* handle,
Packit Service 7c31a4
                       uv_alloc_cb alloc_cb,
Packit Service 7c31a4
                       uv_read_cb read_cb) {
Packit Service 7c31a4
  uv_loop_t* loop = handle->loop;
Packit Service 7c31a4
Packit Service 7c31a4
  handle->flags |= UV_HANDLE_READING;
Packit Service 7c31a4
  INCREASE_ACTIVE_COUNT(loop, handle);
Packit Service 7c31a4
  handle->read_cb = read_cb;
Packit Service 7c31a4
  handle->alloc_cb = alloc_cb;
Packit Service 7c31a4
Packit Service 7c31a4
  /* If reading was stopped and then started again, there could still be a read
Packit Service 7c31a4
   * request pending. */
Packit Service 7c31a4
  if (!(handle->flags & UV_HANDLE_READ_PENDING)) {
Packit Service 7c31a4
    if (handle->flags & UV_HANDLE_EMULATE_IOCP &&
Packit Service 7c31a4
        handle->read_req.event_handle == NULL) {
Packit Service 7c31a4
      handle->read_req.event_handle = CreateEvent(NULL, 0, 0, NULL);
Packit Service 7c31a4
      if (handle->read_req.event_handle == NULL) {
Packit Service 7c31a4
        uv_fatal_error(GetLastError(), "CreateEvent");
Packit Service 7c31a4
      }
Packit Service 7c31a4
    }
Packit Service 7c31a4
    uv_pipe_queue_read(loop, handle);
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void uv_insert_non_overlapped_write_req(uv_pipe_t* handle,
Packit Service 7c31a4
    uv_write_t* req) {
Packit Service 7c31a4
  req->next_req = NULL;
Packit Service 7c31a4
  if (handle->pipe.conn.non_overlapped_writes_tail) {
Packit Service 7c31a4
    req->next_req =
Packit Service 7c31a4
      handle->pipe.conn.non_overlapped_writes_tail->next_req;
Packit Service 7c31a4
    handle->pipe.conn.non_overlapped_writes_tail->next_req = (uv_req_t*)req;
Packit Service 7c31a4
    handle->pipe.conn.non_overlapped_writes_tail = req;
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    req->next_req = (uv_req_t*)req;
Packit Service 7c31a4
    handle->pipe.conn.non_overlapped_writes_tail = req;
Packit Service 7c31a4
  }
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static uv_write_t* uv_remove_non_overlapped_write_req(uv_pipe_t* handle) {
Packit Service 7c31a4
  uv_write_t* req;
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->pipe.conn.non_overlapped_writes_tail) {
Packit Service 7c31a4
    req = (uv_write_t*)handle->pipe.conn.non_overlapped_writes_tail->next_req;
Packit Service 7c31a4
Packit Service 7c31a4
    if (req == handle->pipe.conn.non_overlapped_writes_tail) {
Packit Service 7c31a4
      handle->pipe.conn.non_overlapped_writes_tail = NULL;
Packit Service 7c31a4
    } else {
Packit Service 7c31a4
      handle->pipe.conn.non_overlapped_writes_tail->next_req =
Packit Service 7c31a4
        req->next_req;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    return req;
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    /* queue empty */
Packit Service 7c31a4
    return NULL;
Packit Service 7c31a4
  }
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void uv_queue_non_overlapped_write(uv_pipe_t* handle) {
Packit Service 7c31a4
  uv_write_t* req = uv_remove_non_overlapped_write_req(handle);
Packit Service 7c31a4
  if (req) {
Packit Service 7c31a4
    if (!QueueUserWorkItem(&uv_pipe_writefile_thread_proc,
Packit Service 7c31a4
                           req,
Packit Service 7c31a4
                           WT_EXECUTELONGFUNCTION)) {
Packit Service 7c31a4
      uv_fatal_error(GetLastError(), "QueueUserWorkItem");
Packit Service 7c31a4
    }
Packit Service 7c31a4
  }
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static int uv__build_coalesced_write_req(uv_write_t* user_req,
Packit Service 7c31a4
                                         const uv_buf_t bufs[],
Packit Service 7c31a4
                                         size_t nbufs,
Packit Service 7c31a4
                                         uv_write_t** req_out,
Packit Service 7c31a4
                                         uv_buf_t* write_buf_out) {
Packit Service 7c31a4
  /* Pack into a single heap-allocated buffer:
Packit Service 7c31a4
   *   (a) a uv_write_t structure where libuv stores the actual state.
Packit Service 7c31a4
   *   (b) a pointer to the original uv_write_t.
Packit Service 7c31a4
   *   (c) data from all `bufs` entries.
Packit Service 7c31a4
   */
Packit Service 7c31a4
  char* heap_buffer;
Packit Service 7c31a4
  size_t heap_buffer_length, heap_buffer_offset;
Packit Service 7c31a4
  uv__coalesced_write_t* coalesced_write_req; /* (a) + (b) */
Packit Service 7c31a4
  char* data_start;                           /* (c) */
Packit Service 7c31a4
  size_t data_length;
Packit Service 7c31a4
  unsigned int i;
Packit Service 7c31a4
Packit Service 7c31a4
  /* Compute combined size of all combined buffers from `bufs`. */
Packit Service 7c31a4
  data_length = 0;
Packit Service 7c31a4
  for (i = 0; i < nbufs; i++)
Packit Service 7c31a4
    data_length += bufs[i].len;
Packit Service 7c31a4
Packit Service 7c31a4
  /* The total combined size of data buffers should not exceed UINT32_MAX,
Packit Service 7c31a4
   * because WriteFile() won't accept buffers larger than that. */
Packit Service 7c31a4
  if (data_length > UINT32_MAX)
Packit Service 7c31a4
    return WSAENOBUFS; /* Maps to UV_ENOBUFS. */
Packit Service 7c31a4
Packit Service 7c31a4
  /* Compute heap buffer size. */
Packit Service 7c31a4
  heap_buffer_length = sizeof *coalesced_write_req + /* (a) + (b) */
Packit Service 7c31a4
                       data_length;                  /* (c) */
Packit Service 7c31a4
Packit Service 7c31a4
  /* Allocate buffer. */
Packit Service 7c31a4
  heap_buffer = uv__malloc(heap_buffer_length);
Packit Service 7c31a4
  if (heap_buffer == NULL)
Packit Service 7c31a4
    return ERROR_NOT_ENOUGH_MEMORY; /* Maps to UV_ENOMEM. */
Packit Service 7c31a4
Packit Service 7c31a4
  /* Copy uv_write_t information to the buffer. */
Packit Service 7c31a4
  coalesced_write_req = (uv__coalesced_write_t*) heap_buffer;
Packit Service 7c31a4
  coalesced_write_req->req = *user_req; /* copy (a) */
Packit Service 7c31a4
  coalesced_write_req->req.coalesced = 1;
Packit Service 7c31a4
  coalesced_write_req->user_req = user_req;         /* copy (b) */
Packit Service 7c31a4
  heap_buffer_offset = sizeof *coalesced_write_req; /* offset (a) + (b) */
Packit Service 7c31a4
Packit Service 7c31a4
  /* Copy data buffers to the heap buffer. */
Packit Service 7c31a4
  data_start = &heap_buffer[heap_buffer_offset];
Packit Service 7c31a4
  for (i = 0; i < nbufs; i++) {
Packit Service 7c31a4
    memcpy(&heap_buffer[heap_buffer_offset],
Packit Service 7c31a4
           bufs[i].base,
Packit Service 7c31a4
           bufs[i].len);               /* copy (c) */
Packit Service 7c31a4
    heap_buffer_offset += bufs[i].len; /* offset (c) */
Packit Service 7c31a4
  }
Packit Service 7c31a4
  assert(heap_buffer_offset == heap_buffer_length);
Packit Service 7c31a4
Packit Service 7c31a4
  /* Set out arguments and return. */
Packit Service 7c31a4
  *req_out = &coalesced_write_req->req;
Packit Service 7c31a4
  *write_buf_out = uv_buf_init(data_start, (unsigned int) data_length);
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static int uv__pipe_write_data(uv_loop_t* loop,
Packit Service 7c31a4
                               uv_write_t* req,
Packit Service 7c31a4
                               uv_pipe_t* handle,
Packit Service 7c31a4
                               const uv_buf_t bufs[],
Packit Service 7c31a4
                               size_t nbufs,
Packit Service 7c31a4
                               uv_write_cb cb,
Packit Service 7c31a4
                               int copy_always) {
Packit Service 7c31a4
  int err;
Packit Service 7c31a4
  int result;
Packit Service 7c31a4
  uv_buf_t write_buf;
Packit Service 7c31a4
Packit Service 7c31a4
  assert(handle->handle != INVALID_HANDLE_VALUE);
Packit Service 7c31a4
Packit Service 7c31a4
  UV_REQ_INIT(req, UV_WRITE);
Packit Service 7c31a4
  req->handle = (uv_stream_t*) handle;
Packit Service 7c31a4
  req->send_handle = NULL;
Packit Service 7c31a4
  req->cb = cb;
Packit Service 7c31a4
  /* Private fields. */
Packit Service 7c31a4
  req->coalesced = 0;
Packit Service 7c31a4
  req->event_handle = NULL;
Packit Service 7c31a4
  req->wait_handle = INVALID_HANDLE_VALUE;
Packit Service 7c31a4
Packit Service 7c31a4
  /* Prepare the overlapped structure. */
Packit Service 7c31a4
  memset(&req->u.io.overlapped, 0, sizeof(req->u.io.overlapped));
Packit Service 7c31a4
  if (handle->flags & (UV_HANDLE_EMULATE_IOCP | UV_HANDLE_BLOCKING_WRITES)) {
Packit Service 7c31a4
    req->event_handle = CreateEvent(NULL, 0, 0, NULL);
Packit Service 7c31a4
    if (req->event_handle == NULL) {
Packit Service 7c31a4
      uv_fatal_error(GetLastError(), "CreateEvent");
Packit Service 7c31a4
    }
Packit Service 7c31a4
    req->u.io.overlapped.hEvent = (HANDLE) ((uintptr_t) req->event_handle | 1);
Packit Service 7c31a4
  }
Packit Service 7c31a4
  req->write_buffer = uv_null_buf_;
Packit Service 7c31a4
Packit Service 7c31a4
  if (nbufs == 0) {
Packit Service 7c31a4
    /* Write empty buffer. */
Packit Service 7c31a4
    write_buf = uv_null_buf_;
Packit Service 7c31a4
  } else if (nbufs == 1 && !copy_always) {
Packit Service 7c31a4
    /* Write directly from bufs[0]. */
Packit Service 7c31a4
    write_buf = bufs[0];
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    /* Coalesce all `bufs` into one big buffer. This also creates a new
Packit Service 7c31a4
     * write-request structure that replaces the old one. */
Packit Service 7c31a4
    err = uv__build_coalesced_write_req(req, bufs, nbufs, &req, &write_buf);
Packit Service 7c31a4
    if (err != 0)
Packit Service 7c31a4
      return err;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if ((handle->flags &
Packit Service 7c31a4
      (UV_HANDLE_BLOCKING_WRITES | UV_HANDLE_NON_OVERLAPPED_PIPE)) ==
Packit Service 7c31a4
      (UV_HANDLE_BLOCKING_WRITES | UV_HANDLE_NON_OVERLAPPED_PIPE)) {
Packit Service 7c31a4
    DWORD bytes;
Packit Service 7c31a4
    result =
Packit Service 7c31a4
        WriteFile(handle->handle, write_buf.base, write_buf.len, &bytes, NULL);
Packit Service 7c31a4
Packit Service 7c31a4
    if (!result) {
Packit Service 7c31a4
      err = GetLastError();
Packit Service 7c31a4
      return err;
Packit Service 7c31a4
    } else {
Packit Service 7c31a4
      /* Request completed immediately. */
Packit Service 7c31a4
      req->u.io.queued_bytes = 0;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    REGISTER_HANDLE_REQ(loop, handle, req);
Packit Service 7c31a4
    handle->reqs_pending++;
Packit Service 7c31a4
    handle->stream.conn.write_reqs_pending++;
Packit Service 7c31a4
    POST_COMPLETION_FOR_REQ(loop, req);
Packit Service 7c31a4
    return 0;
Packit Service 7c31a4
  } else if (handle->flags & UV_HANDLE_NON_OVERLAPPED_PIPE) {
Packit Service 7c31a4
    req->write_buffer = write_buf;
Packit Service 7c31a4
    uv_insert_non_overlapped_write_req(handle, req);
Packit Service 7c31a4
    if (handle->stream.conn.write_reqs_pending == 0) {
Packit Service 7c31a4
      uv_queue_non_overlapped_write(handle);
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    /* Request queued by the kernel. */
Packit Service 7c31a4
    req->u.io.queued_bytes = write_buf.len;
Packit Service 7c31a4
    handle->write_queue_size += req->u.io.queued_bytes;
Packit Service 7c31a4
  } else if (handle->flags & UV_HANDLE_BLOCKING_WRITES) {
Packit Service 7c31a4
    /* Using overlapped IO, but wait for completion before returning */
Packit Service 7c31a4
    result = WriteFile(handle->handle,
Packit Service 7c31a4
                       write_buf.base,
Packit Service 7c31a4
                       write_buf.len,
Packit Service 7c31a4
                       NULL,
Packit Service 7c31a4
                       &req->u.io.overlapped);
Packit Service 7c31a4
Packit Service 7c31a4
    if (!result && GetLastError() != ERROR_IO_PENDING) {
Packit Service 7c31a4
      err = GetLastError();
Packit Service 7c31a4
      CloseHandle(req->event_handle);
Packit Service 7c31a4
      req->event_handle = NULL;
Packit Service 7c31a4
      return err;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    if (result) {
Packit Service 7c31a4
      /* Request completed immediately. */
Packit Service 7c31a4
      req->u.io.queued_bytes = 0;
Packit Service 7c31a4
    } else {
Packit Service 7c31a4
      /* Request queued by the kernel. */
Packit Service 7c31a4
      req->u.io.queued_bytes = write_buf.len;
Packit Service 7c31a4
      handle->write_queue_size += req->u.io.queued_bytes;
Packit Service 7c31a4
      if (WaitForSingleObject(req->event_handle, INFINITE) !=
Packit Service 7c31a4
          WAIT_OBJECT_0) {
Packit Service 7c31a4
        err = GetLastError();
Packit Service 7c31a4
        CloseHandle(req->event_handle);
Packit Service 7c31a4
        req->event_handle = NULL;
Packit Service 7c31a4
        return err;
Packit Service 7c31a4
      }
Packit Service 7c31a4
    }
Packit Service 7c31a4
    CloseHandle(req->event_handle);
Packit Service 7c31a4
    req->event_handle = NULL;
Packit Service 7c31a4
Packit Service 7c31a4
    REGISTER_HANDLE_REQ(loop, handle, req);
Packit Service 7c31a4
    handle->reqs_pending++;
Packit Service 7c31a4
    handle->stream.conn.write_reqs_pending++;
Packit Service 7c31a4
    return 0;
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    result = WriteFile(handle->handle,
Packit Service 7c31a4
                       write_buf.base,
Packit Service 7c31a4
                       write_buf.len,
Packit Service 7c31a4
                       NULL,
Packit Service 7c31a4
                       &req->u.io.overlapped);
Packit Service 7c31a4
Packit Service 7c31a4
    if (!result && GetLastError() != ERROR_IO_PENDING) {
Packit Service 7c31a4
      return GetLastError();
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    if (result) {
Packit Service 7c31a4
      /* Request completed immediately. */
Packit Service 7c31a4
      req->u.io.queued_bytes = 0;
Packit Service 7c31a4
    } else {
Packit Service 7c31a4
      /* Request queued by the kernel. */
Packit Service 7c31a4
      req->u.io.queued_bytes = write_buf.len;
Packit Service 7c31a4
      handle->write_queue_size += req->u.io.queued_bytes;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    if (handle->flags & UV_HANDLE_EMULATE_IOCP) {
Packit Service 7c31a4
      if (!RegisterWaitForSingleObject(&req->wait_handle,
Packit Service 7c31a4
          req->event_handle, post_completion_write_wait, (void*) req,
Packit Service 7c31a4
          INFINITE, WT_EXECUTEINWAITTHREAD)) {
Packit Service 7c31a4
        return GetLastError();
Packit Service 7c31a4
      }
Packit Service 7c31a4
    }
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  REGISTER_HANDLE_REQ(loop, handle, req);
Packit Service 7c31a4
  handle->reqs_pending++;
Packit Service 7c31a4
  handle->stream.conn.write_reqs_pending++;
Packit Service 7c31a4
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static DWORD uv__pipe_get_ipc_remote_pid(uv_pipe_t* handle) {
Packit Service 7c31a4
  DWORD* pid = &handle->pipe.conn.ipc_remote_pid;
Packit Service 7c31a4
Packit Service 7c31a4
  /* If the both ends of the IPC pipe are owned by the same process,
Packit Service 7c31a4
   * the remote end pid may not yet be set. If so, do it here.
Packit Service 7c31a4
   * TODO: this is weird; it'd probably better to use a handshake. */
Packit Service 7c31a4
  if (*pid == 0)
Packit Service 7c31a4
    *pid = GetCurrentProcessId();
Packit Service 7c31a4
Packit Service 7c31a4
  return *pid;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
int uv__pipe_write_ipc(uv_loop_t* loop,
Packit Service 7c31a4
                       uv_write_t* req,
Packit Service 7c31a4
                       uv_pipe_t* handle,
Packit Service 7c31a4
                       const uv_buf_t data_bufs[],
Packit Service 7c31a4
                       size_t data_buf_count,
Packit Service 7c31a4
                       uv_stream_t* send_handle,
Packit Service 7c31a4
                       uv_write_cb cb) {
Packit Service 7c31a4
  uv_buf_t stack_bufs[6];
Packit Service 7c31a4
  uv_buf_t* bufs;
Packit Service 7c31a4
  size_t buf_count, buf_index;
Packit Service 7c31a4
  uv__ipc_frame_header_t frame_header;
Packit Service 7c31a4
  uv__ipc_socket_xfer_type_t xfer_type = UV__IPC_SOCKET_XFER_NONE;
Packit Service 7c31a4
  uv__ipc_socket_xfer_info_t xfer_info;
Packit Service 7c31a4
  uint64_t data_length;
Packit Service 7c31a4
  size_t i;
Packit Service 7c31a4
  int err;
Packit Service 7c31a4
Packit Service 7c31a4
  /* Compute the combined size of data buffers. */
Packit Service 7c31a4
  data_length = 0;
Packit Service 7c31a4
  for (i = 0; i < data_buf_count; i++)
Packit Service 7c31a4
    data_length += data_bufs[i].len;
Packit Service 7c31a4
  if (data_length > UINT32_MAX)
Packit Service 7c31a4
    return WSAENOBUFS; /* Maps to UV_ENOBUFS. */
Packit Service 7c31a4
Packit Service 7c31a4
  /* Prepare the frame's socket xfer payload. */
Packit Service 7c31a4
  if (send_handle != NULL) {
Packit Service 7c31a4
    uv_tcp_t* send_tcp_handle = (uv_tcp_t*) send_handle;
Packit Service 7c31a4
Packit Service 7c31a4
    /* Verify that `send_handle` it is indeed a tcp handle. */
Packit Service 7c31a4
    if (send_tcp_handle->type != UV_TCP)
Packit Service 7c31a4
      return ERROR_NOT_SUPPORTED;
Packit Service 7c31a4
Packit Service 7c31a4
    /* Export the tcp handle. */
Packit Service 7c31a4
    err = uv__tcp_xfer_export(send_tcp_handle,
Packit Service 7c31a4
                              uv__pipe_get_ipc_remote_pid(handle),
Packit Service 7c31a4
                              &xfer_type,
Packit Service 7c31a4
                              &xfer_info);
Packit Service 7c31a4
    if (err != 0)
Packit Service 7c31a4
      return err;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* Compute the number of uv_buf_t's required. */
Packit Service 7c31a4
  buf_count = 1 + data_buf_count; /* Frame header and data buffers. */
Packit Service 7c31a4
  if (send_handle != NULL)
Packit Service 7c31a4
    buf_count += 1; /* One extra for the socket xfer information. */
Packit Service 7c31a4
Packit Service 7c31a4
  /* Use the on-stack buffer array if it is big enough; otherwise allocate
Packit Service 7c31a4
   * space for it on the heap. */
Packit Service 7c31a4
  if (buf_count < ARRAY_SIZE(stack_bufs)) {
Packit Service 7c31a4
    /* Use on-stack buffer array. */
Packit Service 7c31a4
    bufs = stack_bufs;
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    /* Use heap-allocated buffer array. */
Packit Service 7c31a4
    bufs = uv__calloc(buf_count, sizeof(uv_buf_t));
Packit Service 7c31a4
    if (bufs == NULL)
Packit Service 7c31a4
      return ERROR_NOT_ENOUGH_MEMORY; /* Maps to UV_ENOMEM. */
Packit Service 7c31a4
  }
Packit Service 7c31a4
  buf_index = 0;
Packit Service 7c31a4
Packit Service 7c31a4
  /* Initialize frame header and add it to the buffers list. */
Packit Service 7c31a4
  memset(&frame_header, 0, sizeof frame_header);
Packit Service 7c31a4
  bufs[buf_index++] = uv_buf_init((char*) &frame_header, sizeof frame_header);
Packit Service 7c31a4
Packit Service 7c31a4
  if (send_handle != NULL) {
Packit Service 7c31a4
    /* Add frame header flags. */
Packit Service 7c31a4
    switch (xfer_type) {
Packit Service 7c31a4
      case UV__IPC_SOCKET_XFER_TCP_CONNECTION:
Packit Service 7c31a4
        frame_header.flags |= UV__IPC_FRAME_HAS_SOCKET_XFER |
Packit Service 7c31a4
                              UV__IPC_FRAME_XFER_IS_TCP_CONNECTION;
Packit Service 7c31a4
        break;
Packit Service 7c31a4
      case UV__IPC_SOCKET_XFER_TCP_SERVER:
Packit Service 7c31a4
        frame_header.flags |= UV__IPC_FRAME_HAS_SOCKET_XFER;
Packit Service 7c31a4
        break;
Packit Service 7c31a4
      default:
Packit Service 7c31a4
        assert(0);  /* Unreachable. */
Packit Service 7c31a4
    }
Packit Service 7c31a4
    /* Add xfer info buffer. */
Packit Service 7c31a4
    bufs[buf_index++] = uv_buf_init((char*) &xfer_info, sizeof xfer_info);
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (data_length > 0) {
Packit Service 7c31a4
    /* Update frame header. */
Packit Service 7c31a4
    frame_header.flags |= UV__IPC_FRAME_HAS_DATA;
Packit Service 7c31a4
    frame_header.data_length = (uint32_t) data_length;
Packit Service 7c31a4
    /* Add data buffers to buffers list. */
Packit Service 7c31a4
    for (i = 0; i < data_buf_count; i++)
Packit Service 7c31a4
      bufs[buf_index++] = data_bufs[i];
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* Write buffers. We set the `always_copy` flag, so it is not a problem that
Packit Service 7c31a4
   * some of the written data lives on the stack. */
Packit Service 7c31a4
  err = uv__pipe_write_data(loop, req, handle, bufs, buf_count, cb, 1);
Packit Service 7c31a4
Packit Service 7c31a4
  /* If we had to heap-allocate the bufs array, free it now. */
Packit Service 7c31a4
  if (bufs != stack_bufs) {
Packit Service 7c31a4
    uv__free(bufs);
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  return err;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
int uv__pipe_write(uv_loop_t* loop,
Packit Service 7c31a4
                   uv_write_t* req,
Packit Service 7c31a4
                   uv_pipe_t* handle,
Packit Service 7c31a4
                   const uv_buf_t bufs[],
Packit Service 7c31a4
                   size_t nbufs,
Packit Service 7c31a4
                   uv_stream_t* send_handle,
Packit Service 7c31a4
                   uv_write_cb cb) {
Packit Service 7c31a4
  if (handle->ipc) {
Packit Service 7c31a4
    /* IPC pipe write: use framing protocol. */
Packit Service 7c31a4
    return uv__pipe_write_ipc(loop, req, handle, bufs, nbufs, send_handle, cb);
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    /* Non-IPC pipe write: put data on the wire directly. */
Packit Service 7c31a4
    assert(send_handle == NULL);
Packit Service 7c31a4
    return uv__pipe_write_data(loop, req, handle, bufs, nbufs, cb, 0);
Packit Service 7c31a4
  }
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void uv_pipe_read_eof(uv_loop_t* loop, uv_pipe_t* handle,
Packit Service 7c31a4
    uv_buf_t buf) {
Packit Service 7c31a4
  /* If there is an eof timer running, we don't need it any more, so discard
Packit Service 7c31a4
   * it. */
Packit Service 7c31a4
  eof_timer_destroy(handle);
Packit Service 7c31a4
Packit Service 7c31a4
  handle->flags &= ~UV_HANDLE_READABLE;
Packit Service 7c31a4
  uv_read_stop((uv_stream_t*) handle);
Packit Service 7c31a4
Packit Service 7c31a4
  handle->read_cb((uv_stream_t*) handle, UV_EOF, &buf;;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void uv_pipe_read_error(uv_loop_t* loop, uv_pipe_t* handle, int error,
Packit Service 7c31a4
    uv_buf_t buf) {
Packit Service 7c31a4
  /* If there is an eof timer running, we don't need it any more, so discard
Packit Service 7c31a4
   * it. */
Packit Service 7c31a4
  eof_timer_destroy(handle);
Packit Service 7c31a4
Packit Service 7c31a4
  uv_read_stop((uv_stream_t*) handle);
Packit Service 7c31a4
Packit Service 7c31a4
  handle->read_cb((uv_stream_t*)handle, uv_translate_sys_error(error), &buf;;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void uv_pipe_read_error_or_eof(uv_loop_t* loop, uv_pipe_t* handle,
Packit Service 7c31a4
    int error, uv_buf_t buf) {
Packit Service 7c31a4
  if (error == ERROR_BROKEN_PIPE) {
Packit Service 7c31a4
    uv_pipe_read_eof(loop, handle, buf);
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    uv_pipe_read_error(loop, handle, error, buf);
Packit Service 7c31a4
  }
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void uv__pipe_queue_ipc_xfer_info(
Packit Service 7c31a4
    uv_pipe_t* handle,
Packit Service 7c31a4
    uv__ipc_socket_xfer_type_t xfer_type,
Packit Service 7c31a4
    uv__ipc_socket_xfer_info_t* xfer_info) {
Packit Service 7c31a4
  uv__ipc_xfer_queue_item_t* item;
Packit Service 7c31a4
Packit Service 7c31a4
  item = (uv__ipc_xfer_queue_item_t*) uv__malloc(sizeof(*item));
Packit Service 7c31a4
  if (item == NULL)
Packit Service 7c31a4
    uv_fatal_error(ERROR_OUTOFMEMORY, "uv__malloc");
Packit Service 7c31a4
Packit Service 7c31a4
  item->xfer_type = xfer_type;
Packit Service 7c31a4
  item->xfer_info = *xfer_info;
Packit Service 7c31a4
Packit Service 7c31a4
  QUEUE_INSERT_TAIL(&handle->pipe.conn.ipc_xfer_queue, &item->member);
Packit Service 7c31a4
  handle->pipe.conn.ipc_xfer_queue_length++;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
/* Read an exact number of bytes from a pipe. If an error or end-of-file is
Packit Service 7c31a4
 * encountered before the requested number of bytes are read, an error is
Packit Service 7c31a4
 * returned. */
Packit Service 7c31a4
static int uv__pipe_read_exactly(HANDLE h, void* buffer, DWORD count) {
Packit Service 7c31a4
  DWORD bytes_read, bytes_read_now;
Packit Service 7c31a4
Packit Service 7c31a4
  bytes_read = 0;
Packit Service 7c31a4
  while (bytes_read < count) {
Packit Service 7c31a4
    if (!ReadFile(h,
Packit Service 7c31a4
                  (char*) buffer + bytes_read,
Packit Service 7c31a4
                  count - bytes_read,
Packit Service 7c31a4
                  &bytes_read_now,
Packit Service 7c31a4
                  NULL)) {
Packit Service 7c31a4
      return GetLastError();
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    bytes_read += bytes_read_now;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  assert(bytes_read == count);
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static DWORD uv__pipe_read_data(uv_loop_t* loop,
Packit Service 7c31a4
                                uv_pipe_t* handle,
Packit Service 7c31a4
                                DWORD suggested_bytes,
Packit Service 7c31a4
                                DWORD max_bytes) {
Packit Service 7c31a4
  DWORD bytes_read;
Packit Service 7c31a4
  uv_buf_t buf;
Packit Service 7c31a4
Packit Service 7c31a4
  /* Ask the user for a buffer to read data into. */
Packit Service 7c31a4
  buf = uv_buf_init(NULL, 0);
Packit Service 7c31a4
  handle->alloc_cb((uv_handle_t*) handle, suggested_bytes, &buf;;
Packit Service 7c31a4
  if (buf.base == NULL || buf.len == 0) {
Packit Service 7c31a4
    handle->read_cb((uv_stream_t*) handle, UV_ENOBUFS, &buf;;
Packit Service 7c31a4
    return 0; /* Break out of read loop. */
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* Ensure we read at most the smaller of:
Packit Service 7c31a4
   *   (a) the length of the user-allocated buffer.
Packit Service 7c31a4
   *   (b) the maximum data length as specified by the `max_bytes` argument.
Packit Service 7c31a4
   */
Packit Service 7c31a4
  if (max_bytes > buf.len)
Packit Service 7c31a4
    max_bytes = buf.len;
Packit Service 7c31a4
Packit Service 7c31a4
  /* Read into the user buffer. */
Packit Service 7c31a4
  if (!ReadFile(handle->handle, buf.base, max_bytes, &bytes_read, NULL)) {
Packit Service 7c31a4
    uv_pipe_read_error_or_eof(loop, handle, GetLastError(), buf);
Packit Service 7c31a4
    return 0; /* Break out of read loop. */
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* Call the read callback. */
Packit Service 7c31a4
  handle->read_cb((uv_stream_t*) handle, bytes_read, &buf;;
Packit Service 7c31a4
Packit Service 7c31a4
  return bytes_read;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static DWORD uv__pipe_read_ipc(uv_loop_t* loop, uv_pipe_t* handle) {
Packit Service 7c31a4
  uint32_t* data_remaining = &handle->pipe.conn.ipc_data_frame.payload_remaining;
Packit Service 7c31a4
  int err;
Packit Service 7c31a4
Packit Service 7c31a4
  if (*data_remaining > 0) {
Packit Service 7c31a4
    /* Read frame data payload. */
Packit Service 7c31a4
    DWORD bytes_read =
Packit Service 7c31a4
        uv__pipe_read_data(loop, handle, *data_remaining, *data_remaining);
Packit Service 7c31a4
    *data_remaining -= bytes_read;
Packit Service 7c31a4
    return bytes_read;
Packit Service 7c31a4
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    /* Start of a new IPC frame. */
Packit Service 7c31a4
    uv__ipc_frame_header_t frame_header;
Packit Service 7c31a4
    uint32_t xfer_flags;
Packit Service 7c31a4
    uv__ipc_socket_xfer_type_t xfer_type;
Packit Service 7c31a4
    uv__ipc_socket_xfer_info_t xfer_info;
Packit Service 7c31a4
Packit Service 7c31a4
    /* Read the IPC frame header. */
Packit Service 7c31a4
    err = uv__pipe_read_exactly(
Packit Service 7c31a4
        handle->handle, &frame_header, sizeof frame_header);
Packit Service 7c31a4
    if (err)
Packit Service 7c31a4
      goto error;
Packit Service 7c31a4
Packit Service 7c31a4
    /* Validate that flags are valid. */
Packit Service 7c31a4
    if ((frame_header.flags & ~UV__IPC_FRAME_VALID_FLAGS) != 0)
Packit Service 7c31a4
      goto invalid;
Packit Service 7c31a4
    /* Validate that reserved2 is zero. */
Packit Service 7c31a4
    if (frame_header.reserved2 != 0)
Packit Service 7c31a4
      goto invalid;
Packit Service 7c31a4
Packit Service 7c31a4
    /* Parse xfer flags. */
Packit Service 7c31a4
    xfer_flags = frame_header.flags & UV__IPC_FRAME_XFER_FLAGS;
Packit Service 7c31a4
    if (xfer_flags & UV__IPC_FRAME_HAS_SOCKET_XFER) {
Packit Service 7c31a4
      /* Socket coming -- determine the type. */
Packit Service 7c31a4
      xfer_type = xfer_flags & UV__IPC_FRAME_XFER_IS_TCP_CONNECTION
Packit Service 7c31a4
                      ? UV__IPC_SOCKET_XFER_TCP_CONNECTION
Packit Service 7c31a4
                      : UV__IPC_SOCKET_XFER_TCP_SERVER;
Packit Service 7c31a4
    } else if (xfer_flags == 0) {
Packit Service 7c31a4
      /* No socket. */
Packit Service 7c31a4
      xfer_type = UV__IPC_SOCKET_XFER_NONE;
Packit Service 7c31a4
    } else {
Packit Service 7c31a4
      /* Invalid flags. */
Packit Service 7c31a4
      goto invalid;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    /* Parse data frame information. */
Packit Service 7c31a4
    if (frame_header.flags & UV__IPC_FRAME_HAS_DATA) {
Packit Service 7c31a4
      *data_remaining = frame_header.data_length;
Packit Service 7c31a4
    } else if (frame_header.data_length != 0) {
Packit Service 7c31a4
      /* Data length greater than zero but data flag not set -- invalid. */
Packit Service 7c31a4
      goto invalid;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    /* If no socket xfer info follows, return here. Data will be read in a
Packit Service 7c31a4
     * subsequent invocation of uv__pipe_read_ipc(). */
Packit Service 7c31a4
    if (xfer_type == UV__IPC_SOCKET_XFER_NONE)
Packit Service 7c31a4
      return sizeof frame_header; /* Number of bytes read. */
Packit Service 7c31a4
Packit Service 7c31a4
    /* Read transferred socket information. */
Packit Service 7c31a4
    err = uv__pipe_read_exactly(handle->handle, &xfer_info, sizeof xfer_info);
Packit Service 7c31a4
    if (err)
Packit Service 7c31a4
      goto error;
Packit Service 7c31a4
Packit Service 7c31a4
    /* Store the pending socket info. */
Packit Service 7c31a4
    uv__pipe_queue_ipc_xfer_info(handle, xfer_type, &xfer_info);
Packit Service 7c31a4
Packit Service 7c31a4
    /* Return number of bytes read. */
Packit Service 7c31a4
    return sizeof frame_header + sizeof xfer_info;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
invalid:
Packit Service 7c31a4
  /* Invalid frame. */
Packit Service 7c31a4
  err = WSAECONNABORTED; /* Maps to UV_ECONNABORTED. */
Packit Service 7c31a4
Packit Service 7c31a4
error:
Packit Service 7c31a4
  uv_pipe_read_error_or_eof(loop, handle, err, uv_null_buf_);
Packit Service 7c31a4
  return 0; /* Break out of read loop. */
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
void uv_process_pipe_read_req(uv_loop_t* loop,
Packit Service 7c31a4
                              uv_pipe_t* handle,
Packit Service 7c31a4
                              uv_req_t* req) {
Packit Service 7c31a4
  assert(handle->type == UV_NAMED_PIPE);
Packit Service 7c31a4
Packit Service 7c31a4
  handle->flags &= ~(UV_HANDLE_READ_PENDING | UV_HANDLE_CANCELLATION_PENDING);
Packit Service 7c31a4
  DECREASE_PENDING_REQ_COUNT(handle);
Packit Service 7c31a4
  eof_timer_stop(handle);
Packit Service 7c31a4
Packit Service 7c31a4
  /* At this point, we're done with bookkeeping. If the user has stopped
Packit Service 7c31a4
   * reading the pipe in the meantime, there is nothing left to do, since there
Packit Service 7c31a4
   * is no callback that we can call. */
Packit Service 7c31a4
  if (!(handle->flags & UV_HANDLE_READING))
Packit Service 7c31a4
    return;
Packit Service 7c31a4
Packit Service 7c31a4
  if (!REQ_SUCCESS(req)) {
Packit Service 7c31a4
    /* An error occurred doing the zero-read. */
Packit Service 7c31a4
    DWORD err = GET_REQ_ERROR(req);
Packit Service 7c31a4
Packit Service 7c31a4
    /* If the read was cancelled by uv__pipe_interrupt_read(), the request may
Packit Service 7c31a4
     * indicate an ERROR_OPERATION_ABORTED error. This error isn't relevant to
Packit Service 7c31a4
     * the user; we'll start a new zero-read at the end of this function. */
Packit Service 7c31a4
    if (err != ERROR_OPERATION_ABORTED)
Packit Service 7c31a4
      uv_pipe_read_error_or_eof(loop, handle, err, uv_null_buf_);
Packit Service 7c31a4
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    /* The zero-read completed without error, indicating there is data
Packit Service 7c31a4
     * available in the kernel buffer. */
Packit Service 7c31a4
    DWORD avail;
Packit Service 7c31a4
Packit Service 7c31a4
    /* Get the number of bytes available. */
Packit Service 7c31a4
    avail = 0;
Packit Service 7c31a4
    if (!PeekNamedPipe(handle->handle, NULL, 0, NULL, &avail, NULL))
Packit Service 7c31a4
      uv_pipe_read_error_or_eof(loop, handle, GetLastError(), uv_null_buf_);
Packit Service 7c31a4
Packit Service 7c31a4
    /* Read until we've either read all the bytes available, or the 'reading'
Packit Service 7c31a4
     * flag is cleared. */
Packit Service 7c31a4
    while (avail > 0 && handle->flags & UV_HANDLE_READING) {
Packit Service 7c31a4
      /* Depending on the type of pipe, read either IPC frames or raw data. */
Packit Service 7c31a4
      DWORD bytes_read =
Packit Service 7c31a4
          handle->ipc ? uv__pipe_read_ipc(loop, handle)
Packit Service 7c31a4
                      : uv__pipe_read_data(loop, handle, avail, (DWORD) -1);
Packit Service 7c31a4
Packit Service 7c31a4
      /* If no bytes were read, treat this as an indication that an error
Packit Service 7c31a4
       * occurred, and break out of the read loop. */
Packit Service 7c31a4
      if (bytes_read == 0)
Packit Service 7c31a4
        break;
Packit Service 7c31a4
Packit Service 7c31a4
      /* It is possible that more bytes were read than we thought were
Packit Service 7c31a4
       * available. To prevent `avail` from underflowing, break out of the loop
Packit Service 7c31a4
       * if this is the case. */
Packit Service 7c31a4
      if (bytes_read > avail)
Packit Service 7c31a4
        break;
Packit Service 7c31a4
Packit Service 7c31a4
      /* Recompute the number of bytes available. */
Packit Service 7c31a4
      avail -= bytes_read;
Packit Service 7c31a4
    }
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* Start another zero-read request if necessary. */
Packit Service 7c31a4
  if ((handle->flags & UV_HANDLE_READING) &&
Packit Service 7c31a4
      !(handle->flags & UV_HANDLE_READ_PENDING)) {
Packit Service 7c31a4
    uv_pipe_queue_read(loop, handle);
Packit Service 7c31a4
  }
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
void uv_process_pipe_write_req(uv_loop_t* loop, uv_pipe_t* handle,
Packit Service 7c31a4
    uv_write_t* req) {
Packit Service 7c31a4
  int err;
Packit Service 7c31a4
Packit Service 7c31a4
  assert(handle->type == UV_NAMED_PIPE);
Packit Service 7c31a4
Packit Service 7c31a4
  assert(handle->write_queue_size >= req->u.io.queued_bytes);
Packit Service 7c31a4
  handle->write_queue_size -= req->u.io.queued_bytes;
Packit Service 7c31a4
Packit Service 7c31a4
  UNREGISTER_HANDLE_REQ(loop, handle, req);
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_EMULATE_IOCP) {
Packit Service 7c31a4
    if (req->wait_handle != INVALID_HANDLE_VALUE) {
Packit Service 7c31a4
      UnregisterWait(req->wait_handle);
Packit Service 7c31a4
      req->wait_handle = INVALID_HANDLE_VALUE;
Packit Service 7c31a4
    }
Packit Service 7c31a4
    if (req->event_handle) {
Packit Service 7c31a4
      CloseHandle(req->event_handle);
Packit Service 7c31a4
      req->event_handle = NULL;
Packit Service 7c31a4
    }
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  err = GET_REQ_ERROR(req);
Packit Service 7c31a4
Packit Service 7c31a4
  /* If this was a coalesced write, extract pointer to the user_provided
Packit Service 7c31a4
   * uv_write_t structure so we can pass the expected pointer to the callback,
Packit Service 7c31a4
   * then free the heap-allocated write req. */
Packit Service 7c31a4
  if (req->coalesced) {
Packit Service 7c31a4
    uv__coalesced_write_t* coalesced_write =
Packit Service 7c31a4
        container_of(req, uv__coalesced_write_t, req);
Packit Service 7c31a4
    req = coalesced_write->user_req;
Packit Service 7c31a4
    uv__free(coalesced_write);
Packit Service 7c31a4
  }
Packit Service 7c31a4
  if (req->cb) {
Packit Service 7c31a4
    req->cb(req, uv_translate_sys_error(err));
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  handle->stream.conn.write_reqs_pending--;
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_NON_OVERLAPPED_PIPE &&
Packit Service 7c31a4
      handle->pipe.conn.non_overlapped_writes_tail) {
Packit Service 7c31a4
    assert(handle->stream.conn.write_reqs_pending > 0);
Packit Service 7c31a4
    uv_queue_non_overlapped_write(handle);
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->stream.conn.shutdown_req != NULL &&
Packit Service 7c31a4
      handle->stream.conn.write_reqs_pending == 0) {
Packit Service 7c31a4
    uv_want_endgame(loop, (uv_handle_t*)handle);
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  DECREASE_PENDING_REQ_COUNT(handle);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
void uv_process_pipe_accept_req(uv_loop_t* loop, uv_pipe_t* handle,
Packit Service 7c31a4
    uv_req_t* raw_req) {
Packit Service 7c31a4
  uv_pipe_accept_t* req = (uv_pipe_accept_t*) raw_req;
Packit Service 7c31a4
Packit Service 7c31a4
  assert(handle->type == UV_NAMED_PIPE);
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_CLOSING) {
Packit Service 7c31a4
    /* The req->pipeHandle should be freed already in uv_pipe_cleanup(). */
Packit Service 7c31a4
    assert(req->pipeHandle == INVALID_HANDLE_VALUE);
Packit Service 7c31a4
    DECREASE_PENDING_REQ_COUNT(handle);
Packit Service 7c31a4
    return;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (REQ_SUCCESS(req)) {
Packit Service 7c31a4
    assert(req->pipeHandle != INVALID_HANDLE_VALUE);
Packit Service 7c31a4
    req->next_pending = handle->pipe.serv.pending_accepts;
Packit Service 7c31a4
    handle->pipe.serv.pending_accepts = req;
Packit Service 7c31a4
Packit Service 7c31a4
    if (handle->stream.serv.connection_cb) {
Packit Service 7c31a4
      handle->stream.serv.connection_cb((uv_stream_t*)handle, 0);
Packit Service 7c31a4
    }
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    if (req->pipeHandle != INVALID_HANDLE_VALUE) {
Packit Service 7c31a4
      CloseHandle(req->pipeHandle);
Packit Service 7c31a4
      req->pipeHandle = INVALID_HANDLE_VALUE;
Packit Service 7c31a4
    }
Packit Service 7c31a4
    if (!(handle->flags & UV_HANDLE_CLOSING)) {
Packit Service 7c31a4
      uv_pipe_queue_accept(loop, handle, req, FALSE);
Packit Service 7c31a4
    }
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  DECREASE_PENDING_REQ_COUNT(handle);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
void uv_process_pipe_connect_req(uv_loop_t* loop, uv_pipe_t* handle,
Packit Service 7c31a4
    uv_connect_t* req) {
Packit Service 7c31a4
  int err;
Packit Service 7c31a4
Packit Service 7c31a4
  assert(handle->type == UV_NAMED_PIPE);
Packit Service 7c31a4
Packit Service 7c31a4
  UNREGISTER_HANDLE_REQ(loop, handle, req);
Packit Service 7c31a4
Packit Service 7c31a4
  if (req->cb) {
Packit Service 7c31a4
    err = 0;
Packit Service 7c31a4
    if (REQ_SUCCESS(req)) {
Packit Service 7c31a4
      uv_pipe_connection_init(handle);
Packit Service 7c31a4
    } else {
Packit Service 7c31a4
      err = GET_REQ_ERROR(req);
Packit Service 7c31a4
    }
Packit Service 7c31a4
    req->cb(req, uv_translate_sys_error(err));
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  DECREASE_PENDING_REQ_COUNT(handle);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
void uv_process_pipe_shutdown_req(uv_loop_t* loop, uv_pipe_t* handle,
Packit Service 7c31a4
    uv_shutdown_t* req) {
Packit Service 7c31a4
  assert(handle->type == UV_NAMED_PIPE);
Packit Service 7c31a4
Packit Service 7c31a4
  UNREGISTER_HANDLE_REQ(loop, handle, req);
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_READABLE) {
Packit Service 7c31a4
    /* Initialize and optionally start the eof timer. Only do this if the pipe
Packit Service 7c31a4
     * is readable and we haven't seen EOF come in ourselves. */
Packit Service 7c31a4
    eof_timer_init(handle);
Packit Service 7c31a4
Packit Service 7c31a4
    /* If reading start the timer right now. Otherwise uv_pipe_queue_read will
Packit Service 7c31a4
     * start it. */
Packit Service 7c31a4
    if (handle->flags & UV_HANDLE_READ_PENDING) {
Packit Service 7c31a4
      eof_timer_start(handle);
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    /* This pipe is not readable. We can just close it to let the other end
Packit Service 7c31a4
     * know that we're done writing. */
Packit Service 7c31a4
    close_pipe(handle);
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (req->cb) {
Packit Service 7c31a4
    req->cb(req, 0);
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  DECREASE_PENDING_REQ_COUNT(handle);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void eof_timer_init(uv_pipe_t* pipe) {
Packit Service 7c31a4
  int r;
Packit Service 7c31a4
Packit Service 7c31a4
  assert(pipe->pipe.conn.eof_timer == NULL);
Packit Service 7c31a4
  assert(pipe->flags & UV_HANDLE_CONNECTION);
Packit Service 7c31a4
Packit Service 7c31a4
  pipe->pipe.conn.eof_timer = (uv_timer_t*) uv__malloc(sizeof *pipe->pipe.conn.eof_timer);
Packit Service 7c31a4
Packit Service 7c31a4
  r = uv_timer_init(pipe->loop, pipe->pipe.conn.eof_timer);
Packit Service 7c31a4
  assert(r == 0); /* timers can't fail */
Packit Service 7c31a4
  pipe->pipe.conn.eof_timer->data = pipe;
Packit Service 7c31a4
  uv_unref((uv_handle_t*) pipe->pipe.conn.eof_timer);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void eof_timer_start(uv_pipe_t* pipe) {
Packit Service 7c31a4
  assert(pipe->flags & UV_HANDLE_CONNECTION);
Packit Service 7c31a4
Packit Service 7c31a4
  if (pipe->pipe.conn.eof_timer != NULL) {
Packit Service 7c31a4
    uv_timer_start(pipe->pipe.conn.eof_timer, eof_timer_cb, eof_timeout, 0);
Packit Service 7c31a4
  }
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void eof_timer_stop(uv_pipe_t* pipe) {
Packit Service 7c31a4
  assert(pipe->flags & UV_HANDLE_CONNECTION);
Packit Service 7c31a4
Packit Service 7c31a4
  if (pipe->pipe.conn.eof_timer != NULL) {
Packit Service 7c31a4
    uv_timer_stop(pipe->pipe.conn.eof_timer);
Packit Service 7c31a4
  }
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void eof_timer_cb(uv_timer_t* timer) {
Packit Service 7c31a4
  uv_pipe_t* pipe = (uv_pipe_t*) timer->data;
Packit Service 7c31a4
  uv_loop_t* loop = timer->loop;
Packit Service 7c31a4
Packit Service 7c31a4
  assert(pipe->type == UV_NAMED_PIPE);
Packit Service 7c31a4
Packit Service 7c31a4
  /* This should always be true, since we start the timer only in
Packit Service 7c31a4
   * uv_pipe_queue_read after successfully calling ReadFile, or in
Packit Service 7c31a4
   * uv_process_pipe_shutdown_req if a read is pending, and we always
Packit Service 7c31a4
   * immediately stop the timer in uv_process_pipe_read_req. */
Packit Service 7c31a4
  assert(pipe->flags & UV_HANDLE_READ_PENDING);
Packit Service 7c31a4
Packit Service 7c31a4
  /* If there are many packets coming off the iocp then the timer callback may
Packit Service 7c31a4
   * be called before the read request is coming off the queue. Therefore we
Packit Service 7c31a4
   * check here if the read request has completed but will be processed later.
Packit Service 7c31a4
   */
Packit Service 7c31a4
  if ((pipe->flags & UV_HANDLE_READ_PENDING) &&
Packit Service 7c31a4
      HasOverlappedIoCompleted(&pipe->read_req.u.io.overlapped)) {
Packit Service 7c31a4
    return;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* Force both ends off the pipe. */
Packit Service 7c31a4
  close_pipe(pipe);
Packit Service 7c31a4
Packit Service 7c31a4
  /* Stop reading, so the pending read that is going to fail will not be
Packit Service 7c31a4
   * reported to the user. */
Packit Service 7c31a4
  uv_read_stop((uv_stream_t*) pipe);
Packit Service 7c31a4
Packit Service 7c31a4
  /* Report the eof and update flags. This will get reported even if the user
Packit Service 7c31a4
   * stopped reading in the meantime. TODO: is that okay? */
Packit Service 7c31a4
  uv_pipe_read_eof(loop, pipe, uv_null_buf_);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void eof_timer_destroy(uv_pipe_t* pipe) {
Packit Service 7c31a4
  assert(pipe->flags & UV_HANDLE_CONNECTION);
Packit Service 7c31a4
Packit Service 7c31a4
  if (pipe->pipe.conn.eof_timer) {
Packit Service 7c31a4
    uv_close((uv_handle_t*) pipe->pipe.conn.eof_timer, eof_timer_close_cb);
Packit Service 7c31a4
    pipe->pipe.conn.eof_timer = NULL;
Packit Service 7c31a4
  }
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static void eof_timer_close_cb(uv_handle_t* handle) {
Packit Service 7c31a4
  assert(handle->type == UV_TIMER);
Packit Service 7c31a4
  uv__free(handle);
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
int uv_pipe_open(uv_pipe_t* pipe, uv_file file) {
Packit Service 7c31a4
  HANDLE os_handle = uv__get_osfhandle(file);
Packit Service 7c31a4
  NTSTATUS nt_status;
Packit Service 7c31a4
  IO_STATUS_BLOCK io_status;
Packit Service 7c31a4
  FILE_ACCESS_INFORMATION access;
Packit Service 7c31a4
  DWORD duplex_flags = 0;
Packit Service 7c31a4
Packit Service 7c31a4
  if (os_handle == INVALID_HANDLE_VALUE)
Packit Service 7c31a4
    return UV_EBADF;
Packit Service 7c31a4
Packit Service 7c31a4
  uv__once_init();
Packit Service 7c31a4
  /* In order to avoid closing a stdio file descriptor 0-2, duplicate the
Packit Service 7c31a4
   * underlying OS handle and forget about the original fd.
Packit Service 7c31a4
   * We could also opt to use the original OS handle and just never close it,
Packit Service 7c31a4
   * but then there would be no reliable way to cancel pending read operations
Packit Service 7c31a4
   * upon close.
Packit Service 7c31a4
   */
Packit Service 7c31a4
  if (file <= 2) {
Packit Service 7c31a4
    if (!DuplicateHandle(INVALID_HANDLE_VALUE,
Packit Service 7c31a4
                         os_handle,
Packit Service 7c31a4
                         INVALID_HANDLE_VALUE,
Packit Service 7c31a4
                         &os_handle,
Packit Service 7c31a4
                         0,
Packit Service 7c31a4
                         FALSE,
Packit Service 7c31a4
                         DUPLICATE_SAME_ACCESS))
Packit Service 7c31a4
      return uv_translate_sys_error(GetLastError());
Packit Service 7c31a4
    file = -1;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* Determine what kind of permissions we have on this handle.
Packit Service 7c31a4
   * Cygwin opens the pipe in message mode, but we can support it,
Packit Service 7c31a4
   * just query the access flags and set the stream flags accordingly.
Packit Service 7c31a4
   */
Packit Service 7c31a4
  nt_status = pNtQueryInformationFile(os_handle,
Packit Service 7c31a4
                                      &io_status,
Packit Service 7c31a4
                                      &access,
Packit Service 7c31a4
                                      sizeof(access),
Packit Service 7c31a4
                                      FileAccessInformation);
Packit Service 7c31a4
  if (nt_status != STATUS_SUCCESS)
Packit Service 7c31a4
    return UV_EINVAL;
Packit Service 7c31a4
Packit Service 7c31a4
  if (pipe->ipc) {
Packit Service 7c31a4
    if (!(access.AccessFlags & FILE_WRITE_DATA) ||
Packit Service 7c31a4
        !(access.AccessFlags & FILE_READ_DATA)) {
Packit Service 7c31a4
      return UV_EINVAL;
Packit Service 7c31a4
    }
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (access.AccessFlags & FILE_WRITE_DATA)
Packit Service 7c31a4
    duplex_flags |= UV_HANDLE_WRITABLE;
Packit Service 7c31a4
  if (access.AccessFlags & FILE_READ_DATA)
Packit Service 7c31a4
    duplex_flags |= UV_HANDLE_READABLE;
Packit Service 7c31a4
Packit Service 7c31a4
  if (os_handle == INVALID_HANDLE_VALUE ||
Packit Service 7c31a4
      uv_set_pipe_handle(pipe->loop,
Packit Service 7c31a4
                         pipe,
Packit Service 7c31a4
                         os_handle,
Packit Service 7c31a4
                         file,
Packit Service 7c31a4
                         duplex_flags) == -1) {
Packit Service 7c31a4
    return UV_EINVAL;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  uv_pipe_connection_init(pipe);
Packit Service 7c31a4
Packit Service 7c31a4
  if (pipe->ipc) {
Packit Service 7c31a4
    assert(!(pipe->flags & UV_HANDLE_NON_OVERLAPPED_PIPE));
Packit Service 7c31a4
    pipe->pipe.conn.ipc_remote_pid = uv_os_getppid();
Packit Service e2ebee
    assert(pipe->pipe.conn.ipc_remote_pid != (DWORD)(uv_pid_t) -1);
Packit Service 7c31a4
  }
Packit Service 7c31a4
  return 0;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
static int uv__pipe_getname(const uv_pipe_t* handle, char* buffer, size_t* size) {
Packit Service 7c31a4
  NTSTATUS nt_status;
Packit Service 7c31a4
  IO_STATUS_BLOCK io_status;
Packit Service 7c31a4
  FILE_NAME_INFORMATION tmp_name_info;
Packit Service 7c31a4
  FILE_NAME_INFORMATION* name_info;
Packit Service 7c31a4
  WCHAR* name_buf;
Packit Service 7c31a4
  unsigned int addrlen;
Packit Service 7c31a4
  unsigned int name_size;
Packit Service 7c31a4
  unsigned int name_len;
Packit Service 7c31a4
  int err;
Packit Service 7c31a4
Packit Service 7c31a4
  uv__once_init();
Packit Service 7c31a4
  name_info = NULL;
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->handle == INVALID_HANDLE_VALUE) {
Packit Service 7c31a4
    *size = 0;
Packit Service 7c31a4
    return UV_EINVAL;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  /* NtQueryInformationFile will block if another thread is performing a
Packit Service 7c31a4
   * blocking operation on the queried handle. If the pipe handle is
Packit Service 7c31a4
   * synchronous, there may be a worker thread currently calling ReadFile() on
Packit Service 7c31a4
   * the pipe handle, which could cause a deadlock. To avoid this, interrupt
Packit Service 7c31a4
   * the read. */
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_CONNECTION &&
Packit Service 7c31a4
      handle->flags & UV_HANDLE_NON_OVERLAPPED_PIPE) {
Packit Service 7c31a4
    uv__pipe_interrupt_read((uv_pipe_t*) handle); /* cast away const warning */
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  nt_status = pNtQueryInformationFile(handle->handle,
Packit Service 7c31a4
                                      &io_status,
Packit Service 7c31a4
                                      &tmp_name_info,
Packit Service 7c31a4
                                      sizeof tmp_name_info,
Packit Service 7c31a4
                                      FileNameInformation);
Packit Service 7c31a4
  if (nt_status == STATUS_BUFFER_OVERFLOW) {
Packit Service 7c31a4
    name_size = sizeof(*name_info) + tmp_name_info.FileNameLength;
Packit Service 7c31a4
    name_info = uv__malloc(name_size);
Packit Service 7c31a4
    if (!name_info) {
Packit Service 7c31a4
      *size = 0;
Packit Service 7c31a4
      err = UV_ENOMEM;
Packit Service 7c31a4
      goto cleanup;
Packit Service 7c31a4
    }
Packit Service 7c31a4
Packit Service 7c31a4
    nt_status = pNtQueryInformationFile(handle->handle,
Packit Service 7c31a4
                                        &io_status,
Packit Service 7c31a4
                                        name_info,
Packit Service 7c31a4
                                        name_size,
Packit Service 7c31a4
                                        FileNameInformation);
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (nt_status != STATUS_SUCCESS) {
Packit Service 7c31a4
    *size = 0;
Packit Service 7c31a4
    err = uv_translate_sys_error(pRtlNtStatusToDosError(nt_status));
Packit Service 7c31a4
    goto error;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (!name_info) {
Packit Service 7c31a4
    /* the struct on stack was used */
Packit Service 7c31a4
    name_buf = tmp_name_info.FileName;
Packit Service 7c31a4
    name_len = tmp_name_info.FileNameLength;
Packit Service 7c31a4
  } else {
Packit Service 7c31a4
    name_buf = name_info->FileName;
Packit Service 7c31a4
    name_len = name_info->FileNameLength;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (name_len == 0) {
Packit Service 7c31a4
    *size = 0;
Packit Service 7c31a4
    err = 0;
Packit Service 7c31a4
    goto error;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  name_len /= sizeof(WCHAR);
Packit Service 7c31a4
Packit Service 7c31a4
  /* check how much space we need */
Packit Service 7c31a4
  addrlen = WideCharToMultiByte(CP_UTF8,
Packit Service 7c31a4
                                0,
Packit Service 7c31a4
                                name_buf,
Packit Service 7c31a4
                                name_len,
Packit Service 7c31a4
                                NULL,
Packit Service 7c31a4
                                0,
Packit Service 7c31a4
                                NULL,
Packit Service 7c31a4
                                NULL);
Packit Service 7c31a4
  if (!addrlen) {
Packit Service 7c31a4
    *size = 0;
Packit Service 7c31a4
    err = uv_translate_sys_error(GetLastError());
Packit Service 7c31a4
    goto error;
Packit Service 7c31a4
  } else if (pipe_prefix_len + addrlen >= *size) {
Packit Service 7c31a4
    /* "\\\\.\\pipe" + name */
Packit Service 7c31a4
    *size = pipe_prefix_len + addrlen + 1;
Packit Service 7c31a4
    err = UV_ENOBUFS;
Packit Service 7c31a4
    goto error;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  memcpy(buffer, pipe_prefix, pipe_prefix_len);
Packit Service 7c31a4
  addrlen = WideCharToMultiByte(CP_UTF8,
Packit Service 7c31a4
                                0,
Packit Service 7c31a4
                                name_buf,
Packit Service 7c31a4
                                name_len,
Packit Service 7c31a4
                                buffer+pipe_prefix_len,
Packit Service 7c31a4
                                *size-pipe_prefix_len,
Packit Service 7c31a4
                                NULL,
Packit Service 7c31a4
                                NULL);
Packit Service 7c31a4
  if (!addrlen) {
Packit Service 7c31a4
    *size = 0;
Packit Service 7c31a4
    err = uv_translate_sys_error(GetLastError());
Packit Service 7c31a4
    goto error;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  addrlen += pipe_prefix_len;
Packit Service 7c31a4
  *size = addrlen;
Packit Service 7c31a4
  buffer[addrlen] = '\0';
Packit Service 7c31a4
Packit Service 7c31a4
  err = 0;
Packit Service 7c31a4
Packit Service 7c31a4
error:
Packit Service 7c31a4
  uv__free(name_info);
Packit Service 7c31a4
Packit Service 7c31a4
cleanup:
Packit Service 7c31a4
  return err;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
int uv_pipe_pending_count(uv_pipe_t* handle) {
Packit Service 7c31a4
  if (!handle->ipc)
Packit Service 7c31a4
    return 0;
Packit Service 7c31a4
  return handle->pipe.conn.ipc_xfer_queue_length;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
int uv_pipe_getsockname(const uv_pipe_t* handle, char* buffer, size_t* size) {
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_BOUND)
Packit Service 7c31a4
    return uv__pipe_getname(handle, buffer, size);
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_CONNECTION ||
Packit Service 7c31a4
      handle->handle != INVALID_HANDLE_VALUE) {
Packit Service 7c31a4
    *size = 0;
Packit Service 7c31a4
    return 0;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  return UV_EBADF;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
int uv_pipe_getpeername(const uv_pipe_t* handle, char* buffer, size_t* size) {
Packit Service 7c31a4
  /* emulate unix behaviour */
Packit Service 7c31a4
  if (handle->flags & UV_HANDLE_BOUND)
Packit Service 7c31a4
    return UV_ENOTCONN;
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle->handle != INVALID_HANDLE_VALUE)
Packit Service 7c31a4
    return uv__pipe_getname(handle, buffer, size);
Packit Service 7c31a4
Packit Service 7c31a4
  return UV_EBADF;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
Packit Service 7c31a4
uv_handle_type uv_pipe_pending_type(uv_pipe_t* handle) {
Packit Service 7c31a4
  if (!handle->ipc)
Packit Service 7c31a4
    return UV_UNKNOWN_HANDLE;
Packit Service 7c31a4
  if (handle->pipe.conn.ipc_xfer_queue_length == 0)
Packit Service 7c31a4
    return UV_UNKNOWN_HANDLE;
Packit Service 7c31a4
  else
Packit Service 7c31a4
    return UV_TCP;
Packit Service 7c31a4
}
Packit Service 7c31a4
Packit Service 7c31a4
int uv_pipe_chmod(uv_pipe_t* handle, int mode) {
Packit Service 7c31a4
  SID_IDENTIFIER_AUTHORITY sid_world = { SECURITY_WORLD_SID_AUTHORITY };
Packit Service 7c31a4
  PACL old_dacl, new_dacl;
Packit Service 7c31a4
  PSECURITY_DESCRIPTOR sd;
Packit Service 7c31a4
  EXPLICIT_ACCESS ea;
Packit Service 7c31a4
  PSID everyone;
Packit Service 7c31a4
  int error;
Packit Service 7c31a4
Packit Service 7c31a4
  if (handle == NULL || handle->handle == INVALID_HANDLE_VALUE)
Packit Service 7c31a4
    return UV_EBADF;
Packit Service 7c31a4
Packit Service 7c31a4
  if (mode != UV_READABLE &&
Packit Service 7c31a4
      mode != UV_WRITABLE &&
Packit Service 7c31a4
      mode != (UV_WRITABLE | UV_READABLE))
Packit Service 7c31a4
    return UV_EINVAL;
Packit Service 7c31a4
Packit Service 7c31a4
  if (!AllocateAndInitializeSid(&sid_world,
Packit Service 7c31a4
                                1,
Packit Service 7c31a4
                                SECURITY_WORLD_RID,
Packit Service 7c31a4
                                0, 0, 0, 0, 0, 0, 0,
Packit Service 7c31a4
                                &everyone)) {
Packit Service 7c31a4
    error = GetLastError();
Packit Service 7c31a4
    goto done;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (GetSecurityInfo(handle->handle,
Packit Service 7c31a4
                      SE_KERNEL_OBJECT,
Packit Service 7c31a4
                      DACL_SECURITY_INFORMATION,
Packit Service 7c31a4
                      NULL,
Packit Service 7c31a4
                      NULL,
Packit Service 7c31a4
                      &old_dacl,
Packit Service 7c31a4
                      NULL,
Packit Service 7c31a4
                      &sd)) {
Packit Service 7c31a4
    error = GetLastError();
Packit Service 7c31a4
    goto clean_sid;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  memset(&ea, 0, sizeof(EXPLICIT_ACCESS));
Packit Service 7c31a4
  if (mode & UV_READABLE)
Packit Service 7c31a4
    ea.grfAccessPermissions |= GENERIC_READ | FILE_WRITE_ATTRIBUTES;
Packit Service 7c31a4
  if (mode & UV_WRITABLE)
Packit Service 7c31a4
    ea.grfAccessPermissions |= GENERIC_WRITE | FILE_READ_ATTRIBUTES;
Packit Service 7c31a4
  ea.grfAccessPermissions |= SYNCHRONIZE;
Packit Service 7c31a4
  ea.grfAccessMode = SET_ACCESS;
Packit Service 7c31a4
  ea.grfInheritance = NO_INHERITANCE;
Packit Service 7c31a4
  ea.Trustee.TrusteeForm = TRUSTEE_IS_SID;
Packit Service 7c31a4
  ea.Trustee.TrusteeType = TRUSTEE_IS_WELL_KNOWN_GROUP;
Packit Service 7c31a4
  ea.Trustee.ptstrName = (LPTSTR)everyone;
Packit Service 7c31a4
Packit Service 7c31a4
  if (SetEntriesInAcl(1, &ea, old_dacl, &new_dacl)) {
Packit Service 7c31a4
    error = GetLastError();
Packit Service 7c31a4
    goto clean_sd;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  if (SetSecurityInfo(handle->handle,
Packit Service 7c31a4
                      SE_KERNEL_OBJECT,
Packit Service 7c31a4
                      DACL_SECURITY_INFORMATION,
Packit Service 7c31a4
                      NULL,
Packit Service 7c31a4
                      NULL,
Packit Service 7c31a4
                      new_dacl,
Packit Service 7c31a4
                      NULL)) {
Packit Service 7c31a4
    error = GetLastError();
Packit Service 7c31a4
    goto clean_dacl;
Packit Service 7c31a4
  }
Packit Service 7c31a4
Packit Service 7c31a4
  error = 0;
Packit Service 7c31a4
Packit Service 7c31a4
clean_dacl:
Packit Service 7c31a4
  LocalFree((HLOCAL) new_dacl);
Packit Service 7c31a4
clean_sd:
Packit Service 7c31a4
  LocalFree((HLOCAL) sd);
Packit Service 7c31a4
clean_sid:
Packit Service 7c31a4
  FreeSid(everyone);
Packit Service 7c31a4
done:
Packit Service 7c31a4
  return uv_translate_sys_error(error);
Packit Service 7c31a4
}