xref: /openbmc/qemu/hw/xen/xen-bus.c (revision 3c6ef471ee67bf5a22a9e0ecfdc45ca7d2393216)
1108f7bbaSPaul Durrant /*
2108f7bbaSPaul Durrant  * Copyright (c) 2018  Citrix Systems Inc.
3108f7bbaSPaul Durrant  *
4108f7bbaSPaul Durrant  * This work is licensed under the terms of the GNU GPL, version 2 or later.
5108f7bbaSPaul Durrant  * See the COPYING file in the top-level directory.
6108f7bbaSPaul Durrant  */
7108f7bbaSPaul Durrant 
8108f7bbaSPaul Durrant #include "qemu/osdep.h"
982a29e30SPaul Durrant #include "qemu/main-loop.h"
100b8fa32fSMarkus Armbruster #include "qemu/module.h"
1182a29e30SPaul Durrant #include "qemu/uuid.h"
12a27bd6c7SMarkus Armbruster #include "hw/qdev-properties.h"
13108f7bbaSPaul Durrant #include "hw/sysbus.h"
14094a2239SPaul Durrant #include "hw/xen/xen.h"
15a783f8adSPaul Durrant #include "hw/xen/xen-backend.h"
16108f7bbaSPaul Durrant #include "hw/xen/xen-bus.h"
17094a2239SPaul Durrant #include "hw/xen/xen-bus-helper.h"
18094a2239SPaul Durrant #include "monitor/monitor.h"
19108f7bbaSPaul Durrant #include "qapi/error.h"
20a783f8adSPaul Durrant #include "qapi/qmp/qdict.h"
21094a2239SPaul Durrant #include "sysemu/sysemu.h"
22108f7bbaSPaul Durrant #include "trace.h"
23108f7bbaSPaul Durrant 
24094a2239SPaul Durrant static char *xen_device_get_backend_path(XenDevice *xendev)
25094a2239SPaul Durrant {
26094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
27094a2239SPaul Durrant     XenDeviceClass *xendev_class = XEN_DEVICE_GET_CLASS(xendev);
28094a2239SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
29094a2239SPaul Durrant     const char *backend = xendev_class->backend;
30094a2239SPaul Durrant 
31094a2239SPaul Durrant     if (!backend) {
32094a2239SPaul Durrant         backend = type;
33094a2239SPaul Durrant     }
34094a2239SPaul Durrant 
35094a2239SPaul Durrant     return g_strdup_printf("/local/domain/%u/backend/%s/%u/%s",
36094a2239SPaul Durrant                            xenbus->backend_id, backend, xendev->frontend_id,
37094a2239SPaul Durrant                            xendev->name);
38094a2239SPaul Durrant }
39094a2239SPaul Durrant 
40094a2239SPaul Durrant static char *xen_device_get_frontend_path(XenDevice *xendev)
41094a2239SPaul Durrant {
42094a2239SPaul Durrant     XenDeviceClass *xendev_class = XEN_DEVICE_GET_CLASS(xendev);
43094a2239SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
44094a2239SPaul Durrant     const char *device = xendev_class->device;
45094a2239SPaul Durrant 
46094a2239SPaul Durrant     if (!device) {
47094a2239SPaul Durrant         device = type;
48094a2239SPaul Durrant     }
49094a2239SPaul Durrant 
50094a2239SPaul Durrant     return g_strdup_printf("/local/domain/%u/device/%s/%s",
51094a2239SPaul Durrant                            xendev->frontend_id, device, xendev->name);
52094a2239SPaul Durrant }
53094a2239SPaul Durrant 
54b6af8926SPaul Durrant static void xen_device_unplug(XenDevice *xendev, Error **errp)
55b6af8926SPaul Durrant {
56b6af8926SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
57b6af8926SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
58b6af8926SPaul Durrant     Error *local_err = NULL;
59b6af8926SPaul Durrant     xs_transaction_t tid;
60b6af8926SPaul Durrant 
61b6af8926SPaul Durrant     trace_xen_device_unplug(type, xendev->name);
62b6af8926SPaul Durrant 
63b6af8926SPaul Durrant     /* Mimic the way the Xen toolstack does an unplug */
64b6af8926SPaul Durrant again:
65b6af8926SPaul Durrant     tid = xs_transaction_start(xenbus->xsh);
66b6af8926SPaul Durrant     if (tid == XBT_NULL) {
67b6af8926SPaul Durrant         error_setg_errno(errp, errno, "failed xs_transaction_start");
68b6af8926SPaul Durrant         return;
69b6af8926SPaul Durrant     }
70b6af8926SPaul Durrant 
71b6af8926SPaul Durrant     xs_node_printf(xenbus->xsh, tid, xendev->backend_path, "online",
72b6af8926SPaul Durrant                    &local_err, "%u", 0);
73b6af8926SPaul Durrant     if (local_err) {
74b6af8926SPaul Durrant         goto abort;
75b6af8926SPaul Durrant     }
76b6af8926SPaul Durrant 
77b6af8926SPaul Durrant     xs_node_printf(xenbus->xsh, tid, xendev->backend_path, "state",
78b6af8926SPaul Durrant                    &local_err, "%u", XenbusStateClosing);
79b6af8926SPaul Durrant     if (local_err) {
80b6af8926SPaul Durrant         goto abort;
81b6af8926SPaul Durrant     }
82b6af8926SPaul Durrant 
83b6af8926SPaul Durrant     if (!xs_transaction_end(xenbus->xsh, tid, false)) {
84b6af8926SPaul Durrant         if (errno == EAGAIN) {
85b6af8926SPaul Durrant             goto again;
86b6af8926SPaul Durrant         }
87b6af8926SPaul Durrant 
88b6af8926SPaul Durrant         error_setg_errno(errp, errno, "failed xs_transaction_end");
89b6af8926SPaul Durrant     }
90b6af8926SPaul Durrant 
91b6af8926SPaul Durrant     return;
92b6af8926SPaul Durrant 
93b6af8926SPaul Durrant abort:
94b6af8926SPaul Durrant     /*
95b6af8926SPaul Durrant      * We only abort if there is already a failure so ignore any error
96b6af8926SPaul Durrant      * from ending the transaction.
97b6af8926SPaul Durrant      */
98b6af8926SPaul Durrant     xs_transaction_end(xenbus->xsh, tid, true);
99b6af8926SPaul Durrant     error_propagate(errp, local_err);
100b6af8926SPaul Durrant }
101b6af8926SPaul Durrant 
102094a2239SPaul Durrant static void xen_bus_print_dev(Monitor *mon, DeviceState *dev, int indent)
103094a2239SPaul Durrant {
104094a2239SPaul Durrant     XenDevice *xendev = XEN_DEVICE(dev);
105094a2239SPaul Durrant 
106094a2239SPaul Durrant     monitor_printf(mon, "%*sname = '%s' frontend_id = %u\n",
107094a2239SPaul Durrant                    indent, "", xendev->name, xendev->frontend_id);
108094a2239SPaul Durrant }
109094a2239SPaul Durrant 
110094a2239SPaul Durrant static char *xen_bus_get_dev_path(DeviceState *dev)
111094a2239SPaul Durrant {
112094a2239SPaul Durrant     return xen_device_get_backend_path(XEN_DEVICE(dev));
113094a2239SPaul Durrant }
114094a2239SPaul Durrant 
11582a29e30SPaul Durrant struct XenWatch {
11682a29e30SPaul Durrant     char *node, *key;
11782a29e30SPaul Durrant     char *token;
11882a29e30SPaul Durrant     XenWatchHandler handler;
11982a29e30SPaul Durrant     void *opaque;
12082a29e30SPaul Durrant     Notifier notifier;
12182a29e30SPaul Durrant };
12282a29e30SPaul Durrant 
12382a29e30SPaul Durrant static void watch_notify(Notifier *n, void *data)
12482a29e30SPaul Durrant {
12582a29e30SPaul Durrant     XenWatch *watch = container_of(n, XenWatch, notifier);
12682a29e30SPaul Durrant     const char *token = data;
12782a29e30SPaul Durrant 
12882a29e30SPaul Durrant     if (!strcmp(watch->token, token)) {
12982a29e30SPaul Durrant         watch->handler(watch->opaque);
13082a29e30SPaul Durrant     }
13182a29e30SPaul Durrant }
13282a29e30SPaul Durrant 
13382a29e30SPaul Durrant static XenWatch *new_watch(const char *node, const char *key,
13482a29e30SPaul Durrant                            XenWatchHandler handler, void *opaque)
13582a29e30SPaul Durrant {
13682a29e30SPaul Durrant     XenWatch *watch = g_new0(XenWatch, 1);
13782a29e30SPaul Durrant     QemuUUID uuid;
13882a29e30SPaul Durrant 
13982a29e30SPaul Durrant     qemu_uuid_generate(&uuid);
14082a29e30SPaul Durrant 
14182a29e30SPaul Durrant     watch->token = qemu_uuid_unparse_strdup(&uuid);
14282a29e30SPaul Durrant     watch->node = g_strdup(node);
14382a29e30SPaul Durrant     watch->key = g_strdup(key);
14482a29e30SPaul Durrant     watch->handler = handler;
14582a29e30SPaul Durrant     watch->opaque = opaque;
14682a29e30SPaul Durrant     watch->notifier.notify = watch_notify;
14782a29e30SPaul Durrant 
14882a29e30SPaul Durrant     return watch;
14982a29e30SPaul Durrant }
15082a29e30SPaul Durrant 
15182a29e30SPaul Durrant static void free_watch(XenWatch *watch)
15282a29e30SPaul Durrant {
15382a29e30SPaul Durrant     g_free(watch->token);
15482a29e30SPaul Durrant     g_free(watch->key);
15582a29e30SPaul Durrant     g_free(watch->node);
15682a29e30SPaul Durrant 
15782a29e30SPaul Durrant     g_free(watch);
15882a29e30SPaul Durrant }
15982a29e30SPaul Durrant 
160374752a2SPaul Durrant struct XenWatchList {
161374752a2SPaul Durrant     struct xs_handle *xsh;
162374752a2SPaul Durrant     NotifierList notifiers;
163374752a2SPaul Durrant };
164374752a2SPaul Durrant 
165374752a2SPaul Durrant static void watch_list_event(void *opaque)
166374752a2SPaul Durrant {
167374752a2SPaul Durrant     XenWatchList *watch_list = opaque;
168374752a2SPaul Durrant     char **v;
169374752a2SPaul Durrant     const char *token;
170374752a2SPaul Durrant 
171374752a2SPaul Durrant     v = xs_check_watch(watch_list->xsh);
172374752a2SPaul Durrant     if (!v) {
173374752a2SPaul Durrant         return;
174374752a2SPaul Durrant     }
175374752a2SPaul Durrant 
176374752a2SPaul Durrant     token = v[XS_WATCH_TOKEN];
177374752a2SPaul Durrant 
178374752a2SPaul Durrant     notifier_list_notify(&watch_list->notifiers, (void *)token);
179374752a2SPaul Durrant 
180374752a2SPaul Durrant     free(v);
181374752a2SPaul Durrant }
182374752a2SPaul Durrant 
183374752a2SPaul Durrant static XenWatchList *watch_list_create(struct xs_handle *xsh)
184374752a2SPaul Durrant {
185374752a2SPaul Durrant     XenWatchList *watch_list = g_new0(XenWatchList, 1);
186374752a2SPaul Durrant 
187374752a2SPaul Durrant     g_assert(xsh);
188374752a2SPaul Durrant 
189374752a2SPaul Durrant     watch_list->xsh = xsh;
190374752a2SPaul Durrant     notifier_list_init(&watch_list->notifiers);
191374752a2SPaul Durrant     qemu_set_fd_handler(xs_fileno(watch_list->xsh), watch_list_event, NULL,
192374752a2SPaul Durrant                         watch_list);
193374752a2SPaul Durrant 
194374752a2SPaul Durrant     return watch_list;
195374752a2SPaul Durrant }
196374752a2SPaul Durrant 
197374752a2SPaul Durrant static void watch_list_destroy(XenWatchList *watch_list)
198374752a2SPaul Durrant {
199374752a2SPaul Durrant     g_assert(notifier_list_empty(&watch_list->notifiers));
200374752a2SPaul Durrant     qemu_set_fd_handler(xs_fileno(watch_list->xsh), NULL, NULL, NULL);
201374752a2SPaul Durrant     g_free(watch_list);
202374752a2SPaul Durrant }
203374752a2SPaul Durrant 
204374752a2SPaul Durrant static XenWatch *watch_list_add(XenWatchList *watch_list, const char *node,
20582a29e30SPaul Durrant                                 const char *key, XenWatchHandler handler,
20682a29e30SPaul Durrant                                 void *opaque, Error **errp)
20782a29e30SPaul Durrant {
20882a29e30SPaul Durrant     XenWatch *watch = new_watch(node, key, handler, opaque);
20982a29e30SPaul Durrant     Error *local_err = NULL;
21082a29e30SPaul Durrant 
211374752a2SPaul Durrant     notifier_list_add(&watch_list->notifiers, &watch->notifier);
21282a29e30SPaul Durrant 
213374752a2SPaul Durrant     xs_node_watch(watch_list->xsh, node, key, watch->token, &local_err);
21482a29e30SPaul Durrant     if (local_err) {
21582a29e30SPaul Durrant         error_propagate(errp, local_err);
21682a29e30SPaul Durrant 
21782a29e30SPaul Durrant         notifier_remove(&watch->notifier);
21882a29e30SPaul Durrant         free_watch(watch);
21982a29e30SPaul Durrant 
22082a29e30SPaul Durrant         return NULL;
22182a29e30SPaul Durrant     }
22282a29e30SPaul Durrant 
22382a29e30SPaul Durrant     return watch;
22482a29e30SPaul Durrant }
22582a29e30SPaul Durrant 
226374752a2SPaul Durrant static void watch_list_remove(XenWatchList *watch_list, XenWatch *watch,
22782a29e30SPaul Durrant                               Error **errp)
22882a29e30SPaul Durrant {
229374752a2SPaul Durrant     xs_node_unwatch(watch_list->xsh, watch->node, watch->key, watch->token,
23082a29e30SPaul Durrant                     errp);
23182a29e30SPaul Durrant 
23282a29e30SPaul Durrant     notifier_remove(&watch->notifier);
23382a29e30SPaul Durrant     free_watch(watch);
23482a29e30SPaul Durrant }
23582a29e30SPaul Durrant 
236374752a2SPaul Durrant static XenWatch *xen_bus_add_watch(XenBus *xenbus, const char *node,
237374752a2SPaul Durrant                                    const char *key, XenWatchHandler handler,
238d198b711SPaul Durrant                                    Error **errp)
239374752a2SPaul Durrant {
240374752a2SPaul Durrant     trace_xen_bus_add_watch(node, key);
241374752a2SPaul Durrant 
242d198b711SPaul Durrant     return watch_list_add(xenbus->watch_list, node, key, handler, xenbus,
243374752a2SPaul Durrant                           errp);
244374752a2SPaul Durrant }
245374752a2SPaul Durrant 
246374752a2SPaul Durrant static void xen_bus_remove_watch(XenBus *xenbus, XenWatch *watch,
247374752a2SPaul Durrant                                  Error **errp)
248374752a2SPaul Durrant {
249374752a2SPaul Durrant     trace_xen_bus_remove_watch(watch->node, watch->key);
250374752a2SPaul Durrant 
251374752a2SPaul Durrant     watch_list_remove(xenbus->watch_list, watch, errp);
252374752a2SPaul Durrant }
253374752a2SPaul Durrant 
254a783f8adSPaul Durrant static void xen_bus_backend_create(XenBus *xenbus, const char *type,
255a783f8adSPaul Durrant                                    const char *name, char *path,
256a783f8adSPaul Durrant                                    Error **errp)
257a783f8adSPaul Durrant {
258a783f8adSPaul Durrant     xs_transaction_t tid;
259a783f8adSPaul Durrant     char **key;
260a783f8adSPaul Durrant     QDict *opts;
261a783f8adSPaul Durrant     unsigned int i, n;
262a783f8adSPaul Durrant     Error *local_err = NULL;
263a783f8adSPaul Durrant 
264a783f8adSPaul Durrant     trace_xen_bus_backend_create(type, path);
265a783f8adSPaul Durrant 
266a783f8adSPaul Durrant again:
267a783f8adSPaul Durrant     tid = xs_transaction_start(xenbus->xsh);
268a783f8adSPaul Durrant     if (tid == XBT_NULL) {
269a783f8adSPaul Durrant         error_setg(errp, "failed xs_transaction_start");
270a783f8adSPaul Durrant         return;
271a783f8adSPaul Durrant     }
272a783f8adSPaul Durrant 
273a783f8adSPaul Durrant     key = xs_directory(xenbus->xsh, tid, path, &n);
274a783f8adSPaul Durrant     if (!key) {
275a783f8adSPaul Durrant         if (!xs_transaction_end(xenbus->xsh, tid, true)) {
276a783f8adSPaul Durrant             error_setg_errno(errp, errno, "failed xs_transaction_end");
277a783f8adSPaul Durrant         }
278a783f8adSPaul Durrant         return;
279a783f8adSPaul Durrant     }
280a783f8adSPaul Durrant 
281a783f8adSPaul Durrant     opts = qdict_new();
282a783f8adSPaul Durrant     for (i = 0; i < n; i++) {
283a783f8adSPaul Durrant         char *val;
284a783f8adSPaul Durrant 
285a783f8adSPaul Durrant         /*
286a783f8adSPaul Durrant          * Assume anything found in the xenstore backend area, other than
287a783f8adSPaul Durrant          * the keys created for a generic XenDevice, are parameters
288a783f8adSPaul Durrant          * to be used to configure the backend.
289a783f8adSPaul Durrant          */
290a783f8adSPaul Durrant         if (!strcmp(key[i], "state") ||
291a783f8adSPaul Durrant             !strcmp(key[i], "online") ||
292a783f8adSPaul Durrant             !strcmp(key[i], "frontend") ||
293a783f8adSPaul Durrant             !strcmp(key[i], "frontend-id") ||
294a783f8adSPaul Durrant             !strcmp(key[i], "hotplug-status"))
295a783f8adSPaul Durrant             continue;
296a783f8adSPaul Durrant 
297a783f8adSPaul Durrant         if (xs_node_scanf(xenbus->xsh, tid, path, key[i], NULL, "%ms",
298a783f8adSPaul Durrant                           &val) == 1) {
299a783f8adSPaul Durrant             qdict_put_str(opts, key[i], val);
300a783f8adSPaul Durrant             free(val);
301a783f8adSPaul Durrant         }
302a783f8adSPaul Durrant     }
303a783f8adSPaul Durrant 
304a783f8adSPaul Durrant     free(key);
305a783f8adSPaul Durrant 
306a783f8adSPaul Durrant     if (!xs_transaction_end(xenbus->xsh, tid, false)) {
307a783f8adSPaul Durrant         qobject_unref(opts);
308a783f8adSPaul Durrant 
309a783f8adSPaul Durrant         if (errno == EAGAIN) {
310a783f8adSPaul Durrant             goto again;
311a783f8adSPaul Durrant         }
312a783f8adSPaul Durrant 
313a783f8adSPaul Durrant         error_setg_errno(errp, errno, "failed xs_transaction_end");
314a783f8adSPaul Durrant         return;
315a783f8adSPaul Durrant     }
316a783f8adSPaul Durrant 
317a783f8adSPaul Durrant     xen_backend_device_create(xenbus, type, name, opts, &local_err);
318a783f8adSPaul Durrant     qobject_unref(opts);
319a783f8adSPaul Durrant 
320a783f8adSPaul Durrant     if (local_err) {
321a783f8adSPaul Durrant         error_propagate_prepend(errp, local_err,
322a783f8adSPaul Durrant                                 "failed to create '%s' device '%s': ",
323a783f8adSPaul Durrant                                 type, name);
324a783f8adSPaul Durrant     }
325a783f8adSPaul Durrant }
326a783f8adSPaul Durrant 
327a783f8adSPaul Durrant static void xen_bus_type_enumerate(XenBus *xenbus, const char *type)
328a783f8adSPaul Durrant {
329a783f8adSPaul Durrant     char *domain_path = g_strdup_printf("backend/%s/%u", type, xen_domid);
330a783f8adSPaul Durrant     char **backend;
331a783f8adSPaul Durrant     unsigned int i, n;
332a783f8adSPaul Durrant 
333a783f8adSPaul Durrant     trace_xen_bus_type_enumerate(type);
334a783f8adSPaul Durrant 
335a783f8adSPaul Durrant     backend = xs_directory(xenbus->xsh, XBT_NULL, domain_path, &n);
336a783f8adSPaul Durrant     if (!backend) {
337a783f8adSPaul Durrant         goto out;
338a783f8adSPaul Durrant     }
339a783f8adSPaul Durrant 
340a783f8adSPaul Durrant     for (i = 0; i < n; i++) {
341a783f8adSPaul Durrant         char *backend_path = g_strdup_printf("%s/%s", domain_path,
342a783f8adSPaul Durrant                                              backend[i]);
3433809f758SPaul Durrant         enum xenbus_state state;
3443809f758SPaul Durrant         unsigned int online;
345a783f8adSPaul Durrant 
346a783f8adSPaul Durrant         if (xs_node_scanf(xenbus->xsh, XBT_NULL, backend_path, "state",
3473809f758SPaul Durrant                           NULL, "%u", &state) != 1)
3483809f758SPaul Durrant             state = XenbusStateUnknown;
349a783f8adSPaul Durrant 
3503809f758SPaul Durrant         if (xs_node_scanf(xenbus->xsh, XBT_NULL, backend_path, "online",
3513809f758SPaul Durrant                           NULL, "%u", &online) != 1)
3523809f758SPaul Durrant             online = 0;
3533809f758SPaul Durrant 
3543809f758SPaul Durrant         if (online && state == XenbusStateInitialising) {
355a783f8adSPaul Durrant             Error *local_err = NULL;
356a783f8adSPaul Durrant 
357a783f8adSPaul Durrant             xen_bus_backend_create(xenbus, type, backend[i], backend_path,
358a783f8adSPaul Durrant                                    &local_err);
359a783f8adSPaul Durrant             if (local_err) {
360a783f8adSPaul Durrant                 error_report_err(local_err);
361a783f8adSPaul Durrant             }
362a783f8adSPaul Durrant         }
363a783f8adSPaul Durrant 
364a783f8adSPaul Durrant         g_free(backend_path);
365a783f8adSPaul Durrant     }
366a783f8adSPaul Durrant 
367a783f8adSPaul Durrant     free(backend);
368a783f8adSPaul Durrant 
369a783f8adSPaul Durrant out:
370a783f8adSPaul Durrant     g_free(domain_path);
371a783f8adSPaul Durrant }
372a783f8adSPaul Durrant 
3733809f758SPaul Durrant static void xen_bus_enumerate(XenBus *xenbus)
374a783f8adSPaul Durrant {
375a783f8adSPaul Durrant     char **type;
376a783f8adSPaul Durrant     unsigned int i, n;
377a783f8adSPaul Durrant 
378a783f8adSPaul Durrant     trace_xen_bus_enumerate();
379a783f8adSPaul Durrant 
380a783f8adSPaul Durrant     type = xs_directory(xenbus->xsh, XBT_NULL, "backend", &n);
381a783f8adSPaul Durrant     if (!type) {
382a783f8adSPaul Durrant         return;
383a783f8adSPaul Durrant     }
384a783f8adSPaul Durrant 
385a783f8adSPaul Durrant     for (i = 0; i < n; i++) {
386a783f8adSPaul Durrant         xen_bus_type_enumerate(xenbus, type[i]);
387a783f8adSPaul Durrant     }
388a783f8adSPaul Durrant 
389a783f8adSPaul Durrant     free(type);
390a783f8adSPaul Durrant }
391a783f8adSPaul Durrant 
3923809f758SPaul Durrant static void xen_bus_device_cleanup(XenDevice *xendev)
3933809f758SPaul Durrant {
3943809f758SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
3953809f758SPaul Durrant     Error *local_err = NULL;
3963809f758SPaul Durrant 
3973809f758SPaul Durrant     trace_xen_bus_device_cleanup(type, xendev->name);
3983809f758SPaul Durrant 
3993809f758SPaul Durrant     g_assert(!xendev->backend_online);
4003809f758SPaul Durrant 
4013809f758SPaul Durrant     if (!xen_backend_try_device_destroy(xendev, &local_err)) {
4023809f758SPaul Durrant         object_unparent(OBJECT(xendev));
4033809f758SPaul Durrant     }
4043809f758SPaul Durrant 
4053809f758SPaul Durrant     if (local_err) {
4063809f758SPaul Durrant         error_report_err(local_err);
4073809f758SPaul Durrant     }
4083809f758SPaul Durrant }
4093809f758SPaul Durrant 
4103809f758SPaul Durrant static void xen_bus_cleanup(XenBus *xenbus)
4113809f758SPaul Durrant {
4123809f758SPaul Durrant     XenDevice *xendev, *next;
4133809f758SPaul Durrant 
4143809f758SPaul Durrant     trace_xen_bus_cleanup();
4153809f758SPaul Durrant 
4163809f758SPaul Durrant     QLIST_FOREACH_SAFE(xendev, &xenbus->inactive_devices, list, next) {
4173809f758SPaul Durrant         g_assert(xendev->inactive);
4183809f758SPaul Durrant         QLIST_REMOVE(xendev, list);
4193809f758SPaul Durrant         xen_bus_device_cleanup(xendev);
4203809f758SPaul Durrant     }
4213809f758SPaul Durrant }
4223809f758SPaul Durrant 
4233809f758SPaul Durrant static void xen_bus_backend_changed(void *opaque)
4243809f758SPaul Durrant {
4253809f758SPaul Durrant     XenBus *xenbus = opaque;
4263809f758SPaul Durrant 
4273809f758SPaul Durrant     xen_bus_enumerate(xenbus);
4283809f758SPaul Durrant     xen_bus_cleanup(xenbus);
4293809f758SPaul Durrant }
4303809f758SPaul Durrant 
431b69c3c21SMarkus Armbruster static void xen_bus_unrealize(BusState *bus)
432108f7bbaSPaul Durrant {
433094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(bus);
434094a2239SPaul Durrant 
435108f7bbaSPaul Durrant     trace_xen_bus_unrealize();
436094a2239SPaul Durrant 
437a783f8adSPaul Durrant     if (xenbus->backend_watch) {
438a783f8adSPaul Durrant         xen_bus_remove_watch(xenbus, xenbus->backend_watch, NULL);
439a783f8adSPaul Durrant         xenbus->backend_watch = NULL;
440a783f8adSPaul Durrant     }
441a783f8adSPaul Durrant 
442374752a2SPaul Durrant     if (xenbus->watch_list) {
443374752a2SPaul Durrant         watch_list_destroy(xenbus->watch_list);
444374752a2SPaul Durrant         xenbus->watch_list = NULL;
445094a2239SPaul Durrant     }
446094a2239SPaul Durrant 
447374752a2SPaul Durrant     if (xenbus->xsh) {
448094a2239SPaul Durrant         xs_close(xenbus->xsh);
449108f7bbaSPaul Durrant     }
45082a29e30SPaul Durrant }
45182a29e30SPaul Durrant 
452108f7bbaSPaul Durrant static void xen_bus_realize(BusState *bus, Error **errp)
453108f7bbaSPaul Durrant {
454094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(bus);
455094a2239SPaul Durrant     unsigned int domid;
456a783f8adSPaul Durrant     Error *local_err = NULL;
457094a2239SPaul Durrant 
458108f7bbaSPaul Durrant     trace_xen_bus_realize();
459094a2239SPaul Durrant 
460094a2239SPaul Durrant     xenbus->xsh = xs_open(0);
461094a2239SPaul Durrant     if (!xenbus->xsh) {
462094a2239SPaul Durrant         error_setg_errno(errp, errno, "failed xs_open");
463094a2239SPaul Durrant         goto fail;
464094a2239SPaul Durrant     }
465094a2239SPaul Durrant 
466094a2239SPaul Durrant     if (xs_node_scanf(xenbus->xsh, XBT_NULL, "", /* domain root node */
467094a2239SPaul Durrant                       "domid", NULL, "%u", &domid) == 1) {
468094a2239SPaul Durrant         xenbus->backend_id = domid;
469094a2239SPaul Durrant     } else {
470094a2239SPaul Durrant         xenbus->backend_id = 0; /* Assume lack of node means dom0 */
471094a2239SPaul Durrant     }
472094a2239SPaul Durrant 
473374752a2SPaul Durrant     xenbus->watch_list = watch_list_create(xenbus->xsh);
474a783f8adSPaul Durrant 
475a783f8adSPaul Durrant     module_call_init(MODULE_INIT_XEN_BACKEND);
476a783f8adSPaul Durrant 
477a783f8adSPaul Durrant     xenbus->backend_watch =
478a783f8adSPaul Durrant         xen_bus_add_watch(xenbus, "", /* domain root node */
4793809f758SPaul Durrant                           "backend", xen_bus_backend_changed, &local_err);
480a783f8adSPaul Durrant     if (local_err) {
481a783f8adSPaul Durrant         /* This need not be treated as a hard error so don't propagate */
482a783f8adSPaul Durrant         error_reportf_err(local_err,
483a783f8adSPaul Durrant                           "failed to set up enumeration watch: ");
484a783f8adSPaul Durrant     }
485a783f8adSPaul Durrant 
486094a2239SPaul Durrant     return;
487094a2239SPaul Durrant 
488094a2239SPaul Durrant fail:
489b69c3c21SMarkus Armbruster     xen_bus_unrealize(bus);
490108f7bbaSPaul Durrant }
491108f7bbaSPaul Durrant 
492b6af8926SPaul Durrant static void xen_bus_unplug_request(HotplugHandler *hotplug,
493b6af8926SPaul Durrant                                    DeviceState *dev,
494b6af8926SPaul Durrant                                    Error **errp)
495b6af8926SPaul Durrant {
496b6af8926SPaul Durrant     XenDevice *xendev = XEN_DEVICE(dev);
497b6af8926SPaul Durrant 
498b6af8926SPaul Durrant     xen_device_unplug(xendev, errp);
499b6af8926SPaul Durrant }
500b6af8926SPaul Durrant 
501108f7bbaSPaul Durrant static void xen_bus_class_init(ObjectClass *class, void *data)
502108f7bbaSPaul Durrant {
503108f7bbaSPaul Durrant     BusClass *bus_class = BUS_CLASS(class);
504b6af8926SPaul Durrant     HotplugHandlerClass *hotplug_class = HOTPLUG_HANDLER_CLASS(class);
505108f7bbaSPaul Durrant 
506094a2239SPaul Durrant     bus_class->print_dev = xen_bus_print_dev;
507094a2239SPaul Durrant     bus_class->get_dev_path = xen_bus_get_dev_path;
508108f7bbaSPaul Durrant     bus_class->realize = xen_bus_realize;
509108f7bbaSPaul Durrant     bus_class->unrealize = xen_bus_unrealize;
510b6af8926SPaul Durrant 
511b6af8926SPaul Durrant     hotplug_class->unplug_request = xen_bus_unplug_request;
512108f7bbaSPaul Durrant }
513108f7bbaSPaul Durrant 
514108f7bbaSPaul Durrant static const TypeInfo xen_bus_type_info = {
515108f7bbaSPaul Durrant     .name = TYPE_XEN_BUS,
516108f7bbaSPaul Durrant     .parent = TYPE_BUS,
517108f7bbaSPaul Durrant     .instance_size = sizeof(XenBus),
518108f7bbaSPaul Durrant     .class_size = sizeof(XenBusClass),
519108f7bbaSPaul Durrant     .class_init = xen_bus_class_init,
520108f7bbaSPaul Durrant     .interfaces = (InterfaceInfo[]) {
521108f7bbaSPaul Durrant         { TYPE_HOTPLUG_HANDLER },
522108f7bbaSPaul Durrant         { }
523108f7bbaSPaul Durrant     },
524108f7bbaSPaul Durrant };
525108f7bbaSPaul Durrant 
526b6af8926SPaul Durrant void xen_device_backend_printf(XenDevice *xendev, const char *key,
527094a2239SPaul Durrant                                const char *fmt, ...)
528094a2239SPaul Durrant {
529094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
530094a2239SPaul Durrant     Error *local_err = NULL;
531094a2239SPaul Durrant     va_list ap;
532094a2239SPaul Durrant 
533094a2239SPaul Durrant     g_assert(xenbus->xsh);
534094a2239SPaul Durrant 
535094a2239SPaul Durrant     va_start(ap, fmt);
536094a2239SPaul Durrant     xs_node_vprintf(xenbus->xsh, XBT_NULL, xendev->backend_path, key,
537094a2239SPaul Durrant                     &local_err, fmt, ap);
538094a2239SPaul Durrant     va_end(ap);
539094a2239SPaul Durrant 
540094a2239SPaul Durrant     if (local_err) {
541094a2239SPaul Durrant         error_report_err(local_err);
542094a2239SPaul Durrant     }
543094a2239SPaul Durrant }
544094a2239SPaul Durrant 
54582a29e30SPaul Durrant static int xen_device_backend_scanf(XenDevice *xendev, const char *key,
54682a29e30SPaul Durrant                                     const char *fmt, ...)
54782a29e30SPaul Durrant {
54882a29e30SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
54982a29e30SPaul Durrant     va_list ap;
55082a29e30SPaul Durrant     int rc;
55182a29e30SPaul Durrant 
55282a29e30SPaul Durrant     g_assert(xenbus->xsh);
55382a29e30SPaul Durrant 
55482a29e30SPaul Durrant     va_start(ap, fmt);
55582a29e30SPaul Durrant     rc = xs_node_vscanf(xenbus->xsh, XBT_NULL, xendev->backend_path, key,
55682a29e30SPaul Durrant                         NULL, fmt, ap);
55782a29e30SPaul Durrant     va_end(ap);
55882a29e30SPaul Durrant 
55982a29e30SPaul Durrant     return rc;
56082a29e30SPaul Durrant }
56182a29e30SPaul Durrant 
56282a29e30SPaul Durrant void xen_device_backend_set_state(XenDevice *xendev,
563094a2239SPaul Durrant                                   enum xenbus_state state)
564094a2239SPaul Durrant {
565094a2239SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
566094a2239SPaul Durrant 
567094a2239SPaul Durrant     if (xendev->backend_state == state) {
568094a2239SPaul Durrant         return;
569094a2239SPaul Durrant     }
570094a2239SPaul Durrant 
571094a2239SPaul Durrant     trace_xen_device_backend_state(type, xendev->name,
572094a2239SPaul Durrant                                    xs_strstate(state));
573094a2239SPaul Durrant 
574094a2239SPaul Durrant     xendev->backend_state = state;
575094a2239SPaul Durrant     xen_device_backend_printf(xendev, "state", "%u", state);
576094a2239SPaul Durrant }
577094a2239SPaul Durrant 
57882a29e30SPaul Durrant enum xenbus_state xen_device_backend_get_state(XenDevice *xendev)
57982a29e30SPaul Durrant {
58082a29e30SPaul Durrant     return xendev->backend_state;
58182a29e30SPaul Durrant }
58282a29e30SPaul Durrant 
583b6af8926SPaul Durrant static void xen_device_backend_set_online(XenDevice *xendev, bool online)
584b6af8926SPaul Durrant {
585b6af8926SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
586b6af8926SPaul Durrant 
587b6af8926SPaul Durrant     if (xendev->backend_online == online) {
588b6af8926SPaul Durrant         return;
589b6af8926SPaul Durrant     }
590b6af8926SPaul Durrant 
591b6af8926SPaul Durrant     trace_xen_device_backend_online(type, xendev->name, online);
592b6af8926SPaul Durrant 
593b6af8926SPaul Durrant     xendev->backend_online = online;
594b6af8926SPaul Durrant     xen_device_backend_printf(xendev, "online", "%u", online);
595b6af8926SPaul Durrant }
596b6af8926SPaul Durrant 
597cb323146SAnthony PERARD /*
598cb323146SAnthony PERARD  * Tell from the state whether the frontend is likely alive,
599cb323146SAnthony PERARD  * i.e. it will react to a change of state of the backend.
600cb323146SAnthony PERARD  */
6013809f758SPaul Durrant static bool xen_device_frontend_is_active(XenDevice *xendev)
602cb323146SAnthony PERARD {
6033809f758SPaul Durrant     switch (xendev->frontend_state) {
604cb323146SAnthony PERARD     case XenbusStateInitWait:
605cb323146SAnthony PERARD     case XenbusStateInitialised:
606cb323146SAnthony PERARD     case XenbusStateConnected:
607cb323146SAnthony PERARD     case XenbusStateClosing:
608cb323146SAnthony PERARD         return true;
609cb323146SAnthony PERARD     default:
610cb323146SAnthony PERARD         return false;
611cb323146SAnthony PERARD     }
612cb323146SAnthony PERARD }
613cb323146SAnthony PERARD 
614b6af8926SPaul Durrant static void xen_device_backend_changed(void *opaque)
615b6af8926SPaul Durrant {
616b6af8926SPaul Durrant     XenDevice *xendev = opaque;
617b6af8926SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
618b6af8926SPaul Durrant     enum xenbus_state state;
619b6af8926SPaul Durrant     unsigned int online;
620b6af8926SPaul Durrant 
621b6af8926SPaul Durrant     trace_xen_device_backend_changed(type, xendev->name);
622b6af8926SPaul Durrant 
623b6af8926SPaul Durrant     if (xen_device_backend_scanf(xendev, "state", "%u", &state) != 1) {
624b6af8926SPaul Durrant         state = XenbusStateUnknown;
625b6af8926SPaul Durrant     }
626b6af8926SPaul Durrant 
627b6af8926SPaul Durrant     xen_device_backend_set_state(xendev, state);
628b6af8926SPaul Durrant 
629b6af8926SPaul Durrant     if (xen_device_backend_scanf(xendev, "online", "%u", &online) != 1) {
630b6af8926SPaul Durrant         online = 0;
631b6af8926SPaul Durrant     }
632b6af8926SPaul Durrant 
633b6af8926SPaul Durrant     xen_device_backend_set_online(xendev, !!online);
634b6af8926SPaul Durrant 
635b6af8926SPaul Durrant     /*
636b6af8926SPaul Durrant      * If the toolstack (or unplug request callback) has set the backend
637cb323146SAnthony PERARD      * state to Closing, but there is no active frontend then set the
638cb323146SAnthony PERARD      * backend state to Closed.
639b6af8926SPaul Durrant      */
6403809f758SPaul Durrant     if (state == XenbusStateClosing &&
6413809f758SPaul Durrant         !xen_device_frontend_is_active(xendev)) {
642b6af8926SPaul Durrant         xen_device_backend_set_state(xendev, XenbusStateClosed);
643b6af8926SPaul Durrant     }
644b6af8926SPaul Durrant 
645b6af8926SPaul Durrant     /*
64667bc8e00SPaul Durrant      * If a backend is still 'online' then we should leave it alone but,
6473809f758SPaul Durrant      * if a backend is not 'online', then the device is a candidate
6483809f758SPaul Durrant      * for destruction. Hence add it to the 'inactive' list to be cleaned
6493809f758SPaul Durrant      * by xen_bus_cleanup().
650b6af8926SPaul Durrant      */
6513809f758SPaul Durrant     if (!online &&
6523809f758SPaul Durrant         (state == XenbusStateClosed ||  state == XenbusStateInitialising ||
6533809f758SPaul Durrant          state == XenbusStateInitWait || state == XenbusStateUnknown) &&
6543809f758SPaul Durrant         !xendev->inactive) {
6553809f758SPaul Durrant         XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
656a783f8adSPaul Durrant 
6573809f758SPaul Durrant         xendev->inactive = true;
6583809f758SPaul Durrant         QLIST_INSERT_HEAD(&xenbus->inactive_devices, xendev, list);
659a783f8adSPaul Durrant 
6603809f758SPaul Durrant         /*
6613809f758SPaul Durrant          * Re-write the state to cause a XenBus backend_watch notification,
6623809f758SPaul Durrant          * resulting in a call to xen_bus_cleanup().
6633809f758SPaul Durrant          */
6643809f758SPaul Durrant         xen_device_backend_printf(xendev, "state", "%u", state);
665a783f8adSPaul Durrant     }
666b6af8926SPaul Durrant }
667b6af8926SPaul Durrant 
668d198b711SPaul Durrant static XenWatch *xen_device_add_watch(XenDevice *xendev, const char *node,
669d198b711SPaul Durrant                                       const char *key,
670d198b711SPaul Durrant                                       XenWatchHandler handler,
671d198b711SPaul Durrant                                       Error **errp)
672d198b711SPaul Durrant {
673d198b711SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
674d198b711SPaul Durrant 
675d198b711SPaul Durrant     trace_xen_device_add_watch(type, xendev->name, node, key);
676d198b711SPaul Durrant 
677d198b711SPaul Durrant     return watch_list_add(xendev->watch_list, node, key, handler, xendev,
678d198b711SPaul Durrant                           errp);
679d198b711SPaul Durrant }
680d198b711SPaul Durrant 
681d198b711SPaul Durrant static void xen_device_remove_watch(XenDevice *xendev, XenWatch *watch,
682d198b711SPaul Durrant                                     Error **errp)
683d198b711SPaul Durrant {
684d198b711SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
685d198b711SPaul Durrant 
686d198b711SPaul Durrant     trace_xen_device_remove_watch(type, xendev->name, watch->node,
687d198b711SPaul Durrant                                   watch->key);
688d198b711SPaul Durrant 
689d198b711SPaul Durrant     watch_list_remove(xendev->watch_list, watch, errp);
690d198b711SPaul Durrant }
691d198b711SPaul Durrant 
692d198b711SPaul Durrant 
693094a2239SPaul Durrant static void xen_device_backend_create(XenDevice *xendev, Error **errp)
694094a2239SPaul Durrant {
695094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
696094a2239SPaul Durrant     struct xs_permissions perms[2];
697094a2239SPaul Durrant     Error *local_err = NULL;
698094a2239SPaul Durrant 
699094a2239SPaul Durrant     xendev->backend_path = xen_device_get_backend_path(xendev);
700094a2239SPaul Durrant 
701094a2239SPaul Durrant     perms[0].id = xenbus->backend_id;
702094a2239SPaul Durrant     perms[0].perms = XS_PERM_NONE;
703094a2239SPaul Durrant     perms[1].id = xendev->frontend_id;
704094a2239SPaul Durrant     perms[1].perms = XS_PERM_READ;
705094a2239SPaul Durrant 
706094a2239SPaul Durrant     g_assert(xenbus->xsh);
707094a2239SPaul Durrant 
708094a2239SPaul Durrant     xs_node_create(xenbus->xsh, XBT_NULL, xendev->backend_path, perms,
709094a2239SPaul Durrant                    ARRAY_SIZE(perms), &local_err);
710094a2239SPaul Durrant     if (local_err) {
711094a2239SPaul Durrant         error_propagate_prepend(errp, local_err,
712094a2239SPaul Durrant                                 "failed to create backend: ");
713b6af8926SPaul Durrant         return;
714b6af8926SPaul Durrant     }
715b6af8926SPaul Durrant 
716b6af8926SPaul Durrant     xendev->backend_state_watch =
717d198b711SPaul Durrant         xen_device_add_watch(xendev, xendev->backend_path,
718b6af8926SPaul Durrant                              "state", xen_device_backend_changed,
719d198b711SPaul Durrant                              &local_err);
720b6af8926SPaul Durrant     if (local_err) {
721b6af8926SPaul Durrant         error_propagate_prepend(errp, local_err,
722b6af8926SPaul Durrant                                 "failed to watch backend state: ");
723b6af8926SPaul Durrant         return;
724b6af8926SPaul Durrant     }
725b6af8926SPaul Durrant 
726b6af8926SPaul Durrant     xendev->backend_online_watch =
727d198b711SPaul Durrant         xen_device_add_watch(xendev, xendev->backend_path,
728b6af8926SPaul Durrant                              "online", xen_device_backend_changed,
729d198b711SPaul Durrant                              &local_err);
730b6af8926SPaul Durrant     if (local_err) {
731b6af8926SPaul Durrant         error_propagate_prepend(errp, local_err,
732b6af8926SPaul Durrant                                 "failed to watch backend online: ");
733b6af8926SPaul Durrant         return;
734094a2239SPaul Durrant     }
735094a2239SPaul Durrant }
736094a2239SPaul Durrant 
737094a2239SPaul Durrant static void xen_device_backend_destroy(XenDevice *xendev)
738094a2239SPaul Durrant {
739094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
740094a2239SPaul Durrant     Error *local_err = NULL;
741094a2239SPaul Durrant 
742b6af8926SPaul Durrant     if (xendev->backend_online_watch) {
743d198b711SPaul Durrant         xen_device_remove_watch(xendev, xendev->backend_online_watch, NULL);
744b6af8926SPaul Durrant         xendev->backend_online_watch = NULL;
745b6af8926SPaul Durrant     }
746b6af8926SPaul Durrant 
747b6af8926SPaul Durrant     if (xendev->backend_state_watch) {
748d198b711SPaul Durrant         xen_device_remove_watch(xendev, xendev->backend_state_watch, NULL);
749b6af8926SPaul Durrant         xendev->backend_state_watch = NULL;
750b6af8926SPaul Durrant     }
751b6af8926SPaul Durrant 
752094a2239SPaul Durrant     if (!xendev->backend_path) {
753094a2239SPaul Durrant         return;
754094a2239SPaul Durrant     }
755094a2239SPaul Durrant 
756094a2239SPaul Durrant     g_assert(xenbus->xsh);
757094a2239SPaul Durrant 
758094a2239SPaul Durrant     xs_node_destroy(xenbus->xsh, XBT_NULL, xendev->backend_path,
759094a2239SPaul Durrant                     &local_err);
760094a2239SPaul Durrant     g_free(xendev->backend_path);
761094a2239SPaul Durrant     xendev->backend_path = NULL;
762094a2239SPaul Durrant 
763094a2239SPaul Durrant     if (local_err) {
764094a2239SPaul Durrant         error_report_err(local_err);
765094a2239SPaul Durrant     }
766094a2239SPaul Durrant }
767094a2239SPaul Durrant 
768b6af8926SPaul Durrant void xen_device_frontend_printf(XenDevice *xendev, const char *key,
769094a2239SPaul Durrant                                 const char *fmt, ...)
770094a2239SPaul Durrant {
771094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
772094a2239SPaul Durrant     Error *local_err = NULL;
773094a2239SPaul Durrant     va_list ap;
774094a2239SPaul Durrant 
775094a2239SPaul Durrant     g_assert(xenbus->xsh);
776094a2239SPaul Durrant 
777094a2239SPaul Durrant     va_start(ap, fmt);
778094a2239SPaul Durrant     xs_node_vprintf(xenbus->xsh, XBT_NULL, xendev->frontend_path, key,
779094a2239SPaul Durrant                     &local_err, fmt, ap);
780094a2239SPaul Durrant     va_end(ap);
781094a2239SPaul Durrant 
782094a2239SPaul Durrant     if (local_err) {
783094a2239SPaul Durrant         error_report_err(local_err);
784094a2239SPaul Durrant     }
785094a2239SPaul Durrant }
786094a2239SPaul Durrant 
787b6af8926SPaul Durrant int xen_device_frontend_scanf(XenDevice *xendev, const char *key,
78882a29e30SPaul Durrant                               const char *fmt, ...)
78982a29e30SPaul Durrant {
79082a29e30SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
79182a29e30SPaul Durrant     va_list ap;
79282a29e30SPaul Durrant     int rc;
79382a29e30SPaul Durrant 
79482a29e30SPaul Durrant     g_assert(xenbus->xsh);
79582a29e30SPaul Durrant 
79682a29e30SPaul Durrant     va_start(ap, fmt);
79782a29e30SPaul Durrant     rc = xs_node_vscanf(xenbus->xsh, XBT_NULL, xendev->frontend_path, key,
79882a29e30SPaul Durrant                         NULL, fmt, ap);
79982a29e30SPaul Durrant     va_end(ap);
80082a29e30SPaul Durrant 
80182a29e30SPaul Durrant     return rc;
80282a29e30SPaul Durrant }
80382a29e30SPaul Durrant 
804094a2239SPaul Durrant static void xen_device_frontend_set_state(XenDevice *xendev,
805705be570SAnthony PERARD                                           enum xenbus_state state,
806705be570SAnthony PERARD                                           bool publish)
807094a2239SPaul Durrant {
808094a2239SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
809094a2239SPaul Durrant 
810094a2239SPaul Durrant     if (xendev->frontend_state == state) {
811094a2239SPaul Durrant         return;
812094a2239SPaul Durrant     }
813094a2239SPaul Durrant 
814094a2239SPaul Durrant     trace_xen_device_frontend_state(type, xendev->name,
815094a2239SPaul Durrant                                     xs_strstate(state));
816094a2239SPaul Durrant 
817094a2239SPaul Durrant     xendev->frontend_state = state;
818705be570SAnthony PERARD     if (publish) {
819094a2239SPaul Durrant         xen_device_frontend_printf(xendev, "state", "%u", state);
820094a2239SPaul Durrant     }
821705be570SAnthony PERARD }
822094a2239SPaul Durrant 
82382a29e30SPaul Durrant static void xen_device_frontend_changed(void *opaque)
82482a29e30SPaul Durrant {
82582a29e30SPaul Durrant     XenDevice *xendev = opaque;
82682a29e30SPaul Durrant     XenDeviceClass *xendev_class = XEN_DEVICE_GET_CLASS(xendev);
82782a29e30SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
82882a29e30SPaul Durrant     enum xenbus_state state;
82982a29e30SPaul Durrant 
83082a29e30SPaul Durrant     trace_xen_device_frontend_changed(type, xendev->name);
83182a29e30SPaul Durrant 
83282a29e30SPaul Durrant     if (xen_device_frontend_scanf(xendev, "state", "%u", &state) != 1) {
83382a29e30SPaul Durrant         state = XenbusStateUnknown;
83482a29e30SPaul Durrant     }
83582a29e30SPaul Durrant 
836705be570SAnthony PERARD     xen_device_frontend_set_state(xendev, state, false);
83782a29e30SPaul Durrant 
83867bc8e00SPaul Durrant     if (state == XenbusStateInitialising &&
83967bc8e00SPaul Durrant         xendev->backend_state == XenbusStateClosed &&
84067bc8e00SPaul Durrant         xendev->backend_online) {
84167bc8e00SPaul Durrant         /*
84267bc8e00SPaul Durrant          * The frontend is re-initializing so switch back to
84367bc8e00SPaul Durrant          * InitWait.
84467bc8e00SPaul Durrant          */
84567bc8e00SPaul Durrant         xen_device_backend_set_state(xendev, XenbusStateInitWait);
84667bc8e00SPaul Durrant         return;
84767bc8e00SPaul Durrant     }
84867bc8e00SPaul Durrant 
84982a29e30SPaul Durrant     if (xendev_class->frontend_changed) {
85082a29e30SPaul Durrant         Error *local_err = NULL;
85182a29e30SPaul Durrant 
85282a29e30SPaul Durrant         xendev_class->frontend_changed(xendev, state, &local_err);
85382a29e30SPaul Durrant 
85482a29e30SPaul Durrant         if (local_err) {
85582a29e30SPaul Durrant             error_reportf_err(local_err, "frontend change error: ");
85682a29e30SPaul Durrant         }
85782a29e30SPaul Durrant     }
85882a29e30SPaul Durrant }
85982a29e30SPaul Durrant 
8606bd6b955SMark Syms static bool xen_device_frontend_exists(XenDevice *xendev)
8616bd6b955SMark Syms {
8626bd6b955SMark Syms     enum xenbus_state state;
8636bd6b955SMark Syms 
8646bd6b955SMark Syms     return (xen_device_frontend_scanf(xendev, "state", "%u", &state) == 1);
8656bd6b955SMark Syms }
8666bd6b955SMark Syms 
867094a2239SPaul Durrant static void xen_device_frontend_create(XenDevice *xendev, Error **errp)
868094a2239SPaul Durrant {
869094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
870094a2239SPaul Durrant     struct xs_permissions perms[2];
871094a2239SPaul Durrant     Error *local_err = NULL;
872094a2239SPaul Durrant 
873094a2239SPaul Durrant     xendev->frontend_path = xen_device_get_frontend_path(xendev);
874094a2239SPaul Durrant 
8756bd6b955SMark Syms     /*
8766bd6b955SMark Syms      * The frontend area may have already been created by a legacy
8776bd6b955SMark Syms      * toolstack.
8786bd6b955SMark Syms      */
8796bd6b955SMark Syms     if (!xen_device_frontend_exists(xendev)) {
880094a2239SPaul Durrant         perms[0].id = xendev->frontend_id;
881094a2239SPaul Durrant         perms[0].perms = XS_PERM_NONE;
882094a2239SPaul Durrant         perms[1].id = xenbus->backend_id;
883094a2239SPaul Durrant         perms[1].perms = XS_PERM_READ | XS_PERM_WRITE;
884094a2239SPaul Durrant 
885094a2239SPaul Durrant         g_assert(xenbus->xsh);
886094a2239SPaul Durrant 
887094a2239SPaul Durrant         xs_node_create(xenbus->xsh, XBT_NULL, xendev->frontend_path, perms,
888094a2239SPaul Durrant                        ARRAY_SIZE(perms), &local_err);
889094a2239SPaul Durrant         if (local_err) {
890094a2239SPaul Durrant             error_propagate_prepend(errp, local_err,
891094a2239SPaul Durrant                                     "failed to create frontend: ");
89282a29e30SPaul Durrant             return;
89382a29e30SPaul Durrant         }
8946bd6b955SMark Syms     }
89582a29e30SPaul Durrant 
89682a29e30SPaul Durrant     xendev->frontend_state_watch =
897d198b711SPaul Durrant         xen_device_add_watch(xendev, xendev->frontend_path, "state",
898d198b711SPaul Durrant                              xen_device_frontend_changed, &local_err);
89982a29e30SPaul Durrant     if (local_err) {
90082a29e30SPaul Durrant         error_propagate_prepend(errp, local_err,
90182a29e30SPaul Durrant                                 "failed to watch frontend state: ");
902094a2239SPaul Durrant     }
903094a2239SPaul Durrant }
904094a2239SPaul Durrant 
905094a2239SPaul Durrant static void xen_device_frontend_destroy(XenDevice *xendev)
906094a2239SPaul Durrant {
907094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
908094a2239SPaul Durrant     Error *local_err = NULL;
909094a2239SPaul Durrant 
91082a29e30SPaul Durrant     if (xendev->frontend_state_watch) {
911d198b711SPaul Durrant         xen_device_remove_watch(xendev, xendev->frontend_state_watch,
912d198b711SPaul Durrant                                 NULL);
91382a29e30SPaul Durrant         xendev->frontend_state_watch = NULL;
91482a29e30SPaul Durrant     }
91582a29e30SPaul Durrant 
916094a2239SPaul Durrant     if (!xendev->frontend_path) {
917094a2239SPaul Durrant         return;
918094a2239SPaul Durrant     }
919094a2239SPaul Durrant 
920094a2239SPaul Durrant     g_assert(xenbus->xsh);
921094a2239SPaul Durrant 
922094a2239SPaul Durrant     xs_node_destroy(xenbus->xsh, XBT_NULL, xendev->frontend_path,
923094a2239SPaul Durrant                     &local_err);
924094a2239SPaul Durrant     g_free(xendev->frontend_path);
925094a2239SPaul Durrant     xendev->frontend_path = NULL;
926094a2239SPaul Durrant 
927094a2239SPaul Durrant     if (local_err) {
928094a2239SPaul Durrant         error_report_err(local_err);
929094a2239SPaul Durrant     }
930094a2239SPaul Durrant }
931094a2239SPaul Durrant 
9324b34b5b1SPaul Durrant void xen_device_set_max_grant_refs(XenDevice *xendev, unsigned int nr_refs,
9334b34b5b1SPaul Durrant                                    Error **errp)
9344b34b5b1SPaul Durrant {
9354b34b5b1SPaul Durrant     if (xengnttab_set_max_grants(xendev->xgth, nr_refs)) {
9364b34b5b1SPaul Durrant         error_setg_errno(errp, errno, "xengnttab_set_max_grants failed");
9374b34b5b1SPaul Durrant     }
9384b34b5b1SPaul Durrant }
9394b34b5b1SPaul Durrant 
9404b34b5b1SPaul Durrant void *xen_device_map_grant_refs(XenDevice *xendev, uint32_t *refs,
9414b34b5b1SPaul Durrant                                 unsigned int nr_refs, int prot,
9424b34b5b1SPaul Durrant                                 Error **errp)
9434b34b5b1SPaul Durrant {
9444b34b5b1SPaul Durrant     void *map = xengnttab_map_domain_grant_refs(xendev->xgth, nr_refs,
9454b34b5b1SPaul Durrant                                                 xendev->frontend_id, refs,
9464b34b5b1SPaul Durrant                                                 prot);
9474b34b5b1SPaul Durrant 
9484b34b5b1SPaul Durrant     if (!map) {
9494b34b5b1SPaul Durrant         error_setg_errno(errp, errno,
9504b34b5b1SPaul Durrant                          "xengnttab_map_domain_grant_refs failed");
9514b34b5b1SPaul Durrant     }
9524b34b5b1SPaul Durrant 
9534b34b5b1SPaul Durrant     return map;
9544b34b5b1SPaul Durrant }
9554b34b5b1SPaul Durrant 
9564b34b5b1SPaul Durrant void xen_device_unmap_grant_refs(XenDevice *xendev, void *map,
9574b34b5b1SPaul Durrant                                  unsigned int nr_refs, Error **errp)
9584b34b5b1SPaul Durrant {
9594b34b5b1SPaul Durrant     if (xengnttab_unmap(xendev->xgth, map, nr_refs)) {
9604b34b5b1SPaul Durrant         error_setg_errno(errp, errno, "xengnttab_unmap failed");
9614b34b5b1SPaul Durrant     }
9624b34b5b1SPaul Durrant }
9634b34b5b1SPaul Durrant 
9644b34b5b1SPaul Durrant static void compat_copy_grant_refs(XenDevice *xendev, bool to_domain,
9654b34b5b1SPaul Durrant                                    XenDeviceGrantCopySegment segs[],
9664b34b5b1SPaul Durrant                                    unsigned int nr_segs, Error **errp)
9674b34b5b1SPaul Durrant {
9684b34b5b1SPaul Durrant     uint32_t *refs = g_new(uint32_t, nr_segs);
9694b34b5b1SPaul Durrant     int prot = to_domain ? PROT_WRITE : PROT_READ;
9704b34b5b1SPaul Durrant     void *map;
9714b34b5b1SPaul Durrant     unsigned int i;
9724b34b5b1SPaul Durrant 
9734b34b5b1SPaul Durrant     for (i = 0; i < nr_segs; i++) {
9744b34b5b1SPaul Durrant         XenDeviceGrantCopySegment *seg = &segs[i];
9754b34b5b1SPaul Durrant 
9764b34b5b1SPaul Durrant         refs[i] = to_domain ? seg->dest.foreign.ref :
9774b34b5b1SPaul Durrant             seg->source.foreign.ref;
9784b34b5b1SPaul Durrant     }
9794b34b5b1SPaul Durrant 
9804b34b5b1SPaul Durrant     map = xengnttab_map_domain_grant_refs(xendev->xgth, nr_segs,
9814b34b5b1SPaul Durrant                                           xendev->frontend_id, refs,
9824b34b5b1SPaul Durrant                                           prot);
9834b34b5b1SPaul Durrant     if (!map) {
9844b34b5b1SPaul Durrant         error_setg_errno(errp, errno,
9854b34b5b1SPaul Durrant                          "xengnttab_map_domain_grant_refs failed");
9864b34b5b1SPaul Durrant         goto done;
9874b34b5b1SPaul Durrant     }
9884b34b5b1SPaul Durrant 
9894b34b5b1SPaul Durrant     for (i = 0; i < nr_segs; i++) {
9904b34b5b1SPaul Durrant         XenDeviceGrantCopySegment *seg = &segs[i];
9914b34b5b1SPaul Durrant         void *page = map + (i * XC_PAGE_SIZE);
9924b34b5b1SPaul Durrant 
9934b34b5b1SPaul Durrant         if (to_domain) {
9944b34b5b1SPaul Durrant             memcpy(page + seg->dest.foreign.offset, seg->source.virt,
9954b34b5b1SPaul Durrant                    seg->len);
9964b34b5b1SPaul Durrant         } else {
9974b34b5b1SPaul Durrant             memcpy(seg->dest.virt, page + seg->source.foreign.offset,
9984b34b5b1SPaul Durrant                    seg->len);
9994b34b5b1SPaul Durrant         }
10004b34b5b1SPaul Durrant     }
10014b34b5b1SPaul Durrant 
10024b34b5b1SPaul Durrant     if (xengnttab_unmap(xendev->xgth, map, nr_segs)) {
10034b34b5b1SPaul Durrant         error_setg_errno(errp, errno, "xengnttab_unmap failed");
10044b34b5b1SPaul Durrant     }
10054b34b5b1SPaul Durrant 
10064b34b5b1SPaul Durrant done:
10074b34b5b1SPaul Durrant     g_free(refs);
10084b34b5b1SPaul Durrant }
10094b34b5b1SPaul Durrant 
10104b34b5b1SPaul Durrant void xen_device_copy_grant_refs(XenDevice *xendev, bool to_domain,
10114b34b5b1SPaul Durrant                                 XenDeviceGrantCopySegment segs[],
10124b34b5b1SPaul Durrant                                 unsigned int nr_segs, Error **errp)
10134b34b5b1SPaul Durrant {
10144b34b5b1SPaul Durrant     xengnttab_grant_copy_segment_t *xengnttab_segs;
10154b34b5b1SPaul Durrant     unsigned int i;
10164b34b5b1SPaul Durrant 
10174b34b5b1SPaul Durrant     if (!xendev->feature_grant_copy) {
10184b34b5b1SPaul Durrant         compat_copy_grant_refs(xendev, to_domain, segs, nr_segs, errp);
10194b34b5b1SPaul Durrant         return;
10204b34b5b1SPaul Durrant     }
10214b34b5b1SPaul Durrant 
10224b34b5b1SPaul Durrant     xengnttab_segs = g_new0(xengnttab_grant_copy_segment_t, nr_segs);
10234b34b5b1SPaul Durrant 
10244b34b5b1SPaul Durrant     for (i = 0; i < nr_segs; i++) {
10254b34b5b1SPaul Durrant         XenDeviceGrantCopySegment *seg = &segs[i];
10264b34b5b1SPaul Durrant         xengnttab_grant_copy_segment_t *xengnttab_seg = &xengnttab_segs[i];
10274b34b5b1SPaul Durrant 
10284b34b5b1SPaul Durrant         if (to_domain) {
10294b34b5b1SPaul Durrant             xengnttab_seg->flags = GNTCOPY_dest_gref;
10304b34b5b1SPaul Durrant             xengnttab_seg->dest.foreign.domid = xendev->frontend_id;
10314b34b5b1SPaul Durrant             xengnttab_seg->dest.foreign.ref = seg->dest.foreign.ref;
10324b34b5b1SPaul Durrant             xengnttab_seg->dest.foreign.offset = seg->dest.foreign.offset;
10334b34b5b1SPaul Durrant             xengnttab_seg->source.virt = seg->source.virt;
10344b34b5b1SPaul Durrant         } else {
10354b34b5b1SPaul Durrant             xengnttab_seg->flags = GNTCOPY_source_gref;
10364b34b5b1SPaul Durrant             xengnttab_seg->source.foreign.domid = xendev->frontend_id;
10374b34b5b1SPaul Durrant             xengnttab_seg->source.foreign.ref = seg->source.foreign.ref;
10384b34b5b1SPaul Durrant             xengnttab_seg->source.foreign.offset =
10394b34b5b1SPaul Durrant                 seg->source.foreign.offset;
10404b34b5b1SPaul Durrant             xengnttab_seg->dest.virt = seg->dest.virt;
10414b34b5b1SPaul Durrant         }
10424b34b5b1SPaul Durrant 
10434b34b5b1SPaul Durrant         xengnttab_seg->len = seg->len;
10444b34b5b1SPaul Durrant     }
10454b34b5b1SPaul Durrant 
10464b34b5b1SPaul Durrant     if (xengnttab_grant_copy(xendev->xgth, nr_segs, xengnttab_segs)) {
10474b34b5b1SPaul Durrant         error_setg_errno(errp, errno, "xengnttab_grant_copy failed");
10484b34b5b1SPaul Durrant         goto done;
10494b34b5b1SPaul Durrant     }
10504b34b5b1SPaul Durrant 
10514b34b5b1SPaul Durrant     for (i = 0; i < nr_segs; i++) {
10524b34b5b1SPaul Durrant         xengnttab_grant_copy_segment_t *xengnttab_seg = &xengnttab_segs[i];
10534b34b5b1SPaul Durrant 
10544b34b5b1SPaul Durrant         if (xengnttab_seg->status != GNTST_okay) {
10554b34b5b1SPaul Durrant             error_setg(errp, "xengnttab_grant_copy seg[%u] failed", i);
10564b34b5b1SPaul Durrant             break;
10574b34b5b1SPaul Durrant         }
10584b34b5b1SPaul Durrant     }
10594b34b5b1SPaul Durrant 
10604b34b5b1SPaul Durrant done:
10614b34b5b1SPaul Durrant     g_free(xengnttab_segs);
10624b34b5b1SPaul Durrant }
10634b34b5b1SPaul Durrant 
1064a3d669c8SPaul Durrant struct XenEventChannel {
1065c0b336eaSPaul Durrant     QLIST_ENTRY(XenEventChannel) list;
106683361a8aSPaul Durrant     AioContext *ctx;
1067c0b336eaSPaul Durrant     xenevtchn_handle *xeh;
1068a3d669c8SPaul Durrant     evtchn_port_t local_port;
1069a3d669c8SPaul Durrant     XenEventHandler handler;
1070a3d669c8SPaul Durrant     void *opaque;
1071a3d669c8SPaul Durrant };
1072a3d669c8SPaul Durrant 
1073345f42b4SPaul Durrant static bool xen_device_poll(void *opaque)
1074345f42b4SPaul Durrant {
1075345f42b4SPaul Durrant     XenEventChannel *channel = opaque;
1076345f42b4SPaul Durrant 
1077345f42b4SPaul Durrant     return channel->handler(channel->opaque);
1078345f42b4SPaul Durrant }
1079345f42b4SPaul Durrant 
1080c0b336eaSPaul Durrant static void xen_device_event(void *opaque)
1081a3d669c8SPaul Durrant {
1082c0b336eaSPaul Durrant     XenEventChannel *channel = opaque;
1083c0b336eaSPaul Durrant     unsigned long port = xenevtchn_pending(channel->xeh);
1084a3d669c8SPaul Durrant 
1085a3d669c8SPaul Durrant     if (port == channel->local_port) {
1086345f42b4SPaul Durrant         xen_device_poll(channel);
1087c0b336eaSPaul Durrant 
1088c0b336eaSPaul Durrant         xenevtchn_unmask(channel->xeh, port);
1089a3d669c8SPaul Durrant     }
1090a3d669c8SPaul Durrant }
1091a3d669c8SPaul Durrant 
109232d0b7beSPaul Durrant void xen_device_set_event_channel_context(XenDevice *xendev,
109332d0b7beSPaul Durrant                                           XenEventChannel *channel,
109483361a8aSPaul Durrant                                           AioContext *ctx,
109532d0b7beSPaul Durrant                                           Error **errp)
109632d0b7beSPaul Durrant {
109732d0b7beSPaul Durrant     if (!channel) {
109832d0b7beSPaul Durrant         error_setg(errp, "bad channel");
109932d0b7beSPaul Durrant         return;
110032d0b7beSPaul Durrant     }
110132d0b7beSPaul Durrant 
110232d0b7beSPaul Durrant     if (channel->ctx)
110332d0b7beSPaul Durrant         aio_set_fd_handler(channel->ctx, xenevtchn_fd(channel->xeh), true,
110432d0b7beSPaul Durrant                            NULL, NULL, NULL, NULL);
110532d0b7beSPaul Durrant 
110632d0b7beSPaul Durrant     channel->ctx = ctx;
110732d0b7beSPaul Durrant     aio_set_fd_handler(channel->ctx, xenevtchn_fd(channel->xeh), true,
110832d0b7beSPaul Durrant                        xen_device_event, NULL, xen_device_poll, channel);
110932d0b7beSPaul Durrant }
111032d0b7beSPaul Durrant 
111132d0b7beSPaul Durrant XenEventChannel *xen_device_bind_event_channel(XenDevice *xendev,
1112a3d669c8SPaul Durrant                                                unsigned int port,
1113a3d669c8SPaul Durrant                                                XenEventHandler handler,
1114a3d669c8SPaul Durrant                                                void *opaque, Error **errp)
1115a3d669c8SPaul Durrant {
1116a3d669c8SPaul Durrant     XenEventChannel *channel = g_new0(XenEventChannel, 1);
1117a3d669c8SPaul Durrant     xenevtchn_port_or_error_t local_port;
1118a3d669c8SPaul Durrant 
1119c0b336eaSPaul Durrant     channel->xeh = xenevtchn_open(NULL, 0);
1120c0b336eaSPaul Durrant     if (!channel->xeh) {
1121c0b336eaSPaul Durrant         error_setg_errno(errp, errno, "failed xenevtchn_open");
1122c0b336eaSPaul Durrant         goto fail;
1123c0b336eaSPaul Durrant     }
1124c0b336eaSPaul Durrant 
1125c0b336eaSPaul Durrant     local_port = xenevtchn_bind_interdomain(channel->xeh,
1126a3d669c8SPaul Durrant                                             xendev->frontend_id,
1127a3d669c8SPaul Durrant                                             port);
1128a3d669c8SPaul Durrant     if (local_port < 0) {
1129a3d669c8SPaul Durrant         error_setg_errno(errp, errno, "xenevtchn_bind_interdomain failed");
1130c0b336eaSPaul Durrant         goto fail;
1131a3d669c8SPaul Durrant     }
1132a3d669c8SPaul Durrant 
1133a3d669c8SPaul Durrant     channel->local_port = local_port;
1134a3d669c8SPaul Durrant     channel->handler = handler;
1135a3d669c8SPaul Durrant     channel->opaque = opaque;
1136a3d669c8SPaul Durrant 
113732d0b7beSPaul Durrant     /* Only reason for failure is a NULL channel */
113832d0b7beSPaul Durrant     xen_device_set_event_channel_context(xendev, channel,
113932d0b7beSPaul Durrant                                          qemu_get_aio_context(),
114032d0b7beSPaul Durrant                                          &error_abort);
1141c0b336eaSPaul Durrant 
1142c0b336eaSPaul Durrant     QLIST_INSERT_HEAD(&xendev->event_channels, channel, list);
1143a3d669c8SPaul Durrant 
1144a3d669c8SPaul Durrant     return channel;
1145c0b336eaSPaul Durrant 
1146c0b336eaSPaul Durrant fail:
1147c0b336eaSPaul Durrant     if (channel->xeh) {
1148c0b336eaSPaul Durrant         xenevtchn_close(channel->xeh);
1149c0b336eaSPaul Durrant     }
1150c0b336eaSPaul Durrant 
1151c0b336eaSPaul Durrant     g_free(channel);
1152c0b336eaSPaul Durrant 
1153c0b336eaSPaul Durrant     return NULL;
1154a3d669c8SPaul Durrant }
1155a3d669c8SPaul Durrant 
1156a3d669c8SPaul Durrant void xen_device_notify_event_channel(XenDevice *xendev,
1157a3d669c8SPaul Durrant                                      XenEventChannel *channel,
1158a3d669c8SPaul Durrant                                      Error **errp)
1159a3d669c8SPaul Durrant {
1160a3d669c8SPaul Durrant     if (!channel) {
1161a3d669c8SPaul Durrant         error_setg(errp, "bad channel");
1162a3d669c8SPaul Durrant         return;
1163a3d669c8SPaul Durrant     }
1164a3d669c8SPaul Durrant 
1165c0b336eaSPaul Durrant     if (xenevtchn_notify(channel->xeh, channel->local_port) < 0) {
1166a3d669c8SPaul Durrant         error_setg_errno(errp, errno, "xenevtchn_notify failed");
1167a3d669c8SPaul Durrant     }
1168a3d669c8SPaul Durrant }
1169a3d669c8SPaul Durrant 
1170a3d669c8SPaul Durrant void xen_device_unbind_event_channel(XenDevice *xendev,
1171a3d669c8SPaul Durrant                                      XenEventChannel *channel,
1172a3d669c8SPaul Durrant                                      Error **errp)
1173a3d669c8SPaul Durrant {
1174a3d669c8SPaul Durrant     if (!channel) {
1175a3d669c8SPaul Durrant         error_setg(errp, "bad channel");
1176a3d669c8SPaul Durrant         return;
1177a3d669c8SPaul Durrant     }
1178a3d669c8SPaul Durrant 
1179c0b336eaSPaul Durrant     QLIST_REMOVE(channel, list);
1180a3d669c8SPaul Durrant 
118183361a8aSPaul Durrant     aio_set_fd_handler(channel->ctx, xenevtchn_fd(channel->xeh), true,
118283361a8aSPaul Durrant                        NULL, NULL, NULL, NULL);
1183c0b336eaSPaul Durrant 
1184c0b336eaSPaul Durrant     if (xenevtchn_unbind(channel->xeh, channel->local_port) < 0) {
1185a3d669c8SPaul Durrant         error_setg_errno(errp, errno, "xenevtchn_unbind failed");
1186a3d669c8SPaul Durrant     }
1187a3d669c8SPaul Durrant 
1188c0b336eaSPaul Durrant     xenevtchn_close(channel->xeh);
1189a3d669c8SPaul Durrant     g_free(channel);
1190a3d669c8SPaul Durrant }
1191a3d669c8SPaul Durrant 
1192b69c3c21SMarkus Armbruster static void xen_device_unrealize(DeviceState *dev)
1193108f7bbaSPaul Durrant {
1194108f7bbaSPaul Durrant     XenDevice *xendev = XEN_DEVICE(dev);
1195108f7bbaSPaul Durrant     XenDeviceClass *xendev_class = XEN_DEVICE_GET_CLASS(xendev);
1196108f7bbaSPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
1197c0b336eaSPaul Durrant     XenEventChannel *channel, *next;
1198108f7bbaSPaul Durrant 
1199094a2239SPaul Durrant     if (!xendev->name) {
1200094a2239SPaul Durrant         return;
1201094a2239SPaul Durrant     }
1202094a2239SPaul Durrant 
1203094a2239SPaul Durrant     trace_xen_device_unrealize(type, xendev->name);
1204094a2239SPaul Durrant 
1205094a2239SPaul Durrant     if (xendev->exit.notify) {
1206094a2239SPaul Durrant         qemu_remove_exit_notifier(&xendev->exit);
1207094a2239SPaul Durrant         xendev->exit.notify = NULL;
1208094a2239SPaul Durrant     }
1209108f7bbaSPaul Durrant 
1210108f7bbaSPaul Durrant     if (xendev_class->unrealize) {
1211b69c3c21SMarkus Armbruster         xendev_class->unrealize(xendev);
1212108f7bbaSPaul Durrant     }
1213094a2239SPaul Durrant 
1214c0b336eaSPaul Durrant     /* Make sure all event channels are cleaned up */
1215c0b336eaSPaul Durrant     QLIST_FOREACH_SAFE(channel, &xendev->event_channels, list, next) {
1216c0b336eaSPaul Durrant         xen_device_unbind_event_channel(xendev, channel, NULL);
1217c0b336eaSPaul Durrant     }
1218c0b336eaSPaul Durrant 
1219094a2239SPaul Durrant     xen_device_frontend_destroy(xendev);
1220094a2239SPaul Durrant     xen_device_backend_destroy(xendev);
1221094a2239SPaul Durrant 
12224b34b5b1SPaul Durrant     if (xendev->xgth) {
12234b34b5b1SPaul Durrant         xengnttab_close(xendev->xgth);
12244b34b5b1SPaul Durrant         xendev->xgth = NULL;
12254b34b5b1SPaul Durrant     }
12264b34b5b1SPaul Durrant 
1227d198b711SPaul Durrant     if (xendev->watch_list) {
1228d198b711SPaul Durrant         watch_list_destroy(xendev->watch_list);
1229d198b711SPaul Durrant         xendev->watch_list = NULL;
1230d198b711SPaul Durrant     }
1231d198b711SPaul Durrant 
1232d198b711SPaul Durrant     if (xendev->xsh) {
1233d198b711SPaul Durrant         xs_close(xendev->xsh);
1234d198b711SPaul Durrant         xendev->xsh = NULL;
1235d198b711SPaul Durrant     }
1236d198b711SPaul Durrant 
1237094a2239SPaul Durrant     g_free(xendev->name);
1238094a2239SPaul Durrant     xendev->name = NULL;
1239094a2239SPaul Durrant }
1240094a2239SPaul Durrant 
1241094a2239SPaul Durrant static void xen_device_exit(Notifier *n, void *data)
1242094a2239SPaul Durrant {
1243094a2239SPaul Durrant     XenDevice *xendev = container_of(n, XenDevice, exit);
1244094a2239SPaul Durrant 
1245b69c3c21SMarkus Armbruster     xen_device_unrealize(DEVICE(xendev));
1246108f7bbaSPaul Durrant }
1247108f7bbaSPaul Durrant 
1248108f7bbaSPaul Durrant static void xen_device_realize(DeviceState *dev, Error **errp)
1249108f7bbaSPaul Durrant {
1250108f7bbaSPaul Durrant     XenDevice *xendev = XEN_DEVICE(dev);
1251108f7bbaSPaul Durrant     XenDeviceClass *xendev_class = XEN_DEVICE_GET_CLASS(xendev);
1252094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
1253108f7bbaSPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
1254108f7bbaSPaul Durrant     Error *local_err = NULL;
1255108f7bbaSPaul Durrant 
1256094a2239SPaul Durrant     if (xendev->frontend_id == DOMID_INVALID) {
1257094a2239SPaul Durrant         xendev->frontend_id = xen_domid;
1258094a2239SPaul Durrant     }
1259094a2239SPaul Durrant 
1260094a2239SPaul Durrant     if (xendev->frontend_id >= DOMID_FIRST_RESERVED) {
1261094a2239SPaul Durrant         error_setg(errp, "invalid frontend-id");
1262094a2239SPaul Durrant         goto unrealize;
1263094a2239SPaul Durrant     }
1264094a2239SPaul Durrant 
1265094a2239SPaul Durrant     if (!xendev_class->get_name) {
1266094a2239SPaul Durrant         error_setg(errp, "get_name method not implemented");
1267094a2239SPaul Durrant         goto unrealize;
1268094a2239SPaul Durrant     }
1269094a2239SPaul Durrant 
1270094a2239SPaul Durrant     xendev->name = xendev_class->get_name(xendev, &local_err);
1271094a2239SPaul Durrant     if (local_err) {
1272094a2239SPaul Durrant         error_propagate_prepend(errp, local_err,
1273094a2239SPaul Durrant                                 "failed to get device name: ");
1274094a2239SPaul Durrant         goto unrealize;
1275094a2239SPaul Durrant     }
1276094a2239SPaul Durrant 
1277094a2239SPaul Durrant     trace_xen_device_realize(type, xendev->name);
1278094a2239SPaul Durrant 
1279d198b711SPaul Durrant     xendev->xsh = xs_open(0);
1280d198b711SPaul Durrant     if (!xendev->xsh) {
1281d198b711SPaul Durrant         error_setg_errno(errp, errno, "failed xs_open");
1282d198b711SPaul Durrant         goto unrealize;
1283d198b711SPaul Durrant     }
1284d198b711SPaul Durrant 
1285d198b711SPaul Durrant     xendev->watch_list = watch_list_create(xendev->xsh);
1286d198b711SPaul Durrant 
12874b34b5b1SPaul Durrant     xendev->xgth = xengnttab_open(NULL, 0);
12884b34b5b1SPaul Durrant     if (!xendev->xgth) {
12894b34b5b1SPaul Durrant         error_setg_errno(errp, errno, "failed xengnttab_open");
12904b34b5b1SPaul Durrant         goto unrealize;
12914b34b5b1SPaul Durrant     }
12924b34b5b1SPaul Durrant 
12934b34b5b1SPaul Durrant     xendev->feature_grant_copy =
12944b34b5b1SPaul Durrant         (xengnttab_grant_copy(xendev->xgth, 0, NULL) == 0);
12954b34b5b1SPaul Durrant 
1296094a2239SPaul Durrant     xen_device_backend_create(xendev, &local_err);
1297094a2239SPaul Durrant     if (local_err) {
1298094a2239SPaul Durrant         error_propagate(errp, local_err);
1299094a2239SPaul Durrant         goto unrealize;
1300094a2239SPaul Durrant     }
1301094a2239SPaul Durrant 
1302094a2239SPaul Durrant     xen_device_frontend_create(xendev, &local_err);
1303094a2239SPaul Durrant     if (local_err) {
1304094a2239SPaul Durrant         error_propagate(errp, local_err);
1305094a2239SPaul Durrant         goto unrealize;
1306094a2239SPaul Durrant     }
1307108f7bbaSPaul Durrant 
1308108f7bbaSPaul Durrant     if (xendev_class->realize) {
1309108f7bbaSPaul Durrant         xendev_class->realize(xendev, &local_err);
1310108f7bbaSPaul Durrant         if (local_err) {
1311108f7bbaSPaul Durrant             error_propagate(errp, local_err);
1312108f7bbaSPaul Durrant             goto unrealize;
1313108f7bbaSPaul Durrant         }
1314108f7bbaSPaul Durrant     }
1315108f7bbaSPaul Durrant 
1316094a2239SPaul Durrant     xen_device_backend_printf(xendev, "frontend", "%s",
1317094a2239SPaul Durrant                               xendev->frontend_path);
1318094a2239SPaul Durrant     xen_device_backend_printf(xendev, "frontend-id", "%u",
1319094a2239SPaul Durrant                               xendev->frontend_id);
1320094a2239SPaul Durrant     xen_device_backend_printf(xendev, "hotplug-status", "connected");
1321094a2239SPaul Durrant 
1322b6af8926SPaul Durrant     xen_device_backend_set_online(xendev, true);
1323094a2239SPaul Durrant     xen_device_backend_set_state(xendev, XenbusStateInitWait);
1324094a2239SPaul Durrant 
13256bd6b955SMark Syms     if (!xen_device_frontend_exists(xendev)) {
1326094a2239SPaul Durrant         xen_device_frontend_printf(xendev, "backend", "%s",
1327094a2239SPaul Durrant                                    xendev->backend_path);
1328094a2239SPaul Durrant         xen_device_frontend_printf(xendev, "backend-id", "%u",
1329094a2239SPaul Durrant                                    xenbus->backend_id);
1330094a2239SPaul Durrant 
1331705be570SAnthony PERARD         xen_device_frontend_set_state(xendev, XenbusStateInitialising, true);
13326bd6b955SMark Syms     }
1333094a2239SPaul Durrant 
1334094a2239SPaul Durrant     xendev->exit.notify = xen_device_exit;
1335094a2239SPaul Durrant     qemu_add_exit_notifier(&xendev->exit);
1336108f7bbaSPaul Durrant     return;
1337108f7bbaSPaul Durrant 
1338108f7bbaSPaul Durrant unrealize:
1339b69c3c21SMarkus Armbruster     xen_device_unrealize(dev);
1340108f7bbaSPaul Durrant }
1341108f7bbaSPaul Durrant 
1342094a2239SPaul Durrant static Property xen_device_props[] = {
1343094a2239SPaul Durrant     DEFINE_PROP_UINT16("frontend-id", XenDevice, frontend_id,
1344094a2239SPaul Durrant                        DOMID_INVALID),
1345094a2239SPaul Durrant     DEFINE_PROP_END_OF_LIST()
1346094a2239SPaul Durrant };
1347094a2239SPaul Durrant 
1348108f7bbaSPaul Durrant static void xen_device_class_init(ObjectClass *class, void *data)
1349108f7bbaSPaul Durrant {
1350108f7bbaSPaul Durrant     DeviceClass *dev_class = DEVICE_CLASS(class);
1351108f7bbaSPaul Durrant 
1352108f7bbaSPaul Durrant     dev_class->realize = xen_device_realize;
1353108f7bbaSPaul Durrant     dev_class->unrealize = xen_device_unrealize;
13544f67d30bSMarc-André Lureau     device_class_set_props(dev_class, xen_device_props);
1355108f7bbaSPaul Durrant     dev_class->bus_type = TYPE_XEN_BUS;
1356108f7bbaSPaul Durrant }
1357108f7bbaSPaul Durrant 
1358108f7bbaSPaul Durrant static const TypeInfo xen_device_type_info = {
1359108f7bbaSPaul Durrant     .name = TYPE_XEN_DEVICE,
1360108f7bbaSPaul Durrant     .parent = TYPE_DEVICE,
1361108f7bbaSPaul Durrant     .instance_size = sizeof(XenDevice),
1362108f7bbaSPaul Durrant     .abstract = true,
1363108f7bbaSPaul Durrant     .class_size = sizeof(XenDeviceClass),
1364108f7bbaSPaul Durrant     .class_init = xen_device_class_init,
1365108f7bbaSPaul Durrant };
1366108f7bbaSPaul Durrant 
1367108f7bbaSPaul Durrant typedef struct XenBridge {
1368108f7bbaSPaul Durrant     SysBusDevice busdev;
1369108f7bbaSPaul Durrant } XenBridge;
1370108f7bbaSPaul Durrant 
1371108f7bbaSPaul Durrant #define TYPE_XEN_BRIDGE "xen-bridge"
1372108f7bbaSPaul Durrant 
1373108f7bbaSPaul Durrant static const TypeInfo xen_bridge_type_info = {
1374108f7bbaSPaul Durrant     .name = TYPE_XEN_BRIDGE,
1375108f7bbaSPaul Durrant     .parent = TYPE_SYS_BUS_DEVICE,
1376108f7bbaSPaul Durrant     .instance_size = sizeof(XenBridge),
1377108f7bbaSPaul Durrant };
1378108f7bbaSPaul Durrant 
1379108f7bbaSPaul Durrant static void xen_register_types(void)
1380108f7bbaSPaul Durrant {
1381108f7bbaSPaul Durrant     type_register_static(&xen_bridge_type_info);
1382108f7bbaSPaul Durrant     type_register_static(&xen_bus_type_info);
1383108f7bbaSPaul Durrant     type_register_static(&xen_device_type_info);
1384108f7bbaSPaul Durrant }
1385108f7bbaSPaul Durrant 
1386108f7bbaSPaul Durrant type_init(xen_register_types)
1387108f7bbaSPaul Durrant 
1388108f7bbaSPaul Durrant void xen_bus_init(void)
1389108f7bbaSPaul Durrant {
13903e80f690SMarkus Armbruster     DeviceState *dev = qdev_new(TYPE_XEN_BRIDGE);
1391108f7bbaSPaul Durrant     BusState *bus = qbus_create(TYPE_XEN_BUS, dev, NULL);
1392108f7bbaSPaul Durrant 
1393*3c6ef471SMarkus Armbruster     sysbus_realize_and_unref(SYS_BUS_DEVICE(dev), &error_fatal);
1394108f7bbaSPaul Durrant     qbus_set_bus_hotplug_handler(bus, &error_abort);
1395108f7bbaSPaul Durrant }
1396