| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  | /*
 | 
					
						
							|  |  |  |  * GThread coroutine initialization code | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2006  Anthony Liguori <anthony@codemonkey.ws> | 
					
						
							|  |  |  |  * Copyright (C) 2011  Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This library is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU Lesser General Public | 
					
						
							|  |  |  |  * License as published by the Free Software Foundation; either | 
					
						
							|  |  |  |  * version 2.0 of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This library is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
					
						
							|  |  |  |  * Lesser General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU Lesser General Public | 
					
						
							|  |  |  |  * License along with this library; if not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <glib.h>
 | 
					
						
							|  |  |  | #include "qemu-common.h"
 | 
					
						
							| 
									
										
										
										
											2015-09-01 14:48:02 +01:00
										 |  |  | #include "qemu/coroutine_int.h"
 | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  |     Coroutine base; | 
					
						
							|  |  |  |     GThread *thread; | 
					
						
							|  |  |  |     bool runnable; | 
					
						
							| 
									
										
										
										
											2012-03-03 04:52:56 +00:00
										 |  |  |     bool free_on_thread_exit; | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  |     CoroutineAction action; | 
					
						
							|  |  |  | } CoroutineGThread; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-08 12:30:46 +04:00
										 |  |  | static CompatGMutex coroutine_lock; | 
					
						
							|  |  |  | static CompatGCond coroutine_cond; | 
					
						
							| 
									
										
										
										
											2012-03-03 04:52:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* GLib 2.31 and beyond deprecated various parts of the thread API,
 | 
					
						
							|  |  |  |  * but the new interfaces are not available in older GLib versions | 
					
						
							|  |  |  |  * so we have to cope with both. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #if GLIB_CHECK_VERSION(2, 31, 0)
 | 
					
						
							|  |  |  | /* Awkwardly, the GPrivate API doesn't provide a way to update the
 | 
					
						
							|  |  |  |  * GDestroyNotify handler for the coroutine key dynamically. So instead | 
					
						
							|  |  |  |  * we track whether or not the CoroutineGThread should be freed on | 
					
						
							|  |  |  |  * thread exit / coroutine key update using the free_on_thread_exit | 
					
						
							|  |  |  |  * field. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void coroutine_destroy_notify(gpointer data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CoroutineGThread *co = data; | 
					
						
							|  |  |  |     if (co && co->free_on_thread_exit) { | 
					
						
							|  |  |  |         g_free(co); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static GPrivate coroutine_key = G_PRIVATE_INIT(coroutine_destroy_notify); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline CoroutineGThread *get_coroutine_key(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return g_private_get(&coroutine_key); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void set_coroutine_key(CoroutineGThread *co, | 
					
						
							|  |  |  |                                      bool free_on_thread_exit) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Unlike g_static_private_set() this does not call the GDestroyNotify
 | 
					
						
							|  |  |  |      * if the previous value of the key was NULL. Fortunately we only need | 
					
						
							|  |  |  |      * the GDestroyNotify in the non-NULL key case. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     co->free_on_thread_exit = free_on_thread_exit; | 
					
						
							|  |  |  |     g_private_replace(&coroutine_key, co); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline GThread *create_thread(GThreadFunc func, gpointer data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return g_thread_new("coroutine", func, data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Handle older GLib versions */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  | static GStaticPrivate coroutine_key = G_STATIC_PRIVATE_INIT; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-03 04:52:56 +00:00
										 |  |  | static inline CoroutineGThread *get_coroutine_key(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return g_static_private_get(&coroutine_key); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void set_coroutine_key(CoroutineGThread *co, | 
					
						
							|  |  |  |                                      bool free_on_thread_exit) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     g_static_private_set(&coroutine_key, co, | 
					
						
							|  |  |  |                          free_on_thread_exit ? (GDestroyNotify)g_free : NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline GThread *create_thread(GThreadFunc func, gpointer data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return g_thread_create_full(func, data, 0, TRUE, TRUE, | 
					
						
							|  |  |  |                                 G_THREAD_PRIORITY_NORMAL, NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  | static void __attribute__((constructor)) coroutine_init(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-20 13:41:04 +02:00
										 |  |  | #if !GLIB_CHECK_VERSION(2, 31, 0)
 | 
					
						
							| 
									
										
										
										
											2014-05-02 18:35:55 +04:00
										 |  |  |     if (!g_thread_supported()) { | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  |         g_thread_init(NULL); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-05-02 18:35:55 +04:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void coroutine_wait_runnable_locked(CoroutineGThread *co) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     while (!co->runnable) { | 
					
						
							| 
									
										
										
										
											2014-05-08 12:30:46 +04:00
										 |  |  |         g_cond_wait(&coroutine_cond, &coroutine_lock); | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void coroutine_wait_runnable(CoroutineGThread *co) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-05-08 12:30:46 +04:00
										 |  |  |     g_mutex_lock(&coroutine_lock); | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  |     coroutine_wait_runnable_locked(co); | 
					
						
							| 
									
										
										
										
											2014-05-08 12:30:46 +04:00
										 |  |  |     g_mutex_unlock(&coroutine_lock); | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static gpointer coroutine_thread(gpointer opaque) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CoroutineGThread *co = opaque; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-03 04:52:56 +00:00
										 |  |  |     set_coroutine_key(co, false); | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  |     coroutine_wait_runnable(co); | 
					
						
							|  |  |  |     co->base.entry(co->base.entry_arg); | 
					
						
							|  |  |  |     qemu_coroutine_switch(&co->base, co->base.caller, COROUTINE_TERMINATE); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Coroutine *qemu_coroutine_new(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CoroutineGThread *co; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-20 22:09:37 -05:00
										 |  |  |     co = g_malloc0(sizeof(*co)); | 
					
						
							| 
									
										
										
										
											2012-03-03 04:52:56 +00:00
										 |  |  |     co->thread = create_thread(coroutine_thread, co); | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  |     if (!co->thread) { | 
					
						
							| 
									
										
										
										
											2011-08-20 22:09:37 -05:00
										 |  |  |         g_free(co); | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return &co->base; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void qemu_coroutine_delete(Coroutine *co_) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CoroutineGThread *co = DO_UPCAST(CoroutineGThread, base, co_); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     g_thread_join(co->thread); | 
					
						
							| 
									
										
										
										
											2011-08-20 22:09:37 -05:00
										 |  |  |     g_free(co); | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CoroutineAction qemu_coroutine_switch(Coroutine *from_, | 
					
						
							|  |  |  |                                       Coroutine *to_, | 
					
						
							|  |  |  |                                       CoroutineAction action) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     CoroutineGThread *from = DO_UPCAST(CoroutineGThread, base, from_); | 
					
						
							|  |  |  |     CoroutineGThread *to = DO_UPCAST(CoroutineGThread, base, to_); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-08 12:30:46 +04:00
										 |  |  |     g_mutex_lock(&coroutine_lock); | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  |     from->runnable = false; | 
					
						
							|  |  |  |     from->action = action; | 
					
						
							|  |  |  |     to->runnable = true; | 
					
						
							|  |  |  |     to->action = action; | 
					
						
							| 
									
										
										
										
											2014-05-08 12:30:46 +04:00
										 |  |  |     g_cond_broadcast(&coroutine_cond); | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  |     if (action != COROUTINE_TERMINATE) { | 
					
						
							|  |  |  |         coroutine_wait_runnable_locked(from); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-05-08 12:30:46 +04:00
										 |  |  |     g_mutex_unlock(&coroutine_lock); | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  |     return from->action; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Coroutine *qemu_coroutine_self(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-03 04:52:56 +00:00
										 |  |  |     CoroutineGThread *co = get_coroutine_key(); | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  |     if (!co) { | 
					
						
							| 
									
										
										
										
											2011-08-20 22:09:37 -05:00
										 |  |  |         co = g_malloc0(sizeof(*co)); | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  |         co->runnable = true; | 
					
						
							| 
									
										
										
										
											2012-03-03 04:52:56 +00:00
										 |  |  |         set_coroutine_key(co, true); | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return &co->base; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool qemu_in_coroutine(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-03 04:52:56 +00:00
										 |  |  |     CoroutineGThread *co = get_coroutine_key(); | 
					
						
							| 
									
										
										
										
											2011-06-09 23:11:06 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  |     return co && co->base.caller; | 
					
						
							|  |  |  | } |