/* Copyright Joyent, Inc. and other Node 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 "uv-common.h" #include #include /* NULL */ #include /* malloc */ #include /* memset */ /* use inet_pton from c-ares if necessary */ #include "ares_config.h" #include "ares/inet_net_pton.h" #include "ares/inet_ntop.h" size_t uv_strlcpy(char* dst, const char* src, size_t size) { size_t n; if (size == 0) return 0; for (n = 0; n < (size - 1) && *src != '\0'; n++) *dst++ = *src++; *dst = '\0'; return n; } size_t uv_strlcat(char* dst, const char* src, size_t size) { size_t n; if (size == 0) return 0; for (n = 0; n < size && *dst != '\0'; n++, dst++); if (n == size) return n; while (n < (size - 1) && *src != '\0') n++, *dst++ = *src++; *dst = '\0'; return n; } uv_buf_t uv_buf_init(char* base, size_t len) { uv_buf_t buf; buf.base = base; buf.len = len; return buf; } const uv_err_t uv_ok_ = { UV_OK, 0 }; #define UV_ERR_NAME_GEN(val, name, s) case UV_##name : return #name; const char* uv_err_name(uv_err_t err) { switch (err.code) { UV_ERRNO_MAP(UV_ERR_NAME_GEN) default: assert(0); return NULL; } } #undef UV_ERR_NAME_GEN #define UV_STRERROR_GEN(val, name, s) case UV_##name : return s; const char* uv_strerror(uv_err_t err) { switch (err.code) { UV_ERRNO_MAP(UV_STRERROR_GEN) default: return "Unknown system error"; } } #undef UV_STRERROR_GEN int uv__set_error(uv_loop_t* loop, uv_err_code code, int sys_error) { loop->last_err.code = code; loop->last_err.sys_errno_ = sys_error; return -1; } int uv__set_sys_error(uv_loop_t* loop, int sys_error) { loop->last_err.code = uv_translate_sys_error(sys_error); loop->last_err.sys_errno_ = sys_error; return -1; } int uv__set_artificial_error(uv_loop_t* loop, uv_err_code code) { loop->last_err = uv__new_artificial_error(code); return -1; } uv_err_t uv__new_sys_error(int sys_error) { uv_err_t error; error.code = uv_translate_sys_error(sys_error); error.sys_errno_ = sys_error; return error; } uv_err_t uv__new_artificial_error(uv_err_code code) { uv_err_t error; error.code = code; error.sys_errno_ = 0; return error; } uv_err_t uv_last_error(uv_loop_t* loop) { return loop->last_err; } struct sockaddr_in uv_ip4_addr(const char* ip, int port) { struct sockaddr_in addr; memset(&addr, 0, sizeof(struct sockaddr_in)); addr.sin_family = AF_INET; addr.sin_port = htons(port); addr.sin_addr.s_addr = inet_addr(ip); return addr; } struct sockaddr_in6 uv_ip6_addr(const char* ip, int port) { struct sockaddr_in6 addr; memset(&addr, 0, sizeof(struct sockaddr_in6)); addr.sin6_family = AF_INET6; addr.sin6_port = htons(port); ares_inet_pton(AF_INET6, ip, &addr.sin6_addr); return addr; } int uv_ip4_name(struct sockaddr_in* src, char* dst, size_t size) { const char* d = ares_inet_ntop(AF_INET, &src->sin_addr, dst, size); return d != dst; } int uv_ip6_name(struct sockaddr_in6* src, char* dst, size_t size) { const char* d = ares_inet_ntop(AF_INET6, &src->sin6_addr, dst, size); return d != dst; } static int cmp_ares_tasks(const uv_ares_task_t* a, const uv_ares_task_t* b) { if (a->sock < b->sock) return -1; if (a->sock > b->sock) return 1; return 0; } RB_GENERATE_STATIC(uv__ares_tasks, uv_ares_task_s, node, cmp_ares_tasks) /* add ares handle to list */ void uv_add_ares_handle(uv_loop_t* loop, uv_ares_task_t* handle) { assert(loop == handle->loop); RB_INSERT(uv__ares_tasks, &loop->uv_ares_handles_, handle); } /* find matching ares handle in list */ uv_ares_task_t* uv_find_ares_handle(uv_loop_t* loop, ares_socket_t sock) { uv_ares_task_t handle; handle.sock = sock; return RB_FIND(uv__ares_tasks, &loop->uv_ares_handles_, &handle); } /* remove ares handle in list */ void uv_remove_ares_handle(uv_ares_task_t* handle) { RB_REMOVE(uv__ares_tasks, &handle->loop->uv_ares_handles_, handle); } /* Returns 1 if the uv_ares_handles_ list is empty. 0 otherwise. */ int uv_ares_handles_empty(uv_loop_t* loop) { return RB_EMPTY(&loop->uv_ares_handles_); } int uv_tcp_bind(uv_tcp_t* handle, struct sockaddr_in addr) { if (handle->type != UV_TCP || addr.sin_family != AF_INET) { uv__set_artificial_error(handle->loop, UV_EFAULT); return -1; } return uv__tcp_bind(handle, addr); } int uv_tcp_bind6(uv_tcp_t* handle, struct sockaddr_in6 addr) { if (handle->type != UV_TCP || addr.sin6_family != AF_INET6) { uv__set_artificial_error(handle->loop, UV_EFAULT); return -1; } return uv__tcp_bind6(handle, addr); } int uv_udp_bind(uv_udp_t* handle, struct sockaddr_in addr, unsigned int flags) { if (handle->type != UV_UDP || addr.sin_family != AF_INET) { uv__set_artificial_error(handle->loop, UV_EFAULT); return -1; } return uv__udp_bind(handle, addr, flags); } int uv_udp_bind6(uv_udp_t* handle, struct sockaddr_in6 addr, unsigned int flags) { if (handle->type != UV_UDP || addr.sin6_family != AF_INET6) { uv__set_artificial_error(handle->loop, UV_EFAULT); return -1; } return uv__udp_bind6(handle, addr, flags); } int uv_tcp_connect(uv_connect_t* req, uv_tcp_t* handle, struct sockaddr_in address, uv_connect_cb cb) { if (handle->type != UV_TCP || address.sin_family != AF_INET) { uv__set_artificial_error(handle->loop, UV_EINVAL); return -1; } return uv__tcp_connect(req, handle, address, cb); } int uv_tcp_connect6(uv_connect_t* req, uv_tcp_t* handle, struct sockaddr_in6 address, uv_connect_cb cb) { if (handle->type != UV_TCP || address.sin6_family != AF_INET6) { uv__set_artificial_error(handle->loop, UV_EINVAL); return -1; } return uv__tcp_connect6(req, handle, address, cb); } #ifdef _WIN32 static UINT __stdcall uv__thread_start(void *ctx_v) #else static void *uv__thread_start(void *ctx_v) #endif { void (*entry)(void *arg); void *arg; struct { void (*entry)(void *arg); void *arg; } *ctx; ctx = ctx_v; arg = ctx->arg; entry = ctx->entry; free(ctx); entry(arg); return 0; } int uv_thread_create(uv_thread_t *tid, void (*entry)(void *arg), void *arg) { struct { void (*entry)(void *arg); void *arg; } *ctx; if ((ctx = malloc(sizeof *ctx)) == NULL) return -1; ctx->entry = entry; ctx->arg = arg; #ifdef _WIN32 *tid = (HANDLE) _beginthreadex(NULL, 0, uv__thread_start, ctx, 0, NULL); if (*tid == 0) { #else if (pthread_create(tid, NULL, uv__thread_start, ctx)) { #endif free(ctx); return -1; } return 0; }