xref: /illumos-gate/usr/src/uts/common/xen/io/xpvd.c (revision 97869ac5121a555decdf3080d02beef2af506da6)
1843e1988Sjohnlev /*
2843e1988Sjohnlev  * CDDL HEADER START
3843e1988Sjohnlev  *
4843e1988Sjohnlev  * The contents of this file are subject to the terms of the
5843e1988Sjohnlev  * Common Development and Distribution License (the "License").
6843e1988Sjohnlev  * You may not use this file except in compliance with the License.
7843e1988Sjohnlev  *
8843e1988Sjohnlev  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9843e1988Sjohnlev  * or http://www.opensolaris.org/os/licensing.
10843e1988Sjohnlev  * See the License for the specific language governing permissions
11843e1988Sjohnlev  * and limitations under the License.
12843e1988Sjohnlev  *
13843e1988Sjohnlev  * When distributing Covered Code, include this CDDL HEADER in each
14843e1988Sjohnlev  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15843e1988Sjohnlev  * If applicable, add the following below this CDDL HEADER, with the
16843e1988Sjohnlev  * fields enclosed by brackets "[]" replaced with your own identifying
17843e1988Sjohnlev  * information: Portions Copyright [yyyy] [name of copyright owner]
18843e1988Sjohnlev  *
19843e1988Sjohnlev  * CDDL HEADER END
20843e1988Sjohnlev  */
21843e1988Sjohnlev 
22843e1988Sjohnlev /*
2306bbe1e0Sedp  * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
24843e1988Sjohnlev  * Use is subject to license terms.
25843e1988Sjohnlev  */
26843e1988Sjohnlev 
27843e1988Sjohnlev #pragma ident	"%Z%%M%	%I%	%E% SMI"
28843e1988Sjohnlev 
29843e1988Sjohnlev /*
30843e1988Sjohnlev  *	Host to hypervisor virtual devices nexus driver
31843e1988Sjohnlev  *
32843e1988Sjohnlev  * TODO:
33843e1988Sjohnlev  * - Add watchpoints on vbd/vif and enumerate/offline on watch callback
34843e1988Sjohnlev  * - Add DR IOCTLs
35843e1988Sjohnlev  * - Filter/restrict property lookups into xenstore
36843e1988Sjohnlev  */
37843e1988Sjohnlev 
38843e1988Sjohnlev #include <sys/conf.h>
39843e1988Sjohnlev #include <sys/kmem.h>
40843e1988Sjohnlev #include <sys/debug.h>
41843e1988Sjohnlev #include <sys/modctl.h>
42843e1988Sjohnlev #include <sys/autoconf.h>
43843e1988Sjohnlev #include <sys/ddi_impldefs.h>
44843e1988Sjohnlev #include <sys/ddi_subrdefs.h>
45843e1988Sjohnlev #include <sys/ddi.h>
46843e1988Sjohnlev #include <sys/sunddi.h>
47843e1988Sjohnlev #include <sys/sunndi.h>
48843e1988Sjohnlev #include <sys/avintr.h>
49843e1988Sjohnlev #include <sys/psm.h>
50843e1988Sjohnlev #include <sys/spl.h>
51843e1988Sjohnlev #include <sys/promif.h>
52843e1988Sjohnlev #include <sys/list.h>
53843e1988Sjohnlev #include <sys/bootconf.h>
54843e1988Sjohnlev #include <sys/bootsvcs.h>
55843e1988Sjohnlev #include <util/sscanf.h>
56551bc2a6Smrj #include <sys/mach_intr.h>
57551bc2a6Smrj #include <sys/bootinfo.h>
58551bc2a6Smrj #ifdef XPV_HVM_DRIVER
59551bc2a6Smrj #include <sys/xpv_support.h>
60551bc2a6Smrj #include <sys/hypervisor.h>
61551bc2a6Smrj #include <sys/archsystm.h>
62551bc2a6Smrj #include <sys/cpu.h>
63551bc2a6Smrj #include <public/xen.h>
64551bc2a6Smrj #include <public/event_channel.h>
65551bc2a6Smrj #include <public/io/xenbus.h>
66551bc2a6Smrj #else
67551bc2a6Smrj #include <sys/hypervisor.h>
68551bc2a6Smrj #include <sys/evtchn_impl.h>
69551bc2a6Smrj #include <sys/xen_mmu.h>
70551bc2a6Smrj #endif
71843e1988Sjohnlev #include <xen/sys/xenbus_impl.h>
72843e1988Sjohnlev #include <xen/sys/xendev.h>
73843e1988Sjohnlev 
74843e1988Sjohnlev /*
75843e1988Sjohnlev  * DDI dev_ops entrypoints
76843e1988Sjohnlev  */
77843e1988Sjohnlev static int xpvd_info(dev_info_t *, ddi_info_cmd_t, void *, void **);
78843e1988Sjohnlev static int xpvd_attach(dev_info_t *devi, ddi_attach_cmd_t cmd);
79843e1988Sjohnlev static int xpvd_detach(dev_info_t *devi, ddi_detach_cmd_t cmd);
80843e1988Sjohnlev 
81843e1988Sjohnlev 
82843e1988Sjohnlev /*
83843e1988Sjohnlev  * NDI bus_ops entrypoints
84843e1988Sjohnlev  */
85843e1988Sjohnlev static int xpvd_ctlops(dev_info_t *, dev_info_t *, ddi_ctl_enum_t, void *,
86843e1988Sjohnlev 	void *);
87843e1988Sjohnlev static int xpvd_intr_ops(dev_info_t *, dev_info_t *, ddi_intr_op_t,
88843e1988Sjohnlev 	ddi_intr_handle_impl_t *, void *);
89843e1988Sjohnlev static int xpvd_prop_op(dev_t, dev_info_t *, dev_info_t *, ddi_prop_op_t,
90843e1988Sjohnlev 	int, char *, caddr_t, int *);
91843e1988Sjohnlev static int xpvd_bus_config(dev_info_t *, uint_t, ddi_bus_config_op_t,
92843e1988Sjohnlev 	void *, dev_info_t **);
93843e1988Sjohnlev static int xpvd_bus_unconfig(dev_info_t *, uint_t, ddi_bus_config_op_t,
94843e1988Sjohnlev     void *);
95843e1988Sjohnlev static int xpvd_get_eventcookie(dev_info_t *, dev_info_t *,
96843e1988Sjohnlev     char *, ddi_eventcookie_t *);
97843e1988Sjohnlev static int xpvd_add_eventcall(dev_info_t *, dev_info_t *,
98843e1988Sjohnlev     ddi_eventcookie_t, void (*)(dev_info_t *,
99843e1988Sjohnlev     ddi_eventcookie_t, void *, void *),
100843e1988Sjohnlev     void *, ddi_callback_id_t *);
101843e1988Sjohnlev static int xpvd_remove_eventcall(dev_info_t *, ddi_callback_id_t);
102843e1988Sjohnlev static int xpvd_post_event(dev_info_t *, dev_info_t *,
103843e1988Sjohnlev     ddi_eventcookie_t, void *);
104843e1988Sjohnlev 
105843e1988Sjohnlev /*
106843e1988Sjohnlev  * misc functions
107843e1988Sjohnlev  */
108843e1988Sjohnlev static int xpvd_enable_intr(dev_info_t *, ddi_intr_handle_impl_t *, int);
109843e1988Sjohnlev static void xpvd_disable_intr(dev_info_t *, ddi_intr_handle_impl_t *, int);
110843e1988Sjohnlev static int xpvd_removechild(dev_info_t *);
111843e1988Sjohnlev static int xpvd_initchild(dev_info_t *);
112843e1988Sjohnlev static int xpvd_name_child(dev_info_t *, char *, int);
113843e1988Sjohnlev static boolean_t i_xpvd_parse_devname(char *, xendev_devclass_t *,
114843e1988Sjohnlev     domid_t *, int *);
115843e1988Sjohnlev 
116843e1988Sjohnlev 
117843e1988Sjohnlev /* Extern declarations */
118843e1988Sjohnlev extern int (*psm_intr_ops)(dev_info_t *, ddi_intr_handle_impl_t *,
119843e1988Sjohnlev     psm_intr_op_t, int *);
120843e1988Sjohnlev 
121843e1988Sjohnlev struct bus_ops xpvd_bus_ops = {
122843e1988Sjohnlev 	BUSO_REV,
123843e1988Sjohnlev 	i_ddi_bus_map,
124843e1988Sjohnlev 	NULL,
125843e1988Sjohnlev 	NULL,
126843e1988Sjohnlev 	NULL,
127843e1988Sjohnlev 	i_ddi_map_fault,
128843e1988Sjohnlev 	ddi_dma_map,
129843e1988Sjohnlev 	ddi_dma_allochdl,
130843e1988Sjohnlev 	ddi_dma_freehdl,
131843e1988Sjohnlev 	ddi_dma_bindhdl,
132843e1988Sjohnlev 	ddi_dma_unbindhdl,
133843e1988Sjohnlev 	ddi_dma_flush,
134843e1988Sjohnlev 	ddi_dma_win,
135843e1988Sjohnlev 	ddi_dma_mctl,
136843e1988Sjohnlev 	xpvd_ctlops,
137843e1988Sjohnlev 	xpvd_prop_op,
138843e1988Sjohnlev 	xpvd_get_eventcookie,
139843e1988Sjohnlev 	xpvd_add_eventcall,
140843e1988Sjohnlev 	xpvd_remove_eventcall,
141843e1988Sjohnlev 	xpvd_post_event,
142843e1988Sjohnlev 	0,		/* (*bus_intr_ctl)(); */
143843e1988Sjohnlev 	xpvd_bus_config,
144843e1988Sjohnlev 	xpvd_bus_unconfig,
145843e1988Sjohnlev 	NULL,		/* (*bus_fm_init)(); */
146843e1988Sjohnlev 	NULL,		/* (*bus_fm_fini)(); */
147843e1988Sjohnlev 	NULL,		/* (*bus_fm_access_enter)(); */
148843e1988Sjohnlev 	NULL,		/* (*bus_fm_access_exit)(); */
149843e1988Sjohnlev 	NULL,		/* (*bus_power)(); */
150843e1988Sjohnlev 	xpvd_intr_ops	/* (*bus_intr_op)(); */
151843e1988Sjohnlev };
152843e1988Sjohnlev 
153843e1988Sjohnlev struct dev_ops xpvd_ops = {
154843e1988Sjohnlev 	DEVO_REV,		/* devo_rev */
155843e1988Sjohnlev 	0,			/* refcnt  */
156843e1988Sjohnlev 	xpvd_info,		/* info */
157843e1988Sjohnlev 	nulldev,		/* identify */
158843e1988Sjohnlev 	nulldev,		/* probe */
159843e1988Sjohnlev 	xpvd_attach,		/* attach */
160843e1988Sjohnlev 	xpvd_detach,		/* detach */
161843e1988Sjohnlev 	nulldev,		/* reset */
162843e1988Sjohnlev 	(struct cb_ops *)0,	/* driver operations */
163843e1988Sjohnlev 	&xpvd_bus_ops		/* bus operations */
164843e1988Sjohnlev };
165843e1988Sjohnlev 
166843e1988Sjohnlev 
167843e1988Sjohnlev dev_info_t *xpvd_dip;
168843e1988Sjohnlev 
169843e1988Sjohnlev #define	CF_DBG		0x1
170843e1988Sjohnlev #define	ALL_DBG		0xff
171843e1988Sjohnlev 
172843e1988Sjohnlev static ndi_event_definition_t xpvd_ndi_event_defs[] = {
173843e1988Sjohnlev 	{ 0, XS_OE_STATE, EPL_KERNEL, NDI_EVENT_POST_TO_TGT },
174843e1988Sjohnlev 	{ 1, XS_HP_STATE, EPL_KERNEL, NDI_EVENT_POST_TO_TGT },
175843e1988Sjohnlev };
176843e1988Sjohnlev 
177843e1988Sjohnlev #define	XENDEV_N_NDI_EVENTS \
178843e1988Sjohnlev 	(sizeof (xpvd_ndi_event_defs) / sizeof (xpvd_ndi_event_defs[0]))
179843e1988Sjohnlev 
180843e1988Sjohnlev static ndi_event_set_t xpvd_ndi_events = {
181843e1988Sjohnlev 	NDI_EVENTS_REV1, XENDEV_N_NDI_EVENTS, xpvd_ndi_event_defs
182843e1988Sjohnlev };
183843e1988Sjohnlev 
184843e1988Sjohnlev static ndi_event_hdl_t xpvd_ndi_event_handle;
185843e1988Sjohnlev 
186843e1988Sjohnlev /*
187843e1988Sjohnlev  * Hypervisor interrupt capabilities
188843e1988Sjohnlev  */
189843e1988Sjohnlev #define	XENDEV_INTR_CAPABILITIES \
190843e1988Sjohnlev 	(DDI_INTR_FLAG_EDGE | DDI_INTR_FLAG_MASKABLE | DDI_INTR_FLAG_PENDING)
191843e1988Sjohnlev 
192843e1988Sjohnlev /*
193843e1988Sjohnlev  * Module linkage information for the kernel.
194843e1988Sjohnlev  */
195843e1988Sjohnlev 
196843e1988Sjohnlev static struct modldrv modldrv = {
197843e1988Sjohnlev 	&mod_driverops, /* Type of module */
198843e1988Sjohnlev 	"virtual device nexus driver %I%",
199843e1988Sjohnlev 	&xpvd_ops,	/* driver ops */
200843e1988Sjohnlev };
201843e1988Sjohnlev 
202843e1988Sjohnlev static struct modlinkage modlinkage = {
203843e1988Sjohnlev 	MODREV_1,
204843e1988Sjohnlev 	(void *)&modldrv,
205843e1988Sjohnlev 	NULL
206843e1988Sjohnlev };
207843e1988Sjohnlev 
208843e1988Sjohnlev int
209843e1988Sjohnlev _init(void)
210843e1988Sjohnlev {
211843e1988Sjohnlev 	return (mod_install(&modlinkage));
212843e1988Sjohnlev }
213843e1988Sjohnlev 
214843e1988Sjohnlev int
215843e1988Sjohnlev _fini(void)
216843e1988Sjohnlev {
217843e1988Sjohnlev 	return (mod_remove(&modlinkage));
218843e1988Sjohnlev }
219843e1988Sjohnlev 
220843e1988Sjohnlev int
221843e1988Sjohnlev _info(struct modinfo *modinfop)
222843e1988Sjohnlev {
223843e1988Sjohnlev 	return (mod_info(&modlinkage, modinfop));
224843e1988Sjohnlev }
225843e1988Sjohnlev 
226843e1988Sjohnlev /* ARGSUSED */
227843e1988Sjohnlev static int
228843e1988Sjohnlev xpvd_info(dev_info_t *dip, ddi_info_cmd_t cmd, void *arg, void **result)
229843e1988Sjohnlev {
230843e1988Sjohnlev 	switch (cmd) {
231843e1988Sjohnlev 	default:
232843e1988Sjohnlev 		return (DDI_FAILURE);
233843e1988Sjohnlev 
234843e1988Sjohnlev 	case DDI_INFO_DEVT2INSTANCE:
235843e1988Sjohnlev 		*result = (void *)0;
236843e1988Sjohnlev 		return (DDI_SUCCESS);
237843e1988Sjohnlev 
238843e1988Sjohnlev 	case DDI_INFO_DEVT2DEVINFO:
239843e1988Sjohnlev 		*result = (void *)xpvd_dip;
240843e1988Sjohnlev 		return (DDI_SUCCESS);
241843e1988Sjohnlev 	}
242843e1988Sjohnlev }
243843e1988Sjohnlev 
244843e1988Sjohnlev /*ARGSUSED*/
245843e1988Sjohnlev static int
246843e1988Sjohnlev xpvd_attach(dev_info_t *devi, ddi_attach_cmd_t cmd)
247843e1988Sjohnlev {
248843e1988Sjohnlev 	extern void xvdi_watch_devices(int);
249551bc2a6Smrj 
250551bc2a6Smrj #ifdef XPV_HVM_DRIVER
251551bc2a6Smrj 	if (xen_info == NULL) {
252551bc2a6Smrj 		if (ddi_hold_installed_driver(ddi_name_to_major("xpv")) ==
253551bc2a6Smrj 		    NULL) {
254551bc2a6Smrj 			cmn_err(CE_WARN, "Couldn't initialize xpv framework");
255551bc2a6Smrj 			return (DDI_FAILURE);
256551bc2a6Smrj 		}
257551bc2a6Smrj 	}
25806bbe1e0Sedp #endif /* XPV_HVM_DRIVER */
259843e1988Sjohnlev 
260843e1988Sjohnlev 	if (ndi_event_alloc_hdl(devi, 0, &xpvd_ndi_event_handle,
261843e1988Sjohnlev 	    NDI_SLEEP) != NDI_SUCCESS) {
262843e1988Sjohnlev 		xpvd_dip = NULL;
263843e1988Sjohnlev 		return (DDI_FAILURE);
264843e1988Sjohnlev 	}
265843e1988Sjohnlev 	if (ndi_event_bind_set(xpvd_ndi_event_handle, &xpvd_ndi_events,
266843e1988Sjohnlev 	    NDI_SLEEP) != NDI_SUCCESS) {
267843e1988Sjohnlev 		(void) ndi_event_free_hdl(xpvd_ndi_event_handle);
268843e1988Sjohnlev 		xpvd_dip = NULL;
269843e1988Sjohnlev 		return (DDI_FAILURE);
270843e1988Sjohnlev 	}
271843e1988Sjohnlev 
27206bbe1e0Sedp #ifdef XPV_HVM_DRIVER
27306bbe1e0Sedp 	(void) ddi_prop_update_int(DDI_DEV_T_NONE, devi, DDI_NO_AUTODETACH, 1);
2746eb35ee7Srab 
2756eb35ee7Srab 	/*
2766eb35ee7Srab 	 * Report our version to dom0.
2776eb35ee7Srab 	 */
2786eb35ee7Srab 	if (xenbus_printf(XBT_NULL, "hvmpv/xpvd", "version", "%d",
2796eb35ee7Srab 	    HVMPV_XPVD_VERS))
2806eb35ee7Srab 		cmn_err(CE_WARN, "xpvd: couldn't write version\n");
28106bbe1e0Sedp #endif /* XPV_HVM_DRIVER */
28206bbe1e0Sedp 
283843e1988Sjohnlev 	/* watch both frontend and backend for new devices */
284843e1988Sjohnlev 	if (DOMAIN_IS_INITDOMAIN(xen_info))
285843e1988Sjohnlev 		(void) xs_register_xenbus_callback(xvdi_watch_devices);
286843e1988Sjohnlev 	else
287843e1988Sjohnlev 		xvdi_watch_devices(XENSTORE_UP);
288843e1988Sjohnlev 
289551bc2a6Smrj 	xpvd_dip = devi;
290843e1988Sjohnlev 	ddi_report_dev(devi);
291843e1988Sjohnlev 
292843e1988Sjohnlev 	return (DDI_SUCCESS);
293843e1988Sjohnlev }
294843e1988Sjohnlev 
295843e1988Sjohnlev /*ARGSUSED*/
296843e1988Sjohnlev static int
297843e1988Sjohnlev xpvd_detach(dev_info_t *devi, ddi_detach_cmd_t cmd)
298843e1988Sjohnlev {
299843e1988Sjohnlev 	return (DDI_FAILURE);
300843e1988Sjohnlev }
301843e1988Sjohnlev 
302843e1988Sjohnlev /*
303843e1988Sjohnlev  * xpvd_prop_op()
304843e1988Sjohnlev  *
305843e1988Sjohnlev  * Query xenstore for the value of properties if DDI_PROP_NOTPROM
306843e1988Sjohnlev  * is not set.  Xenstore property values are represented as ascii strings.
307843e1988Sjohnlev  */
308843e1988Sjohnlev static int
309843e1988Sjohnlev xpvd_prop_op(dev_t dev, dev_info_t *dip, dev_info_t *ch_dip,
310843e1988Sjohnlev     ddi_prop_op_t prop_op, int mod_flags, char *name, caddr_t valuep,
311843e1988Sjohnlev     int *lengthp)
312843e1988Sjohnlev {
313843e1988Sjohnlev 	caddr_t buff;
314843e1988Sjohnlev 	struct xendev_ppd *pdp;
315843e1988Sjohnlev 	void *prop_str;
316843e1988Sjohnlev 	size_t prop_len;
317843e1988Sjohnlev 	unsigned int len;
318843e1988Sjohnlev 	int rv;
319843e1988Sjohnlev 
320843e1988Sjohnlev 	pdp = (struct xendev_ppd *)ddi_get_parent_data(ch_dip);
321843e1988Sjohnlev 
322843e1988Sjohnlev 	if ((pdp == NULL) || !(mod_flags & (DDI_PROP_CANSLEEP)) ||
323843e1988Sjohnlev 	    (mod_flags & DDI_PROP_NOTPROM) || (pdp->xd_xsdev.nodename == NULL))
324843e1988Sjohnlev 		goto toss_off;
325843e1988Sjohnlev 	/*
326843e1988Sjohnlev 	 * First try reading the property off the the frontend. if that
327843e1988Sjohnlev 	 * fails, try and read it from the backend node.  If that
328843e1988Sjohnlev 	 * also fails, pass the request on the DDI framework
329843e1988Sjohnlev 	 */
330843e1988Sjohnlev 	prop_str = NULL;
331843e1988Sjohnlev 	if ((xenbus_read(XBT_NULL, pdp->xd_xsdev.nodename, name, &prop_str,
332843e1988Sjohnlev 	    &len) == 0) && (prop_str != NULL) && (strlen(prop_str) != 0))
333843e1988Sjohnlev 		goto got_xs_prop;
334843e1988Sjohnlev 
335843e1988Sjohnlev 	prop_str = NULL;
336843e1988Sjohnlev 	if ((pdp->xd_xsdev.otherend != NULL) &&
337843e1988Sjohnlev 	    (xenbus_read(XBT_NULL, pdp->xd_xsdev.otherend, name, &prop_str,
338843e1988Sjohnlev 	    &len) == 0) && (prop_str != NULL) && (strlen(prop_str) != 0))
339843e1988Sjohnlev 		goto got_xs_prop;
340843e1988Sjohnlev 
341843e1988Sjohnlev toss_off:
342843e1988Sjohnlev 	return (ddi_bus_prop_op(dev, dip, ch_dip, prop_op,
343843e1988Sjohnlev 	    mod_flags | DDI_PROP_NOTPROM, name, valuep, lengthp));
344843e1988Sjohnlev 
345843e1988Sjohnlev got_xs_prop:
346843e1988Sjohnlev 	prop_len = strlen(prop_str) + 1;
347843e1988Sjohnlev 	rv = DDI_PROP_SUCCESS;
348843e1988Sjohnlev 
349843e1988Sjohnlev 	switch (prop_op) {
350843e1988Sjohnlev 	case PROP_LEN:
351843e1988Sjohnlev 		*lengthp = prop_len;
352843e1988Sjohnlev 		break;
353843e1988Sjohnlev 
354843e1988Sjohnlev 	case PROP_LEN_AND_VAL_ALLOC:
355843e1988Sjohnlev 		buff = kmem_alloc((size_t)prop_len, KM_SLEEP);
356843e1988Sjohnlev 		*(caddr_t *)valuep = (caddr_t)buff;
357843e1988Sjohnlev 		break;
358843e1988Sjohnlev 	case PROP_LEN_AND_VAL_BUF:
359843e1988Sjohnlev 		buff = (caddr_t)valuep;
360843e1988Sjohnlev 		if (*lengthp < prop_len)
361843e1988Sjohnlev 			rv = DDI_PROP_BUF_TOO_SMALL;
362843e1988Sjohnlev 		break;
363843e1988Sjohnlev 	default:
364843e1988Sjohnlev 		rv = DDI_PROP_INVAL_ARG;
365843e1988Sjohnlev 		break;
366843e1988Sjohnlev 	}
367843e1988Sjohnlev 
368843e1988Sjohnlev 	if ((rv == DDI_PROP_SUCCESS) && (prop_len > 0)) {
369843e1988Sjohnlev 		bcopy(prop_str, buff, prop_len);
370843e1988Sjohnlev 		*lengthp = prop_len;
371843e1988Sjohnlev 	}
372843e1988Sjohnlev 	kmem_free(prop_str, len);
373843e1988Sjohnlev 	return (rv);
374843e1988Sjohnlev }
375843e1988Sjohnlev 
376843e1988Sjohnlev 
377843e1988Sjohnlev /*
378843e1988Sjohnlev  * return address of the device's interrupt spec structure.
379843e1988Sjohnlev  */
380843e1988Sjohnlev /*ARGSUSED*/
381843e1988Sjohnlev struct intrspec *
382843e1988Sjohnlev xpvd_get_ispec(dev_info_t *rdip, uint_t inumber)
383843e1988Sjohnlev {
384843e1988Sjohnlev 	struct xendev_ppd *pdp;
385843e1988Sjohnlev 
386843e1988Sjohnlev 	ASSERT(inumber == 0);
387843e1988Sjohnlev 
388843e1988Sjohnlev 	if ((pdp = ddi_get_parent_data(rdip)) == NULL)
389843e1988Sjohnlev 		return (NULL);
390843e1988Sjohnlev 
391843e1988Sjohnlev 	return (&pdp->xd_ispec);
392843e1988Sjohnlev }
393843e1988Sjohnlev 
394843e1988Sjohnlev /*
395843e1988Sjohnlev  * return (and determine) the interrupt priority of the device.
396843e1988Sjohnlev  */
397843e1988Sjohnlev /*ARGSUSED*/
398843e1988Sjohnlev static int
399843e1988Sjohnlev xpvd_get_priority(dev_info_t *dip, int inum, int *pri)
400843e1988Sjohnlev {
401843e1988Sjohnlev 	struct xendev_ppd *pdp;
402843e1988Sjohnlev 	struct intrspec *ispec;
403843e1988Sjohnlev 	int	*intpriorities;
404843e1988Sjohnlev 	uint_t	num_intpriorities;
405843e1988Sjohnlev 
406843e1988Sjohnlev 	DDI_INTR_NEXDBG((CE_CONT, "xpvd_get_priority: dip = 0x%p\n",
407843e1988Sjohnlev 	    (void *)dip));
408843e1988Sjohnlev 
409843e1988Sjohnlev 	ASSERT(inum == 0);
410843e1988Sjohnlev 
411843e1988Sjohnlev 	if ((pdp = ddi_get_parent_data(dip)) == NULL)
412843e1988Sjohnlev 		return (DDI_FAILURE);
413843e1988Sjohnlev 
414843e1988Sjohnlev 	ispec = &pdp->xd_ispec;
415843e1988Sjohnlev 
416843e1988Sjohnlev 	/*
417843e1988Sjohnlev 	 * Set the default priority based on the device class.  The
418843e1988Sjohnlev 	 * "interrupt-priorities" property can be used to override
419843e1988Sjohnlev 	 * the default.
420843e1988Sjohnlev 	 */
421843e1988Sjohnlev 	if (ispec->intrspec_pri == 0) {
422843e1988Sjohnlev 		ispec->intrspec_pri = xendev_devclass_ipl(pdp->xd_devclass);
423843e1988Sjohnlev 		if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY, dip,
424843e1988Sjohnlev 		    DDI_PROP_NOTPROM | DDI_PROP_DONTPASS,
425843e1988Sjohnlev 		    "interrupt-priorities", &intpriorities,
426843e1988Sjohnlev 		    &num_intpriorities) == DDI_PROP_SUCCESS) {
427843e1988Sjohnlev 			ispec->intrspec_pri = intpriorities[0];
428843e1988Sjohnlev 			ddi_prop_free(intpriorities);
429843e1988Sjohnlev 		}
430843e1988Sjohnlev 	}
431843e1988Sjohnlev 	*pri = ispec->intrspec_pri;
432843e1988Sjohnlev 	return (DDI_SUCCESS);
433843e1988Sjohnlev }
434843e1988Sjohnlev 
435843e1988Sjohnlev 
436843e1988Sjohnlev /*
437843e1988Sjohnlev  * xpvd_intr_ops: bus_intr_op() function for interrupt support
438843e1988Sjohnlev  */
439843e1988Sjohnlev /* ARGSUSED */
440843e1988Sjohnlev static int
441843e1988Sjohnlev xpvd_intr_ops(dev_info_t *pdip, dev_info_t *rdip, ddi_intr_op_t intr_op,
442843e1988Sjohnlev     ddi_intr_handle_impl_t *hdlp, void *result)
443843e1988Sjohnlev {
444843e1988Sjohnlev 	int priority = 0;
445843e1988Sjohnlev 	struct intrspec *ispec;
446843e1988Sjohnlev 	struct xendev_ppd *pdp;
447843e1988Sjohnlev 
448843e1988Sjohnlev 	DDI_INTR_NEXDBG((CE_CONT,
449843e1988Sjohnlev 	    "xpvd_intr_ops: pdip 0x%p, rdip 0x%p, op %x handle 0x%p\n",
450843e1988Sjohnlev 	    (void *)pdip, (void *)rdip, intr_op, (void *)hdlp));
451843e1988Sjohnlev 
452843e1988Sjohnlev 	/* Process the request */
453843e1988Sjohnlev 	switch (intr_op) {
454843e1988Sjohnlev 	case DDI_INTROP_SUPPORTED_TYPES:
455843e1988Sjohnlev 		/* Fixed supported by default */
456843e1988Sjohnlev 		*(int *)result = DDI_INTR_TYPE_FIXED;
457843e1988Sjohnlev 		break;
458843e1988Sjohnlev 
459843e1988Sjohnlev 	case DDI_INTROP_NINTRS:
460843e1988Sjohnlev 		*(int *)result = 1;
461843e1988Sjohnlev 		break;
462843e1988Sjohnlev 
463843e1988Sjohnlev 	case DDI_INTROP_ALLOC:
464843e1988Sjohnlev 		/*
465843e1988Sjohnlev 		 * FIXED interrupts: just return available interrupts
466843e1988Sjohnlev 		 */
467843e1988Sjohnlev 		if (hdlp->ih_type == DDI_INTR_TYPE_FIXED) {
468843e1988Sjohnlev 			/*
469843e1988Sjohnlev 			 * event channels are edge-triggered, maskable,
470843e1988Sjohnlev 			 * and support int pending.
471843e1988Sjohnlev 			 */
472843e1988Sjohnlev 			hdlp->ih_cap |= XENDEV_INTR_CAPABILITIES;
473843e1988Sjohnlev 			*(int *)result = 1;	/* DDI_INTR_TYPE_FIXED */
474843e1988Sjohnlev 		} else {
475843e1988Sjohnlev 			return (DDI_FAILURE);
476843e1988Sjohnlev 		}
477843e1988Sjohnlev 		break;
478843e1988Sjohnlev 
479843e1988Sjohnlev 	case DDI_INTROP_FREE:
480843e1988Sjohnlev 		ispec = xpvd_get_ispec(rdip, (int)hdlp->ih_inum);
481843e1988Sjohnlev 		if (ispec == NULL)
482843e1988Sjohnlev 			return (DDI_FAILURE);
483843e1988Sjohnlev 		ispec->intrspec_pri = 0; /* mark as un-initialized */
484843e1988Sjohnlev 		break;
485843e1988Sjohnlev 
486843e1988Sjohnlev 	case DDI_INTROP_GETPRI:
487843e1988Sjohnlev 		if (xpvd_get_priority(rdip, hdlp->ih_inum, &priority) !=
488843e1988Sjohnlev 		    DDI_SUCCESS)
489843e1988Sjohnlev 			return (DDI_FAILURE);
490843e1988Sjohnlev 		DDI_INTR_NEXDBG((CE_CONT, "xpvd_intr_ops: priority = 0x%x\n",
491843e1988Sjohnlev 		    priority));
492843e1988Sjohnlev 		*(int *)result = priority;
493843e1988Sjohnlev 		break;
494843e1988Sjohnlev 
495843e1988Sjohnlev 	case DDI_INTROP_SETPRI:
496843e1988Sjohnlev 		/* Validate the interrupt priority passed */
497843e1988Sjohnlev 		if (*(int *)result > LOCK_LEVEL)
498843e1988Sjohnlev 			return (DDI_FAILURE);
499843e1988Sjohnlev 
500843e1988Sjohnlev 		/* Ensure that PSM is all initialized */
501843e1988Sjohnlev 		if (psm_intr_ops == NULL)
502843e1988Sjohnlev 			return (DDI_FAILURE);
503843e1988Sjohnlev 
504843e1988Sjohnlev 		/* Change the priority */
505843e1988Sjohnlev 		if ((*psm_intr_ops)(rdip, hdlp, PSM_INTR_OP_SET_PRI, result) ==
506843e1988Sjohnlev 		    PSM_FAILURE)
507843e1988Sjohnlev 			return (DDI_FAILURE);
508843e1988Sjohnlev 
509843e1988Sjohnlev 		ispec = xpvd_get_ispec(rdip, (int)hdlp->ih_inum);
510843e1988Sjohnlev 		if (ispec == NULL)
511843e1988Sjohnlev 			return (DDI_FAILURE);
512843e1988Sjohnlev 		ispec->intrspec_pri = *(int *)result;
513843e1988Sjohnlev 		break;
514843e1988Sjohnlev 
515843e1988Sjohnlev 	case DDI_INTROP_ADDISR:
516843e1988Sjohnlev 		/* update ispec */
517843e1988Sjohnlev 		ispec = xpvd_get_ispec(rdip, (int)hdlp->ih_inum);
518843e1988Sjohnlev 		if (ispec == NULL)
519843e1988Sjohnlev 			return (DDI_FAILURE);
520843e1988Sjohnlev 		ispec->intrspec_func = hdlp->ih_cb_func;
521843e1988Sjohnlev 
522843e1988Sjohnlev 		break;
523843e1988Sjohnlev 
524843e1988Sjohnlev 	case DDI_INTROP_REMISR:
525843e1988Sjohnlev 		ispec = xpvd_get_ispec(rdip, (int)hdlp->ih_inum);
526843e1988Sjohnlev 		pdp = (struct xendev_ppd *)ddi_get_parent_data(rdip);
527843e1988Sjohnlev 
528843e1988Sjohnlev 		ASSERT(pdp != NULL);
529843e1988Sjohnlev 		ASSERT(pdp->xd_evtchn != INVALID_EVTCHN);
530843e1988Sjohnlev 
531843e1988Sjohnlev 		if (ispec) {
532843e1988Sjohnlev 			ispec->intrspec_vec = 0;
533843e1988Sjohnlev 			ispec->intrspec_func = (uint_t (*)()) 0;
534843e1988Sjohnlev 		}
535843e1988Sjohnlev 		pdp->xd_evtchn = INVALID_EVTCHN;
536843e1988Sjohnlev 		break;
537843e1988Sjohnlev 
538843e1988Sjohnlev 	case DDI_INTROP_GETCAP:
539843e1988Sjohnlev 		if (hdlp->ih_type ==  DDI_INTR_TYPE_FIXED) {
540843e1988Sjohnlev 			/*
541843e1988Sjohnlev 			 * event channels are edge-triggered, maskable,
542843e1988Sjohnlev 			 * and support int pending.
543843e1988Sjohnlev 			 */
544843e1988Sjohnlev 			*(int *)result = XENDEV_INTR_CAPABILITIES;
545843e1988Sjohnlev 		} else {
546843e1988Sjohnlev 			*(int *)result = 0;
547843e1988Sjohnlev 			return (DDI_FAILURE);
548843e1988Sjohnlev 		}
549843e1988Sjohnlev 		DDI_INTR_NEXDBG((CE_CONT, "xpvd: GETCAP returned = %x\n",
550843e1988Sjohnlev 		    *(int *)result));
551843e1988Sjohnlev 		break;
552843e1988Sjohnlev 	case DDI_INTROP_SETCAP:
553843e1988Sjohnlev 		DDI_INTR_NEXDBG((CE_CONT, "xpvd_intr_ops: SETCAP cap=0x%x\n",
554843e1988Sjohnlev 		    *(int *)result));
555843e1988Sjohnlev 		if (psm_intr_ops == NULL)
556843e1988Sjohnlev 			return (DDI_FAILURE);
557843e1988Sjohnlev 
558843e1988Sjohnlev 		if ((*psm_intr_ops)(rdip, hdlp, PSM_INTR_OP_SET_CAP, result)) {
559843e1988Sjohnlev 			DDI_INTR_NEXDBG((CE_CONT, "GETCAP: psm_intr_ops"
560843e1988Sjohnlev 			    " returned failure\n"));
561843e1988Sjohnlev 			return (DDI_FAILURE);
562843e1988Sjohnlev 		}
563843e1988Sjohnlev 		break;
564843e1988Sjohnlev 
565843e1988Sjohnlev 	case DDI_INTROP_ENABLE:
566843e1988Sjohnlev 		if (psm_intr_ops == NULL)
567843e1988Sjohnlev 			return (DDI_FAILURE);
568843e1988Sjohnlev 
569843e1988Sjohnlev 		if (xpvd_enable_intr(rdip, hdlp, (int)hdlp->ih_inum) !=
570843e1988Sjohnlev 		    DDI_SUCCESS)
571843e1988Sjohnlev 			return (DDI_FAILURE);
572843e1988Sjohnlev 
573843e1988Sjohnlev 		DDI_INTR_NEXDBG((CE_CONT, "xpvd_intr_ops: ENABLE vec=0x%x\n",
574843e1988Sjohnlev 		    hdlp->ih_vector));
575843e1988Sjohnlev 		break;
576843e1988Sjohnlev 
577843e1988Sjohnlev 	case DDI_INTROP_DISABLE:
578843e1988Sjohnlev 		if (psm_intr_ops == NULL)
579843e1988Sjohnlev 			return (DDI_FAILURE);
580843e1988Sjohnlev 		xpvd_disable_intr(rdip, hdlp, hdlp->ih_inum);
581843e1988Sjohnlev 		DDI_INTR_NEXDBG((CE_CONT, "xpvd_intr_ops: DISABLE vec = %x\n",
582843e1988Sjohnlev 		    hdlp->ih_vector));
583843e1988Sjohnlev 		break;
584843e1988Sjohnlev 
585843e1988Sjohnlev 	case DDI_INTROP_BLOCKENABLE:
586843e1988Sjohnlev 	case DDI_INTROP_BLOCKDISABLE:
587843e1988Sjohnlev 		return (DDI_FAILURE);
588843e1988Sjohnlev 
589843e1988Sjohnlev 	case DDI_INTROP_SETMASK:
590843e1988Sjohnlev 	case DDI_INTROP_CLRMASK:
591551bc2a6Smrj #ifdef XPV_HVM_DRIVER
592551bc2a6Smrj 		return (DDI_ENOTSUP);
593551bc2a6Smrj #else
594843e1988Sjohnlev 		/*
595843e1988Sjohnlev 		 * Handle this here
596843e1988Sjohnlev 		 */
597843e1988Sjohnlev 		if (hdlp->ih_type != DDI_INTR_TYPE_FIXED)
598843e1988Sjohnlev 			return (DDI_FAILURE);
599843e1988Sjohnlev 		if (intr_op == DDI_INTROP_SETMASK) {
600843e1988Sjohnlev 			ec_disable_irq(hdlp->ih_vector);
601843e1988Sjohnlev 		} else {
602843e1988Sjohnlev 			ec_enable_irq(hdlp->ih_vector);
603843e1988Sjohnlev 		}
604843e1988Sjohnlev 		break;
605551bc2a6Smrj #endif
606843e1988Sjohnlev 	case DDI_INTROP_GETPENDING:
607551bc2a6Smrj #ifdef XPV_HVM_DRIVER
608551bc2a6Smrj 		return (DDI_ENOTSUP);
609551bc2a6Smrj #else
610843e1988Sjohnlev 		if (hdlp->ih_type != DDI_INTR_TYPE_FIXED)
611843e1988Sjohnlev 			return (DDI_FAILURE);
612843e1988Sjohnlev 		*(int *)result = ec_pending_irq(hdlp->ih_vector);
613843e1988Sjohnlev 		DDI_INTR_NEXDBG((CE_CONT, "xpvd: GETPENDING returned = %x\n",
614843e1988Sjohnlev 		    *(int *)result));
615843e1988Sjohnlev 		break;
616551bc2a6Smrj #endif
617843e1988Sjohnlev 
618843e1988Sjohnlev 	case DDI_INTROP_NAVAIL:
619843e1988Sjohnlev 		*(int *)result = 1;
620843e1988Sjohnlev 		DDI_INTR_NEXDBG((CE_CONT, "xpvd: NAVAIL returned = %x\n",
621843e1988Sjohnlev 		    *(int *)result));
622843e1988Sjohnlev 		break;
623843e1988Sjohnlev 
624843e1988Sjohnlev 	default:
625843e1988Sjohnlev 		return (i_ddi_intr_ops(pdip, rdip, intr_op, hdlp, result));
626843e1988Sjohnlev 	}
627843e1988Sjohnlev 
628843e1988Sjohnlev 	return (DDI_SUCCESS);
629843e1988Sjohnlev }
630843e1988Sjohnlev 
631843e1988Sjohnlev 
632843e1988Sjohnlev static int
633843e1988Sjohnlev xpvd_enable_intr(dev_info_t *rdip, ddi_intr_handle_impl_t *hdlp, int inum)
634843e1988Sjohnlev {
635843e1988Sjohnlev 	int		vector;
636843e1988Sjohnlev 	ihdl_plat_t	*ihdl_plat_datap = (ihdl_plat_t *)hdlp->ih_private;
637843e1988Sjohnlev 
638843e1988Sjohnlev 	DDI_INTR_NEXDBG((CE_CONT, "xpvd_enable_intr: hdlp %p inum %x\n",
639843e1988Sjohnlev 	    (void *)hdlp, inum));
640843e1988Sjohnlev 
641843e1988Sjohnlev 	ihdl_plat_datap->ip_ispecp = xpvd_get_ispec(rdip, inum);
642843e1988Sjohnlev 	if (ihdl_plat_datap->ip_ispecp == NULL)
643843e1988Sjohnlev 		return (DDI_FAILURE);
644843e1988Sjohnlev 
645843e1988Sjohnlev 	/* translate the interrupt if needed */
646843e1988Sjohnlev 	(void) (*psm_intr_ops)(rdip, hdlp, PSM_INTR_OP_XLATE_VECTOR, &vector);
647843e1988Sjohnlev 	DDI_INTR_NEXDBG((CE_CONT, "xpvd_enable_intr: priority=%x vector=%x\n",
648843e1988Sjohnlev 	    hdlp->ih_pri, vector));
649843e1988Sjohnlev 
650843e1988Sjohnlev 	/* Add the interrupt handler */
651843e1988Sjohnlev 	if (!add_avintr((void *)hdlp, hdlp->ih_pri, hdlp->ih_cb_func,
652843e1988Sjohnlev 	    DEVI(rdip)->devi_name, vector, hdlp->ih_cb_arg1,
653843e1988Sjohnlev 	    hdlp->ih_cb_arg2, NULL, rdip))
654843e1988Sjohnlev 		return (DDI_FAILURE);
655843e1988Sjohnlev 
656843e1988Sjohnlev 	/* Note this really is an irq. */
657843e1988Sjohnlev 	hdlp->ih_vector = (ushort_t)vector;
658843e1988Sjohnlev 
659843e1988Sjohnlev 	return (DDI_SUCCESS);
660843e1988Sjohnlev }
661843e1988Sjohnlev 
662843e1988Sjohnlev 
663843e1988Sjohnlev static void
664843e1988Sjohnlev xpvd_disable_intr(dev_info_t *rdip, ddi_intr_handle_impl_t *hdlp, int inum)
665843e1988Sjohnlev {
666843e1988Sjohnlev 	int		vector;
667843e1988Sjohnlev 	ihdl_plat_t	*ihdl_plat_datap = (ihdl_plat_t *)hdlp->ih_private;
668843e1988Sjohnlev 
669843e1988Sjohnlev 	DDI_INTR_NEXDBG((CE_CONT, "xpvd_disable_intr: \n"));
670843e1988Sjohnlev 	ihdl_plat_datap->ip_ispecp = xpvd_get_ispec(rdip, inum);
671843e1988Sjohnlev 	if (ihdl_plat_datap->ip_ispecp == NULL)
672843e1988Sjohnlev 		return;
673843e1988Sjohnlev 
674843e1988Sjohnlev 	/* translate the interrupt if needed */
675843e1988Sjohnlev 	(void) (*psm_intr_ops)(rdip, hdlp, PSM_INTR_OP_XLATE_VECTOR, &vector);
676843e1988Sjohnlev 
677843e1988Sjohnlev 	/* Disable the interrupt handler */
678843e1988Sjohnlev 	rem_avintr((void *)hdlp, hdlp->ih_pri, hdlp->ih_cb_func, vector);
679843e1988Sjohnlev 	ihdl_plat_datap->ip_ispecp = NULL;
680843e1988Sjohnlev }
681843e1988Sjohnlev 
682843e1988Sjohnlev /*ARGSUSED*/
683843e1988Sjohnlev static int
684843e1988Sjohnlev xpvd_ctlops(dev_info_t *dip, dev_info_t *rdip,
685843e1988Sjohnlev 	ddi_ctl_enum_t ctlop, void *arg, void *result)
686843e1988Sjohnlev {
687843e1988Sjohnlev 	switch (ctlop) {
688843e1988Sjohnlev 	case DDI_CTLOPS_REPORTDEV:
689843e1988Sjohnlev 		if (rdip == (dev_info_t *)0)
690843e1988Sjohnlev 			return (DDI_FAILURE);
691843e1988Sjohnlev 		cmn_err(CE_CONT, "?%s@%s, %s%d\n", ddi_node_name(rdip),
692843e1988Sjohnlev 		    ddi_get_name_addr(rdip), ddi_driver_name(rdip),
693843e1988Sjohnlev 		    ddi_get_instance(rdip));
694843e1988Sjohnlev 		return (DDI_SUCCESS);
695843e1988Sjohnlev 
696843e1988Sjohnlev 	case DDI_CTLOPS_INITCHILD:
697843e1988Sjohnlev 		return (xpvd_initchild((dev_info_t *)arg));
698843e1988Sjohnlev 
699843e1988Sjohnlev 	case DDI_CTLOPS_UNINITCHILD:
700843e1988Sjohnlev 		return (xpvd_removechild((dev_info_t *)arg));
701843e1988Sjohnlev 
702843e1988Sjohnlev 	case DDI_CTLOPS_SIDDEV:
703843e1988Sjohnlev 		return (DDI_SUCCESS);
704843e1988Sjohnlev 
705843e1988Sjohnlev 	case DDI_CTLOPS_REGSIZE:
706843e1988Sjohnlev 	case DDI_CTLOPS_NREGS:
707843e1988Sjohnlev 		return (DDI_FAILURE);
708843e1988Sjohnlev 
709843e1988Sjohnlev 	case DDI_CTLOPS_POWER: {
710843e1988Sjohnlev 		return (ddi_ctlops(dip, rdip, ctlop, arg, result));
711843e1988Sjohnlev 	}
712843e1988Sjohnlev 
713843e1988Sjohnlev 	default:
714843e1988Sjohnlev 		return (ddi_ctlops(dip, rdip, ctlop, arg, result));
715843e1988Sjohnlev 	}
716843e1988Sjohnlev 
717843e1988Sjohnlev 	/* NOTREACHED */
718843e1988Sjohnlev 
719843e1988Sjohnlev }
720843e1988Sjohnlev 
721843e1988Sjohnlev /*
722843e1988Sjohnlev  * Assign the address portion of the node name
723843e1988Sjohnlev  */
724843e1988Sjohnlev static int
72506bbe1e0Sedp xpvd_name_child(dev_info_t *child, char *addr, int addrlen)
726843e1988Sjohnlev {
727843e1988Sjohnlev 	int *domain, *vdev;
728843e1988Sjohnlev 	uint_t ndomain, nvdev;
72906bbe1e0Sedp 	char *prop_str;
730843e1988Sjohnlev 
731843e1988Sjohnlev 	/*
732843e1988Sjohnlev 	 * i_xpvd_parse_devname() knows the formats used by this
733843e1988Sjohnlev 	 * routine.  If this code changes, so must that.
734843e1988Sjohnlev 	 */
735843e1988Sjohnlev 
736843e1988Sjohnlev 	if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY, child, DDI_PROP_DONTPASS,
737843e1988Sjohnlev 	    "domain", &domain, &ndomain) != DDI_PROP_SUCCESS)
738843e1988Sjohnlev 		return (DDI_FAILURE);
739843e1988Sjohnlev 	ASSERT(ndomain == 1);
740843e1988Sjohnlev 
741843e1988Sjohnlev 	/*
742843e1988Sjohnlev 	 * Use "domain" and "vdev" properties (backend drivers).
743843e1988Sjohnlev 	 */
744843e1988Sjohnlev 	if (*domain != DOMID_SELF) {
745843e1988Sjohnlev 		if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY, child,
746843e1988Sjohnlev 		    DDI_PROP_DONTPASS, "vdev", &vdev, &nvdev)
747843e1988Sjohnlev 		    != DDI_PROP_SUCCESS) {
748843e1988Sjohnlev 			ddi_prop_free(domain);
749843e1988Sjohnlev 			return (DDI_FAILURE);
750843e1988Sjohnlev 		}
751843e1988Sjohnlev 		ASSERT(nvdev == 1);
752843e1988Sjohnlev 
75306bbe1e0Sedp 		(void) snprintf(addr, addrlen, "%d,%d", domain[0], vdev[0]);
754843e1988Sjohnlev 		ddi_prop_free(vdev);
755843e1988Sjohnlev 		ddi_prop_free(domain);
756843e1988Sjohnlev 		return (DDI_SUCCESS);
757843e1988Sjohnlev 	}
758843e1988Sjohnlev 	ddi_prop_free(domain);
759843e1988Sjohnlev 
760843e1988Sjohnlev 	/*
761843e1988Sjohnlev 	 * Use "unit-address" property (frontend/softdev drivers).
762843e1988Sjohnlev 	 */
76306bbe1e0Sedp 	if (ddi_prop_lookup_string(DDI_DEV_T_ANY, child, DDI_PROP_DONTPASS,
76406bbe1e0Sedp 	    "unit-address", &prop_str) != DDI_PROP_SUCCESS)
765551bc2a6Smrj 		return (DDI_FAILURE);
76606bbe1e0Sedp 	(void) strlcpy(addr, prop_str, addrlen);
76706bbe1e0Sedp 	ddi_prop_free(prop_str);
768843e1988Sjohnlev 	return (DDI_SUCCESS);
769843e1988Sjohnlev }
770843e1988Sjohnlev 
771843e1988Sjohnlev static int
772843e1988Sjohnlev xpvd_initchild(dev_info_t *child)
773843e1988Sjohnlev {
77406bbe1e0Sedp 	char addr[80];
775843e1988Sjohnlev 
776843e1988Sjohnlev 	/*
777843e1988Sjohnlev 	 * Pseudo nodes indicate a prototype node with per-instance
778843e1988Sjohnlev 	 * properties to be merged into the real h/w device node.
779843e1988Sjohnlev 	 */
780843e1988Sjohnlev 	if (ndi_dev_is_persistent_node(child) == 0) {
781843e1988Sjohnlev 		ddi_set_parent_data(child, NULL);
782843e1988Sjohnlev 
783843e1988Sjohnlev 		/*
784843e1988Sjohnlev 		 * Try to merge the properties from this prototype
785843e1988Sjohnlev 		 * node into real h/w nodes.
786843e1988Sjohnlev 		 */
787843e1988Sjohnlev 		if (ndi_merge_node(child, xpvd_name_child) == DDI_SUCCESS) {
788843e1988Sjohnlev 			/*
789843e1988Sjohnlev 			 * Merged ok - return failure to remove the node.
790843e1988Sjohnlev 			 */
791843e1988Sjohnlev 			ddi_set_name_addr(child, NULL);
792843e1988Sjohnlev 			return (DDI_FAILURE);
793843e1988Sjohnlev 		}
794843e1988Sjohnlev 
795843e1988Sjohnlev 		/*
796843e1988Sjohnlev 		 * The child was not merged into a h/w node,
797843e1988Sjohnlev 		 * but there's not much we can do with it other
798843e1988Sjohnlev 		 * than return failure to cause the node to be removed.
799843e1988Sjohnlev 		 */
800843e1988Sjohnlev 		cmn_err(CE_WARN, "!%s@%s: %s.conf properties not merged",
801843e1988Sjohnlev 		    ddi_get_name(child), ddi_get_name_addr(child),
802843e1988Sjohnlev 		    ddi_get_name(child));
803843e1988Sjohnlev 		ddi_set_name_addr(child, NULL);
804843e1988Sjohnlev 		return (DDI_NOT_WELL_FORMED);
805843e1988Sjohnlev 	}
806843e1988Sjohnlev 
807843e1988Sjohnlev 	if (xvdi_init_dev(child) != DDI_SUCCESS)
808843e1988Sjohnlev 		return (DDI_FAILURE);
809843e1988Sjohnlev 
81006bbe1e0Sedp 	if (xpvd_name_child(child, addr, sizeof (addr)) != DDI_SUCCESS) {
811843e1988Sjohnlev 		xvdi_uninit_dev(child);
812843e1988Sjohnlev 		return (DDI_FAILURE);
813843e1988Sjohnlev 	}
81406bbe1e0Sedp 	ddi_set_name_addr(child, addr);
815843e1988Sjohnlev 
816843e1988Sjohnlev 	return (DDI_SUCCESS);
817843e1988Sjohnlev }
818843e1988Sjohnlev 
819843e1988Sjohnlev static int
820843e1988Sjohnlev xpvd_removechild(dev_info_t *dip)
821843e1988Sjohnlev {
822843e1988Sjohnlev 	xvdi_uninit_dev(dip);
823843e1988Sjohnlev 
824843e1988Sjohnlev 	ddi_set_name_addr(dip, NULL);
825843e1988Sjohnlev 
826843e1988Sjohnlev 	/*
827843e1988Sjohnlev 	 * Strip the node to properly convert it back to prototype
828843e1988Sjohnlev 	 * form.
829843e1988Sjohnlev 	 */
830843e1988Sjohnlev 	ddi_remove_minor_node(dip, NULL);
831843e1988Sjohnlev 
832843e1988Sjohnlev 	return (DDI_SUCCESS);
833843e1988Sjohnlev }
834843e1988Sjohnlev 
835843e1988Sjohnlev static int
836843e1988Sjohnlev xpvd_bus_unconfig(dev_info_t *parent, uint_t flag, ddi_bus_config_op_t op,
837843e1988Sjohnlev     void *device_name)
838843e1988Sjohnlev {
839843e1988Sjohnlev 	return (ndi_busop_bus_unconfig(parent, flag, op, device_name));
840843e1988Sjohnlev }
841843e1988Sjohnlev 
842843e1988Sjohnlev /*
843843e1988Sjohnlev  * Given the name of a child of xpvd, determine the device class,
844843e1988Sjohnlev  * domain and vdevnum to which it refers.
845843e1988Sjohnlev  */
846843e1988Sjohnlev static boolean_t
847843e1988Sjohnlev i_xpvd_parse_devname(char *name, xendev_devclass_t *devclassp,
848843e1988Sjohnlev     domid_t *domp, int *vdevp)
849843e1988Sjohnlev {
850843e1988Sjohnlev 	int len = strlen(name) + 1;
851843e1988Sjohnlev 	char *device_name = i_ddi_strdup(name, KM_SLEEP);
852843e1988Sjohnlev 	char *cname = NULL, *caddr = NULL;
853843e1988Sjohnlev 	boolean_t ret;
854843e1988Sjohnlev 
855843e1988Sjohnlev 	i_ddi_parse_name(device_name, &cname, &caddr, NULL);
856843e1988Sjohnlev 
857843e1988Sjohnlev 	if ((cname == NULL) || (strlen(cname) == 0) ||
858843e1988Sjohnlev 	    (caddr == NULL) || (strlen(caddr) == 0)) {
859843e1988Sjohnlev 		ret = B_FALSE;
860843e1988Sjohnlev 		goto done;
861843e1988Sjohnlev 	}
862843e1988Sjohnlev 
863843e1988Sjohnlev 	*devclassp = xendev_nodename_to_devclass(cname);
864843e1988Sjohnlev 	if (*devclassp < 0) {
865843e1988Sjohnlev 		ret = B_FALSE;
866843e1988Sjohnlev 		goto done;
867843e1988Sjohnlev 	}
868843e1988Sjohnlev 
869843e1988Sjohnlev 	/*
870843e1988Sjohnlev 	 * Parsing the address component requires knowledge of how
871843e1988Sjohnlev 	 * xpvd_name_child() works.  If that code changes, so must
872843e1988Sjohnlev 	 * this.
873843e1988Sjohnlev 	 */
874843e1988Sjohnlev 
875843e1988Sjohnlev 	/* Backend format is "<domain>,<vdev>". */
876843e1988Sjohnlev 	if (sscanf(caddr, "%d,%d", domp, vdevp) == 2) {
877843e1988Sjohnlev 		ret = B_TRUE;
878843e1988Sjohnlev 		goto done;
879843e1988Sjohnlev 	}
880843e1988Sjohnlev 
881843e1988Sjohnlev 	/* Frontend format is "<vdev>". */
882843e1988Sjohnlev 	*domp = DOMID_SELF;
883*97869ac5Sjhd 	if (sscanf(caddr, "%d", vdevp) == 1)
884843e1988Sjohnlev 		ret = B_TRUE;
885843e1988Sjohnlev done:
886843e1988Sjohnlev 	kmem_free(device_name, len);
887843e1988Sjohnlev 	return (ret);
888843e1988Sjohnlev }
889843e1988Sjohnlev 
890843e1988Sjohnlev /*
891843e1988Sjohnlev  * xpvd_bus_config()
892843e1988Sjohnlev  *
893843e1988Sjohnlev  * BUS_CONFIG_ONE:
894843e1988Sjohnlev  *	Enumerate the exact instance of a driver.
895843e1988Sjohnlev  *
896843e1988Sjohnlev  * BUS_CONFIG_ALL:
897843e1988Sjohnlev  *	Enumerate all the instances of all the possible children (seen before
898843e1988Sjohnlev  *	and never seen before).
899843e1988Sjohnlev  *
900843e1988Sjohnlev  * BUS_CONFIG_DRIVER:
901843e1988Sjohnlev  *	Enumerate all the instances of a particular driver.
902843e1988Sjohnlev  */
903843e1988Sjohnlev static int
904843e1988Sjohnlev xpvd_bus_config(dev_info_t *parent, uint_t flag, ddi_bus_config_op_t op,
905843e1988Sjohnlev 	void *arg, dev_info_t **childp)
906843e1988Sjohnlev {
907843e1988Sjohnlev 	int circ;
908843e1988Sjohnlev 	char *cname = NULL;
909843e1988Sjohnlev 
910843e1988Sjohnlev 	ndi_devi_enter(parent, &circ);
911843e1988Sjohnlev 
912843e1988Sjohnlev 	switch (op) {
913843e1988Sjohnlev 	case BUS_CONFIG_ONE: {
914843e1988Sjohnlev 		xendev_devclass_t devclass;
915843e1988Sjohnlev 		domid_t dom;
916843e1988Sjohnlev 		int vdev;
917843e1988Sjohnlev 
918843e1988Sjohnlev 		if (!i_xpvd_parse_devname(arg, &devclass, &dom, &vdev)) {
919843e1988Sjohnlev 			ndi_devi_exit(parent, circ);
920843e1988Sjohnlev 			return (NDI_FAILURE);
921843e1988Sjohnlev 		}
922843e1988Sjohnlev 
923843e1988Sjohnlev 		*childp = xvdi_find_dev(parent, devclass, dom, vdev);
924843e1988Sjohnlev 		if (*childp == NULL)
925843e1988Sjohnlev 			*childp = xvdi_create_dev(parent, devclass, dom, vdev);
926843e1988Sjohnlev 
927843e1988Sjohnlev 		ndi_devi_exit(parent, circ);
928843e1988Sjohnlev 
929843e1988Sjohnlev 		if (*childp == NULL)
930843e1988Sjohnlev 			return (NDI_FAILURE);
931843e1988Sjohnlev 		else
932843e1988Sjohnlev 			return (ndi_busop_bus_config(parent, flag,
933843e1988Sjohnlev 			    op, arg, childp, 0));
934843e1988Sjohnlev 	}
935843e1988Sjohnlev 
936843e1988Sjohnlev 	case BUS_CONFIG_DRIVER: {
937843e1988Sjohnlev 		xendev_devclass_t devclass = XEN_INVAL;
938843e1988Sjohnlev 
939843e1988Sjohnlev 		cname = ddi_major_to_name((major_t)(uintptr_t)arg);
940843e1988Sjohnlev 		if (cname != NULL)
941843e1988Sjohnlev 			devclass = xendev_nodename_to_devclass(cname);
942843e1988Sjohnlev 
943843e1988Sjohnlev 		if (devclass == XEN_INVAL) {
944843e1988Sjohnlev 			ndi_devi_exit(parent, circ);
945843e1988Sjohnlev 			return (NDI_FAILURE);
946843e1988Sjohnlev 		} else {
947843e1988Sjohnlev 			xendev_enum_class(parent, devclass);
948843e1988Sjohnlev 			ndi_devi_exit(parent, circ);
949843e1988Sjohnlev 			return (ndi_busop_bus_config(parent, flag, op,
950843e1988Sjohnlev 			    arg, childp, 0));
951843e1988Sjohnlev 		}
952843e1988Sjohnlev 		/* NOTREACHED */
953843e1988Sjohnlev 	}
954843e1988Sjohnlev 
955843e1988Sjohnlev 	case BUS_CONFIG_ALL:
956843e1988Sjohnlev 		xendev_enum_all(parent, B_FALSE);
957843e1988Sjohnlev 		ndi_devi_exit(parent, circ);
958843e1988Sjohnlev 
959843e1988Sjohnlev 		return (ndi_busop_bus_config(parent, flag, op,
960843e1988Sjohnlev 		    arg, childp, 0));
961843e1988Sjohnlev 
962843e1988Sjohnlev 	default:
963843e1988Sjohnlev 		ndi_devi_exit(parent, circ);
964843e1988Sjohnlev 		return (NDI_FAILURE);
965843e1988Sjohnlev 	}
966843e1988Sjohnlev }
967843e1988Sjohnlev 
968843e1988Sjohnlev /*ARGSUSED*/
969843e1988Sjohnlev static int
970843e1988Sjohnlev xpvd_get_eventcookie(dev_info_t *dip, dev_info_t *rdip,
971843e1988Sjohnlev     char *eventname, ddi_eventcookie_t *cookie)
972843e1988Sjohnlev {
973843e1988Sjohnlev 	return (ndi_event_retrieve_cookie(xpvd_ndi_event_handle,
974843e1988Sjohnlev 	    rdip, eventname, cookie, NDI_EVENT_NOPASS));
975843e1988Sjohnlev }
976843e1988Sjohnlev 
977843e1988Sjohnlev /*ARGSUSED*/
978843e1988Sjohnlev static int
979843e1988Sjohnlev xpvd_add_eventcall(dev_info_t *dip, dev_info_t *rdip,
980843e1988Sjohnlev     ddi_eventcookie_t cookie, void (*callback)(dev_info_t *dip,
981843e1988Sjohnlev     ddi_eventcookie_t cookie, void *arg, void *bus_impldata),
982843e1988Sjohnlev     void *arg, ddi_callback_id_t *cb_id)
983843e1988Sjohnlev {
984843e1988Sjohnlev 	return (ndi_event_add_callback(xpvd_ndi_event_handle,
985843e1988Sjohnlev 	    rdip, cookie, callback, arg, NDI_SLEEP, cb_id));
986843e1988Sjohnlev }
987843e1988Sjohnlev 
988843e1988Sjohnlev /*ARGSUSED*/
989843e1988Sjohnlev static int
990843e1988Sjohnlev xpvd_remove_eventcall(dev_info_t *dip, ddi_callback_id_t cb_id)
991843e1988Sjohnlev {
992843e1988Sjohnlev 	return (ndi_event_remove_callback(xpvd_ndi_event_handle,
993843e1988Sjohnlev 	    cb_id));
994843e1988Sjohnlev }
995843e1988Sjohnlev 
996843e1988Sjohnlev /*ARGSUSED*/
997843e1988Sjohnlev static int
998843e1988Sjohnlev xpvd_post_event(dev_info_t *dip, dev_info_t *rdip,
999843e1988Sjohnlev     ddi_eventcookie_t cookie, void *bus_impldata)
1000843e1988Sjohnlev {
1001843e1988Sjohnlev 	return (ndi_event_run_callbacks(xpvd_ndi_event_handle, rdip,
1002843e1988Sjohnlev 	    cookie, bus_impldata));
1003843e1988Sjohnlev }
1004