12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879 |
- /*
- * QEMU coroutine sleep
- *
- * Copyright IBM, Corp. 2011
- *
- * Authors:
- * Stefan Hajnoczi <stefanha@linux.vnet.ibm.com>
- *
- * This work is licensed under the terms of the GNU LGPL, version 2 or later.
- * See the COPYING.LIB file in the top-level directory.
- *
- */
- #include "qemu/osdep.h"
- #include "qemu/coroutine.h"
- #include "qemu/coroutine_int.h"
- #include "qemu/timer.h"
- #include "block/aio.h"
- static const char *qemu_co_sleep_ns__scheduled = "qemu_co_sleep_ns";
- struct QemuCoSleepState {
- Coroutine *co;
- QEMUTimer *ts;
- QemuCoSleepState **user_state_pointer;
- };
- void qemu_co_sleep_wake(QemuCoSleepState *sleep_state)
- {
- /* Write of schedule protected by barrier write in aio_co_schedule */
- const char *scheduled = atomic_cmpxchg(&sleep_state->co->scheduled,
- qemu_co_sleep_ns__scheduled, NULL);
- assert(scheduled == qemu_co_sleep_ns__scheduled);
- if (sleep_state->user_state_pointer) {
- *sleep_state->user_state_pointer = NULL;
- }
- timer_del(sleep_state->ts);
- aio_co_wake(sleep_state->co);
- }
- static void co_sleep_cb(void *opaque)
- {
- qemu_co_sleep_wake(opaque);
- }
- void coroutine_fn qemu_co_sleep_ns_wakeable(QEMUClockType type, int64_t ns,
- QemuCoSleepState **sleep_state)
- {
- AioContext *ctx = qemu_get_current_aio_context();
- QemuCoSleepState state = {
- .co = qemu_coroutine_self(),
- .ts = aio_timer_new(ctx, type, SCALE_NS, co_sleep_cb, &state),
- .user_state_pointer = sleep_state,
- };
- const char *scheduled = atomic_cmpxchg(&state.co->scheduled, NULL,
- qemu_co_sleep_ns__scheduled);
- if (scheduled) {
- fprintf(stderr,
- "%s: Co-routine was already scheduled in '%s'\n",
- __func__, scheduled);
- abort();
- }
- if (sleep_state) {
- *sleep_state = &state;
- }
- timer_mod(state.ts, qemu_clock_get_ns(type) + ns);
- qemu_coroutine_yield();
- if (sleep_state) {
- /*
- * Note that *sleep_state is cleared during qemu_co_sleep_wake
- * before resuming this coroutine.
- */
- assert(*sleep_state == NULL);
- }
- timer_free(state.ts);
- }
|