mirror of
https://github.com/danog/ext-uv.git
synced 2024-11-26 20:14:47 +01:00
6463 lines
157 KiB
C
6463 lines
157 KiB
C
/*
|
|
+----------------------------------------------------------------------+
|
|
| PHP Version 5 |
|
|
+----------------------------------------------------------------------+
|
|
| Copyright (c) 1997-2012 The PHP Group |
|
|
+----------------------------------------------------------------------+
|
|
| This source file is subject to version 3.01 of the PHP license, |
|
|
| that is bundled with this package in the file LICENSE, and is |
|
|
| available through the world-wide-web at the following url: |
|
|
| http://www.php.net/license/3_01.txt |
|
|
| If you did not receive a copy of the PHP license and are unable to |
|
|
| obtain it through the world-wide-web, please send a note to |
|
|
| license@php.net so we can mail you a copy immediately. |
|
|
+----------------------------------------------------------------------+
|
|
| Authors: Shuhei Tanuma <chobieeee@php.net> |
|
|
+----------------------------------------------------------------------+
|
|
*/
|
|
|
|
|
|
#include "php_uv.h"
|
|
#include "php_main.h"
|
|
#include "ext/standard/info.h"
|
|
|
|
#ifndef PHP_UV_DEBUG
|
|
#define PHP_UV_DEBUG 0
|
|
#endif
|
|
|
|
#ifdef ZTS
|
|
#undef TSRMLS_C
|
|
#undef TSRMLS_CC
|
|
#undef TSRMLS_D
|
|
#undef TSRMLS_DC
|
|
#define TSRMLS_C tsrm_ls
|
|
#define TSRMLS_CC , TSRMLS_C
|
|
#define TSRMLS_D void *tsrm_ls
|
|
#define TSRMLS_DC , TSRMLS_D
|
|
#endif
|
|
|
|
#define PHP_UV_INIT_UV(uv, uv_type) \
|
|
uv = (php_uv_t *)emalloc(sizeof(php_uv_t)); \
|
|
if (!uv) { \
|
|
php_error_docref(NULL, E_ERROR, "emalloc failed"); \
|
|
RETURN_FALSE; \
|
|
} else { \
|
|
uv->type = uv_type; \
|
|
PHP_UV_INIT_ZVALS(uv) \
|
|
TSRMLS_SET_CTX(uv->thread_ctx); \
|
|
uv->resource_id = PHP_UV_LIST_INSERT(uv, uv_resource_handle); \
|
|
}
|
|
|
|
#define PHP_UV_INIT_TIMER(uv, uv_type) \
|
|
uv = (php_uv_t *)emalloc(sizeof(php_uv_t)); \
|
|
if (!uv) { \
|
|
php_error_docref(NULL, E_ERROR, "emalloc failed"); \
|
|
RETURN_FALSE; \
|
|
} \
|
|
r = uv_timer_init(loop, &uv->uv.timer); \
|
|
if (r) { \
|
|
php_error_docref(NULL, E_WARNING, "uv_timer_init failed");\
|
|
RETURN_FALSE;\
|
|
} \
|
|
uv->type = uv_type; \
|
|
PHP_UV_INIT_ZVALS(uv) \
|
|
TSRMLS_SET_CTX(uv->thread_ctx); \
|
|
uv->resource_id = PHP_UV_LIST_INSERT(uv, uv_resource_handle); \
|
|
|
|
#define PHP_UV_INIT_SIGNAL(uv, uv_type) \
|
|
uv = (php_uv_t *)emalloc(sizeof(php_uv_t)); \
|
|
if (!uv) { \
|
|
php_error_docref(NULL, E_ERROR, "emalloc failed"); \
|
|
RETURN_FALSE; \
|
|
} \
|
|
if (uv_signal_init(loop, &uv->uv.signal)) { \
|
|
php_error_docref(NULL, E_WARNING, "uv_signal_init failed"); \
|
|
RETURN_FALSE;\
|
|
} \
|
|
uv->type = uv_type; \
|
|
PHP_UV_INIT_ZVALS(uv) \
|
|
TSRMLS_SET_CTX(uv->thread_ctx); \
|
|
uv->resource_id = PHP_UV_LIST_INSERT(uv, uv_resource_handle); \
|
|
|
|
#define PHP_UV_INIT_CONNECT(req, uv) \
|
|
req = (uv_connect_t*)emalloc(sizeof(uv_connect_t)); \
|
|
req->data = uv;
|
|
|
|
#define PHP_UV_INIT_WRITE_REQ(w, uv, str, strlen) \
|
|
w = emalloc(sizeof(write_req_t)); \
|
|
w->req.data = uv; \
|
|
w->buf = uv_buf_init(estrndup(str, strlen), strlen); \
|
|
|
|
#define PHP_UV_INIT_SEND_REQ(w, uv, str, strlen) \
|
|
w = emalloc(sizeof(send_req_t)); \
|
|
w->req.data = uv; \
|
|
w->buf = uv_buf_init(estrndup(str, strlen), strlen); \
|
|
|
|
#define PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop) \
|
|
if (zloop != NULL) { \
|
|
ZEND_FETCH_RESOURCE(loop, uv_loop_t*, zloop, -1, PHP_UV_LOOP_RESOURCE_NAME, uv_loop_handle); \
|
|
} else { \
|
|
loop = uv_default_loop(); \
|
|
} \
|
|
|
|
#define PHP_UV_INIT_LOCK(lock, lock_type) \
|
|
lock = emalloc(sizeof(php_uv_lock_t)); \
|
|
if (!lock) { \
|
|
php_error_docref(NULL, E_ERROR, "emalloc failed"); \
|
|
RETURN_FALSE; \
|
|
} \
|
|
lock->type = lock_type; \
|
|
lock->locked = 0; \
|
|
|
|
#define PHP_UV_ZVAL_TO_VALID_POLL_FD(fd, zstream) \
|
|
{ \
|
|
fd = php_uv_zval_to_valid_poll_fd(zstream); \
|
|
if (fd < 0) { \
|
|
php_error_docref(NULL, E_WARNING, "invalid variable passed. can't convert to fd."); \
|
|
RETURN_FALSE; \
|
|
} \
|
|
if (uv->fs_fd == NULL) { \
|
|
uv->fs_fd = Z_RES_P(zstream);\
|
|
Z_ADDREF_P(zstream);\
|
|
}\
|
|
}
|
|
|
|
#define PHP_UV_ZVAL_TO_FD(fd, zstream) \
|
|
{ \
|
|
fd = php_uv_zval_to_fd(zstream); \
|
|
if (fd < 0) { \
|
|
php_error_docref(NULL, E_WARNING, "invalid variable passed. can't convert to fd."); \
|
|
RETURN_FALSE; \
|
|
} \
|
|
if (uv->fs_fd == NULL) { \
|
|
uv->fs_fd = Z_RES_P(zstream);\
|
|
Z_ADDREF_P(zstream);\
|
|
}\
|
|
}
|
|
|
|
#define PHP_UV_FS_ASYNC(loop, func, ...) \
|
|
error = uv_fs_##func(loop, (uv_fs_t*)&uv->uv.fs, __VA_ARGS__, php_uv_fs_cb); \
|
|
if (error) { \
|
|
php_error_docref(NULL, E_WARNING, "uv_##func failed"); \
|
|
return; \
|
|
}
|
|
|
|
#define PHP_UV_INIT_ZVALS(uv) \
|
|
{ \
|
|
int ix = 0;\
|
|
for (ix = 0; ix < PHP_UV_CB_MAX; ix++) {\
|
|
uv->callback[ix] = NULL;\
|
|
} \
|
|
uv->fs_fd = NULL; \
|
|
uv->in_free = 0;\
|
|
}
|
|
|
|
#define PHP_UV_SOCKADDR_INIT(sockaddr, ip_type) \
|
|
sockaddr = (php_uv_sockaddr_t*)emalloc(sizeof(php_uv_sockaddr_t)); \
|
|
if (!sockaddr) { \
|
|
php_error_docref(NULL, E_ERROR, "emalloc failed"); \
|
|
RETURN_FALSE; \
|
|
} \
|
|
sockaddr->is_ipv4 = ip_type;
|
|
|
|
#define PHP_UV_SOCKADDR_IS_IPV4(sockaddr) (sockaddr->is_ipv4 == 1)
|
|
#define PHP_UV_SOCKADDR_IS_IPV6(sockaddr) (sockaddr->is_ipv4 == 0)
|
|
|
|
#define PHP_UV_SOCKADDR_IPV4(sockaddr) sockaddr->addr.ipv4
|
|
#define PHP_UV_SOCKADDR_IPV4_P(sockaddr) &sockaddr->addr.ipv4
|
|
|
|
#define PHP_UV_SOCKADDR_IPV6(sockaddr) sockaddr->addr.ipv6
|
|
#define PHP_UV_SOCKADDR_IPV6_P(sockaddr) &sockaddr->addr.ipv6
|
|
|
|
#define PHP_UV_LOCK_RWLOCK_P(resource) &resource->lock.rwlock
|
|
#define PHP_UV_LOCK_MUTEX_P(resource) &resource->lock.mutex
|
|
#define PHP_UV_LOCK_SEM_P(resource) &resource->lock.semaphore
|
|
|
|
#define PHP_UV_FD_TO_ZVAL(zv, fd) { php_stream *_stream = php_stream_fopen_from_fd(fd, "w+", NULL); zval *_z = (zv); php_stream_to_zval(_stream, _z); Z_ADDREF_P(_z); }
|
|
|
|
#if PHP_UV_DEBUG>=1
|
|
#define PHP_UV_DEBUG_PRINT(format, ...) fprintf(stderr, format, ## __VA_ARGS__)
|
|
#else
|
|
#define PHP_UV_DEBUG_PRINT(format, ...)
|
|
#endif
|
|
|
|
#if PHP_UV_DEBUG>=1
|
|
#define PHP_UV_DEBUG_RESOURCE_REFCOUNT(name, resource_id) \
|
|
{ \
|
|
zend_rsrc_list_entry *le; \
|
|
if (zend_hash_index_find(&EG(regular_list), resource_id, (void **) &le)==SUCCESS) { \
|
|
printf("# %s del(%d): %d->%d\n", #name, resource_id, le->refcount, le->refcount-1); \
|
|
} else { \
|
|
printf("# can't find (%s)", #name); \
|
|
} \
|
|
}
|
|
#else
|
|
#define PHP_UV_DEBUG_RESOURCE_REFCOUNT(name, resource_id)
|
|
#endif
|
|
|
|
#ifdef ZTS
|
|
#define UV_FETCH_ALL(ls, id, type) ((type) (*((void ***) ls))[TSRM_UNSHUFFLE_RSRC_ID(id)])
|
|
#define UV_FETCH_CTX(ls, id, type, element) (((type) (*((void ***) ls))[TSRM_UNSHUFFLE_RSRC_ID(id)])->element)
|
|
#define UV_CG(ls, v) UV_FETCH_CTX(ls, compiler_globals_id, zend_compiler_globals*, v)
|
|
#define UV_CG_ALL(ls) UV_FETCH_ALL(ls, compiler_globals_id, zend_compiler_globals*)
|
|
#define UV_EG(ls, v) UV_FETCH_CTX(ls, executor_globals_id, zend_executor_globals*, v)
|
|
#define UV_SG(ls, v) UV_FETCH_CTX(ls, sapi_globals_id, sapi_globals_struct*, v)
|
|
#define UV_EG_ALL(ls) UV_FETCH_ALL(ls, executor_globals_id, zend_executor_globals*)
|
|
#endif
|
|
|
|
#if !defined(PHP_WIN32) && !defined(HAVE_SOCKETS)
|
|
int (*php_sockets_le_socket)(void) = NULL;
|
|
#endif
|
|
extern void php_uv_init();
|
|
|
|
typedef struct {
|
|
uv_write_t req;
|
|
uv_buf_t buf;
|
|
} write_req_t;
|
|
|
|
typedef struct {
|
|
uv_udp_send_t req;
|
|
uv_buf_t buf;
|
|
} send_req_t;
|
|
|
|
enum php_uv_socket_type {
|
|
PHP_UV_TCP_IPV4 = 1,
|
|
PHP_UV_TCP_IPV6 = 2,
|
|
PHP_UV_TCP = 3,
|
|
PHP_UV_UDP_IPV4 = 16,
|
|
PHP_UV_UDP_IPV6 = 32,
|
|
PHP_UV_UDP = 48,
|
|
};
|
|
|
|
/* static variables */
|
|
|
|
static uv_loop_t *_php_uv_default_loop;
|
|
|
|
/* resources */
|
|
|
|
static int uv_resource_handle;
|
|
|
|
static int uv_loop_handle;
|
|
|
|
static int uv_sockaddr_handle;
|
|
|
|
static int uv_lock_handle;
|
|
|
|
static int uv_stdio_handle;
|
|
|
|
char *php_uv_resource_map[IS_UV_MAX] = {
|
|
"uv_tcp",
|
|
"uv_udp",
|
|
"uv_pipe",
|
|
"uv_idle",
|
|
"uv_timer",
|
|
"uv_async",
|
|
"uv_loop",
|
|
"uv_handle",
|
|
"uv_stream",
|
|
"uv_addrinfo",
|
|
"uv_process",
|
|
"uv_prepare",
|
|
"uv_check",
|
|
"uv_work",
|
|
"uv_fs",
|
|
"uv_fs_event",
|
|
"uv_tty",
|
|
"uv_fs_poll",
|
|
"uv_poll",
|
|
};
|
|
|
|
/* declarations */
|
|
|
|
static inline uv_stream_t* php_uv_get_current_stream(php_uv_t *uv);
|
|
|
|
static void php_uv_fs_cb(uv_fs_t* req);
|
|
/**
|
|
* execute callback
|
|
*
|
|
* @param zval* retval_ptr non-initialized pointer. this will be allocate from zend_call_function
|
|
* @param zval* callback callable object
|
|
* @param zval* params parameters.
|
|
* @param int param_count
|
|
* @return int (maybe..)
|
|
*/
|
|
static int php_uv_do_callback(zval *retval_ptr, zval *callback, zval *params, int param_count TSRMLS_DC);
|
|
|
|
void static destruct_uv(zend_resource *rsrc);
|
|
|
|
static void php_uv_tcp_connect_cb(uv_connect_t *conn_req, int status);
|
|
|
|
static void php_uv_write_cb(uv_write_t* req, int status);
|
|
|
|
static void php_uv_listen_cb(uv_stream_t* server, int status);
|
|
|
|
static void php_uv_close_cb2(uv_handle_t *handle);
|
|
|
|
static void php_uv_shutdown_cb(uv_shutdown_t* req, int status);
|
|
|
|
static void php_uv_read_cb(uv_stream_t* handle, ssize_t nread,const uv_buf_t* buf);
|
|
|
|
static void php_uv_read2_cb(uv_pipe_t* handle, ssize_t nread, uv_buf_t buf, uv_handle_type pending);
|
|
|
|
static void php_uv_read_alloc(uv_handle_t* handle, size_t suggested_size, uv_buf_t* buf);
|
|
|
|
static void php_uv_close_cb(uv_handle_t *handle);
|
|
|
|
static void php_uv_timer_cb(uv_timer_t *handle);
|
|
|
|
static void php_uv_idle_cb(uv_timer_t *handle);
|
|
|
|
static void php_uv_signal_cb(uv_signal_t *handle, int sig_num);
|
|
|
|
|
|
static char *php_uv_map_resource_name(enum php_uv_resource_type type)
|
|
{
|
|
if (php_uv_resource_map[type] != NULL) {
|
|
return php_uv_resource_map[type];
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
#define PHP_UV_TYPE_CHECK(uv, uv_type) \
|
|
if (uv->type != uv_type) { \
|
|
php_error_docref(NULL, E_WARNING, "the passed resource does not initialize for %s resource.", php_uv_map_resource_name(uv_type)); \
|
|
RETURN_FALSE; \
|
|
}
|
|
|
|
static php_socket_t php_uv_zval_to_valid_poll_fd(zval *ptr)
|
|
{
|
|
php_socket_t fd = -1;
|
|
php_stream *stream;
|
|
php_uv_t *uv;
|
|
|
|
/* Validate Checks */
|
|
const char check_php[] = "PHP";
|
|
const char check_file[] = "plainfile";
|
|
|
|
#ifndef PHP_WIN32
|
|
php_socket *socket;
|
|
#endif
|
|
/* TODO: is this correct on windows platform? */
|
|
if (Z_TYPE_P(ptr) == IS_RESOURCE) {
|
|
if (ZEND_FETCH_RESOURCE_NO_RETURN(stream, php_stream *, ptr, -1, NULL, php_file_le_stream())) {
|
|
/* Some streams (specifically STDIO and encrypted streams) can be cast to FDs */
|
|
if (php_stream_cast(stream, PHP_STREAM_AS_FD_FOR_SELECT | PHP_STREAM_CAST_INTERNAL, (void*)&fd, 1) == SUCCESS && fd >= 0) {
|
|
return fd;
|
|
}
|
|
if (stream->wrapper) {
|
|
/* make sure only valid resource streams are passed - plainfiles and most php streams are invalid */
|
|
if (!strcmp((char *)stream->wrapper->wops->label, check_file) || !strcmp((char *)stream->wrapper->wops->label, check_php)) {
|
|
php_error_docref(NULL, E_ERROR, "invalid resource passed, this resource is not supported");
|
|
return -1;
|
|
}
|
|
}
|
|
fd = -1;
|
|
} else if (ZEND_FETCH_RESOURCE_NO_RETURN(uv, php_uv_t*, ptr, -1, NULL, uv_resource_handle)) {
|
|
php_error_docref(NULL, E_WARNING, "uv resource does not support yet");
|
|
fd = -1;
|
|
#ifndef PHP_WIN32
|
|
} else if (php_sockets_le_socket && ZEND_FETCH_RESOURCE_NO_RETURN(socket, php_socket *, ptr, -1, NULL, php_sockets_le_socket())) {
|
|
/* TODO: is this correct on windows platform? */
|
|
fd = socket->bsd_socket;
|
|
#endif
|
|
} else {
|
|
php_error_docref(NULL, E_WARNING, "unhandled resource type detected.");
|
|
fd = -1;
|
|
}
|
|
}
|
|
|
|
return fd;
|
|
}
|
|
|
|
static php_socket_t php_uv_zval_to_fd(zval *ptr)
|
|
{
|
|
php_socket_t fd = -1;
|
|
php_stream *stream;
|
|
php_uv_t *uv;
|
|
#ifndef PHP_WIN32
|
|
php_socket *socket;
|
|
#endif
|
|
/* TODO: is this correct on windows platform? */
|
|
if (Z_TYPE_P(ptr) == IS_RESOURCE) {
|
|
if (ZEND_FETCH_RESOURCE_NO_RETURN(stream, php_stream *, ptr, -1, NULL, php_file_le_stream())) {
|
|
if (php_stream_cast(stream, PHP_STREAM_AS_FD | PHP_STREAM_CAST_INTERNAL, (void*)&fd, 1) != SUCCESS || fd < 0) {
|
|
fd = -1;
|
|
}
|
|
} else if (ZEND_FETCH_RESOURCE_NO_RETURN(uv, php_uv_t*, ptr, -1, NULL, uv_resource_handle)) {
|
|
php_error_docref(NULL, E_WARNING, "uv resource does not support yet");
|
|
fd = -1;
|
|
#ifndef PHP_WIN32
|
|
} else if (php_sockets_le_socket && ZEND_FETCH_RESOURCE_NO_RETURN(socket, php_socket *, ptr, -1, NULL, php_sockets_le_socket())) {
|
|
/* TODO: is this correct on windows platform? */
|
|
fd = socket->bsd_socket;
|
|
#endif
|
|
} else {
|
|
php_error_docref(NULL, E_WARNING, "unhandled resource type detected.");
|
|
fd = -1;
|
|
}
|
|
} else if (Z_TYPE_P(ptr) == IS_LONG) {
|
|
fd = Z_LVAL_P(ptr);
|
|
if (fd < 0) {
|
|
fd = -1;
|
|
}
|
|
|
|
{
|
|
/* make sure that a valid resource handle was passed - issue #36 */
|
|
int err = uv_guess_handle((uv_file) fd);
|
|
if (err == UV_UNKNOWN_HANDLE) {
|
|
php_error_docref(NULL, E_WARNING, "invalid resource type detected");
|
|
fd = -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
return fd;
|
|
}
|
|
|
|
static const char* php_uv_strerror(long error_code)
|
|
{
|
|
/* Note: uv_strerror doesn't use assert. we don't need check value here */
|
|
return uv_strerror(error_code);
|
|
}
|
|
|
|
/**
|
|
* common uv initializer.
|
|
*
|
|
* @param php_uv_t** result this expects non allocated pointer.
|
|
* @param uv_loop_t* loop
|
|
* @param enum php_uv_resource_type type
|
|
* @param zval* return_value|NULL register as a uv_resource when return_value is not null.
|
|
* @return int error
|
|
*/
|
|
static inline int php_uv_common_init(php_uv_t **result, uv_loop_t *loop, enum php_uv_resource_type type, zval *return_value)
|
|
{
|
|
php_uv_t *uv;
|
|
int r = 0;
|
|
|
|
uv = (php_uv_t *)emalloc(sizeof(php_uv_t));
|
|
if (!uv) {
|
|
php_error_docref(NULL, E_ERROR, "php_uv_common_init: emalloc failed");
|
|
goto cleanup;
|
|
}
|
|
|
|
uv->type = type;
|
|
switch (uv->type) {
|
|
case IS_UV_TCP:
|
|
{
|
|
r = uv_tcp_init(loop, &uv->uv.tcp);
|
|
if (r) {
|
|
php_error_docref(NULL, E_ERROR, "uv_tcp_init failed");
|
|
goto cleanup;
|
|
}
|
|
|
|
uv->uv.tcp.data = uv;
|
|
}
|
|
break;
|
|
case IS_UV_IDLE:
|
|
{
|
|
r = uv_idle_init(loop, &uv->uv.idle);
|
|
if (r) {
|
|
php_error_docref(NULL, E_ERROR, "uv_idle_init failed");
|
|
goto cleanup;
|
|
}
|
|
|
|
uv->uv.idle.data = uv;
|
|
}
|
|
break;
|
|
case IS_UV_UDP:
|
|
{
|
|
r = uv_udp_init(loop, &uv->uv.udp);
|
|
if (r) {
|
|
php_error_docref(NULL, E_ERROR, "uv_udp_init failed");
|
|
goto cleanup;
|
|
}
|
|
|
|
uv->uv.udp.data = uv;
|
|
}
|
|
break;
|
|
case IS_UV_PREPARE:
|
|
{
|
|
r = uv_prepare_init(loop, &uv->uv.prepare);
|
|
if (r) {
|
|
php_error_docref(NULL, E_ERROR, "uv_prepare_init failed");
|
|
goto cleanup;
|
|
}
|
|
|
|
uv->uv.prepare.data = uv;
|
|
}
|
|
break;
|
|
case IS_UV_CHECK:
|
|
{
|
|
r = uv_check_init(loop, &uv->uv.check);
|
|
if (r) {
|
|
php_error_docref(NULL, E_ERROR, "uv_check_init failed");
|
|
goto cleanup;
|
|
}
|
|
|
|
uv->uv.check.data = uv;
|
|
}
|
|
break;
|
|
default:
|
|
php_error_docref(NULL, E_ERROR, "unexpected type");
|
|
goto cleanup;
|
|
}
|
|
|
|
PHP_UV_INIT_ZVALS(uv)
|
|
TSRMLS_SET_CTX(uv->thread_ctx);
|
|
|
|
if (return_value != NULL) {
|
|
ZEND_REGISTER_RESOURCE(return_value, uv, uv_resource_handle);
|
|
uv->resource_id = Z_RES_P(return_value);
|
|
}
|
|
|
|
*result = uv;
|
|
return r;
|
|
|
|
cleanup:
|
|
efree(uv);
|
|
return r;
|
|
}
|
|
|
|
static void php_uv_cb_init(php_uv_cb_t **result, php_uv_t *uv, zend_fcall_info *fci, zend_fcall_info_cache *fcc, enum php_uv_callback_type type)
|
|
{
|
|
php_uv_cb_t *cb;
|
|
|
|
if (uv->callback[type] == NULL) {
|
|
cb = emalloc(sizeof(php_uv_cb_t));
|
|
} else {
|
|
cb = uv->callback[type];
|
|
|
|
if (Z_TYPE(cb->fci.function_name) != IS_UNDEF) {
|
|
zval_dtor(&cb->fci.function_name);
|
|
}
|
|
if (fci->object) {
|
|
OBJ_RELEASE(fci->object);
|
|
}
|
|
}
|
|
|
|
memcpy(&cb->fci, fci, sizeof(zend_fcall_info));
|
|
memcpy(&cb->fcc, fcc, sizeof(zend_fcall_info_cache));
|
|
|
|
if (ZEND_FCI_INITIALIZED(*fci)) {
|
|
Z_TRY_ADDREF(cb->fci.function_name);
|
|
if (fci->object) {
|
|
GC_REFCOUNT(cb->fci.object)++;
|
|
}
|
|
}
|
|
|
|
uv->callback[type] = cb;
|
|
}
|
|
|
|
static void php_uv_lock_init(enum php_uv_lock_type lock_type, INTERNAL_FUNCTION_PARAMETERS)
|
|
{
|
|
php_uv_lock_t *lock;
|
|
int error;
|
|
|
|
switch (lock_type) {
|
|
case IS_UV_RWLOCK:
|
|
case IS_UV_RWLOCK_WR:
|
|
case IS_UV_RWLOCK_RD:
|
|
{
|
|
PHP_UV_INIT_LOCK(lock, IS_UV_RWLOCK);
|
|
error = uv_rwlock_init(PHP_UV_LOCK_RWLOCK_P(lock));
|
|
}
|
|
break;
|
|
case IS_UV_MUTEX:
|
|
{
|
|
PHP_UV_INIT_LOCK(lock, IS_UV_MUTEX);
|
|
error = uv_mutex_init(PHP_UV_LOCK_MUTEX_P(lock));
|
|
}
|
|
break;
|
|
case IS_UV_SEMAPHORE:
|
|
{
|
|
unsigned long val = 0;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"l", &val) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_INIT_LOCK(lock, IS_UV_SEMAPHORE);
|
|
error = uv_sem_init(PHP_UV_LOCK_SEM_P(lock), val);
|
|
}
|
|
break;
|
|
default:
|
|
php_error_docref(NULL, E_ERROR, "unexpected type");
|
|
break;
|
|
}
|
|
|
|
if (error == 0) {
|
|
ZEND_REGISTER_RESOURCE(return_value, lock, uv_lock_handle);
|
|
} else {
|
|
efree(lock);
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
|
|
static void php_uv_lock_lock(enum php_uv_lock_type lock_type, INTERNAL_FUNCTION_PARAMETERS)
|
|
{
|
|
php_uv_lock_t *lock;
|
|
zval *handle;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &handle) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(lock, php_uv_lock_t *, handle, -1, PHP_UV_LOCK_RESOURCE_NAME, uv_lock_handle);
|
|
|
|
switch (lock_type) {
|
|
case IS_UV_RWLOCK:
|
|
case IS_UV_RWLOCK_RD:
|
|
{
|
|
uv_rwlock_rdlock(PHP_UV_LOCK_RWLOCK_P(lock));
|
|
lock->locked = 0x01;
|
|
}
|
|
break;
|
|
case IS_UV_RWLOCK_WR:
|
|
{
|
|
uv_rwlock_wrlock(PHP_UV_LOCK_RWLOCK_P(lock));
|
|
lock->locked = 0x02;
|
|
}
|
|
break;
|
|
case IS_UV_MUTEX:
|
|
{
|
|
uv_mutex_lock(PHP_UV_LOCK_MUTEX_P(lock));
|
|
lock->locked = 0x01;
|
|
}
|
|
break;
|
|
case IS_UV_SEMAPHORE:
|
|
{
|
|
uv_sem_post(PHP_UV_LOCK_SEM_P(lock));
|
|
}
|
|
break;
|
|
default:
|
|
php_error_docref(NULL, E_ERROR, "unexpected type");
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void php_uv_lock_unlock(enum php_uv_lock_type lock_type, INTERNAL_FUNCTION_PARAMETERS)
|
|
{
|
|
php_uv_lock_t *lock;
|
|
zval *handle;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &handle) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(lock, php_uv_lock_t *, handle, -1, PHP_UV_LOCK_RESOURCE_NAME, uv_lock_handle);
|
|
|
|
switch (lock_type) {
|
|
case IS_UV_RWLOCK:
|
|
case IS_UV_RWLOCK_RD:
|
|
{
|
|
if (lock->locked == 0x01) {
|
|
uv_rwlock_rdunlock(PHP_UV_LOCK_RWLOCK_P(lock));
|
|
lock->locked = 0x00;
|
|
}
|
|
}
|
|
break;
|
|
case IS_UV_RWLOCK_WR:
|
|
{
|
|
if (lock->locked == 0x02) {
|
|
uv_rwlock_wrunlock(PHP_UV_LOCK_RWLOCK_P(lock));
|
|
lock->locked = 0x00;
|
|
}
|
|
}
|
|
break;
|
|
case IS_UV_MUTEX:
|
|
{
|
|
if (lock->locked == 0x01) {
|
|
uv_mutex_unlock(PHP_UV_LOCK_MUTEX_P(lock));
|
|
lock->locked = 0x00;
|
|
}
|
|
}
|
|
break;
|
|
case IS_UV_SEMAPHORE:
|
|
{
|
|
uv_sem_wait(PHP_UV_LOCK_SEM_P(lock));
|
|
}
|
|
break;
|
|
default:
|
|
php_error_docref(NULL, E_ERROR, "unexpected type");
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void php_uv_lock_trylock(enum php_uv_lock_type lock_type, INTERNAL_FUNCTION_PARAMETERS)
|
|
{
|
|
php_uv_lock_t *lock;
|
|
zval *handle;
|
|
int error = 0;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &handle) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(lock, php_uv_lock_t *, handle, -1, PHP_UV_LOCK_RESOURCE_NAME, uv_lock_handle);
|
|
|
|
switch(lock_type) {
|
|
case IS_UV_RWLOCK:
|
|
case IS_UV_RWLOCK_RD:
|
|
{
|
|
error = uv_rwlock_tryrdlock(PHP_UV_LOCK_RWLOCK_P(lock));
|
|
if (error == 0) {
|
|
lock->locked = 0x01;
|
|
RETURN_TRUE;
|
|
} else {
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
break;
|
|
case IS_UV_RWLOCK_WR:
|
|
{
|
|
error = uv_rwlock_trywrlock(PHP_UV_LOCK_RWLOCK_P(lock));
|
|
if (error == 0) {
|
|
lock->locked = 0x02;
|
|
RETURN_TRUE;
|
|
} else {
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
break;
|
|
case IS_UV_MUTEX:
|
|
{
|
|
error = uv_mutex_trylock(PHP_UV_LOCK_MUTEX_P(lock));
|
|
|
|
if (error == 0) {
|
|
lock->locked = 0x01;
|
|
RETURN_TRUE;
|
|
} else {
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
break;
|
|
case IS_UV_SEMAPHORE:
|
|
{
|
|
error = uv_sem_trywait(PHP_UV_LOCK_SEM_P(lock));
|
|
RETURN_LONG(error);
|
|
}
|
|
default:
|
|
php_error_docref(NULL, E_ERROR, "unexpected type");
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
static void php_uv_fs_common(uv_fs_type fs_type, INTERNAL_FUNCTION_PARAMETERS)
|
|
{
|
|
int error = 0;
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
php_uv_t *uv;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
#define PHP_UV_FS_PARSE_PARAMETERS(specs, ...) \
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(), \
|
|
specs, __VA_ARGS__) == FAILURE) { \
|
|
return; \
|
|
} \
|
|
|
|
#define PHP_UV_FS_SETUP() \
|
|
PHP_UV_INIT_UV(uv, IS_UV_FS); \
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop); \
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_FS_CB); \
|
|
uv->uv.fs.data = uv; \
|
|
|
|
#define PHP_UV_FS_SETUP_AND_EXECUTE(command, ...) \
|
|
PHP_UV_FS_SETUP(); \
|
|
PHP_UV_FS_ASYNC(loop, command, __VA_ARGS__); \
|
|
|
|
switch (fs_type) {
|
|
case UV_FS_SYMLINK:
|
|
{
|
|
zend_string *from, *to;
|
|
long flags;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rSSlf", &zloop, &from, &to, &flags, &fci, &fcc);
|
|
PHP_UV_FS_SETUP_AND_EXECUTE(symlink, from->val, to->val, flags);
|
|
break;
|
|
}
|
|
case UV_FS_LINK:
|
|
{
|
|
zend_string *from, *to;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rSSf", &zloop, &from, &to, &fci, &fcc);
|
|
PHP_UV_FS_SETUP_AND_EXECUTE(link, from->val, to->val);
|
|
break;
|
|
}
|
|
case UV_FS_CHMOD:
|
|
{
|
|
long mode;
|
|
zend_string *path;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rSlf", &zloop, &path, &mode, &fci, &fcc);
|
|
PHP_UV_FS_SETUP_AND_EXECUTE(chmod, path->val, mode);
|
|
break;
|
|
}
|
|
case UV_FS_FCHMOD:
|
|
{
|
|
zval *zstream = NULL;
|
|
long mode;
|
|
unsigned long fd;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rrllf!", &zloop, &zstream, &mode, &fci, &fcc);
|
|
PHP_UV_ZVAL_TO_FD(fd, zstream);
|
|
PHP_UV_FS_SETUP_AND_EXECUTE(fchmod, fd, mode);
|
|
break;
|
|
}
|
|
case UV_FS_RENAME:
|
|
{
|
|
zend_string *from, *to;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rSSf!", &zloop, &from, &to, &fci, &fcc);
|
|
PHP_UV_FS_SETUP_AND_EXECUTE(rename, from->val, to->val);
|
|
break;
|
|
}
|
|
case UV_FS_UNLINK:
|
|
{
|
|
zend_string *path;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rSf!", &zloop, &path, &fci, &fcc);
|
|
PHP_UV_FS_SETUP_AND_EXECUTE(unlink, path->val);
|
|
break;
|
|
}
|
|
case UV_FS_RMDIR:
|
|
{
|
|
zend_string *path;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rSf!", &zloop, &path, &fci, &fcc);
|
|
PHP_UV_FS_SETUP_AND_EXECUTE(rmdir, path->val);
|
|
break;
|
|
}
|
|
case UV_FS_MKDIR:
|
|
{
|
|
zend_string *path;
|
|
long mode = 0;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rSlf", &zloop, &path, &mode, &fci, &fcc);
|
|
PHP_UV_FS_SETUP_AND_EXECUTE(mkdir, path->val, mode);
|
|
break;
|
|
}
|
|
case UV_FS_FTRUNCATE:
|
|
{
|
|
zval *zstream = NULL;
|
|
long offset = 0;
|
|
unsigned long fd;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rrlf", &zloop, &zstream, &offset, &fci, &fcc);
|
|
PHP_UV_FS_SETUP()
|
|
PHP_UV_ZVAL_TO_FD(fd, zstream);
|
|
PHP_UV_FS_ASYNC(loop, ftruncate, fd, offset);
|
|
break;
|
|
}
|
|
case UV_FS_FDATASYNC:
|
|
{
|
|
zval *zstream = NULL;
|
|
unsigned long fd;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rrf", &zloop, &zstream, &fci, &fcc);
|
|
PHP_UV_FS_SETUP()
|
|
PHP_UV_ZVAL_TO_FD(fd, zstream);
|
|
PHP_UV_FS_ASYNC(loop, fdatasync, fd);
|
|
break;
|
|
}
|
|
case UV_FS_FSYNC:
|
|
{
|
|
zval *zstream = NULL;
|
|
unsigned long fd;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rrf!", &zloop, &zstream, &fci, &fcc);
|
|
PHP_UV_FS_SETUP()
|
|
PHP_UV_ZVAL_TO_FD(fd, zstream);
|
|
PHP_UV_FS_ASYNC(loop, fsync, fd);
|
|
break;
|
|
}
|
|
case UV_FS_CLOSE:
|
|
{
|
|
zval *zstream = NULL;
|
|
unsigned long fd;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rrf", &zloop, &zstream, &fci, &fcc);
|
|
PHP_UV_FS_SETUP()
|
|
PHP_UV_ZVAL_TO_FD(fd, zstream);
|
|
PHP_UV_FS_ASYNC(loop, close, fd);
|
|
break;
|
|
}
|
|
case UV_FS_CHOWN:
|
|
{
|
|
long uid, gid;
|
|
zend_string *path;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rSlf", &zloop, &path, &uid, &gid, &fci, &fcc);
|
|
PHP_UV_FS_SETUP_AND_EXECUTE(chown, path->val, uid, gid);
|
|
break;
|
|
}
|
|
case UV_FS_FCHOWN:
|
|
{
|
|
zval *zstream = NULL;
|
|
long uid, gid;
|
|
unsigned long fd;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rrllf!", &zloop, &zstream, &uid, &gid, &fci, &fcc);
|
|
PHP_UV_FS_SETUP()
|
|
PHP_UV_ZVAL_TO_FD(fd, zstream);
|
|
PHP_UV_FS_ASYNC(loop, fchown, fd, uid, gid);
|
|
break;
|
|
}
|
|
case UV_FS_OPEN:
|
|
{
|
|
zend_string *path;
|
|
long flag, mode;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rSllf!", &zloop, &path, &flag, &mode, &fci, &fcc);
|
|
PHP_UV_FS_SETUP_AND_EXECUTE(open, path->val, flag, mode);
|
|
break;
|
|
}
|
|
case UV_FS_SCANDIR:
|
|
{
|
|
zend_string *path;
|
|
long flags;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rSlf!", &zloop, &path, &flags, &fci, &fcc);
|
|
PHP_UV_FS_SETUP_AND_EXECUTE(scandir, path->val, flags);
|
|
break;
|
|
}
|
|
case UV_FS_LSTAT:
|
|
{
|
|
zend_string *path;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rSf", &zloop, &path, &fci, &fcc);
|
|
PHP_UV_FS_SETUP_AND_EXECUTE(lstat, path->val);
|
|
break;
|
|
}
|
|
case UV_FS_FSTAT:
|
|
{
|
|
zval *zstream = NULL;
|
|
unsigned long fd;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rrf", &zloop, &zstream, &fci, &fcc);
|
|
PHP_UV_FS_SETUP()
|
|
PHP_UV_ZVAL_TO_FD(fd, zstream);
|
|
PHP_UV_FS_ASYNC(loop, fstat, fd);
|
|
break;
|
|
}
|
|
case UV_FS_STAT:
|
|
{
|
|
zend_string *path;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rSf", &zloop, &path, &fci, &fcc);
|
|
PHP_UV_FS_SETUP_AND_EXECUTE(stat, path->val);
|
|
break;
|
|
}
|
|
case UV_FS_UTIME:
|
|
{
|
|
long utime, atime;
|
|
zend_string *path;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rSllf", &zloop, &path, &utime, &atime, &fci, &fcc);
|
|
PHP_UV_FS_SETUP_AND_EXECUTE(utime, path->val, utime, atime);
|
|
break;
|
|
}
|
|
case UV_FS_FUTIME:
|
|
{
|
|
zval *zstream = NULL;
|
|
long utime, atime;
|
|
unsigned long fd;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rrllf", &zloop, &zstream, &utime, &atime, &fci, &fcc);
|
|
PHP_UV_FS_SETUP()
|
|
PHP_UV_ZVAL_TO_FD(fd, zstream);
|
|
PHP_UV_FS_ASYNC(loop, futime, fd, utime, atime);
|
|
break;
|
|
}
|
|
case UV_FS_READLINK:
|
|
{
|
|
zend_string *path;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rSf", &zloop, &path, &fci, &fcc);
|
|
PHP_UV_FS_SETUP_AND_EXECUTE(readlink, path->val);
|
|
break;
|
|
}
|
|
case UV_FS_READ:
|
|
{
|
|
zval *zstream = NULL;
|
|
unsigned long fd;
|
|
long length;
|
|
long offset;
|
|
uv_buf_t buf;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rrllf", &zloop, &zstream, &offset, &length, &fci, &fcc);
|
|
if (length <= 0) {
|
|
length = -1;
|
|
}
|
|
if (offset < 0) {
|
|
offset = 0;
|
|
}
|
|
PHP_UV_FS_SETUP()
|
|
PHP_UV_ZVAL_TO_FD(fd, zstream);
|
|
|
|
uv->buffer = (char*) emalloc(length+1);
|
|
buf = uv_buf_init(uv->buffer, length+1);
|
|
|
|
PHP_UV_FS_ASYNC(loop, read, fd, &buf, 1, offset);
|
|
break;
|
|
}
|
|
case UV_FS_SENDFILE:
|
|
{
|
|
zval *z_instream, *z_outstream = NULL;
|
|
unsigned long in_fd, out_fd;
|
|
long offset, length = 0;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rrrllf!", &zloop, &z_instream, &z_outstream, &offset, &length, &fci, &fcc);
|
|
PHP_UV_FS_SETUP()
|
|
/* TODO */
|
|
PHP_UV_ZVAL_TO_FD(in_fd, z_instream);
|
|
PHP_UV_ZVAL_TO_FD(out_fd, z_outstream);
|
|
PHP_UV_FS_ASYNC(loop, sendfile, in_fd, out_fd, offset, length);
|
|
break;
|
|
}
|
|
case UV_FS_WRITE:
|
|
{
|
|
zval *zstream = NULL;
|
|
zend_string *buffer;
|
|
long fd, offset = -1;
|
|
uv_buf_t uv_fs_write_buf_t;
|
|
|
|
PHP_UV_FS_PARSE_PARAMETERS("rrSlf", &zloop, &zstream, &buffer, &offset, &fci, &fcc);
|
|
PHP_UV_FS_SETUP();
|
|
PHP_UV_ZVAL_TO_FD(fd, zstream);
|
|
uv->buffer = estrndup(buffer->val, buffer->len);
|
|
|
|
/* TODO: is this right?! */
|
|
uv_fs_write_buf_t = uv_buf_init(uv->buffer, buffer->len);
|
|
PHP_UV_FS_ASYNC(loop, write, fd, &uv_fs_write_buf_t, 1, offset);
|
|
break;
|
|
}
|
|
case UV_FS_UNKNOWN:
|
|
case UV_FS_CUSTOM:
|
|
default: {
|
|
php_error_docref(NULL, E_ERROR, "type; %d does not support yet.", fs_type);
|
|
break;
|
|
}
|
|
}
|
|
|
|
#undef PHP_UV_FS_PARSE_PARAMETERS
|
|
#undef PHP_UV_FS_SETUP
|
|
#undef PHP_UV_FS_SETUP_AND_EXECUTE
|
|
|
|
}
|
|
/* util */
|
|
|
|
static zval php_uv_address_to_zval(const struct sockaddr *addr)
|
|
{
|
|
zval tmp = {{0}};
|
|
char ip[INET6_ADDRSTRLEN];
|
|
const struct sockaddr_in *a4;
|
|
const struct sockaddr_in6 *a6;
|
|
int port;
|
|
|
|
array_init(&tmp);
|
|
|
|
switch (addr->sa_family) {
|
|
case AF_INET6:
|
|
{
|
|
a6 = (const struct sockaddr_in6 *)addr;
|
|
uv_inet_ntop(AF_INET, &a6->sin6_addr, ip, sizeof ip);
|
|
port = ntohs(a6->sin6_port);
|
|
|
|
add_assoc_string_ex(&tmp, ZEND_STRL("address"), ip);
|
|
add_assoc_long_ex(&tmp, ZEND_STRL("port"), port);
|
|
add_assoc_string_ex(&tmp, ZEND_STRL("family"), "IPv6");
|
|
break;
|
|
}
|
|
case AF_INET:
|
|
{
|
|
a4 = (const struct sockaddr_in *)addr;
|
|
uv_inet_ntop(AF_INET, &a4->sin_addr, ip, sizeof ip);
|
|
port = ntohs(a4->sin_port);
|
|
|
|
add_assoc_string_ex(&tmp, ZEND_STRL("address"), ip);
|
|
add_assoc_long_ex(&tmp, ZEND_STRL("port"), port);
|
|
add_assoc_string_ex(&tmp, ZEND_STRL("family"), "IPv4");
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return tmp;
|
|
}
|
|
|
|
static zval php_uv_make_stat(const uv_stat_t *s)
|
|
{
|
|
zval tmp = {{0}};
|
|
array_init(&tmp);
|
|
|
|
add_assoc_long_ex(&tmp, ZEND_STRL("dev"), s->st_dev);
|
|
add_assoc_long_ex(&tmp, ZEND_STRL("ino"), s->st_ino);
|
|
add_assoc_long_ex(&tmp, ZEND_STRL("mode"), s->st_mode);
|
|
add_assoc_long_ex(&tmp, ZEND_STRL("link"), s->st_nlink);
|
|
add_assoc_long_ex(&tmp, ZEND_STRL("uid"), s->st_uid);
|
|
add_assoc_long_ex(&tmp, ZEND_STRL("gid"), s->st_gid);
|
|
add_assoc_long_ex(&tmp, ZEND_STRL("rdev"), s->st_rdev);
|
|
add_assoc_long_ex(&tmp, ZEND_STRL("size"), s->st_size);
|
|
|
|
#ifndef PHP_WIN32
|
|
add_assoc_long_ex(&tmp, ZEND_STRL("blksize"), s->st_blksize);
|
|
add_assoc_long_ex(&tmp, ZEND_STRL("blocks"), s->st_blocks);
|
|
#endif
|
|
|
|
add_assoc_long_ex(&tmp, ZEND_STRL("atime"), s->st_atim.tv_sec);
|
|
add_assoc_long_ex(&tmp, ZEND_STRL("mtime"), s->st_mtim.tv_sec);
|
|
add_assoc_long_ex(&tmp, ZEND_STRL("ctime"), s->st_ctim.tv_sec);
|
|
|
|
return tmp;
|
|
}
|
|
|
|
/* destructor */
|
|
|
|
void static destruct_uv_lock(zend_resource *rsrc)
|
|
{
|
|
php_uv_lock_t *lock = (php_uv_lock_t *)rsrc->ptr;
|
|
|
|
if (lock->type == IS_UV_RWLOCK) {
|
|
if (lock->locked == 0x01) {
|
|
php_error_docref(NULL, E_NOTICE, "uv_rwlock: unlocked resoruce detected. force rdunlock resource.");
|
|
uv_rwlock_rdunlock(PHP_UV_LOCK_RWLOCK_P(lock));
|
|
lock->locked = 0x00;
|
|
} else if (lock->locked == 0x02) {
|
|
php_error_docref(NULL, E_NOTICE, "uv_rwlock: unlocked resoruce detected. force wrunlock resource.");
|
|
uv_rwlock_wrunlock(PHP_UV_LOCK_RWLOCK_P(lock));
|
|
lock->locked = 0x00;
|
|
}
|
|
uv_rwlock_destroy(PHP_UV_LOCK_RWLOCK_P(lock));
|
|
} else if (lock->type == IS_UV_MUTEX) {
|
|
if (lock->locked == 0x01) {
|
|
php_error_docref(NULL, E_NOTICE, "uv_mutex: unlocked resoruce detected. force unlock resource.");
|
|
uv_mutex_unlock(PHP_UV_LOCK_MUTEX_P(lock));
|
|
lock->locked = 0x00;
|
|
}
|
|
uv_mutex_destroy(PHP_UV_LOCK_MUTEX_P(lock));
|
|
} else if (lock->type == IS_UV_SEMAPHORE) {
|
|
if (lock->locked == 0x01) {
|
|
php_error_docref(NULL, E_NOTICE, "uv_sem: unlocked resoruce detected. force unlock resource.");
|
|
uv_sem_post(PHP_UV_LOCK_SEM_P(lock));
|
|
lock->locked = 0x00;
|
|
}
|
|
uv_sem_destroy(PHP_UV_LOCK_SEM_P(lock));
|
|
}
|
|
|
|
efree(lock);
|
|
}
|
|
|
|
static void destruct_uv_loop_walk_cb(uv_handle_t* handle, void* arg)
|
|
{
|
|
if (!uv_is_closing(handle)) {
|
|
uv_close(handle, NULL);
|
|
}
|
|
}
|
|
|
|
void static destruct_uv_loop(zend_resource *rsrc)
|
|
{
|
|
uv_loop_t *loop = (uv_loop_t *)rsrc->ptr;
|
|
if (loop != _php_uv_default_loop) {
|
|
/* updated for libuv 1.0 as uv_loop_delete is deprecated */
|
|
uv_walk(loop, destruct_uv_loop_walk_cb, NULL);
|
|
uv_loop_close(loop);
|
|
efree(loop);
|
|
}
|
|
}
|
|
|
|
void static destruct_uv_sockaddr(zend_resource *rsrc)
|
|
{
|
|
php_uv_sockaddr_t *addr = (php_uv_sockaddr_t *)rsrc->ptr;
|
|
efree(addr);
|
|
}
|
|
|
|
static uv_loop_t *php_uv_default_loop()
|
|
{
|
|
if (_php_uv_default_loop == NULL) {
|
|
_php_uv_default_loop = uv_default_loop();
|
|
}
|
|
|
|
return _php_uv_default_loop;
|
|
}
|
|
|
|
void static destruct_uv(zend_resource *rsrc)
|
|
{
|
|
int i = 0;
|
|
php_uv_t *obj = NULL;
|
|
|
|
if (rsrc->ptr == NULL) {
|
|
return;
|
|
}
|
|
|
|
obj = (php_uv_t *)rsrc->ptr;
|
|
if (obj == NULL) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_DEBUG_PRINT("# will be free: (resource_id: %d)\n", obj->resource_id);
|
|
|
|
if (obj->in_free > 0) {
|
|
PHP_UV_DEBUG_PRINT("# resource_id: %d is freeing. prevent double free.\n", obj->resource_id);
|
|
return;
|
|
}
|
|
|
|
obj->in_free = 1;
|
|
|
|
/* for now */
|
|
for (i = 0; i < PHP_UV_CB_MAX; i++) {
|
|
php_uv_cb_t *cb = obj->callback[i];
|
|
if (cb != NULL) {
|
|
zval_dtor(&cb->fci.function_name);
|
|
|
|
if (cb->fci.object != NULL) {
|
|
OBJ_RELEASE(cb->fci.object);
|
|
}
|
|
|
|
efree(cb);
|
|
cb = NULL;
|
|
}
|
|
}
|
|
|
|
if (obj->fs_fd != NULL) {
|
|
zend_list_delete(obj->fs_fd);
|
|
obj->fs_fd = NULL;
|
|
}
|
|
|
|
if (obj->resource_id) {
|
|
obj->resource_id = 0;
|
|
}
|
|
|
|
if (obj != NULL) {
|
|
efree(obj);
|
|
obj = NULL;
|
|
rsrc->ptr = NULL;
|
|
}
|
|
}
|
|
|
|
/* callback */
|
|
|
|
static int php_uv_do_callback(zval *retval_ptr, zval *callback, zval *params, int param_count TSRMLS_DC)
|
|
{
|
|
zend_fcall_info fci;
|
|
zend_fcall_info_cache fcc;
|
|
char *is_callable_error = NULL;
|
|
int error;
|
|
#ifdef ZTS
|
|
void *old = tsrm_set_interpreter_context(tsrm_ls);
|
|
#endif
|
|
|
|
if(zend_fcall_info_init(callback, 0, &fci, &fcc, NULL, &is_callable_error) == SUCCESS) {
|
|
if (is_callable_error) {
|
|
php_error_docref(NULL, E_ERROR, "to be a valid callback");
|
|
}
|
|
}
|
|
fci.retval = retval_ptr;
|
|
fci.params = params;
|
|
fci.param_count = param_count;
|
|
|
|
error = zend_call_function(&fci, &fcc);
|
|
|
|
#ifdef ZTS
|
|
tsrm_set_interpreter_context(old);
|
|
#endif
|
|
|
|
return error;
|
|
}
|
|
|
|
static int php_uv_do_callback2(zval *retval_ptr, php_uv_t *uv, zval *params, int param_count, enum php_uv_callback_type type TSRMLS_DC)
|
|
{
|
|
int error = 0;
|
|
#ifdef ZTS
|
|
void *old = tsrm_set_interpreter_context(tsrm_ls);
|
|
#endif
|
|
|
|
if (ZEND_FCI_INITIALIZED(uv->callback[type]->fci)) {
|
|
uv->callback[type]->fci.params = params;
|
|
uv->callback[type]->fci.retval = retval_ptr;
|
|
uv->callback[type]->fci.param_count = param_count;
|
|
uv->callback[type]->fci.no_separation = 1;
|
|
|
|
if (zend_call_function(&uv->callback[type]->fci, &uv->callback[type]->fcc) != SUCCESS) {
|
|
error = -1;
|
|
}
|
|
} else {
|
|
error = -2;
|
|
}
|
|
|
|
#ifdef ZTS
|
|
tsrm_set_interpreter_context(old);
|
|
#endif
|
|
//zend_fcall_info_args_clear(&uv->callback[type]->fci, 0);
|
|
|
|
return error;
|
|
}
|
|
|
|
#ifdef ZTS
|
|
|
|
static int php_uv_do_callback3(zval *retval_ptr, php_uv_t *uv, zval *params, int param_count, enum php_uv_callback_type type)
|
|
{
|
|
int error = 0;
|
|
// zend_executor_globals *ZEG = NULL;
|
|
void *tsrm_ls, *old;
|
|
zend_op_array *old_rtc;
|
|
|
|
if (ZEND_FCI_INITIALIZED(uv->callback[type]->fci)) {
|
|
tsrm_ls = tsrm_new_interpreter_context();
|
|
old = tsrm_set_interpreter_context(tsrm_ls);
|
|
|
|
PG(expose_php) = 0;
|
|
PG(auto_globals_jit) = 0;
|
|
|
|
php_request_startup();
|
|
// ZEG = UV_EG_ALL(tsrm_ls);
|
|
EG(current_execute_data) = NULL;
|
|
EG(current_module) = phpext_uv_ptr;
|
|
|
|
uv->callback[type]->fci.params = params;
|
|
uv->callback[type]->fci.retval = retval_ptr;
|
|
uv->callback[type]->fci.param_count = param_count;
|
|
uv->callback[type]->fci.no_separation = 1;
|
|
uv->callback[type]->fci.object = NULL;
|
|
uv->callback[type]->fcc.initialized = 1;
|
|
|
|
uv->callback[type]->fcc.calling_scope = NULL;
|
|
uv->callback[type]->fcc.called_scope = NULL;
|
|
uv->callback[type]->fcc.object = NULL;
|
|
|
|
old_rtc = uv->callback[type]->fcc.function_handler->op_array.run_time_cache;
|
|
uv->callback[type]->fcc.function_handler->op_array.run_time_cache = NULL;
|
|
|
|
zend_try {
|
|
if (zend_call_function(&uv->callback[type]->fci, &uv->callback[type]->fcc) != SUCCESS) {
|
|
error = -1;
|
|
}
|
|
} zend_catch {
|
|
error = -1;
|
|
} zend_end_try();
|
|
|
|
{
|
|
zend_op_array *ops = &uv->callback[type]->fcc.function_handler->op_array;
|
|
if (ops->run_time_cache) {
|
|
efree(ops->run_time_cache);
|
|
ops->run_time_cache = NULL;
|
|
}
|
|
ops->run_time_cache = old_rtc;
|
|
}
|
|
|
|
php_request_shutdown(NULL);
|
|
tsrm_set_interpreter_context(old);
|
|
tsrm_free_interpreter_context(tsrm_ls);
|
|
} else {
|
|
error = -2;
|
|
}
|
|
|
|
//zend_fcall_info_args_clear(&uv->callback[type]->fci, 0);
|
|
|
|
return error;
|
|
}
|
|
#else
|
|
static int php_uv_do_callback3(zval *retval_ptr, php_uv_t *uv, zval *params, int param_count, enum php_uv_callback_type type)
|
|
{
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
static void php_uv_tcp_connect_cb(uv_connect_t *req, int status)
|
|
{
|
|
zval retval;
|
|
zval params[2] = {{{0}}};
|
|
php_uv_t *uv = (php_uv_t*)req->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
ZVAL_LONG(¶ms[1], status);
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 2, PHP_UV_CONNECT_CB TSRMLS_CC);
|
|
|
|
zval_dtor(&retval);
|
|
|
|
efree(req);
|
|
zval_dtor(¶ms[0]);
|
|
zval_dtor(¶ms[1]);
|
|
}
|
|
|
|
/* TODO: Not sure how PHP will deal with int64_t */
|
|
static void php_uv_process_close_cb(uv_process_t* process, int64_t exit_status, int term_signal)
|
|
{
|
|
zval retval;
|
|
zval params[3] = {{{0}}};
|
|
php_uv_t *uv = (php_uv_t*)process->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
ZVAL_LONG(¶ms[1], exit_status);
|
|
ZVAL_LONG(¶ms[2], term_signal);
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 3, PHP_UV_PROC_CLOSE_CB TSRMLS_CC);
|
|
|
|
zval_dtor(¶ms[0]);
|
|
zval_dtor(¶ms[1]);
|
|
zval_dtor(¶ms[2]);
|
|
zval_dtor(&retval);
|
|
}
|
|
|
|
|
|
static void php_uv_pipe_connect_cb(uv_connect_t *req, int status)
|
|
{
|
|
zval retval;
|
|
zval params[2] = {{{0}}};
|
|
php_uv_t *uv = (php_uv_t*)req->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
ZVAL_LONG(¶ms[0], status);
|
|
ZVAL_RES(¶ms[1], uv->resource_id);
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 2, PHP_UV_PIPE_CONNECT_CB TSRMLS_CC);
|
|
|
|
zval_dtor(¶ms[0]);
|
|
zval_dtor(¶ms[1]);
|
|
zval_dtor(&retval);
|
|
efree(req);
|
|
}
|
|
|
|
|
|
static void php_uv_walk_cb(uv_handle_t* handle, void* arg)
|
|
{
|
|
/*
|
|
zval retval;
|
|
zval params[2] = {{{0}}};
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
ZVAL_LONG(¶ms[0], status);
|
|
ZVAL_RES(¶ms[1], uv->resource_id);
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 2, PHP_UV_PIPE_CONNECT_CB TSRMLS_CC);
|
|
|
|
zval_dtor(¶ms[0]);
|
|
zval_dtor(¶ms[1]);
|
|
zval_dtor(&retval);
|
|
efree(req);
|
|
*/
|
|
}
|
|
|
|
static void php_uv_write_cb(uv_write_t* req, int status)
|
|
{
|
|
write_req_t* wr = (write_req_t*) req;
|
|
zval retval;
|
|
zval params[2] = {{{0}}};
|
|
php_uv_t *uv = (php_uv_t*)req->handle->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
PHP_UV_DEBUG_PRINT("uv_write_cb: status: %d\n", status);
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
ZVAL_LONG(¶ms[1], status);
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 2, PHP_UV_WRITE_CB TSRMLS_CC);
|
|
|
|
zval_dtor(&retval);
|
|
|
|
zval_dtor(¶ms[0]);
|
|
zval_dtor(¶ms[1]);
|
|
|
|
if (wr->buf.base) {
|
|
efree(wr->buf.base);
|
|
}
|
|
|
|
efree(wr);
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_write_cb, uv->resource_id);
|
|
}
|
|
|
|
static void php_uv_udp_send_cb(uv_udp_send_t* req, int status)
|
|
{
|
|
|
|
send_req_t* wr = (send_req_t*) req;
|
|
zval retval;
|
|
zval params[2] = {{{0}}};
|
|
php_uv_t *uv = (php_uv_t*)req->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
ZVAL_LONG(¶ms[1], status);
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 2, PHP_UV_SEND_CB TSRMLS_CC);
|
|
|
|
zval_dtor(¶ms[0]);
|
|
zval_dtor(¶ms[1]);
|
|
|
|
zval_dtor(&retval);
|
|
|
|
if (wr->buf.base) {
|
|
efree(wr->buf.base);
|
|
}
|
|
efree(wr);
|
|
}
|
|
|
|
static void php_uv_listen_cb(uv_stream_t* server, int status)
|
|
{
|
|
zval retval;
|
|
zval params[2] = {{{0}}};
|
|
php_uv_t *uv = (php_uv_t*)server->data;
|
|
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
ZVAL_LONG(¶ms[1], status);
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 2, PHP_UV_LISTEN_CB TSRMLS_CC);
|
|
|
|
zval_dtor(¶ms[0]);
|
|
zval_dtor(¶ms[1]);
|
|
zval_dtor(&retval);
|
|
}
|
|
|
|
static void php_uv_close_cb2(uv_handle_t *handle)
|
|
{
|
|
php_uv_t *uv = (php_uv_t*)handle->data;
|
|
|
|
#ifdef ZTS
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
void *old = tsrm_set_interpreter_context(tsrm_ls);
|
|
#endif
|
|
|
|
PHP_UV_DEBUG_PRINT("uv_close_cb2:\n");
|
|
|
|
zend_list_delete(uv->resource_id);
|
|
|
|
#ifdef ZTS
|
|
tsrm_set_interpreter_context(old);
|
|
#endif
|
|
|
|
// maybe we can't call efree at here.
|
|
//efree(handle);
|
|
}
|
|
|
|
static void php_uv_shutdown_cb(uv_shutdown_t* handle, int status)
|
|
{
|
|
zval retval;
|
|
zval params[2] = {{{0}}};
|
|
php_uv_t *uv = (php_uv_t*)handle->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
ZVAL_LONG(¶ms[1], status);
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 2, PHP_UV_SHUTDOWN_CB TSRMLS_CC);
|
|
|
|
zval_dtor(&retval);
|
|
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_shutdown_cb, uv->resource_id);
|
|
|
|
zval_dtor(¶ms[0]);
|
|
zval_dtor(¶ms[1]);
|
|
efree(handle);
|
|
}
|
|
|
|
static void php_uv_read_cb(uv_stream_t* handle, ssize_t nread, const uv_buf_t* buf)
|
|
{
|
|
zval retval;
|
|
zval params[3] = {{{0}}};
|
|
php_uv_t *uv = (php_uv_t*)handle->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
PHP_UV_DEBUG_PRINT("uv_read_cb\n");
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
|
|
ZVAL_LONG(¶ms[1], nread);
|
|
if (nread > 0) {
|
|
ZVAL_STRINGL(¶ms[2], buf->base, nread);
|
|
} else {
|
|
ZVAL_NULL(¶ms[2]);
|
|
}
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 3, PHP_UV_READ_CB TSRMLS_CC);
|
|
|
|
zval_dtor(¶ms[0]);
|
|
zval_dtor(¶ms[1]);
|
|
zval_dtor(¶ms[2]);
|
|
|
|
zval_dtor(&retval);
|
|
|
|
if (buf->base) {
|
|
efree(buf->base);
|
|
}
|
|
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_read_cb, uv->resource_id);
|
|
}
|
|
|
|
static void php_uv_read2_cb(uv_pipe_t* handle, ssize_t nread, uv_buf_t buf, uv_handle_type pending)
|
|
{
|
|
zval retval;
|
|
zval params[4] = {{{0}}};
|
|
php_uv_t *uv = (php_uv_t*)handle->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
PHP_UV_DEBUG_PRINT("uv_read2_cb\n");
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
ZVAL_LONG(¶ms[1], nread);
|
|
if (nread > 0) {
|
|
ZVAL_STRINGL(¶ms[2], buf.base,nread);
|
|
} else {
|
|
ZVAL_NULL(¶ms[2]);
|
|
}
|
|
ZVAL_LONG(¶ms[3], pending);
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 4, PHP_UV_READ2_CB TSRMLS_CC);
|
|
|
|
zval_dtor(¶ms[0]);
|
|
zval_dtor(¶ms[1]);
|
|
zval_dtor(¶ms[2]);
|
|
zval_dtor(¶ms[3]);
|
|
|
|
zval_dtor(&retval);
|
|
|
|
if (buf.base) {
|
|
efree(buf.base);
|
|
}
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_read2_cb, uv->resource_id);
|
|
}
|
|
|
|
static void php_uv_prepare_cb(uv_prepare_t* handle)
|
|
{
|
|
zval retval;
|
|
zval params[1];
|
|
php_uv_t *uv = (php_uv_t*)handle->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
PHP_UV_DEBUG_PRINT("prepare_cb\n");
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 1, PHP_UV_PREPARE_CB TSRMLS_CC);
|
|
|
|
zval_dtor(¶ms[0]);
|
|
|
|
zval_dtor(&retval);
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_prepare_cb, uv->resource_id);
|
|
}
|
|
|
|
static void php_uv_check_cb(uv_check_t* handle)
|
|
{
|
|
zval retval;
|
|
zval params[1];
|
|
php_uv_t *uv = (php_uv_t*)handle->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
PHP_UV_DEBUG_PRINT("check_cb\n");
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 1, PHP_UV_CHECK_CB TSRMLS_CC);
|
|
|
|
zval_dtor(¶ms[0]);
|
|
|
|
zval_dtor(&retval);
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_check_cb, uv->resource_id);
|
|
}
|
|
|
|
|
|
static void php_uv_async_cb(uv_async_t* handle)
|
|
{
|
|
zval retval;
|
|
zval params[1];
|
|
php_uv_t *uv = (php_uv_t*)handle->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
PHP_UV_DEBUG_PRINT("async_cb\n");
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 1, PHP_UV_ASYNC_CB TSRMLS_CC);
|
|
|
|
zval_dtor(¶ms[0]);
|
|
|
|
zval_dtor(&retval);
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_async_cb, uv->resource_id);
|
|
}
|
|
|
|
|
|
static void php_uv_work_cb(uv_work_t* req)
|
|
{
|
|
zval retval;
|
|
php_uv_t *uv = (php_uv_t*)req->data;
|
|
|
|
uv = (php_uv_t*)req->data;
|
|
|
|
PHP_UV_DEBUG_PRINT("work_cb\n");
|
|
|
|
php_uv_do_callback3(&retval, uv, NULL, 0, PHP_UV_WORK_CB);
|
|
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_work_cb, uv->resource_id);
|
|
}
|
|
|
|
static void php_uv_after_work_cb(uv_work_t* req, int status)
|
|
{
|
|
zval retval;
|
|
php_uv_t *uv = (php_uv_t*)req->data;
|
|
|
|
uv = (php_uv_t*)req->data;
|
|
|
|
PHP_UV_DEBUG_PRINT("after_work_cb\n");
|
|
|
|
php_uv_do_callback3(&retval, uv, NULL, 0, PHP_UV_AFTER_WORK_CB);
|
|
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_work_cb, uv->resource_id);
|
|
}
|
|
|
|
static void php_uv_fs_cb(uv_fs_t* req)
|
|
{
|
|
zval params[3] = {{{0}}};
|
|
zval retval;
|
|
php_uv_t *uv = (php_uv_t*)req->data;
|
|
int argc = 2, i = 0;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
PHP_UV_DEBUG_PRINT("# php_uv_fs_cb %d\n", uv->resource_id);
|
|
|
|
if (uv->fs_fd != NULL) {
|
|
GC_REFCOUNT(uv->fs_fd)++;
|
|
ZVAL_RES(¶ms[0], uv->fs_fd);
|
|
} else {
|
|
if (uv->uv.fs.result < 0) {
|
|
ZVAL_FALSE(¶ms[0]);
|
|
} else {
|
|
PHP_UV_FD_TO_ZVAL(¶ms[0], uv->uv.fs.result)
|
|
}
|
|
}
|
|
|
|
switch (uv->uv.fs.fs_type) {
|
|
case UV_FS_SYMLINK:
|
|
case UV_FS_LINK:
|
|
case UV_FS_CHMOD:
|
|
case UV_FS_FCHMOD:
|
|
case UV_FS_RENAME:
|
|
case UV_FS_UNLINK:
|
|
case UV_FS_RMDIR:
|
|
case UV_FS_MKDIR:
|
|
case UV_FS_FTRUNCATE:
|
|
case UV_FS_FDATASYNC:
|
|
case UV_FS_FSYNC:
|
|
case UV_FS_CLOSE:
|
|
case UV_FS_CHOWN:
|
|
case UV_FS_FCHOWN:
|
|
argc = 1;
|
|
break;
|
|
case UV_FS_OPEN:
|
|
{
|
|
argc = 1;
|
|
break;
|
|
}
|
|
case UV_FS_SCANDIR:
|
|
{
|
|
uv_dirent_t dent;
|
|
|
|
array_init(¶ms[1]);
|
|
while (UV_EOF != uv_fs_scandir_next(req, &dent)) {
|
|
add_next_index_string(¶ms[1], dent.name);
|
|
}
|
|
break;
|
|
}
|
|
case UV_FS_LSTAT:
|
|
case UV_FS_FSTAT:
|
|
case UV_FS_STAT:
|
|
{
|
|
if (Z_RES(params[0]) && req->ptr != NULL) {
|
|
params[1] = php_uv_make_stat((const uv_stat_t*)req->ptr);
|
|
} else {
|
|
ZVAL_NULL(¶ms[1]);
|
|
}
|
|
break;
|
|
}
|
|
case UV_FS_UTIME:
|
|
case UV_FS_FUTIME:
|
|
argc = 0;
|
|
zval_dtor(¶ms[0]);
|
|
break;
|
|
case UV_FS_READLINK:
|
|
{
|
|
ZVAL_STRING(¶ms[1], req->ptr);
|
|
break;
|
|
}
|
|
case UV_FS_READ:
|
|
{
|
|
argc = 3;
|
|
|
|
if (uv->uv.fs.result > 0) {
|
|
ZVAL_STRINGL(¶ms[2], uv->buffer, uv->uv.fs.result);
|
|
} else {
|
|
ZVAL_NULL(¶ms[2]);
|
|
}
|
|
ZVAL_LONG(¶ms[1], uv->uv.fs.result);
|
|
efree(uv->buffer);
|
|
|
|
break;
|
|
}
|
|
case UV_FS_SENDFILE:
|
|
{
|
|
argc = 2;
|
|
|
|
ZVAL_LONG(¶ms[1], uv->uv.fs.result);
|
|
|
|
break;
|
|
}
|
|
case UV_FS_WRITE:
|
|
{
|
|
argc = 2;
|
|
ZVAL_LONG(¶ms[1], uv->uv.fs.result);
|
|
|
|
efree(uv->buffer);
|
|
break;
|
|
}
|
|
case UV_FS_UNKNOWN:
|
|
case UV_FS_CUSTOM:
|
|
default: {
|
|
php_error_docref(NULL, E_ERROR, "type; %d does not support yet.", uv->uv.fs.fs_type);
|
|
break;
|
|
}
|
|
}
|
|
|
|
php_uv_do_callback2(&retval, uv, params, argc, PHP_UV_FS_CB TSRMLS_CC);
|
|
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_fs_cb, uv->resource_id);
|
|
|
|
zval_dtor(&retval);
|
|
|
|
for (i = 0; i < argc; i++) {
|
|
zval_dtor(¶ms[i]);
|
|
}
|
|
|
|
uv_fs_req_cleanup(req);
|
|
}
|
|
|
|
static void php_uv_fs_event_cb(uv_fs_event_t* req, const char* filename, int events, int status)
|
|
{
|
|
zval params[4] = {{{0}}};
|
|
zval retval;
|
|
php_uv_t *uv = (php_uv_t*)req->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
PHP_UV_DEBUG_PRINT("fs_event_cb: %s, %d\n", filename, status);
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
if (filename) {
|
|
ZVAL_STRING(¶ms[1], filename);
|
|
} else {
|
|
ZVAL_NULL(¶ms[1]);
|
|
}
|
|
ZVAL_LONG(¶ms[2], events);
|
|
ZVAL_LONG(¶ms[3], status);
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 4, PHP_UV_FS_EVENT_CB TSRMLS_CC);
|
|
|
|
zval_dtor(&retval);
|
|
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_fs_event_cb, uv->resource_id);
|
|
zval_dtor(¶ms[0]);
|
|
zval_dtor(¶ms[1]);
|
|
zval_dtor(¶ms[2]);
|
|
zval_dtor(¶ms[3]);
|
|
}
|
|
|
|
static zval php_uv_stat_to_zval(const uv_stat_t *stat)
|
|
{
|
|
zval result = {{0}};
|
|
array_init(&result);
|
|
|
|
add_assoc_long_ex(&result, ZEND_STRL("dev"), stat->st_dev);
|
|
add_assoc_long_ex(&result, ZEND_STRL("ino"), stat->st_ino);
|
|
add_assoc_long_ex(&result, ZEND_STRL("mode"), stat->st_mode);
|
|
add_assoc_long_ex(&result, ZEND_STRL("nlink"), stat->st_nlink);
|
|
add_assoc_long_ex(&result, ZEND_STRL("uid"), stat->st_uid);
|
|
add_assoc_long_ex(&result, ZEND_STRL("gid"), stat->st_gid);
|
|
add_assoc_long_ex(&result, ZEND_STRL("rdev"), stat->st_rdev);
|
|
add_assoc_long_ex(&result, ZEND_STRL("size"), stat->st_size);
|
|
|
|
#ifndef PHP_WIN32
|
|
add_assoc_long_ex(&result, ZEND_STRL("blksize"), stat->st_blksize);
|
|
add_assoc_long_ex(&result, ZEND_STRL("blocks"), stat->st_blocks);
|
|
#endif
|
|
|
|
add_assoc_long_ex(&result, ZEND_STRL("atime"), stat->st_atim.tv_sec);
|
|
add_assoc_long_ex(&result, ZEND_STRL("mtime"), stat->st_mtim.tv_sec);
|
|
add_assoc_long_ex(&result, ZEND_STRL("ctime"), stat->st_ctim.tv_sec);
|
|
|
|
return result;
|
|
}
|
|
|
|
static void php_uv_fs_poll_cb(uv_fs_poll_t* handle, int status, const uv_stat_t* prev, const uv_stat_t* curr)
|
|
{
|
|
zval params[4] = {{{0}}};
|
|
zval retval;
|
|
php_uv_t *uv = (php_uv_t*)handle->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
ZVAL_LONG(¶ms[1], status);
|
|
params[2] = php_uv_stat_to_zval(prev);
|
|
params[3] = php_uv_stat_to_zval(curr);
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 4, PHP_UV_FS_POLL_CB TSRMLS_CC);
|
|
|
|
zval_dtor(¶ms[0]);
|
|
zval_dtor(¶ms[1]);
|
|
zval_dtor(¶ms[2]);
|
|
zval_dtor(¶ms[3]);
|
|
|
|
zval_dtor(&retval);
|
|
}
|
|
|
|
static void php_uv_poll_cb(uv_poll_t* handle, int status, int events)
|
|
{
|
|
zval params[4] = {{{0}}};
|
|
zval retval;
|
|
php_uv_t *uv = (php_uv_t*)handle->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
ZVAL_LONG(¶ms[1], status);
|
|
ZVAL_LONG(¶ms[2], events);
|
|
if (uv->fs_fd != NULL) {
|
|
GC_REFCOUNT(uv->fs_fd)++;
|
|
ZVAL_RES(¶ms[3], uv->fs_fd);
|
|
} else {
|
|
PHP_UV_FD_TO_ZVAL(¶ms[3], uv->sock);
|
|
}
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 4, PHP_UV_POLL_CB TSRMLS_CC);
|
|
|
|
zval_dtor(¶ms[0]);
|
|
zval_dtor(¶ms[1]);
|
|
zval_dtor(¶ms[2]);
|
|
zval_dtor(¶ms[3]);
|
|
|
|
zval_dtor(&retval);
|
|
}
|
|
|
|
|
|
static void php_uv_udp_recv_cb(uv_udp_t* handle, ssize_t nread, const uv_buf_t* buf, const struct sockaddr* addr, unsigned flags)
|
|
{
|
|
/* TODO: is this correctly implmented? */
|
|
zval retval;
|
|
zval params[3] = {{{0}}};
|
|
php_uv_t *uv = (php_uv_t*)handle->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
ZVAL_LONG(¶ms[1], nread);
|
|
ZVAL_STRINGL(¶ms[2], buf->base, nread);
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 3, PHP_UV_RECV_CB TSRMLS_CC);
|
|
|
|
zval_dtor(¶ms[0]);
|
|
zval_dtor(¶ms[1]);
|
|
zval_dtor(¶ms[2]);
|
|
zval_dtor(&retval);
|
|
|
|
if (buf->base) {
|
|
efree(buf->base);
|
|
}
|
|
}
|
|
|
|
static void php_uv_read_alloc(uv_handle_t* handle, size_t suggested_size, uv_buf_t* buf)
|
|
{
|
|
buf->base = emalloc(suggested_size);
|
|
buf->len = suggested_size;
|
|
//return uv_buf_init(emalloc(suggested_size), suggested_size);
|
|
}
|
|
|
|
|
|
static void php_uv_close_cb(uv_handle_t *handle)
|
|
{
|
|
zval retval;
|
|
zval params[1] = {{{0}}};
|
|
|
|
php_uv_t *uv = (php_uv_t*)handle->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 1, PHP_UV_CLOSE_CB TSRMLS_CC);
|
|
zval_dtor(&retval);
|
|
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_close_cb, uv->resource_id);
|
|
zend_list_delete(uv->resource_id);
|
|
|
|
zval_dtor(¶ms[0]); /* call destruct_uv */
|
|
}
|
|
|
|
|
|
static void php_uv_idle_cb(uv_timer_t *handle)
|
|
{
|
|
zval retval;
|
|
zval params[1] = {{{0}}};
|
|
|
|
php_uv_t *uv = (php_uv_t*)handle->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 1, PHP_UV_IDLE_CB TSRMLS_CC);
|
|
|
|
zval_dtor(&retval);
|
|
zval_dtor(¶ms[0]);
|
|
}
|
|
|
|
static void php_uv_getaddrinfo_cb(uv_getaddrinfo_t* handle, int status, struct addrinfo* res)
|
|
{
|
|
zval retval;
|
|
zval params[2] = {{{0}}};
|
|
struct addrinfo *address;
|
|
char ip[INET6_ADDRSTRLEN];
|
|
const char *addr;
|
|
|
|
php_uv_t *uv = (php_uv_t*)handle->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
ZVAL_LONG(¶ms[0], status);
|
|
array_init(¶ms[1]);
|
|
|
|
address = res;
|
|
while (address) {
|
|
if (address->ai_family == AF_INET) {
|
|
const char *c;
|
|
|
|
addr = (char*) &((struct sockaddr_in*) address->ai_addr)->sin_addr;
|
|
c = uv_inet_ntop(address->ai_family, addr, ip, INET6_ADDRSTRLEN);
|
|
add_next_index_string(¶ms[1], c);
|
|
}
|
|
|
|
address = address->ai_next;
|
|
}
|
|
|
|
address = res;
|
|
while (address) {
|
|
if (address->ai_family == AF_INET6) {
|
|
const char *c;
|
|
|
|
addr = (char*) &((struct sockaddr_in6*) address->ai_addr)->sin6_addr;
|
|
c = uv_inet_ntop(address->ai_family, addr, ip, INET6_ADDRSTRLEN);
|
|
add_next_index_string(¶ms[1], c);
|
|
}
|
|
|
|
address = address->ai_next;
|
|
}
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 2, PHP_UV_GETADDR_CB TSRMLS_CC);
|
|
|
|
zval_dtor(&retval);
|
|
zval_dtor(¶ms[0]);
|
|
zval_dtor(¶ms[1]);
|
|
|
|
zend_list_delete(uv->resource_id);
|
|
uv_freeaddrinfo(res);
|
|
}
|
|
|
|
static void php_uv_timer_cb(uv_timer_t *handle)
|
|
{
|
|
zval retval;
|
|
zval params[1] = {{{0}}};
|
|
php_uv_t *uv = (php_uv_t*)handle->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 1, PHP_UV_TIMER_CB TSRMLS_CC);
|
|
|
|
zval_dtor(&retval);
|
|
zval_dtor(¶ms[0]);
|
|
}
|
|
|
|
static void php_uv_signal_cb(uv_signal_t *handle, int sig_num)
|
|
{
|
|
zval retval;
|
|
zval params[2] = {{{0}}};
|
|
php_uv_t *uv = (php_uv_t*)handle->data;
|
|
TSRMLS_FETCH_FROM_CTX(uv->thread_ctx);
|
|
|
|
ZVAL_RES(¶ms[0], uv->resource_id);
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
ZVAL_LONG(¶ms[1], sig_num);
|
|
|
|
php_uv_do_callback2(&retval, uv, params, 2, PHP_UV_SIGNAL_CB TSRMLS_CC);
|
|
|
|
zval_dtor(&retval);
|
|
zval_dtor(¶ms[0]);
|
|
zval_dtor(¶ms[1]);
|
|
}
|
|
|
|
static inline uv_stream_t* php_uv_get_current_stream(php_uv_t *uv)
|
|
{
|
|
uv_stream_t *stream;
|
|
switch(uv->type) {
|
|
case IS_UV_TCP:
|
|
stream = (uv_stream_t*)&uv->uv.tcp;
|
|
break;
|
|
case IS_UV_UDP:
|
|
stream = (uv_stream_t*)&uv->uv.udp;
|
|
break;
|
|
case IS_UV_PIPE:
|
|
stream = (uv_stream_t*)&uv->uv.pipe;
|
|
break;
|
|
case IS_UV_IDLE:
|
|
stream = (uv_stream_t*)&uv->uv.idle;
|
|
break;
|
|
case IS_UV_TIMER:
|
|
stream = (uv_stream_t*)&uv->uv.timer;
|
|
break;
|
|
case IS_UV_ASYNC:
|
|
stream = (uv_stream_t*)&uv->uv.async;
|
|
break;
|
|
case IS_UV_LOOP:
|
|
stream = (uv_stream_t*)&uv->uv.loop;
|
|
break;
|
|
case IS_UV_HANDLE:
|
|
stream = (uv_stream_t*)&uv->uv.handle;
|
|
break;
|
|
case IS_UV_STREAM:
|
|
stream = (uv_stream_t*)&uv->uv.stream;
|
|
break;
|
|
case IS_UV_PROCESS:
|
|
stream = (uv_stream_t*)&uv->uv.process;
|
|
break;
|
|
case IS_UV_PREPARE:
|
|
stream = (uv_stream_t*)&uv->uv.prepare;
|
|
break;
|
|
case IS_UV_CHECK:
|
|
stream = (uv_stream_t*)&uv->uv.check;
|
|
break;
|
|
case IS_UV_FS_EVENT:
|
|
stream = (uv_stream_t*)&uv->uv.fs_event;
|
|
break;
|
|
case IS_UV_FS_POLL:
|
|
stream = (uv_stream_t*)&uv->uv.fs_poll;
|
|
break;
|
|
case IS_UV_POLL:
|
|
stream = (uv_stream_t*)&uv->uv.poll;
|
|
break;
|
|
default: {
|
|
TSRMLS_FETCH();
|
|
php_error_docref(NULL, E_ERROR, "unexpected type found");
|
|
break;
|
|
}
|
|
}
|
|
|
|
return stream;
|
|
}
|
|
|
|
void static destruct_uv_stdio(zend_resource *rsrc)
|
|
{
|
|
php_uv_stdio_t *obj = (php_uv_stdio_t *)rsrc->ptr;
|
|
|
|
if (obj->stream != NULL) {
|
|
zend_list_delete(obj->stream);
|
|
obj->stream = NULL;
|
|
}
|
|
|
|
efree(obj);
|
|
}
|
|
|
|
|
|
/* common functions */
|
|
|
|
static void php_uv_ip_common(int ip_type, INTERNAL_FUNCTION_PARAMETERS)
|
|
{
|
|
zval *address;
|
|
php_uv_sockaddr_t *addr;
|
|
char ip[INET6_ADDRSTRLEN];
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &address) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(addr, php_uv_sockaddr_t *, address, -1, PHP_UV_SOCKADDR_RESOURCE_NAME, uv_sockaddr_handle);
|
|
|
|
if (ip_type == 1) {
|
|
if (!PHP_UV_SOCKADDR_IS_IPV4(addr)) {
|
|
RETURN_FALSE;
|
|
}
|
|
uv_ip4_name(PHP_UV_SOCKADDR_IPV4_P(addr), ip, INET6_ADDRSTRLEN);
|
|
RETVAL_STRING(ip);
|
|
} else if (ip_type == 2) {
|
|
if (!PHP_UV_SOCKADDR_IS_IPV6(addr)) {
|
|
RETURN_FALSE;
|
|
}
|
|
uv_ip6_name(PHP_UV_SOCKADDR_IPV6_P(addr), ip, INET6_ADDRSTRLEN);
|
|
RETVAL_STRING(ip);
|
|
}
|
|
}
|
|
|
|
static void php_uv_socket_bind(enum php_uv_socket_type ip_type, INTERNAL_FUNCTION_PARAMETERS)
|
|
{
|
|
zval *resource, *address;
|
|
php_uv_sockaddr_t *addr;
|
|
php_uv_t *uv;
|
|
long flags = 0;
|
|
int r;
|
|
|
|
if (ip_type & PHP_UV_UDP) {
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rr|l", &resource, &address, &flags) == FAILURE) {
|
|
return;
|
|
}
|
|
} else {
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rr", &resource, &address) == FAILURE) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, resource, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
ZEND_FETCH_RESOURCE(addr, php_uv_sockaddr_t *, address, -1, PHP_UV_SOCKADDR_RESOURCE_NAME, uv_sockaddr_handle);
|
|
|
|
if (ip_type & PHP_UV_TCP && uv->type != IS_UV_TCP) {
|
|
php_error_docref(NULL, E_WARNING, "expects uv_tcp resource");
|
|
RETURN_FALSE;
|
|
} else if (ip_type & PHP_UV_UDP && uv->type != IS_UV_UDP) {
|
|
php_error_docref(NULL, E_WARNING, "expects uv_udp resource");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
if ((ip_type & PHP_UV_TCP_IPV4 || ip_type & PHP_UV_UDP_IPV4) && !PHP_UV_SOCKADDR_IS_IPV4(addr)) {
|
|
php_error_docref(NULL, E_WARNING, "expects uv ipv4 addr resource");
|
|
RETURN_FALSE;
|
|
} else if ((ip_type & PHP_UV_TCP_IPV6 || ip_type & PHP_UV_UDP_IPV6) && !PHP_UV_SOCKADDR_IS_IPV6(addr)) {
|
|
php_error_docref(NULL, E_WARNING, "expects uv ipv6 addr resource");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
switch (ip_type) {
|
|
case PHP_UV_TCP_IPV4:
|
|
r = uv_tcp_bind((uv_tcp_t*)&uv->uv.tcp, (const struct sockaddr*)&PHP_UV_SOCKADDR_IPV4(addr), 0);
|
|
break;
|
|
case PHP_UV_TCP_IPV6:
|
|
r = uv_tcp_bind((uv_tcp_t*)&uv->uv.tcp, (const struct sockaddr*)&PHP_UV_SOCKADDR_IPV6(addr), 0);
|
|
break;
|
|
case PHP_UV_UDP_IPV4:
|
|
r = uv_udp_bind((uv_udp_t*)&uv->uv.udp, (const struct sockaddr*)&PHP_UV_SOCKADDR_IPV4(addr), flags);
|
|
break;
|
|
case PHP_UV_UDP_IPV6:
|
|
r = uv_udp_bind((uv_udp_t*)&uv->uv.udp, (const struct sockaddr*)&PHP_UV_SOCKADDR_IPV6(addr), flags);
|
|
break;
|
|
case PHP_UV_TCP:
|
|
case PHP_UV_UDP:
|
|
default:
|
|
php_error_docref(NULL, E_ERROR, "unhandled type");
|
|
break;
|
|
}
|
|
|
|
if (r) {
|
|
php_error_docref(NULL, E_WARNING, "bind failed");
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
|
|
static void php_uv_socket_getname(int type, INTERNAL_FUNCTION_PARAMETERS)
|
|
{
|
|
php_uv_t *uv;
|
|
zval *handle, result;
|
|
int addr_len;
|
|
struct sockaddr_storage addr;
|
|
addr_len = sizeof(struct sockaddr_storage);
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &handle) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t*, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
|
|
switch (type) {
|
|
case 1:
|
|
if (uv->type != IS_UV_TCP) {
|
|
php_error_docref(NULL, E_WARNING, "passed resource doesn't initialize for uv_tcp");
|
|
RETURN_FALSE;
|
|
}
|
|
uv_tcp_getsockname(&uv->uv.tcp, (struct sockaddr*)&addr, &addr_len);
|
|
break;
|
|
case 2:
|
|
if (uv->type != IS_UV_TCP) {
|
|
php_error_docref(NULL, E_WARNING, "passed resource doesn't initialize for uv_tcp");
|
|
RETURN_FALSE;
|
|
}
|
|
uv_tcp_getpeername(&uv->uv.tcp, (struct sockaddr*)&addr, &addr_len);
|
|
break;
|
|
case 3:
|
|
if (uv->type != IS_UV_UDP) {
|
|
php_error_docref(NULL, E_WARNING, "passed resource doesn't initialize for uv_tcp");
|
|
RETURN_FALSE;
|
|
}
|
|
uv_udp_getsockname(&uv->uv.udp, (struct sockaddr*)&addr, &addr_len);
|
|
break;
|
|
default:
|
|
php_error_docref(NULL, E_ERROR, "unexpected type");
|
|
break;
|
|
}
|
|
|
|
result = php_uv_address_to_zval((struct sockaddr*)&addr);
|
|
RETURN_ZVAL(&result, 0, 1);
|
|
}
|
|
|
|
static void php_uv_udp_send(int type, INTERNAL_FUNCTION_PARAMETERS)
|
|
{
|
|
zval *z_cli,*z_addr;
|
|
zend_string *data;
|
|
php_uv_t *client;
|
|
send_req_t *w;
|
|
php_uv_sockaddr_t *addr;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rSrf!", &z_cli, &data, &z_addr, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(client, php_uv_t *, z_cli, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
ZEND_FETCH_RESOURCE(addr, php_uv_sockaddr_t *, z_addr, -1, PHP_UV_SOCKADDR_RESOURCE_NAME, uv_sockaddr_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(client, IS_UV_UDP);
|
|
|
|
GC_REFCOUNT(client->resource_id)++;
|
|
|
|
PHP_UV_INIT_SEND_REQ(w, client, data->val, data->len);
|
|
php_uv_cb_init(&cb, client, &fci, &fcc, PHP_UV_SEND_CB);
|
|
|
|
if (type == 1) {
|
|
uv_udp_send(&w->req, &client->uv.udp, &w->buf, 1, (const struct sockaddr*)&PHP_UV_SOCKADDR_IPV4(addr), php_uv_udp_send_cb);
|
|
} else if (type == 2) {
|
|
uv_udp_send(&w->req, &client->uv.udp, &w->buf, 1, (const struct sockaddr*)&PHP_UV_SOCKADDR_IPV6(addr), php_uv_udp_send_cb);
|
|
}
|
|
}
|
|
|
|
static void php_uv_tcp_connect(enum php_uv_socket_type type, INTERNAL_FUNCTION_PARAMETERS)
|
|
{
|
|
zval *resource,*address;
|
|
php_uv_t *uv;
|
|
php_uv_sockaddr_t *addr;
|
|
uv_connect_t *req;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rrf", &resource,&address, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, resource, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
ZEND_FETCH_RESOURCE(addr, php_uv_sockaddr_t *, address, -1, PHP_UV_SOCKADDR_RESOURCE_NAME, uv_sockaddr_handle);
|
|
|
|
if (uv->type != IS_UV_TCP) {
|
|
php_error_docref(NULL, E_WARNING, "passed uv resource is not initialized for tcp");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
PHP_UV_INIT_CONNECT(req, uv)
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_CONNECT_CB);
|
|
|
|
if (type == PHP_UV_TCP_IPV4) {
|
|
if (!PHP_UV_SOCKADDR_IS_IPV4(addr)) {
|
|
php_error_docref(NULL, E_WARNING, "passed uv sockaddr resource is not initialized for ipv4");
|
|
goto clean;
|
|
}
|
|
|
|
uv_tcp_connect(req, &uv->uv.tcp, (const struct sockaddr*)&PHP_UV_SOCKADDR_IPV4(addr), php_uv_tcp_connect_cb);
|
|
} else {
|
|
if (!PHP_UV_SOCKADDR_IS_IPV6(addr)) {
|
|
php_error_docref(NULL, E_WARNING, "passed uv sockaddr resource is not initialized for ipv6");
|
|
goto clean;
|
|
}
|
|
|
|
uv_tcp_connect(req, &uv->uv.tcp, (const struct sockaddr*)&PHP_UV_SOCKADDR_IPV6(addr), php_uv_tcp_connect_cb);
|
|
}
|
|
|
|
return;
|
|
|
|
clean:
|
|
/* callback zval will be free'd by uv destructor */
|
|
zend_list_delete(uv->resource_id);
|
|
efree(req);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
/* zend */
|
|
|
|
PHP_MINIT_FUNCTION(uv)
|
|
{
|
|
PHP_UV_PROBE(MINIT);
|
|
php_uv_init();
|
|
|
|
uv_resource_handle = zend_register_list_destructors_ex(destruct_uv, NULL, PHP_UV_RESOURCE_NAME, module_number);
|
|
uv_loop_handle = zend_register_list_destructors_ex(destruct_uv_loop, NULL, PHP_UV_LOOP_RESOURCE_NAME, module_number);
|
|
uv_sockaddr_handle = zend_register_list_destructors_ex(destruct_uv_sockaddr, NULL, PHP_UV_SOCKADDR_RESOURCE_NAME, module_number);
|
|
uv_lock_handle = zend_register_list_destructors_ex(destruct_uv_lock, NULL, PHP_UV_LOCK_RESOURCE_NAME, module_number);
|
|
uv_stdio_handle = zend_register_list_destructors_ex(destruct_uv_stdio, NULL, PHP_UV_STDIO_RESOURCE_NAME, module_number);
|
|
|
|
#ifdef ENABLE_HTTPPARSER
|
|
register_httpparser(module_number);
|
|
#endif
|
|
|
|
#if !defined(PHP_WIN32) && !defined(HAVE_SOCKETS)
|
|
{
|
|
zend_module_entry *sockets;
|
|
if ((sockets = zend_hash_str_find_ptr(&module_registry, ZEND_STRL("sockets")))) {
|
|
if (sockets->handle) { // shared
|
|
php_sockets_le_socket = (int (*)(void)) DL_FETCH_SYMBOL(sockets->handle, "php_sockets_le_socket");
|
|
if (php_sockets_le_socket == NULL) {
|
|
php_sockets_le_socket = (int (*)(void)) DL_FETCH_SYMBOL(sockets->handle, "_php_sockets_le_socket");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
PHP_RSHUTDOWN_FUNCTION(uv)
|
|
{
|
|
return SUCCESS;
|
|
}
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_run_once, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_run, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_stop, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_loop_delete, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_now, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_tcp_connect, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, resource)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_tcp_connect6, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, resource)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_tcp_init, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_listen, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, resource)
|
|
ZEND_ARG_INFO(0, backlog)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_accept, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, server)
|
|
ZEND_ARG_INFO(0, client)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_read_start, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, server)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_read2_start, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, server)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_read_stop, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, server)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_write, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, client)
|
|
ZEND_ARG_INFO(0, data)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_write2, 0, 0, 4)
|
|
ZEND_ARG_INFO(0, client)
|
|
ZEND_ARG_INFO(0, data)
|
|
ZEND_ARG_INFO(0, send)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_strerror, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, error)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_err_name, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, error)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_timer_init, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_idle_stop, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, idle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_again, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, idle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_timer_start, 0, 0, 4)
|
|
ZEND_ARG_INFO(0, timer)
|
|
ZEND_ARG_INFO(0, timeout)
|
|
ZEND_ARG_INFO(0, repeat)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_timer_stop, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, timer)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_timer_again, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, timer)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_timer_set_repeat, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, timer)
|
|
ZEND_ARG_INFO(0, timeout)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_timer_get_repeat, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, timer)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_idle_start, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, timer)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_tcp_bind, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, resource)
|
|
ZEND_ARG_INFO(0, address)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_tcp_bind6, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, resource)
|
|
ZEND_ARG_INFO(0, address)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_shutdown, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, stream)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_close, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, stream)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_idle_init, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_update_time, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_is_active, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_is_readable, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_is_writable, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_walk, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_ARG_INFO(0, opaque)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_guess_handle, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, fd)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_handle_type, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_ref, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_unref, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_tcp_nodelay, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, tcp)
|
|
ZEND_ARG_INFO(0, enabled)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_ip4_addr, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, address)
|
|
ZEND_ARG_INFO(0, port)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_ip6_addr, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, address)
|
|
ZEND_ARG_INFO(0, port)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_udp_init, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_udp_bind, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, resource)
|
|
ZEND_ARG_INFO(0, address)
|
|
ZEND_ARG_INFO(0, flags)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_udp_bind6, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, resource)
|
|
ZEND_ARG_INFO(0, address)
|
|
ZEND_ARG_INFO(0, flags)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_udp_recv_start, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, server)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_udp_recv_stop, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, server)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_udp_set_multicast_loop, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, server)
|
|
ZEND_ARG_INFO(0, enabled)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_udp_set_multicast_ttl, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, server)
|
|
ZEND_ARG_INFO(0, ttl)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_udp_set_broadcast, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, server)
|
|
ZEND_ARG_INFO(0, enabled)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_udp_send, 0, 0, 4)
|
|
ZEND_ARG_INFO(0, server)
|
|
ZEND_ARG_INFO(0, buffer)
|
|
ZEND_ARG_INFO(0, address)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_udp_send6, 0, 0, 4)
|
|
ZEND_ARG_INFO(0, server)
|
|
ZEND_ARG_INFO(0, buffer)
|
|
ZEND_ARG_INFO(0, address)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_pipe_open, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, file)
|
|
ZEND_ARG_INFO(0, pipe)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_pipe_init, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, file)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_pipe_bind, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_ARG_INFO(0, name)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_pipe_connect, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_ARG_INFO(0, name)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_pipe_pending_count, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_pipe_pending_type, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_pipe_pending_instances, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_ARG_INFO(0, count)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
/*
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_spawn, 0, 0, 5)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, command)
|
|
ZEND_ARG_INFO(0, args)
|
|
ZEND_ARG_INFO(0, options)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
*/
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_kill, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, pid)
|
|
ZEND_ARG_INFO(0, signal)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_process_kill, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, process)
|
|
ZEND_ARG_INFO(0, signal)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_chdir, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, dir)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_tty_get_winsize, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, tty)
|
|
ZEND_ARG_INFO(1, width)
|
|
ZEND_ARG_INFO(1, height)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_tty_init, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, fd)
|
|
ZEND_ARG_INFO(0, readable)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_event_init, 0, 0, 4)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, path)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_ARG_INFO(0, flags)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_sendfile, 0, 0, 6)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, in)
|
|
ZEND_ARG_INFO(0, out)
|
|
ZEND_ARG_INFO(0, offset)
|
|
ZEND_ARG_INFO(0, length)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_readdir, 0, 0, 4)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, path)
|
|
ZEND_ARG_INFO(0, flags)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_scandir, 0, 0, 4)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, path)
|
|
ZEND_ARG_INFO(0, flags)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_fstat, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, fd)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_lstat, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, path)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_stat, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, path)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_readlink, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, path)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_symlink, 0, 0, 5)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, from)
|
|
ZEND_ARG_INFO(0, to)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_ARG_INFO(0, flags)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_link, 0, 0, 4)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, from)
|
|
ZEND_ARG_INFO(0, to)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_fchown, 0, 0, 5)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, fd)
|
|
ZEND_ARG_INFO(0, uid)
|
|
ZEND_ARG_INFO(0, gid)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_chown, 0, 0, 5)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, path)
|
|
ZEND_ARG_INFO(0, uid)
|
|
ZEND_ARG_INFO(0, gid)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_fchmod, 0, 0, 4)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, fd)
|
|
ZEND_ARG_INFO(0, mode)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_chmod, 0, 0, 4)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, path)
|
|
ZEND_ARG_INFO(0, mode)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_futime, 0, 0, 5)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, fd)
|
|
ZEND_ARG_INFO(0, utime)
|
|
ZEND_ARG_INFO(0, atime)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_utime, 0, 0, 5)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, path)
|
|
ZEND_ARG_INFO(0, utime)
|
|
ZEND_ARG_INFO(0, atime)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_open, 0, 0, 5)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, path)
|
|
ZEND_ARG_INFO(0, flag)
|
|
ZEND_ARG_INFO(0, mode)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_read, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, fd)
|
|
ZEND_ARG_INFO(0, offset)
|
|
ZEND_ARG_INFO(0, size)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_close, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, fd)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_write, 0, 0, 5)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, fd)
|
|
ZEND_ARG_INFO(0, buffer)
|
|
ZEND_ARG_INFO(0, offset)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_fsync, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, fd)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_fdatasync, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, fd)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_ftruncate, 0, 0, 4)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, fd)
|
|
ZEND_ARG_INFO(0, offset)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_mkdir, 0, 0, 4)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, path)
|
|
ZEND_ARG_INFO(0, mode)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_rmdir, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, path)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_unlink, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, path)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_rename, 0, 0, 4)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, from)
|
|
ZEND_ARG_INFO(0, to)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_rwlock_rdlock, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_rwlock_tryrdlock, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_rwlock_rdunlock, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_rwlock_wrlock, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_rwlock_trywrlock, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_rwlock_wrunlock, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_mutex_lock, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_mutex_trylock, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_mutex_unlock, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_sem_init, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, val)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_sem_post, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, resource)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_sem_wait, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, resource)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_sem_trywait, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, resource)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_prepare_init, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_prepare_start, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_prepare_stop, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_check_init, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_check_start, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_check_stop, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_async_send, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_async_init, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_ares_gethostbyname, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_ARG_INFO(0, name)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_ares_init_options, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, options)
|
|
ZEND_ARG_INFO(0, mask)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_tcp_getsockname, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_tcp_getpeername, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_udp_getsockname, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_udp_set_membership, 0, 0, 4)
|
|
ZEND_ARG_INFO(0, client)
|
|
ZEND_ARG_INFO(0, multicast_addr)
|
|
ZEND_ARG_INFO(0, interface_addr)
|
|
ZEND_ARG_INFO(0, membership)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_ip6_name, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_ip4_name, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_poll_init, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_poll_start, 0, 0, 4)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_ARG_INFO(0, path)
|
|
ZEND_ARG_INFO(0, interval)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_fs_poll_stop, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_poll_init, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, fd)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_poll_init_socket, 0, 0, 2)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_ARG_INFO(0, fd)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_poll_start, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_ARG_INFO(0, events)
|
|
ZEND_ARG_INFO(0, callback)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_poll_stop, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_signal_init, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, loop)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_signal_start, 0, 0, 3)
|
|
ZEND_ARG_INFO(0, sig_handle)
|
|
ZEND_ARG_INFO(0, sig_callback)
|
|
ZEND_ARG_INFO(0, sig_num)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_signal_stop, 0, 0, 1)
|
|
ZEND_ARG_INFO(0, sig_handle)
|
|
ZEND_END_ARG_INFO()
|
|
|
|
/* PHP Functions */
|
|
|
|
/* {{{ proto void uv_unref(resource $uv_t)
|
|
*/
|
|
PHP_FUNCTION(uv_unref)
|
|
{
|
|
zval *handle = NULL;
|
|
php_uv_t *uv;
|
|
uv_loop_t *loop;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &handle) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
if (ZEND_FETCH_RESOURCE_NO_RETURN(loop, uv_loop_t*, handle, -1, NULL, uv_loop_handle)) {
|
|
uv_unref((uv_handle_t *)loop);
|
|
} else if (ZEND_FETCH_RESOURCE_NO_RETURN(uv, php_uv_t*, handle, -1, NULL, uv_resource_handle)) {
|
|
uv_unref((uv_handle_t *)php_uv_get_current_stream(uv));
|
|
} else {
|
|
php_error_docref(NULL, E_ERROR, "passes unexpected resource.");
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string uv_err_name(long $error_code)
|
|
*/
|
|
PHP_FUNCTION(uv_err_name)
|
|
{
|
|
long error_code;
|
|
const char *error_msg;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"l", &error_code) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
if (error_code < UV_ERRNO_MAX || error_code > 0) {
|
|
php_error_docref(NULL, E_NOTICE, "passes unexpected value.");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
error_msg = uv_err_name(error_code);
|
|
|
|
RETVAL_STRING(error_msg);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto string uv_strerror(long $error_code)
|
|
*/
|
|
PHP_FUNCTION(uv_strerror)
|
|
{
|
|
long error_code;
|
|
const char *error_msg;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"l", &error_code) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
error_msg = php_uv_strerror(error_code);
|
|
RETVAL_STRING(error_msg);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_update_time(resource $uv_loop)
|
|
*/
|
|
PHP_FUNCTION(uv_update_time)
|
|
{
|
|
zval *z_loop = NULL;
|
|
uv_loop_t *loop;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &z_loop) == FAILURE) {
|
|
return;
|
|
}
|
|
ZEND_FETCH_RESOURCE(loop, uv_loop_t *, z_loop, -1, PHP_UV_LOOP_RESOURCE_NAME, uv_loop_handle);
|
|
uv_update_time(loop);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_ref(resource $uv_handle)
|
|
*/
|
|
PHP_FUNCTION(uv_ref)
|
|
{
|
|
zval *handle = NULL;
|
|
php_uv_t *uv;
|
|
uv_loop_t *loop;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &handle) == FAILURE) {
|
|
return;
|
|
}
|
|
if (ZEND_FETCH_RESOURCE_NO_RETURN(loop, uv_loop_t*, handle, -1, NULL, uv_loop_handle)) {
|
|
uv_ref((uv_handle_t *)loop);
|
|
zend_list_delete(Z_RES_P(handle));
|
|
} else if (ZEND_FETCH_RESOURCE_NO_RETURN(uv, php_uv_t*, handle, -1, NULL, uv_resource_handle)) {
|
|
uv_ref((uv_handle_t *)php_uv_get_current_stream(uv));
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
} else {
|
|
php_error_docref(NULL, E_ERROR, "passes unexpected resource.");
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_run([resource $uv_loop, long $run_mode])
|
|
*/
|
|
PHP_FUNCTION(uv_run)
|
|
{
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
long run_mode = UV_RUN_DEFAULT;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"|rl", &zloop, &run_mode) == FAILURE) {
|
|
return;
|
|
}
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
uv_run(loop, run_mode);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_stop([resource $uv_loop])
|
|
*/
|
|
PHP_FUNCTION(uv_stop)
|
|
{
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"|r", &zloop) == FAILURE) {
|
|
return;
|
|
}
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
uv_stop(loop);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto resource uv_signal_init([resource $uv_loop])
|
|
*/
|
|
PHP_FUNCTION(uv_signal_init)
|
|
{
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
php_uv_t *uv;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"|r", &zloop) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
PHP_UV_INIT_SIGNAL(uv, IS_UV_SIGNAL)
|
|
|
|
uv->uv.signal.data = uv;
|
|
|
|
ZVAL_RES(return_value, uv->resource_id);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_signal_start(resource $sig_handle, callable $sig_callback, int $sig_num)
|
|
*/
|
|
PHP_FUNCTION(uv_signal_start)
|
|
{
|
|
zval *sig_handle;
|
|
long sig_num;
|
|
php_uv_t *uv;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rfl", &sig_handle, &fci, &fcc, &sig_num) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, sig_handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_SIGNAL);
|
|
|
|
if (uv_is_active((uv_handle_t*)&uv->uv.signal)) {
|
|
php_error_docref(NULL, E_NOTICE, "passed uv signal resource has been started. you don't have to call this method");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_SIGNAL_CB);
|
|
|
|
uv_signal_start((uv_signal_t*)&uv->uv.signal, php_uv_signal_cb, sig_num);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto int uv_signal_stop(resource $sig_handle)
|
|
*/
|
|
PHP_FUNCTION(uv_signal_stop)
|
|
{
|
|
zval *sig_handle;
|
|
php_uv_t *uv;
|
|
int r = 0;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &sig_handle) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, sig_handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_SIGNAL);
|
|
|
|
if (!uv_is_active((uv_handle_t*)&uv->uv.signal)) {
|
|
php_error_docref(NULL, E_NOTICE, "passed uv signal resource has been stopped. you don't have to call this method");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
r = uv_signal_stop((uv_signal_t*)&uv->uv.signal);
|
|
|
|
RETURN_LONG(r);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto long uv_run_once([resource $uv_loop])
|
|
*/
|
|
PHP_FUNCTION(uv_run_once)
|
|
{
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"|r", &zloop) == FAILURE) {
|
|
return;
|
|
}
|
|
php_error_docref(NULL, E_DEPRECATED, "uv_run_once is deprecated; use uv_run(uv_default_loop(), UV::RUN_ONCE) instead");
|
|
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
RETURN_LONG(uv_run(loop, UV_RUN_ONCE));
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_loop_delete(resource $uv_loop)
|
|
*/
|
|
PHP_FUNCTION(uv_loop_delete)
|
|
{
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &zloop) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
|
|
if (loop != _php_uv_default_loop) {
|
|
/* uv_loop_delete deprecated with libuv 1.0 */
|
|
//uv_loop_delete(loop);
|
|
uv_loop_close(loop);
|
|
efree(loop);
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto long uv_now(resource $uv_loop)
|
|
*/
|
|
PHP_FUNCTION(uv_now)
|
|
{
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
int64_t now;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &zloop) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
now = uv_now(loop);
|
|
RETURN_LONG((long)now);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto void uv_tcp_bind(resource $uv_tcp, resource $uv_sockaddr)
|
|
*/
|
|
PHP_FUNCTION(uv_tcp_bind)
|
|
{
|
|
php_uv_socket_bind(PHP_UV_TCP_IPV4, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_tcp_bind6(resource $uv_tcp, resource $uv_sockaddr)
|
|
*/
|
|
PHP_FUNCTION(uv_tcp_bind6)
|
|
{
|
|
php_uv_socket_bind(PHP_UV_TCP_IPV6, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto void uv_write(resource $handle, string $data, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_write)
|
|
{
|
|
zval *z_cli;
|
|
zend_string *data;
|
|
int r;
|
|
php_uv_t *uv;
|
|
write_req_t *w;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rS|f!", &z_cli, &data, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, z_cli, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
if (uv->type != IS_UV_TCP && uv->type != IS_UV_PIPE && uv->type != IS_UV_TTY) {
|
|
php_error_docref(NULL, E_WARNING, "passed resource does not support yet");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_WRITE_CB);
|
|
|
|
PHP_UV_INIT_WRITE_REQ(w, uv, data->val, data->len)
|
|
|
|
r = uv_write(&w->req, (uv_stream_t*)php_uv_get_current_stream(uv), &w->buf, 1, php_uv_write_cb);
|
|
if (r) {
|
|
php_error_docref(NULL, E_WARNING, "write failed");
|
|
}
|
|
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_write, uv->resource_id);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_write2(resource $handle, string $data, resource $send, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_write2)
|
|
{
|
|
zval *z_cli, *z_send;
|
|
zend_string *data;
|
|
int r;
|
|
php_uv_t *uv, *send;
|
|
write_req_t *w;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rSrf", &z_cli, &data, &z_send, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, z_cli, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
if (uv->type != IS_UV_TCP && uv->type != IS_UV_PIPE && uv->type != IS_UV_TTY) {
|
|
php_error_docref(NULL, E_WARNING, "passed resource does not support yet");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(send, php_uv_t *, z_send, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_WRITE_CB);
|
|
PHP_UV_INIT_WRITE_REQ(w, uv, data->val, data->len)
|
|
|
|
r = uv_write2(&w->req, (uv_stream_t*)php_uv_get_current_stream(uv), &w->buf, 1, (uv_stream_t*)php_uv_get_current_stream(send), php_uv_write_cb);
|
|
if (r) {
|
|
php_error_docref(NULL, E_ERROR, "write2 failed");
|
|
}
|
|
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_write2, uv->resource_id);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_tcp_nodelay(resource $handle, bool $enable)
|
|
*/
|
|
PHP_FUNCTION(uv_tcp_nodelay)
|
|
{
|
|
zval *z_cli;
|
|
php_uv_t *client;
|
|
long bval = 1;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rl", &z_cli, &bval) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(client, php_uv_t *, z_cli, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(client, IS_UV_TCP);
|
|
|
|
uv_tcp_nodelay(&client->uv.tcp, bval);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_accept(resource $server, resource $client)
|
|
*/
|
|
PHP_FUNCTION(uv_accept)
|
|
{
|
|
zval *z_svr,*z_cli;
|
|
php_uv_t *server, *client;
|
|
int r;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rr", &z_svr, &z_cli) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(server, php_uv_t *, z_svr, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
ZEND_FETCH_RESOURCE(client, php_uv_t *, z_cli, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
if ((server->type == IS_UV_TCP && client->type != IS_UV_TCP) ||
|
|
(server->type == IS_UV_PIPE && client->type != IS_UV_PIPE)
|
|
) {
|
|
php_error_docref(NULL, E_WARNING, "both resource type should be same.");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
r = uv_accept((uv_stream_t *)php_uv_get_current_stream(server), (uv_stream_t *)php_uv_get_current_stream(client));
|
|
if (r) {
|
|
php_error_docref(NULL, E_WARNING, "%s", php_uv_strerror(r));
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto void uv_shutdown(resource $handle, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_shutdown)
|
|
{
|
|
zval *client = NULL;
|
|
php_uv_t *uv;
|
|
uv_shutdown_t *shutdown;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
int r = 0;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r|f!", &client, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
#ifndef PHP_WIN32
|
|
/* uv_shutdown (unix) only supports uv_handle_t right now */
|
|
if (uv->type != IS_UV_TCP && uv->type != IS_UV_PIPE) {
|
|
php_error_docref(NULL, E_WARNING, "passed resource didn't initialize for uv_tcp or uv_pipe");
|
|
RETURN_FALSE;
|
|
}
|
|
#endif
|
|
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_SHUTDOWN_CB);
|
|
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
shutdown = emalloc(sizeof(uv_shutdown_t));
|
|
shutdown->data = uv;
|
|
|
|
r = uv_shutdown(shutdown, (uv_stream_t*)php_uv_get_current_stream(uv), (uv_shutdown_cb)php_uv_shutdown_cb);
|
|
if (r) {
|
|
php_error_docref(NULL, E_WARNING, "%s", php_uv_strerror(r));
|
|
}
|
|
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_close(resource $handle, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_close)
|
|
{
|
|
zval *client = NULL;
|
|
php_uv_t *uv;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r|f!", &client, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
switch (uv->type) {
|
|
/* TODO: use libuv enum */
|
|
case IS_UV_PIPE:
|
|
case IS_UV_TTY:
|
|
case IS_UV_TCP:
|
|
case IS_UV_UDP:
|
|
case IS_UV_PREPARE:
|
|
case IS_UV_CHECK:
|
|
case IS_UV_IDLE:
|
|
case IS_UV_ASYNC:
|
|
case IS_UV_TIMER:
|
|
case IS_UV_PROCESS:
|
|
case IS_UV_FS_EVENT:
|
|
case IS_UV_POLL:
|
|
case IS_UV_FS_POLL:
|
|
break;
|
|
default:
|
|
php_error_docref(NULL, E_WARNING, "passed resource didn't initialize for uv_close (%d)", uv->type);
|
|
RETURN_FALSE;
|
|
break;
|
|
}
|
|
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_CLOSE_CB);
|
|
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
uv_close((uv_handle_t*)php_uv_get_current_stream(uv), (uv_close_cb)php_uv_close_cb);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_read_start(resource $handle, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_read_start)
|
|
{
|
|
zval *client;
|
|
php_uv_t *uv;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
int r;
|
|
|
|
PHP_UV_DEBUG_PRINT("uv_read_start\n");
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rf!", &client, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
if (uv->type != IS_UV_TCP && uv->type != IS_UV_PIPE && uv->type != IS_UV_TTY) {
|
|
php_error_docref(NULL, E_WARNING, "passed resource didn't intialize for uv_tcp, uv_pipe or uv_tty.");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
|
|
if (uv->type == IS_UV_TCP) {
|
|
uv->uv.tcp.data = uv;
|
|
} else if(uv->type == IS_UV_PIPE) {
|
|
uv->uv.pipe.data = uv;
|
|
} else if (uv->type == IS_UV_TTY) {
|
|
uv->uv.tty.data = uv;
|
|
} else {
|
|
php_error_docref(NULL, E_NOTICE, "this type does not support yet");
|
|
}
|
|
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_READ_CB);
|
|
|
|
r = uv_read_start((uv_stream_t*)php_uv_get_current_stream(uv), php_uv_read_alloc, php_uv_read_cb);
|
|
if (r) {
|
|
php_error_docref(NULL, E_NOTICE, "read failed");
|
|
}
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_read_start, uv->resource_id);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_read2_start(resource $handle, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_read2_start)
|
|
{
|
|
/* TODO: determine how to make this backwards compatible? */
|
|
php_error_docref(NULL, E_ERROR, "uv_read2_start is no longer supported.");
|
|
/*
|
|
zval *client;
|
|
php_uv_t *uv;
|
|
int r;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
PHP_UV_DEBUG_PRINT("uv_read2_start\n");
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rf", &client, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
if (uv->type != IS_UV_TCP && uv->type != IS_UV_PIPE && uv->type != IS_UV_TTY) {
|
|
php_error_docref(NULL, E_WARNING, "passed resource didn't intialize for uv_tcp, uv_pipe or uv_tty.");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
|
|
if(uv->type == IS_UV_TCP) {
|
|
uv->uv.tcp.data = uv;
|
|
} else if(uv->type == IS_UV_PIPE) {
|
|
uv->uv.pipe.data = uv;
|
|
} else if (uv->type == IS_UV_TTY) {
|
|
uv->uv.tty.data = uv;
|
|
} else {
|
|
php_error_docref(NULL, E_NOTICE, "this type does not support yet");
|
|
}
|
|
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_READ2_CB);
|
|
r = uv_read2_start((uv_stream_t*)php_uv_get_current_stream(uv), php_uv_read_alloc, php_uv_read2_cb);
|
|
if (r) {
|
|
php_error_docref(NULL, E_NOTICE, "read2 failed");
|
|
}
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_read2_start, uv->resource_id);
|
|
*/
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_read_stop(resource $handle)
|
|
*/
|
|
PHP_FUNCTION(uv_read_stop)
|
|
{
|
|
zval *server;
|
|
php_uv_t *uv;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &server) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, server, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
if (uv->type != IS_UV_TCP && uv->type != IS_UV_PIPE && uv->type != IS_UV_TTY) {
|
|
php_error_docref(NULL, E_WARNING, "passed resource didn't intialize for uv_tcp, uv_pipe or uv_tty.");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
uv_read_stop((uv_stream_t*)php_uv_get_current_stream(uv));
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_read_stop, uv->resource_id);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto resource uv_ip4_addr(string $ipv4_addr, long $port)
|
|
*/
|
|
PHP_FUNCTION(uv_ip4_addr)
|
|
{
|
|
zend_string *address;
|
|
long port = 0;
|
|
php_uv_sockaddr_t *sockaddr;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"Sl", &address, &port) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_SOCKADDR_INIT(sockaddr, 1);
|
|
//PHP_UV_SOCKADDR_IPV4(sockaddr) = uv_ip4_addr(address, port);
|
|
/* TODO: is this right?! */
|
|
uv_ip4_addr(address->val, port, &PHP_UV_SOCKADDR_IPV4(sockaddr));
|
|
|
|
ZEND_REGISTER_RESOURCE(return_value, sockaddr, uv_sockaddr_handle);
|
|
sockaddr->resource_id = Z_RES_P(return_value);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto resource uv_ip6_addr(string $ipv6_addr, long $port)
|
|
*/
|
|
PHP_FUNCTION(uv_ip6_addr)
|
|
{
|
|
zend_string *address;
|
|
long port = 0;
|
|
php_uv_sockaddr_t *sockaddr;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"Sl", &address, &port) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_SOCKADDR_INIT(sockaddr, 0);
|
|
//PHP_UV_SOCKADDR_IPV6(sockaddr) = uv_ip6_addr(address, port);
|
|
/* TODO: is this right?! */
|
|
uv_ip6_addr(address->val, port, &PHP_UV_SOCKADDR_IPV6(sockaddr));
|
|
|
|
ZEND_REGISTER_RESOURCE(return_value, sockaddr, uv_sockaddr_handle);
|
|
sockaddr->resource_id = Z_RES_P(return_value);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto void uv_listen(resource $handle, long $backlog, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_listen)
|
|
{
|
|
zval *resource;
|
|
long backlog = SOMAXCONN;
|
|
php_uv_t *uv;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
int r;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rlf", &resource, &backlog, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, resource, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
switch (uv->type) {
|
|
case IS_UV_TCP:
|
|
case IS_UV_PIPE:
|
|
break;
|
|
default:
|
|
php_error_docref(NULL, E_WARNING, "expects uv_tcp or uv_pipe resource.");
|
|
RETURN_FALSE;
|
|
break;
|
|
}
|
|
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_LISTEN_CB);
|
|
|
|
r = uv_listen((uv_stream_t*)php_uv_get_current_stream(uv), backlog, php_uv_listen_cb);
|
|
if (r) {
|
|
php_error_docref(NULL, E_WARNING, "%s", php_uv_strerror(r));
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_tcp_connect(resource $handle, resource $ipv4_addr, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_tcp_connect)
|
|
{
|
|
php_uv_tcp_connect(PHP_UV_TCP_IPV4, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto void uv_tcp_connect6(resource $handle, resource $ipv6_addr, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_tcp_connect6)
|
|
{
|
|
php_uv_tcp_connect(PHP_UV_TCP_IPV6, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto resource uv_timer_init([resource $loop])
|
|
*/
|
|
PHP_FUNCTION(uv_timer_init)
|
|
{
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
php_uv_t *uv;
|
|
int r;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"|r", &zloop) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
PHP_UV_INIT_TIMER(uv, IS_UV_TIMER)
|
|
|
|
uv->uv.timer.data = uv;
|
|
PHP_UV_DEBUG_PRINT("uv_timer_init: resource: %d\n", uv->resource_id);
|
|
|
|
ZVAL_RES(return_value, uv->resource_id);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_timer_start(resource $timer, long $timeout, long $repeat, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_timer_start)
|
|
{
|
|
zval *timer;
|
|
php_uv_t *uv;
|
|
long timeout, repeat = 0;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rllf!", &timer, &timeout, &repeat, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, timer, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_TIMER);
|
|
|
|
if (timeout < 0) {
|
|
php_error_docref(NULL, E_WARNING, "timeout value have to be larger than 0. given %ld", timeout);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
if (repeat < 0) {
|
|
php_error_docref(NULL, E_WARNING, "repeat value have to be larger than 0. given %ld", repeat);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
if (uv_is_active((uv_handle_t*)&uv->uv.timer)) {
|
|
php_error_docref(NULL, E_NOTICE, "passed uv timer resource has been started. you don't have to call this method");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_TIMER_CB);
|
|
|
|
uv_timer_start((uv_timer_t*)&uv->uv.timer, php_uv_timer_cb, timeout, repeat);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_timer_stop(resource $timer)
|
|
*/
|
|
PHP_FUNCTION(uv_timer_stop)
|
|
{
|
|
zval *timer;
|
|
php_uv_t *uv;
|
|
int r = 0;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &timer) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, timer, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_TIMER);
|
|
|
|
if (!uv_is_active((uv_handle_t*)&uv->uv.timer)) {
|
|
php_error_docref(NULL, E_NOTICE, "passed uv timer resource has been stopped. you don't have to call this method");
|
|
RETURN_FALSE;
|
|
}
|
|
PHP_UV_DEBUG_PRINT("uv_timer_stop: resource: %d\n", uv->resource_id);
|
|
r = uv_timer_stop((uv_timer_t*)&uv->uv.timer);
|
|
|
|
RETURN_LONG(r);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_timer_again(resource $timer)
|
|
*/
|
|
PHP_FUNCTION(uv_timer_again)
|
|
{
|
|
zval *timer;
|
|
php_uv_t *uv;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &timer) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, timer, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_TIMER);
|
|
|
|
if (uv_is_active((uv_handle_t*)&uv->uv.timer)) {
|
|
php_error_docref(NULL, E_NOTICE, "passed uv timer resource has been started. you don't have to call this method");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
uv_timer_again((uv_timer_t*)&uv->uv.timer);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_timer_set_repeat(resource $timer, long $repeat)
|
|
*/
|
|
PHP_FUNCTION(uv_timer_set_repeat)
|
|
{
|
|
zval *timer;
|
|
php_uv_t *uv;
|
|
long repeat;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rl", &timer,&repeat) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, timer, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_TIMER);
|
|
|
|
uv_timer_set_repeat((uv_timer_t*)&uv->uv.timer,repeat);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto long uv_timer_get_repeat(resource $timer)
|
|
*/
|
|
PHP_FUNCTION(uv_timer_get_repeat)
|
|
{
|
|
zval *timer;
|
|
php_uv_t *uv;
|
|
int64_t repeat;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &timer) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, timer, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_TIMER);
|
|
|
|
repeat = uv_timer_get_repeat((uv_timer_t*)&uv->uv.timer);
|
|
RETURN_LONG(repeat);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto resource uv_idle_init([resource $loop])
|
|
*/
|
|
PHP_FUNCTION(uv_idle_init)
|
|
{
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
php_uv_t *uv;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"|r", &zloop) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
php_uv_common_init(&uv, loop, IS_UV_IDLE, return_value);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_idle_start(resource $idle, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_idle_start)
|
|
{
|
|
zval *idle;
|
|
php_uv_t *uv;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
int r = 0;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rf", &idle, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, idle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_IDLE);
|
|
|
|
if (uv_is_active((uv_handle_t*)&uv->uv.idle)) {
|
|
php_error_docref(NULL, E_WARNING, "passed uv_idle resource has already started.");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_IDLE_CB);
|
|
|
|
r = uv_idle_start((uv_idle_t*)&uv->uv.idle, (uv_idle_cb)php_uv_idle_cb);
|
|
|
|
RETURN_LONG(r);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto void uv_idle_stop(resource $idle)
|
|
*/
|
|
PHP_FUNCTION(uv_idle_stop)
|
|
{
|
|
zval *idle;
|
|
php_uv_t *uv;
|
|
int r = 0;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &idle) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, idle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_IDLE);
|
|
|
|
if (!uv_is_active((uv_handle_t*)&uv->uv.idle)) {
|
|
php_error_docref(NULL, E_NOTICE, "passed uv_idle resource does not start yet.");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
r = uv_idle_stop((uv_idle_t*)&uv->uv.idle);
|
|
RETURN_LONG(r);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto void uv_getaddrinfo(resource $loop, callable $callback, string $node, string $service, array $hints)
|
|
*/
|
|
PHP_FUNCTION(uv_getaddrinfo)
|
|
{
|
|
zval *z_loop, *hints = NULL;
|
|
uv_loop_t *loop;
|
|
php_uv_t *uv = NULL;
|
|
struct addrinfo hint = {0};
|
|
zend_string *node, *service;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rfSS|a", &z_loop, &fci, &fcc, &node, &service, &hints) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(loop, uv_loop_t *, z_loop, -1, PHP_UV_LOOP_RESOURCE_NAME, uv_loop_handle);
|
|
|
|
if (Z_TYPE_P(hints) == IS_ARRAY) {
|
|
HashTable *h;
|
|
zval *data;
|
|
|
|
h = Z_ARRVAL_P(hints);
|
|
if ((data = zend_hash_str_find(h, ZEND_STRL("ai_family")))) {
|
|
hint.ai_family = Z_LVAL_P(data);
|
|
}
|
|
if ((data = zend_hash_str_find(h, ZEND_STRL("ai_socktype")))) {
|
|
hint.ai_socktype = Z_LVAL_P(data);
|
|
}
|
|
if ((data = zend_hash_str_find(h, ZEND_STRL("ai_protocol")))) {
|
|
hint.ai_socktype = Z_LVAL_P(data);
|
|
}
|
|
if ((data = zend_hash_str_find(h, ZEND_STRL("ai_flags")))) {
|
|
hint.ai_flags = Z_LVAL_P(data);
|
|
}
|
|
}
|
|
|
|
uv = (php_uv_t *)emalloc(sizeof(php_uv_t));
|
|
PHP_UV_INIT_ZVALS(uv)
|
|
TSRMLS_SET_CTX(uv->thread_ctx);
|
|
uv->uv.addrinfo.data = uv;
|
|
uv->resource_id = PHP_UV_LIST_INSERT(uv, uv_resource_handle);
|
|
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_GETADDR_CB);
|
|
uv_getaddrinfo(loop, &uv->uv.addrinfo, php_uv_getaddrinfo_cb, node->val, service->val, &hint);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto resource uv_tcp_init([resource $loop])
|
|
*/
|
|
PHP_FUNCTION(uv_tcp_init)
|
|
{
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
php_uv_t *uv;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"|r", &zloop) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
php_uv_common_init(&uv, loop, IS_UV_TCP, return_value);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto resource uv_default_loop()
|
|
*/
|
|
PHP_FUNCTION(uv_default_loop)
|
|
{
|
|
/* TODO: implement this correctly */
|
|
ZEND_REGISTER_RESOURCE(return_value, php_uv_default_loop(), uv_loop_handle);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto resource uv_loop_new()
|
|
*/
|
|
PHP_FUNCTION(uv_loop_new)
|
|
{
|
|
uv_loop_t *loop = emalloc(sizeof(*loop));
|
|
uv_loop_init(loop);
|
|
|
|
//loop = uv_loop_new();
|
|
|
|
ZEND_REGISTER_RESOURCE(return_value, loop, uv_loop_handle);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto resource uv_udp_init([resource $loop])
|
|
*/
|
|
PHP_FUNCTION(uv_udp_init)
|
|
{
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop = NULL;
|
|
php_uv_t *uv;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"|r", &zloop) == FAILURE) {
|
|
return;
|
|
}
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
php_uv_common_init(&uv, loop, IS_UV_UDP, return_value);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_udp_bind(resource $resource, resource $address, long $flags)
|
|
*/
|
|
PHP_FUNCTION(uv_udp_bind)
|
|
{
|
|
php_uv_socket_bind(PHP_UV_UDP_IPV4, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_udp_bind6(resource $resource, resource $address, long $flags)
|
|
*/
|
|
PHP_FUNCTION(uv_udp_bind6)
|
|
{
|
|
php_uv_socket_bind(PHP_UV_UDP_IPV6, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_udp_recv_start(resource $handle, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_udp_recv_start)
|
|
{
|
|
zval *client;
|
|
php_uv_t *uv;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
int r;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rf", &client, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_UDP);
|
|
|
|
if (uv_is_active((uv_handle_t*)&uv->uv.udp)) {
|
|
php_error_docref(NULL, E_WARNING, "passed uv_resource has already activated.");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
|
|
uv->uv.udp.data = uv;
|
|
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_RECV_CB);
|
|
r = uv_udp_recv_start((uv_udp_t*)&uv->uv.udp, php_uv_read_alloc, php_uv_udp_recv_cb);
|
|
if (r) {
|
|
php_error_docref(NULL, E_NOTICE, "read failed");
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_udp_recv_stop(resource $handle)
|
|
*/
|
|
PHP_FUNCTION(uv_udp_recv_stop)
|
|
{
|
|
zval *client;
|
|
php_uv_t *uv;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &client) == FAILURE) {
|
|
return;
|
|
}
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_UDP);
|
|
|
|
if (!uv_is_active((uv_handle_t*)&uv->uv.udp)) {
|
|
php_error_docref(NULL, E_NOTICE, "passed uv_resource has already stopped.");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
uv_udp_recv_stop((uv_udp_t*)&uv->uv.udp);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto long uv_udp_set_membership(resource $handle, string $multicast_addr, string $interface_addr, long $membership)
|
|
*/
|
|
PHP_FUNCTION(uv_udp_set_membership)
|
|
{
|
|
zval *client;
|
|
php_uv_t *uv;
|
|
zend_string *multicast_addr, *interface_addr;
|
|
int error;
|
|
int membership;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rSSl", &client, &multicast_addr, &interface_addr, &membership) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_UDP);
|
|
|
|
error = uv_udp_set_membership((uv_udp_t*)&uv->uv.udp, (const char*)multicast_addr->val, (const char*)interface_addr->val, (int)membership);
|
|
|
|
RETURN_LONG(error);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto void uv_udp_set_multicast_loop(resource $handle, long $enabled)
|
|
*/
|
|
PHP_FUNCTION(uv_udp_set_multicast_loop)
|
|
{
|
|
zval *client;
|
|
php_uv_t *uv;
|
|
long enabled = 0;
|
|
int r;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rl", &client, &enabled) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_UDP);
|
|
|
|
r = uv_udp_set_multicast_loop((uv_udp_t*)&uv->uv.udp, enabled);
|
|
if (r) {
|
|
php_error_docref(NULL, E_NOTICE, "uv_udp_set_muticast_loop failed");
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_udp_set_multicast_ttl(resource $handle, long $ttl)
|
|
*/
|
|
PHP_FUNCTION(uv_udp_set_multicast_ttl)
|
|
{
|
|
zval *client;
|
|
php_uv_t *uv;
|
|
long ttl = 0; /* 1 through 255 */
|
|
int r;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rl", &client, &ttl) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_UDP);
|
|
|
|
if (ttl > 255) {
|
|
php_error_docref(NULL, E_NOTICE, "uv_udp_set_muticast_ttl: ttl parameter expected smaller than 255.");
|
|
ttl = 255;
|
|
} else if (ttl < 1) {
|
|
php_error_docref(NULL, E_NOTICE, "uv_udp_set_muticast_ttl: ttl parameter expected larger than 0.");
|
|
ttl = 1;
|
|
}
|
|
|
|
r = uv_udp_set_multicast_ttl((uv_udp_t*)&uv->uv.udp, ttl);
|
|
if (r) {
|
|
php_error_docref(NULL, E_NOTICE, "uv_udp_set_muticast_ttl failed");
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_udp_set_broadcast(resource $handle, bool $enabled)
|
|
*/
|
|
PHP_FUNCTION(uv_udp_set_broadcast)
|
|
{
|
|
zval *client;
|
|
php_uv_t *uv;
|
|
long enabled = 0;
|
|
int r;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rl", &client, &enabled) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_UDP);
|
|
|
|
r = uv_udp_set_broadcast((uv_udp_t*)&uv->uv.udp, enabled);
|
|
if (r) {
|
|
php_error_docref(NULL, E_NOTICE, "uv_udp_set_muticast_loop failed");
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_udp_send(resource $handle, string $data, resource $uv_addr, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_udp_send)
|
|
{
|
|
php_uv_udp_send(1, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_udp_send6(resource $handle, string $data, resource $uv_addr6, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_udp_send6)
|
|
{
|
|
php_uv_udp_send(2, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto bool uv_is_active(resource $handle)
|
|
*/
|
|
PHP_FUNCTION(uv_is_active)
|
|
{
|
|
zval *handle;
|
|
php_uv_t *uv;
|
|
int r;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &handle) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
r = uv_is_active((uv_handle_t*)php_uv_get_current_stream(uv));
|
|
RETURN_BOOL(r);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto bool uv_is_readable(resource $handle)
|
|
*/
|
|
PHP_FUNCTION(uv_is_readable)
|
|
{
|
|
zval *handle;
|
|
php_uv_t *uv;
|
|
int r;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &handle) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
r = uv_is_readable((uv_stream_t*)php_uv_get_current_stream(uv));
|
|
RETURN_BOOL(r);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto bool uv_is_writable(resource $handle)
|
|
*/
|
|
PHP_FUNCTION(uv_is_writable)
|
|
{
|
|
zval *handle;
|
|
php_uv_t *uv;
|
|
int r;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &handle) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
r = uv_is_writable((uv_stream_t*)php_uv_get_current_stream(uv));
|
|
RETURN_BOOL(r);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto bool uv_walk(resource $loop, callable $closure[, array $opaque])
|
|
*/
|
|
PHP_FUNCTION(uv_walk)
|
|
{
|
|
zval *zloop, *opaque;
|
|
uv_loop_t *loop;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
//php_uv_cb_t *cb;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rf|a", &zloop, &fci, &fcc, &opaque) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
php_error_docref(NULL, E_ERROR, "uv_walk not yet supported");
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
uv_walk(loop, php_uv_walk_cb, NULL);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto long uv_guess_handle(resource $uv)
|
|
*/
|
|
PHP_FUNCTION(uv_guess_handle)
|
|
{
|
|
zval *handle;
|
|
long fd = -1;
|
|
uv_handle_type type;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &handle) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
fd = php_uv_zval_to_fd(handle);
|
|
if (fd < 0) {
|
|
php_error_docref(NULL, E_WARNING, "invalid variable passed. can't convert to fd.");
|
|
return;
|
|
}
|
|
type = uv_guess_handle(fd);
|
|
|
|
RETURN_LONG(type);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto long uv_handle_type(resource $uv)
|
|
*/
|
|
PHP_FUNCTION(uv_handle_type)
|
|
{
|
|
zval *handle;
|
|
php_uv_t *uv = NULL;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &handle) == FAILURE) {
|
|
return;
|
|
}
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
RETURN_LONG(uv->type);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto resource uv_pipe_init(resource $loop, long $ipc)
|
|
*/
|
|
PHP_FUNCTION(uv_pipe_init)
|
|
{
|
|
php_uv_t *uv;
|
|
uv_loop_t *loop;
|
|
zval *zloop = NULL;
|
|
zend_bool ipc = 0;
|
|
int r;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"|rb", &zloop, &ipc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
|
|
uv = (php_uv_t *)emalloc(sizeof(php_uv_t));
|
|
if (!uv) {
|
|
php_error_docref(NULL, E_ERROR, "uv_pipe_init emalloc failed");
|
|
return;
|
|
}
|
|
|
|
uv->type = IS_UV_PIPE;
|
|
r = uv_pipe_init(loop, &uv->uv.pipe, (int)ipc);
|
|
|
|
if (r) {
|
|
php_error_docref(NULL, E_ERROR, "uv_pipe_init failed");
|
|
return;
|
|
}
|
|
|
|
uv->uv.pipe.data = uv;
|
|
PHP_UV_INIT_ZVALS(uv)
|
|
TSRMLS_SET_CTX(uv->thread_ctx);
|
|
|
|
ZEND_REGISTER_RESOURCE(return_value, uv, uv_resource_handle);
|
|
uv->resource_id = Z_RES_P(return_value);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_pipe_open(resource $handle, long $pipe)
|
|
*/
|
|
PHP_FUNCTION(uv_pipe_open)
|
|
{
|
|
php_uv_t *uv;
|
|
zval *handle;
|
|
long pipe = -1; // file handle
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rl", &handle, &pipe) == FAILURE) {
|
|
return;
|
|
}
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_PIPE);
|
|
|
|
if (pipe < 0) {
|
|
php_error_docref(NULL, E_WARNING, "pipe parameter have to be unsigned value");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
uv_pipe_open(&uv->uv.pipe, pipe);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto long uv_pipe_bind(resource $handle, string $name)
|
|
*/
|
|
PHP_FUNCTION(uv_pipe_bind)
|
|
{
|
|
php_uv_t *uv;
|
|
zval *handle;
|
|
zend_string *name;
|
|
int error;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rS", &handle, &name) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_PIPE);
|
|
|
|
error = uv_pipe_bind(&uv->uv.pipe, name->val);
|
|
if (error) {
|
|
php_error_docref(NULL, E_WARNING, "%s", php_uv_strerror(error));
|
|
}
|
|
|
|
RETURN_LONG(error);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_pipe_connect(resource $handle, string $path, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_pipe_connect)
|
|
{
|
|
zval *resource = NULL;
|
|
php_uv_t *uv;
|
|
zend_string *name;
|
|
uv_connect_t *req;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rSf", &resource, &name, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, resource, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_PIPE);
|
|
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
|
|
req = (uv_connect_t*)emalloc(sizeof(uv_connect_t));
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_PIPE_CONNECT_CB);
|
|
|
|
req->data = uv;
|
|
uv_pipe_connect(req, (uv_pipe_t*)php_uv_get_current_stream(uv), name->val, php_uv_pipe_connect_cb);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_pipe_pending_instances(resource $handle, long $count)
|
|
*/
|
|
PHP_FUNCTION(uv_pipe_pending_instances)
|
|
{
|
|
php_uv_t *uv;
|
|
zval *handle;
|
|
long count;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rl", &handle, &count) == FAILURE) {
|
|
return;
|
|
}
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_PIPE);
|
|
|
|
uv_pipe_pending_instances(&uv->uv.pipe, count);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_pipe_pending_count(resource $handle)
|
|
*/
|
|
PHP_FUNCTION(uv_pipe_pending_count)
|
|
{
|
|
php_uv_t *uv;
|
|
zval *handle;
|
|
int count;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &handle) == FAILURE) {
|
|
return;
|
|
}
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_PIPE);
|
|
|
|
count = uv_pipe_pending_count(&uv->uv.pipe);
|
|
RETURN_LONG(count);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_pipe_pending_type(resource $handle)
|
|
*/
|
|
PHP_FUNCTION(uv_pipe_pending_type)
|
|
{
|
|
php_uv_t *uv;
|
|
zval *handle;
|
|
uv_handle_type ret;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &handle) == FAILURE) {
|
|
return;
|
|
}
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_PIPE);
|
|
|
|
ret = uv_pipe_pending_type(&uv->uv.pipe);
|
|
RETURN_LONG(ret);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_stdio_new(zval $fd, long $flags)
|
|
*/
|
|
PHP_FUNCTION(uv_stdio_new)
|
|
{
|
|
php_uv_stdio_t *stdio;
|
|
zval *handle;
|
|
long flags = 0;
|
|
php_uv_t *uv;
|
|
php_socket *socket;
|
|
php_socket_t fd = -1;
|
|
php_stream *stream;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r|l", &handle, &flags) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
if(Z_TYPE_P(handle) == IS_NULL) {
|
|
php_error_docref(NULL, E_WARNING, "passed unexpected resource");
|
|
RETURN_FALSE;
|
|
} else if (Z_TYPE_P(handle) == IS_LONG) {
|
|
fd = Z_LVAL_P(handle);
|
|
} else if (Z_TYPE_P(handle) == IS_RESOURCE) {
|
|
if (ZEND_FETCH_RESOURCE_NO_RETURN(stream, php_stream*, handle, -1, NULL, php_file_le_stream())) {
|
|
if (php_stream_cast(stream, PHP_STREAM_AS_FD | PHP_STREAM_CAST_INTERNAL, (void*)&fd, 1) != SUCCESS || fd < 0) {
|
|
fd = -1;
|
|
}
|
|
} else if (ZEND_FETCH_RESOURCE_NO_RETURN(socket, php_socket *, handle, -1, NULL, php_sockets_le_socket())) {
|
|
fd = socket->bsd_socket;
|
|
} else if (ZEND_FETCH_RESOURCE_NO_RETURN(uv, php_uv_t*, handle, -1, NULL, uv_resource_handle)) {
|
|
fd = -1;
|
|
} else {
|
|
php_error_docref(NULL, E_WARNING, "passed unexpected resource");
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
|
|
stdio = (php_uv_stdio_t*)emalloc(sizeof(php_uv_stdio_t));
|
|
stdio->flags = flags;
|
|
stdio->stream = NULL;
|
|
|
|
stdio->fd = fd;
|
|
|
|
if (Z_TYPE_P(handle) == IS_RESOURCE) {
|
|
stdio->stream = Z_RES_P(handle);
|
|
Z_ADDREF_P(handle);
|
|
}
|
|
|
|
ZEND_REGISTER_RESOURCE(return_value, stdio, uv_stdio_handle);
|
|
stdio->resource_id = Z_RES_P(return_value);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto array uv_loadavg(void)
|
|
*/
|
|
PHP_FUNCTION(uv_loadavg)
|
|
{
|
|
double average[3];
|
|
|
|
if (zend_parse_parameters_none() == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
uv_loadavg(average);
|
|
|
|
array_init(return_value);
|
|
add_next_index_double(return_value, average[0]);
|
|
add_next_index_double(return_value, average[1]);
|
|
add_next_index_double(return_value, average[2]);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto double uv_uptime(void)
|
|
*/
|
|
PHP_FUNCTION(uv_uptime)
|
|
{
|
|
double uptime;
|
|
|
|
if (zend_parse_parameters_none() == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
uv_uptime(&uptime);
|
|
|
|
RETURN_DOUBLE(uptime);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto long uv_get_free_memory(void)
|
|
*/
|
|
PHP_FUNCTION(uv_get_free_memory)
|
|
{
|
|
if (zend_parse_parameters_none() == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
RETURN_LONG(uv_get_free_memory());
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto long uv_get_total_memory(void)
|
|
*/
|
|
PHP_FUNCTION(uv_get_total_memory)
|
|
{
|
|
if (zend_parse_parameters_none() == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
RETURN_LONG(uv_get_total_memory());
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto long uv_hrtime(void)
|
|
*/
|
|
PHP_FUNCTION(uv_hrtime)
|
|
{
|
|
if (zend_parse_parameters_none() == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
/* TODO: is this correct? */
|
|
RETURN_LONG(uv_hrtime());
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string uv_exepath(void)
|
|
*/
|
|
PHP_FUNCTION(uv_exepath)
|
|
{
|
|
char buffer[MAXPATHLEN];
|
|
size_t buffer_sz = sizeof(buffer) / sizeof(buffer[0]);
|
|
|
|
if (zend_parse_parameters_none() == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
if (uv_exepath(buffer, &buffer_sz) == UV_EINVAL) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
RETURN_STRINGL(buffer, buffer_sz);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string uv_cwd(void)
|
|
*/
|
|
PHP_FUNCTION(uv_cwd)
|
|
{
|
|
char buffer[MAXPATHLEN];
|
|
size_t buffer_sz = sizeof(buffer) / sizeof(buffer[0]);
|
|
|
|
if (zend_parse_parameters_none() == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
uv_cwd(buffer, &buffer_sz);
|
|
|
|
RETURN_STRING(buffer);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto array uv_cpu_info(void)
|
|
*/
|
|
PHP_FUNCTION(uv_cpu_info)
|
|
{
|
|
uv_cpu_info_t *cpus;
|
|
int error;
|
|
int i, count;
|
|
|
|
if (zend_parse_parameters_none() == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
error = uv_cpu_info(&cpus, &count);
|
|
if (0 == error) {
|
|
array_init(return_value);
|
|
|
|
for (i = 0; i < count; i++) {
|
|
zval tmp, times;
|
|
|
|
array_init(&tmp);
|
|
array_init(×);
|
|
|
|
add_assoc_string_ex(&tmp, ZEND_STRL("model"), cpus[i].model);
|
|
add_assoc_long_ex(&tmp, ZEND_STRL("speed"), cpus[i].speed);
|
|
|
|
add_assoc_long_ex(×, ZEND_STRL("sys"), (size_t)cpus[i].cpu_times.sys);
|
|
add_assoc_long_ex(×, ZEND_STRL("user"), (size_t)cpus[i].cpu_times.user);
|
|
add_assoc_long_ex(×, ZEND_STRL("idle"), (size_t)cpus[i].cpu_times.idle);
|
|
add_assoc_long_ex(×, ZEND_STRL("irq"), (size_t)cpus[i].cpu_times.irq);
|
|
add_assoc_long_ex(×, ZEND_STRL("nice"), (size_t)cpus[i].cpu_times.nice);
|
|
add_assoc_zval_ex(&tmp, ZEND_STRL("times"), ×);
|
|
|
|
add_next_index_zval(return_value, &tmp);
|
|
}
|
|
|
|
uv_free_cpu_info(cpus, count);
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto array uv_interface_addresses(void)
|
|
*/
|
|
PHP_FUNCTION(uv_interface_addresses)
|
|
{
|
|
uv_interface_address_t *interfaces;
|
|
int error;
|
|
char buffer[512];
|
|
int i, count;
|
|
|
|
if (zend_parse_parameters_none() == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
error = uv_interface_addresses(&interfaces, &count);
|
|
if (0 == error) {
|
|
array_init(return_value);
|
|
|
|
for (i = 0; i < count; i++) {
|
|
zval tmp;
|
|
|
|
array_init(&tmp);
|
|
|
|
add_assoc_string_ex(&tmp, ZEND_STRL("name"), interfaces[i].name);
|
|
add_assoc_bool_ex(&tmp, ZEND_STRL("is_internal"), interfaces[i].is_internal);
|
|
|
|
if (interfaces[i].address.address4.sin_family == AF_INET) {
|
|
uv_ip4_name(&interfaces[i].address.address4, buffer, sizeof(buffer));
|
|
} else if (interfaces[i].address.address4.sin_family == AF_INET6) {
|
|
uv_ip6_name(&interfaces[i].address.address6, buffer, sizeof(buffer));
|
|
}
|
|
add_assoc_string_ex(&tmp, ZEND_STRL("address"), buffer);
|
|
|
|
add_next_index_zval(return_value, &tmp);
|
|
}
|
|
uv_free_interface_addresses(interfaces, count);
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto resource uv_spawn(resource $loop, string $command, array $args, array $stdio, string $cwd, array $env = array(), callable $callback [,long $flags, array $options])
|
|
*/
|
|
PHP_FUNCTION(uv_spawn)
|
|
{
|
|
uv_loop_t *loop;
|
|
uv_process_options_t options = {0};
|
|
uv_stdio_container_t *stdio = NULL;
|
|
php_uv_t *proc;
|
|
zval *zloop, *args, *env, *zoptions = NULL, *zstdio = NULL, *value;
|
|
char **command_args, **zenv;
|
|
zend_string *command, *cwd;
|
|
int uid = 0, gid = 0, stdio_count = 0;
|
|
long flags = 0;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rSaaSafl|a", &zloop, &command, &args, &zstdio, &cwd, &env, &fci, &fcc, &flags, &zoptions) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
memset(&options, 0, sizeof(uv_process_options_t));
|
|
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
|
|
{/* process stdio */
|
|
HashTable *stdio_container;
|
|
int x = 0;
|
|
|
|
stdio_container = Z_ARRVAL_P(zstdio);
|
|
stdio_count = zend_hash_num_elements(stdio_container);
|
|
|
|
stdio = emalloc(sizeof(uv_stdio_container_t) * stdio_count);
|
|
|
|
ZEND_HASH_FOREACH_VAL(stdio_container, value) {
|
|
php_uv_stdio_t *stdio_tmp;
|
|
|
|
if (Z_TYPE_P(value) != IS_RESOURCE) {
|
|
php_error_docref(NULL, E_ERROR, "must be uv_stdio resource");
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(stdio_tmp, php_uv_stdio_t*, value, -1, PHP_UV_STDIO_RESOURCE_NAME, uv_stdio_handle);
|
|
|
|
stdio[x].flags = stdio_tmp->flags;
|
|
|
|
if (stdio_tmp->flags & UV_INHERIT_FD) {
|
|
stdio[x].data.fd = stdio_tmp->fd;
|
|
} else if (stdio_tmp->flags & (UV_CREATE_PIPE | UV_INHERIT_STREAM)) {
|
|
php_uv_t* uv_pipe;
|
|
zval passed_id;
|
|
ZVAL_RES(&passed_id, stdio_tmp->stream);
|
|
|
|
ZEND_FETCH_RESOURCE(uv_pipe, php_uv_t*, &passed_id, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
stdio[x].data.stream = (uv_stream_t*)&uv_pipe->uv.pipe;
|
|
} else {
|
|
php_error_docref(NULL, E_WARNING, "passes unexpected stdio flags");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
x++;
|
|
} ZEND_HASH_FOREACH_END();
|
|
}
|
|
|
|
{
|
|
HashTable *h;
|
|
zval *value;
|
|
int n = 0;
|
|
int hash_len = 0;
|
|
|
|
h = Z_ARRVAL_P(args);
|
|
|
|
hash_len = zend_hash_num_elements(h);
|
|
|
|
command_args = ecalloc(hash_len+2, sizeof(char**));
|
|
command_args[n] = command->val;
|
|
|
|
n++;
|
|
ZEND_HASH_FOREACH_VAL(h, value) {
|
|
command_args[n] = Z_STRVAL_P(value);
|
|
n++;
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
command_args[n] = NULL;
|
|
}
|
|
|
|
{ /* env */
|
|
HashTable *tmp_env;
|
|
zend_string *key;
|
|
int i = 0;
|
|
zval *value;
|
|
|
|
tmp_env = Z_ARRVAL_P(env);
|
|
|
|
zenv = ecalloc(zend_hash_num_elements(tmp_env)+1, sizeof(char*));
|
|
ZEND_HASH_FOREACH_STR_KEY_VAL(tmp_env, key, value) {
|
|
char *tmp_env_entry;
|
|
|
|
tmp_env_entry = emalloc(sizeof(char) * (key->len + 2 + Z_STRLEN_P(value)));
|
|
slprintf(tmp_env_entry, key->len + 1 + Z_STRLEN_P(value), "%s=%s", key->val, Z_STRVAL_P(value));
|
|
|
|
zenv[i++] = tmp_env_entry;
|
|
} ZEND_HASH_FOREACH_END();
|
|
zenv[i] = NULL;
|
|
}
|
|
|
|
if (zoptions != NULL && Z_TYPE_P(zoptions) != IS_NULL){
|
|
HashTable *opts;
|
|
zval *data;
|
|
|
|
opts = Z_ARRVAL_P(zoptions);
|
|
|
|
if ((data = zend_hash_str_find(opts, ZEND_STRL("uid")))) {
|
|
uid = Z_LVAL_P(data);
|
|
}
|
|
|
|
if ((data = zend_hash_str_find(opts, ZEND_STRL("gid")))) {
|
|
gid = Z_LVAL_P(data);
|
|
}
|
|
}
|
|
|
|
options.file = command->val;
|
|
options.stdio = stdio;
|
|
options.exit_cb = php_uv_process_close_cb;
|
|
options.env = zenv;
|
|
options.args = command_args;
|
|
options.cwd = cwd->val;
|
|
options.stdio = stdio;
|
|
options.stdio_count = stdio_count;
|
|
options.flags = flags;
|
|
options.uid = uid;
|
|
options.gid = gid;
|
|
|
|
proc = (php_uv_t *)emalloc(sizeof(php_uv_t));
|
|
PHP_UV_INIT_ZVALS(proc);
|
|
php_uv_cb_init(&cb, proc, &fci, &fcc, PHP_UV_PROC_CLOSE_CB);
|
|
TSRMLS_SET_CTX(proc->thread_ctx);
|
|
|
|
proc->type = IS_UV_PROCESS;
|
|
proc->uv.process.data = proc;
|
|
|
|
ZEND_REGISTER_RESOURCE(return_value, proc, uv_resource_handle);
|
|
proc->resource_id = Z_RES_P(return_value);
|
|
zval_copy_ctor(return_value);
|
|
|
|
uv_spawn(loop, &proc->uv.process, &options);
|
|
|
|
if (zenv != NULL) {
|
|
char **p = zenv;
|
|
while(*p != NULL) {
|
|
efree(*p);
|
|
p++;
|
|
}
|
|
efree(zenv);
|
|
}
|
|
if (command_args != NULL) {
|
|
efree(command_args);
|
|
}
|
|
|
|
if (stdio != NULL) {
|
|
efree(stdio);
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto void uv_process_kill(resource $handle, long $signal)
|
|
*/
|
|
PHP_FUNCTION(uv_process_kill)
|
|
{
|
|
php_uv_t *uv;
|
|
zval *handle;
|
|
int signal;
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rl", &handle, &signal) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
if (uv->type != IS_UV_PROCESS) {
|
|
php_error_docref(NULL, E_WARNING, "passed resource didn't initialize for uv_process");
|
|
}
|
|
|
|
uv_process_kill(&uv->uv.process, signal);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_kill(long $pid, long $signal)
|
|
*/
|
|
PHP_FUNCTION(uv_kill)
|
|
{
|
|
long pid, signal;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"ll", &pid, &signal) == FAILURE) {
|
|
return;
|
|
}
|
|
uv_kill(pid, signal);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto bool uv_chdir(string $directory)
|
|
*/
|
|
PHP_FUNCTION(uv_chdir)
|
|
{
|
|
int error;
|
|
zend_string *directory;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"S", &directory) == FAILURE) {
|
|
return;
|
|
}
|
|
error = uv_chdir(directory->val);
|
|
if (error == 0) {
|
|
RETURN_TRUE;
|
|
} else {
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto resource uv_rwlock_init(void)
|
|
*/
|
|
PHP_FUNCTION(uv_rwlock_init)
|
|
{
|
|
php_uv_lock_init(IS_UV_RWLOCK, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto uv_rwlock_rdlock(resource $handle)
|
|
*/
|
|
PHP_FUNCTION(uv_rwlock_rdlock)
|
|
{
|
|
php_uv_lock_lock(IS_UV_RWLOCK_RD, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto bool uv_rwlock_tryrdlock(resource $handle)
|
|
*/
|
|
PHP_FUNCTION(uv_rwlock_tryrdlock)
|
|
{
|
|
php_uv_lock_trylock(IS_UV_RWLOCK_RD, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_rwlock_rdunlock(resource $handle)
|
|
*/
|
|
PHP_FUNCTION(uv_rwlock_rdunlock)
|
|
{
|
|
php_uv_lock_unlock(IS_UV_RWLOCK_RD, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto uv_rwlock_wrlock(resource $handle)
|
|
*/
|
|
PHP_FUNCTION(uv_rwlock_wrlock)
|
|
{
|
|
php_uv_lock_lock(IS_UV_RWLOCK_WR, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto uv_rwlock_trywrlock(resource $handle)
|
|
*/
|
|
PHP_FUNCTION(uv_rwlock_trywrlock)
|
|
{
|
|
php_uv_lock_trylock(IS_UV_RWLOCK_WR, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto uv_rwlock_wrunlock(resource $handle)
|
|
*/
|
|
PHP_FUNCTION(uv_rwlock_wrunlock)
|
|
{
|
|
php_uv_lock_unlock(IS_UV_RWLOCK_WR, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto uv_lock uv_mutex_init(void)
|
|
*/
|
|
PHP_FUNCTION(uv_mutex_init)
|
|
{
|
|
php_uv_lock_init(IS_UV_MUTEX, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_mutex_lock(uv_lock $lock)
|
|
*/
|
|
PHP_FUNCTION(uv_mutex_lock)
|
|
{
|
|
php_uv_lock_lock(IS_UV_MUTEX, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto bool uv_mutex_trylock(uv_lock $lock)
|
|
*/
|
|
PHP_FUNCTION(uv_mutex_trylock)
|
|
{
|
|
php_uv_lock_trylock(IS_UV_MUTEX, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ void uv_mutex_unlock(uv_lock $lock)
|
|
|
|
##### *Description*
|
|
|
|
unlock mutex
|
|
|
|
##### *Parameters*
|
|
|
|
*resource $handle*: uv resource handle (uv mutex)
|
|
|
|
##### *Return Value*
|
|
|
|
*void *:
|
|
|
|
##### *Example*
|
|
|
|
*/
|
|
PHP_FUNCTION(uv_mutex_unlock)
|
|
{
|
|
php_uv_lock_unlock(IS_UV_MUTEX, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto uv_lock uv_sem_init(long $value)
|
|
*/
|
|
PHP_FUNCTION(uv_sem_init)
|
|
{
|
|
php_uv_lock_init(IS_UV_SEMAPHORE, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_sem_post(uv_lock $sem)
|
|
*/
|
|
PHP_FUNCTION(uv_sem_post)
|
|
{
|
|
php_uv_lock_lock(IS_UV_SEMAPHORE, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_sem_wait(uv_lock $sem)
|
|
*/
|
|
PHP_FUNCTION(uv_sem_wait)
|
|
{
|
|
php_uv_lock_unlock(IS_UV_SEMAPHORE, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_sem_trywait(uv_lock $sem)
|
|
*/
|
|
PHP_FUNCTION(uv_sem_trywait)
|
|
{
|
|
php_uv_lock_trylock(IS_UV_SEMAPHORE, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto resource uv_prepare_init(resource $loop)
|
|
*/
|
|
PHP_FUNCTION(uv_prepare_init)
|
|
{
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
php_uv_t *uv;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"|r", &zloop) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
php_uv_common_init(&uv, loop, IS_UV_PREPARE, return_value);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_prepare_start(resource $handle, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_prepare_start)
|
|
{
|
|
zval *handle;
|
|
php_uv_t *uv;
|
|
int r;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
PHP_UV_DEBUG_PRINT("uv_prepare_start\n");
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rf", &handle, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_PREPARE);
|
|
|
|
if (uv_is_active((uv_handle_t*)&uv->uv.prepare)) {
|
|
php_error_docref(NULL, E_WARNING, "passed uv_prepare resource has been started.");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
uv->uv.prepare.data = uv;
|
|
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_PREPARE_CB);
|
|
r = uv_prepare_start((uv_prepare_t*)php_uv_get_current_stream(uv), php_uv_prepare_cb);
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_prepare_start, uv->resource_id);
|
|
|
|
RETURN_LONG(r);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_prepare_stop(resource $handle)
|
|
*/
|
|
PHP_FUNCTION(uv_prepare_stop)
|
|
{
|
|
zval *handle;
|
|
php_uv_t *uv;
|
|
int r = 0;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &handle) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_PREPARE);
|
|
|
|
if (!uv_is_active((uv_handle_t*)&uv->uv.prepare)) {
|
|
php_error_docref(NULL, E_NOTICE, "passed uv_prepare resource has been stopped.");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
r = uv_prepare_stop((uv_prepare_t*)php_uv_get_current_stream(uv));
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_prepare_stop, uv->resource_id);
|
|
|
|
RETURN_LONG(r);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto resoruce uv_check_init([resource $loop])
|
|
*/
|
|
PHP_FUNCTION(uv_check_init)
|
|
{
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
php_uv_t *uv;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"|r", &zloop) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
php_uv_common_init(&uv, loop, IS_UV_CHECK, return_value);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_check_start(resource $handle, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_check_start)
|
|
{
|
|
zval *handle;
|
|
php_uv_t *uv;
|
|
int r;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
PHP_UV_DEBUG_PRINT("uv_check_start");
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rf", &handle, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_CHECK);
|
|
|
|
if (uv_is_active((uv_handle_t*)&uv->uv.idle)) {
|
|
php_error_docref(NULL, E_WARNING, "passed uv check resource has already started");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
|
|
uv->uv.check.data = uv;
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_CHECK_CB);
|
|
|
|
r = uv_check_start((uv_check_t*)php_uv_get_current_stream(uv), php_uv_check_cb);
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_check_start, uv->resource_id);
|
|
|
|
RETURN_LONG(r);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_check_stop(resource $handle)
|
|
*/
|
|
PHP_FUNCTION(uv_check_stop)
|
|
{
|
|
zval *handle;
|
|
php_uv_t *uv;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &handle) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_CHECK);
|
|
|
|
if (!uv_is_active((uv_handle_t*)&uv->uv.check)) {
|
|
php_error_docref(NULL, E_NOTICE, "passed uv_check resource hasn't start yet.");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
uv_check_stop((uv_check_t*)php_uv_get_current_stream(uv));
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_check_stop, uv->resource_id);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto resource uv_async_init(resource $loop, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_async_init)
|
|
{
|
|
int r;
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
php_uv_t *uv;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rf",&zloop, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
PHP_UV_INIT_UV(uv, IS_UV_ASYNC);
|
|
|
|
r = uv_async_init(loop, &uv->uv.async, php_uv_async_cb);
|
|
if (r) {
|
|
php_error_docref(NULL, E_ERROR, "uv_async_init failed");
|
|
return;
|
|
}
|
|
|
|
uv->uv.async.data = uv;
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_ASYNC_CB);
|
|
|
|
ZVAL_RES(return_value, uv->resource_id);
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_async_send(resource $handle)
|
|
*/
|
|
PHP_FUNCTION(uv_async_send)
|
|
{
|
|
zval *handle;
|
|
php_uv_t *uv;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &handle) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
PHP_UV_TYPE_CHECK(uv, IS_UV_ASYNC)
|
|
|
|
uv_async_send((uv_async_t*)php_uv_get_current_stream(uv));
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_async_send, uv->resource_id);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_queue_work(resource $loop, callable $callback, callable $after_callback)
|
|
*/
|
|
PHP_FUNCTION(uv_queue_work)
|
|
{
|
|
#ifdef ZTS
|
|
int r;
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
php_uv_t *uv;
|
|
zend_fcall_info work_fci, after_fci = empty_fcall_info;
|
|
zend_fcall_info_cache work_fcc, after_fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *work_cb, *after_cb;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rff", &zloop, &work_fci, &work_fcc, &after_fci, &after_fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
PHP_UV_INIT_UV(uv, IS_UV_WORK)
|
|
|
|
php_uv_cb_init(&work_cb, uv, &work_fci, &work_fcc, PHP_UV_WORK_CB);
|
|
php_uv_cb_init(&after_cb, uv, &after_fci, &after_fcc, PHP_UV_AFTER_WORK_CB);
|
|
|
|
uv->uv.work.data = uv;
|
|
|
|
r = uv_queue_work(loop, (uv_work_t*)&uv->uv.work, php_uv_work_cb, php_uv_after_work_cb);
|
|
|
|
if (r) {
|
|
php_error_docref(NULL, E_ERROR, "uv_queue_work failed");
|
|
return;
|
|
}
|
|
#else
|
|
php_error_docref(NULL, E_ERROR, "this PHP doesn't support this uv_queue_work. please rebuild with --enable-maintainer-zts");
|
|
#endif
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto resource uv_fs_open(resource $loop, string $path, long $flag, long $mode, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_open)
|
|
{
|
|
php_uv_fs_common(UV_FS_OPEN, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto void uv_fs_read(resource $loop, zval $fd, long $offset, long $length, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_read)
|
|
{
|
|
php_uv_fs_common(UV_FS_READ, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto void uv_fs_close(resource $loop, zval $fd, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_close)
|
|
{
|
|
php_uv_fs_common(UV_FS_CLOSE, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto void uv_fs_write(resource $loop, zval $fd, string $buffer, long $offset, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_write)
|
|
{
|
|
php_uv_fs_common(UV_FS_WRITE, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_fsync(resource $loop, zval $fd, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_fsync)
|
|
{
|
|
php_uv_fs_common(UV_FS_FSYNC, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_fdatasync(resource $loop, zval $fd, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_fdatasync)
|
|
{
|
|
php_uv_fs_common(UV_FS_FDATASYNC, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_ftruncate(resource $loop, zval $fd, long $offset, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_ftruncate)
|
|
{
|
|
php_uv_fs_common(UV_FS_FTRUNCATE, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_mkdir(resource $loop, string $path, long $mode, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_mkdir)
|
|
{
|
|
php_uv_fs_common(UV_FS_MKDIR, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto void uv_fs_rmdir(resource $loop, string $path, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_rmdir)
|
|
{
|
|
php_uv_fs_common(UV_FS_RMDIR, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_unlink(resource $loop, string $path, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_unlink)
|
|
{
|
|
php_uv_fs_common(UV_FS_UNLINK, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_rename(resource $loop, string $from, string $to, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_rename)
|
|
{
|
|
php_uv_fs_common(UV_FS_RENAME, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_utime(resource $loop, string $path, long $utime, long $atime, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_utime)
|
|
{
|
|
php_uv_fs_common(UV_FS_UTIME, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_futime(resource $loop, zval $fd, long $utime, long $atime callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_futime)
|
|
{
|
|
php_uv_fs_common(UV_FS_FUTIME, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_chmod(resource $loop, string $path, long $mode, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_chmod)
|
|
{
|
|
php_uv_fs_common(UV_FS_CHMOD, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto void uv_fs_fchmod(resource $loop, zval $fd, long $mode, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_fchmod)
|
|
{
|
|
php_uv_fs_common(UV_FS_FCHMOD, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto void uv_fs_chown(resource $loop, string $path, long $uid, long $gid, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_chown)
|
|
{
|
|
php_uv_fs_common(UV_FS_CHOWN, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_fchown(resource $loop, zval $fd, long $uid, $long $gid, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_fchown)
|
|
{
|
|
php_uv_fs_common(UV_FS_FCHOWN, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_link(resource $loop, string $from, string $to, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_link)
|
|
{
|
|
php_uv_fs_common(UV_FS_LINK, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto void uv_fs_symlink(resource $loop, string $from, string $to, long $flags, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_symlink)
|
|
{
|
|
php_uv_fs_common(UV_FS_SYMLINK, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_readlink(resource $loop, string $path, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_readlink)
|
|
{
|
|
php_uv_fs_common(UV_FS_READLINK, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_stat(resource $loop, string $path, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_stat)
|
|
{
|
|
php_uv_fs_common(UV_FS_STAT, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_lstat(resource $loop, string $path, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_lstat)
|
|
{
|
|
php_uv_fs_common(UV_FS_LSTAT, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_fstat(resource $loop, zval $fd, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_fstat)
|
|
{
|
|
php_uv_fs_common(UV_FS_FSTAT, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto uv_fs_readdir(resource $loop, string $path, long $flags, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_readdir)
|
|
{
|
|
php_uv_fs_common(UV_FS_SCANDIR, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto uv_fs_scandir(resource $loop, string $path, long $flags, callable $callback)
|
|
* */
|
|
PHP_FUNCTION(uv_fs_scandir)
|
|
{
|
|
php_uv_fs_common(UV_FS_SCANDIR, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_sendfile(resource $loop, zval $in_fd, zval $out_fd, long $offset, long $length, callable $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_sendfile)
|
|
{
|
|
php_uv_fs_common(UV_FS_SENDFILE, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto resource uv_fs_event_init(resource $loop, string $path, callable $callback, long $flags = 0)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_event_init)
|
|
{
|
|
int error;
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
php_uv_t *uv;
|
|
zend_string *path;
|
|
long flags = 0;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rSfl", &zloop, &path, &fci, &fcc, &flags) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_INIT_UV(uv, IS_UV_FS_EVENT);
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_FS_EVENT_CB);
|
|
|
|
uv->uv.fs_event.data = uv;
|
|
|
|
/* TODO: is this right?! */
|
|
//error = uv_fs_event_init(loop, (uv_fs_event_t*)&uv->uv.fs_event, path, php_uv_fs_event_cb, flags);
|
|
error = uv_fs_event_init(loop, (uv_fs_event_t*)&uv->uv.fs_event);
|
|
if (error < 0) {
|
|
php_error_docref(NULL, E_ERROR, "uv_fs_event_init failed");
|
|
return;
|
|
}
|
|
|
|
error = uv_fs_event_start((uv_fs_event_t*)&uv->uv.fs_event, php_uv_fs_event_cb, path->val, flags);
|
|
if (error < 0) {
|
|
php_error_docref(NULL, E_ERROR, "uv_fs_event_start failed");
|
|
return;
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto resource uv_tty_init(resource $loop, zval $fd, long $readable)
|
|
*/
|
|
PHP_FUNCTION(uv_tty_init)
|
|
{
|
|
int error;
|
|
zval *zstream, *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
php_uv_t *uv;
|
|
long readable = 1;
|
|
unsigned long fd;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rrl", &zloop, &zstream, &readable) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_INIT_UV(uv, IS_UV_TTY);
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
PHP_UV_ZVAL_TO_FD(fd, zstream);
|
|
|
|
uv->uv.tty.data = uv;
|
|
|
|
error = uv_tty_init(loop, (uv_tty_t*)&uv->uv.tty, fd, readable);
|
|
if (error) {
|
|
php_error_docref(NULL, E_WARNING, "uv_tty_init failed");
|
|
return;
|
|
}
|
|
|
|
ZVAL_RES(return_value, uv->resource_id);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto long uv_tty_get_winsize(resource $tty, long &$width, long &$height)
|
|
*/
|
|
PHP_FUNCTION(uv_tty_get_winsize)
|
|
{
|
|
php_uv_t *uv;
|
|
zval *handle, *w, *h = NULL;
|
|
int error, width, height = 0;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r/z/z", &handle, &w, &h) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t*, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
if (uv->type != IS_UV_TTY) {
|
|
php_error_docref(NULL, E_WARNING, "passed resource doesn't initialize for uv_tty");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
error = uv_tty_get_winsize(&uv->uv.tty, &width, &height);
|
|
|
|
zval_dtor(w);
|
|
zval_dtor(h);
|
|
|
|
ZVAL_LONG(w, width);
|
|
ZVAL_LONG(h, height);
|
|
|
|
RETURN_LONG(error);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto long uv_tty_set_mode(resource $tty, long $mode)
|
|
*/
|
|
PHP_FUNCTION(uv_tty_set_mode)
|
|
{
|
|
php_uv_t *uv;
|
|
zval *handle;
|
|
long mode, error = 0;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rl", &handle, &mode) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t*, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
if (uv->type != IS_UV_TTY) {
|
|
php_error_docref(NULL, E_WARNING, "passed resource doesn't initialize for uv_tty");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
error = uv_tty_set_mode(&uv->uv.tty, mode);
|
|
RETURN_LONG(error);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_tty_reset_mode(void)
|
|
*/
|
|
PHP_FUNCTION(uv_tty_reset_mode)
|
|
{
|
|
if (zend_parse_parameters_none() == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
uv_tty_reset_mode();
|
|
}
|
|
/* }}} */
|
|
|
|
#ifdef PHP_WIN32
|
|
/* {{{ */
|
|
PHP_FUNCTION(uv_tcp_simultaneous_accepts)
|
|
{
|
|
php_uv_t *uv;
|
|
zval *handle, *result;
|
|
long enable, error = 0;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rl", &handle, &enable) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t*, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
if (uv->type != IS_UV_TCP) {
|
|
php_error_docref(NULL, E_WARNING, "passed resource doesn't initialize for uv_tcp");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
error = uv_tcp_simultaneous_accepts(&uv->uv.tcp, enable);
|
|
RETURN_LONG(error);
|
|
}
|
|
/* }}} */
|
|
#endif
|
|
|
|
/* {{{ proto string uv_tcp_getsockname(resource $uv_sockaddr)
|
|
*/
|
|
PHP_FUNCTION(uv_tcp_getsockname)
|
|
{
|
|
php_uv_socket_getname(1, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string uv_tcp_getpeername(resource $uv_sockaddr)
|
|
*/
|
|
PHP_FUNCTION(uv_tcp_getpeername)
|
|
{
|
|
php_uv_socket_getname(2, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string uv_udp_getsockname(resource $uv_sockaddr)
|
|
*/
|
|
PHP_FUNCTION(uv_udp_getsockname)
|
|
{
|
|
php_uv_socket_getname(3, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto long uv_resident_set_memory(void)
|
|
*/
|
|
PHP_FUNCTION(uv_resident_set_memory)
|
|
{
|
|
size_t rss;
|
|
|
|
if (zend_parse_parameters_none() == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
uv_resident_set_memory(&rss);
|
|
|
|
RETURN_LONG(rss);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string uv_ip4_name(resource uv_sockaddr $address)
|
|
*/
|
|
PHP_FUNCTION(uv_ip4_name)
|
|
{
|
|
php_uv_ip_common(1, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string uv_ip6_name(resource uv_sockaddr $address)
|
|
*/
|
|
PHP_FUNCTION(uv_ip6_name)
|
|
{
|
|
php_uv_ip_common(2, INTERNAL_FUNCTION_PARAM_PASSTHRU);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto uv uv_poll_init([resource $uv_loop], zval fd)
|
|
*/
|
|
PHP_FUNCTION(uv_poll_init)
|
|
{
|
|
zval *zstream, *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
php_uv_t *uv;
|
|
int error;
|
|
unsigned long fd = 0;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rr", &zloop, &zstream) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_INIT_UV(uv, IS_UV_POLL);
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
PHP_UV_ZVAL_TO_VALID_POLL_FD(fd, zstream);
|
|
|
|
error = uv_poll_init(loop, &uv->uv.poll, fd);
|
|
if (error) {
|
|
php_error_docref(NULL, E_ERROR, "uv_poll_init failed");
|
|
return;
|
|
}
|
|
|
|
uv->sock = fd;
|
|
ZVAL_RES(return_value, uv->resource_id);
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
/* {{{ proto uv uv_poll_init_socket([resource $uv_loop], zval fd)
|
|
*/
|
|
PHP_FUNCTION(uv_poll_init_socket)
|
|
{
|
|
zval *zstream, *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
php_uv_t *uv;
|
|
int error;
|
|
unsigned long fd = 0;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rr", &zloop, &zstream) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_INIT_UV(uv, IS_UV_POLL);
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
PHP_UV_ZVAL_TO_FD(fd, zstream);
|
|
|
|
error = uv_poll_init_socket(loop, &uv->uv.poll, fd);
|
|
if (error) {
|
|
php_error_docref(NULL, E_ERROR, "uv_poll_init_socket failed");
|
|
return;
|
|
}
|
|
|
|
uv->sock = fd;
|
|
ZVAL_RES(return_value, uv->resource_id);
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ proto uv uv_poll_start(resource $handle, $events, $callback)
|
|
*/
|
|
PHP_FUNCTION(uv_poll_start)
|
|
{
|
|
zval *handle = NULL;
|
|
php_uv_t *uv;
|
|
long events = 0;
|
|
int error;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rlf", &handle, &events, &fci, &fcc) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
if (uv->type != IS_UV_POLL) {
|
|
php_error_docref(NULL, E_WARNING, "passed resource didn't initialize for uv_poll");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_POLL_CB);
|
|
uv->uv.poll.data = uv;
|
|
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
|
|
error = uv_poll_start(&uv->uv.poll, events, php_uv_poll_cb);
|
|
if (error) {
|
|
php_error_docref(NULL, E_ERROR, "uv_poll_start failed");
|
|
return;
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_poll_stop(resource $poll)
|
|
*/
|
|
PHP_FUNCTION(uv_poll_stop)
|
|
{
|
|
zval *poll;
|
|
php_uv_t *uv;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &poll) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, poll, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
if (uv->type != IS_UV_POLL) {
|
|
php_error_docref(NULL, E_WARNING, "passed resource didn't initialize for uv_poll");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
uv_poll_stop(&uv->uv.poll);
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_poll_stop, uv->resource_id);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto uv uv_fs_poll_init([resource $uv_loop])
|
|
*/
|
|
PHP_FUNCTION(uv_fs_poll_init)
|
|
{
|
|
zval *zloop = NULL;
|
|
uv_loop_t *loop;
|
|
php_uv_t *uv;
|
|
int error;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &zloop) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
PHP_UV_INIT_UV(uv, IS_UV_FS_POLL);
|
|
PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop);
|
|
|
|
error = uv_fs_poll_init(loop, &uv->uv.fs_poll);
|
|
if (error) {
|
|
php_error_docref(NULL, E_ERROR, "uv_fs_poll_init failed");
|
|
return;
|
|
}
|
|
|
|
ZVAL_RES(return_value, uv->resource_id);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto uv uv_fs_poll_start(resource $handle, $callback, string $path, long $interval)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_poll_start)
|
|
{
|
|
zval *handle = NULL;
|
|
php_uv_t *uv;
|
|
zend_string *path;
|
|
unsigned long interval = 0;
|
|
int error;
|
|
zend_fcall_info fci = empty_fcall_info;
|
|
zend_fcall_info_cache fcc = empty_fcall_info_cache;
|
|
php_uv_cb_t *cb;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"rfSl", &handle, &fci, &fcc, &path, &interval) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
if (uv->type != IS_UV_FS_POLL) {
|
|
php_error_docref(NULL, E_WARNING, "passed uv resource didn't initialize for uv_fs_poll");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_FS_POLL_CB);
|
|
uv->uv.fs_poll.data = uv;
|
|
GC_REFCOUNT(uv->resource_id)++;
|
|
|
|
error = uv_fs_poll_start(&uv->uv.fs_poll, php_uv_fs_poll_cb, (const char*)path->val, interval);
|
|
if (error) {
|
|
php_error_docref(NULL, E_ERROR, "uv_fs_poll_start failed");
|
|
return;
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto void uv_fs_poll_stop(resource $poll)
|
|
*/
|
|
PHP_FUNCTION(uv_fs_poll_stop)
|
|
{
|
|
zval *poll;
|
|
php_uv_t *uv;
|
|
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS(),
|
|
"r", &poll) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
ZEND_FETCH_RESOURCE(uv, php_uv_t *, poll, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle);
|
|
|
|
if (uv->type != IS_UV_FS_POLL) {
|
|
php_error_docref(NULL, E_WARNING, "passed uv resource didn't initialize for uv_fs_poll");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
uv_fs_poll_stop(&uv->uv.fs_poll);
|
|
PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_fs_poll_stop, uv->resource_id);
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
|
|
|
|
static zend_function_entry uv_functions[] = {
|
|
/* general */
|
|
PHP_FE(uv_update_time, arginfo_uv_update_time)
|
|
PHP_FE(uv_ref, arginfo_uv_ref)
|
|
PHP_FE(uv_unref, arginfo_uv_unref)
|
|
PHP_FE(uv_loop_new, NULL)
|
|
PHP_FE(uv_default_loop, NULL)
|
|
PHP_FE(uv_stop, arginfo_uv_stop)
|
|
PHP_FE(uv_run, arginfo_uv_run)
|
|
PHP_FE(uv_run_once, arginfo_uv_run_once)
|
|
PHP_FE(uv_ip4_addr, arginfo_uv_ip4_addr)
|
|
PHP_FE(uv_ip6_addr, arginfo_uv_ip6_addr)
|
|
PHP_FE(uv_ip4_name, arginfo_uv_ip4_name)
|
|
PHP_FE(uv_ip6_name, arginfo_uv_ip6_name)
|
|
PHP_FE(uv_write, arginfo_uv_write)
|
|
PHP_FE(uv_write2, arginfo_uv_write2)
|
|
PHP_FE(uv_shutdown, arginfo_uv_shutdown)
|
|
PHP_FE(uv_close, arginfo_uv_close)
|
|
PHP_FE(uv_now, arginfo_uv_now)
|
|
PHP_FE(uv_loop_delete, arginfo_uv_loop_delete)
|
|
PHP_FE(uv_read_start, arginfo_uv_read_start)
|
|
PHP_FE(uv_read2_start, arginfo_uv_read2_start)
|
|
PHP_FE(uv_read_stop, arginfo_uv_read_stop)
|
|
PHP_FE(uv_err_name, arginfo_uv_err_name)
|
|
PHP_FE(uv_strerror, arginfo_uv_strerror)
|
|
PHP_FE(uv_is_active, arginfo_uv_is_active)
|
|
PHP_FE(uv_is_readable, arginfo_uv_is_readable)
|
|
PHP_FE(uv_is_writable, arginfo_uv_is_writable)
|
|
PHP_FE(uv_walk, arginfo_uv_walk)
|
|
PHP_FE(uv_guess_handle, arginfo_uv_guess_handle)
|
|
PHP_FE(uv_handle_type, arginfo_uv_handle_type)
|
|
/* idle */
|
|
PHP_FE(uv_idle_init, arginfo_uv_idle_init)
|
|
PHP_FE(uv_idle_start, arginfo_uv_idle_start)
|
|
PHP_FE(uv_idle_stop, arginfo_uv_idle_stop)
|
|
/* timer */
|
|
PHP_FE(uv_timer_init, arginfo_uv_timer_init)
|
|
PHP_FE(uv_timer_start, arginfo_uv_timer_start)
|
|
PHP_FE(uv_timer_stop, arginfo_uv_timer_stop)
|
|
PHP_FE(uv_timer_again, arginfo_uv_timer_again)
|
|
PHP_FE(uv_timer_set_repeat, arginfo_uv_timer_set_repeat)
|
|
PHP_FE(uv_timer_get_repeat, arginfo_uv_timer_get_repeat)
|
|
/* tcp */
|
|
PHP_FE(uv_tcp_init, arginfo_uv_tcp_init)
|
|
PHP_FE(uv_tcp_nodelay, arginfo_uv_tcp_nodelay)
|
|
PHP_FE(uv_tcp_bind, arginfo_uv_tcp_bind)
|
|
PHP_FE(uv_tcp_bind6, arginfo_uv_tcp_bind6)
|
|
PHP_FE(uv_listen, arginfo_uv_listen)
|
|
PHP_FE(uv_accept, arginfo_uv_accept)
|
|
PHP_FE(uv_tcp_connect, arginfo_uv_tcp_connect)
|
|
PHP_FE(uv_tcp_connect6, arginfo_uv_tcp_connect6)
|
|
/* udp */
|
|
PHP_FE(uv_udp_init, arginfo_uv_udp_init)
|
|
PHP_FE(uv_udp_bind, arginfo_uv_udp_bind)
|
|
PHP_FE(uv_udp_bind6, arginfo_uv_udp_bind6)
|
|
PHP_FE(uv_udp_set_multicast_loop, arginfo_uv_udp_set_multicast_loop)
|
|
PHP_FE(uv_udp_set_multicast_ttl, arginfo_uv_udp_set_multicast_ttl)
|
|
PHP_FE(uv_udp_send, arginfo_uv_udp_send)
|
|
PHP_FE(uv_udp_send6, arginfo_uv_udp_send6)
|
|
PHP_FE(uv_udp_recv_start, arginfo_uv_udp_recv_start)
|
|
PHP_FE(uv_udp_recv_stop, arginfo_uv_udp_recv_stop)
|
|
PHP_FE(uv_udp_set_membership, arginfo_uv_udp_set_membership)
|
|
/* poll */
|
|
PHP_FE(uv_poll_init, arginfo_uv_poll_init)
|
|
PHP_FE(uv_poll_init_socket, arginfo_uv_poll_init_socket)
|
|
PHP_FE(uv_poll_start, arginfo_uv_poll_start)
|
|
PHP_FE(uv_poll_stop, arginfo_uv_poll_stop)
|
|
PHP_FE(uv_fs_poll_init, arginfo_uv_fs_poll_init)
|
|
PHP_FE(uv_fs_poll_start, arginfo_uv_fs_poll_start)
|
|
PHP_FE(uv_fs_poll_stop, arginfo_uv_fs_poll_stop)
|
|
/* other network functions */
|
|
PHP_FE(uv_tcp_getsockname, arginfo_uv_tcp_getsockname)
|
|
PHP_FE(uv_tcp_getpeername, arginfo_uv_tcp_getpeername)
|
|
PHP_FE(uv_udp_getsockname, arginfo_uv_udp_getsockname)
|
|
#ifdef PHP_WIN32
|
|
PHP_FE(uv_tcp_simultaneous_accepts, NULL)
|
|
#endif
|
|
/* pipe */
|
|
PHP_FE(uv_pipe_init, arginfo_uv_pipe_init)
|
|
PHP_FE(uv_pipe_bind, arginfo_uv_pipe_bind)
|
|
PHP_FE(uv_pipe_open, arginfo_uv_pipe_open)
|
|
PHP_FE(uv_pipe_connect, arginfo_uv_pipe_connect)
|
|
PHP_FE(uv_pipe_pending_instances, arginfo_uv_pipe_pending_instances)
|
|
PHP_FE(uv_pipe_pending_count, arginfo_uv_pipe_pending_count)
|
|
PHP_FE(uv_pipe_pending_type, arginfo_uv_pipe_pending_type)
|
|
PHP_FE(uv_stdio_new, NULL)
|
|
/* spawn */
|
|
PHP_FE(uv_spawn, NULL)
|
|
PHP_FE(uv_process_kill, arginfo_uv_process_kill)
|
|
PHP_FE(uv_kill, arginfo_uv_kill)
|
|
/* c-ares */
|
|
PHP_FE(uv_getaddrinfo, arginfo_uv_tcp_connect)
|
|
/* rwlock */
|
|
PHP_FE(uv_rwlock_init, NULL)
|
|
PHP_FE(uv_rwlock_rdlock, arginfo_uv_rwlock_rdlock)
|
|
PHP_FE(uv_rwlock_tryrdlock, arginfo_uv_rwlock_tryrdlock)
|
|
PHP_FE(uv_rwlock_rdunlock, arginfo_uv_rwlock_rdunlock)
|
|
PHP_FE(uv_rwlock_wrlock, arginfo_uv_rwlock_wrlock)
|
|
PHP_FE(uv_rwlock_trywrlock, arginfo_uv_rwlock_trywrlock)
|
|
PHP_FE(uv_rwlock_wrunlock, arginfo_uv_rwlock_wrunlock)
|
|
/* mutex */
|
|
PHP_FE(uv_mutex_init, NULL)
|
|
PHP_FE(uv_mutex_lock, arginfo_uv_mutex_lock)
|
|
PHP_FE(uv_mutex_trylock, arginfo_uv_mutex_trylock)
|
|
PHP_FE(uv_mutex_unlock, arginfo_uv_mutex_unlock)
|
|
/* semaphore */
|
|
PHP_FE(uv_sem_init, arginfo_uv_sem_init)
|
|
PHP_FE(uv_sem_post, arginfo_uv_sem_post)
|
|
PHP_FE(uv_sem_wait, arginfo_uv_sem_wait)
|
|
PHP_FE(uv_sem_trywait, arginfo_uv_sem_trywait)
|
|
/* prepare (before poll hook) */
|
|
PHP_FE(uv_prepare_init, NULL)
|
|
PHP_FE(uv_prepare_start, arginfo_uv_prepare_start)
|
|
PHP_FE(uv_prepare_stop, arginfo_uv_prepare_stop)
|
|
/* check (after poll hook) */
|
|
PHP_FE(uv_check_init, arginfo_uv_check_init)
|
|
PHP_FE(uv_check_start, arginfo_uv_check_start)
|
|
PHP_FE(uv_check_stop, arginfo_uv_check_stop)
|
|
/* async */
|
|
PHP_FE(uv_async_init, arginfo_uv_async_init)
|
|
PHP_FE(uv_async_send, arginfo_uv_async_send)
|
|
/* queue (does not work yet) */
|
|
PHP_FE(uv_queue_work, NULL)
|
|
/* fs */
|
|
PHP_FE(uv_fs_open, arginfo_uv_fs_open)
|
|
PHP_FE(uv_fs_read, arginfo_uv_fs_read)
|
|
PHP_FE(uv_fs_write, arginfo_uv_fs_write)
|
|
PHP_FE(uv_fs_close, arginfo_uv_fs_close)
|
|
PHP_FE(uv_fs_fsync, arginfo_uv_fs_fsync)
|
|
PHP_FE(uv_fs_fdatasync, arginfo_uv_fs_ftruncate)
|
|
PHP_FE(uv_fs_ftruncate, arginfo_uv_fs_ftruncate)
|
|
PHP_FE(uv_fs_mkdir, arginfo_uv_fs_mkdir)
|
|
PHP_FE(uv_fs_rmdir, arginfo_uv_fs_rmdir)
|
|
PHP_FE(uv_fs_unlink, arginfo_uv_fs_unlink)
|
|
PHP_FE(uv_fs_rename, arginfo_uv_fs_rename)
|
|
PHP_FE(uv_fs_utime, arginfo_uv_fs_utime)
|
|
PHP_FE(uv_fs_futime, arginfo_uv_fs_futime)
|
|
PHP_FE(uv_fs_chmod, arginfo_uv_fs_chmod)
|
|
PHP_FE(uv_fs_fchmod, arginfo_uv_fs_fchmod)
|
|
PHP_FE(uv_fs_chown, arginfo_uv_fs_chown)
|
|
PHP_FE(uv_fs_fchown, arginfo_uv_fs_fchown)
|
|
PHP_FE(uv_fs_link, arginfo_uv_fs_link)
|
|
PHP_FE(uv_fs_symlink, arginfo_uv_fs_symlink)
|
|
PHP_FE(uv_fs_readlink, arginfo_uv_fs_readlink)
|
|
PHP_FE(uv_fs_stat, arginfo_uv_fs_stat)
|
|
PHP_FE(uv_fs_lstat, arginfo_uv_fs_lstat)
|
|
PHP_FE(uv_fs_fstat, arginfo_uv_fs_fstat)
|
|
PHP_FE(uv_fs_readdir, arginfo_uv_fs_readdir)
|
|
PHP_FE(uv_fs_scandir, arginfo_uv_fs_scandir)
|
|
PHP_FE(uv_fs_sendfile, arginfo_uv_fs_sendfile)
|
|
PHP_FE(uv_fs_event_init, arginfo_uv_fs_event_init)
|
|
/* tty */
|
|
PHP_FE(uv_tty_init, arginfo_uv_tty_init)
|
|
PHP_FE(uv_tty_get_winsize, arginfo_uv_tty_get_winsize)
|
|
PHP_FE(uv_tty_set_mode, NULL)
|
|
PHP_FE(uv_tty_reset_mode, NULL)
|
|
/* info */
|
|
PHP_FE(uv_loadavg, NULL)
|
|
PHP_FE(uv_uptime, NULL)
|
|
PHP_FE(uv_cpu_info, NULL)
|
|
PHP_FE(uv_interface_addresses, NULL)
|
|
PHP_FE(uv_get_free_memory, NULL)
|
|
PHP_FE(uv_get_total_memory, NULL)
|
|
PHP_FE(uv_hrtime, NULL)
|
|
PHP_FE(uv_exepath, NULL)
|
|
PHP_FE(uv_cwd, NULL)
|
|
PHP_FE(uv_chdir, arginfo_uv_chdir)
|
|
PHP_FE(uv_resident_set_memory, NULL)
|
|
/* signal handling */
|
|
PHP_FE(uv_signal_init, arginfo_uv_signal_init)
|
|
PHP_FE(uv_signal_start, arginfo_uv_signal_start)
|
|
PHP_FE(uv_signal_stop, arginfo_uv_signal_stop)
|
|
#ifdef ENABLE_HTTPPARSER
|
|
/* http parser */
|
|
PHP_FE(uv_http_parser_init, arginfo_uv_http_parser_init)
|
|
PHP_FE(uv_http_parser_execute, arginfo_uv_http_parser_execute)
|
|
#endif
|
|
{NULL, NULL, NULL}
|
|
};
|
|
|
|
PHP_MINFO_FUNCTION(uv)
|
|
{
|
|
char uv_version[20];
|
|
// char http_parser_version[20];
|
|
|
|
sprintf(uv_version, "%d.%d",UV_VERSION_MAJOR, UV_VERSION_MINOR);
|
|
// sprintf(http_parser_version, "%d.%d",HTTP_PARSER_VERSION_MAJOR, HTTP_PARSER_VERSION_MINOR);
|
|
|
|
php_printf("PHP libuv Extension\n");
|
|
php_info_print_table_start();
|
|
php_info_print_table_header(2,"libuv Support", "enabled");
|
|
php_info_print_table_row(2,"Version", PHP_UV_EXTVER);
|
|
php_info_print_table_row(2,"libuv Version", uv_version);
|
|
// php_info_print_table_row(2,"http-parser Version", http_parser_version);
|
|
php_info_print_table_end();
|
|
}
|
|
|
|
zend_module_entry uv_module_entry = {
|
|
STANDARD_MODULE_HEADER,
|
|
"uv",
|
|
uv_functions, /* Functions */
|
|
PHP_MINIT(uv), /* MINIT */
|
|
NULL, /* MSHUTDOWN */
|
|
NULL, /* RINIT */
|
|
PHP_RSHUTDOWN(uv), /* RSHUTDOWN */
|
|
PHP_MINFO(uv), /* MINFO */
|
|
PHP_UV_EXTVER,
|
|
STANDARD_MODULE_PROPERTIES
|
|
};
|
|
|
|
|
|
#ifdef COMPILE_DL_UV
|
|
ZEND_GET_MODULE(uv)
|
|
#endif
|