mirror of https://github.com/nodejs/node.git
494 lines
13 KiB
C
494 lines
13 KiB
C
/* 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.
|
|
*/
|
|
|
|
#ifndef UV_H
|
|
#define UV_H
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
#define UV_VERSION_MAJOR 0
|
|
#define UV_VERSION_MINOR 1
|
|
|
|
#define CARES_STATICLIB 1
|
|
|
|
#include <stdint.h> /* int64_t */
|
|
#include <sys/types.h> /* size_t */
|
|
|
|
#include "c-ares/ares.h"
|
|
|
|
#ifndef _SSIZE_T_
|
|
typedef intptr_t ssize_t;
|
|
#endif
|
|
|
|
typedef struct uv_err_s uv_err_t;
|
|
typedef struct uv_handle_s uv_handle_t;
|
|
typedef struct uv_stream_s uv_stream_t;
|
|
typedef struct uv_tcp_s uv_tcp_t;
|
|
typedef struct uv_timer_s uv_timer_t;
|
|
typedef struct uv_prepare_s uv_prepare_t;
|
|
typedef struct uv_check_s uv_check_t;
|
|
typedef struct uv_idle_s uv_idle_t;
|
|
typedef struct uv_req_s uv_req_t;
|
|
typedef struct uv_async_s uv_async_t;
|
|
typedef struct uv_getaddrinfo_s uv_getaddrinfo_t;
|
|
|
|
|
|
#if defined(__unix__) || defined(__POSIX__) || defined(__APPLE__)
|
|
# include "uv-unix.h"
|
|
#else
|
|
# include "uv-win.h"
|
|
#endif
|
|
|
|
|
|
/* The status parameter is 0 if the request completed successfully,
|
|
* and should be -1 if the request was cancelled or failed.
|
|
* For uv_close_cb, -1 means that the handle was closed due to an error.
|
|
* Error details can be obtained by calling uv_last_error().
|
|
*
|
|
* In the case of uv_read_cb the uv_buf_t returned should be freed by the
|
|
* user.
|
|
*/
|
|
typedef uv_buf_t (*uv_alloc_cb)(uv_stream_t* tcp, size_t suggested_size);
|
|
typedef void (*uv_read_cb)(uv_stream_t* tcp, ssize_t nread, uv_buf_t buf);
|
|
typedef void (*uv_write_cb)(uv_req_t* req, int status);
|
|
typedef void (*uv_connect_cb)(uv_req_t* req, int status);
|
|
typedef void (*uv_shutdown_cb)(uv_req_t* req, int status);
|
|
typedef void (*uv_connection_cb)(uv_handle_t* server, int status);
|
|
typedef void (*uv_close_cb)(uv_handle_t* handle);
|
|
typedef void (*uv_timer_cb)(uv_timer_t* handle, int status);
|
|
/* TODO: do these really need a status argument? */
|
|
typedef void (*uv_async_cb)(uv_async_t* handle, int status);
|
|
typedef void (*uv_prepare_cb)(uv_prepare_t* handle, int status);
|
|
typedef void (*uv_check_cb)(uv_check_t* handle, int status);
|
|
typedef void (*uv_idle_cb)(uv_idle_t* handle, int status);
|
|
typedef void (*uv_getaddrinfo_cb)(uv_getaddrinfo_t* handle, int status, struct addrinfo* res);
|
|
|
|
|
|
/* Expand this list if necessary. */
|
|
typedef enum {
|
|
UV_UNKNOWN = -1,
|
|
UV_OK = 0,
|
|
UV_EOF,
|
|
UV_EACCESS,
|
|
UV_EAGAIN,
|
|
UV_EADDRINUSE,
|
|
UV_EADDRNOTAVAIL,
|
|
UV_EAFNOSUPPORT,
|
|
UV_EALREADY,
|
|
UV_EBADF,
|
|
UV_EBUSY,
|
|
UV_ECONNABORTED,
|
|
UV_ECONNREFUSED,
|
|
UV_ECONNRESET,
|
|
UV_EDESTADDRREQ,
|
|
UV_EFAULT,
|
|
UV_EHOSTUNREACH,
|
|
UV_EINTR,
|
|
UV_EINVAL,
|
|
UV_EISCONN,
|
|
UV_EMFILE,
|
|
UV_ENETDOWN,
|
|
UV_ENETUNREACH,
|
|
UV_ENFILE,
|
|
UV_ENOBUFS,
|
|
UV_ENOMEM,
|
|
UV_ENONET,
|
|
UV_ENOPROTOOPT,
|
|
UV_ENOTCONN,
|
|
UV_ENOTSOCK,
|
|
UV_ENOTSUP,
|
|
UV_EPROTO,
|
|
UV_EPROTONOSUPPORT,
|
|
UV_EPROTOTYPE,
|
|
UV_ETIMEDOUT,
|
|
UV_ECHARSET,
|
|
UV_EAIFAMNOSUPPORT,
|
|
UV_EAINONAME,
|
|
UV_EAISERVICE,
|
|
UV_EAISOCKTYPE
|
|
} uv_err_code;
|
|
|
|
typedef enum {
|
|
UV_UNKNOWN_HANDLE = 0,
|
|
UV_TCP,
|
|
UV_NAMED_PIPE,
|
|
UV_TTY,
|
|
UV_FILE,
|
|
UV_TIMER,
|
|
UV_PREPARE,
|
|
UV_CHECK,
|
|
UV_IDLE,
|
|
UV_ASYNC,
|
|
UV_ARES,
|
|
UV_ARES_TASK,
|
|
UV_GETADDRINFO
|
|
} uv_handle_type;
|
|
|
|
typedef enum {
|
|
UV_UNKNOWN_REQ = 0,
|
|
UV_CONNECT,
|
|
UV_ACCEPT,
|
|
UV_READ,
|
|
UV_WRITE,
|
|
UV_SHUTDOWN,
|
|
UV_WAKEUP
|
|
} uv_req_type;
|
|
|
|
|
|
struct uv_err_s {
|
|
/* read-only */
|
|
uv_err_code code;
|
|
/* private */
|
|
int sys_errno_;
|
|
};
|
|
|
|
|
|
struct uv_req_s {
|
|
/* read-only */
|
|
uv_req_type type;
|
|
/* public */
|
|
uv_handle_t* handle;
|
|
void* cb;
|
|
void* data;
|
|
/* private */
|
|
UV_REQ_PRIVATE_FIELDS
|
|
};
|
|
|
|
/*
|
|
* Initialize a request for use with uv_write, uv_shutdown, or uv_connect.
|
|
*/
|
|
void uv_req_init(uv_req_t* req, uv_handle_t* handle, void* cb);
|
|
|
|
int uv_shutdown(uv_req_t* req);
|
|
|
|
|
|
#define UV_HANDLE_FIELDS \
|
|
/* read-only */ \
|
|
uv_handle_type type; \
|
|
/* public */ \
|
|
uv_close_cb close_cb; \
|
|
void* data; \
|
|
/* private */ \
|
|
UV_HANDLE_PRIVATE_FIELDS \
|
|
|
|
/* The abstract base class of all handles. */
|
|
struct uv_handle_s {
|
|
UV_HANDLE_FIELDS
|
|
};
|
|
|
|
/*
|
|
* Returns 1 if the prepare/check/idle handle has been started, 0 otherwise.
|
|
* For other handle types this always returns 1.
|
|
*/
|
|
int uv_is_active(uv_handle_t* handle);
|
|
|
|
/*
|
|
* Request handle to be closed. close_cb will be called asynchronously after
|
|
* this call. This MUST be called on each handle before memory is released.
|
|
*/
|
|
int uv_close(uv_handle_t* handle, uv_close_cb close_cb);
|
|
|
|
|
|
#define UV_STREAM_FIELDS \
|
|
/* number of bytes queued for writing */ \
|
|
size_t write_queue_size; \
|
|
/* private */ \
|
|
UV_STREAM_PRIVATE_FIELDS \
|
|
|
|
/* The abstract base class for all streams. */
|
|
struct uv_stream_s {
|
|
UV_HANDLE_FIELDS
|
|
UV_STREAM_FIELDS
|
|
};
|
|
|
|
/* This call is used in conjunction with uv_listen() to accept incoming
|
|
* connections. Call uv_accept after receiving a uv_connection_cb to accept
|
|
* the connection. Before calling uv_accept use uv_*_init() must be
|
|
* called on the client. Non-zero return value indicates an error.
|
|
*
|
|
* When the uv_connection_cb is called it is guaranteed that uv_accept will
|
|
* complete successfully the first time. If you attempt to use it more than
|
|
* once, it may fail. It is suggested to only call uv_accept once per
|
|
* uv_connection_cb call.
|
|
*/
|
|
int uv_accept(uv_handle_t* server, uv_stream_t* client);
|
|
|
|
/* Read data from an incoming stream. The callback will be made several
|
|
* several times until there is no more data to read or uv_read_stop is
|
|
* called. When we've reached EOF nread will be set to -1 and the error is
|
|
* set to UV_EOF. When nread == -1 the buf parameter might not point to a
|
|
* valid buffer; in that case buf.len and buf.base are both set to 0.
|
|
* Note that nread might also be 0, which does *not* indicate an error or
|
|
* eof; it happens when libuv requested a buffer through the alloc callback
|
|
* but then decided that it didn't need that buffer.
|
|
*/
|
|
int uv_read_start(uv_stream_t*, uv_alloc_cb alloc_cb, uv_read_cb read_cb);
|
|
|
|
int uv_read_stop(uv_stream_t*);
|
|
|
|
/* Write data to stream. Buffers are written in order. Example:
|
|
*
|
|
* uv_buf_t a[] = {
|
|
* { .base = "1", .len = 1 },
|
|
* { .base = "2", .len = 1 }
|
|
* };
|
|
*
|
|
* uv_buf_t b[] = {
|
|
* { .base = "3", .len = 1 },
|
|
* { .base = "4", .len = 1 }
|
|
* };
|
|
*
|
|
* // writes "1234"
|
|
* uv_write(req, a, 2);
|
|
* uv_write(req, b, 2);
|
|
*
|
|
*/
|
|
int uv_write(uv_req_t* req, uv_buf_t bufs[], int bufcnt);
|
|
|
|
|
|
/*
|
|
* A subclass of uv_stream_t representing a TCP stream or TCP server. In the
|
|
* future this will probably be split into two classes - one a stream and
|
|
* the other a server.
|
|
*/
|
|
struct uv_tcp_s {
|
|
UV_HANDLE_FIELDS
|
|
UV_STREAM_FIELDS
|
|
UV_TCP_PRIVATE_FIELDS
|
|
};
|
|
|
|
int uv_tcp_init(uv_tcp_t* handle);
|
|
|
|
int uv_tcp_bind(uv_tcp_t* handle, struct sockaddr_in);
|
|
int uv_tcp_bind6(uv_tcp_t* handle, struct sockaddr_in6);
|
|
|
|
int uv_tcp_connect(uv_req_t* req, struct sockaddr_in);
|
|
|
|
int uv_tcp_listen(uv_tcp_t* handle, int backlog, uv_connection_cb cb);
|
|
|
|
|
|
/*
|
|
* Subclass of uv_handle_t. libev wrapper. Every active prepare handle gets
|
|
* its callback called exactly once per loop iteration, just before the
|
|
* system blocks to wait for completed i/o.
|
|
*/
|
|
struct uv_prepare_s {
|
|
UV_HANDLE_FIELDS
|
|
UV_PREPARE_PRIVATE_FIELDS
|
|
};
|
|
|
|
int uv_prepare_init(uv_prepare_t* prepare);
|
|
|
|
int uv_prepare_start(uv_prepare_t* prepare, uv_prepare_cb cb);
|
|
|
|
int uv_prepare_stop(uv_prepare_t* prepare);
|
|
|
|
|
|
/*
|
|
* Subclass of uv_handle_t. libev wrapper. Every active check handle gets
|
|
* its callback called exactly once per loop iteration, just after the
|
|
* system returns from blocking.
|
|
*/
|
|
struct uv_check_s {
|
|
UV_HANDLE_FIELDS
|
|
UV_CHECK_PRIVATE_FIELDS
|
|
};
|
|
|
|
int uv_check_init(uv_check_t* check);
|
|
|
|
int uv_check_start(uv_check_t* check, uv_check_cb cb);
|
|
|
|
int uv_check_stop(uv_check_t* check);
|
|
|
|
|
|
/*
|
|
* Subclass of uv_handle_t. libev wrapper. Every active idle handle gets its
|
|
* callback called repeatedly until it is stopped. This happens after all
|
|
* other types of callbacks are processed. When there are multiple "idle"
|
|
* handles active, their callbacks are called in turn.
|
|
*/
|
|
struct uv_idle_s {
|
|
UV_HANDLE_FIELDS
|
|
UV_IDLE_PRIVATE_FIELDS
|
|
};
|
|
|
|
int uv_idle_init(uv_idle_t* idle);
|
|
|
|
int uv_idle_start(uv_idle_t* idle, uv_idle_cb cb);
|
|
|
|
int uv_idle_stop(uv_idle_t* idle);
|
|
|
|
|
|
/*
|
|
* Subclass of uv_handle_t. libev wrapper. uv_async_send wakes up the event
|
|
* loop and calls the async handle's callback There is no guarantee that
|
|
* every uv_async_send call leads to exactly one invocation of the callback;
|
|
* The only guarantee is that the callback function is called at least once
|
|
* after the call to async_send. Unlike all other libuv functions,
|
|
* uv_async_send can be called from another thread.
|
|
*/
|
|
struct uv_async_s {
|
|
UV_HANDLE_FIELDS
|
|
UV_ASYNC_PRIVATE_FIELDS
|
|
};
|
|
|
|
int uv_async_init(uv_async_t* async, uv_async_cb async_cb);
|
|
|
|
int uv_async_send(uv_async_t* async);
|
|
|
|
|
|
/*
|
|
* Subclass of uv_handle_t. Wraps libev's ev_timer watcher. Used to get
|
|
* woken up at a specified time in the future.
|
|
*/
|
|
struct uv_timer_s {
|
|
UV_HANDLE_FIELDS
|
|
UV_TIMER_PRIVATE_FIELDS
|
|
};
|
|
|
|
int uv_timer_init(uv_timer_t* timer);
|
|
|
|
int uv_timer_start(uv_timer_t* timer, uv_timer_cb cb, int64_t timeout, int64_t repeat);
|
|
|
|
int uv_timer_stop(uv_timer_t* timer);
|
|
|
|
/*
|
|
* Stop the timer, and if it is repeating restart it using the repeat value
|
|
* as the timeout. If the timer has never been started before it returns -1 and
|
|
* sets the error to UV_EINVAL.
|
|
*/
|
|
int uv_timer_again(uv_timer_t* timer);
|
|
|
|
/*
|
|
* Set the repeat value. Note that if the repeat value is set from a timer
|
|
* callback it does not immediately take effect. If the timer was nonrepeating
|
|
* before, it will have been stopped. If it was repeating, then the old repeat
|
|
* value will have been used to schedule the next timeout.
|
|
*/
|
|
void uv_timer_set_repeat(uv_timer_t* timer, int64_t repeat);
|
|
|
|
int64_t uv_timer_get_repeat(uv_timer_t* timer);
|
|
|
|
|
|
/* c-ares integration initialize and terminate */
|
|
int uv_ares_init_options(ares_channel *channelptr,
|
|
struct ares_options *options,
|
|
int optmask);
|
|
|
|
void uv_ares_destroy(ares_channel channel);
|
|
|
|
|
|
/*
|
|
* Subclass of uv_handle_t. Used for integration of getaddrinfo.
|
|
*/
|
|
struct uv_getaddrinfo_s {
|
|
UV_HANDLE_FIELDS
|
|
UV_GETADDRINFO_PRIVATE_FIELDS
|
|
};
|
|
|
|
|
|
/* uv_getaddrinfo
|
|
* return code of UV_OK means that request is accepted,
|
|
* and callback will be called with result.
|
|
* Other return codes mean that there will not be a callback.
|
|
* Input arguments may be released after return from this call.
|
|
* Callback must not call freeaddrinfo
|
|
*/
|
|
int uv_getaddrinfo(uv_getaddrinfo_t* handle,
|
|
uv_getaddrinfo_cb getaddrinfo_cb,
|
|
const char* node,
|
|
const char* service,
|
|
const struct addrinfo* hints);
|
|
|
|
|
|
/*
|
|
* Most functions return boolean: 0 for success and -1 for failure.
|
|
* On error the user should then call uv_last_error() to determine
|
|
* the error code.
|
|
*/
|
|
uv_err_t uv_last_error();
|
|
char* uv_strerror(uv_err_t err);
|
|
const char* uv_err_name(uv_err_t err);
|
|
|
|
void uv_init();
|
|
int uv_run();
|
|
|
|
/*
|
|
* Manually modify the event loop's reference count. Useful if the user wants
|
|
* to have a handle or timeout that doesn't keep the loop alive.
|
|
*/
|
|
void uv_ref();
|
|
void uv_unref();
|
|
|
|
void uv_update_time();
|
|
int64_t uv_now();
|
|
|
|
|
|
/* Utility */
|
|
struct sockaddr_in uv_ip4_addr(const char* ip, int port);
|
|
struct sockaddr_in6 uv_ip6_addr(const char* ip, int port);
|
|
|
|
/* Gets the executable path */
|
|
int uv_exepath(char* buffer, size_t* size);
|
|
|
|
/*
|
|
* Returns the current high-resolution real time. This is expressed in
|
|
* nanoseconds. It is relative to an arbitrary time in the past. It is not
|
|
* related to the time of day and therefore not subject to clock drift. The
|
|
* primary use is for measuring performance between intervals.
|
|
*
|
|
* Note not every platform can support nanosecond resolution; however, this
|
|
* value will always be in nanoseconds.
|
|
*/
|
|
extern uint64_t uv_hrtime(void);
|
|
|
|
|
|
/* the presence of this union forces similar struct layout */
|
|
union uv_any_handle {
|
|
uv_tcp_t tcp;
|
|
uv_prepare_t prepare;
|
|
uv_check_t check;
|
|
uv_idle_t idle;
|
|
uv_async_t async;
|
|
uv_timer_t timer;
|
|
uv_getaddrinfo_t getaddrinfo;
|
|
};
|
|
|
|
/* Diagnostic counters */
|
|
typedef struct {
|
|
uint64_t req_init;
|
|
uint64_t handle_init;
|
|
uint64_t tcp_init;
|
|
uint64_t prepare_init;
|
|
uint64_t check_init;
|
|
uint64_t idle_init;
|
|
uint64_t async_init;
|
|
uint64_t timer_init;
|
|
} uv_counters_t;
|
|
|
|
uv_counters_t* uv_counters();
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
#endif /* UV_H */
|