1
0
mirror of https://github.com/libuv/libuv synced 2025-03-28 21:13:16 +00:00

Merge a0ea334bf91df91aa15cf69b77164e9a7bca3b00 into ea1cf034bef1ee39cf453b5b37b98be153541453

This commit is contained in:
reito 2025-03-19 01:42:25 -04:00 committed by GitHub
commit 4a8c187c81
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
11 changed files with 874 additions and 96 deletions

View File

@ -605,6 +605,7 @@ if(LIBUV_BUILD_TESTS)
test/test-pipe-server-close.c
test/test-pipe-set-fchmod.c
test/test-pipe-set-non-blocking.c
test/test-pipe-win-uds.c
test/test-platform-output.c
test/test-poll-close-doesnt-corrupt-stack.c
test/test-poll-close.c

View File

@ -857,7 +857,8 @@ enum {
* uv_pipe_t is a subclass of uv_stream_t.
*
* Representing a pipe stream or pipe server. On Windows this is a Named
* Pipe. On Unix this is a Unix domain socket.
* Pipe or a Unix domain socket depends on the init flags. On Unix this is
* a Unix domain socket.
*/
struct uv_pipe_s {
UV_HANDLE_FIELDS
@ -866,7 +867,22 @@ struct uv_pipe_s {
UV_PIPE_PRIVATE_FIELDS
};
enum uv_pipe_init_flags {
/*
* Enable IPC mode of pipe.
*/
UV_PIPE_INIT_IPC = 1u << 0,
/*
* Use Unix Domain Socket instead of NamedPipe on Windows. Will validate
* the `name` as a file path when calling bind / connect if this flag is set.
* On non-Windows platform this flag is ignored.
*/
UV_PIPE_INIT_WIN_UDS = 1u << 1,
};
UV_EXTERN int uv_pipe_init(uv_loop_t*, uv_pipe_t* handle, int ipc);
UV_EXTERN int uv_pipe_init_ex(uv_loop_t* loop, uv_pipe_t* handle, unsigned int flags);
UV_EXTERN int uv_pipe_open(uv_pipe_t*, uv_file file);
UV_EXTERN int uv_pipe_bind(uv_pipe_t* handle, const char* name);
UV_EXTERN int uv_pipe_bind2(uv_pipe_t* handle,

View File

@ -376,7 +376,14 @@ typedef struct {
ULONG_PTR result; /* overlapped.Internal is reused to hold the result */\
HANDLE pipeHandle; \
DWORD duplex_flags; \
WCHAR* name; \
/* When using unix domain socket, ConnectEx IOCP result will overwrite
* result + pipeHandle, to keep the ABI, reusing the name field to store
* the pending uds_socket for connect handler.
*/ \
union { \
WCHAR* name; \
SOCKET uds_socket; \
}; \
} connect; \
} u; \
struct uv_req_s* next_req;
@ -486,7 +493,14 @@ typedef struct {
#define UV_PIPE_PRIVATE_FIELDS \
HANDLE handle; \
WCHAR* name; \
/*
* The uds doesn't use wchar name, reuse this memory to store raw utf-8 name
* while keep the ABI compact.
*/ \
union { \
char* pathname; \
WCHAR* name; \
}; \
union { \
struct { uv_pipe_server_fields } serv; \
struct { uv_pipe_connection_fields } conn; \

View File

@ -44,6 +44,13 @@ static int includes_nul(const char *s, size_t n) {
int uv_pipe_init(uv_loop_t* loop, uv_pipe_t* handle, int ipc) {
return uv_pipe_init_ex(loop, handle, ipc ? UV_PIPE_INIT_IPC : 0);
}
int uv_pipe_init_ex(uv_loop_t* loop, uv_pipe_t* handle, unsigned int flags) {
int ipc = (flags & UV_PIPE_INIT_IPC) != 0;
uv__stream_init(loop, (uv_stream_t*)handle, UV_NAMED_PIPE);
handle->shutdown_req = NULL;
handle->connect_req = NULL;

View File

@ -123,6 +123,7 @@ enum {
/* Only used by uv_pipe_t handles. */
UV_HANDLE_NON_OVERLAPPED_PIPE = 0x01000000,
UV_HANDLE_PIPESERVER = 0x02000000,
UV_HANDLE_WIN_UDS_PIPE = 0x04000000,
/* Only used by uv_tty_t handles. */
UV_HANDLE_TTY_READABLE = 0x01000000,

View File

@ -298,9 +298,6 @@ void uv__winsock_init(void);
int uv__ntstatus_to_winsock_error(NTSTATUS status);
BOOL uv__get_acceptex_function(SOCKET socket, LPFN_ACCEPTEX* target);
BOOL uv__get_connectex_function(SOCKET socket, LPFN_CONNECTEX* target);
int WSAAPI uv__wsarecv_workaround(SOCKET socket, WSABUF* buffers,
DWORD buffer_count, DWORD* bytes, DWORD* flags, WSAOVERLAPPED *overlapped,
LPWSAOVERLAPPED_COMPLETION_ROUTINE completion_routine);
@ -320,6 +317,10 @@ extern int uv_tcp_non_ifs_lsp_ipv6;
extern struct sockaddr_in uv_addr_ip4_any_;
extern struct sockaddr_in6 uv_addr_ip6_any_;
/* WSA function pointers */
extern LPFN_ACCEPTEX uv_wsa_acceptex;
extern LPFN_CONNECTEX uv_wsa_connectex;
/*
* Wake all loops with fake message
*/

View File

@ -35,6 +35,11 @@
#include <aclapi.h>
#include <accctrl.h>
#if defined(_WIN32) && !defined(__MINGW32__) && !defined(__MINGW64__)
#define UV__ENABLE_WIN_UDS_PIPE
#include <afunix.h>
#endif
/* A zero-size buffer for use by uv_pipe_read */
static char uv_zero_[] = "";
@ -97,6 +102,47 @@ static void eof_timer_cb(uv_timer_t* timer);
static void eof_timer_destroy(uv_pipe_t* pipe);
static void eof_timer_close_cb(uv_handle_t* handle);
/*
* When use with bind, the file path must not exist for successful creation
* of server socket. When use with connect, the file path must exist so that
* the client socket can connect to.
* The UDS on Windows only supports this pathname mode.
*/
static int uv__win_uds_pipe_file_exists(const char* path, int* exists) {
int err;
WCHAR* wpath;
DWORD attrib;
if (exists == NULL || path == NULL) {
return UV_EINVAL;
}
err = uv__convert_utf8_to_utf16(path, &wpath);
if (err) {
return err;
}
attrib = GetFileAttributesW(wpath);
*exists = (attrib != INVALID_FILE_ATTRIBUTES &&
!(attrib & FILE_ATTRIBUTE_DIRECTORY));
free(wpath);
return 0;
}
static void uv__close_pipe_handle(uv_pipe_t* handle, HANDLE h) {
/* Microsoft says don't close socket with CloseHandle, so when we use
* unix domain socket we should use closesocket instead of CloseHandle.
* https://learn.microsoft.com/en-us/windows/win32/api/handleapi/nf-handleapi-closehandle
*/
if (handle->flags & UV_HANDLE_WIN_UDS_PIPE) {
closesocket((SOCKET) h);
} else {
CloseHandle(h);
}
}
/* Does the file path contain embedded nul bytes? */
static int includes_nul(const char *s, size_t n) {
@ -112,8 +158,23 @@ static void uv__unique_pipe_name(unsigned long long ptr, char* name, size_t size
int uv_pipe_init(uv_loop_t* loop, uv_pipe_t* handle, int ipc) {
uv__stream_init(loop, (uv_stream_t*)handle, UV_NAMED_PIPE);
return uv_pipe_init_ex(loop, handle, ipc ? UV_PIPE_INIT_IPC : 0);
}
/*
* Init pipe with flags.
*/
int uv_pipe_init_ex(uv_loop_t* loop, uv_pipe_t* handle, unsigned int flags) {
int ipc = (flags & UV_PIPE_INIT_IPC) != 0;
int uds = (flags & UV_PIPE_INIT_WIN_UDS) != 0;
if (ipc && uds) {
/* Unix domain socket on Windows doesn't work with IPC mode currently. */
return EINVAL;
}
uv__stream_init(loop, (uv_stream_t*)handle, UV_NAMED_PIPE);
handle->reqs_pending = 0;
handle->handle = INVALID_HANDLE_VALUE;
handle->name = NULL;
@ -124,6 +185,10 @@ int uv_pipe_init(uv_loop_t* loop, uv_pipe_t* handle, int ipc) {
handle->ipc = ipc;
handle->pipe.conn.non_overlapped_writes_tail = NULL;
if (uds) {
handle->flags |= UV_HANDLE_WIN_UDS_PIPE;
}
return 0;
}
@ -197,7 +262,7 @@ static HANDLE open_named_pipe(const WCHAR* name, DWORD* duplex_flags) {
static void close_pipe(uv_pipe_t* pipe) {
assert(pipe->u.fd == -1 || pipe->u.fd > 2);
if (pipe->u.fd == -1)
CloseHandle(pipe->handle);
uv__close_pipe_handle(pipe, pipe->handle);
else
_close(pipe->u.fd);
@ -205,7 +270,7 @@ static void close_pipe(uv_pipe_t* pipe) {
pipe->handle = INVALID_HANDLE_VALUE;
}
/* Pipe pair use named pipe only, no uds support. */
static int uv__pipe_server(
HANDLE* pipeHandle_ptr, DWORD access,
char* name, size_t nameSize, unsigned long long random) {
@ -246,6 +311,7 @@ static int uv__pipe_server(
}
/* Pipe pair use named pipe only, no uds support. */
static int uv__create_pipe_pair(
HANDLE* server_pipe_ptr, HANDLE* client_pipe_ptr,
unsigned int server_flags, unsigned int client_flags,
@ -346,6 +412,7 @@ static int uv__create_pipe_pair(
}
/* Pipe pair use named pipe only, no uds support. */
int uv_pipe(uv_file fds[2], int read_flags, int write_flags) {
uv_file temp[2];
int err;
@ -391,6 +458,7 @@ int uv_pipe(uv_file fds[2], int read_flags, int write_flags) {
}
/* Pipe pair use named pipe only, no uds support. */
int uv__create_stdio_pipe_pair(uv_loop_t* loop,
uv_pipe_t* parent_pipe, HANDLE* child_pipe_ptr, unsigned int flags) {
/* The parent_pipe is always the server_pipe and kept by libuv.
@ -479,6 +547,13 @@ static int uv__set_pipe_handle(uv_loop_t* loop,
if (handle->handle != INVALID_HANDLE_VALUE)
return UV_EBUSY;
if (handle->flags & UV_HANDLE_WIN_UDS_PIPE) {
/* Skip if the handle is a unix domain socket.
* We already created IOCP at connect2.
*/
goto uds_pipe;
}
if (!SetNamedPipeHandleState(pipeHandle, &mode, NULL, NULL)) {
err = GetLastError();
if (err == ERROR_ACCESS_DENIED) {
@ -530,6 +605,7 @@ static int uv__set_pipe_handle(uv_loop_t* loop,
}
}
uds_pipe:
handle->handle = pipeHandle;
handle->u.fd = fd;
handle->flags |= duplex_flags;
@ -538,7 +614,61 @@ static int uv__set_pipe_handle(uv_loop_t* loop,
}
static int pipe_alloc_accept(uv_loop_t* loop, uv_pipe_t* handle,
#if defined(UV__ENABLE_WIN_UDS_PIPE)
static int pipe_alloc_accept_unix_domain_socket(uv_loop_t* loop, uv_pipe_t* handle,
uv_pipe_accept_t* req, const char* name, BOOL firstInstance) {
int ret = 0, err = 0;
SOCKET accept_fd;
SOCKET server_fd;
struct sockaddr_un addr = {0};
assert(req->pipeHandle == INVALID_HANDLE_VALUE);
/* Create a non bound socket for AcceptEx second parameter. */
accept_fd = socket(AF_UNIX, SOCK_STREAM, IPPROTO_IP);
if (accept_fd == INVALID_SOCKET) {
return WSAGetLastError();
}
req->pipeHandle = (HANDLE) accept_fd;
if (firstInstance) {
/* First instance, only possible at bind, create the server socket. */
server_fd = socket(AF_UNIX, SOCK_STREAM, IPPROTO_IP);
if (server_fd == INVALID_SOCKET) {
return WSAGetLastError();
}
addr.sun_family = AF_UNIX;
ret = uv__strscpy(addr.sun_path, name, UNIX_PATH_MAX);
if (ret < 0) {
return ret;
}
ret = bind(server_fd, (const struct sockaddr*)&addr, sizeof(addr));
if (ret == SOCKET_ERROR) {
err = WSAGetLastError();
closesocket(server_fd);
return err;
}
/* Associate it with IOCP so we can get events. */
if (CreateIoCompletionPort((HANDLE) server_fd,
loop->iocp,
(ULONG_PTR) handle,
0) == NULL) {
return GetLastError();
}
/* First instance, save for AcceptEx first parameter. */
handle->handle = (HANDLE) server_fd;
}
return 0;
}
#endif
static int pipe_alloc_accept_named_pipe(uv_loop_t* loop, uv_pipe_t* handle,
uv_pipe_accept_t* req, BOOL firstInstance) {
assert(req->pipeHandle == INVALID_HANDLE_VALUE);
@ -607,6 +737,12 @@ void uv__pipe_shutdown(uv_loop_t* loop, uv_pipe_t* handle, uv_shutdown_t *req) {
return;
}
if (handle->flags & UV_HANDLE_WIN_UDS_PIPE) {
/* Unix domain socket support neither query 'named pipe' info, nor FlushFileBuffer. */
uv__insert_pending_req(loop, (uv_req_t*) req);
return;
}
/* Try to avoid flushing the pipe buffer in the thread pool. */
nt_status = pNtQueryInformationFile(handle->handle,
&io_status,
@ -695,6 +831,11 @@ void uv__pipe_endgame(uv_loop_t* loop, uv_pipe_t* handle) {
void uv_pipe_pending_instances(uv_pipe_t* handle, int count) {
if (handle->flags & UV_HANDLE_BOUND)
return;
/* Unix domain socket only use 1 pending instance. */
if (handle->flags & UV_HANDLE_WIN_UDS_PIPE)
return;
handle->pipe.serv.pending_instances = count;
handle->flags |= UV_HANDLE_PIPESERVER;
}
@ -711,9 +852,12 @@ int uv_pipe_bind2(uv_pipe_t* handle,
size_t namelen,
unsigned int flags) {
uv_loop_t* loop = handle->loop;
int i, err;
int i, err = 0;
uv_pipe_accept_t* req;
char* name_copy;
int use_uds_pipe;
int uds_file_exists;
int uds_err;
if (flags & ~UV_PIPE_NO_TRUNCATE) {
return UV_EINVAL;
@ -731,6 +875,26 @@ int uv_pipe_bind2(uv_pipe_t* handle,
return UV_EINVAL;
}
use_uds_pipe = (handle->flags & UV_HANDLE_WIN_UDS_PIPE) != 0;
#if !defined(UV__ENABLE_WIN_UDS_PIPE)
if (use_uds_pipe) {
return UV_ENOSYS;
}
#endif
#if defined(UV__ENABLE_WIN_UDS_PIPE)
if (use_uds_pipe) {
if (flags & UV_PIPE_NO_TRUNCATE)
if (namelen >= UNIX_PATH_MAX)
return UV_EINVAL;
if (namelen >= UNIX_PATH_MAX)
namelen = UNIX_PATH_MAX - 1;
}
#endif
/* Already bound? */
if (handle->flags & UV_HANDLE_BOUND) {
return UV_EINVAL;
}
@ -751,10 +915,17 @@ int uv_pipe_bind2(uv_pipe_t* handle,
handle->pipe.serv.pending_instances = default_pending_pipe_instances;
}
err = UV_ENOMEM;
if (use_uds_pipe) {
/* Only use 1 pending instance when use unix domain socket, cause
* call AcceptEx multiple times seems result in multiple accept events.
* Not the expected queue behavior, that only one of them is triggered. */
handle->pipe.serv.pending_instances = 1;
}
handle->pipe.serv.accept_reqs = (uv_pipe_accept_t*)
uv__malloc(sizeof(uv_pipe_accept_t) * handle->pipe.serv.pending_instances);
if (handle->pipe.serv.accept_reqs == NULL) {
err = UV_ENOMEM;
goto error;
}
@ -766,20 +937,58 @@ int uv_pipe_bind2(uv_pipe_t* handle,
req->next_pending = NULL;
}
/* TODO(bnoordhuis) Add converters that take a |length| parameter. */
err = uv__convert_utf8_to_utf16(name_copy, &handle->name);
uv__free(name_copy);
name_copy = NULL;
if (!use_uds_pipe) {
/* TODO(bnoordhuis) Add converters that take a |length| parameter. */
err = uv__convert_utf8_to_utf16(name_copy, &handle->name);
uv__free(name_copy);
name_copy = NULL;
} else {
/* Use unix domain socket we save the original path name copy. */
handle->pathname = name_copy;
name_copy = NULL;
}
if (err) {
goto error;
}
#if defined(UV__ENABLE_WIN_UDS_PIPE)
if (use_uds_pipe) {
err = uv__win_uds_pipe_file_exists(handle->pathname, &uds_file_exists);
if (err) {
goto error;
}
/* The uds file must not exist before bind. */
if (uds_file_exists) {
err = UV_EEXIST;
goto error;
}
uds_err = pipe_alloc_accept_unix_domain_socket(
loop, handle, &handle->pipe.serv.accept_reqs[0], handle->pathname, TRUE);
if (uds_err) {
if (uds_err == WSAENETDOWN) {
/*
* Typically it means the file at pathname cannot be created,
* possibly bad parent directory in path.
*/
err = UV_EINVAL;
} else {
err = uv_translate_sys_error(uds_err);
}
goto error;
}
goto uds_pipe;
}
#endif
/*
* Attempt to create the first pipe with FILE_FLAG_FIRST_PIPE_INSTANCE.
* If this fails then there's already a pipe server for the given pipe name.
*/
if (!pipe_alloc_accept(loop,
if (!pipe_alloc_accept_named_pipe(loop,
handle,
&handle->pipe.serv.accept_reqs[0],
TRUE)) {
@ -794,6 +1003,7 @@ int uv_pipe_bind2(uv_pipe_t* handle,
goto error;
}
uds_pipe:
handle->pipe.serv.pending_accepts = NULL;
handle->flags |= UV_HANDLE_PIPESERVER;
handle->flags |= UV_HANDLE_BOUND;
@ -880,12 +1090,33 @@ int uv_pipe_connect2(uv_connect_t* req,
unsigned int flags,
uv_connect_cb cb) {
uv_loop_t* loop;
int err;
int err = 0;
size_t nameSize;
HANDLE pipeHandle = INVALID_HANDLE_VALUE;
DWORD duplex_flags;
char* name_copy;
int use_uds_pipe;
#if defined(UV__ENABLE_WIN_UDS_PIPE)
int uds_file_exists;
SOCKET uds_client_fd;
struct sockaddr_un uds_addr_bind = {0};
struct sockaddr_un uds_addr_real = {0};
DWORD uds_dummy_send_cnt = 0;
/* ConnectEx seems has a bug when using with 'sockaddr_un'.
* It seems corrupting stack if no such buffer present on stack.
* Looks like needs at least 316 bytes to not overwriting some data onto
* valid stack spaces, allocating 512 bytes for 'safety'.
*
* TODO: It still overflow write to this buffer instead of valid
* stack, so it is just a dangerous workaround to write to a controlled
* dummy memory instead of causing stack corruption.
*/
char uds_dummy_send_buffer[512] = {0};
#endif
loop = handle->loop;
UV_REQ_INIT(req, UV_CONNECT);
req->handle = (uv_stream_t*) handle;
@ -910,6 +1141,27 @@ int uv_pipe_connect2(uv_connect_t* req,
return UV_EINVAL;
}
use_uds_pipe = (handle->flags & UV_HANDLE_WIN_UDS_PIPE) != 0;
#if !defined(UV__ENABLE_WIN_UDS_PIPE)
if (use_uds_pipe) {
return UV_ENOSYS;
}
#endif
#if defined(UV__ENABLE_WIN_UDS_PIPE)
if (use_uds_pipe) {
/* https://devblogs.microsoft.com/commandline/af_unix-comes-to-windows/ */
/* a null-terminated UTF-8 file system path */
if (flags & UV_PIPE_NO_TRUNCATE)
if (namelen >= UNIX_PATH_MAX)
return UV_EINVAL;
if (namelen >= UNIX_PATH_MAX)
namelen = UNIX_PATH_MAX - 1;
}
#endif
name_copy = uv__malloc(namelen + 1);
if (name_copy == NULL) {
return UV_ENOMEM;
@ -928,16 +1180,112 @@ int uv_pipe_connect2(uv_connect_t* req,
}
uv__pipe_connection_init(handle);
/* TODO(bnoordhuis) Add converters that take a |length| parameter. */
err = uv__convert_utf8_to_utf16(name_copy, &handle->name);
uv__free(name_copy);
name_copy = NULL;
if (!use_uds_pipe) {
/* TODO(bnoordhuis) Add converters that take a |length| parameter. */
err = uv__convert_utf8_to_utf16(name_copy, &handle->name);
uv__free(name_copy);
name_copy = NULL;
} else {
/* Use unix domain socket we save the original path name copy. */
handle->pathname = name_copy;
name_copy = NULL;
}
if (err) {
err = ERROR_NO_UNICODE_TRANSLATION;
goto error;
}
#if defined(UV__ENABLE_WIN_UDS_PIPE)
if (use_uds_pipe) {
err = uv__win_uds_pipe_file_exists(handle->pathname, &uds_file_exists);
if (err) {
goto error;
}
/* The uds file must have been created by server. */
if (!uds_file_exists) {
err = UV_ENOENT;
goto error;
}
/*
* If prefix is not "\\.\pipe", we assume it is a Unix Domain Socket.
* Although "NamedPipe" is a Windows concept, however in libuv, it has
* been abstracted to be a general concept that can be used on all platforms.
* Thus, we use Unix Domain Socket as a "named_pipe" backend when the prefix
* of the pipe is not matched.
*/
uds_client_fd = socket(AF_UNIX, SOCK_STREAM, IPPROTO_IP);
if (uds_client_fd == INVALID_SOCKET) {
err = WSAGetLastError();
goto error;
}
uds_addr_bind.sun_family = AF_UNIX;
/* ConnectEx need to be initially bound */
int ret = bind(uds_client_fd, (const struct sockaddr*)&uds_addr_bind, sizeof(uds_addr_bind));
if (ret != 0) {
err = WSAGetLastError();
goto error;
}
/* Associate it with IOCP so we can get events. */
if (CreateIoCompletionPort((HANDLE) uds_client_fd,
loop->iocp,
(ULONG_PTR) handle,
0) == NULL) {
closesocket(uds_client_fd);
err = GetLastError();
goto error;
}
uds_addr_real.sun_family = AF_UNIX;
/* The namelen was guaranteed to be < UNIX_PATH_MAX above. */
memcpy(uds_addr_real.sun_path, name, namelen);
uds_addr_real.sun_path[namelen] = '\0';
memset(&req->u.io.overlapped, 0, sizeof(req->u.io.overlapped));
/*
* https://learn.microsoft.com/en-us/windows/win32/api/mswsock/nc-mswsock-lpfn_connectex
* Although doc says the send buffer can be ignored, it will corrupt the
* stack if we don't actually allocate them on stack and pass them.
*/
ret = uv_wsa_connectex(uds_client_fd,
(const struct sockaddr*)&uds_addr_real,
sizeof(uds_addr_real),
uds_dummy_send_buffer,
0,
&uds_dummy_send_cnt,
&req->u.io.overlapped);
if (!ret) {
err = WSAGetLastError();
if (err != ERROR_IO_PENDING) {
closesocket(uds_client_fd);
goto error;
}
}
/* Since we use IOCP, we can't set value to u.connect.pipeHandle
* as it will be rewritten by the result of IOCP. Thus, we set the socket
* to uds_socket (reuse the `name`) and set it to pipeHandle later at req
* handler.
*/
req->u.connect.uds_socket = uds_client_fd;
req->u.connect.duplex_flags = UV_HANDLE_WRITABLE | UV_HANDLE_READABLE;
/* The req will be processed with IOCP. */
handle->reqs_pending++;
REGISTER_HANDLE_REQ(loop, handle);
return 0;
}
#endif
pipeHandle = open_named_pipe(handle->name, &duplex_flags);
if (pipeHandle == INVALID_HANDLE_VALUE) {
if (GetLastError() == ERROR_PIPE_BUSY) {
@ -1056,6 +1404,7 @@ void uv__pipe_read_stop(uv_pipe_t* handle) {
void uv__pipe_close(uv_loop_t* loop, uv_pipe_t* handle) {
int i;
HANDLE pipeHandle;
struct linger uds_linger = {1, 0};
if (handle->flags & UV_HANDLE_READING) {
handle->flags &= ~UV_HANDLE_READING;
@ -1078,11 +1427,20 @@ void uv__pipe_close(uv_loop_t* loop, uv_pipe_t* handle) {
handle->name = NULL;
}
if (handle->flags & UV_HANDLE_WIN_UDS_PIPE) {
/* Force the subsequent closesocket to be abortative. */
setsockopt((SOCKET)handle->handle,
SOL_SOCKET,
SO_LINGER,
(const char*)&uds_linger,
sizeof(uds_linger));
}
if (handle->flags & UV_HANDLE_PIPESERVER) {
for (i = 0; i < handle->pipe.serv.pending_instances; i++) {
pipeHandle = handle->pipe.serv.accept_reqs[i].pipeHandle;
if (pipeHandle != INVALID_HANDLE_VALUE) {
CloseHandle(pipeHandle);
uv__close_pipe_handle(handle, pipeHandle);
handle->pipe.serv.accept_reqs[i].pipeHandle = INVALID_HANDLE_VALUE;
}
}
@ -1106,33 +1464,78 @@ void uv__pipe_close(uv_loop_t* loop, uv_pipe_t* handle) {
static void uv__pipe_queue_accept(uv_loop_t* loop, uv_pipe_t* handle,
uv_pipe_accept_t* req, BOOL firstInstance) {
int uds_err;
int wsa_err;
DWORD bytes_received;
CHAR accept_buf[2 * (sizeof(SOCKADDR_STORAGE) + 16)];
assert(handle->flags & UV_HANDLE_LISTENING);
if (!firstInstance && !pipe_alloc_accept(loop, handle, req, FALSE)) {
SET_REQ_ERROR(req, GetLastError());
uv__insert_pending_req(loop, (uv_req_t*) req);
handle->reqs_pending++;
return;
if (!firstInstance) {
#if defined(UV__ENABLE_WIN_UDS_PIPE)
if (handle->flags & UV_HANDLE_WIN_UDS_PIPE) {
uds_err = pipe_alloc_accept_unix_domain_socket(
loop, handle, req, handle->pathname, FALSE);
if (uds_err) {
SET_REQ_ERROR(req, uds_err);
uv__insert_pending_req(loop, (uv_req_t *) req);
handle->reqs_pending++;
return;
}
goto uds_pipe;
}
#endif
if (!pipe_alloc_accept_named_pipe(loop, handle, req, FALSE)) {
SET_REQ_ERROR(req, GetLastError());
uv__insert_pending_req(loop, (uv_req_t*) req);
handle->reqs_pending++;
return;
}
}
uds_pipe:
assert(req->pipeHandle != INVALID_HANDLE_VALUE);
/* Prepare the overlapped structure. */
memset(&(req->u.io.overlapped), 0, sizeof(req->u.io.overlapped));
if (!ConnectNamedPipe(req->pipeHandle, &req->u.io.overlapped) &&
GetLastError() != ERROR_IO_PENDING) {
if (GetLastError() == ERROR_PIPE_CONNECTED) {
SET_REQ_SUCCESS(req);
} else {
CloseHandle(req->pipeHandle);
req->pipeHandle = INVALID_HANDLE_VALUE;
/* Make this req pending reporting an error. */
SET_REQ_ERROR(req, GetLastError());
if (handle->flags & UV_HANDLE_WIN_UDS_PIPE) {
if (!uv_wsa_acceptex((SOCKET)handle->handle,
(SOCKET)req->pipeHandle,
accept_buf,
0,
sizeof(SOCKADDR_STORAGE) + 16,
sizeof(SOCKADDR_STORAGE) + 16,
&bytes_received,
&req->u.io.overlapped)) {
wsa_err = WSAGetLastError();
if (wsa_err != ERROR_IO_PENDING) {
closesocket((SOCKET) req->pipeHandle);
req->pipeHandle = INVALID_HANDLE_VALUE;
SET_REQ_ERROR(req, wsa_err);
uv__insert_pending_req(loop, (uv_req_t*) req);
handle->reqs_pending++;
return;
}
}
} else {
if (!ConnectNamedPipe(req->pipeHandle, &req->u.io.overlapped) &&
GetLastError() != ERROR_IO_PENDING) {
if (GetLastError() == ERROR_PIPE_CONNECTED) {
SET_REQ_SUCCESS(req);
} else {
CloseHandle(req->pipeHandle);
req->pipeHandle = INVALID_HANDLE_VALUE;
/* Make this req pending reporting an error. */
SET_REQ_ERROR(req, GetLastError());
}
uv__insert_pending_req(loop, (uv_req_t*) req);
handle->reqs_pending++;
return;
}
uv__insert_pending_req(loop, (uv_req_t*) req);
handle->reqs_pending++;
return;
}
/* Wait for completion via IOCP */
@ -1184,12 +1587,32 @@ int uv__pipe_accept(uv_pipe_t* server, uv_stream_t* client) {
pipe_client->handle = req->pipeHandle;
pipe_client->flags |= UV_HANDLE_READABLE | UV_HANDLE_WRITABLE;
/* A unix domain socket server */
if (server->flags & UV_HANDLE_WIN_UDS_PIPE) {
/* Associate it with the I/O completion port. Use uv_handle_t pointer as
* completion key. */
if (CreateIoCompletionPort(req->pipeHandle,
pipe_client->loop->iocp,
(ULONG_PTR) req->pipeHandle,
0) == NULL) {
return GetLastError();
}
/* AcceptEx() implicitly binds the accepted socket. */
pipe_client->flags |= UV_HANDLE_BOUND;
pipe_client->flags |= UV_HANDLE_WIN_UDS_PIPE;
}
/* Prepare the req to pick up a new connection */
server->pipe.serv.pending_accepts = req->next_pending;
req->next_pending = NULL;
req->pipeHandle = INVALID_HANDLE_VALUE;
server->handle = INVALID_HANDLE_VALUE;
if (!(server->flags & UV_HANDLE_WIN_UDS_PIPE)) {
/* Unix domain socket doesn't transfer to client ownership, so do not reset here.*/
server->handle = INVALID_HANDLE_VALUE;
}
if (!(server->flags & UV_HANDLE_CLOSING)) {
uv__pipe_queue_accept(loop, server, req, FALSE);
}
@ -1203,6 +1626,7 @@ int uv__pipe_accept(uv_pipe_t* server, uv_stream_t* client) {
int uv__pipe_listen(uv_pipe_t* handle, int backlog, uv_connection_cb cb) {
uv_loop_t* loop = handle->loop;
int i;
int err;
if (handle->flags & UV_HANDLE_LISTENING) {
handle->stream.serv.connection_cb = cb;
@ -1224,6 +1648,13 @@ int uv__pipe_listen(uv_pipe_t* handle, int backlog, uv_connection_cb cb) {
return WSAEINVAL;
}
if (handle->flags & UV_HANDLE_WIN_UDS_PIPE) {
err = listen((SOCKET) handle->handle, backlog);
if (err) {
return err;
}
}
handle->flags |= UV_HANDLE_LISTENING;
INCREASE_ACTIVE_COUNT(loop, handle);
handle->stream.serv.connection_cb = cb;
@ -2137,14 +2568,31 @@ void uv__process_pipe_read_req(uv_loop_t* loop,
return;
if (!REQ_SUCCESS(req)) {
/* An error occurred doing the zero-read. */
err = GET_REQ_ERROR(req);
if (handle->flags & UV_HANDLE_WIN_UDS_PIPE) {
err = GET_REQ_SOCK_ERROR(req);
if (err == WSAECONNABORTED) {
/* Turn WSAECONNABORTED into UV_ECONNRESET to be consistent with Unix.
*/
err = WSAECONNRESET;
}
handle->flags &= ~(UV_HANDLE_READABLE | UV_HANDLE_WRITABLE);
/* If the read was cancelled by uv__pipe_interrupt_read(), the request may
* indicate an ERROR_OPERATION_ABORTED error. This error isn't relevant to
* the user; we'll start a new zero-read at the end of this function. */
if (err != ERROR_OPERATION_ABORTED)
uv__pipe_read_error_or_eof(loop, handle, err, uv_null_buf_);
if (err == WSAECONNRESET) {
uv__pipe_read_eof(loop, handle, uv_null_buf_);
} else {
uv__pipe_read_error(loop, handle, err, uv_null_buf_);
}
} else {
/* An error occurred doing the zero-read. */
err = GET_REQ_ERROR(req);
/* If the read was cancelled by uv__pipe_interrupt_read(), the request may
* indicate an ERROR_OPERATION_ABORTED error. This error isn't relevant to
* the user; we'll start a new zero-read at the end of this function. */
if (err != ERROR_OPERATION_ABORTED)
uv__pipe_read_error_or_eof(loop, handle, err, uv_null_buf_);
}
} else {
/* The zero-read completed without error, indicating there is data
@ -2236,6 +2684,15 @@ void uv__process_pipe_accept_req(uv_loop_t* loop, uv_pipe_t* handle,
return;
}
if (handle->flags & UV_HANDLE_WIN_UDS_PIPE) {
/* If it is unix domain handle, the event comes from AcceptEx IOCP. */
setsockopt((SOCKET)req->pipeHandle,
SOL_SOCKET,
SO_UPDATE_ACCEPT_CONTEXT,
(char*)&handle->handle,
sizeof(handle->handle));
}
if (REQ_SUCCESS(req)) {
assert(req->pipeHandle != INVALID_HANDLE_VALUE);
req->next_pending = handle->pipe.serv.pending_accepts;
@ -2246,7 +2703,7 @@ void uv__process_pipe_accept_req(uv_loop_t* loop, uv_pipe_t* handle,
}
} else {
if (req->pipeHandle != INVALID_HANDLE_VALUE) {
CloseHandle(req->pipeHandle);
uv__close_pipe_handle(handle, req->pipeHandle);
req->pipeHandle = INVALID_HANDLE_VALUE;
}
if (!(handle->flags & UV_HANDLE_CLOSING)) {
@ -2266,6 +2723,20 @@ void uv__process_pipe_connect_req(uv_loop_t* loop, uv_pipe_t* handle,
assert(handle->type == UV_NAMED_PIPE);
if (handle->flags & UV_HANDLE_WIN_UDS_PIPE) {
/* IOCP overwrites the connect.pipeHandle, so workaround here. */
req->u.connect.pipeHandle = (HANDLE) req->u.connect.uds_socket;
if (req->u.connect.pipeHandle) {
/* If it is unix domain handle, the event comes from ConnectEx IOCP. */
setsockopt((SOCKET) req->u.connect.pipeHandle,
SOL_SOCKET,
SO_UPDATE_CONNECT_CONTEXT,
NULL,
0);
}
}
UNREGISTER_HANDLE_REQ(loop, handle);
err = 0;
@ -2277,7 +2748,7 @@ void uv__process_pipe_connect_req(uv_loop_t* loop, uv_pipe_t* handle,
else
err = uv__set_pipe_handle(loop, handle, pipeHandle, -1, duplex_flags);
if (err)
CloseHandle(pipeHandle);
uv__close_pipe_handle(handle, pipeHandle);
} else {
err = uv_translate_sys_error(GET_REQ_ERROR(req));
}
@ -2602,9 +3073,25 @@ int uv_pipe_pending_count(uv_pipe_t* handle) {
int uv_pipe_getsockname(const uv_pipe_t* handle, char* buffer, size_t* size) {
size_t len;
if (buffer == NULL || size == NULL || *size == 0)
return UV_EINVAL;
if (handle->flags & UV_HANDLE_WIN_UDS_PIPE) {
if (handle->pathname != NULL) {
len = strlen(handle->pathname);
if (len > *size) {
return UV_ENOBUFS;
}
*size = len;
memcpy(buffer, handle->pathname, len);
}
return 0;
}
if (handle->flags & UV_HANDLE_BOUND)
return uv__pipe_getname(handle, buffer, size);
@ -2619,6 +3106,8 @@ int uv_pipe_getsockname(const uv_pipe_t* handle, char* buffer, size_t* size) {
int uv_pipe_getpeername(const uv_pipe_t* handle, char* buffer, size_t* size) {
size_t len;
if (buffer == NULL || size == NULL || *size == 0)
return UV_EINVAL;
@ -2626,6 +3115,20 @@ int uv_pipe_getpeername(const uv_pipe_t* handle, char* buffer, size_t* size) {
if (handle->flags & UV_HANDLE_BOUND)
return UV_ENOTCONN;
if (handle->flags & UV_HANDLE_WIN_UDS_PIPE) {
if (handle->pathname != NULL) {
len = strlen(handle->pathname);
if (len > *size) {
return UV_ENOBUFS;
}
*size = len;
memcpy(buffer, handle->pathname, len);
}
return 0;
}
if (handle->handle != INVALID_HANDLE_VALUE)
return uv__pipe_getname(handle, buffer, size);
@ -2658,6 +3161,11 @@ int uv_pipe_chmod(uv_pipe_t* handle, int mode) {
if (handle == NULL || handle->handle == INVALID_HANDLE_VALUE)
return UV_EBADF;
if (handle->flags & UV_HANDLE_WIN_UDS_PIPE) {
/* Unix domain socket doesn't support this. */
return UV_ENOSYS;
}
if (mode != UV_READABLE &&
mode != UV_WRITABLE &&
mode != (UV_WRITABLE | UV_READABLE))

View File

@ -432,14 +432,14 @@ static void uv__tcp_queue_accept(uv_tcp_t* handle, uv_tcp_accept_t* req) {
req->u.io.overlapped.hEvent = (HANDLE) ((ULONG_PTR) req->event_handle | 1);
}
success = handle->tcp.serv.func_acceptex(handle->socket,
accept_socket,
(void*)req->accept_buffer,
0,
sizeof(struct sockaddr_storage),
sizeof(struct sockaddr_storage),
&bytes,
&req->u.io.overlapped);
success = uv_wsa_acceptex(handle->socket,
accept_socket,
(void*)req->accept_buffer,
0,
sizeof(struct sockaddr_storage),
sizeof(struct sockaddr_storage),
&bytes,
&req->u.io.overlapped);
if (UV_SUCCEEDED_WITHOUT_IOCP(success)) {
/* Process the req without IOCP. */
@ -576,10 +576,8 @@ int uv__tcp_listen(uv_tcp_t* handle, int backlog, uv_connection_cb cb) {
return handle->delayed_error;
}
if (!handle->tcp.serv.func_acceptex) {
if (!uv__get_acceptex_function(handle->socket, &handle->tcp.serv.func_acceptex)) {
return WSAEAFNOSUPPORT;
}
if (uv_wsa_acceptex == NULL) {
return WSAEAFNOSUPPORT;
}
/* If this flag is set, we already made this listen call in xfer. */
@ -799,10 +797,8 @@ static int uv__tcp_try_connect(uv_connect_t* req,
goto out;
}
if (!handle->tcp.conn.func_connectex) {
if (!uv__get_connectex_function(handle->socket, &handle->tcp.conn.func_connectex)) {
return WSAEAFNOSUPPORT;
}
if (uv_wsa_connectex == NULL) {
return WSAEAFNOSUPPORT;
}
/* This makes connect() fail instantly if the target port on the localhost
@ -839,13 +835,13 @@ out:
return 0;
}
success = handle->tcp.conn.func_connectex(handle->socket,
(const struct sockaddr*) &converted,
addrlen,
NULL,
0,
&bytes,
&req->u.io.overlapped);
success = uv_wsa_connectex(handle->socket,
(const struct sockaddr*)&converted,
addrlen,
NULL,
0,
&bytes,
&req->u.io.overlapped);
if (UV_SUCCEEDED_WITHOUT_IOCP(success)) {
/* Process the req without IOCP. */
@ -1569,7 +1565,6 @@ int uv_socketpair(int type, int protocol, uv_os_sock_t fds[2], int flags0, int f
SOCKET client0 = INVALID_SOCKET;
SOCKET client1 = INVALID_SOCKET;
SOCKADDR_IN name;
LPFN_ACCEPTEX func_acceptex;
WSAOVERLAPPED overlap;
char accept_buffer[sizeof(struct sockaddr_storage) * 2 + 32];
int namelen;
@ -1612,19 +1607,19 @@ int uv_socketpair(int type, int protocol, uv_os_sock_t fds[2], int flags0, int f
goto wsaerror;
if (!SetHandleInformation((HANDLE) client1, HANDLE_FLAG_INHERIT, 0))
goto error;
if (!uv__get_acceptex_function(server, &func_acceptex)) {
if (uv_wsa_acceptex == NULL) {
err = WSAEAFNOSUPPORT;
goto cleanup;
}
memset(&overlap, 0, sizeof(overlap));
if (!func_acceptex(server,
client1,
accept_buffer,
0,
sizeof(struct sockaddr_storage),
sizeof(struct sockaddr_storage),
&bytes,
&overlap)) {
if (!uv_wsa_acceptex(server,
client1,
accept_buffer,
0,
sizeof(struct sockaddr_storage),
sizeof(struct sockaddr_storage),
&bytes,
&overlap)) {
err = WSAGetLastError();
if (err == ERROR_IO_PENDING) {
/* Result should complete immediately, since we already called connect,

View File

@ -34,6 +34,10 @@ int uv_tcp_non_ifs_lsp_ipv6;
struct sockaddr_in uv_addr_ip4_any_;
struct sockaddr_in6 uv_addr_ip6_any_;
/* WSA function pointers */
LPFN_ACCEPTEX uv_wsa_acceptex = NULL;
LPFN_CONNECTEX uv_wsa_connectex = NULL;
/*
* Retrieves the pointer to a winsock extension function.
@ -62,19 +66,6 @@ static BOOL uv__get_extension_function(SOCKET socket, GUID guid,
}
BOOL uv__get_acceptex_function(SOCKET socket, LPFN_ACCEPTEX* target) {
const GUID wsaid_acceptex = WSAID_ACCEPTEX;
return uv__get_extension_function(socket, wsaid_acceptex, (void**)target);
}
BOOL uv__get_connectex_function(SOCKET socket, LPFN_CONNECTEX* target) {
const GUID wsaid_connectex = WSAID_CONNECTEX;
return uv__get_extension_function(socket, wsaid_connectex, (void**)target);
}
void uv__winsock_init(void) {
WSADATA wsa_data;
int errorno;
@ -82,6 +73,9 @@ void uv__winsock_init(void) {
WSAPROTOCOL_INFOW protocol_info;
int opt_len;
const GUID wsaid_acceptex = WSAID_ACCEPTEX;
const GUID wsaid_connectex = WSAID_CONNECTEX;
/* Set implicit binding address used by connectEx */
if (uv_ip4_addr("0.0.0.0", 0, &uv_addr_ip4_any_)) {
abort();
@ -131,6 +125,19 @@ void uv__winsock_init(void) {
}
closesocket(dummy);
}
/* Try to get WSA function pointers */
dummy = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (dummy != INVALID_SOCKET) {
if (!uv__get_extension_function(
dummy, wsaid_acceptex, (void**)&uv_wsa_acceptex) ||
!uv__get_extension_function(
dummy, wsaid_connectex, (void**)&uv_wsa_connectex)) {
uv_fatal_error(WSAGetLastError(), "WSAIoctl");
}
closesocket(dummy);
}
}

View File

@ -215,6 +215,11 @@ TEST_DECLARE (pipe_getsockname_blocking)
TEST_DECLARE (pipe_pending_instances)
TEST_DECLARE (pipe_sendmsg)
TEST_DECLARE (pipe_server_close)
#ifdef _WIN32
TEST_DECLARE (pipe_win_uds)
TEST_DECLARE (pipe_win_uds_bad_name)
TEST_DECLARE (pipe_win_uds_shutdown)
#endif
TEST_DECLARE (connection_fail)
TEST_DECLARE (connection_fail_doesnt_auto_close)
TEST_DECLARE (shutdown_close_tcp)
@ -630,6 +635,12 @@ TASK_LIST_START
#endif
/* Seems to be either about 0.5s or 5s, depending on the OS. */
TEST_ENTRY_CUSTOM (pipe_set_non_blocking, 0, 0, 20000)
#ifdef _WIN32
TEST_ENTRY (pipe_win_uds)
TEST_ENTRY (pipe_win_uds_bad_name)
TEST_ENTRY (pipe_win_uds_shutdown)
#endif
TEST_ENTRY (pipe_set_chmod)
TEST_ENTRY (tty)
#ifdef _WIN32

217
test/test-pipe-win-uds.c Normal file
View File

@ -0,0 +1,217 @@
/* Copyright libuv contributors. All rights reserved.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to
* deal in the Software without restriction, including without limitation the
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
* sell copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
* IN THE SOFTWARE.
*/
#include "uv.h"
#include "task.h"
#ifdef _WIN32
#if !defined(__MINGW32__) && !defined(__MINGW64__)
#define UV_SUPPORTS_WIN_UDS
#endif
static int use_shutdown = 0;
static uv_shutdown_t shutdown_client;
static int close_cb_called = 0;
static int shutdown_cb_called = 0;
static int server_connect_cb_called = 0;
static int client_connect_cb_called = 0;
static uv_pipe_t pipe_server;
static uv_pipe_t pipe_client;
const char pipe_test_data[] = "send test through win uds pipe";
static void alloc_cb(uv_handle_t* handle, size_t size, uv_buf_t* buf) {
buf->base = malloc(size);
buf->len = size;
}
static void close_cb(uv_handle_t* handle) {
ASSERT_NOT_NULL(handle);
close_cb_called++;
}
static void shutdown_cb(uv_shutdown_t* req, int status) {
ASSERT_NOT_NULL(req);
uv_close((uv_handle_t*) req->handle, close_cb);
shutdown_cb_called++;
}
static void after_write_cb(uv_write_t* req, int status) {
ASSERT_OK(status);
free(req->data);
free(req);
}
static void client_connect_cb(uv_connect_t* connect_req, int status) {
uv_buf_t bufs[1];
uv_write_t *req;
ASSERT_EQ(status, 0);
client_connect_cb_called++;
// Server connected, send test data.
bufs[0] = uv_buf_init(pipe_test_data, strlen(pipe_test_data));
req = malloc(sizeof(*req));
req->data = NULL;
uv_write(req, connect_req->handle, bufs, 1, after_write_cb);
}
static void read_cb(uv_stream_t* stream, ssize_t nread, const uv_buf_t* buf) {
// Ignore read error.
if (nread < 0 || !buf)
return;
// Test if data equal.
ASSERT_EQ(nread, (int) sizeof(pipe_test_data) - 1);
ASSERT_MEM_EQ(buf->base, pipe_test_data, nread);
if (use_shutdown) {
uv_shutdown(&shutdown_client, (uv_stream_t*) &pipe_client, shutdown_cb);
} else {
uv_close((uv_handle_t*) &pipe_client, close_cb);
}
uv_close((uv_handle_t*) &pipe_server, close_cb);
}
static void server_connect_cb(uv_stream_t* handle, int status) {
uv_pipe_t* conn;
ASSERT_EQ(status, 0);
server_connect_cb_called++;
// Client accepted, start reading.
conn = malloc(sizeof(*conn));
ASSERT_OK(uv_pipe_init_ex(handle->loop, conn, UV_PIPE_INIT_WIN_UDS));
ASSERT_OK(uv_accept(handle, (uv_stream_t*) conn));
ASSERT_OK(uv_read_start((uv_stream_t*) conn, alloc_cb, read_cb));
}
int test_pipe_win_uds() {
#if defined(UV_SUPPORTS_WIN_UDS)
int r;
uv_fs_t fs;
uv_connect_t req;
size_t size = MAX_PATH;
char path[MAX_PATH];
// The windows UDS needs to be created on disk, create in temp dir.
r = uv_os_tmpdir(path, &size);
ASSERT_OK(r);
snprintf(path, sizeof(path), "\\uv_pipe_win_uds");
// Remove the existing file, the file must not exist before server bind.
uv_fs_unlink(uv_default_loop(), &fs, path, NULL);
uv_run(uv_default_loop(), UV_RUN_DEFAULT);
// Bind server
r = uv_pipe_init_ex(uv_default_loop(), &pipe_server, UV_PIPE_INIT_WIN_UDS);
ASSERT_OK(r);
r = uv_pipe_bind(&pipe_server, path);
ASSERT_OK(r);
uv_listen((uv_stream_t*) &pipe_server, SOMAXCONN, server_connect_cb);
uv_read_start((uv_stream_t*) &pipe_server, alloc_cb, read_cb);
// Connect client to server
r = uv_pipe_init_ex(uv_default_loop(), &pipe_client, UV_PIPE_INIT_WIN_UDS);
ASSERT_OK(r);
uv_pipe_connect(&req, &pipe_client, path, client_connect_cb);
// Run the loop
uv_run(uv_default_loop(), UV_RUN_DEFAULT);
if (use_shutdown) {
ASSERT_EQ(1, shutdown_cb_called);
}
ASSERT_EQ(2, close_cb_called);
ASSERT_EQ(1, server_connect_cb_called);
ASSERT_EQ(1, client_connect_cb_called);
#endif
MAKE_VALGRIND_HAPPY(uv_default_loop());
return 0;
}
TEST_IMPL(pipe_win_uds) {
use_shutdown = 0;
return test_pipe_win_uds();
}
TEST_IMPL(pipe_win_uds_shutdown) {
use_shutdown = 1;
return test_pipe_win_uds();
}
static void bad_name_connect_cb(uv_connect_t* connect_req, int status) {
ASSERT_EQ(status, UV_ENOENT);
}
TEST_IMPL(pipe_win_uds_bad_name) {
#if defined(UV_SUPPORTS_WIN_UDS)
int r;
uv_connect_t req;
uv_pipe_t pipe_server_1;
uv_pipe_t pipe_server_2;
uv_pipe_t pipe_client_1;
const char* path_1 = "not/exist/file/path";
const char* path_2 = "test/fixtures/empty_file";
// Bind server 1 which has a bad path
r = uv_pipe_init_ex(uv_default_loop(), &pipe_server_1, UV_PIPE_INIT_WIN_UDS);
ASSERT_OK(r);
r = uv_pipe_bind(&pipe_server_1, path_1);
ASSERT_EQ(r, UV_EINVAL);
// Bind server 2 which file exists
r = uv_pipe_init_ex(uv_default_loop(), &pipe_server_2, UV_PIPE_INIT_WIN_UDS);
ASSERT_OK(r);
r = uv_pipe_bind(&pipe_server_2, path_2);
ASSERT_EQ(r, UV_EEXIST);
// Connect client to server with bad name
r = uv_pipe_init_ex(uv_default_loop(), &pipe_client_1, UV_PIPE_INIT_WIN_UDS);
ASSERT_OK(r);
uv_pipe_connect(&req, &pipe_client_1, path_1, bad_name_connect_cb);
// Run the loop
uv_run(uv_default_loop(), UV_RUN_DEFAULT);
#endif
MAKE_VALGRIND_HAPPY(uv_default_loop());
return 0;
}
#endif