xref: /onnv-gate/usr/src/uts/common/xen/io/xpvd.c (revision 7656)
15084Sjohnlev /*
25084Sjohnlev  * CDDL HEADER START
35084Sjohnlev  *
45084Sjohnlev  * The contents of this file are subject to the terms of the
55084Sjohnlev  * Common Development and Distribution License (the "License").
65084Sjohnlev  * You may not use this file except in compliance with the License.
75084Sjohnlev  *
85084Sjohnlev  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
95084Sjohnlev  * or http://www.opensolaris.org/os/licensing.
105084Sjohnlev  * See the License for the specific language governing permissions
115084Sjohnlev  * and limitations under the License.
125084Sjohnlev  *
135084Sjohnlev  * When distributing Covered Code, include this CDDL HEADER in each
145084Sjohnlev  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
155084Sjohnlev  * If applicable, add the following below this CDDL HEADER, with the
165084Sjohnlev  * fields enclosed by brackets "[]" replaced with your own identifying
175084Sjohnlev  * information: Portions Copyright [yyyy] [name of copyright owner]
185084Sjohnlev  *
195084Sjohnlev  * CDDL HEADER END
205084Sjohnlev  */
215084Sjohnlev 
225084Sjohnlev /*
236318Sedp  * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
245084Sjohnlev  * Use is subject to license terms.
255084Sjohnlev  */
265084Sjohnlev 
275084Sjohnlev /*
285084Sjohnlev  *	Host to hypervisor virtual devices nexus driver
295084Sjohnlev  *
305084Sjohnlev  * TODO:
315084Sjohnlev  * - Add watchpoints on vbd/vif and enumerate/offline on watch callback
325084Sjohnlev  * - Add DR IOCTLs
335084Sjohnlev  * - Filter/restrict property lookups into xenstore
345084Sjohnlev  */
355084Sjohnlev 
365084Sjohnlev #include <sys/conf.h>
375084Sjohnlev #include <sys/kmem.h>
385084Sjohnlev #include <sys/debug.h>
395084Sjohnlev #include <sys/modctl.h>
405084Sjohnlev #include <sys/autoconf.h>
415084Sjohnlev #include <sys/ddi_impldefs.h>
425084Sjohnlev #include <sys/ddi_subrdefs.h>
435084Sjohnlev #include <sys/ddi.h>
445084Sjohnlev #include <sys/sunddi.h>
455084Sjohnlev #include <sys/sunndi.h>
465084Sjohnlev #include <sys/avintr.h>
475084Sjohnlev #include <sys/psm.h>
485084Sjohnlev #include <sys/spl.h>
495084Sjohnlev #include <sys/promif.h>
505084Sjohnlev #include <sys/list.h>
515084Sjohnlev #include <sys/bootconf.h>
525084Sjohnlev #include <sys/bootsvcs.h>
535741Smrj #include <util/sscanf.h>
545741Smrj #include <sys/mach_intr.h>
555084Sjohnlev #include <sys/bootinfo.h>
565741Smrj #ifdef XPV_HVM_DRIVER
575741Smrj #include <sys/xpv_support.h>
585741Smrj #include <sys/hypervisor.h>
595741Smrj #include <sys/archsystm.h>
605741Smrj #include <sys/cpu.h>
615741Smrj #include <public/xen.h>
625741Smrj #include <public/event_channel.h>
635741Smrj #include <public/io/xenbus.h>
645741Smrj #else
655741Smrj #include <sys/hypervisor.h>
665741Smrj #include <sys/evtchn_impl.h>
675741Smrj #include <sys/xen_mmu.h>
685741Smrj #endif
695084Sjohnlev #include <xen/sys/xenbus_impl.h>
705084Sjohnlev #include <xen/sys/xendev.h>
715084Sjohnlev 
725084Sjohnlev /*
735084Sjohnlev  * DDI dev_ops entrypoints
745084Sjohnlev  */
755084Sjohnlev static int xpvd_info(dev_info_t *, ddi_info_cmd_t, void *, void **);
765084Sjohnlev static int xpvd_attach(dev_info_t *devi, ddi_attach_cmd_t cmd);
775084Sjohnlev static int xpvd_detach(dev_info_t *devi, ddi_detach_cmd_t cmd);
785084Sjohnlev 
795084Sjohnlev 
805084Sjohnlev /*
815084Sjohnlev  * NDI bus_ops entrypoints
825084Sjohnlev  */
835084Sjohnlev static int xpvd_ctlops(dev_info_t *, dev_info_t *, ddi_ctl_enum_t, void *,
845084Sjohnlev 	void *);
855084Sjohnlev static int xpvd_intr_ops(dev_info_t *, dev_info_t *, ddi_intr_op_t,
865084Sjohnlev 	ddi_intr_handle_impl_t *, void *);
875084Sjohnlev static int xpvd_prop_op(dev_t, dev_info_t *, dev_info_t *, ddi_prop_op_t,
885084Sjohnlev 	int, char *, caddr_t, int *);
895084Sjohnlev static int xpvd_bus_config(dev_info_t *, uint_t, ddi_bus_config_op_t,
905084Sjohnlev 	void *, dev_info_t **);
915084Sjohnlev static int xpvd_bus_unconfig(dev_info_t *, uint_t, ddi_bus_config_op_t,
925084Sjohnlev     void *);
935084Sjohnlev static int xpvd_get_eventcookie(dev_info_t *, dev_info_t *,
945084Sjohnlev     char *, ddi_eventcookie_t *);
955084Sjohnlev static int xpvd_add_eventcall(dev_info_t *, dev_info_t *,
965084Sjohnlev     ddi_eventcookie_t, void (*)(dev_info_t *,
975084Sjohnlev     ddi_eventcookie_t, void *, void *),
985084Sjohnlev     void *, ddi_callback_id_t *);
995084Sjohnlev static int xpvd_remove_eventcall(dev_info_t *, ddi_callback_id_t);
1005084Sjohnlev static int xpvd_post_event(dev_info_t *, dev_info_t *,
1015084Sjohnlev     ddi_eventcookie_t, void *);
1025084Sjohnlev 
1035084Sjohnlev /*
1045084Sjohnlev  * misc functions
1055084Sjohnlev  */
1065084Sjohnlev static int xpvd_enable_intr(dev_info_t *, ddi_intr_handle_impl_t *, int);
1075084Sjohnlev static void xpvd_disable_intr(dev_info_t *, ddi_intr_handle_impl_t *, int);
1085084Sjohnlev static int xpvd_removechild(dev_info_t *);
1095084Sjohnlev static int xpvd_initchild(dev_info_t *);
1105084Sjohnlev static int xpvd_name_child(dev_info_t *, char *, int);
1115084Sjohnlev static boolean_t i_xpvd_parse_devname(char *, xendev_devclass_t *,
1125084Sjohnlev     domid_t *, int *);
1135084Sjohnlev 
1145084Sjohnlev 
1155084Sjohnlev /* Extern declarations */
1165084Sjohnlev extern int (*psm_intr_ops)(dev_info_t *, ddi_intr_handle_impl_t *,
1175084Sjohnlev     psm_intr_op_t, int *);
1185084Sjohnlev 
1195084Sjohnlev struct bus_ops xpvd_bus_ops = {
1205084Sjohnlev 	BUSO_REV,
1215084Sjohnlev 	i_ddi_bus_map,
1225084Sjohnlev 	NULL,
1235084Sjohnlev 	NULL,
1245084Sjohnlev 	NULL,
1255084Sjohnlev 	i_ddi_map_fault,
1265084Sjohnlev 	ddi_dma_map,
1275084Sjohnlev 	ddi_dma_allochdl,
1285084Sjohnlev 	ddi_dma_freehdl,
1295084Sjohnlev 	ddi_dma_bindhdl,
1305084Sjohnlev 	ddi_dma_unbindhdl,
1315084Sjohnlev 	ddi_dma_flush,
1325084Sjohnlev 	ddi_dma_win,
1335084Sjohnlev 	ddi_dma_mctl,
1345084Sjohnlev 	xpvd_ctlops,
1355084Sjohnlev 	xpvd_prop_op,
1365084Sjohnlev 	xpvd_get_eventcookie,
1375084Sjohnlev 	xpvd_add_eventcall,
1385084Sjohnlev 	xpvd_remove_eventcall,
1395084Sjohnlev 	xpvd_post_event,
1405084Sjohnlev 	0,		/* (*bus_intr_ctl)(); */
1415084Sjohnlev 	xpvd_bus_config,
1425084Sjohnlev 	xpvd_bus_unconfig,
1435084Sjohnlev 	NULL,		/* (*bus_fm_init)(); */
1445084Sjohnlev 	NULL,		/* (*bus_fm_fini)(); */
1455084Sjohnlev 	NULL,		/* (*bus_fm_access_enter)(); */
1465084Sjohnlev 	NULL,		/* (*bus_fm_access_exit)(); */
1475084Sjohnlev 	NULL,		/* (*bus_power)(); */
1485084Sjohnlev 	xpvd_intr_ops	/* (*bus_intr_op)(); */
1495084Sjohnlev };
1505084Sjohnlev 
1515084Sjohnlev struct dev_ops xpvd_ops = {
1525084Sjohnlev 	DEVO_REV,		/* devo_rev */
1535084Sjohnlev 	0,			/* refcnt  */
1545084Sjohnlev 	xpvd_info,		/* info */
1555084Sjohnlev 	nulldev,		/* identify */
1565084Sjohnlev 	nulldev,		/* probe */
1575084Sjohnlev 	xpvd_attach,		/* attach */
1585084Sjohnlev 	xpvd_detach,		/* detach */
1595084Sjohnlev 	nulldev,		/* reset */
1605084Sjohnlev 	(struct cb_ops *)0,	/* driver operations */
161*7656SSherry.Moore@Sun.COM 	&xpvd_bus_ops,		/* bus operations */
162*7656SSherry.Moore@Sun.COM 	NULL,			/* power */
163*7656SSherry.Moore@Sun.COM 	ddi_quiesce_not_needed,		/* quiesce */
1645084Sjohnlev };
1655084Sjohnlev 
1665084Sjohnlev 
1675084Sjohnlev dev_info_t *xpvd_dip;
1685084Sjohnlev 
1695084Sjohnlev #define	CF_DBG		0x1
1705084Sjohnlev #define	ALL_DBG		0xff
1715084Sjohnlev 
1725084Sjohnlev static ndi_event_definition_t xpvd_ndi_event_defs[] = {
1735084Sjohnlev 	{ 0, XS_OE_STATE, EPL_KERNEL, NDI_EVENT_POST_TO_TGT },
1745084Sjohnlev 	{ 1, XS_HP_STATE, EPL_KERNEL, NDI_EVENT_POST_TO_TGT },
1755084Sjohnlev };
1765084Sjohnlev 
1775084Sjohnlev #define	XENDEV_N_NDI_EVENTS \
1785084Sjohnlev 	(sizeof (xpvd_ndi_event_defs) / sizeof (xpvd_ndi_event_defs[0]))
1795084Sjohnlev 
1805084Sjohnlev static ndi_event_set_t xpvd_ndi_events = {
1815084Sjohnlev 	NDI_EVENTS_REV1, XENDEV_N_NDI_EVENTS, xpvd_ndi_event_defs
1825084Sjohnlev };
1835084Sjohnlev 
1845084Sjohnlev static ndi_event_hdl_t xpvd_ndi_event_handle;
1855084Sjohnlev 
1865084Sjohnlev /*
1875084Sjohnlev  * Hypervisor interrupt capabilities
1885084Sjohnlev  */
1895084Sjohnlev #define	XENDEV_INTR_CAPABILITIES \
1905084Sjohnlev 	(DDI_INTR_FLAG_EDGE | DDI_INTR_FLAG_MASKABLE | DDI_INTR_FLAG_PENDING)
1915084Sjohnlev 
1925084Sjohnlev /*
1935084Sjohnlev  * Module linkage information for the kernel.
1945084Sjohnlev  */
1955084Sjohnlev 
1965084Sjohnlev static struct modldrv modldrv = {
1975084Sjohnlev 	&mod_driverops, /* Type of module */
1987632SNick.Todd@Sun.COM 	"virtual device nexus driver",
1995084Sjohnlev 	&xpvd_ops,	/* driver ops */
2005084Sjohnlev };
2015084Sjohnlev 
2025084Sjohnlev static struct modlinkage modlinkage = {
2035084Sjohnlev 	MODREV_1,
2045084Sjohnlev 	(void *)&modldrv,
2055084Sjohnlev 	NULL
2065084Sjohnlev };
2075084Sjohnlev 
2085084Sjohnlev int
2095084Sjohnlev _init(void)
2105084Sjohnlev {
2115084Sjohnlev 	return (mod_install(&modlinkage));
2125084Sjohnlev }
2135084Sjohnlev 
2145084Sjohnlev int
2155084Sjohnlev _fini(void)
2165084Sjohnlev {
2175084Sjohnlev 	return (mod_remove(&modlinkage));
2185084Sjohnlev }
2195084Sjohnlev 
2205084Sjohnlev int
2215084Sjohnlev _info(struct modinfo *modinfop)
2225084Sjohnlev {
2235084Sjohnlev 	return (mod_info(&modlinkage, modinfop));
2245084Sjohnlev }
2255084Sjohnlev 
2265084Sjohnlev /* ARGSUSED */
2275084Sjohnlev static int
2285084Sjohnlev xpvd_info(dev_info_t *dip, ddi_info_cmd_t cmd, void *arg, void **result)
2295084Sjohnlev {
2305084Sjohnlev 	switch (cmd) {
2315084Sjohnlev 	default:
2325084Sjohnlev 		return (DDI_FAILURE);
2335084Sjohnlev 
2345084Sjohnlev 	case DDI_INFO_DEVT2INSTANCE:
2355084Sjohnlev 		*result = (void *)0;
2365084Sjohnlev 		return (DDI_SUCCESS);
2375084Sjohnlev 
2385084Sjohnlev 	case DDI_INFO_DEVT2DEVINFO:
2395084Sjohnlev 		*result = (void *)xpvd_dip;
2405084Sjohnlev 		return (DDI_SUCCESS);
2415084Sjohnlev 	}
2425084Sjohnlev }
2435084Sjohnlev 
2445084Sjohnlev /*ARGSUSED*/
2455084Sjohnlev static int
2465084Sjohnlev xpvd_attach(dev_info_t *devi, ddi_attach_cmd_t cmd)
2475084Sjohnlev {
2485084Sjohnlev 	extern void xvdi_watch_devices(int);
2495741Smrj 
2505741Smrj #ifdef XPV_HVM_DRIVER
2515741Smrj 	if (xen_info == NULL) {
2525741Smrj 		if (ddi_hold_installed_driver(ddi_name_to_major("xpv")) ==
2535741Smrj 		    NULL) {
2545741Smrj 			cmn_err(CE_WARN, "Couldn't initialize xpv framework");
2555741Smrj 			return (DDI_FAILURE);
2565741Smrj 		}
2575741Smrj 	}
2586318Sedp #endif /* XPV_HVM_DRIVER */
2595084Sjohnlev 
2605084Sjohnlev 	if (ndi_event_alloc_hdl(devi, 0, &xpvd_ndi_event_handle,
2615084Sjohnlev 	    NDI_SLEEP) != NDI_SUCCESS) {
2625084Sjohnlev 		xpvd_dip = NULL;
2635084Sjohnlev 		return (DDI_FAILURE);
2645084Sjohnlev 	}
2655084Sjohnlev 	if (ndi_event_bind_set(xpvd_ndi_event_handle, &xpvd_ndi_events,
2665084Sjohnlev 	    NDI_SLEEP) != NDI_SUCCESS) {
2675084Sjohnlev 		(void) ndi_event_free_hdl(xpvd_ndi_event_handle);
2685084Sjohnlev 		xpvd_dip = NULL;
2695084Sjohnlev 		return (DDI_FAILURE);
2705084Sjohnlev 	}
2715084Sjohnlev 
2726318Sedp #ifdef XPV_HVM_DRIVER
2736318Sedp 	(void) ddi_prop_update_int(DDI_DEV_T_NONE, devi, DDI_NO_AUTODETACH, 1);
2746450Srab 
2756450Srab 	/*
2766450Srab 	 * Report our version to dom0.
2776450Srab 	 */
2786450Srab 	if (xenbus_printf(XBT_NULL, "hvmpv/xpvd", "version", "%d",
2796450Srab 	    HVMPV_XPVD_VERS))
2806450Srab 		cmn_err(CE_WARN, "xpvd: couldn't write version\n");
2816318Sedp #endif /* XPV_HVM_DRIVER */
2826318Sedp 
2835084Sjohnlev 	/* watch both frontend and backend for new devices */
2845084Sjohnlev 	if (DOMAIN_IS_INITDOMAIN(xen_info))
2855084Sjohnlev 		(void) xs_register_xenbus_callback(xvdi_watch_devices);
2865084Sjohnlev 	else
2875084Sjohnlev 		xvdi_watch_devices(XENSTORE_UP);
2885084Sjohnlev 
2895741Smrj 	xpvd_dip = devi;
2905084Sjohnlev 	ddi_report_dev(devi);
2915084Sjohnlev 
2925084Sjohnlev 	return (DDI_SUCCESS);
2935084Sjohnlev }
2945084Sjohnlev 
2955084Sjohnlev /*ARGSUSED*/
2965084Sjohnlev static int
2975084Sjohnlev xpvd_detach(dev_info_t *devi, ddi_detach_cmd_t cmd)
2985084Sjohnlev {
2995084Sjohnlev 	return (DDI_FAILURE);
3005084Sjohnlev }
3015084Sjohnlev 
3025084Sjohnlev /*
3035084Sjohnlev  * xpvd_prop_op()
3045084Sjohnlev  *
3055084Sjohnlev  * Query xenstore for the value of properties if DDI_PROP_NOTPROM
3065084Sjohnlev  * is not set.  Xenstore property values are represented as ascii strings.
3075084Sjohnlev  */
3085084Sjohnlev static int
3095084Sjohnlev xpvd_prop_op(dev_t dev, dev_info_t *dip, dev_info_t *ch_dip,
3105084Sjohnlev     ddi_prop_op_t prop_op, int mod_flags, char *name, caddr_t valuep,
3115084Sjohnlev     int *lengthp)
3125084Sjohnlev {
3135084Sjohnlev 	caddr_t buff;
3145084Sjohnlev 	struct xendev_ppd *pdp;
3155084Sjohnlev 	void *prop_str;
3165084Sjohnlev 	size_t prop_len;
3175084Sjohnlev 	unsigned int len;
3185084Sjohnlev 	int rv;
3195084Sjohnlev 
3205084Sjohnlev 	pdp = (struct xendev_ppd *)ddi_get_parent_data(ch_dip);
3215084Sjohnlev 
3225084Sjohnlev 	if ((pdp == NULL) || !(mod_flags & (DDI_PROP_CANSLEEP)) ||
3235084Sjohnlev 	    (mod_flags & DDI_PROP_NOTPROM) || (pdp->xd_xsdev.nodename == NULL))
3245084Sjohnlev 		goto toss_off;
3255084Sjohnlev 	/*
3265084Sjohnlev 	 * First try reading the property off the the frontend. if that
3275084Sjohnlev 	 * fails, try and read it from the backend node.  If that
3285084Sjohnlev 	 * also fails, pass the request on the DDI framework
3295084Sjohnlev 	 */
3305084Sjohnlev 	prop_str = NULL;
3315084Sjohnlev 	if ((xenbus_read(XBT_NULL, pdp->xd_xsdev.nodename, name, &prop_str,
3325084Sjohnlev 	    &len) == 0) && (prop_str != NULL) && (strlen(prop_str) != 0))
3335084Sjohnlev 		goto got_xs_prop;
3345084Sjohnlev 
3355084Sjohnlev 	prop_str = NULL;
3365084Sjohnlev 	if ((pdp->xd_xsdev.otherend != NULL) &&
3375084Sjohnlev 	    (xenbus_read(XBT_NULL, pdp->xd_xsdev.otherend, name, &prop_str,
3385084Sjohnlev 	    &len) == 0) && (prop_str != NULL) && (strlen(prop_str) != 0))
3395084Sjohnlev 		goto got_xs_prop;
3405084Sjohnlev 
3415084Sjohnlev toss_off:
3425084Sjohnlev 	return (ddi_bus_prop_op(dev, dip, ch_dip, prop_op,
3435084Sjohnlev 	    mod_flags | DDI_PROP_NOTPROM, name, valuep, lengthp));
3445084Sjohnlev 
3455084Sjohnlev got_xs_prop:
3465084Sjohnlev 	prop_len = strlen(prop_str) + 1;
3475084Sjohnlev 	rv = DDI_PROP_SUCCESS;
3485084Sjohnlev 
3495084Sjohnlev 	switch (prop_op) {
3505084Sjohnlev 	case PROP_LEN:
3515084Sjohnlev 		*lengthp = prop_len;
3525084Sjohnlev 		break;
3535084Sjohnlev 
3545084Sjohnlev 	case PROP_LEN_AND_VAL_ALLOC:
3555084Sjohnlev 		buff = kmem_alloc((size_t)prop_len, KM_SLEEP);
3565084Sjohnlev 		*(caddr_t *)valuep = (caddr_t)buff;
3575084Sjohnlev 		break;
3585084Sjohnlev 	case PROP_LEN_AND_VAL_BUF:
3595084Sjohnlev 		buff = (caddr_t)valuep;
3605084Sjohnlev 		if (*lengthp < prop_len)
3615084Sjohnlev 			rv = DDI_PROP_BUF_TOO_SMALL;
3625084Sjohnlev 		break;
3635084Sjohnlev 	default:
3645084Sjohnlev 		rv = DDI_PROP_INVAL_ARG;
3655084Sjohnlev 		break;
3665084Sjohnlev 	}
3675084Sjohnlev 
3685084Sjohnlev 	if ((rv == DDI_PROP_SUCCESS) && (prop_len > 0)) {
3695084Sjohnlev 		bcopy(prop_str, buff, prop_len);
3705084Sjohnlev 		*lengthp = prop_len;
3715084Sjohnlev 	}
3725084Sjohnlev 	kmem_free(prop_str, len);
3735084Sjohnlev 	return (rv);
3745084Sjohnlev }
3755084Sjohnlev 
3765084Sjohnlev 
3775084Sjohnlev /*
3785084Sjohnlev  * return address of the device's interrupt spec structure.
3795084Sjohnlev  */
3805084Sjohnlev /*ARGSUSED*/
3815084Sjohnlev struct intrspec *
3825084Sjohnlev xpvd_get_ispec(dev_info_t *rdip, uint_t inumber)
3835084Sjohnlev {
3845084Sjohnlev 	struct xendev_ppd *pdp;
3855084Sjohnlev 
3865084Sjohnlev 	ASSERT(inumber == 0);
3875084Sjohnlev 
3885084Sjohnlev 	if ((pdp = ddi_get_parent_data(rdip)) == NULL)
3895084Sjohnlev 		return (NULL);
3905084Sjohnlev 
3915084Sjohnlev 	return (&pdp->xd_ispec);
3925084Sjohnlev }
3935084Sjohnlev 
3945084Sjohnlev /*
3955084Sjohnlev  * return (and determine) the interrupt priority of the device.
3965084Sjohnlev  */
3975084Sjohnlev /*ARGSUSED*/
3985084Sjohnlev static int
3995084Sjohnlev xpvd_get_priority(dev_info_t *dip, int inum, int *pri)
4005084Sjohnlev {
4015084Sjohnlev 	struct xendev_ppd *pdp;
4025084Sjohnlev 	struct intrspec *ispec;
4035084Sjohnlev 	int	*intpriorities;
4045084Sjohnlev 	uint_t	num_intpriorities;
4055084Sjohnlev 
4065084Sjohnlev 	DDI_INTR_NEXDBG((CE_CONT, "xpvd_get_priority: dip = 0x%p\n",
4075084Sjohnlev 	    (void *)dip));
4085084Sjohnlev 
4095084Sjohnlev 	ASSERT(inum == 0);
4105084Sjohnlev 
4115084Sjohnlev 	if ((pdp = ddi_get_parent_data(dip)) == NULL)
4125084Sjohnlev 		return (DDI_FAILURE);
4135084Sjohnlev 
4145084Sjohnlev 	ispec = &pdp->xd_ispec;
4155084Sjohnlev 
4165084Sjohnlev 	/*
4175084Sjohnlev 	 * Set the default priority based on the device class.  The
4185084Sjohnlev 	 * "interrupt-priorities" property can be used to override
4195084Sjohnlev 	 * the default.
4205084Sjohnlev 	 */
4215084Sjohnlev 	if (ispec->intrspec_pri == 0) {
4225084Sjohnlev 		ispec->intrspec_pri = xendev_devclass_ipl(pdp->xd_devclass);
4235084Sjohnlev 		if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY, dip,
4245084Sjohnlev 		    DDI_PROP_NOTPROM | DDI_PROP_DONTPASS,
4255084Sjohnlev 		    "interrupt-priorities", &intpriorities,
4265084Sjohnlev 		    &num_intpriorities) == DDI_PROP_SUCCESS) {
4275084Sjohnlev 			ispec->intrspec_pri = intpriorities[0];
4285084Sjohnlev 			ddi_prop_free(intpriorities);
4295084Sjohnlev 		}
4305084Sjohnlev 	}
4315084Sjohnlev 	*pri = ispec->intrspec_pri;
4325084Sjohnlev 	return (DDI_SUCCESS);
4335084Sjohnlev }
4345084Sjohnlev 
4355084Sjohnlev 
4365084Sjohnlev /*
4375084Sjohnlev  * xpvd_intr_ops: bus_intr_op() function for interrupt support
4385084Sjohnlev  */
4395084Sjohnlev /* ARGSUSED */
4405084Sjohnlev static int
4415084Sjohnlev xpvd_intr_ops(dev_info_t *pdip, dev_info_t *rdip, ddi_intr_op_t intr_op,
4425084Sjohnlev     ddi_intr_handle_impl_t *hdlp, void *result)
4435084Sjohnlev {
4445084Sjohnlev 	int priority = 0;
4455084Sjohnlev 	struct intrspec *ispec;
4465084Sjohnlev 	struct xendev_ppd *pdp;
4475084Sjohnlev 
4485084Sjohnlev 	DDI_INTR_NEXDBG((CE_CONT,
4495084Sjohnlev 	    "xpvd_intr_ops: pdip 0x%p, rdip 0x%p, op %x handle 0x%p\n",
4505084Sjohnlev 	    (void *)pdip, (void *)rdip, intr_op, (void *)hdlp));
4515084Sjohnlev 
4525084Sjohnlev 	/* Process the request */
4535084Sjohnlev 	switch (intr_op) {
4545084Sjohnlev 	case DDI_INTROP_SUPPORTED_TYPES:
4555084Sjohnlev 		/* Fixed supported by default */
4565084Sjohnlev 		*(int *)result = DDI_INTR_TYPE_FIXED;
4575084Sjohnlev 		break;
4585084Sjohnlev 
4595084Sjohnlev 	case DDI_INTROP_NINTRS:
4605084Sjohnlev 		*(int *)result = 1;
4615084Sjohnlev 		break;
4625084Sjohnlev 
4635084Sjohnlev 	case DDI_INTROP_ALLOC:
4645084Sjohnlev 		/*
4655084Sjohnlev 		 * FIXED interrupts: just return available interrupts
4665084Sjohnlev 		 */
4675084Sjohnlev 		if (hdlp->ih_type == DDI_INTR_TYPE_FIXED) {
4685084Sjohnlev 			/*
4695084Sjohnlev 			 * event channels are edge-triggered, maskable,
4705084Sjohnlev 			 * and support int pending.
4715084Sjohnlev 			 */
4725084Sjohnlev 			hdlp->ih_cap |= XENDEV_INTR_CAPABILITIES;
4735084Sjohnlev 			*(int *)result = 1;	/* DDI_INTR_TYPE_FIXED */
4745084Sjohnlev 		} else {
4755084Sjohnlev 			return (DDI_FAILURE);
4765084Sjohnlev 		}
4775084Sjohnlev 		break;
4785084Sjohnlev 
4795084Sjohnlev 	case DDI_INTROP_FREE:
4805084Sjohnlev 		ispec = xpvd_get_ispec(rdip, (int)hdlp->ih_inum);
4815084Sjohnlev 		if (ispec == NULL)
4825084Sjohnlev 			return (DDI_FAILURE);
4835084Sjohnlev 		ispec->intrspec_pri = 0; /* mark as un-initialized */
4845084Sjohnlev 		break;
4855084Sjohnlev 
4865084Sjohnlev 	case DDI_INTROP_GETPRI:
4875084Sjohnlev 		if (xpvd_get_priority(rdip, hdlp->ih_inum, &priority) !=
4885084Sjohnlev 		    DDI_SUCCESS)
4895084Sjohnlev 			return (DDI_FAILURE);
4905084Sjohnlev 		DDI_INTR_NEXDBG((CE_CONT, "xpvd_intr_ops: priority = 0x%x\n",
4915084Sjohnlev 		    priority));
4925084Sjohnlev 		*(int *)result = priority;
4935084Sjohnlev 		break;
4945084Sjohnlev 
4955084Sjohnlev 	case DDI_INTROP_SETPRI:
4965084Sjohnlev 		/* Validate the interrupt priority passed */
4975084Sjohnlev 		if (*(int *)result > LOCK_LEVEL)
4985084Sjohnlev 			return (DDI_FAILURE);
4995084Sjohnlev 
5005084Sjohnlev 		/* Ensure that PSM is all initialized */
5015084Sjohnlev 		if (psm_intr_ops == NULL)
5025084Sjohnlev 			return (DDI_FAILURE);
5035084Sjohnlev 
5045084Sjohnlev 		/* Change the priority */
5055084Sjohnlev 		if ((*psm_intr_ops)(rdip, hdlp, PSM_INTR_OP_SET_PRI, result) ==
5065084Sjohnlev 		    PSM_FAILURE)
5075084Sjohnlev 			return (DDI_FAILURE);
5085084Sjohnlev 
5095084Sjohnlev 		ispec = xpvd_get_ispec(rdip, (int)hdlp->ih_inum);
5105084Sjohnlev 		if (ispec == NULL)
5115084Sjohnlev 			return (DDI_FAILURE);
5125084Sjohnlev 		ispec->intrspec_pri = *(int *)result;
5135084Sjohnlev 		break;
5145084Sjohnlev 
5155084Sjohnlev 	case DDI_INTROP_ADDISR:
5165084Sjohnlev 		/* update ispec */
5175084Sjohnlev 		ispec = xpvd_get_ispec(rdip, (int)hdlp->ih_inum);
5185084Sjohnlev 		if (ispec == NULL)
5195084Sjohnlev 			return (DDI_FAILURE);
5205084Sjohnlev 		ispec->intrspec_func = hdlp->ih_cb_func;
5215084Sjohnlev 
5225084Sjohnlev 		break;
5235084Sjohnlev 
5245084Sjohnlev 	case DDI_INTROP_REMISR:
5255084Sjohnlev 		ispec = xpvd_get_ispec(rdip, (int)hdlp->ih_inum);
5265084Sjohnlev 		pdp = (struct xendev_ppd *)ddi_get_parent_data(rdip);
5275084Sjohnlev 
5285084Sjohnlev 		ASSERT(pdp != NULL);
5295084Sjohnlev 		ASSERT(pdp->xd_evtchn != INVALID_EVTCHN);
5305084Sjohnlev 
5315084Sjohnlev 		if (ispec) {
5325084Sjohnlev 			ispec->intrspec_vec = 0;
5335084Sjohnlev 			ispec->intrspec_func = (uint_t (*)()) 0;
5345084Sjohnlev 		}
5355084Sjohnlev 		pdp->xd_evtchn = INVALID_EVTCHN;
5365084Sjohnlev 		break;
5375084Sjohnlev 
5385084Sjohnlev 	case DDI_INTROP_GETCAP:
5395084Sjohnlev 		if (hdlp->ih_type ==  DDI_INTR_TYPE_FIXED) {
5405084Sjohnlev 			/*
5415084Sjohnlev 			 * event channels are edge-triggered, maskable,
5425084Sjohnlev 			 * and support int pending.
5435084Sjohnlev 			 */
5445084Sjohnlev 			*(int *)result = XENDEV_INTR_CAPABILITIES;
5455084Sjohnlev 		} else {
5465084Sjohnlev 			*(int *)result = 0;
5475084Sjohnlev 			return (DDI_FAILURE);
5485084Sjohnlev 		}
5495084Sjohnlev 		DDI_INTR_NEXDBG((CE_CONT, "xpvd: GETCAP returned = %x\n",
5505084Sjohnlev 		    *(int *)result));
5515084Sjohnlev 		break;
5525084Sjohnlev 	case DDI_INTROP_SETCAP:
5535084Sjohnlev 		DDI_INTR_NEXDBG((CE_CONT, "xpvd_intr_ops: SETCAP cap=0x%x\n",
5545084Sjohnlev 		    *(int *)result));
5555084Sjohnlev 		if (psm_intr_ops == NULL)
5565084Sjohnlev 			return (DDI_FAILURE);
5575084Sjohnlev 
5585084Sjohnlev 		if ((*psm_intr_ops)(rdip, hdlp, PSM_INTR_OP_SET_CAP, result)) {
5595084Sjohnlev 			DDI_INTR_NEXDBG((CE_CONT, "GETCAP: psm_intr_ops"
5605084Sjohnlev 			    " returned failure\n"));
5615084Sjohnlev 			return (DDI_FAILURE);
5625084Sjohnlev 		}
5635084Sjohnlev 		break;
5645084Sjohnlev 
5655084Sjohnlev 	case DDI_INTROP_ENABLE:
5665084Sjohnlev 		if (psm_intr_ops == NULL)
5675084Sjohnlev 			return (DDI_FAILURE);
5685084Sjohnlev 
5695084Sjohnlev 		if (xpvd_enable_intr(rdip, hdlp, (int)hdlp->ih_inum) !=
5705084Sjohnlev 		    DDI_SUCCESS)
5715084Sjohnlev 			return (DDI_FAILURE);
5725084Sjohnlev 
5735084Sjohnlev 		DDI_INTR_NEXDBG((CE_CONT, "xpvd_intr_ops: ENABLE vec=0x%x\n",
5745084Sjohnlev 		    hdlp->ih_vector));
5755084Sjohnlev 		break;
5765084Sjohnlev 
5775084Sjohnlev 	case DDI_INTROP_DISABLE:
5785084Sjohnlev 		if (psm_intr_ops == NULL)
5795084Sjohnlev 			return (DDI_FAILURE);
5805084Sjohnlev 		xpvd_disable_intr(rdip, hdlp, hdlp->ih_inum);
5815084Sjohnlev 		DDI_INTR_NEXDBG((CE_CONT, "xpvd_intr_ops: DISABLE vec = %x\n",
5825084Sjohnlev 		    hdlp->ih_vector));
5835084Sjohnlev 		break;
5845084Sjohnlev 
5855084Sjohnlev 	case DDI_INTROP_BLOCKENABLE:
5865084Sjohnlev 	case DDI_INTROP_BLOCKDISABLE:
5875084Sjohnlev 		return (DDI_FAILURE);
5885084Sjohnlev 
5895084Sjohnlev 	case DDI_INTROP_SETMASK:
5905084Sjohnlev 	case DDI_INTROP_CLRMASK:
5915741Smrj #ifdef XPV_HVM_DRIVER
5925741Smrj 		return (DDI_ENOTSUP);
5935741Smrj #else
5945084Sjohnlev 		/*
5955084Sjohnlev 		 * Handle this here
5965084Sjohnlev 		 */
5975084Sjohnlev 		if (hdlp->ih_type != DDI_INTR_TYPE_FIXED)
5985084Sjohnlev 			return (DDI_FAILURE);
5995084Sjohnlev 		if (intr_op == DDI_INTROP_SETMASK) {
6005084Sjohnlev 			ec_disable_irq(hdlp->ih_vector);
6015084Sjohnlev 		} else {
6025084Sjohnlev 			ec_enable_irq(hdlp->ih_vector);
6035084Sjohnlev 		}
6045084Sjohnlev 		break;
6055741Smrj #endif
6065084Sjohnlev 	case DDI_INTROP_GETPENDING:
6075741Smrj #ifdef XPV_HVM_DRIVER
6085741Smrj 		return (DDI_ENOTSUP);
6095741Smrj #else
6105084Sjohnlev 		if (hdlp->ih_type != DDI_INTR_TYPE_FIXED)
6115084Sjohnlev 			return (DDI_FAILURE);
6125084Sjohnlev 		*(int *)result = ec_pending_irq(hdlp->ih_vector);
6135084Sjohnlev 		DDI_INTR_NEXDBG((CE_CONT, "xpvd: GETPENDING returned = %x\n",
6145084Sjohnlev 		    *(int *)result));
6155084Sjohnlev 		break;
6165741Smrj #endif
6175084Sjohnlev 
6185084Sjohnlev 	case DDI_INTROP_NAVAIL:
6195084Sjohnlev 		*(int *)result = 1;
6205084Sjohnlev 		DDI_INTR_NEXDBG((CE_CONT, "xpvd: NAVAIL returned = %x\n",
6215084Sjohnlev 		    *(int *)result));
6225084Sjohnlev 		break;
6235084Sjohnlev 
6245084Sjohnlev 	default:
6255084Sjohnlev 		return (i_ddi_intr_ops(pdip, rdip, intr_op, hdlp, result));
6265084Sjohnlev 	}
6275084Sjohnlev 
6285084Sjohnlev 	return (DDI_SUCCESS);
6295084Sjohnlev }
6305084Sjohnlev 
6315084Sjohnlev 
6325084Sjohnlev static int
6335084Sjohnlev xpvd_enable_intr(dev_info_t *rdip, ddi_intr_handle_impl_t *hdlp, int inum)
6345084Sjohnlev {
6355084Sjohnlev 	int		vector;
6365084Sjohnlev 	ihdl_plat_t	*ihdl_plat_datap = (ihdl_plat_t *)hdlp->ih_private;
6375084Sjohnlev 
6385084Sjohnlev 	DDI_INTR_NEXDBG((CE_CONT, "xpvd_enable_intr: hdlp %p inum %x\n",
6395084Sjohnlev 	    (void *)hdlp, inum));
6405084Sjohnlev 
6415084Sjohnlev 	ihdl_plat_datap->ip_ispecp = xpvd_get_ispec(rdip, inum);
6425084Sjohnlev 	if (ihdl_plat_datap->ip_ispecp == NULL)
6435084Sjohnlev 		return (DDI_FAILURE);
6445084Sjohnlev 
6455084Sjohnlev 	/* translate the interrupt if needed */
6465084Sjohnlev 	(void) (*psm_intr_ops)(rdip, hdlp, PSM_INTR_OP_XLATE_VECTOR, &vector);
6475084Sjohnlev 	DDI_INTR_NEXDBG((CE_CONT, "xpvd_enable_intr: priority=%x vector=%x\n",
6485084Sjohnlev 	    hdlp->ih_pri, vector));
6495084Sjohnlev 
6505084Sjohnlev 	/* Add the interrupt handler */
6515084Sjohnlev 	if (!add_avintr((void *)hdlp, hdlp->ih_pri, hdlp->ih_cb_func,
6525084Sjohnlev 	    DEVI(rdip)->devi_name, vector, hdlp->ih_cb_arg1,
6535084Sjohnlev 	    hdlp->ih_cb_arg2, NULL, rdip))
6545084Sjohnlev 		return (DDI_FAILURE);
6555084Sjohnlev 
6565084Sjohnlev 	/* Note this really is an irq. */
6575084Sjohnlev 	hdlp->ih_vector = (ushort_t)vector;
6585084Sjohnlev 
6595084Sjohnlev 	return (DDI_SUCCESS);
6605084Sjohnlev }
6615084Sjohnlev 
6625084Sjohnlev 
6635084Sjohnlev static void
6645084Sjohnlev xpvd_disable_intr(dev_info_t *rdip, ddi_intr_handle_impl_t *hdlp, int inum)
6655084Sjohnlev {
6665084Sjohnlev 	int		vector;
6675084Sjohnlev 	ihdl_plat_t	*ihdl_plat_datap = (ihdl_plat_t *)hdlp->ih_private;
6685084Sjohnlev 
6695084Sjohnlev 	DDI_INTR_NEXDBG((CE_CONT, "xpvd_disable_intr: \n"));
6705084Sjohnlev 	ihdl_plat_datap->ip_ispecp = xpvd_get_ispec(rdip, inum);
6715084Sjohnlev 	if (ihdl_plat_datap->ip_ispecp == NULL)
6725084Sjohnlev 		return;
6735084Sjohnlev 
6745084Sjohnlev 	/* translate the interrupt if needed */
6755084Sjohnlev 	(void) (*psm_intr_ops)(rdip, hdlp, PSM_INTR_OP_XLATE_VECTOR, &vector);
6765084Sjohnlev 
6775084Sjohnlev 	/* Disable the interrupt handler */
6785084Sjohnlev 	rem_avintr((void *)hdlp, hdlp->ih_pri, hdlp->ih_cb_func, vector);
6795084Sjohnlev 	ihdl_plat_datap->ip_ispecp = NULL;
6805084Sjohnlev }
6815084Sjohnlev 
6825084Sjohnlev /*ARGSUSED*/
6835084Sjohnlev static int
6845084Sjohnlev xpvd_ctlops(dev_info_t *dip, dev_info_t *rdip,
6855084Sjohnlev 	ddi_ctl_enum_t ctlop, void *arg, void *result)
6865084Sjohnlev {
6875084Sjohnlev 	switch (ctlop) {
6885084Sjohnlev 	case DDI_CTLOPS_REPORTDEV:
6895084Sjohnlev 		if (rdip == (dev_info_t *)0)
6905084Sjohnlev 			return (DDI_FAILURE);
6915084Sjohnlev 		cmn_err(CE_CONT, "?%s@%s, %s%d\n", ddi_node_name(rdip),
6925084Sjohnlev 		    ddi_get_name_addr(rdip), ddi_driver_name(rdip),
6935084Sjohnlev 		    ddi_get_instance(rdip));
6945084Sjohnlev 		return (DDI_SUCCESS);
6955084Sjohnlev 
6965084Sjohnlev 	case DDI_CTLOPS_INITCHILD:
6975084Sjohnlev 		return (xpvd_initchild((dev_info_t *)arg));
6985084Sjohnlev 
6995084Sjohnlev 	case DDI_CTLOPS_UNINITCHILD:
7005084Sjohnlev 		return (xpvd_removechild((dev_info_t *)arg));
7015084Sjohnlev 
7025084Sjohnlev 	case DDI_CTLOPS_SIDDEV:
7035084Sjohnlev 		return (DDI_SUCCESS);
7045084Sjohnlev 
7055084Sjohnlev 	case DDI_CTLOPS_REGSIZE:
7065084Sjohnlev 	case DDI_CTLOPS_NREGS:
7075084Sjohnlev 		return (DDI_FAILURE);
7085084Sjohnlev 
7095084Sjohnlev 	case DDI_CTLOPS_POWER: {
7105084Sjohnlev 		return (ddi_ctlops(dip, rdip, ctlop, arg, result));
7115084Sjohnlev 	}
7125084Sjohnlev 
7135084Sjohnlev 	default:
7145084Sjohnlev 		return (ddi_ctlops(dip, rdip, ctlop, arg, result));
7155084Sjohnlev 	}
7165084Sjohnlev 
7175084Sjohnlev 	/* NOTREACHED */
7185084Sjohnlev 
7195084Sjohnlev }
7205084Sjohnlev 
7215084Sjohnlev /*
7225084Sjohnlev  * Assign the address portion of the node name
7235084Sjohnlev  */
7245084Sjohnlev static int
7256318Sedp xpvd_name_child(dev_info_t *child, char *addr, int addrlen)
7265084Sjohnlev {
7275084Sjohnlev 	int *domain, *vdev;
7285084Sjohnlev 	uint_t ndomain, nvdev;
7296318Sedp 	char *prop_str;
7305084Sjohnlev 
7315084Sjohnlev 	/*
7325084Sjohnlev 	 * i_xpvd_parse_devname() knows the formats used by this
7335084Sjohnlev 	 * routine.  If this code changes, so must that.
7345084Sjohnlev 	 */
7355084Sjohnlev 
7365084Sjohnlev 	if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY, child, DDI_PROP_DONTPASS,
7375084Sjohnlev 	    "domain", &domain, &ndomain) != DDI_PROP_SUCCESS)
7385084Sjohnlev 		return (DDI_FAILURE);
7395084Sjohnlev 	ASSERT(ndomain == 1);
7405084Sjohnlev 
7415084Sjohnlev 	/*
7425084Sjohnlev 	 * Use "domain" and "vdev" properties (backend drivers).
7435084Sjohnlev 	 */
7445084Sjohnlev 	if (*domain != DOMID_SELF) {
7455084Sjohnlev 		if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY, child,
7465084Sjohnlev 		    DDI_PROP_DONTPASS, "vdev", &vdev, &nvdev)
7475084Sjohnlev 		    != DDI_PROP_SUCCESS) {
7485084Sjohnlev 			ddi_prop_free(domain);
7495084Sjohnlev 			return (DDI_FAILURE);
7505084Sjohnlev 		}
7515084Sjohnlev 		ASSERT(nvdev == 1);
7525084Sjohnlev 
7536318Sedp 		(void) snprintf(addr, addrlen, "%d,%d", domain[0], vdev[0]);
7545084Sjohnlev 		ddi_prop_free(vdev);
7555084Sjohnlev 		ddi_prop_free(domain);
7565084Sjohnlev 		return (DDI_SUCCESS);
7575084Sjohnlev 	}
7585084Sjohnlev 	ddi_prop_free(domain);
7595084Sjohnlev 
7605084Sjohnlev 	/*
7615084Sjohnlev 	 * Use "unit-address" property (frontend/softdev drivers).
7625084Sjohnlev 	 */
7636318Sedp 	if (ddi_prop_lookup_string(DDI_DEV_T_ANY, child, DDI_PROP_DONTPASS,
7646318Sedp 	    "unit-address", &prop_str) != DDI_PROP_SUCCESS)
7656318Sedp 		return (DDI_FAILURE);
7666318Sedp 	(void) strlcpy(addr, prop_str, addrlen);
7676318Sedp 	ddi_prop_free(prop_str);
7686318Sedp 	return (DDI_SUCCESS);
7695084Sjohnlev }
7705084Sjohnlev 
7715084Sjohnlev static int
7725084Sjohnlev xpvd_initchild(dev_info_t *child)
7735084Sjohnlev {
7746318Sedp 	char addr[80];
7755084Sjohnlev 
7765084Sjohnlev 	/*
7775084Sjohnlev 	 * Pseudo nodes indicate a prototype node with per-instance
7785084Sjohnlev 	 * properties to be merged into the real h/w device node.
7795084Sjohnlev 	 */
7805084Sjohnlev 	if (ndi_dev_is_persistent_node(child) == 0) {
7815084Sjohnlev 		ddi_set_parent_data(child, NULL);
7825084Sjohnlev 
7835084Sjohnlev 		/*
7845084Sjohnlev 		 * Try to merge the properties from this prototype
7855084Sjohnlev 		 * node into real h/w nodes.
7865084Sjohnlev 		 */
7875084Sjohnlev 		if (ndi_merge_node(child, xpvd_name_child) == DDI_SUCCESS) {
7885084Sjohnlev 			/*
7895084Sjohnlev 			 * Merged ok - return failure to remove the node.
7905084Sjohnlev 			 */
7915084Sjohnlev 			ddi_set_name_addr(child, NULL);
7925084Sjohnlev 			return (DDI_FAILURE);
7935084Sjohnlev 		}
7945084Sjohnlev 
7955084Sjohnlev 		/*
7965084Sjohnlev 		 * The child was not merged into a h/w node,
7975084Sjohnlev 		 * but there's not much we can do with it other
7985084Sjohnlev 		 * than return failure to cause the node to be removed.
7995084Sjohnlev 		 */
8005084Sjohnlev 		cmn_err(CE_WARN, "!%s@%s: %s.conf properties not merged",
8015084Sjohnlev 		    ddi_get_name(child), ddi_get_name_addr(child),
8025084Sjohnlev 		    ddi_get_name(child));
8035084Sjohnlev 		ddi_set_name_addr(child, NULL);
8045084Sjohnlev 		return (DDI_NOT_WELL_FORMED);
8055084Sjohnlev 	}
8065084Sjohnlev 
8075084Sjohnlev 	if (xvdi_init_dev(child) != DDI_SUCCESS)
8085084Sjohnlev 		return (DDI_FAILURE);
8095084Sjohnlev 
8106318Sedp 	if (xpvd_name_child(child, addr, sizeof (addr)) != DDI_SUCCESS) {
8115084Sjohnlev 		xvdi_uninit_dev(child);
8125084Sjohnlev 		return (DDI_FAILURE);
8135084Sjohnlev 	}
8146318Sedp 	ddi_set_name_addr(child, addr);
8155084Sjohnlev 
8165084Sjohnlev 	return (DDI_SUCCESS);
8175084Sjohnlev }
8185084Sjohnlev 
8195084Sjohnlev static int
8205084Sjohnlev xpvd_removechild(dev_info_t *dip)
8215084Sjohnlev {
8225084Sjohnlev 	xvdi_uninit_dev(dip);
8235084Sjohnlev 
8245084Sjohnlev 	ddi_set_name_addr(dip, NULL);
8255084Sjohnlev 
8265084Sjohnlev 	/*
8275084Sjohnlev 	 * Strip the node to properly convert it back to prototype
8285084Sjohnlev 	 * form.
8295084Sjohnlev 	 */
8305084Sjohnlev 	ddi_remove_minor_node(dip, NULL);
8315084Sjohnlev 
8325084Sjohnlev 	return (DDI_SUCCESS);
8335084Sjohnlev }
8345084Sjohnlev 
8355084Sjohnlev static int
8365084Sjohnlev xpvd_bus_unconfig(dev_info_t *parent, uint_t flag, ddi_bus_config_op_t op,
8375084Sjohnlev     void *device_name)
8385084Sjohnlev {
8395084Sjohnlev 	return (ndi_busop_bus_unconfig(parent, flag, op, device_name));
8405084Sjohnlev }
8415084Sjohnlev 
8425084Sjohnlev /*
8435084Sjohnlev  * Given the name of a child of xpvd, determine the device class,
8445084Sjohnlev  * domain and vdevnum to which it refers.
8455084Sjohnlev  */
8465084Sjohnlev static boolean_t
8475084Sjohnlev i_xpvd_parse_devname(char *name, xendev_devclass_t *devclassp,
8485084Sjohnlev     domid_t *domp, int *vdevp)
8495084Sjohnlev {
8505084Sjohnlev 	int len = strlen(name) + 1;
8515084Sjohnlev 	char *device_name = i_ddi_strdup(name, KM_SLEEP);
8525084Sjohnlev 	char *cname = NULL, *caddr = NULL;
8535084Sjohnlev 	boolean_t ret;
8545084Sjohnlev 
8555084Sjohnlev 	i_ddi_parse_name(device_name, &cname, &caddr, NULL);
8565084Sjohnlev 
8575084Sjohnlev 	if ((cname == NULL) || (strlen(cname) == 0) ||
8585084Sjohnlev 	    (caddr == NULL) || (strlen(caddr) == 0)) {
8595084Sjohnlev 		ret = B_FALSE;
8605084Sjohnlev 		goto done;
8615084Sjohnlev 	}
8625084Sjohnlev 
8635084Sjohnlev 	*devclassp = xendev_nodename_to_devclass(cname);
8645084Sjohnlev 	if (*devclassp < 0) {
8655084Sjohnlev 		ret = B_FALSE;
8665084Sjohnlev 		goto done;
8675084Sjohnlev 	}
8685084Sjohnlev 
8695084Sjohnlev 	/*
8705084Sjohnlev 	 * Parsing the address component requires knowledge of how
8715084Sjohnlev 	 * xpvd_name_child() works.  If that code changes, so must
8725084Sjohnlev 	 * this.
8735084Sjohnlev 	 */
8745084Sjohnlev 
8755084Sjohnlev 	/* Backend format is "<domain>,<vdev>". */
8767632SNick.Todd@Sun.COM 	if (sscanf(caddr, "%hu,%d", domp, vdevp) == 2) {
8775084Sjohnlev 		ret = B_TRUE;
8785084Sjohnlev 		goto done;
8795084Sjohnlev 	}
8805084Sjohnlev 
8815084Sjohnlev 	/* Frontend format is "<vdev>". */
8825084Sjohnlev 	*domp = DOMID_SELF;
8836500Sjhd 	if (sscanf(caddr, "%d", vdevp) == 1)
8845084Sjohnlev 		ret = B_TRUE;
8855084Sjohnlev done:
8865084Sjohnlev 	kmem_free(device_name, len);
8875084Sjohnlev 	return (ret);
8885084Sjohnlev }
8895084Sjohnlev 
8905084Sjohnlev /*
8915084Sjohnlev  * xpvd_bus_config()
8925084Sjohnlev  *
8935084Sjohnlev  * BUS_CONFIG_ONE:
8945084Sjohnlev  *	Enumerate the exact instance of a driver.
8955084Sjohnlev  *
8965084Sjohnlev  * BUS_CONFIG_ALL:
8975084Sjohnlev  *	Enumerate all the instances of all the possible children (seen before
8985084Sjohnlev  *	and never seen before).
8995084Sjohnlev  *
9005084Sjohnlev  * BUS_CONFIG_DRIVER:
9015084Sjohnlev  *	Enumerate all the instances of a particular driver.
9025084Sjohnlev  */
9035084Sjohnlev static int
9045084Sjohnlev xpvd_bus_config(dev_info_t *parent, uint_t flag, ddi_bus_config_op_t op,
9055084Sjohnlev 	void *arg, dev_info_t **childp)
9065084Sjohnlev {
9075084Sjohnlev 	int circ;
9085084Sjohnlev 	char *cname = NULL;
9095084Sjohnlev 
9105084Sjohnlev 	ndi_devi_enter(parent, &circ);
9115084Sjohnlev 
9125084Sjohnlev 	switch (op) {
9135084Sjohnlev 	case BUS_CONFIG_ONE: {
9145084Sjohnlev 		xendev_devclass_t devclass;
9155084Sjohnlev 		domid_t dom;
9165084Sjohnlev 		int vdev;
9175084Sjohnlev 
9185084Sjohnlev 		if (!i_xpvd_parse_devname(arg, &devclass, &dom, &vdev)) {
9195084Sjohnlev 			ndi_devi_exit(parent, circ);
9205084Sjohnlev 			return (NDI_FAILURE);
9215084Sjohnlev 		}
9225084Sjohnlev 
9235084Sjohnlev 		*childp = xvdi_find_dev(parent, devclass, dom, vdev);
9245084Sjohnlev 		if (*childp == NULL)
9255084Sjohnlev 			*childp = xvdi_create_dev(parent, devclass, dom, vdev);
9265084Sjohnlev 
9275084Sjohnlev 		ndi_devi_exit(parent, circ);
9285084Sjohnlev 
9295084Sjohnlev 		if (*childp == NULL)
9305084Sjohnlev 			return (NDI_FAILURE);
9315084Sjohnlev 		else
9325084Sjohnlev 			return (ndi_busop_bus_config(parent, flag,
9335084Sjohnlev 			    op, arg, childp, 0));
9345084Sjohnlev 	}
9355084Sjohnlev 
9365084Sjohnlev 	case BUS_CONFIG_DRIVER: {
9375084Sjohnlev 		xendev_devclass_t devclass = XEN_INVAL;
9385084Sjohnlev 
9395084Sjohnlev 		cname = ddi_major_to_name((major_t)(uintptr_t)arg);
9405084Sjohnlev 		if (cname != NULL)
9415084Sjohnlev 			devclass = xendev_nodename_to_devclass(cname);
9425084Sjohnlev 
9435084Sjohnlev 		if (devclass == XEN_INVAL) {
9445084Sjohnlev 			ndi_devi_exit(parent, circ);
9455084Sjohnlev 			return (NDI_FAILURE);
9465084Sjohnlev 		} else {
9475084Sjohnlev 			xendev_enum_class(parent, devclass);
9485084Sjohnlev 			ndi_devi_exit(parent, circ);
9495084Sjohnlev 			return (ndi_busop_bus_config(parent, flag, op,
9505084Sjohnlev 			    arg, childp, 0));
9515084Sjohnlev 		}
9525084Sjohnlev 		/* NOTREACHED */
9535084Sjohnlev 	}
9545084Sjohnlev 
9555084Sjohnlev 	case BUS_CONFIG_ALL:
9565084Sjohnlev 		xendev_enum_all(parent, B_FALSE);
9575084Sjohnlev 		ndi_devi_exit(parent, circ);
9585084Sjohnlev 
9595084Sjohnlev 		return (ndi_busop_bus_config(parent, flag, op,
9605084Sjohnlev 		    arg, childp, 0));
9615084Sjohnlev 
9625084Sjohnlev 	default:
9635084Sjohnlev 		ndi_devi_exit(parent, circ);
9645084Sjohnlev 		return (NDI_FAILURE);
9655084Sjohnlev 	}
9665084Sjohnlev }
9675084Sjohnlev 
9685084Sjohnlev /*ARGSUSED*/
9695084Sjohnlev static int
9705084Sjohnlev xpvd_get_eventcookie(dev_info_t *dip, dev_info_t *rdip,
9715084Sjohnlev     char *eventname, ddi_eventcookie_t *cookie)
9725084Sjohnlev {
9735084Sjohnlev 	return (ndi_event_retrieve_cookie(xpvd_ndi_event_handle,
9745084Sjohnlev 	    rdip, eventname, cookie, NDI_EVENT_NOPASS));
9755084Sjohnlev }
9765084Sjohnlev 
9775084Sjohnlev /*ARGSUSED*/
9785084Sjohnlev static int
9795084Sjohnlev xpvd_add_eventcall(dev_info_t *dip, dev_info_t *rdip,
9805084Sjohnlev     ddi_eventcookie_t cookie, void (*callback)(dev_info_t *dip,
9815084Sjohnlev     ddi_eventcookie_t cookie, void *arg, void *bus_impldata),
9825084Sjohnlev     void *arg, ddi_callback_id_t *cb_id)
9835084Sjohnlev {
9845084Sjohnlev 	return (ndi_event_add_callback(xpvd_ndi_event_handle,
9855084Sjohnlev 	    rdip, cookie, callback, arg, NDI_SLEEP, cb_id));
9865084Sjohnlev }
9875084Sjohnlev 
9885084Sjohnlev /*ARGSUSED*/
9895084Sjohnlev static int
9905084Sjohnlev xpvd_remove_eventcall(dev_info_t *dip, ddi_callback_id_t cb_id)
9915084Sjohnlev {
9925084Sjohnlev 	return (ndi_event_remove_callback(xpvd_ndi_event_handle,
9935084Sjohnlev 	    cb_id));
9945084Sjohnlev }
9955084Sjohnlev 
9965084Sjohnlev /*ARGSUSED*/
9975084Sjohnlev static int
9985084Sjohnlev xpvd_post_event(dev_info_t *dip, dev_info_t *rdip,
9995084Sjohnlev     ddi_eventcookie_t cookie, void *bus_impldata)
10005084Sjohnlev {
10015084Sjohnlev 	return (ndi_event_run_callbacks(xpvd_ndi_event_handle, rdip,
10025084Sjohnlev 	    cookie, bus_impldata));
10035084Sjohnlev }
1004