xref: /openbmc/qemu/hw/xen/xen-bus.c (revision b6cacfea0b38300e3ea5fd6d486d5085122554eb)
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 {
561de7096dSVladimir Sementsov-Ogievskiy     ERRP_GUARD();
57b6af8926SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
58b6af8926SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
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",
721de7096dSVladimir Sementsov-Ogievskiy                    errp, "%u", 0);
731de7096dSVladimir Sementsov-Ogievskiy     if (*errp) {
74b6af8926SPaul Durrant         goto abort;
75b6af8926SPaul Durrant     }
76b6af8926SPaul Durrant 
77b6af8926SPaul Durrant     xs_node_printf(xenbus->xsh, tid, xendev->backend_path, "state",
781de7096dSVladimir Sementsov-Ogievskiy                    errp, "%u", XenbusStateClosing);
791de7096dSVladimir Sementsov-Ogievskiy     if (*errp) {
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 }
100b6af8926SPaul Durrant 
101094a2239SPaul Durrant static void xen_bus_print_dev(Monitor *mon, DeviceState *dev, int indent)
102094a2239SPaul Durrant {
103094a2239SPaul Durrant     XenDevice *xendev = XEN_DEVICE(dev);
104094a2239SPaul Durrant 
105094a2239SPaul Durrant     monitor_printf(mon, "%*sname = '%s' frontend_id = %u\n",
106094a2239SPaul Durrant                    indent, "", xendev->name, xendev->frontend_id);
107094a2239SPaul Durrant }
108094a2239SPaul Durrant 
109094a2239SPaul Durrant static char *xen_bus_get_dev_path(DeviceState *dev)
110094a2239SPaul Durrant {
111094a2239SPaul Durrant     return xen_device_get_backend_path(XEN_DEVICE(dev));
112094a2239SPaul Durrant }
113094a2239SPaul Durrant 
11482a29e30SPaul Durrant struct XenWatch {
11582a29e30SPaul Durrant     char *node, *key;
11682a29e30SPaul Durrant     char *token;
11782a29e30SPaul Durrant     XenWatchHandler handler;
11882a29e30SPaul Durrant     void *opaque;
11982a29e30SPaul Durrant     Notifier notifier;
12082a29e30SPaul Durrant };
12182a29e30SPaul Durrant 
12282a29e30SPaul Durrant static void watch_notify(Notifier *n, void *data)
12382a29e30SPaul Durrant {
12482a29e30SPaul Durrant     XenWatch *watch = container_of(n, XenWatch, notifier);
12582a29e30SPaul Durrant     const char *token = data;
12682a29e30SPaul Durrant 
12782a29e30SPaul Durrant     if (!strcmp(watch->token, token)) {
12882a29e30SPaul Durrant         watch->handler(watch->opaque);
12982a29e30SPaul Durrant     }
13082a29e30SPaul Durrant }
13182a29e30SPaul Durrant 
13282a29e30SPaul Durrant static XenWatch *new_watch(const char *node, const char *key,
13382a29e30SPaul Durrant                            XenWatchHandler handler, void *opaque)
13482a29e30SPaul Durrant {
13582a29e30SPaul Durrant     XenWatch *watch = g_new0(XenWatch, 1);
13682a29e30SPaul Durrant     QemuUUID uuid;
13782a29e30SPaul Durrant 
13882a29e30SPaul Durrant     qemu_uuid_generate(&uuid);
13982a29e30SPaul Durrant 
14082a29e30SPaul Durrant     watch->token = qemu_uuid_unparse_strdup(&uuid);
14182a29e30SPaul Durrant     watch->node = g_strdup(node);
14282a29e30SPaul Durrant     watch->key = g_strdup(key);
14382a29e30SPaul Durrant     watch->handler = handler;
14482a29e30SPaul Durrant     watch->opaque = opaque;
14582a29e30SPaul Durrant     watch->notifier.notify = watch_notify;
14682a29e30SPaul Durrant 
14782a29e30SPaul Durrant     return watch;
14882a29e30SPaul Durrant }
14982a29e30SPaul Durrant 
15082a29e30SPaul Durrant static void free_watch(XenWatch *watch)
15182a29e30SPaul Durrant {
15282a29e30SPaul Durrant     g_free(watch->token);
15382a29e30SPaul Durrant     g_free(watch->key);
15482a29e30SPaul Durrant     g_free(watch->node);
15582a29e30SPaul Durrant 
15682a29e30SPaul Durrant     g_free(watch);
15782a29e30SPaul Durrant }
15882a29e30SPaul Durrant 
159374752a2SPaul Durrant struct XenWatchList {
160374752a2SPaul Durrant     struct xs_handle *xsh;
161374752a2SPaul Durrant     NotifierList notifiers;
162374752a2SPaul Durrant };
163374752a2SPaul Durrant 
164374752a2SPaul Durrant static void watch_list_event(void *opaque)
165374752a2SPaul Durrant {
166374752a2SPaul Durrant     XenWatchList *watch_list = opaque;
167374752a2SPaul Durrant     char **v;
168374752a2SPaul Durrant     const char *token;
169374752a2SPaul Durrant 
170374752a2SPaul Durrant     v = xs_check_watch(watch_list->xsh);
171374752a2SPaul Durrant     if (!v) {
172374752a2SPaul Durrant         return;
173374752a2SPaul Durrant     }
174374752a2SPaul Durrant 
175374752a2SPaul Durrant     token = v[XS_WATCH_TOKEN];
176374752a2SPaul Durrant 
177374752a2SPaul Durrant     notifier_list_notify(&watch_list->notifiers, (void *)token);
178374752a2SPaul Durrant 
179374752a2SPaul Durrant     free(v);
180374752a2SPaul Durrant }
181374752a2SPaul Durrant 
182374752a2SPaul Durrant static XenWatchList *watch_list_create(struct xs_handle *xsh)
183374752a2SPaul Durrant {
184374752a2SPaul Durrant     XenWatchList *watch_list = g_new0(XenWatchList, 1);
185374752a2SPaul Durrant 
186374752a2SPaul Durrant     g_assert(xsh);
187374752a2SPaul Durrant 
188374752a2SPaul Durrant     watch_list->xsh = xsh;
189374752a2SPaul Durrant     notifier_list_init(&watch_list->notifiers);
190374752a2SPaul Durrant     qemu_set_fd_handler(xs_fileno(watch_list->xsh), watch_list_event, NULL,
191374752a2SPaul Durrant                         watch_list);
192374752a2SPaul Durrant 
193374752a2SPaul Durrant     return watch_list;
194374752a2SPaul Durrant }
195374752a2SPaul Durrant 
196374752a2SPaul Durrant static void watch_list_destroy(XenWatchList *watch_list)
197374752a2SPaul Durrant {
198374752a2SPaul Durrant     g_assert(notifier_list_empty(&watch_list->notifiers));
199374752a2SPaul Durrant     qemu_set_fd_handler(xs_fileno(watch_list->xsh), NULL, NULL, NULL);
200374752a2SPaul Durrant     g_free(watch_list);
201374752a2SPaul Durrant }
202374752a2SPaul Durrant 
203374752a2SPaul Durrant static XenWatch *watch_list_add(XenWatchList *watch_list, const char *node,
20482a29e30SPaul Durrant                                 const char *key, XenWatchHandler handler,
20582a29e30SPaul Durrant                                 void *opaque, Error **errp)
20682a29e30SPaul Durrant {
2071de7096dSVladimir Sementsov-Ogievskiy     ERRP_GUARD();
20882a29e30SPaul Durrant     XenWatch *watch = new_watch(node, key, handler, opaque);
20982a29e30SPaul Durrant 
210374752a2SPaul Durrant     notifier_list_add(&watch_list->notifiers, &watch->notifier);
21182a29e30SPaul Durrant 
2121de7096dSVladimir Sementsov-Ogievskiy     xs_node_watch(watch_list->xsh, node, key, watch->token, errp);
2131de7096dSVladimir Sementsov-Ogievskiy     if (*errp) {
21482a29e30SPaul Durrant         notifier_remove(&watch->notifier);
21582a29e30SPaul Durrant         free_watch(watch);
21682a29e30SPaul Durrant 
21782a29e30SPaul Durrant         return NULL;
21882a29e30SPaul Durrant     }
21982a29e30SPaul Durrant 
22082a29e30SPaul Durrant     return watch;
22182a29e30SPaul Durrant }
22282a29e30SPaul Durrant 
223374752a2SPaul Durrant static void watch_list_remove(XenWatchList *watch_list, XenWatch *watch,
22482a29e30SPaul Durrant                               Error **errp)
22582a29e30SPaul Durrant {
226374752a2SPaul Durrant     xs_node_unwatch(watch_list->xsh, watch->node, watch->key, watch->token,
22782a29e30SPaul Durrant                     errp);
22882a29e30SPaul Durrant 
22982a29e30SPaul Durrant     notifier_remove(&watch->notifier);
23082a29e30SPaul Durrant     free_watch(watch);
23182a29e30SPaul Durrant }
23282a29e30SPaul Durrant 
233374752a2SPaul Durrant static XenWatch *xen_bus_add_watch(XenBus *xenbus, const char *node,
234374752a2SPaul Durrant                                    const char *key, XenWatchHandler handler,
235d198b711SPaul Durrant                                    Error **errp)
236374752a2SPaul Durrant {
237374752a2SPaul Durrant     trace_xen_bus_add_watch(node, key);
238374752a2SPaul Durrant 
239d198b711SPaul Durrant     return watch_list_add(xenbus->watch_list, node, key, handler, xenbus,
240374752a2SPaul Durrant                           errp);
241374752a2SPaul Durrant }
242374752a2SPaul Durrant 
243374752a2SPaul Durrant static void xen_bus_remove_watch(XenBus *xenbus, XenWatch *watch,
244374752a2SPaul Durrant                                  Error **errp)
245374752a2SPaul Durrant {
246374752a2SPaul Durrant     trace_xen_bus_remove_watch(watch->node, watch->key);
247374752a2SPaul Durrant 
248374752a2SPaul Durrant     watch_list_remove(xenbus->watch_list, watch, errp);
249374752a2SPaul Durrant }
250374752a2SPaul Durrant 
251a783f8adSPaul Durrant static void xen_bus_backend_create(XenBus *xenbus, const char *type,
252a783f8adSPaul Durrant                                    const char *name, char *path,
253a783f8adSPaul Durrant                                    Error **errp)
254a783f8adSPaul Durrant {
2551de7096dSVladimir Sementsov-Ogievskiy     ERRP_GUARD();
256a783f8adSPaul Durrant     xs_transaction_t tid;
257a783f8adSPaul Durrant     char **key;
258a783f8adSPaul Durrant     QDict *opts;
259a783f8adSPaul Durrant     unsigned int i, n;
260a783f8adSPaul Durrant 
261a783f8adSPaul Durrant     trace_xen_bus_backend_create(type, path);
262a783f8adSPaul Durrant 
263a783f8adSPaul Durrant again:
264a783f8adSPaul Durrant     tid = xs_transaction_start(xenbus->xsh);
265a783f8adSPaul Durrant     if (tid == XBT_NULL) {
266a783f8adSPaul Durrant         error_setg(errp, "failed xs_transaction_start");
267a783f8adSPaul Durrant         return;
268a783f8adSPaul Durrant     }
269a783f8adSPaul Durrant 
270a783f8adSPaul Durrant     key = xs_directory(xenbus->xsh, tid, path, &n);
271a783f8adSPaul Durrant     if (!key) {
272a783f8adSPaul Durrant         if (!xs_transaction_end(xenbus->xsh, tid, true)) {
273a783f8adSPaul Durrant             error_setg_errno(errp, errno, "failed xs_transaction_end");
274a783f8adSPaul Durrant         }
275a783f8adSPaul Durrant         return;
276a783f8adSPaul Durrant     }
277a783f8adSPaul Durrant 
278a783f8adSPaul Durrant     opts = qdict_new();
279a783f8adSPaul Durrant     for (i = 0; i < n; i++) {
280a783f8adSPaul Durrant         char *val;
281a783f8adSPaul Durrant 
282a783f8adSPaul Durrant         /*
283a783f8adSPaul Durrant          * Assume anything found in the xenstore backend area, other than
284a783f8adSPaul Durrant          * the keys created for a generic XenDevice, are parameters
285a783f8adSPaul Durrant          * to be used to configure the backend.
286a783f8adSPaul Durrant          */
287a783f8adSPaul Durrant         if (!strcmp(key[i], "state") ||
288a783f8adSPaul Durrant             !strcmp(key[i], "online") ||
289a783f8adSPaul Durrant             !strcmp(key[i], "frontend") ||
290a783f8adSPaul Durrant             !strcmp(key[i], "frontend-id") ||
291a783f8adSPaul Durrant             !strcmp(key[i], "hotplug-status"))
292a783f8adSPaul Durrant             continue;
293a783f8adSPaul Durrant 
294a783f8adSPaul Durrant         if (xs_node_scanf(xenbus->xsh, tid, path, key[i], NULL, "%ms",
295a783f8adSPaul Durrant                           &val) == 1) {
296a783f8adSPaul Durrant             qdict_put_str(opts, key[i], val);
297a783f8adSPaul Durrant             free(val);
298a783f8adSPaul Durrant         }
299a783f8adSPaul Durrant     }
300a783f8adSPaul Durrant 
301a783f8adSPaul Durrant     free(key);
302a783f8adSPaul Durrant 
303a783f8adSPaul Durrant     if (!xs_transaction_end(xenbus->xsh, tid, false)) {
304a783f8adSPaul Durrant         qobject_unref(opts);
305a783f8adSPaul Durrant 
306a783f8adSPaul Durrant         if (errno == EAGAIN) {
307a783f8adSPaul Durrant             goto again;
308a783f8adSPaul Durrant         }
309a783f8adSPaul Durrant 
310a783f8adSPaul Durrant         error_setg_errno(errp, errno, "failed xs_transaction_end");
311a783f8adSPaul Durrant         return;
312a783f8adSPaul Durrant     }
313a783f8adSPaul Durrant 
3141de7096dSVladimir Sementsov-Ogievskiy     xen_backend_device_create(xenbus, type, name, opts, errp);
315a783f8adSPaul Durrant     qobject_unref(opts);
316a783f8adSPaul Durrant 
3171de7096dSVladimir Sementsov-Ogievskiy     if (*errp) {
3181de7096dSVladimir Sementsov-Ogievskiy         error_prepend(errp, "failed to create '%s' device '%s': ", type, name);
319a783f8adSPaul Durrant     }
320a783f8adSPaul Durrant }
321a783f8adSPaul Durrant 
322a783f8adSPaul Durrant static void xen_bus_type_enumerate(XenBus *xenbus, const char *type)
323a783f8adSPaul Durrant {
324a783f8adSPaul Durrant     char *domain_path = g_strdup_printf("backend/%s/%u", type, xen_domid);
325a783f8adSPaul Durrant     char **backend;
326a783f8adSPaul Durrant     unsigned int i, n;
327a783f8adSPaul Durrant 
328a783f8adSPaul Durrant     trace_xen_bus_type_enumerate(type);
329a783f8adSPaul Durrant 
330a783f8adSPaul Durrant     backend = xs_directory(xenbus->xsh, XBT_NULL, domain_path, &n);
331a783f8adSPaul Durrant     if (!backend) {
332a783f8adSPaul Durrant         goto out;
333a783f8adSPaul Durrant     }
334a783f8adSPaul Durrant 
335a783f8adSPaul Durrant     for (i = 0; i < n; i++) {
336a783f8adSPaul Durrant         char *backend_path = g_strdup_printf("%s/%s", domain_path,
337a783f8adSPaul Durrant                                              backend[i]);
3383809f758SPaul Durrant         enum xenbus_state state;
3393809f758SPaul Durrant         unsigned int online;
340a783f8adSPaul Durrant 
341a783f8adSPaul Durrant         if (xs_node_scanf(xenbus->xsh, XBT_NULL, backend_path, "state",
3423809f758SPaul Durrant                           NULL, "%u", &state) != 1)
3433809f758SPaul Durrant             state = XenbusStateUnknown;
344a783f8adSPaul Durrant 
3453809f758SPaul Durrant         if (xs_node_scanf(xenbus->xsh, XBT_NULL, backend_path, "online",
3463809f758SPaul Durrant                           NULL, "%u", &online) != 1)
3473809f758SPaul Durrant             online = 0;
3483809f758SPaul Durrant 
3493809f758SPaul Durrant         if (online && state == XenbusStateInitialising) {
350a783f8adSPaul Durrant             Error *local_err = NULL;
351a783f8adSPaul Durrant 
352a783f8adSPaul Durrant             xen_bus_backend_create(xenbus, type, backend[i], backend_path,
353a783f8adSPaul Durrant                                    &local_err);
354a783f8adSPaul Durrant             if (local_err) {
355a783f8adSPaul Durrant                 error_report_err(local_err);
356a783f8adSPaul Durrant             }
357a783f8adSPaul Durrant         }
358a783f8adSPaul Durrant 
359a783f8adSPaul Durrant         g_free(backend_path);
360a783f8adSPaul Durrant     }
361a783f8adSPaul Durrant 
362a783f8adSPaul Durrant     free(backend);
363a783f8adSPaul Durrant 
364a783f8adSPaul Durrant out:
365a783f8adSPaul Durrant     g_free(domain_path);
366a783f8adSPaul Durrant }
367a783f8adSPaul Durrant 
3683809f758SPaul Durrant static void xen_bus_enumerate(XenBus *xenbus)
369a783f8adSPaul Durrant {
370a783f8adSPaul Durrant     char **type;
371a783f8adSPaul Durrant     unsigned int i, n;
372a783f8adSPaul Durrant 
373a783f8adSPaul Durrant     trace_xen_bus_enumerate();
374a783f8adSPaul Durrant 
375a783f8adSPaul Durrant     type = xs_directory(xenbus->xsh, XBT_NULL, "backend", &n);
376a783f8adSPaul Durrant     if (!type) {
377a783f8adSPaul Durrant         return;
378a783f8adSPaul Durrant     }
379a783f8adSPaul Durrant 
380a783f8adSPaul Durrant     for (i = 0; i < n; i++) {
381a783f8adSPaul Durrant         xen_bus_type_enumerate(xenbus, type[i]);
382a783f8adSPaul Durrant     }
383a783f8adSPaul Durrant 
384a783f8adSPaul Durrant     free(type);
385a783f8adSPaul Durrant }
386a783f8adSPaul Durrant 
3873809f758SPaul Durrant static void xen_bus_device_cleanup(XenDevice *xendev)
3883809f758SPaul Durrant {
3893809f758SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
3903809f758SPaul Durrant     Error *local_err = NULL;
3913809f758SPaul Durrant 
3923809f758SPaul Durrant     trace_xen_bus_device_cleanup(type, xendev->name);
3933809f758SPaul Durrant 
3943809f758SPaul Durrant     g_assert(!xendev->backend_online);
3953809f758SPaul Durrant 
3963809f758SPaul Durrant     if (!xen_backend_try_device_destroy(xendev, &local_err)) {
3973809f758SPaul Durrant         object_unparent(OBJECT(xendev));
3983809f758SPaul Durrant     }
3993809f758SPaul Durrant 
4003809f758SPaul Durrant     if (local_err) {
4013809f758SPaul Durrant         error_report_err(local_err);
4023809f758SPaul Durrant     }
4033809f758SPaul Durrant }
4043809f758SPaul Durrant 
4053809f758SPaul Durrant static void xen_bus_cleanup(XenBus *xenbus)
4063809f758SPaul Durrant {
4073809f758SPaul Durrant     XenDevice *xendev, *next;
4083809f758SPaul Durrant 
4093809f758SPaul Durrant     trace_xen_bus_cleanup();
4103809f758SPaul Durrant 
4113809f758SPaul Durrant     QLIST_FOREACH_SAFE(xendev, &xenbus->inactive_devices, list, next) {
4123809f758SPaul Durrant         g_assert(xendev->inactive);
4133809f758SPaul Durrant         QLIST_REMOVE(xendev, list);
4143809f758SPaul Durrant         xen_bus_device_cleanup(xendev);
4153809f758SPaul Durrant     }
4163809f758SPaul Durrant }
4173809f758SPaul Durrant 
4183809f758SPaul Durrant static void xen_bus_backend_changed(void *opaque)
4193809f758SPaul Durrant {
4203809f758SPaul Durrant     XenBus *xenbus = opaque;
4213809f758SPaul Durrant 
4223809f758SPaul Durrant     xen_bus_enumerate(xenbus);
4233809f758SPaul Durrant     xen_bus_cleanup(xenbus);
4243809f758SPaul Durrant }
4253809f758SPaul Durrant 
426b69c3c21SMarkus Armbruster static void xen_bus_unrealize(BusState *bus)
427108f7bbaSPaul Durrant {
428094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(bus);
429094a2239SPaul Durrant 
430108f7bbaSPaul Durrant     trace_xen_bus_unrealize();
431094a2239SPaul Durrant 
432a783f8adSPaul Durrant     if (xenbus->backend_watch) {
433c4583c8cSPaul Durrant         unsigned int i;
434c4583c8cSPaul Durrant 
435c4583c8cSPaul Durrant         for (i = 0; i < xenbus->backend_types; i++) {
436c4583c8cSPaul Durrant             if (xenbus->backend_watch[i]) {
437c4583c8cSPaul Durrant                 xen_bus_remove_watch(xenbus, xenbus->backend_watch[i], NULL);
438c4583c8cSPaul Durrant             }
439c4583c8cSPaul Durrant         }
440c4583c8cSPaul Durrant 
441c4583c8cSPaul Durrant         g_free(xenbus->backend_watch);
442a783f8adSPaul Durrant         xenbus->backend_watch = NULL;
443a783f8adSPaul Durrant     }
444a783f8adSPaul Durrant 
445374752a2SPaul Durrant     if (xenbus->watch_list) {
446374752a2SPaul Durrant         watch_list_destroy(xenbus->watch_list);
447374752a2SPaul Durrant         xenbus->watch_list = NULL;
448094a2239SPaul Durrant     }
449094a2239SPaul Durrant 
450374752a2SPaul Durrant     if (xenbus->xsh) {
451094a2239SPaul Durrant         xs_close(xenbus->xsh);
452108f7bbaSPaul Durrant     }
45382a29e30SPaul Durrant }
45482a29e30SPaul Durrant 
455108f7bbaSPaul Durrant static void xen_bus_realize(BusState *bus, Error **errp)
456108f7bbaSPaul Durrant {
457c4583c8cSPaul Durrant     char *key = g_strdup_printf("%u", xen_domid);
458094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(bus);
459094a2239SPaul Durrant     unsigned int domid;
460c4583c8cSPaul Durrant     const char **type;
461c4583c8cSPaul Durrant     unsigned int i;
462a783f8adSPaul Durrant     Error *local_err = NULL;
463094a2239SPaul Durrant 
464108f7bbaSPaul Durrant     trace_xen_bus_realize();
465094a2239SPaul Durrant 
466094a2239SPaul Durrant     xenbus->xsh = xs_open(0);
467094a2239SPaul Durrant     if (!xenbus->xsh) {
468094a2239SPaul Durrant         error_setg_errno(errp, errno, "failed xs_open");
469094a2239SPaul Durrant         goto fail;
470094a2239SPaul Durrant     }
471094a2239SPaul Durrant 
472094a2239SPaul Durrant     if (xs_node_scanf(xenbus->xsh, XBT_NULL, "", /* domain root node */
473094a2239SPaul Durrant                       "domid", NULL, "%u", &domid) == 1) {
474094a2239SPaul Durrant         xenbus->backend_id = domid;
475094a2239SPaul Durrant     } else {
476094a2239SPaul Durrant         xenbus->backend_id = 0; /* Assume lack of node means dom0 */
477094a2239SPaul Durrant     }
478094a2239SPaul Durrant 
479374752a2SPaul Durrant     xenbus->watch_list = watch_list_create(xenbus->xsh);
480a783f8adSPaul Durrant 
481a783f8adSPaul Durrant     module_call_init(MODULE_INIT_XEN_BACKEND);
482a783f8adSPaul Durrant 
483c4583c8cSPaul Durrant     type = xen_backend_get_types(&xenbus->backend_types);
484c4583c8cSPaul Durrant     xenbus->backend_watch = g_new(XenWatch *, xenbus->backend_types);
485c4583c8cSPaul Durrant 
486c4583c8cSPaul Durrant     for (i = 0; i < xenbus->backend_types; i++) {
487c4583c8cSPaul Durrant         char *node = g_strdup_printf("backend/%s", type[i]);
488c4583c8cSPaul Durrant 
489c4583c8cSPaul Durrant         xenbus->backend_watch[i] =
490c4583c8cSPaul Durrant             xen_bus_add_watch(xenbus, node, key, xen_bus_backend_changed,
491c4583c8cSPaul Durrant                               &local_err);
492a783f8adSPaul Durrant         if (local_err) {
493a783f8adSPaul Durrant             /* This need not be treated as a hard error so don't propagate */
494a783f8adSPaul Durrant             error_reportf_err(local_err,
495c4583c8cSPaul Durrant                               "failed to set up '%s' enumeration watch: ",
496c4583c8cSPaul Durrant                               type[i]);
497a783f8adSPaul Durrant         }
498a783f8adSPaul Durrant 
499c4583c8cSPaul Durrant         g_free(node);
500c4583c8cSPaul Durrant     }
501c4583c8cSPaul Durrant 
502c4583c8cSPaul Durrant     g_free(type);
503c4583c8cSPaul Durrant     g_free(key);
504094a2239SPaul Durrant     return;
505094a2239SPaul Durrant 
506094a2239SPaul Durrant fail:
507b69c3c21SMarkus Armbruster     xen_bus_unrealize(bus);
508c4583c8cSPaul Durrant     g_free(key);
509108f7bbaSPaul Durrant }
510108f7bbaSPaul Durrant 
511b6af8926SPaul Durrant static void xen_bus_unplug_request(HotplugHandler *hotplug,
512b6af8926SPaul Durrant                                    DeviceState *dev,
513b6af8926SPaul Durrant                                    Error **errp)
514b6af8926SPaul Durrant {
515b6af8926SPaul Durrant     XenDevice *xendev = XEN_DEVICE(dev);
516b6af8926SPaul Durrant 
517b6af8926SPaul Durrant     xen_device_unplug(xendev, errp);
518b6af8926SPaul Durrant }
519b6af8926SPaul Durrant 
520108f7bbaSPaul Durrant static void xen_bus_class_init(ObjectClass *class, void *data)
521108f7bbaSPaul Durrant {
522108f7bbaSPaul Durrant     BusClass *bus_class = BUS_CLASS(class);
523b6af8926SPaul Durrant     HotplugHandlerClass *hotplug_class = HOTPLUG_HANDLER_CLASS(class);
524108f7bbaSPaul Durrant 
525094a2239SPaul Durrant     bus_class->print_dev = xen_bus_print_dev;
526094a2239SPaul Durrant     bus_class->get_dev_path = xen_bus_get_dev_path;
527108f7bbaSPaul Durrant     bus_class->realize = xen_bus_realize;
528108f7bbaSPaul Durrant     bus_class->unrealize = xen_bus_unrealize;
529b6af8926SPaul Durrant 
530b6af8926SPaul Durrant     hotplug_class->unplug_request = xen_bus_unplug_request;
531108f7bbaSPaul Durrant }
532108f7bbaSPaul Durrant 
533108f7bbaSPaul Durrant static const TypeInfo xen_bus_type_info = {
534108f7bbaSPaul Durrant     .name = TYPE_XEN_BUS,
535108f7bbaSPaul Durrant     .parent = TYPE_BUS,
536108f7bbaSPaul Durrant     .instance_size = sizeof(XenBus),
537108f7bbaSPaul Durrant     .class_size = sizeof(XenBusClass),
538108f7bbaSPaul Durrant     .class_init = xen_bus_class_init,
539108f7bbaSPaul Durrant     .interfaces = (InterfaceInfo[]) {
540108f7bbaSPaul Durrant         { TYPE_HOTPLUG_HANDLER },
541108f7bbaSPaul Durrant         { }
542108f7bbaSPaul Durrant     },
543108f7bbaSPaul Durrant };
544108f7bbaSPaul Durrant 
545b6af8926SPaul Durrant void xen_device_backend_printf(XenDevice *xendev, const char *key,
546094a2239SPaul Durrant                                const char *fmt, ...)
547094a2239SPaul Durrant {
548094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
549094a2239SPaul Durrant     Error *local_err = NULL;
550094a2239SPaul Durrant     va_list ap;
551094a2239SPaul Durrant 
552094a2239SPaul Durrant     g_assert(xenbus->xsh);
553094a2239SPaul Durrant 
554094a2239SPaul Durrant     va_start(ap, fmt);
555094a2239SPaul Durrant     xs_node_vprintf(xenbus->xsh, XBT_NULL, xendev->backend_path, key,
556094a2239SPaul Durrant                     &local_err, fmt, ap);
557094a2239SPaul Durrant     va_end(ap);
558094a2239SPaul Durrant 
559094a2239SPaul Durrant     if (local_err) {
560094a2239SPaul Durrant         error_report_err(local_err);
561094a2239SPaul Durrant     }
562094a2239SPaul Durrant }
563094a2239SPaul Durrant 
564d62449daSDaniel P. Berrangé G_GNUC_SCANF(3, 4)
56582a29e30SPaul Durrant static int xen_device_backend_scanf(XenDevice *xendev, const char *key,
56682a29e30SPaul Durrant                                     const char *fmt, ...)
56782a29e30SPaul Durrant {
56882a29e30SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
56982a29e30SPaul Durrant     va_list ap;
57082a29e30SPaul Durrant     int rc;
57182a29e30SPaul Durrant 
57282a29e30SPaul Durrant     g_assert(xenbus->xsh);
57382a29e30SPaul Durrant 
57482a29e30SPaul Durrant     va_start(ap, fmt);
57582a29e30SPaul Durrant     rc = xs_node_vscanf(xenbus->xsh, XBT_NULL, xendev->backend_path, key,
57682a29e30SPaul Durrant                         NULL, fmt, ap);
57782a29e30SPaul Durrant     va_end(ap);
57882a29e30SPaul Durrant 
57982a29e30SPaul Durrant     return rc;
58082a29e30SPaul Durrant }
58182a29e30SPaul Durrant 
58282a29e30SPaul Durrant void xen_device_backend_set_state(XenDevice *xendev,
583094a2239SPaul Durrant                                   enum xenbus_state state)
584094a2239SPaul Durrant {
585094a2239SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
586094a2239SPaul Durrant 
587094a2239SPaul Durrant     if (xendev->backend_state == state) {
588094a2239SPaul Durrant         return;
589094a2239SPaul Durrant     }
590094a2239SPaul Durrant 
591094a2239SPaul Durrant     trace_xen_device_backend_state(type, xendev->name,
592094a2239SPaul Durrant                                    xs_strstate(state));
593094a2239SPaul Durrant 
594094a2239SPaul Durrant     xendev->backend_state = state;
595094a2239SPaul Durrant     xen_device_backend_printf(xendev, "state", "%u", state);
596094a2239SPaul Durrant }
597094a2239SPaul Durrant 
59882a29e30SPaul Durrant enum xenbus_state xen_device_backend_get_state(XenDevice *xendev)
59982a29e30SPaul Durrant {
60082a29e30SPaul Durrant     return xendev->backend_state;
60182a29e30SPaul Durrant }
60282a29e30SPaul Durrant 
603b6af8926SPaul Durrant static void xen_device_backend_set_online(XenDevice *xendev, bool online)
604b6af8926SPaul Durrant {
605b6af8926SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
606b6af8926SPaul Durrant 
607b6af8926SPaul Durrant     if (xendev->backend_online == online) {
608b6af8926SPaul Durrant         return;
609b6af8926SPaul Durrant     }
610b6af8926SPaul Durrant 
611b6af8926SPaul Durrant     trace_xen_device_backend_online(type, xendev->name, online);
612b6af8926SPaul Durrant 
613b6af8926SPaul Durrant     xendev->backend_online = online;
614b6af8926SPaul Durrant     xen_device_backend_printf(xendev, "online", "%u", online);
615b6af8926SPaul Durrant }
616b6af8926SPaul Durrant 
617cb323146SAnthony PERARD /*
618cb323146SAnthony PERARD  * Tell from the state whether the frontend is likely alive,
619cb323146SAnthony PERARD  * i.e. it will react to a change of state of the backend.
620cb323146SAnthony PERARD  */
6213809f758SPaul Durrant static bool xen_device_frontend_is_active(XenDevice *xendev)
622cb323146SAnthony PERARD {
6233809f758SPaul Durrant     switch (xendev->frontend_state) {
624cb323146SAnthony PERARD     case XenbusStateInitWait:
625cb323146SAnthony PERARD     case XenbusStateInitialised:
626cb323146SAnthony PERARD     case XenbusStateConnected:
627cb323146SAnthony PERARD     case XenbusStateClosing:
628cb323146SAnthony PERARD         return true;
629cb323146SAnthony PERARD     default:
630cb323146SAnthony PERARD         return false;
631cb323146SAnthony PERARD     }
632cb323146SAnthony PERARD }
633cb323146SAnthony PERARD 
634b6af8926SPaul Durrant static void xen_device_backend_changed(void *opaque)
635b6af8926SPaul Durrant {
636b6af8926SPaul Durrant     XenDevice *xendev = opaque;
637b6af8926SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
638b6af8926SPaul Durrant     enum xenbus_state state;
639b6af8926SPaul Durrant     unsigned int online;
640b6af8926SPaul Durrant 
641b6af8926SPaul Durrant     trace_xen_device_backend_changed(type, xendev->name);
642b6af8926SPaul Durrant 
643b6af8926SPaul Durrant     if (xen_device_backend_scanf(xendev, "state", "%u", &state) != 1) {
644b6af8926SPaul Durrant         state = XenbusStateUnknown;
645b6af8926SPaul Durrant     }
646b6af8926SPaul Durrant 
647b6af8926SPaul Durrant     xen_device_backend_set_state(xendev, state);
648b6af8926SPaul Durrant 
649b6af8926SPaul Durrant     if (xen_device_backend_scanf(xendev, "online", "%u", &online) != 1) {
650b6af8926SPaul Durrant         online = 0;
651b6af8926SPaul Durrant     }
652b6af8926SPaul Durrant 
653b6af8926SPaul Durrant     xen_device_backend_set_online(xendev, !!online);
654b6af8926SPaul Durrant 
655b6af8926SPaul Durrant     /*
656b6af8926SPaul Durrant      * If the toolstack (or unplug request callback) has set the backend
657cb323146SAnthony PERARD      * state to Closing, but there is no active frontend then set the
658cb323146SAnthony PERARD      * backend state to Closed.
659b6af8926SPaul Durrant      */
6603809f758SPaul Durrant     if (state == XenbusStateClosing &&
6613809f758SPaul Durrant         !xen_device_frontend_is_active(xendev)) {
662b6af8926SPaul Durrant         xen_device_backend_set_state(xendev, XenbusStateClosed);
663b6af8926SPaul Durrant     }
664b6af8926SPaul Durrant 
665b6af8926SPaul Durrant     /*
66667bc8e00SPaul Durrant      * If a backend is still 'online' then we should leave it alone but,
6673809f758SPaul Durrant      * if a backend is not 'online', then the device is a candidate
6683809f758SPaul Durrant      * for destruction. Hence add it to the 'inactive' list to be cleaned
6693809f758SPaul Durrant      * by xen_bus_cleanup().
670b6af8926SPaul Durrant      */
6713809f758SPaul Durrant     if (!online &&
6723809f758SPaul Durrant         (state == XenbusStateClosed ||  state == XenbusStateInitialising ||
6733809f758SPaul Durrant          state == XenbusStateInitWait || state == XenbusStateUnknown) &&
6743809f758SPaul Durrant         !xendev->inactive) {
6753809f758SPaul Durrant         XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
676a783f8adSPaul Durrant 
6773809f758SPaul Durrant         xendev->inactive = true;
6783809f758SPaul Durrant         QLIST_INSERT_HEAD(&xenbus->inactive_devices, xendev, list);
679a783f8adSPaul Durrant 
6803809f758SPaul Durrant         /*
6813809f758SPaul Durrant          * Re-write the state to cause a XenBus backend_watch notification,
6823809f758SPaul Durrant          * resulting in a call to xen_bus_cleanup().
6833809f758SPaul Durrant          */
6843809f758SPaul Durrant         xen_device_backend_printf(xendev, "state", "%u", state);
685a783f8adSPaul Durrant     }
686b6af8926SPaul Durrant }
687b6af8926SPaul Durrant 
688d198b711SPaul Durrant static XenWatch *xen_device_add_watch(XenDevice *xendev, const char *node,
689d198b711SPaul Durrant                                       const char *key,
690d198b711SPaul Durrant                                       XenWatchHandler handler,
691d198b711SPaul Durrant                                       Error **errp)
692d198b711SPaul Durrant {
693d198b711SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
694d198b711SPaul Durrant 
695d198b711SPaul Durrant     trace_xen_device_add_watch(type, xendev->name, node, key);
696d198b711SPaul Durrant 
697d198b711SPaul Durrant     return watch_list_add(xendev->watch_list, node, key, handler, xendev,
698d198b711SPaul Durrant                           errp);
699d198b711SPaul Durrant }
700d198b711SPaul Durrant 
701d198b711SPaul Durrant static void xen_device_remove_watch(XenDevice *xendev, XenWatch *watch,
702d198b711SPaul Durrant                                     Error **errp)
703d198b711SPaul Durrant {
704d198b711SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
705d198b711SPaul Durrant 
706d198b711SPaul Durrant     trace_xen_device_remove_watch(type, xendev->name, watch->node,
707d198b711SPaul Durrant                                   watch->key);
708d198b711SPaul Durrant 
709d198b711SPaul Durrant     watch_list_remove(xendev->watch_list, watch, errp);
710d198b711SPaul Durrant }
711d198b711SPaul Durrant 
712d198b711SPaul Durrant 
713094a2239SPaul Durrant static void xen_device_backend_create(XenDevice *xendev, Error **errp)
714094a2239SPaul Durrant {
7151de7096dSVladimir Sementsov-Ogievskiy     ERRP_GUARD();
716094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
717094a2239SPaul Durrant     struct xs_permissions perms[2];
718094a2239SPaul Durrant 
719094a2239SPaul Durrant     xendev->backend_path = xen_device_get_backend_path(xendev);
720094a2239SPaul Durrant 
721094a2239SPaul Durrant     perms[0].id = xenbus->backend_id;
722094a2239SPaul Durrant     perms[0].perms = XS_PERM_NONE;
723094a2239SPaul Durrant     perms[1].id = xendev->frontend_id;
724094a2239SPaul Durrant     perms[1].perms = XS_PERM_READ;
725094a2239SPaul Durrant 
726094a2239SPaul Durrant     g_assert(xenbus->xsh);
727094a2239SPaul Durrant 
728094a2239SPaul Durrant     xs_node_create(xenbus->xsh, XBT_NULL, xendev->backend_path, perms,
7291de7096dSVladimir Sementsov-Ogievskiy                    ARRAY_SIZE(perms), errp);
7301de7096dSVladimir Sementsov-Ogievskiy     if (*errp) {
7311de7096dSVladimir Sementsov-Ogievskiy         error_prepend(errp, "failed to create backend: ");
732b6af8926SPaul Durrant         return;
733b6af8926SPaul Durrant     }
734b6af8926SPaul Durrant 
735b6af8926SPaul Durrant     xendev->backend_state_watch =
736d198b711SPaul Durrant         xen_device_add_watch(xendev, xendev->backend_path,
737b6af8926SPaul Durrant                              "state", xen_device_backend_changed,
7381de7096dSVladimir Sementsov-Ogievskiy                              errp);
7391de7096dSVladimir Sementsov-Ogievskiy     if (*errp) {
7401de7096dSVladimir Sementsov-Ogievskiy         error_prepend(errp, "failed to watch backend state: ");
741b6af8926SPaul Durrant         return;
742b6af8926SPaul Durrant     }
743b6af8926SPaul Durrant 
744b6af8926SPaul Durrant     xendev->backend_online_watch =
745d198b711SPaul Durrant         xen_device_add_watch(xendev, xendev->backend_path,
746b6af8926SPaul Durrant                              "online", xen_device_backend_changed,
7471de7096dSVladimir Sementsov-Ogievskiy                              errp);
7481de7096dSVladimir Sementsov-Ogievskiy     if (*errp) {
7491de7096dSVladimir Sementsov-Ogievskiy         error_prepend(errp, "failed to watch backend online: ");
750b6af8926SPaul Durrant         return;
751094a2239SPaul Durrant     }
752094a2239SPaul Durrant }
753094a2239SPaul Durrant 
754094a2239SPaul Durrant static void xen_device_backend_destroy(XenDevice *xendev)
755094a2239SPaul Durrant {
756094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
757094a2239SPaul Durrant     Error *local_err = NULL;
758094a2239SPaul Durrant 
759b6af8926SPaul Durrant     if (xendev->backend_online_watch) {
760d198b711SPaul Durrant         xen_device_remove_watch(xendev, xendev->backend_online_watch, NULL);
761b6af8926SPaul Durrant         xendev->backend_online_watch = NULL;
762b6af8926SPaul Durrant     }
763b6af8926SPaul Durrant 
764b6af8926SPaul Durrant     if (xendev->backend_state_watch) {
765d198b711SPaul Durrant         xen_device_remove_watch(xendev, xendev->backend_state_watch, NULL);
766b6af8926SPaul Durrant         xendev->backend_state_watch = NULL;
767b6af8926SPaul Durrant     }
768b6af8926SPaul Durrant 
769094a2239SPaul Durrant     if (!xendev->backend_path) {
770094a2239SPaul Durrant         return;
771094a2239SPaul Durrant     }
772094a2239SPaul Durrant 
773094a2239SPaul Durrant     g_assert(xenbus->xsh);
774094a2239SPaul Durrant 
775094a2239SPaul Durrant     xs_node_destroy(xenbus->xsh, XBT_NULL, xendev->backend_path,
776094a2239SPaul Durrant                     &local_err);
777094a2239SPaul Durrant     g_free(xendev->backend_path);
778094a2239SPaul Durrant     xendev->backend_path = NULL;
779094a2239SPaul Durrant 
780094a2239SPaul Durrant     if (local_err) {
781094a2239SPaul Durrant         error_report_err(local_err);
782094a2239SPaul Durrant     }
783094a2239SPaul Durrant }
784094a2239SPaul Durrant 
785b6af8926SPaul Durrant void xen_device_frontend_printf(XenDevice *xendev, const char *key,
786094a2239SPaul Durrant                                 const char *fmt, ...)
787094a2239SPaul Durrant {
788094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
789094a2239SPaul Durrant     Error *local_err = NULL;
790094a2239SPaul Durrant     va_list ap;
791094a2239SPaul Durrant 
792094a2239SPaul Durrant     g_assert(xenbus->xsh);
793094a2239SPaul Durrant 
794094a2239SPaul Durrant     va_start(ap, fmt);
795094a2239SPaul Durrant     xs_node_vprintf(xenbus->xsh, XBT_NULL, xendev->frontend_path, key,
796094a2239SPaul Durrant                     &local_err, fmt, ap);
797094a2239SPaul Durrant     va_end(ap);
798094a2239SPaul Durrant 
799094a2239SPaul Durrant     if (local_err) {
800094a2239SPaul Durrant         error_report_err(local_err);
801094a2239SPaul Durrant     }
802094a2239SPaul Durrant }
803094a2239SPaul Durrant 
804b6af8926SPaul Durrant int xen_device_frontend_scanf(XenDevice *xendev, const char *key,
80582a29e30SPaul Durrant                               const char *fmt, ...)
80682a29e30SPaul Durrant {
80782a29e30SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
80882a29e30SPaul Durrant     va_list ap;
80982a29e30SPaul Durrant     int rc;
81082a29e30SPaul Durrant 
81182a29e30SPaul Durrant     g_assert(xenbus->xsh);
81282a29e30SPaul Durrant 
81382a29e30SPaul Durrant     va_start(ap, fmt);
81482a29e30SPaul Durrant     rc = xs_node_vscanf(xenbus->xsh, XBT_NULL, xendev->frontend_path, key,
81582a29e30SPaul Durrant                         NULL, fmt, ap);
81682a29e30SPaul Durrant     va_end(ap);
81782a29e30SPaul Durrant 
81882a29e30SPaul Durrant     return rc;
81982a29e30SPaul Durrant }
82082a29e30SPaul Durrant 
821094a2239SPaul Durrant static void xen_device_frontend_set_state(XenDevice *xendev,
822705be570SAnthony PERARD                                           enum xenbus_state state,
823705be570SAnthony PERARD                                           bool publish)
824094a2239SPaul Durrant {
825094a2239SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
826094a2239SPaul Durrant 
827094a2239SPaul Durrant     if (xendev->frontend_state == state) {
828094a2239SPaul Durrant         return;
829094a2239SPaul Durrant     }
830094a2239SPaul Durrant 
831094a2239SPaul Durrant     trace_xen_device_frontend_state(type, xendev->name,
832094a2239SPaul Durrant                                     xs_strstate(state));
833094a2239SPaul Durrant 
834094a2239SPaul Durrant     xendev->frontend_state = state;
835705be570SAnthony PERARD     if (publish) {
836094a2239SPaul Durrant         xen_device_frontend_printf(xendev, "state", "%u", state);
837094a2239SPaul Durrant     }
838705be570SAnthony PERARD }
839094a2239SPaul Durrant 
84082a29e30SPaul Durrant static void xen_device_frontend_changed(void *opaque)
84182a29e30SPaul Durrant {
84282a29e30SPaul Durrant     XenDevice *xendev = opaque;
84382a29e30SPaul Durrant     XenDeviceClass *xendev_class = XEN_DEVICE_GET_CLASS(xendev);
84482a29e30SPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
84582a29e30SPaul Durrant     enum xenbus_state state;
84682a29e30SPaul Durrant 
84782a29e30SPaul Durrant     trace_xen_device_frontend_changed(type, xendev->name);
84882a29e30SPaul Durrant 
84982a29e30SPaul Durrant     if (xen_device_frontend_scanf(xendev, "state", "%u", &state) != 1) {
85082a29e30SPaul Durrant         state = XenbusStateUnknown;
85182a29e30SPaul Durrant     }
85282a29e30SPaul Durrant 
853705be570SAnthony PERARD     xen_device_frontend_set_state(xendev, state, false);
85482a29e30SPaul Durrant 
85567bc8e00SPaul Durrant     if (state == XenbusStateInitialising &&
85667bc8e00SPaul Durrant         xendev->backend_state == XenbusStateClosed &&
85767bc8e00SPaul Durrant         xendev->backend_online) {
85867bc8e00SPaul Durrant         /*
85967bc8e00SPaul Durrant          * The frontend is re-initializing so switch back to
86067bc8e00SPaul Durrant          * InitWait.
86167bc8e00SPaul Durrant          */
86267bc8e00SPaul Durrant         xen_device_backend_set_state(xendev, XenbusStateInitWait);
86367bc8e00SPaul Durrant         return;
86467bc8e00SPaul Durrant     }
86567bc8e00SPaul Durrant 
86682a29e30SPaul Durrant     if (xendev_class->frontend_changed) {
86782a29e30SPaul Durrant         Error *local_err = NULL;
86882a29e30SPaul Durrant 
86982a29e30SPaul Durrant         xendev_class->frontend_changed(xendev, state, &local_err);
87082a29e30SPaul Durrant 
87182a29e30SPaul Durrant         if (local_err) {
87282a29e30SPaul Durrant             error_reportf_err(local_err, "frontend change error: ");
87382a29e30SPaul Durrant         }
87482a29e30SPaul Durrant     }
87582a29e30SPaul Durrant }
87682a29e30SPaul Durrant 
8776bd6b955SMark Syms static bool xen_device_frontend_exists(XenDevice *xendev)
8786bd6b955SMark Syms {
8796bd6b955SMark Syms     enum xenbus_state state;
8806bd6b955SMark Syms 
8816bd6b955SMark Syms     return (xen_device_frontend_scanf(xendev, "state", "%u", &state) == 1);
8826bd6b955SMark Syms }
8836bd6b955SMark Syms 
884094a2239SPaul Durrant static void xen_device_frontend_create(XenDevice *xendev, Error **errp)
885094a2239SPaul Durrant {
8861de7096dSVladimir Sementsov-Ogievskiy     ERRP_GUARD();
887094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
888094a2239SPaul Durrant     struct xs_permissions perms[2];
889094a2239SPaul Durrant 
890094a2239SPaul Durrant     xendev->frontend_path = xen_device_get_frontend_path(xendev);
891094a2239SPaul Durrant 
8926bd6b955SMark Syms     /*
8936bd6b955SMark Syms      * The frontend area may have already been created by a legacy
8946bd6b955SMark Syms      * toolstack.
8956bd6b955SMark Syms      */
8966bd6b955SMark Syms     if (!xen_device_frontend_exists(xendev)) {
897094a2239SPaul Durrant         perms[0].id = xendev->frontend_id;
898094a2239SPaul Durrant         perms[0].perms = XS_PERM_NONE;
899094a2239SPaul Durrant         perms[1].id = xenbus->backend_id;
900094a2239SPaul Durrant         perms[1].perms = XS_PERM_READ | XS_PERM_WRITE;
901094a2239SPaul Durrant 
902094a2239SPaul Durrant         g_assert(xenbus->xsh);
903094a2239SPaul Durrant 
904094a2239SPaul Durrant         xs_node_create(xenbus->xsh, XBT_NULL, xendev->frontend_path, perms,
9051de7096dSVladimir Sementsov-Ogievskiy                        ARRAY_SIZE(perms), errp);
9061de7096dSVladimir Sementsov-Ogievskiy         if (*errp) {
9071de7096dSVladimir Sementsov-Ogievskiy             error_prepend(errp, "failed to create frontend: ");
90882a29e30SPaul Durrant             return;
90982a29e30SPaul Durrant         }
9106bd6b955SMark Syms     }
91182a29e30SPaul Durrant 
91282a29e30SPaul Durrant     xendev->frontend_state_watch =
913d198b711SPaul Durrant         xen_device_add_watch(xendev, xendev->frontend_path, "state",
9141de7096dSVladimir Sementsov-Ogievskiy                              xen_device_frontend_changed, errp);
9151de7096dSVladimir Sementsov-Ogievskiy     if (*errp) {
9161de7096dSVladimir Sementsov-Ogievskiy         error_prepend(errp, "failed to watch frontend state: ");
917094a2239SPaul Durrant     }
918094a2239SPaul Durrant }
919094a2239SPaul Durrant 
920094a2239SPaul Durrant static void xen_device_frontend_destroy(XenDevice *xendev)
921094a2239SPaul Durrant {
922094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
923094a2239SPaul Durrant     Error *local_err = NULL;
924094a2239SPaul Durrant 
92582a29e30SPaul Durrant     if (xendev->frontend_state_watch) {
926d198b711SPaul Durrant         xen_device_remove_watch(xendev, xendev->frontend_state_watch,
927d198b711SPaul Durrant                                 NULL);
92882a29e30SPaul Durrant         xendev->frontend_state_watch = NULL;
92982a29e30SPaul Durrant     }
93082a29e30SPaul Durrant 
931094a2239SPaul Durrant     if (!xendev->frontend_path) {
932094a2239SPaul Durrant         return;
933094a2239SPaul Durrant     }
934094a2239SPaul Durrant 
935094a2239SPaul Durrant     g_assert(xenbus->xsh);
936094a2239SPaul Durrant 
937094a2239SPaul Durrant     xs_node_destroy(xenbus->xsh, XBT_NULL, xendev->frontend_path,
938094a2239SPaul Durrant                     &local_err);
939094a2239SPaul Durrant     g_free(xendev->frontend_path);
940094a2239SPaul Durrant     xendev->frontend_path = NULL;
941094a2239SPaul Durrant 
942094a2239SPaul Durrant     if (local_err) {
943094a2239SPaul Durrant         error_report_err(local_err);
944094a2239SPaul Durrant     }
945094a2239SPaul Durrant }
946094a2239SPaul Durrant 
9474b34b5b1SPaul Durrant void xen_device_set_max_grant_refs(XenDevice *xendev, unsigned int nr_refs,
9484b34b5b1SPaul Durrant                                    Error **errp)
9494b34b5b1SPaul Durrant {
9504b34b5b1SPaul Durrant     if (xengnttab_set_max_grants(xendev->xgth, nr_refs)) {
9514b34b5b1SPaul Durrant         error_setg_errno(errp, errno, "xengnttab_set_max_grants failed");
9524b34b5b1SPaul Durrant     }
9534b34b5b1SPaul Durrant }
9544b34b5b1SPaul Durrant 
9554b34b5b1SPaul Durrant void *xen_device_map_grant_refs(XenDevice *xendev, uint32_t *refs,
9564b34b5b1SPaul Durrant                                 unsigned int nr_refs, int prot,
9574b34b5b1SPaul Durrant                                 Error **errp)
9584b34b5b1SPaul Durrant {
9594b34b5b1SPaul Durrant     void *map = xengnttab_map_domain_grant_refs(xendev->xgth, nr_refs,
9604b34b5b1SPaul Durrant                                                 xendev->frontend_id, refs,
9614b34b5b1SPaul Durrant                                                 prot);
9624b34b5b1SPaul Durrant 
9634b34b5b1SPaul Durrant     if (!map) {
9644b34b5b1SPaul Durrant         error_setg_errno(errp, errno,
9654b34b5b1SPaul Durrant                          "xengnttab_map_domain_grant_refs failed");
9664b34b5b1SPaul Durrant     }
9674b34b5b1SPaul Durrant 
9684b34b5b1SPaul Durrant     return map;
9694b34b5b1SPaul Durrant }
9704b34b5b1SPaul Durrant 
9714b34b5b1SPaul Durrant void xen_device_unmap_grant_refs(XenDevice *xendev, void *map,
9724b34b5b1SPaul Durrant                                  unsigned int nr_refs, Error **errp)
9734b34b5b1SPaul Durrant {
9744b34b5b1SPaul Durrant     if (xengnttab_unmap(xendev->xgth, map, nr_refs)) {
9754b34b5b1SPaul Durrant         error_setg_errno(errp, errno, "xengnttab_unmap failed");
9764b34b5b1SPaul Durrant     }
9774b34b5b1SPaul Durrant }
9784b34b5b1SPaul Durrant 
9794b34b5b1SPaul Durrant static void compat_copy_grant_refs(XenDevice *xendev, bool to_domain,
9804b34b5b1SPaul Durrant                                    XenDeviceGrantCopySegment segs[],
9814b34b5b1SPaul Durrant                                    unsigned int nr_segs, Error **errp)
9824b34b5b1SPaul Durrant {
9834b34b5b1SPaul Durrant     uint32_t *refs = g_new(uint32_t, nr_segs);
9844b34b5b1SPaul Durrant     int prot = to_domain ? PROT_WRITE : PROT_READ;
9854b34b5b1SPaul Durrant     void *map;
9864b34b5b1SPaul Durrant     unsigned int i;
9874b34b5b1SPaul Durrant 
9884b34b5b1SPaul Durrant     for (i = 0; i < nr_segs; i++) {
9894b34b5b1SPaul Durrant         XenDeviceGrantCopySegment *seg = &segs[i];
9904b34b5b1SPaul Durrant 
9914b34b5b1SPaul Durrant         refs[i] = to_domain ? seg->dest.foreign.ref :
9924b34b5b1SPaul Durrant             seg->source.foreign.ref;
9934b34b5b1SPaul Durrant     }
9944b34b5b1SPaul Durrant 
9954b34b5b1SPaul Durrant     map = xengnttab_map_domain_grant_refs(xendev->xgth, nr_segs,
9964b34b5b1SPaul Durrant                                           xendev->frontend_id, refs,
9974b34b5b1SPaul Durrant                                           prot);
9984b34b5b1SPaul Durrant     if (!map) {
9994b34b5b1SPaul Durrant         error_setg_errno(errp, errno,
10004b34b5b1SPaul Durrant                          "xengnttab_map_domain_grant_refs failed");
10014b34b5b1SPaul Durrant         goto done;
10024b34b5b1SPaul Durrant     }
10034b34b5b1SPaul Durrant 
10044b34b5b1SPaul Durrant     for (i = 0; i < nr_segs; i++) {
10054b34b5b1SPaul Durrant         XenDeviceGrantCopySegment *seg = &segs[i];
10064b34b5b1SPaul Durrant         void *page = map + (i * XC_PAGE_SIZE);
10074b34b5b1SPaul Durrant 
10084b34b5b1SPaul Durrant         if (to_domain) {
10094b34b5b1SPaul Durrant             memcpy(page + seg->dest.foreign.offset, seg->source.virt,
10104b34b5b1SPaul Durrant                    seg->len);
10114b34b5b1SPaul Durrant         } else {
10124b34b5b1SPaul Durrant             memcpy(seg->dest.virt, page + seg->source.foreign.offset,
10134b34b5b1SPaul Durrant                    seg->len);
10144b34b5b1SPaul Durrant         }
10154b34b5b1SPaul Durrant     }
10164b34b5b1SPaul Durrant 
10174b34b5b1SPaul Durrant     if (xengnttab_unmap(xendev->xgth, map, nr_segs)) {
10184b34b5b1SPaul Durrant         error_setg_errno(errp, errno, "xengnttab_unmap failed");
10194b34b5b1SPaul Durrant     }
10204b34b5b1SPaul Durrant 
10214b34b5b1SPaul Durrant done:
10224b34b5b1SPaul Durrant     g_free(refs);
10234b34b5b1SPaul Durrant }
10244b34b5b1SPaul Durrant 
10254b34b5b1SPaul Durrant void xen_device_copy_grant_refs(XenDevice *xendev, bool to_domain,
10264b34b5b1SPaul Durrant                                 XenDeviceGrantCopySegment segs[],
10274b34b5b1SPaul Durrant                                 unsigned int nr_segs, Error **errp)
10284b34b5b1SPaul Durrant {
10294b34b5b1SPaul Durrant     xengnttab_grant_copy_segment_t *xengnttab_segs;
10304b34b5b1SPaul Durrant     unsigned int i;
10314b34b5b1SPaul Durrant 
10324b34b5b1SPaul Durrant     if (!xendev->feature_grant_copy) {
10334b34b5b1SPaul Durrant         compat_copy_grant_refs(xendev, to_domain, segs, nr_segs, errp);
10344b34b5b1SPaul Durrant         return;
10354b34b5b1SPaul Durrant     }
10364b34b5b1SPaul Durrant 
10374b34b5b1SPaul Durrant     xengnttab_segs = g_new0(xengnttab_grant_copy_segment_t, nr_segs);
10384b34b5b1SPaul Durrant 
10394b34b5b1SPaul Durrant     for (i = 0; i < nr_segs; i++) {
10404b34b5b1SPaul Durrant         XenDeviceGrantCopySegment *seg = &segs[i];
10414b34b5b1SPaul Durrant         xengnttab_grant_copy_segment_t *xengnttab_seg = &xengnttab_segs[i];
10424b34b5b1SPaul Durrant 
10434b34b5b1SPaul Durrant         if (to_domain) {
10444b34b5b1SPaul Durrant             xengnttab_seg->flags = GNTCOPY_dest_gref;
10454b34b5b1SPaul Durrant             xengnttab_seg->dest.foreign.domid = xendev->frontend_id;
10464b34b5b1SPaul Durrant             xengnttab_seg->dest.foreign.ref = seg->dest.foreign.ref;
10474b34b5b1SPaul Durrant             xengnttab_seg->dest.foreign.offset = seg->dest.foreign.offset;
10484b34b5b1SPaul Durrant             xengnttab_seg->source.virt = seg->source.virt;
10494b34b5b1SPaul Durrant         } else {
10504b34b5b1SPaul Durrant             xengnttab_seg->flags = GNTCOPY_source_gref;
10514b34b5b1SPaul Durrant             xengnttab_seg->source.foreign.domid = xendev->frontend_id;
10524b34b5b1SPaul Durrant             xengnttab_seg->source.foreign.ref = seg->source.foreign.ref;
10534b34b5b1SPaul Durrant             xengnttab_seg->source.foreign.offset =
10544b34b5b1SPaul Durrant                 seg->source.foreign.offset;
10554b34b5b1SPaul Durrant             xengnttab_seg->dest.virt = seg->dest.virt;
10564b34b5b1SPaul Durrant         }
10574b34b5b1SPaul Durrant 
10584b34b5b1SPaul Durrant         xengnttab_seg->len = seg->len;
10594b34b5b1SPaul Durrant     }
10604b34b5b1SPaul Durrant 
10614b34b5b1SPaul Durrant     if (xengnttab_grant_copy(xendev->xgth, nr_segs, xengnttab_segs)) {
10624b34b5b1SPaul Durrant         error_setg_errno(errp, errno, "xengnttab_grant_copy failed");
10634b34b5b1SPaul Durrant         goto done;
10644b34b5b1SPaul Durrant     }
10654b34b5b1SPaul Durrant 
10664b34b5b1SPaul Durrant     for (i = 0; i < nr_segs; i++) {
10674b34b5b1SPaul Durrant         xengnttab_grant_copy_segment_t *xengnttab_seg = &xengnttab_segs[i];
10684b34b5b1SPaul Durrant 
10694b34b5b1SPaul Durrant         if (xengnttab_seg->status != GNTST_okay) {
10704b34b5b1SPaul Durrant             error_setg(errp, "xengnttab_grant_copy seg[%u] failed", i);
10714b34b5b1SPaul Durrant             break;
10724b34b5b1SPaul Durrant         }
10734b34b5b1SPaul Durrant     }
10744b34b5b1SPaul Durrant 
10754b34b5b1SPaul Durrant done:
10764b34b5b1SPaul Durrant     g_free(xengnttab_segs);
10774b34b5b1SPaul Durrant }
10784b34b5b1SPaul Durrant 
1079a3d669c8SPaul Durrant struct XenEventChannel {
1080c0b336eaSPaul Durrant     QLIST_ENTRY(XenEventChannel) list;
108183361a8aSPaul Durrant     AioContext *ctx;
1082c0b336eaSPaul Durrant     xenevtchn_handle *xeh;
1083a3d669c8SPaul Durrant     evtchn_port_t local_port;
1084a3d669c8SPaul Durrant     XenEventHandler handler;
1085a3d669c8SPaul Durrant     void *opaque;
1086a3d669c8SPaul Durrant };
1087a3d669c8SPaul Durrant 
1088345f42b4SPaul Durrant static bool xen_device_poll(void *opaque)
1089345f42b4SPaul Durrant {
1090345f42b4SPaul Durrant     XenEventChannel *channel = opaque;
1091345f42b4SPaul Durrant 
1092345f42b4SPaul Durrant     return channel->handler(channel->opaque);
1093345f42b4SPaul Durrant }
1094345f42b4SPaul Durrant 
1095c0b336eaSPaul Durrant static void xen_device_event(void *opaque)
1096a3d669c8SPaul Durrant {
1097c0b336eaSPaul Durrant     XenEventChannel *channel = opaque;
1098*b6cacfeaSDavid Woodhouse     unsigned long port = qemu_xen_evtchn_pending(channel->xeh);
1099a3d669c8SPaul Durrant 
1100a3d669c8SPaul Durrant     if (port == channel->local_port) {
1101345f42b4SPaul Durrant         xen_device_poll(channel);
1102c0b336eaSPaul Durrant 
1103*b6cacfeaSDavid Woodhouse         qemu_xen_evtchn_unmask(channel->xeh, port);
1104a3d669c8SPaul Durrant     }
1105a3d669c8SPaul Durrant }
1106a3d669c8SPaul Durrant 
110732d0b7beSPaul Durrant void xen_device_set_event_channel_context(XenDevice *xendev,
110832d0b7beSPaul Durrant                                           XenEventChannel *channel,
110983361a8aSPaul Durrant                                           AioContext *ctx,
111032d0b7beSPaul Durrant                                           Error **errp)
111132d0b7beSPaul Durrant {
111232d0b7beSPaul Durrant     if (!channel) {
111332d0b7beSPaul Durrant         error_setg(errp, "bad channel");
111432d0b7beSPaul Durrant         return;
111532d0b7beSPaul Durrant     }
111632d0b7beSPaul Durrant 
111732d0b7beSPaul Durrant     if (channel->ctx)
1118*b6cacfeaSDavid Woodhouse         aio_set_fd_handler(channel->ctx, qemu_xen_evtchn_fd(channel->xeh), true,
1119826cc324SStefan Hajnoczi                            NULL, NULL, NULL, NULL, NULL);
112032d0b7beSPaul Durrant 
112132d0b7beSPaul Durrant     channel->ctx = ctx;
1122*b6cacfeaSDavid Woodhouse     aio_set_fd_handler(channel->ctx, qemu_xen_evtchn_fd(channel->xeh), true,
1123826cc324SStefan Hajnoczi                        xen_device_event, NULL, xen_device_poll, NULL, channel);
112432d0b7beSPaul Durrant }
112532d0b7beSPaul Durrant 
112632d0b7beSPaul Durrant XenEventChannel *xen_device_bind_event_channel(XenDevice *xendev,
1127a3d669c8SPaul Durrant                                                unsigned int port,
1128a3d669c8SPaul Durrant                                                XenEventHandler handler,
1129a3d669c8SPaul Durrant                                                void *opaque, Error **errp)
1130a3d669c8SPaul Durrant {
1131a3d669c8SPaul Durrant     XenEventChannel *channel = g_new0(XenEventChannel, 1);
1132a3d669c8SPaul Durrant     xenevtchn_port_or_error_t local_port;
1133a3d669c8SPaul Durrant 
1134*b6cacfeaSDavid Woodhouse     channel->xeh = qemu_xen_evtchn_open();
1135c0b336eaSPaul Durrant     if (!channel->xeh) {
1136c0b336eaSPaul Durrant         error_setg_errno(errp, errno, "failed xenevtchn_open");
1137c0b336eaSPaul Durrant         goto fail;
1138c0b336eaSPaul Durrant     }
1139c0b336eaSPaul Durrant 
1140*b6cacfeaSDavid Woodhouse     local_port = qemu_xen_evtchn_bind_interdomain(channel->xeh,
1141a3d669c8SPaul Durrant                                             xendev->frontend_id,
1142a3d669c8SPaul Durrant                                             port);
1143a3d669c8SPaul Durrant     if (local_port < 0) {
1144a3d669c8SPaul Durrant         error_setg_errno(errp, errno, "xenevtchn_bind_interdomain failed");
1145c0b336eaSPaul Durrant         goto fail;
1146a3d669c8SPaul Durrant     }
1147a3d669c8SPaul Durrant 
1148a3d669c8SPaul Durrant     channel->local_port = local_port;
1149a3d669c8SPaul Durrant     channel->handler = handler;
1150a3d669c8SPaul Durrant     channel->opaque = opaque;
1151a3d669c8SPaul Durrant 
115232d0b7beSPaul Durrant     /* Only reason for failure is a NULL channel */
115332d0b7beSPaul Durrant     xen_device_set_event_channel_context(xendev, channel,
115432d0b7beSPaul Durrant                                          qemu_get_aio_context(),
115532d0b7beSPaul Durrant                                          &error_abort);
1156c0b336eaSPaul Durrant 
1157c0b336eaSPaul Durrant     QLIST_INSERT_HEAD(&xendev->event_channels, channel, list);
1158a3d669c8SPaul Durrant 
1159a3d669c8SPaul Durrant     return channel;
1160c0b336eaSPaul Durrant 
1161c0b336eaSPaul Durrant fail:
1162c0b336eaSPaul Durrant     if (channel->xeh) {
1163*b6cacfeaSDavid Woodhouse         qemu_xen_evtchn_close(channel->xeh);
1164c0b336eaSPaul Durrant     }
1165c0b336eaSPaul Durrant 
1166c0b336eaSPaul Durrant     g_free(channel);
1167c0b336eaSPaul Durrant 
1168c0b336eaSPaul Durrant     return NULL;
1169a3d669c8SPaul Durrant }
1170a3d669c8SPaul Durrant 
1171a3d669c8SPaul Durrant void xen_device_notify_event_channel(XenDevice *xendev,
1172a3d669c8SPaul Durrant                                      XenEventChannel *channel,
1173a3d669c8SPaul Durrant                                      Error **errp)
1174a3d669c8SPaul Durrant {
1175a3d669c8SPaul Durrant     if (!channel) {
1176a3d669c8SPaul Durrant         error_setg(errp, "bad channel");
1177a3d669c8SPaul Durrant         return;
1178a3d669c8SPaul Durrant     }
1179a3d669c8SPaul Durrant 
1180*b6cacfeaSDavid Woodhouse     if (qemu_xen_evtchn_notify(channel->xeh, channel->local_port) < 0) {
1181a3d669c8SPaul Durrant         error_setg_errno(errp, errno, "xenevtchn_notify failed");
1182a3d669c8SPaul Durrant     }
1183a3d669c8SPaul Durrant }
1184a3d669c8SPaul Durrant 
1185a3d669c8SPaul Durrant void xen_device_unbind_event_channel(XenDevice *xendev,
1186a3d669c8SPaul Durrant                                      XenEventChannel *channel,
1187a3d669c8SPaul Durrant                                      Error **errp)
1188a3d669c8SPaul Durrant {
1189a3d669c8SPaul Durrant     if (!channel) {
1190a3d669c8SPaul Durrant         error_setg(errp, "bad channel");
1191a3d669c8SPaul Durrant         return;
1192a3d669c8SPaul Durrant     }
1193a3d669c8SPaul Durrant 
1194c0b336eaSPaul Durrant     QLIST_REMOVE(channel, list);
1195a3d669c8SPaul Durrant 
1196*b6cacfeaSDavid Woodhouse     aio_set_fd_handler(channel->ctx, qemu_xen_evtchn_fd(channel->xeh), true,
1197826cc324SStefan Hajnoczi                        NULL, NULL, NULL, NULL, NULL);
1198c0b336eaSPaul Durrant 
1199*b6cacfeaSDavid Woodhouse     if (qemu_xen_evtchn_unbind(channel->xeh, channel->local_port) < 0) {
1200a3d669c8SPaul Durrant         error_setg_errno(errp, errno, "xenevtchn_unbind failed");
1201a3d669c8SPaul Durrant     }
1202a3d669c8SPaul Durrant 
1203*b6cacfeaSDavid Woodhouse     qemu_xen_evtchn_close(channel->xeh);
1204a3d669c8SPaul Durrant     g_free(channel);
1205a3d669c8SPaul Durrant }
1206a3d669c8SPaul Durrant 
1207b69c3c21SMarkus Armbruster static void xen_device_unrealize(DeviceState *dev)
1208108f7bbaSPaul Durrant {
1209108f7bbaSPaul Durrant     XenDevice *xendev = XEN_DEVICE(dev);
1210108f7bbaSPaul Durrant     XenDeviceClass *xendev_class = XEN_DEVICE_GET_CLASS(xendev);
1211108f7bbaSPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
1212c0b336eaSPaul Durrant     XenEventChannel *channel, *next;
1213108f7bbaSPaul Durrant 
1214094a2239SPaul Durrant     if (!xendev->name) {
1215094a2239SPaul Durrant         return;
1216094a2239SPaul Durrant     }
1217094a2239SPaul Durrant 
1218094a2239SPaul Durrant     trace_xen_device_unrealize(type, xendev->name);
1219094a2239SPaul Durrant 
1220094a2239SPaul Durrant     if (xendev->exit.notify) {
1221094a2239SPaul Durrant         qemu_remove_exit_notifier(&xendev->exit);
1222094a2239SPaul Durrant         xendev->exit.notify = NULL;
1223094a2239SPaul Durrant     }
1224108f7bbaSPaul Durrant 
1225108f7bbaSPaul Durrant     if (xendev_class->unrealize) {
1226b69c3c21SMarkus Armbruster         xendev_class->unrealize(xendev);
1227108f7bbaSPaul Durrant     }
1228094a2239SPaul Durrant 
1229c0b336eaSPaul Durrant     /* Make sure all event channels are cleaned up */
1230c0b336eaSPaul Durrant     QLIST_FOREACH_SAFE(channel, &xendev->event_channels, list, next) {
1231c0b336eaSPaul Durrant         xen_device_unbind_event_channel(xendev, channel, NULL);
1232c0b336eaSPaul Durrant     }
1233c0b336eaSPaul Durrant 
1234094a2239SPaul Durrant     xen_device_frontend_destroy(xendev);
1235094a2239SPaul Durrant     xen_device_backend_destroy(xendev);
1236094a2239SPaul Durrant 
12374b34b5b1SPaul Durrant     if (xendev->xgth) {
12384b34b5b1SPaul Durrant         xengnttab_close(xendev->xgth);
12394b34b5b1SPaul Durrant         xendev->xgth = NULL;
12404b34b5b1SPaul Durrant     }
12414b34b5b1SPaul Durrant 
1242d198b711SPaul Durrant     if (xendev->watch_list) {
1243d198b711SPaul Durrant         watch_list_destroy(xendev->watch_list);
1244d198b711SPaul Durrant         xendev->watch_list = NULL;
1245d198b711SPaul Durrant     }
1246d198b711SPaul Durrant 
1247d198b711SPaul Durrant     if (xendev->xsh) {
1248d198b711SPaul Durrant         xs_close(xendev->xsh);
1249d198b711SPaul Durrant         xendev->xsh = NULL;
1250d198b711SPaul Durrant     }
1251d198b711SPaul Durrant 
1252094a2239SPaul Durrant     g_free(xendev->name);
1253094a2239SPaul Durrant     xendev->name = NULL;
1254094a2239SPaul Durrant }
1255094a2239SPaul Durrant 
1256094a2239SPaul Durrant static void xen_device_exit(Notifier *n, void *data)
1257094a2239SPaul Durrant {
1258094a2239SPaul Durrant     XenDevice *xendev = container_of(n, XenDevice, exit);
1259094a2239SPaul Durrant 
1260b69c3c21SMarkus Armbruster     xen_device_unrealize(DEVICE(xendev));
1261108f7bbaSPaul Durrant }
1262108f7bbaSPaul Durrant 
1263108f7bbaSPaul Durrant static void xen_device_realize(DeviceState *dev, Error **errp)
1264108f7bbaSPaul Durrant {
12651de7096dSVladimir Sementsov-Ogievskiy     ERRP_GUARD();
1266108f7bbaSPaul Durrant     XenDevice *xendev = XEN_DEVICE(dev);
1267108f7bbaSPaul Durrant     XenDeviceClass *xendev_class = XEN_DEVICE_GET_CLASS(xendev);
1268094a2239SPaul Durrant     XenBus *xenbus = XEN_BUS(qdev_get_parent_bus(DEVICE(xendev)));
1269108f7bbaSPaul Durrant     const char *type = object_get_typename(OBJECT(xendev));
1270108f7bbaSPaul Durrant 
1271094a2239SPaul Durrant     if (xendev->frontend_id == DOMID_INVALID) {
1272094a2239SPaul Durrant         xendev->frontend_id = xen_domid;
1273094a2239SPaul Durrant     }
1274094a2239SPaul Durrant 
1275094a2239SPaul Durrant     if (xendev->frontend_id >= DOMID_FIRST_RESERVED) {
1276094a2239SPaul Durrant         error_setg(errp, "invalid frontend-id");
1277094a2239SPaul Durrant         goto unrealize;
1278094a2239SPaul Durrant     }
1279094a2239SPaul Durrant 
1280094a2239SPaul Durrant     if (!xendev_class->get_name) {
1281094a2239SPaul Durrant         error_setg(errp, "get_name method not implemented");
1282094a2239SPaul Durrant         goto unrealize;
1283094a2239SPaul Durrant     }
1284094a2239SPaul Durrant 
12851de7096dSVladimir Sementsov-Ogievskiy     xendev->name = xendev_class->get_name(xendev, errp);
12861de7096dSVladimir Sementsov-Ogievskiy     if (*errp) {
12871de7096dSVladimir Sementsov-Ogievskiy         error_prepend(errp, "failed to get device name: ");
1288094a2239SPaul Durrant         goto unrealize;
1289094a2239SPaul Durrant     }
1290094a2239SPaul Durrant 
1291094a2239SPaul Durrant     trace_xen_device_realize(type, xendev->name);
1292094a2239SPaul Durrant 
1293d198b711SPaul Durrant     xendev->xsh = xs_open(0);
1294d198b711SPaul Durrant     if (!xendev->xsh) {
1295d198b711SPaul Durrant         error_setg_errno(errp, errno, "failed xs_open");
1296d198b711SPaul Durrant         goto unrealize;
1297d198b711SPaul Durrant     }
1298d198b711SPaul Durrant 
1299d198b711SPaul Durrant     xendev->watch_list = watch_list_create(xendev->xsh);
1300d198b711SPaul Durrant 
13014b34b5b1SPaul Durrant     xendev->xgth = xengnttab_open(NULL, 0);
13024b34b5b1SPaul Durrant     if (!xendev->xgth) {
13034b34b5b1SPaul Durrant         error_setg_errno(errp, errno, "failed xengnttab_open");
13044b34b5b1SPaul Durrant         goto unrealize;
13054b34b5b1SPaul Durrant     }
13064b34b5b1SPaul Durrant 
13074b34b5b1SPaul Durrant     xendev->feature_grant_copy =
13084b34b5b1SPaul Durrant         (xengnttab_grant_copy(xendev->xgth, 0, NULL) == 0);
13094b34b5b1SPaul Durrant 
13101de7096dSVladimir Sementsov-Ogievskiy     xen_device_backend_create(xendev, errp);
13111de7096dSVladimir Sementsov-Ogievskiy     if (*errp) {
1312094a2239SPaul Durrant         goto unrealize;
1313094a2239SPaul Durrant     }
1314094a2239SPaul Durrant 
13151de7096dSVladimir Sementsov-Ogievskiy     xen_device_frontend_create(xendev, errp);
13161de7096dSVladimir Sementsov-Ogievskiy     if (*errp) {
1317094a2239SPaul Durrant         goto unrealize;
1318094a2239SPaul Durrant     }
1319108f7bbaSPaul Durrant 
1320108f7bbaSPaul Durrant     if (xendev_class->realize) {
13211de7096dSVladimir Sementsov-Ogievskiy         xendev_class->realize(xendev, errp);
13221de7096dSVladimir Sementsov-Ogievskiy         if (*errp) {
1323108f7bbaSPaul Durrant             goto unrealize;
1324108f7bbaSPaul Durrant         }
1325108f7bbaSPaul Durrant     }
1326108f7bbaSPaul Durrant 
1327094a2239SPaul Durrant     xen_device_backend_printf(xendev, "frontend", "%s",
1328094a2239SPaul Durrant                               xendev->frontend_path);
1329094a2239SPaul Durrant     xen_device_backend_printf(xendev, "frontend-id", "%u",
1330094a2239SPaul Durrant                               xendev->frontend_id);
1331094a2239SPaul Durrant     xen_device_backend_printf(xendev, "hotplug-status", "connected");
1332094a2239SPaul Durrant 
1333b6af8926SPaul Durrant     xen_device_backend_set_online(xendev, true);
1334094a2239SPaul Durrant     xen_device_backend_set_state(xendev, XenbusStateInitWait);
1335094a2239SPaul Durrant 
13366bd6b955SMark Syms     if (!xen_device_frontend_exists(xendev)) {
1337094a2239SPaul Durrant         xen_device_frontend_printf(xendev, "backend", "%s",
1338094a2239SPaul Durrant                                    xendev->backend_path);
1339094a2239SPaul Durrant         xen_device_frontend_printf(xendev, "backend-id", "%u",
1340094a2239SPaul Durrant                                    xenbus->backend_id);
1341094a2239SPaul Durrant 
1342705be570SAnthony PERARD         xen_device_frontend_set_state(xendev, XenbusStateInitialising, true);
13436bd6b955SMark Syms     }
1344094a2239SPaul Durrant 
1345094a2239SPaul Durrant     xendev->exit.notify = xen_device_exit;
1346094a2239SPaul Durrant     qemu_add_exit_notifier(&xendev->exit);
1347108f7bbaSPaul Durrant     return;
1348108f7bbaSPaul Durrant 
1349108f7bbaSPaul Durrant unrealize:
1350b69c3c21SMarkus Armbruster     xen_device_unrealize(dev);
1351108f7bbaSPaul Durrant }
1352108f7bbaSPaul Durrant 
1353094a2239SPaul Durrant static Property xen_device_props[] = {
1354094a2239SPaul Durrant     DEFINE_PROP_UINT16("frontend-id", XenDevice, frontend_id,
1355094a2239SPaul Durrant                        DOMID_INVALID),
1356094a2239SPaul Durrant     DEFINE_PROP_END_OF_LIST()
1357094a2239SPaul Durrant };
1358094a2239SPaul Durrant 
1359108f7bbaSPaul Durrant static void xen_device_class_init(ObjectClass *class, void *data)
1360108f7bbaSPaul Durrant {
1361108f7bbaSPaul Durrant     DeviceClass *dev_class = DEVICE_CLASS(class);
1362108f7bbaSPaul Durrant 
1363108f7bbaSPaul Durrant     dev_class->realize = xen_device_realize;
1364108f7bbaSPaul Durrant     dev_class->unrealize = xen_device_unrealize;
13654f67d30bSMarc-André Lureau     device_class_set_props(dev_class, xen_device_props);
1366108f7bbaSPaul Durrant     dev_class->bus_type = TYPE_XEN_BUS;
1367108f7bbaSPaul Durrant }
1368108f7bbaSPaul Durrant 
1369108f7bbaSPaul Durrant static const TypeInfo xen_device_type_info = {
1370108f7bbaSPaul Durrant     .name = TYPE_XEN_DEVICE,
1371108f7bbaSPaul Durrant     .parent = TYPE_DEVICE,
1372108f7bbaSPaul Durrant     .instance_size = sizeof(XenDevice),
1373108f7bbaSPaul Durrant     .abstract = true,
1374108f7bbaSPaul Durrant     .class_size = sizeof(XenDeviceClass),
1375108f7bbaSPaul Durrant     .class_init = xen_device_class_init,
1376108f7bbaSPaul Durrant };
1377108f7bbaSPaul Durrant 
1378108f7bbaSPaul Durrant typedef struct XenBridge {
1379108f7bbaSPaul Durrant     SysBusDevice busdev;
1380108f7bbaSPaul Durrant } XenBridge;
1381108f7bbaSPaul Durrant 
1382108f7bbaSPaul Durrant #define TYPE_XEN_BRIDGE "xen-bridge"
1383108f7bbaSPaul Durrant 
1384108f7bbaSPaul Durrant static const TypeInfo xen_bridge_type_info = {
1385108f7bbaSPaul Durrant     .name = TYPE_XEN_BRIDGE,
1386108f7bbaSPaul Durrant     .parent = TYPE_SYS_BUS_DEVICE,
1387108f7bbaSPaul Durrant     .instance_size = sizeof(XenBridge),
1388108f7bbaSPaul Durrant };
1389108f7bbaSPaul Durrant 
1390108f7bbaSPaul Durrant static void xen_register_types(void)
1391108f7bbaSPaul Durrant {
1392108f7bbaSPaul Durrant     type_register_static(&xen_bridge_type_info);
1393108f7bbaSPaul Durrant     type_register_static(&xen_bus_type_info);
1394108f7bbaSPaul Durrant     type_register_static(&xen_device_type_info);
1395108f7bbaSPaul Durrant }
1396108f7bbaSPaul Durrant 
1397108f7bbaSPaul Durrant type_init(xen_register_types)
1398108f7bbaSPaul Durrant 
1399108f7bbaSPaul Durrant void xen_bus_init(void)
1400108f7bbaSPaul Durrant {
14013e80f690SMarkus Armbruster     DeviceState *dev = qdev_new(TYPE_XEN_BRIDGE);
14029388d170SPeter Maydell     BusState *bus = qbus_new(TYPE_XEN_BUS, dev, NULL);
1403108f7bbaSPaul Durrant 
14043c6ef471SMarkus Armbruster     sysbus_realize_and_unref(SYS_BUS_DEVICE(dev), &error_fatal);
1405cd7c8660SMarkus Armbruster     qbus_set_bus_hotplug_handler(bus);
1406108f7bbaSPaul Durrant }
1407