/* +----------------------------------------------------------------------+ | 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 | +----------------------------------------------------------------------+ */ #include "php_uv.h" #include "ext/standard/info.h" #ifndef PHP_UV_DEBUG #define PHP_UV_DEBUG 0 #endif #define PHP_UV_INIT_UV(uv, uv_type) \ uv = (php_uv_t *)emalloc(sizeof(php_uv_t)); \ if (!uv) { \ php_error_docref(NULL TSRMLS_CC, E_ERROR, "emalloc failed"); \ return; \ } 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_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_ZVAL_TO_FD(fd, zstream) \ { \ fd = php_uv_zval_to_fd(zstream TSRMLS_CC); \ if (fd < 0) { \ php_error_docref(NULL TSRMLS_CC, E_WARNING, "invalid variable passed. can't convert to fd."); \ return; \ } \ } #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 TSRMLS_CC, E_ERROR, "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->in_free = 0;\ } #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 extern void php_uv_init(TSRMLS_D); extern zend_class_entry *uv_class_entry; 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; /* static variables */ static uv_loop_t *_php_uv_default_loop; /* resources */ static int uv_resource_handle; static int uv_ares_handle; static int uv_loop_handle; static int uv_sockaddr_handle; static int uv_lock_handle; static int uv_httpparser_handle; static int uv_ares_initialized; /* TODO: fix this */ static char uv_fs_read_buf[8192]; /* declarations */ void php_uv_init(TSRMLS_D); static inline uv_stream_t* php_uv_get_current_stream(php_uv_t *uv); /** * 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); static void php_uv_close_cb(uv_handle_t *handle); void static destruct_uv(zend_rsrc_list_entry *rsrc TSRMLS_DC); 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, 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 uv_buf_t php_uv_read_alloc(uv_handle_t* handle, size_t suggested_size); static void php_uv_close_cb(uv_handle_t *handle); static void php_uv_timer_cb(uv_timer_t *handle, int status); static void php_uv_idle_cb(uv_timer_t *handle, int status); static php_socket_t php_uv_zval_to_fd(zval *ptr TSRMLS_DC) { 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 TSRMLS_CC, E_WARNING, "uv resource does not support yet"); fd = -1; #ifndef PHP_WIN32 } else if (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 TSRMLS_CC, 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; } } return fd; } /** * 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 TSRMLS_DC) { php_uv_t *uv; int r = 0; uv = (php_uv_t *)emalloc(sizeof(php_uv_t)); if (!uv) { php_error_docref(NULL TSRMLS_CC, 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 TSRMLS_CC, 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 TSRMLS_CC, 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 TSRMLS_CC, 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 TSRMLS_CC, 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 TSRMLS_CC, E_ERROR, "uv_prepare_init failed"); goto cleanup; } uv->uv.check.data = uv; } break; } 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_LVAL_P(return_value); } *result = uv; return r; cleanup: efree(uv); return r; } static inline 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; cb = emalloc(sizeof(php_uv_cb_t)); memcpy(&cb->fci, fci, sizeof(zend_fcall_info)); memcpy(&cb->fcc, fcc, sizeof(zend_fcall_info_cache)); if (uv->callback[type] != NULL) { efree(uv->callback[type]); } if (ZEND_FCI_INITIALIZED(*fci)) { Z_ADDREF_P(cb->fci.function_name); #if PHP_VERSION_ID >= 50300 if (fci->object_ptr) { Z_ADDREF_P(cb->fci.object_ptr); } #endif } uv->callback[type] = cb; } /* util */ static void php_uv_ares_destroy() { if (uv_ares_initialized == 1) { ares_library_cleanup(); } } static void php_uv_ares_init(TSRMLS_D) { int rc = 0; if (uv_ares_initialized == 0) { rc = ares_library_init(ARES_LIB_INIT_ALL); if (rc != 0) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "failed to initialize ares library"); return; } uv_ares_initialized = 1; } } static zval *php_uv_address_to_zval(const struct sockaddr *addr) { zval *tmp; char ip[INET6_ADDRSTRLEN]; const struct sockaddr_in *a4; const struct sockaddr_in6 *a6; int port; MAKE_STD_ZVAL(tmp); 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, "address",sizeof("address"), ip, 1); add_assoc_long_ex(tmp, "port", sizeof("port"), port); add_assoc_string_ex(tmp, "family",sizeof("family"), "IPv6", 1); 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, "address",sizeof("address"), ip, 1); add_assoc_long_ex(tmp, "port", sizeof("port"), port); add_assoc_string_ex(tmp, "family",sizeof("family"), "IPv4", 1); break; } default: break; } return tmp; } static zval *php_uv_make_stat(const uv_statbuf_t *s) { zval *tmp; MAKE_STD_ZVAL(tmp); array_init(tmp); add_assoc_long_ex(tmp, "dev", sizeof("dev"), s->st_dev); add_assoc_long_ex(tmp, "ino", sizeof("ino"), s->st_ino); add_assoc_long_ex(tmp, "mode", sizeof("mode"), s->st_mode); add_assoc_long_ex(tmp, "nlink", sizeof("nlink"), s->st_nlink); add_assoc_long_ex(tmp, "uid", sizeof("uid"), s->st_uid); add_assoc_long_ex(tmp, "gid", sizeof("gid"), s->st_gid); add_assoc_long_ex(tmp, "rdev", sizeof("rdev"), s->st_rdev); add_assoc_long_ex(tmp, "size", sizeof("size"), s->st_size); #ifndef PHP_WIN32 add_assoc_long_ex(tmp, "blksize", sizeof("blksize"), s->st_blksize); add_assoc_long_ex(tmp, "blocks", sizeof("blocks"), s->st_blocks); #endif add_assoc_long_ex(tmp, "atime", sizeof("atime"), s->st_atime); add_assoc_long_ex(tmp, "mtime", sizeof("mtime"), s->st_mtime); add_assoc_long_ex(tmp, "ctime", sizeof("ctime"), s->st_ctime); return tmp; } /* destructor */ void static destruct_uv_lock(zend_rsrc_list_entry *rsrc TSRMLS_DC) { 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 TSRMLS_CC, E_NOTICE, "uv_rwlock: unlocked resoruce detected. force rdunlock resource."); uv_rwlock_rdunlock(&lock->lock.rwlock); lock->locked = 0x00; } else if (lock->locked == 0x02) { php_error_docref(NULL TSRMLS_CC, E_NOTICE, "uv_rwlock: unlocked resoruce detected. force wrunlock resource."); uv_rwlock_wrunlock(&lock->lock.rwlock); lock->locked = 0x00; } uv_rwlock_destroy(&lock->lock.rwlock); } else if (lock->type == IS_UV_MUTEX) { if (lock->locked == 0x01) { php_error_docref(NULL TSRMLS_CC, E_NOTICE, "uv_mutex: unlocked resoruce detected. force unlock resource."); uv_mutex_unlock(&lock->lock.mutex); lock->locked = 0x00; } uv_mutex_destroy(&lock->lock.mutex); } else if (lock->type == IS_UV_SEMAPHORE) { if (lock->locked == 0x01) { php_error_docref(NULL TSRMLS_CC, E_NOTICE, "uv_sem: unlocked resoruce detected. force unlock resource."); uv_sem_post(&lock->lock.semaphore); lock->locked = 0x00; } uv_sem_destroy(&lock->lock.semaphore); } efree(lock); } void static destruct_uv_loop(zend_rsrc_list_entry *rsrc TSRMLS_DC) { uv_loop_t *loop = (uv_loop_t *)rsrc->ptr; if (loop != _php_uv_default_loop) { uv_loop_delete(loop); } } void static destruct_uv_sockaddr(zend_rsrc_list_entry *rsrc TSRMLS_DC) { 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_ares(zend_rsrc_list_entry *rsrc TSRMLS_DC) { int base_id = -1; php_uv_ares_t *obj = (php_uv_ares_t *)rsrc->ptr; PHP_UV_DEBUG_PRINT("# will be free: (resource_id: %d)", obj->resource_id); if (obj->gethostbyname_cb) { //fprintf(stderr, "udp_send_cb: %d\n", Z_REFCOUNT_P(obj->listen_cb)); zval_ptr_dtor(&obj->gethostbyname_cb); obj->gethostbyname_cb = NULL; } if (obj != NULL) { efree(obj); obj = NULL; } if (base_id) { //fprintf(stderr,"resource_refcount:%d\n",rsrc->refcount); zend_list_delete(base_id); } } void static destruct_uv(zend_rsrc_list_entry *rsrc TSRMLS_DC) { int base_id = -1, 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) { if (cb->fci.function_name != NULL) { zval_ptr_dtor(&cb->fci.function_name); } if (cb->fci.object_ptr != NULL) { zval_ptr_dtor(&cb->fci.object_ptr); } efree(cb); cb = NULL; } } if (obj->resource_id) { base_id = 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; if(zend_fcall_info_init(callback, 0, &fci, &fcc, NULL, &is_callable_error TSRMLS_CC) == SUCCESS) { if (is_callable_error) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "to be a valid callback"); } } fci.retval_ptr_ptr = retval_ptr; fci.params = params; fci.param_count = param_count; error = zend_call_function(&fci, &fcc TSRMLS_CC); 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; if (ZEND_FCI_INITIALIZED(uv->callback[type]->fci)) { uv->callback[type]->fci.params = params; uv->callback[type]->fci.retval_ptr_ptr = 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 TSRMLS_CC) != SUCCESS) { error = -1; } } else { error = -2; } //zend_fcall_info_args_clear(&uv->callback[type]->fci, 0); return error; } static void php_uv_tcp_connect_cb(uv_connect_t *req, int status) { zval *retval_ptr, *stat, *client= NULL; zval **params[2]; php_uv_t *uv = (php_uv_t*)req->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); MAKE_STD_ZVAL(stat); ZVAL_LONG(stat, status); MAKE_STD_ZVAL(client); ZVAL_RESOURCE(client, uv->resource_id); params[0] = &client; params[1] = &stat; php_uv_do_callback2(&retval_ptr, uv, params, 2, PHP_UV_CONNECT_CB TSRMLS_CC); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } zval_ptr_dtor(&stat); zval_ptr_dtor(&client); efree(req); } static void php_uv_process_close_cb(uv_process_t* process, int exit_status, int term_signal) { zval *retval_ptr, *signal, *stat, *proc= NULL; zval **params[3]; php_uv_t *uv = (php_uv_t*)process->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); MAKE_STD_ZVAL(stat); ZVAL_LONG(stat, exit_status); MAKE_STD_ZVAL(signal); ZVAL_LONG(signal, term_signal); MAKE_STD_ZVAL(proc); ZVAL_RESOURCE(proc, uv->resource_id); params[0] = &proc; params[1] = &stat; params[2] = &signal; php_uv_do_callback2(&retval_ptr, uv, params, 3, PHP_UV_PROC_CLOSE_CB TSRMLS_CC); zval_ptr_dtor(&retval_ptr); zval_ptr_dtor(&stat); zval_ptr_dtor(&proc); zval_ptr_dtor(&signal); } static void php_uv_pipe_connect_cb(uv_connect_t *req, int status) { zval *retval_ptr, *stat, *client= NULL; zval **params[2]; php_uv_t *uv = (php_uv_t*)req->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); MAKE_STD_ZVAL(stat); ZVAL_LONG(stat, status); MAKE_STD_ZVAL(client); ZVAL_RESOURCE(client, uv->resource_id); params[0] = &stat; params[1] = &client; php_uv_do_callback2(&retval_ptr, uv, params, 2, PHP_UV_PIPE_CONNECT_CB TSRMLS_CC); zval_ptr_dtor(&retval_ptr); zval_ptr_dtor(&stat); zval_ptr_dtor(&client); efree(req); } static void php_uv_walk_cb(uv_handle_t* handle, void* arg) { /* zval *retval_ptr, *stat, *client= NULL; zval **params[2]; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); MAKE_STD_ZVAL(stat); ZVAL_LONG(stat, status); MAKE_STD_ZVAL(client); ZVAL_RESOURCE(client, uv->resource_id); params[0] = &stat; params[1] = &client; php_uv_do_callback2(&retval_ptr, uv, params, 2, PHP_UV_PIPE_CONNECT_CB TSRMLS_CC); zval_ptr_dtor(&retval_ptr); zval_ptr_dtor(&stat); zval_ptr_dtor(&client); efree(req); */ } static void php_uv_write_cb(uv_write_t* req, int status) { write_req_t* wr = (write_req_t*) req; zval *retval_ptr, *stat, *client= NULL; zval **params[2]; php_uv_t *uv = (php_uv_t*)req->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); PHP_UV_DEBUG_PRINT("uv_write_cb: status: %d\n", status); MAKE_STD_ZVAL(stat); ZVAL_LONG(stat, status); MAKE_STD_ZVAL(client); ZVAL_RESOURCE(client, uv->resource_id); //zend_list_addref(uv->resource_id); params[0] = &client; params[1] = &stat; php_uv_do_callback2(&retval_ptr, uv, params, 2, PHP_UV_WRITE_CB TSRMLS_CC); zval_ptr_dtor(&retval_ptr); zval_ptr_dtor(&stat); zval_ptr_dtor(&client); if (wr->buf.base) { //free(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_ptr, *stat, *client= NULL; zval **params[2]; php_uv_t *uv = (php_uv_t*)req->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); MAKE_STD_ZVAL(stat); ZVAL_LONG(stat, status); MAKE_STD_ZVAL(client); ZVAL_RESOURCE(client, uv->resource_id); params[0] = &client; params[1] = &stat; php_uv_do_callback2(&retval_ptr, uv, params, 2, PHP_UV_SEND_CB TSRMLS_CC); zval_ptr_dtor(&stat); zval_ptr_dtor(&client); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } if (wr->buf.base) { efree(wr->buf.base); } efree(wr); } static void php_uv_listen_cb(uv_stream_t* server, int status) { zval *retval_ptr, *stat, *svr= NULL; zval **params[2]; php_uv_t *uv = (php_uv_t*)server->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); MAKE_STD_ZVAL(svr); ZVAL_RESOURCE(svr, uv->resource_id); zend_list_addref(uv->resource_id); MAKE_STD_ZVAL(stat); ZVAL_LONG(stat, status); params[0] = &svr; params[1] = &stat; php_uv_do_callback2(&retval_ptr, uv, params, 2, PHP_UV_LISTEN_CB TSRMLS_CC); zval_ptr_dtor(&svr); zval_ptr_dtor(&stat); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } } static void php_uv_close_cb2(uv_handle_t *handle) { php_uv_t *uv = (php_uv_t*)handle->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); PHP_UV_DEBUG_PRINT("uv_close_cb2:\n"); zend_list_delete(uv->resource_id); // maybe we can't call efree at here. //efree(handle); } static void php_uv_shutdown_cb(uv_shutdown_t* handle, int status) { zval *retval_ptr = NULL; zval **params[2]; zval *h, *stat; php_uv_t *uv = (php_uv_t*)handle->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); MAKE_STD_ZVAL(h); ZVAL_RESOURCE(h, uv->resource_id); MAKE_STD_ZVAL(stat); ZVAL_LONG(stat, status); params[0] = &h; params[1] = &stat; php_uv_do_callback2(&retval_ptr, uv, params, 2, PHP_UV_SHUTDOWN_CB TSRMLS_CC); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_shutdown_cb, uv->resource_id); zval_ptr_dtor(&h); zval_ptr_dtor(&stat); efree(handle); } static void php_uv_read_cb(uv_stream_t* handle, ssize_t nread, uv_buf_t buf) { zval *rsc, *buffer, *err, *retval_ptr = NULL; zval **params[3]; php_uv_t *uv = (php_uv_t*)handle->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); PHP_UV_DEBUG_PRINT("uv_read_cb\n"); MAKE_STD_ZVAL(buffer); if (nread > 0) { ZVAL_STRINGL(buffer,buf.base,nread, 1); } else { ZVAL_NULL(buffer); } MAKE_STD_ZVAL(rsc); ZVAL_RESOURCE(rsc, uv->resource_id); zend_list_addref(uv->resource_id); MAKE_STD_ZVAL(err) ZVAL_LONG(err, nread); params[0] = &rsc; params[1] = &err; params[2] = &buffer; php_uv_do_callback2(&retval_ptr, uv, params, 3, PHP_UV_READ_CB TSRMLS_CC); zval_ptr_dtor(&buffer); zval_ptr_dtor(&rsc); zval_ptr_dtor(&err); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } 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 *rsc, *buffer, *err, *pend, *retval_ptr = NULL; zval **params[4]; php_uv_t *uv = (php_uv_t*)handle->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); PHP_UV_DEBUG_PRINT("uv_read2_cb\n"); MAKE_STD_ZVAL(buffer); if (nread > 0) { ZVAL_STRINGL(buffer,buf.base,nread, 1); } else { ZVAL_NULL(buffer); } MAKE_STD_ZVAL(rsc); ZVAL_RESOURCE(rsc, uv->resource_id); //zend_list_addref(uv->resource_id); MAKE_STD_ZVAL(err) ZVAL_LONG(err, nread); MAKE_STD_ZVAL(pend); ZVAL_LONG(pend, pending); params[0] = &rsc; params[1] = &err; params[2] = &buffer; params[3] = &pend; php_uv_do_callback2(&retval_ptr, uv, params, 4, PHP_UV_READ2_CB TSRMLS_CC); zval_ptr_dtor(&buffer); zval_ptr_dtor(&rsc); zval_ptr_dtor(&err); zval_ptr_dtor(&pend); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } 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, int status) { zval *retval_ptr = NULL; zval **params[2]; zval *rsc, *zstat; php_uv_t *uv = (php_uv_t*)handle->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); PHP_UV_DEBUG_PRINT("prepare_cb\n"); MAKE_STD_ZVAL(zstat); ZVAL_LONG(zstat, status); MAKE_STD_ZVAL(rsc); ZVAL_RESOURCE(rsc, uv->resource_id); zend_list_addref(uv->resource_id); params[0] = &rsc; params[1] = &zstat; php_uv_do_callback2(&retval_ptr, uv, params, 2, PHP_UV_PREPARE_CB TSRMLS_CC); zval_ptr_dtor(&rsc); zval_ptr_dtor(&zstat); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_prepare_cb, uv->resource_id); } static void php_uv_check_cb(uv_check_t* handle, int status) { zval *retval_ptr = NULL; zval **params[2]; zval *rsc, *zstat; php_uv_t *uv = (php_uv_t*)handle->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); PHP_UV_DEBUG_PRINT("check_cb\n"); MAKE_STD_ZVAL(zstat); ZVAL_LONG(zstat, status); MAKE_STD_ZVAL(rsc); ZVAL_RESOURCE(rsc, uv->resource_id); zend_list_addref(uv->resource_id); params[0] = &rsc; params[1] = &zstat; php_uv_do_callback2(&retval_ptr, uv, params, 2, PHP_UV_CHECK_CB TSRMLS_CC); zval_ptr_dtor(&rsc); zval_ptr_dtor(&zstat); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_check_cb, uv->resource_id); } static void php_uv_async_cb(uv_async_t* handle, int status) { zval *retval_ptr = NULL; zval **params[2]; zval *zstat, *resource; php_uv_t *uv = (php_uv_t*)handle->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); PHP_UV_DEBUG_PRINT("async_cb\n"); MAKE_STD_ZVAL(zstat); ZVAL_LONG(zstat, status); MAKE_STD_ZVAL(resource); ZVAL_RESOURCE(resource, uv->resource_id); zend_list_addref(uv->resource_id); params[0] = &resource; params[1] = &zstat; php_uv_do_callback2(&retval_ptr, uv, params, 2, PHP_UV_ASYNC_CB TSRMLS_CC); zval_ptr_dtor(&resource); zval_ptr_dtor(&zstat); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_async_cb, uv->resource_id); } static void php_uv_work_cb(uv_work_t* req) { zval *retval_ptr = NULL; php_uv_t *uv = (php_uv_t*)req->data; TSRMLS_FETCH_FROM_CTX(uv != NULL ? uv->thread_ctx : NULL); uv = (php_uv_t*)req->data; PHP_UV_DEBUG_PRINT("work_cb\n"); php_uv_do_callback2(&retval_ptr, uv, NULL, 0, PHP_UV_WORK_CB TSRMLS_CC); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_work_cb, uv->resource_id); } static void php_uv_after_work_cb(uv_work_t* req) { zval *retval_ptr = NULL; php_uv_t *uv = (php_uv_t*)req->data; TSRMLS_FETCH_FROM_CTX(uv != NULL ? uv->thread_ctx : NULL); PHP_UV_DEBUG_PRINT("after_work_cb\n"); php_uv_do_callback2(&retval_ptr, uv, NULL, 0, PHP_UV_AFTER_WORK_CB TSRMLS_CC); zval_ptr_dtor(&retval_ptr); PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_after_work_cb, uv->resource_id); } static void php_uv_fs_cb(uv_fs_t* req) { zval **params[3], *result, *retval_ptr = NULL; 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); MAKE_STD_ZVAL(result); ZVAL_LONG(result, uv->uv.fs.result); params[0] = &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_READDIR: { zval *dirent; int nnames, i = 0; char *namebuf = (char *)req->ptr; MAKE_STD_ZVAL(dirent); array_init(dirent); nnames = req->result; for (i = 0; i < nnames; i++) { add_next_index_string(dirent, namebuf, 1); namebuf += strlen(namebuf) + 1; } params[1] = &dirent; break; } case UV_FS_LSTAT: case UV_FS_FSTAT: case UV_FS_STAT: { zval *buffer; buffer = php_uv_make_stat((const uv_statbuf_t*)req->ptr); params[1] = &buffer; break; } case UV_FS_UTIME: case UV_FS_FUTIME: argc = 0; zval_ptr_dtor(&result); break; case UV_FS_READLINK: { zval *buffer; MAKE_STD_ZVAL(buffer); ZVAL_STRING(buffer, req->ptr, 1); params[1] = &buffer; break; } case UV_FS_READ: { zval *nread,*buffer; argc = 3; MAKE_STD_ZVAL(buffer); MAKE_STD_ZVAL(nread); if (uv->uv.fs.result > 0) { ZVAL_STRINGL(buffer, uv_fs_read_buf, uv->uv.fs.result, 1); } else { ZVAL_NULL(buffer); } ZVAL_LONG(nread, uv->uv.fs.result); params[1] = &nread; params[2] = &buffer; break; } case UV_FS_SENDFILE: { zval *res; argc = 2; MAKE_STD_ZVAL(res); ZVAL_LONG(res, uv->uv.fs.result); params[1] = &res; break; } case UV_FS_WRITE: { zval *res; argc = 2; MAKE_STD_ZVAL(res); ZVAL_LONG(res, uv->uv.fs.result); params[1] = &res; efree(uv->buffer); break; } default: { php_error_docref(NULL TSRMLS_CC, E_ERROR, "type; %d does not support yet.", uv->uv.fs.fs_type); break; } } php_uv_do_callback2(&retval_ptr, uv, params, argc, PHP_UV_FS_CB TSRMLS_CC); PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_fs_cb, uv->resource_id); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } for (i = 0; i < argc; i++) { zval_ptr_dtor(params[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]; zval *name,*ev,*stat,*rsc,*retval_ptr = NULL; 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); MAKE_STD_ZVAL(rsc); MAKE_STD_ZVAL(name); MAKE_STD_ZVAL(ev); MAKE_STD_ZVAL(stat); if (filename) { ZVAL_STRING(name, filename, 1); } else { ZVAL_NULL(name); } ZVAL_LONG(ev, events); ZVAL_LONG(stat, status); ZVAL_RESOURCE(rsc, uv->resource_id); zend_list_addref(uv->resource_id); params[0] = &rsc; params[1] = &name; params[2] = &ev; params[3] = &stat; php_uv_do_callback2(&retval_ptr, uv, params, 4, PHP_UV_FS_EVENT_CB TSRMLS_CC); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_fs_event_cb, uv->resource_id); zval_ptr_dtor(params[0]); zval_ptr_dtor(params[1]); zval_ptr_dtor(params[2]); zval_ptr_dtor(params[3]); } static void php_uv_statbuf_to_zval(zval *result, const uv_statbuf_t *stat) { array_init(result); add_assoc_long_ex(result, "dev", sizeof("dev"), stat->st_dev); add_assoc_long_ex(result, "ino", sizeof("ino"), stat->st_ino); add_assoc_long_ex(result, "mode", sizeof("mode"), stat->st_mode); add_assoc_long_ex(result, "nlink", sizeof("nlink"), stat->st_nlink); add_assoc_long_ex(result, "uid", sizeof("uid"), stat->st_uid); add_assoc_long_ex(result, "gid", sizeof("gid"), stat->st_gid); add_assoc_long_ex(result, "rdev", sizeof("rdev"), stat->st_rdev); add_assoc_long_ex(result, "size", sizeof("size"), stat->st_size); #ifndef PHP_WIN32 add_assoc_long_ex(result, "blksize", sizeof("blksize"), stat->st_blksize); add_assoc_long_ex(result, "blocks", sizeof("blocks"), stat->st_blocks); #endif add_assoc_long_ex(result, "atime", sizeof("atime"), stat->st_atime); add_assoc_long_ex(result, "mtime", sizeof("mtime"), stat->st_mtime); add_assoc_long_ex(result, "ctime", sizeof("ctime"), stat->st_ctime); } static void php_uv_fs_poll_cb(uv_fs_poll_t* handle, int status, const uv_statbuf_t* prev, const uv_statbuf_t* curr) { zval **params[4], *retval_ptr, *rsc, *stat, *p, *c = NULL; php_uv_t *uv = (php_uv_t*)handle->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); MAKE_STD_ZVAL(rsc); ZVAL_RESOURCE(rsc, uv->resource_id); zend_list_addref(uv->resource_id); MAKE_STD_ZVAL(stat); ZVAL_LONG(stat, status); MAKE_STD_ZVAL(p); php_uv_statbuf_to_zval(p, prev); MAKE_STD_ZVAL(c); php_uv_statbuf_to_zval(c, curr); params[0] = &rsc; params[1] = &stat; params[2] = &p; params[3] = &c; php_uv_do_callback2(&retval_ptr, uv, params, 4, PHP_UV_FS_POLL_CB TSRMLS_CC); zval_ptr_dtor(&rsc); zval_ptr_dtor(&stat); zval_ptr_dtor(&p); zval_ptr_dtor(&c); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } } static void php_uv_poll_cb(uv_poll_t* handle, int status, int events) { zval **params[4], *retval_ptr, *rsc, *stat, *ev, *fd = NULL; php_uv_t *uv = (php_uv_t*)handle->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); MAKE_STD_ZVAL(rsc); ZVAL_RESOURCE(rsc, uv->resource_id); zend_list_addref(uv->resource_id); MAKE_STD_ZVAL(stat); ZVAL_LONG(stat, status); MAKE_STD_ZVAL(ev); ZVAL_LONG(ev, events); MAKE_STD_ZVAL(fd); ZVAL_LONG(fd, uv->sock); params[0] = &rsc; params[1] = &stat; params[2] = &ev; params[3] = &fd; php_uv_do_callback2(&retval_ptr, uv, params, 4, PHP_UV_POLL_CB TSRMLS_CC); zval_ptr_dtor(&rsc); zval_ptr_dtor(&stat); zval_ptr_dtor(&ev); zval_ptr_dtor(&fd); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } } static void php_uv_udp_recv_cb(uv_udp_t* handle, ssize_t nread, uv_buf_t buf, struct sockaddr* addr, unsigned flags) { /* TODO: is this implment correct? */ zval *retval_ptr = NULL; zval **params[3]; zval *buffer, *rsc, *read; php_uv_t *uv = (php_uv_t*)handle->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); MAKE_STD_ZVAL(buffer); ZVAL_STRINGL(buffer,buf.base,nread, 1); MAKE_STD_ZVAL(rsc); ZVAL_RESOURCE(rsc, uv->resource_id); MAKE_STD_ZVAL(read); ZVAL_LONG(read, nread); params[0] = &rsc; params[1] = &read; params[2] = &buffer; php_uv_do_callback2(&retval_ptr, uv, params, 3, PHP_UV_RECV_CB TSRMLS_CC); zval_ptr_dtor(&buffer); zval_ptr_dtor(&rsc); zval_ptr_dtor(&read); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } if (buf.base) { efree(buf.base); } } static uv_buf_t php_uv_read_alloc(uv_handle_t* handle, size_t suggested_size) { return uv_buf_init(emalloc(suggested_size), suggested_size); } static void php_uv_close_cb(uv_handle_t *handle) { zval *retval_ptr = NULL; zval **params[1]; zval *h; php_uv_t *uv = (php_uv_t*)handle->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); MAKE_STD_ZVAL(h); ZVAL_RESOURCE(h, uv->resource_id); params[0] = &h; php_uv_do_callback2(&retval_ptr, uv, params, 1, PHP_UV_CLOSE_CB TSRMLS_CC); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_close_cb, uv->resource_id); zend_hash_index_del(&EG(regular_list), uv->resource_id); zval_ptr_dtor(&h); /* call destruct_uv */ } static void php_uv_idle_cb(uv_timer_t *handle, int status) { zval *retval_ptr, *idle, *stat = NULL; zval **params[2]; php_uv_t *uv = (php_uv_t*)handle->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); MAKE_STD_ZVAL(idle); ZVAL_RESOURCE(idle, uv->resource_id); zend_list_addref(uv->resource_id); MAKE_STD_ZVAL(stat); ZVAL_LONG(stat, status); params[0] = &idle; params[1] = &stat; php_uv_do_callback2(&retval_ptr, uv, params, 2, PHP_UV_IDLE_CB TSRMLS_CC); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } zval_ptr_dtor(&idle); zval_ptr_dtor(&stat); } static void php_uv_getaddrinfo_cb(uv_getaddrinfo_t* handle, int status, struct addrinfo* res) { zval *tmp, *retval_ptr, *stat = NULL; zval **params[2]; 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); MAKE_STD_ZVAL(stat); ZVAL_LONG(stat, status); params[0] = &stat; MAKE_STD_ZVAL(tmp); array_init(tmp); 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(tmp, c, 1); } 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(tmp, c, 1); } address = address->ai_next; } params[1] = &tmp; php_uv_do_callback2(&retval_ptr, uv, params, 2, PHP_UV_GETADDR_CB TSRMLS_CC); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } zval_ptr_dtor(&stat); zval_ptr_dtor(&tmp); zend_list_delete(uv->resource_id); uv_freeaddrinfo(res); } static void php_uv_timer_cb(uv_timer_t *handle, int status) { zval *retval_ptr, *stat, *client= NULL; zval **params[2]; php_uv_t *uv = (php_uv_t*)handle->data; TSRMLS_FETCH_FROM_CTX(uv->thread_ctx); MAKE_STD_ZVAL(stat); ZVAL_LONG(stat, status); MAKE_STD_ZVAL(client); ZVAL_RESOURCE(client, uv->resource_id); zend_list_addref(uv->resource_id); params[0] = &client; params[1] = &stat; php_uv_do_callback2(&retval_ptr, uv, params, 2, PHP_UV_TIMER_CB TSRMLS_CC); if (retval_ptr != NULL) { zval_ptr_dtor(&retval_ptr); } zval_ptr_dtor(&stat); zval_ptr_dtor(&client); } 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 TSRMLS_CC, E_ERROR, "unexpected type found"); break; } } return stream; } void static destruct_httpparser(zend_rsrc_list_entry *rsrc TSRMLS_DC) { http_parser *obj = (http_parser *)rsrc->ptr; efree(obj); } /* http parser callbacks */ static int on_message_begin(http_parser *p) { return 0; } static int on_headers_complete(http_parser *p) { return 0; } static int on_message_complete(http_parser *p) { php_http_parser_context *result = p->data; result->finished = 1; return 0; } #define PHP_HTTP_PARSER_PARSE_URL(flag, name) \ if (result->handle.field_set & (1 << flag)) { \ const char *name = at+result->handle.field_data[flag].off; \ int length = result->handle.field_data[flag].len; \ add_assoc_stringl(data, #name, (char*)name, length, 1); \ } static int on_url_cb(http_parser *p, const char *at, size_t len) { php_http_parser_context *result = p->data; zval *data = result->data; http_parser_parse_url(at, len, 0, &result->handle); add_assoc_stringl(data, "QUERY_STRING", (char*)at, len, 1); PHP_HTTP_PARSER_PARSE_URL(UF_SCHEMA, scheme); PHP_HTTP_PARSER_PARSE_URL(UF_HOST, host); PHP_HTTP_PARSER_PARSE_URL(UF_PORT, port); PHP_HTTP_PARSER_PARSE_URL(UF_PATH, path); PHP_HTTP_PARSER_PARSE_URL(UF_QUERY, query); PHP_HTTP_PARSER_PARSE_URL(UF_FRAGMENT, fragment); return 0; } static int header_field_cb(http_parser *p, const char *at, size_t len) { php_http_parser_context *result = p->data; /* TODO: */ result->tmp = estrndup(at, len); return 0; } static int header_value_cb(http_parser *p, const char *at, size_t len) { php_http_parser_context *result = p->data; zval *data = result->headers; add_assoc_stringl(data, result->tmp, (char*)at, len, 1); /* TODO: */ efree(result->tmp); result->tmp = NULL; return 0; } static int on_body_cb(http_parser *p, const char *at, size_t len) { php_http_parser_context *result = p->data; zval *data = result->headers; add_assoc_stringl(data, "body", (char*)at, len, 1); return 0; } /* end of callback */ /* common functions */ static void php_uv_ip_common(int ip_type, INTERNAL_FUNCTION_PARAMETERS) { int error = 0; zval *address; php_uv_sockaddr_t *addr; char ip[INET6_ADDRSTRLEN]; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z",&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 (addr->is_ipv4 != 1) { RETURN_FALSE; } error = uv_ip4_name(&addr->addr.ipv4, ip, INET6_ADDRSTRLEN); RETVAL_STRING(ip,1); } else if (ip_type == 2) { if (addr->is_ipv4 == 1) { RETURN_FALSE; } error = uv_ip6_name(&addr->addr.ipv6, ip, INET6_ADDRSTRLEN); RETVAL_STRING(ip,1); } } static void php_uv_socket_bind(int 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 > 2) { if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "zz|l",&resource, &address, &flags) == FAILURE) { return; } } else { if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "zz",&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 == 1) { r = uv_tcp_bind((uv_tcp_t*)&uv->uv.tcp, addr->addr.ipv4); if (r) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "bind failed"); } } else if (ip_type == 2) { r = uv_tcp_bind6((uv_tcp_t*)&uv->uv.tcp, addr->addr.ipv6); if (r) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "bind failed"); } } else if (ip_type == 3) { r = uv_udp_bind((uv_udp_t*)&uv->uv.udp, addr->addr.ipv4, flags); if (r) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "uv_udp_bind failed"); } } else if (ip_type == 4) { r = uv_udp_bind6((uv_udp_t*)&uv->uv.udp, addr->addr.ipv6, flags); if (r) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "uv_udp_bind6 failed"); } } } static void php_uv_socket_getname(int type, INTERNAL_FUNCTION_PARAMETERS) { php_uv_t *uv; zval *handle, *result; int addr_len, error = 0; struct sockaddr_storage addr; addr_len = sizeof(struct sockaddr_storage); if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t*, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); switch (type) { case 1: error = uv_tcp_getsockname(&uv->uv.tcp, (struct sockaddr*)&addr, &addr_len); break; case 2: error = uv_tcp_getpeername(&uv->uv.tcp, (struct sockaddr*)&addr, &addr_len); break; case 3: error = uv_udp_getsockname(&uv->uv.udp, (struct sockaddr*)&addr, &addr_len); 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; char *data; int data_len = 0; 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() TSRMLS_CC, "zszf!",&z_cli, &data, &data_len, &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); zend_list_addref(client->resource_id); w = emalloc(sizeof(send_req_t)); w->req.data = client; w->buf = uv_buf_init(estrndup(data,data_len), 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, addr->addr.ipv4, php_uv_udp_send_cb); } else if (type == 2) { uv_udp_send6(&w->req, &client->uv.udp, &w->buf, 1, addr->addr.ipv6, php_uv_udp_send_cb); } } static void php_uv_tcp_connect(int 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() TSRMLS_CC, "zzf",&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); zend_list_addref(uv->resource_id); req = (uv_connect_t*)emalloc(sizeof(uv_connect_t)); php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_CONNECT_CB); req->data = uv; if (type == 1) { uv_tcp_connect(req, &uv->uv.tcp, addr->addr.ipv4, php_uv_tcp_connect_cb); } else { uv_tcp_connect6(req, &uv->uv.tcp, addr->addr.ipv6, php_uv_tcp_connect_cb); } } /* zend */ PHP_MINIT_FUNCTION(uv) { PHP_UV_PROBE(MINIT); php_uv_init(TSRMLS_C); uv_resource_handle = zend_register_list_destructors_ex(destruct_uv, NULL, PHP_UV_RESOURCE_NAME, module_number); uv_ares_handle = zend_register_list_destructors_ex(destruct_uv_ares, NULL, PHP_UV_ARES_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_httpparser_handle = zend_register_list_destructors_ex(destruct_httpparser, NULL, PHP_UV_HTTPPARSER_RESOURCE_NAME, module_number); return SUCCESS; } PHP_RSHUTDOWN_FUNCTION(uv) { php_uv_ares_destroy(); 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_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_last_error, 0, 0, 1) ZEND_ARG_INFO(0, loop) 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, 1) 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_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_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_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, 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_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() /* PHP Functions */ /* {{{ proto void uv_unref(resource $uv_t) */ PHP_FUNCTION(uv_unref) { zval *handle = NULL; php_uv_t *uv; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z",&handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); uv_unref((uv_handle_t *)php_uv_get_current_stream(uv)); zend_list_delete(uv->resource_id); } /* }}} */ /* {{{ proto long uv_last_error([resource $uv_loop]) */ PHP_FUNCTION(uv_last_error) { uv_loop_t *loop; uv_err_t err; zval *zloop = NULL; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|z",&zloop) == FAILURE) { return; } PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop); err = uv_last_error(loop); RETVAL_LONG(err.code); } /* }}} */ /* {{{ proto string uv_err_name(long $error_code) */ PHP_FUNCTION(uv_err_name) { long error_code; const char *error_msg; uv_err_t error; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l",&error_code) == FAILURE) { return; } error.code = error_code; error_msg = uv_err_name(error); RETVAL_STRING(error_msg,1); } /* }}} */ /* {{{ proto string uv_strerror(long $error_code) */ PHP_FUNCTION(uv_strerror) { long error_code; const char *error_msg; uv_err_t error; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l",&error_code) == FAILURE) { return; } error.code = error_code; error_msg = uv_strerror(error); RETVAL_STRING(error_msg,1); } /* }}} */ /* {{{ 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() TSRMLS_CC, "z",&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; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z",&handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); uv_ref((uv_handle_t *)php_uv_get_current_stream(uv)); zend_list_addref(uv->resource_id); } /* }}} */ /* {{{ proto void uv_run([resource $uv_loop]) */ PHP_FUNCTION(uv_run) { zval *zloop = NULL; uv_loop_t *loop; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|z",&zloop) == FAILURE) { return; } PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop); uv_run(loop); } /* }}} */ /* {{{ proto void 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() TSRMLS_CC, "|z",&zloop) == FAILURE) { return; } PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop); uv_run_once(loop); } /* }}} */ /* {{{ proto void uv_loop_delete(resource $uv_loop) */ PHP_FUNCTION(uv_loop_delete) { zval *z_loop = NULL; uv_loop_t *loop; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z",&z_loop) == FAILURE) { return; } if (z_loop != NULL) { ZEND_FETCH_RESOURCE(loop, uv_loop_t *, &z_loop, -1, PHP_UV_LOOP_RESOURCE_NAME, uv_loop_handle); if (loop != _php_uv_default_loop) { uv_loop_delete(loop); } } } /* }}} */ /* {{{ proto long uv_now(resource $uv_loop) */ PHP_FUNCTION(uv_now) { zval *z_loop = NULL; uv_loop_t *loop; int64_t now; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z",&z_loop) == FAILURE) { return; } if (z_loop != NULL) { ZEND_FETCH_RESOURCE(loop, uv_loop_t *, &z_loop, -1, PHP_UV_LOOP_RESOURCE_NAME, uv_loop_handle); now = uv_now(loop); RETURN_LONG((long)now); } } /* }}} */ /* {{{ proto void uv_tcp_bind(resource $uv_sockaddr) */ PHP_FUNCTION(uv_tcp_bind) { php_uv_socket_bind(1, INTERNAL_FUNCTION_PARAM_PASSTHRU); } /* }}} */ /* {{{ proto void uv_tcp_bind6(resource $uv_sockaddr) */ PHP_FUNCTION(uv_tcp_bind6) { php_uv_socket_bind(2, INTERNAL_FUNCTION_PARAM_PASSTHRU); } /* }}} */ /* {{{ proto void uv_write(resource $handle, string $data, callable $callback) ##### *Description* send buffer to speicified uv resource. ##### *Parameters* *resource $handle*: uv resources (uv_tcp, uv_udp, uv_pipe ...etc.) *string $data*: buffer. *callable $callback*: callable variables. this callback expects (resource $handle, long $status) ##### *Return Value* *void*: ##### *Example* */ PHP_FUNCTION(uv_write) { zval *z_cli; char *data; int r, data_len = 0; 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() TSRMLS_CC, "zsf",&z_cli, &data, &data_len, &fci, &fcc) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &z_cli, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); zend_list_addref(uv->resource_id); php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_WRITE_CB); w = emalloc(sizeof(write_req_t)); w->req.data = uv; w->buf = uv_buf_init(data, 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 TSRMLS_CC, E_ERROR, "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; char *data; int r, data_len = 0; 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() TSRMLS_CC, "zszf",&z_cli, &data, &data_len, &z_send, &fci, &fcc) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &z_cli, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); ZEND_FETCH_RESOURCE(send, php_uv_t *, &z_send, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); zend_list_addref(uv->resource_id); php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_WRITE_CB); w = emalloc(sizeof(write_req_t)); w->req.data = uv; w->buf = uv_buf_init(data, 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 TSRMLS_CC, E_ERROR, "write2 failed"); } PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_write2, uv->resource_id); } /* }}} */ /* {{{ proto void uv_tcp_nodelay(resource $handle, bool $flag) */ PHP_FUNCTION(uv_tcp_nodelay) { zval *z_cli; php_uv_t *client; long bval = 1; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "zl",&z_cli, &bval) == FAILURE) { return; } ZEND_FETCH_RESOURCE(client, php_uv_t *, &z_cli, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); 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() TSRMLS_CC, "zz",&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); 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 TSRMLS_CC, E_NOTICE, "accept"); } } /* }}} */ /* {{{ proto void uv_shutdown(resource $handle, callable $callback) ##### *Description* shutdown uv handle. ##### *Parameters* *resource $handle*: uv resources (uv_tcp, uv_udp, uv_pipe ...etc.) *callable $callback*: callable variables. this callback expects (resource $handle, long $status) ##### *Return Value* *void*: ##### *Example* */ 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() TSRMLS_CC, "r|f!",&client, &fci, &fcc) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_SHUTDOWN_CB); zend_list_addref(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 TSRMLS_CC, E_ERROR, "%s (ERRNO: %d)", uv_strerror(uv_last_error(uv_default_loop())), r); } } /* }}} */ /* {{{ proto void uv_close(resource $handle, callable $callback) ##### *Description* close uv handle. ##### *Parameters* *resource $handle*: uv resources (uv_tcp, uv_udp, uv_pipe ...etc.) *callable $callback*: callable variables. this callback expects (resource $handle, long $status) ##### *Return Value* *void*: ##### *Example* */ 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() TSRMLS_CC, "r|f!",&client, &fci, &fcc) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_CLOSE_CB); zend_list_addref(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) ##### *Description* starts read callback for uv resources. ##### *Parameters* *resource $handle*: uv resources (uv_tcp, uv_udp, uv_pipe ...etc.) *callable $callback*: callable variables. this callback expects (resource $handle, long $nread, string buffer) ##### *Return Value* *void*: ##### *Example* ##### *Note* * You have to handle erorrs correctly. otherwise this will leak. * if you want to use PHP's stream or socket resource. see uv_fs_poll_init and uv_fs_read. */ 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() TSRMLS_CC, "rf!",&client, &fci, &fcc) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); zend_list_addref(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 { php_error_docref(NULL TSRMLS_CC, 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 TSRMLS_CC, 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) { 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() TSRMLS_CC, "rf",&client, &fci, &fcc) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); zend_list_addref(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 { php_error_docref(NULL TSRMLS_CC, 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 TSRMLS_CC, 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() TSRMLS_CC, "r", &server) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &server, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); 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) { char *address; int address_len = 0; long port = 0; php_uv_sockaddr_t *sockaddr; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "sl",&address, &address_len, &port) == FAILURE) { return; } sockaddr = (php_uv_sockaddr_t*)emalloc(sizeof(php_uv_sockaddr_t)); sockaddr->is_ipv4 = 1; sockaddr->addr.ipv4 = uv_ip4_addr(address, port); ZEND_REGISTER_RESOURCE(return_value, sockaddr, uv_sockaddr_handle); sockaddr->resource_id = Z_LVAL_P(return_value); } /* }}} */ /* {{{ proto resource uv_ip6_addr(string $ipv6_addr, long $port) */ PHP_FUNCTION(uv_ip6_addr) { char *address; int address_len = 0; long port = 0; php_uv_sockaddr_t *sockaddr; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "sl",&address, &address_len, &port) == FAILURE) { return; } sockaddr = (php_uv_sockaddr_t*)emalloc(sizeof(php_uv_sockaddr_t)); sockaddr->is_ipv4 = 0; sockaddr->addr.ipv6 = uv_ip6_addr(address, port); ZEND_REGISTER_RESOURCE(return_value, sockaddr, uv_sockaddr_handle); sockaddr->resource_id = Z_LVAL_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() TSRMLS_CC, "zlf",&resource, &backlog, &fci, &fcc) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &resource, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); 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 TSRMLS_CC, E_ERROR, "%s", uv_strerror(uv_last_error(uv_default_loop()))); } } /* }}} */ /* {{{ proto void uv_tcp_connect(resource $handle, resource $ipv4_addr, callable $callback) ##### *Description* connect to specified ip address and port. ##### *Parameters* *resource $handle*: requires `uv_tcp_init()` resource. *resource $ipv4_addr*: requires uv_sockaddr resource. *callable $callback*: callable variables. ##### *Return Value* *void*: ##### *Example* ````php uv.timer); if (r) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "uv_timer_init failed"); return; } uv->type = IS_UV_TIMER; uv->uv.timer.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_LVAL_P(return_value); } /* }}} */ /* {{{ proto void uv_timer_start(resource $timer, long $timeout, long $repeat, callable $callback) ##### *Description* initialize timer handle. ##### *Parameters* *resource $loop*: uv_loop resource. ##### *Return Value* *resource $timer*: initialized timer resource. ##### *Example* ````php 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) ##### *Description* stop specified timer. ##### *Parameters* *resource $timer*: uv timer resource. ##### *Return Value* *void*: ##### *Example* ````php uv.timer); } /* }}} */ /* {{{ proto void uv_timer_again(resource $timer) ##### *Description* restart timer. ##### *Parameters* *resource $timer*: uv_timer resource. ##### *Return Value* *void*: ##### *Example* */ PHP_FUNCTION(uv_timer_again) { zval *timer; php_uv_t *uv; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r",&timer) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &timer, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); 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() TSRMLS_CC, "rl",&timer,&repeat) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &timer, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); 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() TSRMLS_CC, "r",&timer) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &timer, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); repeat = uv_timer_get_repeat((uv_timer_t*)&uv->uv.timer); RETURN_LONG(repeat); } /* }}} */ /* {{{ proto void uv_idle_start(resource $idle, callable $callback) ##### *Description* start idle callback. ##### *Parameters* *resource $idle*: uv_idle resource. *callable $callback*: idle callback. ##### *Return Value* *void*: ##### *Example* ````php 3) { uv_idle_stop($idle); } sleep(1); }); uv_run(); ```` */ 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; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rf",&idle, &fci, &fcc) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &idle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); zend_list_addref(uv->resource_id); php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_IDLE_CB); uv_idle_start((uv_idle_t*)&uv->uv.idle, (uv_idle_cb)php_uv_idle_cb); } /* }}} */ /* {{{ 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}; char *node, *service; int node_len, service_len = 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() TSRMLS_CC, "zfss|a",&z_loop, &fci, &fcc, &node, &node_len, &service, &service_len, &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 (zend_hash_find(h, "ai_family", sizeof("ai_family"), (void **)&data) == SUCCESS) { hint.ai_family = Z_LVAL_PP(data); } if (zend_hash_find(h, "ai_socktype", sizeof("ai_socktype"), (void **)&data) == SUCCESS) { hint.ai_socktype = Z_LVAL_PP(data); } if (zend_hash_find(h, "ai_protocol", sizeof("ai_socktype"), (void **)&data) == SUCCESS) { hint.ai_socktype = Z_LVAL_PP(data); } if (zend_hash_find(h, "ai_flags", sizeof("ai_flags"), (void **)&data) == SUCCESS) { hint.ai_flags = Z_LVAL_PP(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, service, &hint); } /* }}} */ /* {{{ proto void uv_idle_stop(resource $idle) ##### *Description* stop idle callback. ##### *Parameters* *resource $idle*: uv_idle resource. ##### *Return Value* *void*: ##### *Example* ````php 3) { uv_idle_stop($idle); } sleep(1); }); uv_run(); ```` */ PHP_FUNCTION(uv_idle_stop) { zval *idle; php_uv_t *uv; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &idle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &idle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); uv_idle_stop((uv_idle_t*)&uv->uv.idle); } /* }}} */ /* {{{ proto resource uv_tcp_init([resource $loop]) ##### *Description* create a tcp socket. ##### *Parameters* *resource $loop*: loop resource or null. if not specified loop resource then use uv_default_loop resource. ##### *Return Value* *resource php_uv*: uv resource which initialized for tcp. ##### *Example* ````php 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 TSRMLS_CC, 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() TSRMLS_CC, "r", &client) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); 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; char *multicast_addr, interface_addr; int error, multicast_addr_len, interface_addr_len = 0; long membership; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rssl", &client, &multicast_addr, &multicast_addr_len, &interface_addr, &interface_addr_len, &membership) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); error = uv_udp_set_membership((uv_udp_t*)&uv->uv.udp, (const char*)multicast_addr, (const char*)interface_addr, (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() TSRMLS_CC, "rl",&client, &enabled) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); r = uv_udp_set_multicast_loop((uv_udp_t*)&uv->uv.udp, enabled); if (r) { php_error_docref(NULL TSRMLS_CC, 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() TSRMLS_CC, "rl",&client, &ttl) == FAILURE) { return; } if (ttl > 255) { php_error_docref(NULL TSRMLS_CC, E_NOTICE, "uv_udp_set_muticast_ttl: ttl parameter expected less than 255."); ttl = 255; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); r = uv_udp_set_multicast_ttl((uv_udp_t*)&uv->uv.udp, ttl); if (r) { php_error_docref(NULL TSRMLS_CC, 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() TSRMLS_CC, "rl",&client, &enabled) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &client, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); r = uv_udp_set_broadcast((uv_udp_t*)&uv->uv.udp, enabled); if (r) { php_error_docref(NULL TSRMLS_CC, 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() TSRMLS_CC, "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() TSRMLS_CC, "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() TSRMLS_CC, "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() TSRMLS_CC, "rf|a",&zloop, &fci, &fcc, &opaque) == FAILURE) { return; } php_error_docref(NULL TSRMLS_CC, E_ERROR, "uv_walk does not support yet"); PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop); uv_walk(loop, php_uv_walk_cb, NULL); } /* }}} */ /* {{{ 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; long ipc = 0; int r; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|z|l", &zloop, &ipc) == FAILURE) { return; } uv = (php_uv_t *)emalloc(sizeof(php_uv_t)); if (!uv) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "uv_pipe_init emalloc failed"); return; } PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop); uv->type = IS_UV_PIPE; r = uv_pipe_init(loop, &uv->uv.pipe, ipc); if (r) { php_error_docref(NULL TSRMLS_CC, 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_LVAL_P(return_value); } /* }}} */ /* {{{ proto void uv_pipe_open(resource $handle, long $pipe) */ PHP_FUNCTION(uv_pipe_open) { php_uv_t *uv; zval *handle; /* TODO: `pipe` correct? */ long pipe = 0; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "zl",&handle, &pipe) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); 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; char *name; int error, name_len = 0; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "zs",&handle, &name, &name_len) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); error = uv_pipe_bind(&uv->uv.pipe, name); if (error) { php_error_docref(NULL TSRMLS_CC, E_WARNING, "%s", uv_strerror(uv_last_error(uv_default_loop()))); } 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; char *name; int name_len = 0; 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() TSRMLS_CC, "zsf",&resource,&name, &name_len, &fci, &fcc) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &resource, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); zend_list_addref(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, 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() TSRMLS_CC, "zl",&handle, &count) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); uv_pipe_pending_instances(&uv->uv.pipe, count); } /* }}} */ static void php_ares_gethostbyname_cb( void *arg, int status, int timeouts, struct hostent *hostent) { TSRMLS_FETCH(); zval *retval_ptr, *hostname, *addresses = NULL; zval **params[2]; php_uv_ares_t *uv = (php_uv_ares_t*)arg; struct in_addr **ptr; MAKE_STD_ZVAL(hostname); ZVAL_STRING(hostname, hostent->h_name, 1); MAKE_STD_ZVAL(addresses); array_init(addresses); ptr = (struct in_addr **)hostent->h_addr_list; while(*ptr != NULL) { add_next_index_string(addresses, inet_ntoa(**(ptr++)), 1); } params[0] = &hostname; params[1] = &addresses; php_uv_do_callback(&retval_ptr, uv->gethostbyname_cb, params, 2 TSRMLS_CC); zval_ptr_dtor(&retval_ptr); zval_ptr_dtor(&hostname); zval_ptr_dtor(&addresses); } /* {{{ proto resource uv_ares_init_options(resource $loop, array $options, long $optmask) */ PHP_FUNCTION(uv_ares_init_options) { int rc, length; int optmask = ARES_OPT_SERVERS | ARES_OPT_TCP_PORT | ARES_OPT_LOOKUPS | ARES_OPT_FLAGS; zval **data, *zoptions, *zloop = NULL; uv_loop_t *loop = NULL; php_uv_ares_t *uv; HashTable *h; struct in_addr *addresses; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "zal",&zloop,&zoptions, &optmask) == FAILURE) { return; } PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop); uv = (php_uv_ares_t*)emalloc(sizeof(php_uv_ares_t)); uv->gethostbyname_cb = NULL; h = Z_ARRVAL_P(zoptions); if (zend_hash_find(h, "servers", sizeof("servers"), (void **)&data) == SUCCESS) { HashTable *servers = Z_ARRVAL_P(*data); HashPosition pos; char *key; int key_type; uint key_len; ulong key_index; int i = 0; length = zend_hash_num_elements(servers); addresses = (struct in_addr*)ecalloc(length, sizeof(struct in_addr)); for (zend_hash_internal_pointer_reset_ex(servers, &pos); (key_type = zend_hash_get_current_key_ex(servers, &key, &key_len, &key_index, 0, &pos)) != HASH_KEY_NON_EXISTANT; zend_hash_move_forward_ex(servers, &pos)) { struct sockaddr_in address; zval **value; zend_hash_get_current_data_ex(servers, (void *) &value, &pos); if (Z_TYPE_PP(value) != IS_STRING) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "servers value must be an array"); } address = uv_ip4_addr(Z_STRVAL_PP(value),0); addresses[i] = address.sin_addr; } } if (zend_hash_find(h, "port", sizeof("port"), (void **)&data) == SUCCESS) { uv->options.tcp_port = htonl(Z_LVAL_PP(data)); } if (zend_hash_find(h, "lookups", sizeof("lookups"), (void **)&data) == SUCCESS) { uv->options.lookups = Z_STRVAL_PP(data); } uv->options.servers = addresses; uv->options.nservers = length; uv->options.flags = ARES_FLAG_USEVC; if (uv_ares_initialized == 0) { php_uv_ares_init(TSRMLS_C); } rc = uv_ares_init_options(loop, &uv->channel, &uv->options, optmask); if (rc) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "uv_ares_init_options failed"); } efree(addresses); ZEND_REGISTER_RESOURCE(return_value, uv, uv_ares_handle); uv->resource_id = Z_LVAL_P(return_value); } /* {{{ proto void ares_gethostbyname(resource $handle, string $name, long $flag, callable $callback) */ PHP_FUNCTION(ares_gethostbyname) { zval *handle, *byname_cb; long flag = AF_INET; char *name; int name_len; php_uv_ares_t *uv; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "zslz",&handle, &name, &name_len, &flag, &byname_cb) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_ares_t *, &handle, -1, PHP_UV_ARES_RESOURCE_NAME, uv_ares_handle); if (uv->gethostbyname_cb != NULL) { zval_ptr_dtor(&uv->gethostbyname_cb); uv->gethostbyname_cb = NULL; } Z_ADDREF_P(byname_cb); uv->gethostbyname_cb = byname_cb; ares_gethostbyname(uv->channel, name, flag, &php_ares_gethostbyname_cb, uv ); } /* }}} */ /* {{{ proto array uv_loadavg(void) */ PHP_FUNCTION(uv_loadavg) { zval *retval; double average[3]; uv_loadavg(average); MAKE_STD_ZVAL(retval); array_init(retval); add_next_index_double(retval, average[0]); add_next_index_double(retval, average[1]); add_next_index_double(retval, average[2]); RETURN_ZVAL(retval,0,1); } /* }}} */ /* {{{ proto double uv_uptime(void) */ PHP_FUNCTION(uv_uptime) { uv_err_t error; double uptime; error = uv_uptime(&uptime); RETURN_DOUBLE(uptime); } /* }}} */ /* {{{ proto long uv_get_free_memory(void) */ PHP_FUNCTION(uv_get_free_memory) { RETURN_LONG(uv_get_free_memory()); } /* }}} */ /* {{{ proto long uv_get_total_memory(void) */ PHP_FUNCTION(uv_get_total_memory) { RETURN_LONG(uv_get_total_memory()); } /* }}} */ /* {{{ proto long uv_hrtime(void) */ PHP_FUNCTION(uv_hrtime) { /* TODO: is this correct? */ RETURN_LONG(uv_hrtime()); } /* }}} */ /* {{{ proto string uv_exepath(void) */ PHP_FUNCTION(uv_exepath) { char buffer[1024] = {0}; size_t buffer_sz; buffer_sz = sizeof(buffer); uv_exepath(buffer, &buffer_sz); buffer[buffer_sz] = '\0'; RETURN_STRINGL(buffer, buffer_sz, 1); } /* }}} */ /* {{{ proto string uv_cwd(void) */ PHP_FUNCTION(uv_cwd) { char buffer[1024] = {0}; size_t buffer_sz = sizeof(buffer); uv_cwd(buffer, buffer_sz); buffer[buffer_sz] = '\0'; RETURN_STRING(buffer, 1); } /* }}} */ /* {{{ proto array uv_cpu_info(void) */ PHP_FUNCTION(uv_cpu_info) { zval *retval; uv_cpu_info_t *cpus; uv_err_t error; int i, count; error = uv_cpu_info(&cpus, &count); if (UV_OK == error.code) { MAKE_STD_ZVAL(retval); array_init(retval); for (i = 0; i < count; i++) { zval *tmp, *times; MAKE_STD_ZVAL(tmp); MAKE_STD_ZVAL(times); array_init(tmp); array_init(times); add_assoc_string_ex(tmp, "model", sizeof("model"), cpus[i].model, 1); add_assoc_long_ex(tmp, "speed", sizeof("speed"), cpus[i].speed); add_assoc_long_ex(times, "sys", sizeof("sys"), (size_t)cpus[i].cpu_times.sys); add_assoc_long_ex(times, "user", sizeof("user"), (size_t)cpus[i].cpu_times.user); add_assoc_long_ex(times, "idle", sizeof("idle"), (size_t)cpus[i].cpu_times.idle); add_assoc_long_ex(times, "irq", sizeof("irq"), (size_t)cpus[i].cpu_times.irq); add_assoc_long_ex(times, "nice", sizeof("nice"), (size_t)cpus[i].cpu_times.nice); add_assoc_zval_ex(tmp, "times", sizeof("times"), times); add_next_index_zval(retval,tmp); } uv_free_cpu_info(cpus, count); RETURN_ZVAL(retval,0,1); } } /* }}} */ /* {{{ proto array uv_interface_addresses(void) */ PHP_FUNCTION(uv_interface_addresses) { zval *retval; uv_interface_address_t *interfaces; uv_err_t error; char buffer[512]; int i, count; error = uv_interface_addresses(&interfaces, &count); if (UV_OK == error.code) { MAKE_STD_ZVAL(retval); array_init(retval); for (i = 0; i < count; i++) { zval *tmp; MAKE_STD_ZVAL(tmp); array_init(tmp); add_assoc_string_ex(tmp, "name", sizeof("name"), interfaces[i].name , 1); add_assoc_bool_ex(tmp, "is_internal", sizeof("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, "address", sizeof("address"), buffer, 1); add_next_index_zval(retval,tmp); } uv_free_interface_addresses(interfaces, count); RETURN_ZVAL(retval,0,1); } } /* }}} */ /* {{{ proto resource uv_spawn(resource $loop, string $command, array $args, array $context, callable $callback) */ PHP_FUNCTION(uv_spawn) { zval *zloop = NULL; uv_loop_t *loop; uv_process_options_t options = {0}; uv_stdio_container_t stdio[3]; php_uv_t *proc; zval *args, *context; char **zenv; char *command; char **command_args; int command_len = 0; zend_fcall_info fci = empty_fcall_info; zend_fcall_info_cache fcc = empty_fcall_info_cache; php_uv_cb_t *cb; options.stdio = stdio; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "zsaaf", &zloop, &command, &command_len, &args, &context, &fci, &fcc) == FAILURE) { return; } PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop); { HashTable *h; zval **data; h = Z_ARRVAL_P(context); if (zend_hash_find(h, "cwd", sizeof("cwd"), (void **)&data) == SUCCESS) { options.cwd = Z_STRVAL_PP(data); } if (zend_hash_find(h, "env", sizeof("env"), (void **)&data) == SUCCESS) { HashTable *env; HashPosition pos; char *key; int key_type; uint key_len; ulong key_index; int i = 0; env = Z_ARRVAL_P(*data); zenv = emalloc(sizeof(char*) * (zend_hash_num_elements(env)+1)); for (zend_hash_internal_pointer_reset_ex(env, &pos); (key_type = zend_hash_get_current_key_ex(env, &key, &key_len, &key_index, 0, &pos)) != HASH_KEY_NON_EXISTANT; zend_hash_move_forward_ex(env, &pos)) { zval **value; char *hoge; zend_hash_get_current_data_ex(env, (void *) &value, &pos); hoge = emalloc(sizeof(char)*key_len+1+Z_STRLEN_PP(value)); slprintf(hoge,key_len+1+Z_STRLEN_PP(value),"%s=%s",key, Z_STRVAL_PP(value)); zenv[i] = hoge; i++; } zenv[i] = NULL; options.env = zenv; } if (zend_hash_find(h, "pipes", sizeof("pipes"), (void **)&data) == SUCCESS) { HashTable *pipes; HashPosition pos; char *key; int key_type; uint key_len; ulong key_index; pipes = Z_ARRVAL_P(*data); options.stdio_count = zend_hash_num_elements(pipes); for (zend_hash_internal_pointer_reset_ex(pipes, &pos); (key_type = zend_hash_get_current_key_ex(pipes, &key, &key_len, &key_index, 0, &pos)) != HASH_KEY_NON_EXISTANT; zend_hash_move_forward_ex(pipes, &pos)) { zval **value; php_uv_t *pipe; zend_hash_get_current_data_ex(pipes, (void *) &value, &pos); if (Z_TYPE_PP(value) != IS_RESOURCE) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "must be uv_pipe resource"); } ZEND_FETCH_RESOURCE(pipe, php_uv_t *, value, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); if (pos->h == 0) { options.stdio[0].flags = UV_CREATE_PIPE | UV_READABLE_PIPE; options.stdio[0].data.stream = (uv_stream_t *)&pipe->uv.pipe; } else if (pos->h == 1) { options.stdio[1].flags = UV_CREATE_PIPE | UV_WRITABLE_PIPE; options.stdio[1].data.stream = (uv_stream_t *)&pipe->uv.pipe; } else if (pos->h == 2) { options.stdio[2].flags = UV_CREATE_PIPE | UV_WRITABLE_PIPE; options.stdio[2].data.stream = (uv_stream_t *)&pipe->uv.pipe; } } } } { HashTable *h; HashPosition pos; char *key; int key_type; uint key_len; ulong key_index; int hash_len = 0; h = Z_ARRVAL_P(args); hash_len = zend_hash_num_elements(h)+1; command_args = ecalloc(hash_len+1, sizeof(char**)); command_args[0] = options.cwd; for (zend_hash_internal_pointer_reset_ex(h, &pos); (key_type = zend_hash_get_current_key_ex(h, &key, &key_len, &key_index, 0, &pos)) != HASH_KEY_NON_EXISTANT; zend_hash_move_forward_ex(h, &pos)) { zval **value; zend_hash_get_current_data_ex(h, (void *) &value, &pos); command_args[pos->h+1] = Z_STRVAL_PP(value); } command_args[hash_len] = NULL; } 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); options.file = command; if (command_args) { options.args = command_args; } options.exit_cb = php_uv_process_close_cb; proc->type = IS_UV_PROCESS; proc->uv.process.data = proc; ZEND_REGISTER_RESOURCE(return_value, proc, uv_resource_handle); proc->resource_id = Z_LVAL_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) { efree(command_args); } } /* }}} */ /* {{{ proto void uv_process_kill(resource $handle, long $signal) TODO: */ PHP_FUNCTION(uv_process_kill) { php_uv_t *uv; zval *handle; int signal; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "zl", &handle, &signal) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); 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() TSRMLS_CC, "ll", &pid, &signal) == FAILURE) { return; } uv_kill(pid, signal); } /* }}} */ /* {{{ proto bool uv_chdir(string $directory) */ PHP_FUNCTION(uv_chdir) { uv_err_t error; char *directory; int directory_len; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s", &directory, &directory_len) == FAILURE) { return; } error = uv_chdir(directory); if (error.code == UV_OK) { RETURN_TRUE; } else { RETURN_FALSE; } } /* }}} */ /* {{{ proto resource uv_rwlock_init(void) */ PHP_FUNCTION(uv_rwlock_init) { php_uv_lock_t *lock; int error; lock = emalloc(sizeof(php_uv_lock_t)); error = uv_rwlock_init(&lock->lock.rwlock); if (error == 0) { ZEND_REGISTER_RESOURCE(return_value, lock, uv_lock_handle); lock->type = IS_UV_RWLOCK; } else { efree(lock); RETURN_FALSE; } } /* }}} */ /* {{{ proto uv_rwlock_rdlock(resource $handle) */ PHP_FUNCTION(uv_rwlock_rdlock) { php_uv_lock_t *lock; zval *handle; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(lock, php_uv_lock_t *, &handle, -1, PHP_UV_LOCK_RESOURCE_NAME, uv_lock_handle); lock->locked = 0x01; uv_rwlock_rdlock(&lock->lock.rwlock); } /* }}} */ /* {{{ proto bool uv_rwlock_tryrdlock(resource $handle) */ PHP_FUNCTION(uv_rwlock_tryrdlock) { php_uv_lock_t *lock; zval *handle; int error = 0; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(lock, php_uv_lock_t *, &handle, -1, PHP_UV_LOCK_RESOURCE_NAME, uv_lock_handle); error = uv_rwlock_tryrdlock(&lock->lock.rwlock); if (error == 0) { lock->locked = 0x01; RETURN_TRUE; } else { RETURN_FALSE; } } /* }}} */ /* {{{ proto void uv_rwlock_rdunlock(resource $handle) */ PHP_FUNCTION(uv_rwlock_rdunlock) { php_uv_lock_t *lock; zval *handle; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(lock, php_uv_lock_t *, &handle, -1, PHP_UV_LOCK_RESOURCE_NAME, uv_lock_handle); if (lock->locked == 0x01) { uv_rwlock_rdunlock(&lock->lock.rwlock); lock->locked = 0x00; } } /* }}} */ /* {{{ proto uv_rwlock_wrlock(resource $handle) */ PHP_FUNCTION(uv_rwlock_wrlock) { php_uv_lock_t *lock; zval *handle; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(lock, php_uv_lock_t *, &handle, -1, PHP_UV_LOCK_RESOURCE_NAME, uv_lock_handle); lock->locked = 0x02; uv_rwlock_wrlock(&lock->lock.rwlock); } /* }}} */ /* {{{ proto uv_rwlock_trywrlock(resource $handle) */ PHP_FUNCTION(uv_rwlock_trywrlock) { php_uv_lock_t *lock; zval *handle; int error = 0; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(lock, php_uv_lock_t *, &handle, -1, PHP_UV_LOCK_RESOURCE_NAME, uv_lock_handle); error = uv_rwlock_trywrlock(&lock->lock.rwlock); if (error == 0) { lock->locked = 0x02; RETURN_TRUE; } else { RETURN_FALSE; } } /* }}} */ /* {{{ proto uv_rwlock_wrunlock(resource $handle) */ PHP_FUNCTION(uv_rwlock_wrunlock) { php_uv_lock_t *lock; zval *handle; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(lock, php_uv_lock_t *, &handle, -1, PHP_UV_LOCK_RESOURCE_NAME, uv_lock_handle); if (lock->locked == 0x02) { uv_rwlock_wrunlock(&lock->lock.rwlock); lock->locked = 0x00; } } /* }}} */ /* {{{ proto uv_lock uv_mutex_init(void) */ PHP_FUNCTION(uv_mutex_init) { php_uv_lock_t *mutex; int error; mutex = emalloc(sizeof(php_uv_lock_t)); error = uv_mutex_init(&mutex->lock.mutex); if (error == 0) { ZEND_REGISTER_RESOURCE(return_value, mutex, uv_lock_handle); mutex->type = IS_UV_MUTEX; } else { efree(mutex); RETURN_FALSE; } } /* }}} */ /* {{{ proto void uv_mutex_lock(uv_lock $lock)*/ PHP_FUNCTION(uv_mutex_lock) { php_uv_lock_t *mutex; zval *handle; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(mutex, php_uv_lock_t*, &handle, -1, PHP_UV_LOCK_RESOURCE_NAME, uv_lock_handle); uv_mutex_lock(&mutex->lock.mutex); mutex->locked = 0x01; } /* }}} */ /* {{{ proto bool uv_mutex_trylock(uv_lock $lock) */ PHP_FUNCTION(uv_mutex_trylock) { php_uv_lock_t *mutex; zval *handle; int error = 0; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(mutex, php_uv_lock_t *, &handle, -1, PHP_UV_LOCK_RESOURCE_NAME, uv_lock_handle); error = uv_mutex_trylock(&mutex->lock.mutex); if (error == 0) { mutex->locked = 0x01; RETURN_TRUE; } else { RETURN_FALSE; } } /* }}} */ /* {{{ void uv_mutex_unlock(uv_lock $lock) */ PHP_FUNCTION(uv_mutex_unlock) { php_uv_lock_t *mutex; zval *handle; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(mutex, php_uv_lock_t *, &handle, -1, PHP_UV_LOCK_RESOURCE_NAME, uv_lock_handle); if (mutex->locked == 0x01) { uv_mutex_unlock(&mutex->lock.mutex); mutex->locked = 0x00; } } /* }}} */ /* {{{ proto uv_lock uv_sem_init(void) */ PHP_FUNCTION(uv_sem_init) { php_uv_lock_t *semaphore; int error = 0; unsigned long val = 0; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &val) == FAILURE) { return; } semaphore = emalloc(sizeof(php_uv_lock_t)); error = uv_sem_init(&semaphore->lock.semaphore, val); if (error == 0) { ZEND_REGISTER_RESOURCE(return_value, semaphore, uv_lock_handle); semaphore->type = IS_UV_SEMAPHORE; } else { efree(semaphore); RETURN_FALSE; } } /* }}} */ /* {{{ proto void uv_sem_post(uv_lock $sem) */ PHP_FUNCTION(uv_sem_post) { php_uv_lock_t *lock; zval *handle; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(lock, php_uv_lock_t *, &handle, -1, PHP_UV_LOCK_RESOURCE_NAME, uv_lock_handle); uv_sem_post(&lock->lock.semaphore); } /* }}} */ /* {{{ proto void uv_sem_wait(uv_lock $sem) */ PHP_FUNCTION(uv_sem_wait) { php_uv_lock_t *lock; zval *handle; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(lock, php_uv_lock_t *, &handle, -1, PHP_UV_LOCK_RESOURCE_NAME, uv_lock_handle); uv_sem_wait(&lock->lock.semaphore); } /* }}} */ /* {{{ proto void uv_sem_trywait(uv_lock $sem) */ PHP_FUNCTION(uv_sem_trywait) { php_uv_lock_t *lock; zval *handle; int error = 0; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "z", &handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(lock, php_uv_lock_t *, &handle, -1, PHP_UV_LOCK_RESOURCE_NAME, uv_lock_handle); error = uv_sem_trywait(&lock->lock.semaphore); RETURN_LONG(error); } /* }}} */ /* {{{ 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() TSRMLS_CC, "|z",&zloop) == FAILURE) { return; } PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop); php_uv_common_init(&uv, loop, IS_UV_PREPARE, return_value TSRMLS_CC); } /* }}} */ /* {{{ 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() TSRMLS_CC, "rf",&handle, &fci, &fcc) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); zend_list_addref(uv->resource_id); if(uv->type == IS_UV_PREPARE) { uv->uv.prepare.data = uv; } else { php_error_docref(NULL TSRMLS_CC, E_NOTICE, "this type does not support yet"); } 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); if (r) { php_error_docref(NULL TSRMLS_CC, E_NOTICE, "read failed"); } PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_prepare_start, uv->resource_id); } /* }}} */ /* {{{ proto void uv_prepare_stop(resource $handle) */ PHP_FUNCTION(uv_prepare_stop) { zval *handle; php_uv_t *uv; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "r", &handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); uv_prepare_stop((uv_prepare_t*)php_uv_get_current_stream(uv)); PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_prepare_stop, uv->resource_id); } /* }}} */ /* {{{ 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() TSRMLS_CC, "|z",&zloop) == FAILURE) { return; } PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop); php_uv_common_init(&uv, loop, IS_UV_CHECK, return_value TSRMLS_CC); } /* }}} */ /* {{{ 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() TSRMLS_CC, "rf", &handle, &fci, &fcc) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); zend_list_addref(uv->resource_id); if(uv->type == IS_UV_CHECK) { uv->uv.check.data = uv; } else { php_error_docref(NULL TSRMLS_CC, E_NOTICE, "this type does not support yet"); } 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); if (r) { php_error_docref(NULL TSRMLS_CC, E_NOTICE, "read failed"); } PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_check_start, uv->resource_id); } /* }}} */ /* {{{ 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() TSRMLS_CC, "r", &handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); 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() TSRMLS_CC, "zf",&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 TSRMLS_CC, 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_RESOURCE(return_value, uv->resource_id); zend_list_addref(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() TSRMLS_CC, "r", &handle) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); 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) { 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() TSRMLS_CC, "zff",&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 TSRMLS_CC, E_ERROR, "uv_queue_work failed"); return; } } /* }}} */ /* {{{ proto resource uv_fs_open(resource $loop, string $path, long $flag, long $mode, callable $callback) */ PHP_FUNCTION(uv_fs_open) { zval *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; char *path; int r, path_len = 0; long flag, mode; 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() TSRMLS_CC, "zsllf!", &zloop, &path, &path_len, &flag, &mode, &fci, &fcc) == FAILURE) { return; } PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop); PHP_UV_INIT_UV(uv, IS_UV_FS); php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_FS_CB); uv->uv.fs.data = uv; r = uv_fs_open(loop, (uv_fs_t*)&uv->uv.fs, path, flag, mode, php_uv_fs_cb); if (r) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "uv_async_init failed"); return; } } /* }}} */ /* {{{ proto void uv_fs_read(resoruce $loop, zval $fd, callable $callback) */ PHP_FUNCTION(uv_fs_read) { int r; zval *zstream, *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; unsigned long fd; 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() TSRMLS_CC, "zzf", &zloop, &zstream, &fci, &fcc) == FAILURE) { return; } PHP_UV_ZVAL_TO_FD(fd, zstream); PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop); PHP_UV_INIT_UV(uv, IS_UV_FS) php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_FS_CB); uv->uv.fs.data = uv; memset(uv_fs_read_buf, 0, sizeof(uv_fs_read_buf)); r = uv_fs_read(loop, (uv_fs_t*)&uv->uv.fs, fd, uv_fs_read_buf, sizeof(uv_fs_read_buf), -1, php_uv_fs_cb); if (r) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "uv_async_init failed"); return; } } /* }}} */ /* {{{ proto void uv_fs_close(resource $loop, zval $fd, callable $callback) */ PHP_FUNCTION(uv_fs_close) { int r; zval *zstream,*zloop = NULL; uv_loop_t *loop; php_uv_t *uv; unsigned long fd; 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() TSRMLS_CC, "zzf", &zloop, &zstream, &fci, &fcc) == FAILURE) { return; } PHP_UV_ZVAL_TO_FD(fd, zstream); PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop); PHP_UV_INIT_UV(uv, IS_UV_FS); php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_FS_CB); uv->uv.fs.data = uv; memset(uv_fs_read_buf, 0, sizeof(uv_fs_read_buf)); r = uv_fs_close(loop, (uv_fs_t*)&uv->uv.fs, fd, php_uv_fs_cb); if (r) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "uv_fs_close failed"); return; } } /* }}} */ /* {{{ proto void uv_fs_write(resource $loop, zval $fd, string $buffer, long $offset, callable $callback) */ PHP_FUNCTION(uv_fs_write) { int r; zval *zstream, *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; char *buffer; int buffer_len = 0; long fd, offset = -1; 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() TSRMLS_CC, "zzslf", &zloop, &zstream, &buffer, &buffer_len, &offset, &fci, &fcc) == FAILURE) { return; } PHP_UV_ZVAL_TO_FD(fd, zstream); PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop); PHP_UV_INIT_UV(uv, IS_UV_FS); php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_FS_CB); uv->uv.fs.data = uv; uv->buffer = estrndup(buffer, buffer_len); r = uv_fs_write(loop, (uv_fs_t*)&uv->uv.fs, fd, uv->buffer, buffer_len, offset, php_uv_fs_cb); if (r) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "uv_fs_write failed"); return; } } /* }}} */ /* {{{ proto void uv_fs_fsync(resource $loop, zval $fd, callable $callback) */ PHP_FUNCTION(uv_fs_fsync) { int error; zval *zstream, *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; unsigned long fd; 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() TSRMLS_CC, "zzf!", &zloop, &zstream, &fci, &fcc) == FAILURE) { return; } PHP_UV_ZVAL_TO_FD(fd, zstream); 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; PHP_UV_FS_ASYNC(loop, fsync, fd); } /* }}} */ /* {{{ proto void uv_fs_fdatasync(resource $loop, zval $fd, callable $callback) */ PHP_FUNCTION(uv_fs_fdatasync) { int error; zval *zstream, *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; unsigned long fd; 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() TSRMLS_CC, "zzf", &zloop, &zstream, &fci, &fcc) == FAILURE) { return; } PHP_UV_ZVAL_TO_FD(fd, zstream); 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; PHP_UV_FS_ASYNC(loop, fdatasync, fd); } /* }}} */ /* {{{ proto void uv_fs_ftruncate(resource $loop, zval $fd, long $offset, callable $callback) */ PHP_FUNCTION(uv_fs_ftruncate) { int error; zval *zstream,*zloop = NULL; uv_loop_t *loop; php_uv_t *uv; long offset = 0; unsigned long fd; 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() TSRMLS_CC, "zzlf", &zloop, &zstream, &offset, &fci, &fcc) == FAILURE) { return; } PHP_UV_ZVAL_TO_FD(fd, zstream); 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; PHP_UV_FS_ASYNC(loop, ftruncate, fd, offset); } /* }}} */ /* {{{ proto void uv_fs_mkdir(resource $loop, string $path, long $mode, callable $callback) */ PHP_FUNCTION(uv_fs_mkdir) { int error; zval *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; char *path; int path_len = 0; long mode = 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() TSRMLS_CC, "zslf", &zloop, &path, &path_len, &mode, &fci, &fcc) == FAILURE) { return; } 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; PHP_UV_FS_ASYNC(loop, mkdir, path, mode); } /* }}} */ /* {{{ proto void uv_fs_rmdir(resource $loop, string $path, callable $callback) */ PHP_FUNCTION(uv_fs_rmdir) { int error; zval *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; char *path; int path_len = 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() TSRMLS_CC, "zsf!", &zloop, &path, &path_len, &fci, &fcc) == FAILURE) { return; } 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; PHP_UV_FS_ASYNC(loop, rmdir, path); } /* }}} */ /* {{{ proto void uv_fs_unlink(resource $loop, string $path, callable $callback) */ PHP_FUNCTION(uv_fs_unlink) { int error; zval *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; char *path; int path_len = 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() TSRMLS_CC, "zsf!", &zloop, &path, &path_len, &fci, &fcc) == FAILURE) { return; } 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; PHP_UV_FS_ASYNC(loop, unlink, path); } /* }}} */ /* {{{ proto void uv_fs_rename(resource $loop, string $from, string $to, callable $callback) */ PHP_FUNCTION(uv_fs_rename) { int error; zval *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; char *from, *to; int from_len, to_len = 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() TSRMLS_CC, "zssf!", &zloop, &from, &from_len, &to, &to_len, &fci, &fcc) == FAILURE) { return; } 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; PHP_UV_FS_ASYNC(loop, rename, from, to); } /* }}} */ /* {{{ proto void uv_fs_utime(resource $loop, string $path, long $utime, long $atime, callable $callback) */ PHP_FUNCTION(uv_fs_utime) { int error; zval *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; long utime, atime; char *path; int path_len = 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() TSRMLS_CC, "zsllf", &zloop, &path, &path_len, &utime, &atime, &fci, &fcc) == FAILURE) { return; } 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; PHP_UV_FS_ASYNC(loop, utime, path, utime, atime); } /* }}} */ /* {{{ proto void uv_fs_futime(resource $loop, zval $fd, long $utime, long $atime callable $callback) */ PHP_FUNCTION(uv_fs_futime) { int error; zval *zstream, *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; long utime, atime; unsigned long fd; 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() TSRMLS_CC, "zzllf", &zloop, &zstream, &utime, &atime, &fci, &fcc) == FAILURE) { return; } PHP_UV_ZVAL_TO_FD(fd, zstream); 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; PHP_UV_FS_ASYNC(loop, futime, fd, utime, atime); } /* }}} */ /* {{{ proto void uv_fs_chmod(resource $loop, string $path, long $mode, callable $callback) */ PHP_FUNCTION(uv_fs_chmod) { int error; zval *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; long mode; char *path; int path_len = 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() TSRMLS_CC, "zslf", &zloop, &path, &path_len, &mode, &fci, &fcc) == FAILURE) { return; } 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; PHP_UV_FS_ASYNC(loop, chmod, path, mode); } /* }}} */ /* {{{ proto void uv_fs_fchmod(resource $loop, zval $fd, long $mode, callable $callback) */ PHP_FUNCTION(uv_fs_fchmod) { int error; zval *zstream,*zloop = NULL; uv_loop_t *loop; php_uv_t *uv; long mode; unsigned long fd; 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() TSRMLS_CC, "zzllf!", &zloop, &zstream, &mode, &fci, &fcc) == FAILURE) { return; } PHP_UV_ZVAL_TO_FD(fd, zstream); 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; PHP_UV_FS_ASYNC(loop, fchmod, fd, mode); } /* }}} */ /* {{{ proto void uv_fs_chown(resource $loop, string $path, long $uid, long $gid, callable $callback) */ PHP_FUNCTION(uv_fs_chown) { int error; zval *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; long uid, gid; char *path; int path_len = 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() TSRMLS_CC, "zslf", &zloop, &path, &path_len, &uid, &gid, &fci, &fcc) == FAILURE) { return; } 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; PHP_UV_FS_ASYNC(loop, chown, path, uid, gid); } /* }}} */ /* {{{ proto void uv_fs_fchown(resource $loop, zval $fd, long $uid, $long $gid, callable $callback) */ PHP_FUNCTION(uv_fs_fchown) { int error; zval *zstream, *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; long uid, gid; unsigned long fd; 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() TSRMLS_CC, "zzllf!", &zloop, &zstream, &uid, &gid, &fci, &fcc) == FAILURE) { return; } PHP_UV_ZVAL_TO_FD(fd, zstream); 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; PHP_UV_FS_ASYNC(loop, fchown, fd, uid, gid); } /* }}} */ /* {{{ proto void uv_fs_link(resource $loop, string $from, string $to, callable $callback) */ PHP_FUNCTION(uv_fs_link) { int error; zval *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; char *from, *to; int from_len, to_len = 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() TSRMLS_CC, "zssf", &zloop, &from, &from_len, &to, &to_len, &fci, &fcc) == FAILURE) { return; } 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; PHP_UV_FS_ASYNC(loop, link, from, to); } /* }}} */ /* {{{ proto void uv_fs_symlink(resource $loop, string $from, string $to, long $flags, callable $callback) */ PHP_FUNCTION(uv_fs_symlink) { int error; zval *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; char *from, *to; int from_len, to_len = 0; long flags; 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() TSRMLS_CC, "zsslf", &zloop, &from, &from_len, &to, &to_len, &flags, &fci, &fcc) == FAILURE) { return; } 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; PHP_UV_FS_ASYNC(loop, symlink, from, to, flags); } /* }}} */ /* {{{ proto void uv_fs_readlink(resource $loop, string $path, callable $callback) */ PHP_FUNCTION(uv_fs_readlink) { int error; zval *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; char *path; int path_len = 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() TSRMLS_CC, "zsf", &zloop, &path, &path_len, &fci, &fcc) == FAILURE) { return; } 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; PHP_UV_FS_ASYNC(loop, readlink, path); } /* }}} */ /* {{{ proto void uv_fs_stat(resource $loop, string $path, callable $callback) */ PHP_FUNCTION(uv_fs_stat) { int error; zval *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; char *path; int path_len = 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() TSRMLS_CC, "zsf", &zloop, &path, &path_len, &fci, &fcc) == FAILURE) { return; } 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; PHP_UV_FS_ASYNC(loop, stat, path); } /* }}} */ /* {{{ proto void uv_fs_lstat(resource $loop, string $path, callable $callback) */ PHP_FUNCTION(uv_fs_lstat) { int error; zval *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; char *path; int path_len = 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() TSRMLS_CC, "zsf", &zloop, &path, &path_len, &fci, &fcc) == FAILURE) { return; } 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; PHP_UV_FS_ASYNC(loop, lstat, path); } /* }}} */ /* {{{ proto void uv_fs_fstat(resource $loop, zval $fd, callable $callback) */ PHP_FUNCTION(uv_fs_fstat) { int error; zval *zstream, *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; unsigned long fd; 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() TSRMLS_CC, "zzf", &zloop, &zstream, &fci, &fcc) == FAILURE) { return; } PHP_UV_ZVAL_TO_FD(fd, zstream); 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; PHP_UV_FS_ASYNC(loop, fstat, fd); } /* }}} */ /* {{{ proto uv_fs_readdir(resource $loop, string $path, long $flags, callable $callback) */ PHP_FUNCTION(uv_fs_readdir) { int error; zval *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; char *path; int path_len = 0; long flags; 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() TSRMLS_CC, "zslf!", &zloop, &path, &path_len, &flags, &fci, &fcc) == FAILURE) { return; } 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; PHP_UV_FS_ASYNC(loop, readdir, path, flags); } /* }}} */ /* {{{ proto void uv_fs_sendfile(resource $loop, zval $in_fd, zval $out_fd, long $offset, long $length, callable $callback) */ PHP_FUNCTION(uv_fs_sendfile) { int error; zval *z_instream, *z_outstream, *zloop = NULL; uv_loop_t *loop; php_uv_t *uv; unsigned long in_fd, out_fd; long offset, length = 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() TSRMLS_CC, "zzzllf!", &zloop, &z_instream, &z_outstream, &offset, &length, &fci, &fcc) == FAILURE) { return; } PHP_UV_ZVAL_TO_FD(in_fd, z_instream); PHP_UV_ZVAL_TO_FD(out_fd, z_outstream); 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; PHP_UV_FS_ASYNC(loop, sendfile, in_fd, out_fd, offset, length); } /* }}} */ /* {{{ 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; char *path; int path_len = 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() TSRMLS_CC, "zsfl", &zloop, &path, &path_len, &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; error = uv_fs_event_init(loop, (uv_fs_event_t*)&uv->uv.fs_event, path, php_uv_fs_event_cb, flags); if (error) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "uv_fs_event_init 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() TSRMLS_CC, "zzl", &zloop, &zstream, &readable) == FAILURE) { return; } PHP_UV_ZVAL_TO_FD(fd, zstream); PHP_UV_INIT_UV(uv, IS_UV_TTY); PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop); uv->uv.tty.data = uv; error = uv_tty_init(loop, (uv_tty_t*)&uv->uv.tty, fd, readable); \ if (error) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "uv_fs_event_init failed"); \ return; } ZVAL_RESOURCE(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() TSRMLS_CC, "zzz", &handle, &w, &h) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t*, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); error = uv_tty_get_winsize(&uv->uv.tty, &width, &height); 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() TSRMLS_CC, "zl", &handle, &mode) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t*, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); 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) { 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() TSRMLS_CC, "zl", &handle, &enable) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t*, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); 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; 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() TSRMLS_CC, "zz", &zloop, &zstream) == FAILURE) { return; } PHP_UV_ZVAL_TO_FD(fd, zstream); PHP_UV_INIT_UV(uv, IS_UV_POLL); PHP_UV_FETCH_UV_DEFAULT_LOOP(loop, zloop); error = uv_poll_init(loop, &uv->uv.poll, fd); if (error) { php_error_docref(NULL TSRMLS_CC, E_ERROR, "uv_poll_init failed"); return; } uv->sock = fd; ZVAL_RESOURCE(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() TSRMLS_CC, "zlf", &handle, &events, &fci, &fcc) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_POLL_CB); uv->uv.poll.data = uv; zend_list_addref(uv->resource_id); error = uv_poll_start(&uv->uv.poll, events, php_uv_poll_cb); if (error) { php_error_docref(NULL TSRMLS_CC, 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() TSRMLS_CC, "r", &poll) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &poll, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); 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() TSRMLS_CC, "z", &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 TSRMLS_CC, E_ERROR, "uv_fs_poll_init failed"); return; } ZVAL_RESOURCE(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; char *path; unsigned long interval = 0; int error, path_len = 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() TSRMLS_CC, "zfsl", &handle, &fci, &fcc, &path, &path_len, &interval) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &handle, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); php_uv_cb_init(&cb, uv, &fci, &fcc, PHP_UV_FS_POLL_CB); uv->uv.fs_poll.data = uv; zend_list_addref(uv->resource_id); error = uv_fs_poll_start(&uv->uv.fs_poll, php_uv_fs_poll_cb, (const char*)path, interval); if (error) { php_error_docref(NULL TSRMLS_CC, 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() TSRMLS_CC, "r", &poll) == FAILURE) { return; } ZEND_FETCH_RESOURCE(uv, php_uv_t *, &poll, -1, PHP_UV_RESOURCE_NAME, uv_resource_handle); uv_fs_poll_stop(&uv->uv.fs_poll); PHP_UV_DEBUG_RESOURCE_REFCOUNT(uv_fs_poll_stop, uv->resource_id); } /* }}} */ /* HTTP PARSER */ ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_http_parser_init, 0, 0, 1) ZEND_ARG_INFO(0, target) ZEND_END_ARG_INFO() ZEND_BEGIN_ARG_INFO_EX(arginfo_uv_http_parser_execute, 0, 0, 3) ZEND_ARG_INFO(0, resource) ZEND_ARG_INFO(0, buffer) ZEND_ARG_INFO(0, setting) ZEND_END_ARG_INFO() /* {{{ proto resource uv_http_parser_init(long $target = UV::HTTP_REQUEST) ##### *Description* initialize http parser. ##### *Parameters* *long $target*: this expects UV::HTTP_REQUEST, UV::HTTP_RESPONSE or UV::HTTP_BOTH. ##### *Return Value* *resource uv_http*: ##### *Example* ````php parser, target); if (target == HTTP_RESPONSE) { ctx->is_response = 1; } else { ctx->is_response = 0; } memset(&ctx->handle, 0, sizeof(struct http_parser_url)); /* setup callback */ ctx->settings.on_message_begin = on_message_begin; ctx->settings.on_header_field = header_field_cb; ctx->settings.on_header_value = header_value_cb; ctx->settings.on_url = on_url_cb; ctx->settings.on_body = on_body_cb; ctx->settings.on_headers_complete = on_headers_complete; ctx->settings.on_message_complete = on_message_complete; ZEND_REGISTER_RESOURCE(return_value, ctx, uv_httpparser_handle); } /* {{{ proto bool uv_http_parser_execute(resource $parser, string $body, array &$result) ##### *Description* execute http parser. ##### *Parameters* *resource $parser*: uv_http_parser resoruce. *string $body*: http message. *array &$result*: result array. ##### *Return Value* *bool finished*: this parser returns false when specified http message is invalid or not enough message. ##### *Example* ````php // array(8) { // ["Host"]=> // string(10) "chobie.net" // ["User-Agent"]=> // string(81) "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.7; rv:12.0) Gecko/20100101 Firefox/12.0" // ["Accept-Language"]=> // string(14) "en-us,en;q=0.5" // ["Accept-Encoding"]=> // string(13) "gzip, deflate" // ["Connection"]=> // string(10) "keep-alive" // ["Referer"]=> // string(18) "http://chobie.net/" // ["Cookie"]=> // string(9) "key=value" // ["Cache-Control"]=> // string(9) "max-age=0" // } // ["QUERY_STRING"]=> // string(35) "/img/http-parser.png?key=value#frag" // ["path"]=> // string(20) "/img/http-parser.png" // ["query"]=> // string(9) "key=value" // ["fragment"]=> // string(4) "frag" // ["REQUEST_METHOD"]=> // string(3) "GET" //} } ```` */ PHP_FUNCTION(uv_http_parser_execute) { zval *z_parser,*result, *headers; php_http_parser_context *context; char *body; int body_len; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "rs/a",&z_parser, &body, &body_len, &result) == FAILURE) { return; } ZEND_FETCH_RESOURCE(context, php_http_parser_context*, &z_parser, -1, PHP_UV_HTTPPARSER_RESOURCE_NAME, uv_httpparser_handle); MAKE_STD_ZVAL(headers); array_init(headers); add_assoc_zval(result, "headers", headers); context->headers = headers; context->data = result; context->parser.data = context; http_parser_execute(&context->parser, &context->settings, body, body_len); if (context->is_response == 0) { add_assoc_string(result, "REQUEST_METHOD", (char*)http_method_str(context->parser.method), 1); } else { add_assoc_long(result, "status_code", (long)context->parser.status_code); } if (context->finished == 1) { RETURN_TRUE; } else { RETURN_FALSE; } } 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_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_last_error, arginfo_uv_last_error) 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) /* 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_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) /* spawn */ PHP_FE(uv_spawn, arginfo_uv_spawn) 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) PHP_FE(uv_ares_init_options, arginfo_uv_ares_init_options) PHP_FE(ares_gethostbyname, arginfo_ares_gethostbyname) /* 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_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) /* http parser */ PHP_FE(uv_http_parser_init, arginfo_uv_http_parser_init) PHP_FE(uv_http_parser_execute, arginfo_uv_http_parser_execute) {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,"bundled libuv Version", uv_version); php_info_print_table_row(2,"bundled http-parser Version", http_parser_version); php_info_print_table_end(); } zend_module_entry uv_module_entry = { #if ZEND_MODULE_API_NO >= 20010901 STANDARD_MODULE_HEADER, #endif "uv", uv_functions, /* Functions */ PHP_MINIT(uv), /* MINIT */ NULL, /* MSHUTDOWN */ NULL, /* RINIT */ PHP_RSHUTDOWN(uv), /* RSHUTDOWN */ PHP_MINFO(uv), /* MINFO */ #if ZEND_MODULE_API_NO >= 20010901 PHP_UV_EXTVER, #endif STANDARD_MODULE_PROPERTIES }; #ifdef COMPILE_DL_UV ZEND_GET_MODULE(uv) #endif