|  | /* | 
|  | * Event loop thread implementation for unit tests | 
|  | * | 
|  | * Copyright Red Hat Inc., 2013, 2016 | 
|  | * | 
|  | * Authors: | 
|  | *  Stefan Hajnoczi   <stefanha@redhat.com> | 
|  | *  Paolo Bonzini     <pbonzini@redhat.com> | 
|  | * | 
|  | * This work is licensed under the terms of the GNU GPL, version 2 or later. | 
|  | * See the COPYING file in the top-level directory. | 
|  | * | 
|  | */ | 
|  |  | 
|  | #include "qemu/osdep.h" | 
|  | #include "qapi/error.h" | 
|  | #include "block/aio.h" | 
|  | #include "qemu/main-loop.h" | 
|  | #include "qemu/rcu.h" | 
|  | #include "iothread.h" | 
|  |  | 
|  | struct IOThread { | 
|  | AioContext *ctx; | 
|  | GMainContext *worker_context; | 
|  | GMainLoop *main_loop; | 
|  |  | 
|  | QemuThread thread; | 
|  | QemuMutex init_done_lock; | 
|  | QemuCond init_done_cond;    /* is thread initialization done? */ | 
|  | bool stopping; | 
|  | }; | 
|  |  | 
|  | static void iothread_init_gcontext(IOThread *iothread) | 
|  | { | 
|  | GSource *source; | 
|  |  | 
|  | iothread->worker_context = g_main_context_new(); | 
|  | source = aio_get_g_source(iothread_get_aio_context(iothread)); | 
|  | g_source_attach(source, iothread->worker_context); | 
|  | g_source_unref(source); | 
|  | iothread->main_loop = g_main_loop_new(iothread->worker_context, TRUE); | 
|  | } | 
|  |  | 
|  | static void *iothread_run(void *opaque) | 
|  | { | 
|  | IOThread *iothread = opaque; | 
|  |  | 
|  | rcu_register_thread(); | 
|  |  | 
|  | qemu_mutex_lock(&iothread->init_done_lock); | 
|  | iothread->ctx = aio_context_new(&error_abort); | 
|  | qemu_set_current_aio_context(iothread->ctx); | 
|  |  | 
|  | /* | 
|  | * We must connect the ctx to a GMainContext, because in older versions | 
|  | * of glib the g_source_ref()/unref() functions are not threadsafe | 
|  | * on sources without a context. | 
|  | */ | 
|  | iothread_init_gcontext(iothread); | 
|  |  | 
|  | /* | 
|  | * g_main_context_push_thread_default() must be called before anything | 
|  | * in this new thread uses glib. | 
|  | */ | 
|  | g_main_context_push_thread_default(iothread->worker_context); | 
|  |  | 
|  | qemu_cond_signal(&iothread->init_done_cond); | 
|  | qemu_mutex_unlock(&iothread->init_done_lock); | 
|  |  | 
|  | while (!qatomic_read(&iothread->stopping)) { | 
|  | aio_poll(iothread->ctx, true); | 
|  | } | 
|  |  | 
|  | g_main_context_pop_thread_default(iothread->worker_context); | 
|  | rcu_unregister_thread(); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | static void iothread_stop_bh(void *opaque) | 
|  | { | 
|  | IOThread *iothread = opaque; | 
|  |  | 
|  | iothread->stopping = true; | 
|  | } | 
|  |  | 
|  | void iothread_join(IOThread *iothread) | 
|  | { | 
|  | aio_bh_schedule_oneshot(iothread->ctx, iothread_stop_bh, iothread); | 
|  | qemu_thread_join(&iothread->thread); | 
|  | g_main_context_unref(iothread->worker_context); | 
|  | g_main_loop_unref(iothread->main_loop); | 
|  | qemu_cond_destroy(&iothread->init_done_cond); | 
|  | qemu_mutex_destroy(&iothread->init_done_lock); | 
|  | aio_context_unref(iothread->ctx); | 
|  | g_free(iothread); | 
|  | } | 
|  |  | 
|  | IOThread *iothread_new(void) | 
|  | { | 
|  | IOThread *iothread = g_new0(IOThread, 1); | 
|  |  | 
|  | qemu_mutex_init(&iothread->init_done_lock); | 
|  | qemu_cond_init(&iothread->init_done_cond); | 
|  | qemu_thread_create(&iothread->thread, NULL, iothread_run, | 
|  | iothread, QEMU_THREAD_JOINABLE); | 
|  |  | 
|  | /* Wait for initialization to complete */ | 
|  | qemu_mutex_lock(&iothread->init_done_lock); | 
|  | while (iothread->ctx == NULL) { | 
|  | qemu_cond_wait(&iothread->init_done_cond, | 
|  | &iothread->init_done_lock); | 
|  | } | 
|  | qemu_mutex_unlock(&iothread->init_done_lock); | 
|  | return iothread; | 
|  | } | 
|  |  | 
|  | AioContext *iothread_get_aio_context(IOThread *iothread) | 
|  | { | 
|  | return iothread->ctx; | 
|  | } |