 1a92d6d500
			
		
	
	
		1a92d6d500
		
	
	
	
	
		
			
			Remove dependency on qiochannel by removing yank_generic_iochannel and letting migration and chardev use their own yank function for iochannel. Signed-off-by: Lukas Straub <lukasstraub2@web.de> Reviewed-by: Marc-André Lureau <marcandre.lureau@redhat.com> Message-Id: <20ff143fc2db23e27cd41d38043e481376c9cec1.1616521341.git.lukasstraub2@web.de>
		
			
				
	
	
		
			200 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			200 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * QEMU yank feature
 | |
|  *
 | |
|  * Copyright (c) Lukas Straub <lukasstraub2@web.de>
 | |
|  *
 | |
|  * 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 "qemu/thread.h"
 | |
| #include "qemu/queue.h"
 | |
| #include "qemu/lockable.h"
 | |
| #include "qapi/qapi-commands-yank.h"
 | |
| #include "qapi/qapi-visit-yank.h"
 | |
| #include "qapi/clone-visitor.h"
 | |
| #include "qemu/yank.h"
 | |
| 
 | |
| struct YankFuncAndParam {
 | |
|     YankFn *func;
 | |
|     void *opaque;
 | |
|     QLIST_ENTRY(YankFuncAndParam) next;
 | |
| };
 | |
| 
 | |
| struct YankInstanceEntry {
 | |
|     YankInstance *instance;
 | |
|     QLIST_HEAD(, YankFuncAndParam) yankfns;
 | |
|     QLIST_ENTRY(YankInstanceEntry) next;
 | |
| };
 | |
| 
 | |
| typedef struct YankFuncAndParam YankFuncAndParam;
 | |
| typedef struct YankInstanceEntry YankInstanceEntry;
 | |
| 
 | |
| /*
 | |
|  * This lock protects the yank_instance_list below. Because it's taken by
 | |
|  * OOB-capable commands, it must be "fast", i.e. it may only be held for a
 | |
|  * bounded, short time. See docs/devel/qapi-code-gen.txt for additional
 | |
|  * information.
 | |
|  */
 | |
| static QemuMutex yank_lock;
 | |
| 
 | |
| static QLIST_HEAD(, YankInstanceEntry) yank_instance_list
 | |
|     = QLIST_HEAD_INITIALIZER(yank_instance_list);
 | |
| 
 | |
| static bool yank_instance_equal(const YankInstance *a, const YankInstance *b)
 | |
