2 * xen backend driver infrastructure
3 * (c) 2008 Gerd Hoffmann <kraxel@redhat.com>
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; under version 2 of the License.
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License along
15 * with this program; if not, see <http://www.gnu.org/licenses/>.
17 * Contributions after 2012-01-13 are licensed under the terms of the
18 * GNU GPL, version 2 or (at your option) any later version.
22 * TODO: add some xenbus / xenstore concepts overview here.
25 #include "qemu/osdep.h"
26 #include <sys/signal.h>
28 #include "hw/sysbus.h"
29 #include "hw/boards.h"
30 #include "hw/qdev-properties.h"
32 #include "qemu/main-loop.h"
33 #include "qapi/error.h"
34 #include "hw/xen/xen-legacy-backend.h"
35 #include "hw/xen/xen_pvdev.h"
36 #include "monitor/qdev.h"
38 DeviceState
*xen_sysdev
;
41 /* ------------------------------------------------------------- */
44 struct xs_handle
*xenstore
;
45 const char *xen_protocol
;
48 static bool xen_feature_grant_copy
;
51 int xenstore_write_be_str(struct XenLegacyDevice
*xendev
, const char *node
,
54 return xenstore_write_str(xendev
->be
, node
, val
);
57 int xenstore_write_be_int(struct XenLegacyDevice
*xendev
, const char *node
,
60 return xenstore_write_int(xendev
->be
, node
, ival
);
63 int xenstore_write_be_int64(struct XenLegacyDevice
*xendev
, const char *node
,
66 return xenstore_write_int64(xendev
->be
, node
, ival
);
69 char *xenstore_read_be_str(struct XenLegacyDevice
*xendev
, const char *node
)
71 return xenstore_read_str(xendev
->be
, node
);
74 int xenstore_read_be_int(struct XenLegacyDevice
*xendev
, const char *node
,
77 return xenstore_read_int(xendev
->be
, node
, ival
);
80 char *xenstore_read_fe_str(struct XenLegacyDevice
*xendev
, const char *node
)
82 return xenstore_read_str(xendev
->fe
, node
);
85 int xenstore_read_fe_int(struct XenLegacyDevice
*xendev
, const char *node
,
88 return xenstore_read_int(xendev
->fe
, node
, ival
);
91 int xenstore_read_fe_uint64(struct XenLegacyDevice
*xendev
, const char *node
,
94 return xenstore_read_uint64(xendev
->fe
, node
, uval
);
97 /* ------------------------------------------------------------- */
99 int xen_be_set_state(struct XenLegacyDevice
*xendev
, enum xenbus_state state
)
103 rc
= xenstore_write_be_int(xendev
, "state", state
);
107 xen_pv_printf(xendev
, 1, "backend state: %s -> %s\n",
108 xenbus_strstate(xendev
->be_state
), xenbus_strstate(state
));
109 xendev
->be_state
= state
;
113 void xen_be_set_max_grant_refs(struct XenLegacyDevice
*xendev
,
114 unsigned int nr_refs
)
116 assert(xendev
->ops
->flags
& DEVOPS_FLAG_NEED_GNTDEV
);
118 if (xengnttab_set_max_grants(xendev
->gnttabdev
, nr_refs
)) {
119 xen_pv_printf(xendev
, 0, "xengnttab_set_max_grants failed: %s\n",
124 void *xen_be_map_grant_refs(struct XenLegacyDevice
*xendev
, uint32_t *refs
,
125 unsigned int nr_refs
, int prot
)
129 assert(xendev
->ops
->flags
& DEVOPS_FLAG_NEED_GNTDEV
);
131 ptr
= xengnttab_map_domain_grant_refs(xendev
->gnttabdev
, nr_refs
,
132 xen_domid
, refs
, prot
);
134 xen_pv_printf(xendev
, 0,
135 "xengnttab_map_domain_grant_refs failed: %s\n",
142 void xen_be_unmap_grant_refs(struct XenLegacyDevice
*xendev
, void *ptr
,
143 unsigned int nr_refs
)
145 assert(xendev
->ops
->flags
& DEVOPS_FLAG_NEED_GNTDEV
);
147 if (xengnttab_unmap(xendev
->gnttabdev
, ptr
, nr_refs
)) {
148 xen_pv_printf(xendev
, 0, "xengnttab_unmap failed: %s\n",
153 static int compat_copy_grant_refs(struct XenLegacyDevice
*xendev
,
155 XenGrantCopySegment segs
[],
156 unsigned int nr_segs
)
158 uint32_t *refs
= g_new(uint32_t, nr_segs
);
159 int prot
= to_domain
? PROT_WRITE
: PROT_READ
;
163 for (i
= 0; i
< nr_segs
; i
++) {
164 XenGrantCopySegment
*seg
= &segs
[i
];
166 refs
[i
] = to_domain
?
167 seg
->dest
.foreign
.ref
: seg
->source
.foreign
.ref
;
170 pages
= xengnttab_map_domain_grant_refs(xendev
->gnttabdev
, nr_segs
,
171 xen_domid
, refs
, prot
);
173 xen_pv_printf(xendev
, 0,
174 "xengnttab_map_domain_grant_refs failed: %s\n",
180 for (i
= 0; i
< nr_segs
; i
++) {
181 XenGrantCopySegment
*seg
= &segs
[i
];
182 void *page
= pages
+ (i
* XC_PAGE_SIZE
);
185 memcpy(page
+ seg
->dest
.foreign
.offset
, seg
->source
.virt
,
188 memcpy(seg
->dest
.virt
, page
+ seg
->source
.foreign
.offset
,
193 if (xengnttab_unmap(xendev
->gnttabdev
, pages
, nr_segs
)) {
194 xen_pv_printf(xendev
, 0, "xengnttab_unmap failed: %s\n",
202 int xen_be_copy_grant_refs(struct XenLegacyDevice
*xendev
,
204 XenGrantCopySegment segs
[],
205 unsigned int nr_segs
)
207 xengnttab_grant_copy_segment_t
*xengnttab_segs
;
211 assert(xendev
->ops
->flags
& DEVOPS_FLAG_NEED_GNTDEV
);
213 if (!xen_feature_grant_copy
) {
214 return compat_copy_grant_refs(xendev
, to_domain
, segs
, nr_segs
);
217 xengnttab_segs
= g_new0(xengnttab_grant_copy_segment_t
, nr_segs
);
219 for (i
= 0; i
< nr_segs
; i
++) {
220 XenGrantCopySegment
*seg
= &segs
[i
];
221 xengnttab_grant_copy_segment_t
*xengnttab_seg
= &xengnttab_segs
[i
];
224 xengnttab_seg
->flags
= GNTCOPY_dest_gref
;
225 xengnttab_seg
->dest
.foreign
.domid
= xen_domid
;
226 xengnttab_seg
->dest
.foreign
.ref
= seg
->dest
.foreign
.ref
;
227 xengnttab_seg
->dest
.foreign
.offset
= seg
->dest
.foreign
.offset
;
228 xengnttab_seg
->source
.virt
= seg
->source
.virt
;
230 xengnttab_seg
->flags
= GNTCOPY_source_gref
;
231 xengnttab_seg
->source
.foreign
.domid
= xen_domid
;
232 xengnttab_seg
->source
.foreign
.ref
= seg
->source
.foreign
.ref
;
233 xengnttab_seg
->source
.foreign
.offset
=
234 seg
->source
.foreign
.offset
;
235 xengnttab_seg
->dest
.virt
= seg
->dest
.virt
;
238 xengnttab_seg
->len
= seg
->len
;
241 rc
= xengnttab_grant_copy(xendev
->gnttabdev
, nr_segs
, xengnttab_segs
);
244 xen_pv_printf(xendev
, 0, "xengnttab_copy failed: %s\n",
248 for (i
= 0; i
< nr_segs
; i
++) {
249 xengnttab_grant_copy_segment_t
*xengnttab_seg
=
252 if (xengnttab_seg
->status
!= GNTST_okay
) {
253 xen_pv_printf(xendev
, 0, "segment[%u] status: %d\n", i
,
254 xengnttab_seg
->status
);
259 g_free(xengnttab_segs
);
264 * get xen backend device, allocate a new one if it doesn't exist.
266 static struct XenLegacyDevice
*xen_be_get_xendev(const char *type
, int dom
,
268 struct XenDevOps
*ops
)
270 struct XenLegacyDevice
*xendev
;
272 xendev
= xen_pv_find_xendev(type
, dom
, dev
);
277 /* init new xendev */
278 xendev
= g_malloc0(ops
->size
);
279 object_initialize(&xendev
->qdev
, ops
->size
, TYPE_XENBACKEND
);
280 OBJECT(xendev
)->free
= g_free
;
281 qdev_set_parent_bus(DEVICE(xendev
), xen_sysbus
);
282 qdev_set_id(DEVICE(xendev
), g_strdup_printf("xen-%s-%d", type
, dev
));
283 qdev_init_nofail(DEVICE(xendev
));
284 object_unref(OBJECT(xendev
));
291 snprintf(xendev
->be
, sizeof(xendev
->be
), "backend/%s/%d/%d",
292 xendev
->type
, xendev
->dom
, xendev
->dev
);
293 snprintf(xendev
->name
, sizeof(xendev
->name
), "%s-%d",
294 xendev
->type
, xendev
->dev
);
296 xendev
->debug
= debug
;
297 xendev
->local_port
= -1;
299 xendev
->evtchndev
= xenevtchn_open(NULL
, 0);
300 if (xendev
->evtchndev
== NULL
) {
301 xen_pv_printf(NULL
, 0, "can't open evtchn device\n");
302 qdev_unplug(DEVICE(xendev
), NULL
);
305 qemu_set_cloexec(xenevtchn_fd(xendev
->evtchndev
));
307 xen_pv_insert_xendev(xendev
);
309 if (xendev
->ops
->alloc
) {
310 xendev
->ops
->alloc(xendev
);
318 * Sync internal data structures on xenstore updates.
319 * Node specifies the changed field. node = NULL means
320 * update all fields (used for initialization).
322 static void xen_be_backend_changed(struct XenLegacyDevice
*xendev
,
325 if (node
== NULL
|| strcmp(node
, "online") == 0) {
326 if (xenstore_read_be_int(xendev
, "online", &xendev
->online
) == -1) {
332 xen_pv_printf(xendev
, 2, "backend update: %s\n", node
);
333 if (xendev
->ops
->backend_changed
) {
334 xendev
->ops
->backend_changed(xendev
, node
);
339 static void xen_be_frontend_changed(struct XenLegacyDevice
*xendev
,
344 if (node
== NULL
|| strcmp(node
, "state") == 0) {
345 if (xenstore_read_fe_int(xendev
, "state", &fe_state
) == -1) {
346 fe_state
= XenbusStateUnknown
;
348 if (xendev
->fe_state
!= fe_state
) {
349 xen_pv_printf(xendev
, 1, "frontend state: %s -> %s\n",
350 xenbus_strstate(xendev
->fe_state
),
351 xenbus_strstate(fe_state
));
353 xendev
->fe_state
= fe_state
;
355 if (node
== NULL
|| strcmp(node
, "protocol") == 0) {
356 g_free(xendev
->protocol
);
357 xendev
->protocol
= xenstore_read_fe_str(xendev
, "protocol");
358 if (xendev
->protocol
) {
359 xen_pv_printf(xendev
, 1, "frontend protocol: %s\n",
365 xen_pv_printf(xendev
, 2, "frontend update: %s\n", node
);
366 if (xendev
->ops
->frontend_changed
) {
367 xendev
->ops
->frontend_changed(xendev
, node
);
372 /* ------------------------------------------------------------- */
373 /* Check for possible state transitions and perform them. */
376 * Initial xendev setup. Read frontend path, register watch for it.
377 * Should succeed once xend finished setting up the backend device.
379 * Also sets initial state (-> Initializing) when done. Which
380 * only affects the xendev->be_state variable as xenbus should
381 * already be put into that state by xend.
383 static int xen_be_try_setup(struct XenLegacyDevice
*xendev
)
385 char token
[XEN_BUFSIZE
];
388 if (xenstore_read_be_int(xendev
, "state", &be_state
) == -1) {
389 xen_pv_printf(xendev
, 0, "reading backend state failed\n");
393 if (be_state
!= XenbusStateInitialising
) {
394 xen_pv_printf(xendev
, 0, "initial backend state is wrong (%s)\n",
395 xenbus_strstate(be_state
));
399 xendev
->fe
= xenstore_read_be_str(xendev
, "frontend");
400 if (xendev
->fe
== NULL
) {
401 xen_pv_printf(xendev
, 0, "reading frontend path failed\n");
405 /* setup frontend watch */
406 snprintf(token
, sizeof(token
), "fe:%p", xendev
);
407 if (!xs_watch(xenstore
, xendev
->fe
, token
)) {
408 xen_pv_printf(xendev
, 0, "watching frontend path (%s) failed\n",
412 xen_be_set_state(xendev
, XenbusStateInitialising
);
414 xen_be_backend_changed(xendev
, NULL
);
415 xen_be_frontend_changed(xendev
, NULL
);
420 * Try initialize xendev. Prepare everything the backend can do
421 * without synchronizing with the frontend. Fakes hotplug-status. No
422 * hotplug involved here because this is about userspace drivers, thus
423 * there are kernel backend devices which could invoke hotplug.
425 * Goes to InitWait on success.
427 static int xen_be_try_init(struct XenLegacyDevice
*xendev
)
431 if (!xendev
->online
) {
432 xen_pv_printf(xendev
, 1, "not online\n");
436 if (xendev
->ops
->init
) {
437 rc
= xendev
->ops
->init(xendev
);
440 xen_pv_printf(xendev
, 1, "init() failed\n");
444 xenstore_write_be_str(xendev
, "hotplug-status", "connected");
445 xen_be_set_state(xendev
, XenbusStateInitWait
);
450 * Try to initialise xendev. Depends on the frontend being ready
451 * for it (shared ring and evtchn info in xenstore, state being
452 * Initialised or Connected).
454 * Goes to Connected on success.
456 static int xen_be_try_initialise(struct XenLegacyDevice
*xendev
)
460 if (xendev
->fe_state
!= XenbusStateInitialised
&&
461 xendev
->fe_state
!= XenbusStateConnected
) {
462 if (xendev
->ops
->flags
& DEVOPS_FLAG_IGNORE_STATE
) {
463 xen_pv_printf(xendev
, 2, "frontend not ready, ignoring\n");
465 xen_pv_printf(xendev
, 2, "frontend not ready (yet)\n");
470 if (xendev
->ops
->flags
& DEVOPS_FLAG_NEED_GNTDEV
) {
471 xendev
->gnttabdev
= xengnttab_open(NULL
, 0);
472 if (xendev
->gnttabdev
== NULL
) {
473 xen_pv_printf(NULL
, 0, "can't open gnttab device\n");
477 xendev
->gnttabdev
= NULL
;
480 if (xendev
->ops
->initialise
) {
481 rc
= xendev
->ops
->initialise(xendev
);
484 xen_pv_printf(xendev
, 0, "initialise() failed\n");
488 xen_be_set_state(xendev
, XenbusStateConnected
);
493 * Try to let xendev know that it is connected. Depends on the
494 * frontend being Connected. Note that this may be called more
495 * than once since the backend state is not modified.
497 static void xen_be_try_connected(struct XenLegacyDevice
*xendev
)
499 if (!xendev
->ops
->connected
) {
503 if (xendev
->fe_state
!= XenbusStateConnected
) {
504 if (xendev
->ops
->flags
& DEVOPS_FLAG_IGNORE_STATE
) {
505 xen_pv_printf(xendev
, 2, "frontend not ready, ignoring\n");
507 xen_pv_printf(xendev
, 2, "frontend not ready (yet)\n");
512 xendev
->ops
->connected(xendev
);
516 * Teardown connection.
518 * Goes to Closed when done.
520 static void xen_be_disconnect(struct XenLegacyDevice
*xendev
,
521 enum xenbus_state state
)
523 if (xendev
->be_state
!= XenbusStateClosing
&&
524 xendev
->be_state
!= XenbusStateClosed
&&
525 xendev
->ops
->disconnect
) {
526 xendev
->ops
->disconnect(xendev
);
528 if (xendev
->gnttabdev
) {
529 xengnttab_close(xendev
->gnttabdev
);
530 xendev
->gnttabdev
= NULL
;
532 if (xendev
->be_state
!= state
) {
533 xen_be_set_state(xendev
, state
);
538 * Try to reset xendev, for reconnection by another frontend instance.
540 static int xen_be_try_reset(struct XenLegacyDevice
*xendev
)
542 if (xendev
->fe_state
!= XenbusStateInitialising
) {
546 xen_pv_printf(xendev
, 1, "device reset (for re-connect)\n");
547 xen_be_set_state(xendev
, XenbusStateInitialising
);
552 * state change dispatcher function
554 void xen_be_check_state(struct XenLegacyDevice
*xendev
)
558 /* frontend may request shutdown from almost anywhere */
559 if (xendev
->fe_state
== XenbusStateClosing
||
560 xendev
->fe_state
== XenbusStateClosed
) {
561 xen_be_disconnect(xendev
, xendev
->fe_state
);
565 /* check for possible backend state transitions */
567 switch (xendev
->be_state
) {
568 case XenbusStateUnknown
:
569 rc
= xen_be_try_setup(xendev
);
571 case XenbusStateInitialising
:
572 rc
= xen_be_try_init(xendev
);
574 case XenbusStateInitWait
:
575 rc
= xen_be_try_initialise(xendev
);
577 case XenbusStateConnected
:
578 /* xendev->be_state doesn't change */
579 xen_be_try_connected(xendev
);
582 case XenbusStateClosed
:
583 rc
= xen_be_try_reset(xendev
);
594 /* ------------------------------------------------------------- */
596 static int xenstore_scan(const char *type
, int dom
, struct XenDevOps
*ops
)
598 struct XenLegacyDevice
*xendev
;
599 char path
[XEN_BUFSIZE
], token
[XEN_BUFSIZE
];
601 unsigned int cdev
, j
;
604 snprintf(token
, sizeof(token
), "be:%p:%d:%p", type
, dom
, ops
);
605 snprintf(path
, sizeof(path
), "backend/%s/%d", type
, dom
);
606 if (!xs_watch(xenstore
, path
, token
)) {
607 xen_pv_printf(NULL
, 0, "xen be: watching backend path (%s) failed\n",
612 /* look for backends */
613 dev
= xs_directory(xenstore
, 0, path
, &cdev
);
617 for (j
= 0; j
< cdev
; j
++) {
618 xendev
= xen_be_get_xendev(type
, dom
, atoi(dev
[j
]), ops
);
619 if (xendev
== NULL
) {
622 xen_be_check_state(xendev
);
628 void xenstore_update_be(char *watch
, char *type
, int dom
,
629 struct XenDevOps
*ops
)
631 struct XenLegacyDevice
*xendev
;
632 char path
[XEN_BUFSIZE
], *bepath
;
633 unsigned int len
, dev
;
635 len
= snprintf(path
, sizeof(path
), "backend/%s/%d", type
, dom
);
636 if (strncmp(path
, watch
, len
) != 0) {
639 if (sscanf(watch
+ len
, "/%u/%255s", &dev
, path
) != 2) {
641 if (sscanf(watch
+ len
, "/%u", &dev
) != 1) {
649 xendev
= xen_be_get_xendev(type
, dom
, dev
, ops
);
650 if (xendev
!= NULL
) {
651 bepath
= xs_read(xenstore
, 0, xendev
->be
, &len
);
652 if (bepath
== NULL
) {
653 xen_pv_del_xendev(xendev
);
656 xen_be_backend_changed(xendev
, path
);
657 xen_be_check_state(xendev
);
662 void xenstore_update_fe(char *watch
, struct XenLegacyDevice
*xendev
)
667 len
= strlen(xendev
->fe
);
668 if (strncmp(xendev
->fe
, watch
, len
) != 0) {
671 if (watch
[len
] != '/') {
674 node
= watch
+ len
+ 1;
676 xen_be_frontend_changed(xendev
, node
);
677 xen_be_check_state(xendev
);
679 /* -------------------------------------------------------------------- */
681 int xen_be_init(void)
683 xengnttab_handle
*gnttabdev
;
685 xenstore
= xs_daemon_open();
687 xen_pv_printf(NULL
, 0, "can't connect to xenstored\n");
691 qemu_set_fd_handler(xs_fileno(xenstore
), xenstore_update
, NULL
, NULL
);
693 if (xen_xc
== NULL
|| xen_fmem
== NULL
) {
694 /* Check if xen_init() have been called */
698 gnttabdev
= xengnttab_open(NULL
, 0);
699 if (gnttabdev
!= NULL
) {
700 if (xengnttab_grant_copy(gnttabdev
, 0, NULL
) == 0) {
701 xen_feature_grant_copy
= true;
703 xengnttab_close(gnttabdev
);
706 xen_sysdev
= qdev_create(NULL
, TYPE_XENSYSDEV
);
707 qdev_init_nofail(xen_sysdev
);
708 xen_sysbus
= qbus_create(TYPE_XENSYSBUS
, DEVICE(xen_sysdev
), "xen-sysbus");
709 qbus_set_bus_hotplug_handler(xen_sysbus
, &error_abort
);
714 qemu_set_fd_handler(xs_fileno(xenstore
), NULL
, NULL
, NULL
);
715 xs_daemon_close(xenstore
);
721 static void xen_set_dynamic_sysbus(void)
723 Object
*machine
= qdev_get_machine();
724 ObjectClass
*oc
= object_get_class(machine
);
725 MachineClass
*mc
= MACHINE_CLASS(oc
);
727 machine_class_allow_dynamic_sysbus_dev(mc
, TYPE_XENSYSDEV
);
730 int xen_be_register(const char *type
, struct XenDevOps
*ops
)
735 if (ops
->backend_register
) {
736 rc
= ops
->backend_register();
742 snprintf(path
, sizeof(path
), "device-model/%u/backends/%s", xen_domid
,
744 xenstore_mkdir(path
, XS_PERM_NONE
);
746 return xenstore_scan(type
, xen_domid
, ops
);
749 void xen_be_register_common(void)
751 xen_set_dynamic_sysbus();
753 xen_be_register("console", &xen_console_ops
);
754 xen_be_register("vkbd", &xen_kbdmouse_ops
);
756 xen_be_register("9pfs", &xen_9pfs_ops
);
758 #ifdef CONFIG_USB_LIBUSB
759 xen_be_register("qusb", &xen_usb_ops
);
763 int xen_be_bind_evtchn(struct XenLegacyDevice
*xendev
)
765 if (xendev
->local_port
!= -1) {
768 xendev
->local_port
= xenevtchn_bind_interdomain
769 (xendev
->evtchndev
, xendev
->dom
, xendev
->remote_port
);
770 if (xendev
->local_port
== -1) {
771 xen_pv_printf(xendev
, 0, "xenevtchn_bind_interdomain failed\n");
774 xen_pv_printf(xendev
, 2, "bind evtchn port %d\n", xendev
->local_port
);
775 qemu_set_fd_handler(xenevtchn_fd(xendev
->evtchndev
),
776 xen_pv_evtchn_event
, NULL
, xendev
);
781 static Property xendev_properties
[] = {
782 DEFINE_PROP_END_OF_LIST(),
785 static void xendev_class_init(ObjectClass
*klass
, void *data
)
787 DeviceClass
*dc
= DEVICE_CLASS(klass
);
789 device_class_set_props(dc
, xendev_properties
);
790 set_bit(DEVICE_CATEGORY_MISC
, dc
->categories
);
791 /* xen-backend devices can be plugged/unplugged dynamically */
792 dc
->user_creatable
= true;
795 static const TypeInfo xendev_type_info
= {
796 .name
= TYPE_XENBACKEND
,
797 .parent
= TYPE_XENSYSDEV
,
798 .class_init
= xendev_class_init
,
799 .instance_size
= sizeof(struct XenLegacyDevice
),
802 static void xen_sysbus_class_init(ObjectClass
*klass
, void *data
)
804 HotplugHandlerClass
*hc
= HOTPLUG_HANDLER_CLASS(klass
);
806 hc
->unplug
= qdev_simple_device_unplug_cb
;
809 static const TypeInfo xensysbus_info
= {
810 .name
= TYPE_XENSYSBUS
,
812 .class_init
= xen_sysbus_class_init
,
813 .interfaces
= (InterfaceInfo
[]) {
814 { TYPE_HOTPLUG_HANDLER
},
819 static Property xen_sysdev_properties
[] = {
820 {/* end of property list */},
823 static void xen_sysdev_class_init(ObjectClass
*klass
, void *data
)
825 DeviceClass
*dc
= DEVICE_CLASS(klass
);
827 device_class_set_props(dc
, xen_sysdev_properties
);
828 dc
->bus_type
= TYPE_XENSYSBUS
;
831 static const TypeInfo xensysdev_info
= {
832 .name
= TYPE_XENSYSDEV
,
833 .parent
= TYPE_SYS_BUS_DEVICE
,
834 .instance_size
= sizeof(SysBusDevice
),
835 .class_init
= xen_sysdev_class_init
,
838 static void xenbe_register_types(void)
840 type_register_static(&xensysbus_info
);
841 type_register_static(&xensysdev_info
);
842 type_register_static(&xendev_type_info
);
845 type_init(xenbe_register_types
)