10
#include "qemu/osdep.h"
11
#include "qapi/error.h"
12
#include "qemu/thread.h"
13
#include "qemu/queue.h"
14
#include "qemu/lockable.h"
15
#include "qapi/qapi-commands-yank.h"
16
#include "qapi/qapi-visit-yank.h"
17
#include "qapi/clone-visitor.h"
20
struct YankFuncAndParam {
23
QLIST_ENTRY(YankFuncAndParam) next;
26
struct YankInstanceEntry {
27
YankInstance *instance;
28
QLIST_HEAD(, YankFuncAndParam) yankfns;
29
QLIST_ENTRY(YankInstanceEntry) next;
32
typedef struct YankFuncAndParam YankFuncAndParam;
33
typedef struct YankInstanceEntry YankInstanceEntry;
41
static QemuMutex yank_lock;
43
static QLIST_HEAD(, YankInstanceEntry) yank_instance_list
44
= QLIST_HEAD_INITIALIZER(yank_instance_list);
46
static bool yank_instance_equal(const YankInstance *a, const YankInstance *b)
48
if (a->type != b->type) {
53
case YANK_INSTANCE_TYPE_BLOCK_NODE:
54
return g_str_equal(a->u.block_node.node_name,
55
b->u.block_node.node_name);
57
case YANK_INSTANCE_TYPE_CHARDEV:
58
return g_str_equal(a->u.chardev.id, b->u.chardev.id);
60
case YANK_INSTANCE_TYPE_MIGRATION:
68
static YankInstanceEntry *yank_find_entry(const YankInstance *instance)
70
YankInstanceEntry *entry;
72
QLIST_FOREACH(entry, &yank_instance_list, next) {
73
if (yank_instance_equal(entry->instance, instance)) {
80
bool yank_register_instance(const YankInstance *instance, Error **errp)
82
YankInstanceEntry *entry;
84
QEMU_LOCK_GUARD(&yank_lock);
86
if (yank_find_entry(instance)) {
87
error_setg(errp, "duplicate yank instance");
91
entry = g_new0(YankInstanceEntry, 1);
92
entry->instance = QAPI_CLONE(YankInstance, instance);
93
QLIST_INIT(&entry->yankfns);
94
QLIST_INSERT_HEAD(&yank_instance_list, entry, next);
99
void yank_unregister_instance(const YankInstance *instance)
101
YankInstanceEntry *entry;
103
QEMU_LOCK_GUARD(&yank_lock);
104
entry = yank_find_entry(instance);
107
assert(QLIST_EMPTY(&entry->yankfns));
108
QLIST_REMOVE(entry, next);
109
qapi_free_YankInstance(entry->instance);
113
void yank_register_function(const YankInstance *instance,
117
YankInstanceEntry *entry;
118
YankFuncAndParam *func_entry;
120
QEMU_LOCK_GUARD(&yank_lock);
121
entry = yank_find_entry(instance);
124
func_entry = g_new0(YankFuncAndParam, 1);
125
func_entry->func = func;
126
func_entry->opaque = opaque;
128
QLIST_INSERT_HEAD(&entry->yankfns, func_entry, next);
131
void yank_unregister_function(const YankInstance *instance,
135
YankInstanceEntry *entry;
136
YankFuncAndParam *func_entry;
138
QEMU_LOCK_GUARD(&yank_lock);
139
entry = yank_find_entry(instance);
142
QLIST_FOREACH(func_entry, &entry->yankfns, next) {
143
if (func_entry->func == func && func_entry->opaque == opaque) {
144
QLIST_REMOVE(func_entry, next);
153
void qmp_yank(YankInstanceList *instances,
156
YankInstanceList *tail;
157
YankInstanceEntry *entry;
158
YankFuncAndParam *func_entry;
160
QEMU_LOCK_GUARD(&yank_lock);
161
for (tail = instances; tail; tail = tail->next) {
162
entry = yank_find_entry(tail->value);
164
error_set(errp, ERROR_CLASS_DEVICE_NOT_FOUND, "Instance not found");
168
for (tail = instances; tail; tail = tail->next) {
169
entry = yank_find_entry(tail->value);
171
QLIST_FOREACH(func_entry, &entry->yankfns, next) {
172
func_entry->func(func_entry->opaque);
177
YankInstanceList *qmp_query_yank(Error **errp)
179
YankInstanceEntry *entry;
180
YankInstanceList *ret;
184
QEMU_LOCK_GUARD(&yank_lock);
185
QLIST_FOREACH(entry, &yank_instance_list, next) {
186
YankInstanceList *new_entry;
187
new_entry = g_new0(YankInstanceList, 1);
188
new_entry->value = QAPI_CLONE(YankInstance, entry->instance);
189
new_entry->next = ret;
196
static void __attribute__((__constructor__)) yank_init(void)
198
qemu_mutex_init(&yank_lock);