Merge remote-tracking branch 'origin/v0.6'
This commit is contained in:
commit
3de0411591
34
include/uv.h
34
include/uv.h
@ -115,7 +115,8 @@ typedef intptr_t ssize_t;
|
||||
XX( 45, EAISOCKTYPE, "") \
|
||||
XX( 46, ESHUTDOWN, "") \
|
||||
XX( 47, EEXIST, "file already exists") \
|
||||
XX( 48, ESRCH, "no such process")
|
||||
XX( 48, ESRCH, "no such process") \
|
||||
XX( 49, ENAMETOOLONG, "name too long")
|
||||
|
||||
|
||||
#define UV_ERRNO_GEN(val, name, s) UV_##name = val,
|
||||
@ -653,6 +654,20 @@ UV_EXTERN int uv_udp_set_membership(uv_udp_t* handle,
|
||||
const char* multicast_addr, const char* interface_addr,
|
||||
uv_membership membership);
|
||||
|
||||
/*
|
||||
* Set IP multicast loop flag. Makes multicast packets loop back to
|
||||
* local sockets.
|
||||
*
|
||||
* Arguments:
|
||||
* handle UDP handle. Should have been initialized with
|
||||
* `uv_udp_init`.
|
||||
* on 1 for on, 0 for off
|
||||
*
|
||||
* Returns:
|
||||
* 0 on success, -1 on error.
|
||||
*/
|
||||
UV_EXTERN int uv_udp_set_multicast_loop(uv_udp_t* handle, int on);
|
||||
|
||||
/*
|
||||
* Set the multicast ttl
|
||||
*
|
||||
@ -664,7 +679,7 @@ UV_EXTERN int uv_udp_set_membership(uv_udp_t* handle,
|
||||
* Returns:
|
||||
* 0 on success, -1 on error.
|
||||
*/
|
||||
int uv_udp_set_multicast_ttl(uv_udp_t* handle, int ttl);
|
||||
UV_EXTERN int uv_udp_set_multicast_ttl(uv_udp_t* handle, int ttl);
|
||||
|
||||
/*
|
||||
* Set broadcast on or off
|
||||
@ -677,7 +692,20 @@ int uv_udp_set_multicast_ttl(uv_udp_t* handle, int ttl);
|
||||
* Returns:
|
||||
* 0 on success, -1 on error.
|
||||
*/
|
||||
int uv_udp_set_broadcast(uv_udp_t* handle, int on);
|
||||
UV_EXTERN int uv_udp_set_broadcast(uv_udp_t* handle, int on);
|
||||
|
||||
/*
|
||||
* Set the time to live
|
||||
*
|
||||
* Arguments:
|
||||
* handle UDP handle. Should have been initialized with
|
||||
* `uv_udp_init`.
|
||||
* ttl 1 through 255
|
||||
*
|
||||
* Returns:
|
||||
* 0 on success, -1 on error.
|
||||
*/
|
||||
UV_EXTERN int uv_udp_set_ttl(uv_udp_t* handle, int ttl);
|
||||
|
||||
/*
|
||||
* Send data. If the socket has not previously been bound with `uv_udp_bind`
|
||||
|
||||
@ -71,6 +71,7 @@ uv_err_code uv_translate_sys_error(int sys_errno) {
|
||||
case EFAULT: return UV_EFAULT;
|
||||
case EMFILE: return UV_EMFILE;
|
||||
case EMSGSIZE: return UV_EMSGSIZE;
|
||||
case ENAMETOOLONG: return UV_ENAMETOOLONG;
|
||||
case EINVAL: return UV_EINVAL;
|
||||
case ECONNREFUSED: return UV_ECONNREFUSED;
|
||||
case EADDRINUSE: return UV_EADDRINUSE;
|
||||
|
||||
@ -344,6 +344,14 @@ static int uv__bind(uv_udp_t* handle,
|
||||
goto out;
|
||||
}
|
||||
|
||||
#ifdef SO_REUSEPORT /* Apple's version of SO_REUSEADDR... */
|
||||
yes = 1;
|
||||
if (setsockopt(fd, SOL_SOCKET, SO_REUSEPORT, &yes, sizeof yes) == -1) {
|
||||
uv__set_sys_error(handle->loop, errno);
|
||||
goto out;
|
||||
}
|
||||
#endif
|
||||
|
||||
if (flags & UV_UDP_IPV6ONLY) {
|
||||
#ifdef IPV6_V6ONLY
|
||||
yes = 1;
|
||||
@ -513,27 +521,25 @@ int uv_udp_set_membership(uv_udp_t* handle, const char* multicast_addr,
|
||||
return 0;
|
||||
}
|
||||
|
||||
int uv_udp_set_multicast_ttl(uv_udp_t* handle, int ttl) {
|
||||
if (setsockopt(handle->fd, IPPROTO_IP, IP_MULTICAST_TTL, &ttl, sizeof ttl) == -1) {
|
||||
uv__set_sys_error(handle->loop, errno);
|
||||
return -1;
|
||||
|
||||
#define X(name, level, option) \
|
||||
int uv_udp_set_##name(uv_udp_t* handle, int flag) { \
|
||||
if (setsockopt(handle->fd, level, option, &flag, sizeof(flag))) { \
|
||||
uv__set_sys_error(handle->loop, errno); \
|
||||
return -1; \
|
||||
} \
|
||||
return 0; \
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
X(multicast_loop, IPPROTO_IP, IP_MULTICAST_LOOP)
|
||||
X(multicast_ttl, IPPROTO_IP, IP_MULTICAST_TTL)
|
||||
X(broadcast, SOL_SOCKET, SO_BROADCAST)
|
||||
X(ttl, IPPROTO_IP, IP_TTL)
|
||||
|
||||
int uv_udp_set_broadcast(uv_udp_t* handle, int on) {
|
||||
if (setsockopt(handle->fd, SOL_SOCKET, SO_BROADCAST, &on, sizeof on) == -1) {
|
||||
uv__set_sys_error(handle->loop, errno);
|
||||
return -1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
#undef X
|
||||
|
||||
|
||||
int uv_udp_getsockname(uv_udp_t* handle, struct sockaddr* name,
|
||||
int* namelen) {
|
||||
int uv_udp_getsockname(uv_udp_t* handle, struct sockaddr* name, int* namelen) {
|
||||
socklen_t socklen;
|
||||
int saved_errno;
|
||||
int rv = 0;
|
||||
|
||||
@ -94,6 +94,7 @@ uv_err_code uv_translate_sys_error(int sys_errno) {
|
||||
case WSAEMSGSIZE: return UV_EMSGSIZE;
|
||||
case ERROR_NETWORK_UNREACHABLE: return UV_ENETUNREACH;
|
||||
case WSAENETUNREACH: return UV_ENETUNREACH;
|
||||
case WSAENOBUFS: return UV_ENOBUFS;
|
||||
case ERROR_OUTOFMEMORY: return UV_ENOMEM;
|
||||
case ERROR_NOT_CONNECTED: return UV_ENOTCONN;
|
||||
case WSAENOTCONN: return UV_ENOTCONN;
|
||||
|
||||
123
src/win/fs.c
123
src/win/fs.c
@ -489,6 +489,103 @@ void fs__readdir(uv_fs_t* req, const wchar_t* path, int flags) {
|
||||
}
|
||||
|
||||
|
||||
#define IS_SLASH(c) \
|
||||
((wchar_t) c == L'/' || (wchar_t) c == L'\\')
|
||||
#define IS_COLON(c) \
|
||||
((wchar_t) c == L':')
|
||||
#define IS_LETTER(c) \
|
||||
((((wchar_t) c >= L'a') && ((wchar_t) c <= L'z')) || \
|
||||
(((wchar_t) c >= L'A') && ((wchar_t) c <= L'Z')))
|
||||
#define IS_QUESTION(c) \
|
||||
((wchar_t) c == L'?')
|
||||
|
||||
|
||||
static int uv__count_slash_separated_words(const wchar_t* pos,
|
||||
const wchar_t* end,
|
||||
int limit) {
|
||||
char last_was_slash = 1, count = 0;
|
||||
|
||||
for (; pos < end; pos++) {
|
||||
if (IS_SLASH(*pos)) {
|
||||
/* Don't accept double slashes */
|
||||
if (last_was_slash) {
|
||||
return 0;
|
||||
} else {
|
||||
last_was_slash = 1;
|
||||
}
|
||||
} else {
|
||||
if (last_was_slash) {
|
||||
/* Found a new word */
|
||||
count++;
|
||||
if (count > limit) {
|
||||
return -1;
|
||||
}
|
||||
last_was_slash = 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return count;
|
||||
}
|
||||
|
||||
/*
|
||||
* Returns true if the given path is a root directory. The following patterns
|
||||
* are recognized:
|
||||
* \
|
||||
* c:\ (must have trailing slash)
|
||||
* \\server\share (trailing slash optional)
|
||||
* \\?\c: (trailing slash optional)
|
||||
* \\?\UNC\server\share (trailing slash optional)
|
||||
*/
|
||||
static int uv__is_root(const wchar_t* path) {
|
||||
size_t len = wcslen(path);
|
||||
|
||||
/* Test for \ */
|
||||
if (len == 0 && IS_SLASH(path[0])) {
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (len < 3) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* Test for c:\ */
|
||||
if (IS_LETTER(path[0]) && IS_COLON(path[1]) && IS_SLASH(path[2])) {
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (!IS_SLASH(path[0]) || !IS_SLASH(path[1])) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* Test for \\server\share */
|
||||
if (!IS_QUESTION(path[2])) {
|
||||
return uv__count_slash_separated_words(path + 2, path + len, 2) == 2;
|
||||
}
|
||||
|
||||
if (!IS_SLASH(path[3])) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
if ((len == 6 || len == 7) &&
|
||||
IS_LETTER(path[4]) && IS_COLON(path[5]) &&
|
||||
(len == 6 || IS_SLASH(path[6]))) {
|
||||
return 1;
|
||||
}
|
||||
|
||||
/* Test for \\?\UNC\server\share */
|
||||
if (len >= 8 &&
|
||||
(path[4] == L'u' || path[4] == L'U') &&
|
||||
(path[5] == L'n' || path[5] == L'N') &&
|
||||
(path[6] == L'c' || path[6] == L'C') &&
|
||||
IS_SLASH(path[7])) {
|
||||
return uv__count_slash_separated_words(path + 8, path + len, 2) == 2;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
void fs__stat(uv_fs_t* req, const wchar_t* path) {
|
||||
HANDLE file;
|
||||
WIN32_FIND_DATAW ent;
|
||||
@ -496,6 +593,30 @@ void fs__stat(uv_fs_t* req, const wchar_t* path) {
|
||||
|
||||
req->ptr = NULL;
|
||||
|
||||
if (uv__is_root(path)) {
|
||||
/* We can't stat root directories like c:\. _wstati64 can't either, but */
|
||||
/* it will make up something reasonable. */
|
||||
DWORD drive_type = GetDriveTypeW(path);
|
||||
if (drive_type == DRIVE_UNKNOWN || drive_type == DRIVE_NO_ROOT_DIR) {
|
||||
req->last_error = ERROR_PATH_NOT_FOUND;
|
||||
req->errorno = UV_ENOENT;
|
||||
req->result = -1;
|
||||
return;
|
||||
}
|
||||
|
||||
memset(&req->stat, 0, sizeof req->stat);
|
||||
|
||||
req->stat.st_nlink = 1;
|
||||
req->stat.st_mode = ((_S_IREAD|_S_IWRITE) + ((_S_IREAD|_S_IWRITE) >> 3) +
|
||||
((_S_IREAD|_S_IWRITE) >> 6)) | S_IFDIR;
|
||||
|
||||
req->last_error = ERROR_SUCCESS;
|
||||
req->errorno = UV_OK;
|
||||
req->result = 0;
|
||||
req->ptr = &req->stat;
|
||||
return;
|
||||
}
|
||||
|
||||
file = FindFirstFileExW(path, FindExInfoStandard, &ent,
|
||||
FindExSearchNameMatch, NULL, 0);
|
||||
|
||||
@ -516,7 +637,7 @@ void fs__stat(uv_fs_t* req, const wchar_t* path) {
|
||||
if (result != -1) {
|
||||
req->ptr = &req->stat;
|
||||
}
|
||||
|
||||
|
||||
SET_REQ_RESULT(req, result);
|
||||
}
|
||||
|
||||
|
||||
@ -1069,12 +1069,25 @@ static int uv_tty_set_style(uv_tty_t* handle, DWORD* error) {
|
||||
bg_bright = 0;
|
||||
|
||||
} else if (arg == 1) {
|
||||
/* Bright */
|
||||
/* Foreground bright on */
|
||||
fg_bright = 1;
|
||||
|
||||
} else if (arg == 21 || arg == 22) {
|
||||
/* Bright off. */
|
||||
} else if (arg == 2) {
|
||||
/* Both bright off */
|
||||
fg_bright = 0;
|
||||
bg_bright = 0;
|
||||
|
||||
} else if (arg == 5) {
|
||||
/* Background bright on */
|
||||
bg_bright = 1;
|
||||
|
||||
} else if (arg == 21 || arg == 22) {
|
||||
/* Foreground bright off */
|
||||
fg_bright = 0;
|
||||
|
||||
} else if (arg == 25) {
|
||||
/* Background bright off */
|
||||
bg_bright = 0;
|
||||
|
||||
} else if (arg >= 30 && arg <= 37) {
|
||||
/* Set foreground color */
|
||||
@ -1091,6 +1104,17 @@ static int uv_tty_set_style(uv_tty_t* handle, DWORD* error) {
|
||||
} else if (arg == 49) {
|
||||
/* Default background color */
|
||||
bg_color = 0;
|
||||
|
||||
} else if (arg >= 90 && arg <= 97) {
|
||||
/* Set bold foreground color */
|
||||
fg_bright = 1;
|
||||
fg_color = arg - 90;
|
||||
|
||||
} else if (arg >= 100 && arg <= 107) {
|
||||
/* Set bold background color */
|
||||
bg_bright = 1;
|
||||
bg_color = arg - 100;
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@ -575,6 +575,12 @@ void uv_process_udp_send_req(uv_loop_t* loop, uv_udp_t* handle,
|
||||
}
|
||||
|
||||
|
||||
int uv_udp_set_multicast_loop(uv_udp_t* handle, int on) {
|
||||
uv__set_artificial_error(handle->loop, UV_ENOSYS);
|
||||
return -1;
|
||||
}
|
||||
|
||||
|
||||
int uv_udp_set_multicast_ttl(uv_udp_t* handle, int ttl) {
|
||||
if (setsockopt(handle->socket, IPPROTO_IP, IP_MULTICAST_TTL,
|
||||
(const char*)&ttl, sizeof ttl) == -1) {
|
||||
@ -595,3 +601,9 @@ int uv_udp_set_broadcast(uv_udp_t* handle, int on) {
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
int uv_udp_set_ttl(uv_udp_t* handle, int ttl) {
|
||||
uv__set_artificial_error(handle->loop, UV_ENOSYS);
|
||||
return -1;
|
||||
}
|
||||
|
||||
@ -45,6 +45,7 @@
|
||||
# define close _close
|
||||
#endif
|
||||
|
||||
#define TOO_LONG_NAME_LENGTH 8192
|
||||
|
||||
typedef struct {
|
||||
const char* path;
|
||||
@ -416,6 +417,14 @@ static void open_noent_cb(uv_fs_t* req) {
|
||||
uv_fs_req_cleanup(req);
|
||||
}
|
||||
|
||||
static void open_nametoolong_cb(uv_fs_t* req) {
|
||||
ASSERT(req->fs_type == UV_FS_OPEN);
|
||||
ASSERT(req->errorno == UV_ENAMETOOLONG);
|
||||
ASSERT(req->result == -1);
|
||||
open_cb_count++;
|
||||
uv_fs_req_cleanup(req);
|
||||
}
|
||||
|
||||
|
||||
TEST_IMPL(fs_file_noent) {
|
||||
uv_fs_t req;
|
||||
@ -441,6 +450,31 @@ TEST_IMPL(fs_file_noent) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
TEST_IMPL(fs_file_nametoolong) {
|
||||
uv_fs_t req;
|
||||
int r;
|
||||
char name[TOO_LONG_NAME_LENGTH + 1];
|
||||
|
||||
loop = uv_default_loop();
|
||||
|
||||
memset(name, 'a', TOO_LONG_NAME_LENGTH);
|
||||
name[TOO_LONG_NAME_LENGTH] = 0;
|
||||
|
||||
r = uv_fs_open(loop, &req, name, O_RDONLY, 0, NULL);
|
||||
ASSERT(r == -1);
|
||||
ASSERT(req.result == -1);
|
||||
ASSERT(uv_last_error(loop).code == UV_ENAMETOOLONG);
|
||||
uv_fs_req_cleanup(&req);
|
||||
|
||||
r = uv_fs_open(loop, &req, name, O_RDONLY, 0, open_nametoolong_cb);
|
||||
ASSERT(r == 0);
|
||||
|
||||
ASSERT(open_cb_count == 0);
|
||||
uv_run(loop);
|
||||
ASSERT(open_cb_count == 1);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void check_utime(const char* path, double atime, double mtime) {
|
||||
struct stat* s;
|
||||
@ -1254,6 +1288,22 @@ TEST_IMPL(fs_utime) {
|
||||
}
|
||||
|
||||
|
||||
#ifdef _WIN32
|
||||
TEST_IMPL(fs_stat_root) {
|
||||
int r;
|
||||
uv_loop_t* loop = uv_default_loop();
|
||||
|
||||
r = uv_fs_stat(loop, &stat_req, "c:\\", NULL);
|
||||
ASSERT(r == 0);
|
||||
|
||||
r = uv_fs_stat(loop, &stat_req, "\\\\?\\C:\\", NULL);
|
||||
ASSERT(r == 0);
|
||||
|
||||
return 0;
|
||||
}
|
||||
#endif
|
||||
|
||||
|
||||
TEST_IMPL(fs_futime) {
|
||||
utime_check_t checkme;
|
||||
const char* path = "test_file";
|
||||
@ -1543,4 +1593,4 @@ TEST_IMPL(fs_rename_to_existing_file) {
|
||||
unlink("test_file2");
|
||||
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
@ -106,6 +106,7 @@ TEST_DECLARE (spawn_and_kill_with_std)
|
||||
TEST_DECLARE (spawn_and_ping)
|
||||
TEST_DECLARE (kill)
|
||||
TEST_DECLARE (fs_file_noent)
|
||||
TEST_DECLARE (fs_file_nametoolong)
|
||||
TEST_DECLARE (fs_file_async)
|
||||
TEST_DECLARE (fs_file_sync)
|
||||
TEST_DECLARE (fs_async_dir)
|
||||
@ -143,6 +144,7 @@ TEST_DECLARE (argument_escaping)
|
||||
TEST_DECLARE (environment_creation)
|
||||
TEST_DECLARE (listen_with_simultaneous_accepts)
|
||||
TEST_DECLARE (listen_no_simultaneous_accepts)
|
||||
TEST_DECLARE (fs_stat_root)
|
||||
#endif
|
||||
HELPER_DECLARE (tcp4_echo_server)
|
||||
HELPER_DECLARE (tcp6_echo_server)
|
||||
@ -281,9 +283,11 @@ TASK_LIST_START
|
||||
TEST_ENTRY (environment_creation)
|
||||
TEST_ENTRY (listen_with_simultaneous_accepts)
|
||||
TEST_ENTRY (listen_no_simultaneous_accepts)
|
||||
TEST_ENTRY (fs_stat_root)
|
||||
#endif
|
||||
|
||||
TEST_ENTRY (fs_file_noent)
|
||||
TEST_ENTRY (fs_file_nametoolong)
|
||||
TEST_ENTRY (fs_file_async)
|
||||
TEST_ENTRY (fs_file_sync)
|
||||
TEST_ENTRY (fs_async_dir)
|
||||
|
||||
Loading…
Reference in New Issue
Block a user