Merge branch 'v0.8'
Conflicts: src/fs-poll.c
This commit is contained in:
commit
28ff1422e8
19
include/uv.h
19
include/uv.h
@ -1551,14 +1551,17 @@ struct uv_fs_event_s {
|
|||||||
struct uv_fs_poll_s {
|
struct uv_fs_poll_s {
|
||||||
UV_HANDLE_FIELDS
|
UV_HANDLE_FIELDS
|
||||||
/* Private, don't touch. */
|
/* Private, don't touch. */
|
||||||
int busy_polling; /* TODO(bnoordhuis) Fold into flags field. */
|
void* poll_ctx;
|
||||||
unsigned int interval;
|
/* v0.8 ABI compatibility */
|
||||||
uint64_t start_time;
|
char padding[sizeof(int)
|
||||||
char* path;
|
+ sizeof(unsigned int)
|
||||||
uv_fs_poll_cb poll_cb;
|
+ sizeof(uint64_t)
|
||||||
uv_timer_t timer_handle;
|
+ sizeof(char*)
|
||||||
uv_fs_t* fs_req;
|
+ sizeof(uv_fs_poll_cb)
|
||||||
uv_statbuf_t statbuf;
|
+ sizeof(uv_timer_t)
|
||||||
|
+ sizeof(uv_fs_t*)
|
||||||
|
+ sizeof(uv_statbuf_t)
|
||||||
|
- sizeof(void*)];
|
||||||
};
|
};
|
||||||
|
|
||||||
UV_EXTERN int uv_fs_poll_init(uv_loop_t* loop, uv_fs_poll_t* handle);
|
UV_EXTERN int uv_fs_poll_init(uv_loop_t* loop, uv_fs_poll_t* handle);
|
||||||
|
|||||||
148
src/fs-poll.c
148
src/fs-poll.c
@ -26,22 +26,29 @@
|
|||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
|
struct poll_ctx {
|
||||||
|
uv_fs_poll_t* parent_handle; /* NULL if parent has been stopped or closed */
|
||||||
|
int busy_polling;
|
||||||
|
unsigned int interval;
|
||||||
|
uint64_t start_time;
|
||||||
|
uv_loop_t* loop;
|
||||||
|
uv_fs_poll_cb poll_cb;
|
||||||
|
uv_timer_t timer_handle;
|
||||||
|
uv_fs_t fs_req; /* TODO(bnoordhuis) mark fs_req internal */
|
||||||
|
uv_statbuf_t statbuf;
|
||||||
|
char path[1]; /* variable length */
|
||||||
|
};
|
||||||
|
|
||||||
static int statbuf_eq(const uv_statbuf_t* a, const uv_statbuf_t* b);
|
static int statbuf_eq(const uv_statbuf_t* a, const uv_statbuf_t* b);
|
||||||
static void timer_cb(uv_timer_t* timer, int status);
|
|
||||||
static void poll_cb(uv_fs_t* req);
|
static void poll_cb(uv_fs_t* req);
|
||||||
|
static void timer_cb(uv_timer_t* timer, int status);
|
||||||
|
static void timer_close_cb(uv_handle_t* handle);
|
||||||
|
|
||||||
static uv_statbuf_t zero_statbuf;
|
static uv_statbuf_t zero_statbuf;
|
||||||
|
|
||||||
|
|
||||||
int uv_fs_poll_init(uv_loop_t* loop, uv_fs_poll_t* handle) {
|
int uv_fs_poll_init(uv_loop_t* loop, uv_fs_poll_t* handle) {
|
||||||
/* TODO(bnoordhuis) Mark fs_req internal. */
|
|
||||||
uv__handle_init(loop, (uv_handle_t*)handle, UV_FS_POLL);
|
uv__handle_init(loop, (uv_handle_t*)handle, UV_FS_POLL);
|
||||||
if (uv_timer_init(loop, &handle->timer_handle))
|
|
||||||
return -1;
|
|
||||||
|
|
||||||
handle->timer_handle.flags |= UV__HANDLE_INTERNAL;
|
|
||||||
uv__handle_unref(&handle->timer_handle);
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -50,30 +57,37 @@ int uv_fs_poll_start(uv_fs_poll_t* handle,
|
|||||||
uv_fs_poll_cb cb,
|
uv_fs_poll_cb cb,
|
||||||
const char* path,
|
const char* path,
|
||||||
unsigned int interval) {
|
unsigned int interval) {
|
||||||
uv_fs_t* req;
|
struct poll_ctx* ctx;
|
||||||
|
uv_loop_t* loop;
|
||||||
size_t len;
|
size_t len;
|
||||||
|
|
||||||
if (uv__is_active(handle))
|
if (uv__is_active(handle))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
len = strlen(path) + 1;
|
loop = handle->loop;
|
||||||
req = malloc(sizeof(*req) + len);
|
len = strlen(path);
|
||||||
|
ctx = calloc(1, sizeof(*ctx) + len);
|
||||||
|
|
||||||
if (req == NULL)
|
if (ctx == NULL)
|
||||||
return uv__set_artificial_error(handle->loop, UV_ENOMEM);
|
return uv__set_artificial_error(loop, UV_ENOMEM);
|
||||||
|
|
||||||
req->data = handle;
|
ctx->loop = loop;
|
||||||
handle->path = memcpy(req + 1, path, len);
|
ctx->poll_cb = cb;
|
||||||
handle->fs_req = req;
|
ctx->interval = interval ? interval : 1;
|
||||||
handle->poll_cb = cb;
|
ctx->start_time = uv_now(loop);
|
||||||
handle->interval = interval ? interval : 1;
|
ctx->parent_handle = handle;
|
||||||
handle->start_time = uv_now(handle->loop);
|
memcpy(ctx->path, path, len + 1);
|
||||||
handle->busy_polling = 0;
|
|
||||||
memset(&handle->statbuf, 0, sizeof(handle->statbuf));
|
|
||||||
|
|
||||||
if (uv_fs_stat(handle->loop, handle->fs_req, handle->path, poll_cb))
|
if (uv_timer_init(loop, &ctx->timer_handle))
|
||||||
abort();
|
abort();
|
||||||
|
|
||||||
|
ctx->timer_handle.flags |= UV__HANDLE_INTERNAL;
|
||||||
|
uv__handle_unref(&ctx->timer_handle);
|
||||||
|
|
||||||
|
if (uv_fs_stat(loop, &ctx->fs_req, ctx->path, poll_cb))
|
||||||
|
abort();
|
||||||
|
|
||||||
|
handle->poll_ctx = ctx;
|
||||||
uv__handle_start(handle);
|
uv__handle_start(handle);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -81,25 +95,19 @@ int uv_fs_poll_start(uv_fs_poll_t* handle,
|
|||||||
|
|
||||||
|
|
||||||
int uv_fs_poll_stop(uv_fs_poll_t* handle) {
|
int uv_fs_poll_stop(uv_fs_poll_t* handle) {
|
||||||
|
struct poll_ctx* ctx;
|
||||||
|
|
||||||
if (!uv__is_active(handle))
|
if (!uv__is_active(handle))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/* Don't free the fs req if it's active. Signal poll_cb that it needs to free
|
ctx = handle->poll_ctx;
|
||||||
* the req by removing the handle backlink.
|
assert(ctx != NULL);
|
||||||
*
|
assert(ctx->parent_handle != NULL);
|
||||||
* TODO(bnoordhuis) Have uv-unix postpone the close callback until the req
|
|
||||||
* finishes so we don't need this pointer / lifecycle hackery. The callback
|
|
||||||
* always runs on the next tick now.
|
|
||||||
*/
|
|
||||||
if (handle->fs_req->data)
|
|
||||||
handle->fs_req->data = NULL;
|
|
||||||
else
|
|
||||||
free(handle->fs_req);
|
|
||||||
|
|
||||||
handle->fs_req = NULL;
|
ctx->parent_handle = NULL;
|
||||||
handle->path = NULL;
|
uv_timer_stop(&ctx->timer_handle);
|
||||||
|
|
||||||
uv_timer_stop(&handle->timer_handle);
|
handle->poll_ctx = NULL;
|
||||||
uv__handle_stop(handle);
|
uv__handle_stop(handle);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -108,73 +116,75 @@ int uv_fs_poll_stop(uv_fs_poll_t* handle) {
|
|||||||
|
|
||||||
void uv__fs_poll_close(uv_fs_poll_t* handle) {
|
void uv__fs_poll_close(uv_fs_poll_t* handle) {
|
||||||
uv_fs_poll_stop(handle);
|
uv_fs_poll_stop(handle);
|
||||||
uv_close((uv_handle_t*)&handle->timer_handle, NULL);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void timer_cb(uv_timer_t* timer, int status) {
|
static void timer_cb(uv_timer_t* timer, int status) {
|
||||||
uv_fs_poll_t* handle;
|
struct poll_ctx* ctx;
|
||||||
|
|
||||||
handle = container_of(timer, uv_fs_poll_t, timer_handle);
|
ctx = container_of(timer, struct poll_ctx, timer_handle);
|
||||||
handle->start_time = uv_now(handle->loop);
|
|
||||||
handle->fs_req->data = handle;
|
|
||||||
|
|
||||||
if (uv_fs_stat(handle->loop, handle->fs_req, handle->path, poll_cb))
|
if (ctx->parent_handle == NULL) { /* handle has been stopped or closed */
|
||||||
|
uv_close((uv_handle_t*)&ctx->timer_handle, timer_close_cb);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(ctx->parent_handle->poll_ctx == ctx);
|
||||||
|
ctx->start_time = uv_now(ctx->loop);
|
||||||
|
|
||||||
|
if (uv_fs_stat(ctx->loop, &ctx->fs_req, ctx->path, poll_cb))
|
||||||
abort();
|
abort();
|
||||||
|
|
||||||
assert(uv__is_active(handle));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void poll_cb(uv_fs_t* req) {
|
static void poll_cb(uv_fs_t* req) {
|
||||||
uv_statbuf_t* statbuf;
|
uv_statbuf_t* statbuf;
|
||||||
uv_fs_poll_t* handle;
|
struct poll_ctx* ctx;
|
||||||
uint64_t interval;
|
uint64_t interval;
|
||||||
|
|
||||||
handle = req->data;
|
ctx = container_of(req, struct poll_ctx, fs_req);
|
||||||
|
|
||||||
if (handle == NULL) /* Handle has been stopped or closed. */
|
if (ctx->parent_handle == NULL) { /* handle has been stopped or closed */
|
||||||
goto out;
|
uv_close((uv_handle_t*)&ctx->timer_handle, timer_close_cb);
|
||||||
|
uv_fs_req_cleanup(req);
|
||||||
assert(req == handle->fs_req);
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (req->result != 0) {
|
if (req->result != 0) {
|
||||||
if (handle->busy_polling != -req->errorno) {
|
if (ctx->busy_polling != -req->errorno) {
|
||||||
uv__set_artificial_error(handle->loop, req->errorno);
|
uv__set_artificial_error(ctx->loop, req->errorno);
|
||||||
handle->poll_cb(handle, -1, &handle->statbuf, &zero_statbuf);
|
ctx->poll_cb(ctx->parent_handle, -1, &ctx->statbuf, &zero_statbuf);
|
||||||
handle->busy_polling = -req->errorno;
|
ctx->busy_polling = -req->errorno;
|
||||||
}
|
}
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
statbuf = req->ptr;
|
statbuf = req->ptr;
|
||||||
|
|
||||||
if (handle->busy_polling != 0)
|
if (ctx->busy_polling != 0)
|
||||||
if (handle->busy_polling < 0 || !statbuf_eq(&handle->statbuf, statbuf))
|
if (ctx->busy_polling < 0 || !statbuf_eq(&ctx->statbuf, statbuf))
|
||||||
handle->poll_cb(handle, 0, &handle->statbuf, statbuf);
|
ctx->poll_cb(ctx->parent_handle, 0, &ctx->statbuf, statbuf);
|
||||||
|
|
||||||
handle->statbuf = *statbuf;
|
ctx->statbuf = *statbuf;
|
||||||
handle->busy_polling = 1;
|
ctx->busy_polling = 1;
|
||||||
|
|
||||||
out:
|
out:
|
||||||
uv_fs_req_cleanup(req);
|
uv_fs_req_cleanup(req);
|
||||||
|
|
||||||
if (req->data == NULL) { /* Handle has been stopped or closed. */
|
|
||||||
free(req);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
req->data = NULL; /* Tell uv_fs_poll_stop() it's safe to free the req. */
|
|
||||||
|
|
||||||
/* Reschedule timer, subtract the delay from doing the stat(). */
|
/* Reschedule timer, subtract the delay from doing the stat(). */
|
||||||
interval = handle->interval;
|
interval = ctx->interval;
|
||||||
interval -= (uv_now(handle->loop) - handle->start_time) % interval;
|
interval -= (uv_now(ctx->loop) - ctx->start_time) % interval;
|
||||||
|
|
||||||
if (uv_timer_start(&handle->timer_handle, timer_cb, interval, 0))
|
if (uv_timer_start(&ctx->timer_handle, timer_cb, interval, 0))
|
||||||
abort();
|
abort();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void timer_close_cb(uv_handle_t* handle) {
|
||||||
|
free(container_of(handle, struct poll_ctx, timer_handle));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
static int statbuf_eq(const uv_statbuf_t* a, const uv_statbuf_t* b) {
|
static int statbuf_eq(const uv_statbuf_t* a, const uv_statbuf_t* b) {
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
return a->st_mtime == b->st_mtime
|
return a->st_mtime == b->st_mtime
|
||||||
|
|||||||
@ -36,6 +36,7 @@ BENCHMARK_IMPL(sizes) {
|
|||||||
LOGF("uv_idle_t: %u bytes\n", (unsigned int) sizeof(uv_idle_t));
|
LOGF("uv_idle_t: %u bytes\n", (unsigned int) sizeof(uv_idle_t));
|
||||||
LOGF("uv_async_t: %u bytes\n", (unsigned int) sizeof(uv_async_t));
|
LOGF("uv_async_t: %u bytes\n", (unsigned int) sizeof(uv_async_t));
|
||||||
LOGF("uv_timer_t: %u bytes\n", (unsigned int) sizeof(uv_timer_t));
|
LOGF("uv_timer_t: %u bytes\n", (unsigned int) sizeof(uv_timer_t));
|
||||||
|
LOGF("uv_fs_poll_t: %u bytes\n", (unsigned int) sizeof(uv_fs_poll_t));
|
||||||
LOGF("uv_fs_event_t: %u bytes\n", (unsigned int) sizeof(uv_fs_event_t));
|
LOGF("uv_fs_event_t: %u bytes\n", (unsigned int) sizeof(uv_fs_event_t));
|
||||||
LOGF("uv_process_t: %u bytes\n", (unsigned int) sizeof(uv_process_t));
|
LOGF("uv_process_t: %u bytes\n", (unsigned int) sizeof(uv_process_t));
|
||||||
LOGF("uv_poll_t: %u bytes\n", (unsigned int) sizeof(uv_poll_t));
|
LOGF("uv_poll_t: %u bytes\n", (unsigned int) sizeof(uv_poll_t));
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user