Skip to content

Commit

Permalink
aio: add aio_notify
Browse files Browse the repository at this point in the history
With this change async.c does not rely anymore on any service from
main-loop.c, i.e. it is completely self-contained.

Reviewed-by: Anthony Liguori <[email protected]>
Signed-off-by: Paolo Bonzini <[email protected]>
  • Loading branch information
bonzini committed Oct 30, 2012
1 parent e3713e0 commit 2f4dc3c
Show file tree
Hide file tree
Showing 2 changed files with 44 additions and 4 deletions.
30 changes: 26 additions & 4 deletions async.c
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@
/* bottom halves (can be seen as timers which expire ASAP) */

struct QEMUBH {
AioContext *ctx;
QEMUBHFunc *cb;
void *opaque;
QEMUBH *next;
Expand All @@ -42,6 +43,7 @@ QEMUBH *aio_bh_new(AioContext *ctx, QEMUBHFunc *cb, void *opaque)
{
QEMUBH *bh;
bh = g_malloc0(sizeof(QEMUBH));
bh->ctx = ctx;
bh->cb = cb;
bh->opaque = opaque;
bh->next = ctx->first_bh;
Expand Down Expand Up @@ -101,8 +103,7 @@ void qemu_bh_schedule(QEMUBH *bh)
return;
bh->scheduled = 1;
bh->idle = 0;
/* stop the currently executing CPU to execute the BH ASAP */
qemu_notify_event();
aio_notify(bh->ctx);
}

void qemu_bh_cancel(QEMUBH *bh)
Expand Down Expand Up @@ -177,11 +178,20 @@ aio_ctx_dispatch(GSource *source,
return true;
}

static void
aio_ctx_finalize(GSource *source)
{
AioContext *ctx = (AioContext *) source;

aio_set_event_notifier(ctx, &ctx->notifier, NULL, NULL);
event_notifier_cleanup(&ctx->notifier);
}

static GSourceFuncs aio_source_funcs = {
aio_ctx_prepare,
aio_ctx_check,
aio_ctx_dispatch,
NULL
aio_ctx_finalize
};

GSource *aio_get_g_source(AioContext *ctx)
Expand All @@ -190,9 +200,21 @@ GSource *aio_get_g_source(AioContext *ctx)
return &ctx->source;
}

void aio_notify(AioContext *ctx)
{
event_notifier_set(&ctx->notifier);
}

AioContext *aio_context_new(void)
{
return (AioContext *) g_source_new(&aio_source_funcs, sizeof(AioContext));
AioContext *ctx;
ctx = (AioContext *) g_source_new(&aio_source_funcs, sizeof(AioContext));
event_notifier_init(&ctx->notifier, false);
aio_set_event_notifier(ctx, &ctx->notifier,
(EventNotifierHandler *)
event_notifier_test_and_clear, NULL);

return ctx;
}

void aio_context_ref(AioContext *ctx)
Expand Down
18 changes: 18 additions & 0 deletions qemu-aio.h
Original file line number Diff line number Diff line change
Expand Up @@ -62,6 +62,9 @@ typedef struct AioContext {
* no callbacks are removed while we're walking and dispatching callbacks.
*/
int walking_bh;

/* Used for aio_notify. */
EventNotifier notifier;
} AioContext;

/* Returns 1 if there are still outstanding AIO requests; 0 otherwise */
Expand Down Expand Up @@ -101,6 +104,21 @@ void aio_context_unref(AioContext *ctx);
*/
QEMUBH *aio_bh_new(AioContext *ctx, QEMUBHFunc *cb, void *opaque);

/**
* aio_notify: Force processing of pending events.
*
* Similar to signaling a condition variable, aio_notify forces
* aio_wait to exit, so that the next call will re-examine pending events.
* The caller of aio_notify will usually call aio_wait again very soon,
* or go through another iteration of the GLib main loop. Hence, aio_notify
* also has the side effect of recalculating the sets of file descriptors
* that the main loop waits for.
*
* Calling aio_notify is rarely necessary, because for example scheduling
* a bottom half calls it already.
*/
void aio_notify(AioContext *ctx);

/**
* aio_bh_poll: Poll bottom halves for an AioContext.
*
Expand Down

0 comments on commit 2f4dc3c

Please sign in to comment.