| {
 | |
|     if (a->type != b->type) {
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     switch (a->type) {
 | |
|     case YANK_INSTANCE_TYPE_BLOCK_NODE:
 | |
|         return g_str_equal(a->u.block_node.node_name,
 | |
|                            b->u.block_node.node_name);
 | |
| 
 | |
|     case YANK_INSTANCE_TYPE_CHARDEV:
 | |
|         return g_str_equal(a->u.chardev.id, b->u.chardev.id);
 | |
| 
 | |
|     case YANK_INSTANCE_TYPE_MIGRATION:
 | |
|         return true;
 | |
| 
 | |
|     default:
 | |
|         abort();
 | |
|     }
 | |
| }
 | |
| 
 | |
| static YankInstanceEntry *yank_find_entry(const YankInstance *instance)
 | |
| {
 | |
|     YankInstanceEntry *entry;
 | |
| 
 | |
|     QLIST_FOREACH(entry, &yank_instance_list, next) {
 | |
|         if (yank_instance_equal(entry->instance, instance)) {
 | |
|             return entry;
 | |
|         }
 | |
|     }
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| bool yank_register_instance(const YankInstance *instance, Error **errp)
 | |
| {
 | |
|     YankInstanceEntry *entry;
 | |
| 
 | |
|     QEMU_LOCK_GUARD(&yank_lock);
 | |
| 
 | |
|     if (yank_find_entry(instance)) {
 | |
|         error_setg(errp, "duplicate yank instance");
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     entry = g_new0(YankInstanceEntry, 1);
 | |
|     entry->instance = QAPI_CLONE(YankInstance, instance);
 | |
|     QLIST_INIT(&entry->yankfns);
 | |
|     QLIST_INSERT_HEAD(&yank_instance_list, entry, next);
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| void yank_unregister_instance(const YankInstance *instance)
 | |
| {
 | |
|     YankInstanceEntry *entry;
 | |
| 
 | |
|     QEMU_LOCK_GUARD(&yank_lock);
 | |
|     entry = yank_find_entry(instance);
 | |
|     assert(entry);
 | |
| 
 | |
|     assert(QLIST_EMPTY(&entry->yankfns));
 | |
|     QLIST_REMOVE(entry, next);
 | |
|     qapi_free_YankInstance(entry->instance);
 | |
|     g_free(entry);
 | |
| }
 | |
| 
 | |
| void yank_register_function(const YankInstance *instance,
 | |
|                             YankFn *func,
 | |
|                             void *opaque)
 | |
| {
 | |
|     YankInstanceEntry *entry;
 | |
|     YankFuncAndParam *func_entry;
 | |
| 
 | |
|     QEMU_LOCK_GUARD(&yank_lock);
 | |
|     entry = yank_find_entry(instance);
 | |
|     assert(entry);
 | |
| 
 | |
|     func_entry = g_new0(YankFuncAndParam, 1);
 | |
|     func_entry->func = func;
 | |
|     func_entry->opaque = opaque;
 | |
| 
 | |
|     QLIST_INSERT_HEAD(&entry->yankfns, func_entry, next);
 | |
| }
 | |
| 
 | |
| void yank_unregister_function(const YankInstance *instance,
 | |
|                               YankFn *func,
 | |
|                               void *opaque)
 | |
| {
 | |
|     YankInstanceEntry *entry;
 | |
|     YankFuncAndParam *func_entry;
 | |
| 
 | |
|     QEMU_LOCK_GUARD(&yank_lock);
 | |
|     entry = yank_find_entry(instance);
 | |
|     assert(entry);
 | |
| 
 | |
|     QLIST_FOREACH(func_entry, &entry->yankfns, next) {
 | |
|         if (func_entry->func == func && func_entry->opaque == opaque) {
 | |
|             QLIST_REMOVE(func_entry, next);
 | |
|             g_free(func_entry);
 | |
|             return;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     abort();
 | |
| }
 | |
| 
 | |
| void qmp_yank(YankInstanceList *instances,
 | |
|               Error **errp)
 | |
| {
 | |
|     YankInstanceList *tail;
 | |
|     YankInstanceEntry *entry;
 | |
|     YankFuncAndParam *func_entry;
 | |
| 
 | |
|     QEMU_LOCK_GUARD(&yank_lock);
 | |
|     for (tail = instances; tail; tail = tail->next) {
 | |
|         entry = yank_find_entry(tail->value);
 | |
|         if (!entry) {
 | |
|             error_set(errp, ERROR_CLASS_DEVICE_NOT_FOUND, "Instance not found");
 | |
|             return;
 | |
|         }
 | |
|     }
 | |
|     for (tail = instances; tail; tail = tail->next) {
 | |
|         entry = yank_find_entry(tail->value);
 | |
|         assert(entry);
 | |
|         QLIST_FOREACH(func_entry, &entry->yankfns, next) {
 | |
|             func_entry->func(func_entry->opaque);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| YankInstanceList *qmp_query_yank(Error **errp)
 | |
| {
 | |
|     YankInstanceEntry *entry;
 | |
|     YankInstanceList *ret;
 | |
| 
 | |
|     ret = NULL;
 | |
| 
 | |
|     QEMU_LOCK_GUARD(&yank_lock);
 | |
|     QLIST_FOREACH(entry, &yank_instance_list, next) {
 | |
|         YankInstanceList *new_entry;
 | |
|         new_entry = g_new0(YankInstanceList, 1);
 | |
|         new_entry->value = QAPI_CLONE(YankInstance, entry->instance);
 | |
|         new_entry->next = ret;
 | |
|         ret = new_entry;
 | |
|     }
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| static void __attribute__((__constructor__)) yank_init(void)
 | |
| {
 | |
|     qemu_mutex_init(&yank_lock);
 | |
| }
 |