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 /*
23*10175SStuart.Maybee@Sun.COM * Copyright 2009 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 */
1617656SSherry.Moore@Sun.COM &xpvd_bus_ops, /* bus operations */
1627656SSherry.Moore@Sun.COM NULL, /* power */
1637656SSherry.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
_init(void)2095084Sjohnlev _init(void)
2105084Sjohnlev {
2115084Sjohnlev return (mod_install(&modlinkage));
2125084Sjohnlev }
2135084Sjohnlev
2145084Sjohnlev int
_fini(void)2155084Sjohnlev _fini(void)
2165084Sjohnlev {
2175084Sjohnlev return (mod_remove(&modlinkage));
2185084Sjohnlev }
2195084Sjohnlev
2205084Sjohnlev int
_info(struct modinfo * modinfop)2215084Sjohnlev _info(struct modinfo *modinfop)
2225084Sjohnlev {
2235084Sjohnlev return (mod_info(&modlinkage, modinfop));
2245084Sjohnlev }
2255084Sjohnlev
2265084Sjohnlev /* ARGSUSED */
2275084Sjohnlev static int
xpvd_info(dev_info_t * dip,ddi_info_cmd_t cmd,void * arg,void ** result)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
xpvd_attach(dev_info_t * devi,ddi_attach_cmd_t cmd)2465084Sjohnlev xpvd_attach(dev_info_t *devi, ddi_attach_cmd_t cmd)
2475084Sjohnlev {
2485084Sjohnlev extern void xvdi_watch_devices(int);
249*10175SStuart.Maybee@Sun.COM #ifdef XPV_HVM_DRIVER
250*10175SStuart.Maybee@Sun.COM extern dev_info_t *xpv_dip;
2515741Smrj
252*10175SStuart.Maybee@Sun.COM if (xpv_dip == NULL) {
2535741Smrj if (ddi_hold_installed_driver(ddi_name_to_major("xpv")) ==
2545741Smrj NULL) {
2555741Smrj cmn_err(CE_WARN, "Couldn't initialize xpv framework");
2565741Smrj return (DDI_FAILURE);
2575741Smrj }
2585741Smrj }
2596318Sedp #endif /* XPV_HVM_DRIVER */
2605084Sjohnlev
2615084Sjohnlev if (ndi_event_alloc_hdl(devi, 0, &xpvd_ndi_event_handle,
2625084Sjohnlev NDI_SLEEP) != NDI_SUCCESS) {
2635084Sjohnlev xpvd_dip = NULL;
2645084Sjohnlev return (DDI_FAILURE);
2655084Sjohnlev }
2665084Sjohnlev if (ndi_event_bind_set(xpvd_ndi_event_handle, &xpvd_ndi_events,
2675084Sjohnlev NDI_SLEEP) != NDI_SUCCESS) {
2685084Sjohnlev (void) ndi_event_free_hdl(xpvd_ndi_event_handle);
2695084Sjohnlev xpvd_dip = NULL;
2705084Sjohnlev return (DDI_FAILURE);
2715084Sjohnlev }
2725084Sjohnlev
2736318Sedp #ifdef XPV_HVM_DRIVER
2746318Sedp (void) ddi_prop_update_int(DDI_DEV_T_NONE, devi, DDI_NO_AUTODETACH, 1);
2756450Srab
2766450Srab /*
2776450Srab * Report our version to dom0.
2786450Srab */
279*10175SStuart.Maybee@Sun.COM if (xenbus_printf(XBT_NULL, "guest/xpvd", "version", "%d",
2806450Srab HVMPV_XPVD_VERS))
2816450Srab cmn_err(CE_WARN, "xpvd: couldn't write version\n");
2826318Sedp #endif /* XPV_HVM_DRIVER */
2836318Sedp
2845084Sjohnlev /* watch both frontend and backend for new devices */
2855084Sjohnlev if (DOMAIN_IS_INITDOMAIN(xen_info))
2865084Sjohnlev (void) xs_register_xenbus_callback(xvdi_watch_devices);
2875084Sjohnlev else
2885084Sjohnlev xvdi_watch_devices(XENSTORE_UP);
2895084Sjohnlev
2905741Smrj xpvd_dip = devi;
2915084Sjohnlev ddi_report_dev(devi);
2925084Sjohnlev
2935084Sjohnlev return (DDI_SUCCESS);
2945084Sjohnlev }
2955084Sjohnlev
2965084Sjohnlev /*ARGSUSED*/
2975084Sjohnlev static int
xpvd_detach(dev_info_t * devi,ddi_detach_cmd_t cmd)2985084Sjohnlev xpvd_detach(dev_info_t *devi, ddi_detach_cmd_t cmd)
2995084Sjohnlev {
3005084Sjohnlev return (DDI_FAILURE);
3015084Sjohnlev }
3025084Sjohnlev
3035084Sjohnlev /*
3045084Sjohnlev * xpvd_prop_op()
3055084Sjohnlev *
3065084Sjohnlev * Query xenstore for the value of properties if DDI_PROP_NOTPROM
3075084Sjohnlev * is not set. Xenstore property values are represented as ascii strings.
3085084Sjohnlev */
3095084Sjohnlev static int
xpvd_prop_op(dev_t dev,dev_info_t * dip,dev_info_t * ch_dip,ddi_prop_op_t prop_op,int mod_flags,char * name,caddr_t valuep,int * lengthp)3105084Sjohnlev xpvd_prop_op(dev_t dev, dev_info_t *dip, dev_info_t *ch_dip,
3115084Sjohnlev ddi_prop_op_t prop_op, int mod_flags, char *name, caddr_t valuep,
3125084Sjohnlev int *lengthp)
3135084Sjohnlev {
3145084Sjohnlev caddr_t buff;
3155084Sjohnlev struct xendev_ppd *pdp;
3165084Sjohnlev void *prop_str;
3175084Sjohnlev size_t prop_len;
3185084Sjohnlev unsigned int len;
3195084Sjohnlev int rv;
3205084Sjohnlev
3215084Sjohnlev pdp = (struct xendev_ppd *)ddi_get_parent_data(ch_dip);
3225084Sjohnlev
3235084Sjohnlev if ((pdp == NULL) || !(mod_flags & (DDI_PROP_CANSLEEP)) ||
3245084Sjohnlev (mod_flags & DDI_PROP_NOTPROM) || (pdp->xd_xsdev.nodename == NULL))
3255084Sjohnlev goto toss_off;
3265084Sjohnlev /*
3275084Sjohnlev * First try reading the property off the the frontend. if that
3285084Sjohnlev * fails, try and read it from the backend node. If that
3295084Sjohnlev * also fails, pass the request on the DDI framework
3305084Sjohnlev */
3315084Sjohnlev prop_str = NULL;
3325084Sjohnlev if ((xenbus_read(XBT_NULL, pdp->xd_xsdev.nodename, name, &prop_str,
3335084Sjohnlev &len) == 0) && (prop_str != NULL) && (strlen(prop_str) != 0))
3345084Sjohnlev goto got_xs_prop;
3355084Sjohnlev
3365084Sjohnlev prop_str = NULL;
3375084Sjohnlev if ((pdp->xd_xsdev.otherend != NULL) &&
3385084Sjohnlev (xenbus_read(XBT_NULL, pdp->xd_xsdev.otherend, name, &prop_str,
3395084Sjohnlev &len) == 0) && (prop_str != NULL) && (strlen(prop_str) != 0))
3405084Sjohnlev goto got_xs_prop;
3415084Sjohnlev
3425084Sjohnlev toss_off:
3435084Sjohnlev return (ddi_bus_prop_op(dev, dip, ch_dip, prop_op,
3445084Sjohnlev mod_flags | DDI_PROP_NOTPROM, name, valuep, lengthp));
3455084Sjohnlev
3465084Sjohnlev got_xs_prop:
3475084Sjohnlev prop_len = strlen(prop_str) + 1;
3485084Sjohnlev rv = DDI_PROP_SUCCESS;
3495084Sjohnlev
3505084Sjohnlev switch (prop_op) {
3515084Sjohnlev case PROP_LEN:
3525084Sjohnlev *lengthp = prop_len;
3535084Sjohnlev break;
3545084Sjohnlev
3555084Sjohnlev case PROP_LEN_AND_VAL_ALLOC:
3565084Sjohnlev buff = kmem_alloc((size_t)prop_len, KM_SLEEP);
3575084Sjohnlev *(caddr_t *)valuep = (caddr_t)buff;
3585084Sjohnlev break;
3595084Sjohnlev case PROP_LEN_AND_VAL_BUF:
3605084Sjohnlev buff = (caddr_t)valuep;
3615084Sjohnlev if (*lengthp < prop_len)
3625084Sjohnlev rv = DDI_PROP_BUF_TOO_SMALL;
3635084Sjohnlev break;
3645084Sjohnlev default:
3655084Sjohnlev rv = DDI_PROP_INVAL_ARG;
3665084Sjohnlev break;
3675084Sjohnlev }
3685084Sjohnlev
3695084Sjohnlev if ((rv == DDI_PROP_SUCCESS) && (prop_len > 0)) {
3705084Sjohnlev bcopy(prop_str, buff, prop_len);
3715084Sjohnlev *lengthp = prop_len;
3725084Sjohnlev }
3735084Sjohnlev kmem_free(prop_str, len);
3745084Sjohnlev return (rv);
3755084Sjohnlev }
3765084Sjohnlev
3775084Sjohnlev
3785084Sjohnlev /*
3795084Sjohnlev * return address of the device's interrupt spec structure.
3805084Sjohnlev */
3815084Sjohnlev /*ARGSUSED*/
3825084Sjohnlev struct intrspec *
xpvd_get_ispec(dev_info_t * rdip,uint_t inumber)3835084Sjohnlev xpvd_get_ispec(dev_info_t *rdip, uint_t inumber)
3845084Sjohnlev {
3855084Sjohnlev struct xendev_ppd *pdp;
3865084Sjohnlev
3875084Sjohnlev ASSERT(inumber == 0);
3885084Sjohnlev
3895084Sjohnlev if ((pdp = ddi_get_parent_data(rdip)) == NULL)
3905084Sjohnlev return (NULL);
3915084Sjohnlev
3925084Sjohnlev return (&pdp->xd_ispec);
3935084Sjohnlev }
3945084Sjohnlev
3955084Sjohnlev /*
3965084Sjohnlev * return (and determine) the interrupt priority of the device.
3975084Sjohnlev */
3985084Sjohnlev /*ARGSUSED*/
3995084Sjohnlev static int
xpvd_get_priority(dev_info_t * dip,int inum,int * pri)4005084Sjohnlev xpvd_get_priority(dev_info_t *dip, int inum, int *pri)
4015084Sjohnlev {
4025084Sjohnlev struct xendev_ppd *pdp;
4035084Sjohnlev struct intrspec *ispec;
4045084Sjohnlev int *intpriorities;
4055084Sjohnlev uint_t num_intpriorities;
4065084Sjohnlev
4075084Sjohnlev DDI_INTR_NEXDBG((CE_CONT, "xpvd_get_priority: dip = 0x%p\n",
4085084Sjohnlev (void *)dip));
4095084Sjohnlev
4105084Sjohnlev ASSERT(inum == 0);
4115084Sjohnlev
4125084Sjohnlev if ((pdp = ddi_get_parent_data(dip)) == NULL)
4135084Sjohnlev return (DDI_FAILURE);
4145084Sjohnlev
4155084Sjohnlev ispec = &pdp->xd_ispec;
4165084Sjohnlev
4175084Sjohnlev /*
4185084Sjohnlev * Set the default priority based on the device class. The
4195084Sjohnlev * "interrupt-priorities" property can be used to override
4205084Sjohnlev * the default.
4215084Sjohnlev */
4225084Sjohnlev if (ispec->intrspec_pri == 0) {
4235084Sjohnlev ispec->intrspec_pri = xendev_devclass_ipl(pdp->xd_devclass);
4245084Sjohnlev if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY, dip,
4255084Sjohnlev DDI_PROP_NOTPROM | DDI_PROP_DONTPASS,
4265084Sjohnlev "interrupt-priorities", &intpriorities,
4275084Sjohnlev &num_intpriorities) == DDI_PROP_SUCCESS) {
4285084Sjohnlev ispec->intrspec_pri = intpriorities[0];
4295084Sjohnlev ddi_prop_free(intpriorities);
4305084Sjohnlev }
4315084Sjohnlev }
4325084Sjohnlev *pri = ispec->intrspec_pri;
4335084Sjohnlev return (DDI_SUCCESS);
4345084Sjohnlev }
4355084Sjohnlev
4365084Sjohnlev
4375084Sjohnlev /*
4385084Sjohnlev * xpvd_intr_ops: bus_intr_op() function for interrupt support
4395084Sjohnlev */
4405084Sjohnlev /* ARGSUSED */
4415084Sjohnlev static int
xpvd_intr_ops(dev_info_t * pdip,dev_info_t * rdip,ddi_intr_op_t intr_op,ddi_intr_handle_impl_t * hdlp,void * result)4425084Sjohnlev xpvd_intr_ops(dev_info_t *pdip, dev_info_t *rdip, ddi_intr_op_t intr_op,
4435084Sjohnlev ddi_intr_handle_impl_t *hdlp, void *result)
4445084Sjohnlev {
4455084Sjohnlev int priority = 0;
4465084Sjohnlev struct intrspec *ispec;
4475084Sjohnlev struct xendev_ppd *pdp;
4485084Sjohnlev
4495084Sjohnlev DDI_INTR_NEXDBG((CE_CONT,
4505084Sjohnlev "xpvd_intr_ops: pdip 0x%p, rdip 0x%p, op %x handle 0x%p\n",
4515084Sjohnlev (void *)pdip, (void *)rdip, intr_op, (void *)hdlp));
4525084Sjohnlev
4535084Sjohnlev /* Process the request */
4545084Sjohnlev switch (intr_op) {
4555084Sjohnlev case DDI_INTROP_SUPPORTED_TYPES:
4565084Sjohnlev /* Fixed supported by default */
4575084Sjohnlev *(int *)result = DDI_INTR_TYPE_FIXED;
4585084Sjohnlev break;
4595084Sjohnlev
4605084Sjohnlev case DDI_INTROP_NINTRS:
4615084Sjohnlev *(int *)result = 1;
4625084Sjohnlev break;
4635084Sjohnlev
4645084Sjohnlev case DDI_INTROP_ALLOC:
4655084Sjohnlev /*
4665084Sjohnlev * FIXED interrupts: just return available interrupts
4675084Sjohnlev */
4685084Sjohnlev if (hdlp->ih_type == DDI_INTR_TYPE_FIXED) {
4695084Sjohnlev /*
4705084Sjohnlev * event channels are edge-triggered, maskable,
4715084Sjohnlev * and support int pending.
4725084Sjohnlev */
4735084Sjohnlev hdlp->ih_cap |= XENDEV_INTR_CAPABILITIES;
4745084Sjohnlev *(int *)result = 1; /* DDI_INTR_TYPE_FIXED */
4755084Sjohnlev } else {
4765084Sjohnlev return (DDI_FAILURE);
4775084Sjohnlev }
4785084Sjohnlev break;
4795084Sjohnlev
4805084Sjohnlev case DDI_INTROP_FREE:
4815084Sjohnlev ispec = xpvd_get_ispec(rdip, (int)hdlp->ih_inum);
4825084Sjohnlev if (ispec == NULL)
4835084Sjohnlev return (DDI_FAILURE);
4845084Sjohnlev ispec->intrspec_pri = 0; /* mark as un-initialized */
4855084Sjohnlev break;
4865084Sjohnlev
4875084Sjohnlev case DDI_INTROP_GETPRI:
4885084Sjohnlev if (xpvd_get_priority(rdip, hdlp->ih_inum, &priority) !=
4895084Sjohnlev DDI_SUCCESS)
4905084Sjohnlev return (DDI_FAILURE);
4915084Sjohnlev DDI_INTR_NEXDBG((CE_CONT, "xpvd_intr_ops: priority = 0x%x\n",
4925084Sjohnlev priority));
4935084Sjohnlev *(int *)result = priority;
4945084Sjohnlev break;
4955084Sjohnlev
4965084Sjohnlev case DDI_INTROP_SETPRI:
4975084Sjohnlev /* Validate the interrupt priority passed */
4985084Sjohnlev if (*(int *)result > LOCK_LEVEL)
4995084Sjohnlev return (DDI_FAILURE);
5005084Sjohnlev
5015084Sjohnlev /* Ensure that PSM is all initialized */
5025084Sjohnlev if (psm_intr_ops == NULL)
5035084Sjohnlev return (DDI_FAILURE);
5045084Sjohnlev
5055084Sjohnlev /* Change the priority */
5065084Sjohnlev if ((*psm_intr_ops)(rdip, hdlp, PSM_INTR_OP_SET_PRI, result) ==
5075084Sjohnlev PSM_FAILURE)
5085084Sjohnlev return (DDI_FAILURE);
5095084Sjohnlev
5105084Sjohnlev ispec = xpvd_get_ispec(rdip, (int)hdlp->ih_inum);
5115084Sjohnlev if (ispec == NULL)
5125084Sjohnlev return (DDI_FAILURE);
5135084Sjohnlev ispec->intrspec_pri = *(int *)result;
5145084Sjohnlev break;
5155084Sjohnlev
5165084Sjohnlev case DDI_INTROP_ADDISR:
5175084Sjohnlev /* update ispec */
5185084Sjohnlev ispec = xpvd_get_ispec(rdip, (int)hdlp->ih_inum);
5195084Sjohnlev if (ispec == NULL)
5205084Sjohnlev return (DDI_FAILURE);
5215084Sjohnlev ispec->intrspec_func = hdlp->ih_cb_func;
5225084Sjohnlev
5235084Sjohnlev break;
5245084Sjohnlev
5255084Sjohnlev case DDI_INTROP_REMISR:
5265084Sjohnlev ispec = xpvd_get_ispec(rdip, (int)hdlp->ih_inum);
5275084Sjohnlev pdp = (struct xendev_ppd *)ddi_get_parent_data(rdip);
5285084Sjohnlev
5295084Sjohnlev ASSERT(pdp != NULL);
5305084Sjohnlev ASSERT(pdp->xd_evtchn != INVALID_EVTCHN);
5315084Sjohnlev
5325084Sjohnlev if (ispec) {
5335084Sjohnlev ispec->intrspec_vec = 0;
5345084Sjohnlev ispec->intrspec_func = (uint_t (*)()) 0;
5355084Sjohnlev }
5365084Sjohnlev pdp->xd_evtchn = INVALID_EVTCHN;
5375084Sjohnlev break;
5385084Sjohnlev
5395084Sjohnlev case DDI_INTROP_GETCAP:
5405084Sjohnlev if (hdlp->ih_type == DDI_INTR_TYPE_FIXED) {
5415084Sjohnlev /*
5425084Sjohnlev * event channels are edge-triggered, maskable,
5435084Sjohnlev * and support int pending.
5445084Sjohnlev */
5455084Sjohnlev *(int *)result = XENDEV_INTR_CAPABILITIES;
5465084Sjohnlev } else {
5475084Sjohnlev *(int *)result = 0;
5485084Sjohnlev return (DDI_FAILURE);
5495084Sjohnlev }
5505084Sjohnlev DDI_INTR_NEXDBG((CE_CONT, "xpvd: GETCAP returned = %x\n",
5515084Sjohnlev *(int *)result));
5525084Sjohnlev break;
5535084Sjohnlev case DDI_INTROP_SETCAP:
5545084Sjohnlev DDI_INTR_NEXDBG((CE_CONT, "xpvd_intr_ops: SETCAP cap=0x%x\n",
5555084Sjohnlev *(int *)result));
5565084Sjohnlev if (psm_intr_ops == NULL)
5575084Sjohnlev return (DDI_FAILURE);
5585084Sjohnlev
5595084Sjohnlev if ((*psm_intr_ops)(rdip, hdlp, PSM_INTR_OP_SET_CAP, result)) {
5605084Sjohnlev DDI_INTR_NEXDBG((CE_CONT, "GETCAP: psm_intr_ops"
5615084Sjohnlev " returned failure\n"));
5625084Sjohnlev return (DDI_FAILURE);
5635084Sjohnlev }
5645084Sjohnlev break;
5655084Sjohnlev
5665084Sjohnlev case DDI_INTROP_ENABLE:
5675084Sjohnlev if (psm_intr_ops == NULL)
5685084Sjohnlev return (DDI_FAILURE);
5695084Sjohnlev
5705084Sjohnlev if (xpvd_enable_intr(rdip, hdlp, (int)hdlp->ih_inum) !=
5715084Sjohnlev DDI_SUCCESS)
5725084Sjohnlev return (DDI_FAILURE);
5735084Sjohnlev
5745084Sjohnlev DDI_INTR_NEXDBG((CE_CONT, "xpvd_intr_ops: ENABLE vec=0x%x\n",
5755084Sjohnlev hdlp->ih_vector));
5765084Sjohnlev break;
5775084Sjohnlev
5785084Sjohnlev case DDI_INTROP_DISABLE:
5795084Sjohnlev if (psm_intr_ops == NULL)
5805084Sjohnlev return (DDI_FAILURE);
5815084Sjohnlev xpvd_disable_intr(rdip, hdlp, hdlp->ih_inum);
5825084Sjohnlev DDI_INTR_NEXDBG((CE_CONT, "xpvd_intr_ops: DISABLE vec = %x\n",
5835084Sjohnlev hdlp->ih_vector));
5845084Sjohnlev break;
5855084Sjohnlev
5865084Sjohnlev case DDI_INTROP_BLOCKENABLE:
5875084Sjohnlev case DDI_INTROP_BLOCKDISABLE:
5885084Sjohnlev return (DDI_FAILURE);
5895084Sjohnlev
5905084Sjohnlev case DDI_INTROP_SETMASK:
5915084Sjohnlev case DDI_INTROP_CLRMASK:
5925741Smrj #ifdef XPV_HVM_DRIVER
5935741Smrj return (DDI_ENOTSUP);
5945741Smrj #else
5955084Sjohnlev /*
5965084Sjohnlev * Handle this here
5975084Sjohnlev */
5985084Sjohnlev if (hdlp->ih_type != DDI_INTR_TYPE_FIXED)
5995084Sjohnlev return (DDI_FAILURE);
6005084Sjohnlev if (intr_op == DDI_INTROP_SETMASK) {
6015084Sjohnlev ec_disable_irq(hdlp->ih_vector);
6025084Sjohnlev } else {
6035084Sjohnlev ec_enable_irq(hdlp->ih_vector);
6045084Sjohnlev }
6055084Sjohnlev break;
6065741Smrj #endif
6075084Sjohnlev case DDI_INTROP_GETPENDING:
6085741Smrj #ifdef XPV_HVM_DRIVER
6095741Smrj return (DDI_ENOTSUP);
6105741Smrj #else
6115084Sjohnlev if (hdlp->ih_type != DDI_INTR_TYPE_FIXED)
6125084Sjohnlev return (DDI_FAILURE);
6135084Sjohnlev *(int *)result = ec_pending_irq(hdlp->ih_vector);
6145084Sjohnlev DDI_INTR_NEXDBG((CE_CONT, "xpvd: GETPENDING returned = %x\n",
6155084Sjohnlev *(int *)result));
6165084Sjohnlev break;
6175741Smrj #endif
6185084Sjohnlev
6195084Sjohnlev case DDI_INTROP_NAVAIL:
6205084Sjohnlev *(int *)result = 1;
6215084Sjohnlev DDI_INTR_NEXDBG((CE_CONT, "xpvd: NAVAIL returned = %x\n",
6225084Sjohnlev *(int *)result));
6235084Sjohnlev break;
6245084Sjohnlev
6255084Sjohnlev default:
6265084Sjohnlev return (i_ddi_intr_ops(pdip, rdip, intr_op, hdlp, result));
6275084Sjohnlev }
6285084Sjohnlev
6295084Sjohnlev return (DDI_SUCCESS);
6305084Sjohnlev }
6315084Sjohnlev
6325084Sjohnlev
6335084Sjohnlev static int
xpvd_enable_intr(dev_info_t * rdip,ddi_intr_handle_impl_t * hdlp,int inum)6345084Sjohnlev xpvd_enable_intr(dev_info_t *rdip, ddi_intr_handle_impl_t *hdlp, int inum)
6355084Sjohnlev {
6365084Sjohnlev int vector;
6375084Sjohnlev ihdl_plat_t *ihdl_plat_datap = (ihdl_plat_t *)hdlp->ih_private;
6385084Sjohnlev
6395084Sjohnlev DDI_INTR_NEXDBG((CE_CONT, "xpvd_enable_intr: hdlp %p inum %x\n",
6405084Sjohnlev (void *)hdlp, inum));
6415084Sjohnlev
6425084Sjohnlev ihdl_plat_datap->ip_ispecp = xpvd_get_ispec(rdip, inum);
6435084Sjohnlev if (ihdl_plat_datap->ip_ispecp == NULL)
6445084Sjohnlev return (DDI_FAILURE);
6455084Sjohnlev
6465084Sjohnlev /* translate the interrupt if needed */
6475084Sjohnlev (void) (*psm_intr_ops)(rdip, hdlp, PSM_INTR_OP_XLATE_VECTOR, &vector);
6485084Sjohnlev DDI_INTR_NEXDBG((CE_CONT, "xpvd_enable_intr: priority=%x vector=%x\n",
6495084Sjohnlev hdlp->ih_pri, vector));
6505084Sjohnlev
6515084Sjohnlev /* Add the interrupt handler */
6525084Sjohnlev if (!add_avintr((void *)hdlp, hdlp->ih_pri, hdlp->ih_cb_func,
6535084Sjohnlev DEVI(rdip)->devi_name, vector, hdlp->ih_cb_arg1,
6545084Sjohnlev hdlp->ih_cb_arg2, NULL, rdip))
6555084Sjohnlev return (DDI_FAILURE);
6565084Sjohnlev
6575084Sjohnlev /* Note this really is an irq. */
6585084Sjohnlev hdlp->ih_vector = (ushort_t)vector;
6595084Sjohnlev
6605084Sjohnlev return (DDI_SUCCESS);
6615084Sjohnlev }
6625084Sjohnlev
6635084Sjohnlev
6645084Sjohnlev static void
xpvd_disable_intr(dev_info_t * rdip,ddi_intr_handle_impl_t * hdlp,int inum)6655084Sjohnlev xpvd_disable_intr(dev_info_t *rdip, ddi_intr_handle_impl_t *hdlp, int inum)
6665084Sjohnlev {
6675084Sjohnlev int vector;
6685084Sjohnlev ihdl_plat_t *ihdl_plat_datap = (ihdl_plat_t *)hdlp->ih_private;
6695084Sjohnlev
6705084Sjohnlev DDI_INTR_NEXDBG((CE_CONT, "xpvd_disable_intr: \n"));
6715084Sjohnlev ihdl_plat_datap->ip_ispecp = xpvd_get_ispec(rdip, inum);
6725084Sjohnlev if (ihdl_plat_datap->ip_ispecp == NULL)
6735084Sjohnlev return;
6745084Sjohnlev
6755084Sjohnlev /* translate the interrupt if needed */
6765084Sjohnlev (void) (*psm_intr_ops)(rdip, hdlp, PSM_INTR_OP_XLATE_VECTOR, &vector);
6775084Sjohnlev
6785084Sjohnlev /* Disable the interrupt handler */
6795084Sjohnlev rem_avintr((void *)hdlp, hdlp->ih_pri, hdlp->ih_cb_func, vector);
6805084Sjohnlev ihdl_plat_datap->ip_ispecp = NULL;
6815084Sjohnlev }
6825084Sjohnlev
6835084Sjohnlev /*ARGSUSED*/
6845084Sjohnlev static int
xpvd_ctlops(dev_info_t * dip,dev_info_t * rdip,ddi_ctl_enum_t ctlop,void * arg,void * result)6855084Sjohnlev xpvd_ctlops(dev_info_t *dip, dev_info_t *rdip,
6865084Sjohnlev ddi_ctl_enum_t ctlop, void *arg, void *result)
6875084Sjohnlev {
6885084Sjohnlev switch (ctlop) {
6895084Sjohnlev case DDI_CTLOPS_REPORTDEV:
6905084Sjohnlev if (rdip == (dev_info_t *)0)
6915084Sjohnlev return (DDI_FAILURE);
6925084Sjohnlev cmn_err(CE_CONT, "?%s@%s, %s%d\n", ddi_node_name(rdip),
6935084Sjohnlev ddi_get_name_addr(rdip), ddi_driver_name(rdip),
6945084Sjohnlev ddi_get_instance(rdip));
6955084Sjohnlev return (DDI_SUCCESS);
6965084Sjohnlev
6975084Sjohnlev case DDI_CTLOPS_INITCHILD:
6985084Sjohnlev return (xpvd_initchild((dev_info_t *)arg));
6995084Sjohnlev
7005084Sjohnlev case DDI_CTLOPS_UNINITCHILD:
7015084Sjohnlev return (xpvd_removechild((dev_info_t *)arg));
7025084Sjohnlev
7035084Sjohnlev case DDI_CTLOPS_SIDDEV:
7045084Sjohnlev return (DDI_SUCCESS);
7055084Sjohnlev
7065084Sjohnlev case DDI_CTLOPS_REGSIZE:
7075084Sjohnlev case DDI_CTLOPS_NREGS:
7085084Sjohnlev return (DDI_FAILURE);
7095084Sjohnlev
7105084Sjohnlev case DDI_CTLOPS_POWER: {
7115084Sjohnlev return (ddi_ctlops(dip, rdip, ctlop, arg, result));
7125084Sjohnlev }
7135084Sjohnlev
7145084Sjohnlev default:
7155084Sjohnlev return (ddi_ctlops(dip, rdip, ctlop, arg, result));
7165084Sjohnlev }
7175084Sjohnlev
7185084Sjohnlev /* NOTREACHED */
7195084Sjohnlev
7205084Sjohnlev }
7215084Sjohnlev
7225084Sjohnlev /*
7235084Sjohnlev * Assign the address portion of the node name
7245084Sjohnlev */
7255084Sjohnlev static int
xpvd_name_child(dev_info_t * child,char * addr,int addrlen)7266318Sedp xpvd_name_child(dev_info_t *child, char *addr, int addrlen)
7275084Sjohnlev {
7285084Sjohnlev int *domain, *vdev;
7295084Sjohnlev uint_t ndomain, nvdev;
7306318Sedp char *prop_str;
7315084Sjohnlev
7325084Sjohnlev /*
7335084Sjohnlev * i_xpvd_parse_devname() knows the formats used by this
7345084Sjohnlev * routine. If this code changes, so must that.
7355084Sjohnlev */
7365084Sjohnlev
7375084Sjohnlev if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY, child, DDI_PROP_DONTPASS,
7385084Sjohnlev "domain", &domain, &ndomain) != DDI_PROP_SUCCESS)
7395084Sjohnlev return (DDI_FAILURE);
7405084Sjohnlev ASSERT(ndomain == 1);
7415084Sjohnlev
7425084Sjohnlev /*
7435084Sjohnlev * Use "domain" and "vdev" properties (backend drivers).
7445084Sjohnlev */
7455084Sjohnlev if (*domain != DOMID_SELF) {
7465084Sjohnlev if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY, child,
7475084Sjohnlev DDI_PROP_DONTPASS, "vdev", &vdev, &nvdev)
7485084Sjohnlev != DDI_PROP_SUCCESS) {
7495084Sjohnlev ddi_prop_free(domain);
7505084Sjohnlev return (DDI_FAILURE);
7515084Sjohnlev }
7525084Sjohnlev ASSERT(nvdev == 1);
7535084Sjohnlev
7546318Sedp (void) snprintf(addr, addrlen, "%d,%d", domain[0], vdev[0]);
7555084Sjohnlev ddi_prop_free(vdev);
7565084Sjohnlev ddi_prop_free(domain);
7575084Sjohnlev return (DDI_SUCCESS);
7585084Sjohnlev }
7595084Sjohnlev ddi_prop_free(domain);
7605084Sjohnlev
7615084Sjohnlev /*
7625084Sjohnlev * Use "unit-address" property (frontend/softdev drivers).
7635084Sjohnlev */
7646318Sedp if (ddi_prop_lookup_string(DDI_DEV_T_ANY, child, DDI_PROP_DONTPASS,
7656318Sedp "unit-address", &prop_str) != DDI_PROP_SUCCESS)
7666318Sedp return (DDI_FAILURE);
7676318Sedp (void) strlcpy(addr, prop_str, addrlen);
7686318Sedp ddi_prop_free(prop_str);
7696318Sedp return (DDI_SUCCESS);
7705084Sjohnlev }
7715084Sjohnlev
7725084Sjohnlev static int
xpvd_initchild(dev_info_t * child)7735084Sjohnlev xpvd_initchild(dev_info_t *child)
7745084Sjohnlev {
7756318Sedp char addr[80];
7765084Sjohnlev
7775084Sjohnlev /*
7785084Sjohnlev * Pseudo nodes indicate a prototype node with per-instance
7795084Sjohnlev * properties to be merged into the real h/w device node.
7805084Sjohnlev */
7815084Sjohnlev if (ndi_dev_is_persistent_node(child) == 0) {
7825084Sjohnlev ddi_set_parent_data(child, NULL);
7835084Sjohnlev
7845084Sjohnlev /*
7855084Sjohnlev * Try to merge the properties from this prototype
7865084Sjohnlev * node into real h/w nodes.
7875084Sjohnlev */
7885084Sjohnlev if (ndi_merge_node(child, xpvd_name_child) == DDI_SUCCESS) {
7895084Sjohnlev /*
7905084Sjohnlev * Merged ok - return failure to remove the node.
7915084Sjohnlev */
7925084Sjohnlev ddi_set_name_addr(child, NULL);
7935084Sjohnlev return (DDI_FAILURE);
7945084Sjohnlev }
7955084Sjohnlev
7965084Sjohnlev /*
7975084Sjohnlev * The child was not merged into a h/w node,
7985084Sjohnlev * but there's not much we can do with it other
7995084Sjohnlev * than return failure to cause the node to be removed.
8005084Sjohnlev */
8015084Sjohnlev cmn_err(CE_WARN, "!%s@%s: %s.conf properties not merged",
8025084Sjohnlev ddi_get_name(child), ddi_get_name_addr(child),
8035084Sjohnlev ddi_get_name(child));
8045084Sjohnlev ddi_set_name_addr(child, NULL);
8055084Sjohnlev return (DDI_NOT_WELL_FORMED);
8065084Sjohnlev }
8075084Sjohnlev
8085084Sjohnlev if (xvdi_init_dev(child) != DDI_SUCCESS)
8095084Sjohnlev return (DDI_FAILURE);
8105084Sjohnlev
8116318Sedp if (xpvd_name_child(child, addr, sizeof (addr)) != DDI_SUCCESS) {
8125084Sjohnlev xvdi_uninit_dev(child);
8135084Sjohnlev return (DDI_FAILURE);
8145084Sjohnlev }
8156318Sedp ddi_set_name_addr(child, addr);
8165084Sjohnlev
8175084Sjohnlev return (DDI_SUCCESS);
8185084Sjohnlev }
8195084Sjohnlev
8205084Sjohnlev static int
xpvd_removechild(dev_info_t * dip)8215084Sjohnlev xpvd_removechild(dev_info_t *dip)
8225084Sjohnlev {
8235084Sjohnlev xvdi_uninit_dev(dip);
8245084Sjohnlev
8255084Sjohnlev ddi_set_name_addr(dip, NULL);
8265084Sjohnlev
8275084Sjohnlev /*
8285084Sjohnlev * Strip the node to properly convert it back to prototype
8295084Sjohnlev * form.
8305084Sjohnlev */
8315084Sjohnlev ddi_remove_minor_node(dip, NULL);
8325084Sjohnlev
8335084Sjohnlev return (DDI_SUCCESS);
8345084Sjohnlev }
8355084Sjohnlev
8365084Sjohnlev static int
xpvd_bus_unconfig(dev_info_t * parent,uint_t flag,ddi_bus_config_op_t op,void * device_name)8375084Sjohnlev xpvd_bus_unconfig(dev_info_t *parent, uint_t flag, ddi_bus_config_op_t op,
8385084Sjohnlev void *device_name)
8395084Sjohnlev {
8405084Sjohnlev return (ndi_busop_bus_unconfig(parent, flag, op, device_name));
8415084Sjohnlev }
8425084Sjohnlev
8435084Sjohnlev /*
8445084Sjohnlev * Given the name of a child of xpvd, determine the device class,
8455084Sjohnlev * domain and vdevnum to which it refers.
8465084Sjohnlev */
8475084Sjohnlev static boolean_t
i_xpvd_parse_devname(char * name,xendev_devclass_t * devclassp,domid_t * domp,int * vdevp)8485084Sjohnlev i_xpvd_parse_devname(char *name, xendev_devclass_t *devclassp,
8495084Sjohnlev domid_t *domp, int *vdevp)
8505084Sjohnlev {
8515084Sjohnlev int len = strlen(name) + 1;
8525084Sjohnlev char *device_name = i_ddi_strdup(name, KM_SLEEP);
8535084Sjohnlev char *cname = NULL, *caddr = NULL;
8545084Sjohnlev boolean_t ret;
8555084Sjohnlev
8565084Sjohnlev i_ddi_parse_name(device_name, &cname, &caddr, NULL);
8575084Sjohnlev
8585084Sjohnlev if ((cname == NULL) || (strlen(cname) == 0) ||
8595084Sjohnlev (caddr == NULL) || (strlen(caddr) == 0)) {
8605084Sjohnlev ret = B_FALSE;
8615084Sjohnlev goto done;
8625084Sjohnlev }
8635084Sjohnlev
8645084Sjohnlev *devclassp = xendev_nodename_to_devclass(cname);
8655084Sjohnlev if (*devclassp < 0) {
8665084Sjohnlev ret = B_FALSE;
8675084Sjohnlev goto done;
8685084Sjohnlev }
8695084Sjohnlev
8705084Sjohnlev /*
8715084Sjohnlev * Parsing the address component requires knowledge of how
8725084Sjohnlev * xpvd_name_child() works. If that code changes, so must
8735084Sjohnlev * this.
8745084Sjohnlev */
8755084Sjohnlev
8765084Sjohnlev /* Backend format is "<domain>,<vdev>". */
8777632SNick.Todd@Sun.COM if (sscanf(caddr, "%hu,%d", domp, vdevp) == 2) {
8785084Sjohnlev ret = B_TRUE;
8795084Sjohnlev goto done;
8805084Sjohnlev }
8815084Sjohnlev
8825084Sjohnlev /* Frontend format is "<vdev>". */
8835084Sjohnlev *domp = DOMID_SELF;
8846500Sjhd if (sscanf(caddr, "%d", vdevp) == 1)
8855084Sjohnlev ret = B_TRUE;
8865084Sjohnlev done:
8875084Sjohnlev kmem_free(device_name, len);
8885084Sjohnlev return (ret);
8895084Sjohnlev }
8905084Sjohnlev
8915084Sjohnlev /*
8925084Sjohnlev * xpvd_bus_config()
8935084Sjohnlev *
8945084Sjohnlev * BUS_CONFIG_ONE:
8955084Sjohnlev * Enumerate the exact instance of a driver.
8965084Sjohnlev *
8975084Sjohnlev * BUS_CONFIG_ALL:
8985084Sjohnlev * Enumerate all the instances of all the possible children (seen before
8995084Sjohnlev * and never seen before).
9005084Sjohnlev *
9015084Sjohnlev * BUS_CONFIG_DRIVER:
9025084Sjohnlev * Enumerate all the instances of a particular driver.
9035084Sjohnlev */
9045084Sjohnlev static int
xpvd_bus_config(dev_info_t * parent,uint_t flag,ddi_bus_config_op_t op,void * arg,dev_info_t ** childp)9055084Sjohnlev xpvd_bus_config(dev_info_t *parent, uint_t flag, ddi_bus_config_op_t op,
9065084Sjohnlev void *arg, dev_info_t **childp)
9075084Sjohnlev {
9085084Sjohnlev int circ;
9095084Sjohnlev char *cname = NULL;
9105084Sjohnlev
9115084Sjohnlev ndi_devi_enter(parent, &circ);
9125084Sjohnlev
9135084Sjohnlev switch (op) {
9145084Sjohnlev case BUS_CONFIG_ONE: {
9155084Sjohnlev xendev_devclass_t devclass;
9165084Sjohnlev domid_t dom;
9175084Sjohnlev int vdev;
9185084Sjohnlev
9195084Sjohnlev if (!i_xpvd_parse_devname(arg, &devclass, &dom, &vdev)) {
9205084Sjohnlev ndi_devi_exit(parent, circ);
9215084Sjohnlev return (NDI_FAILURE);
9225084Sjohnlev }
9235084Sjohnlev
9245084Sjohnlev *childp = xvdi_find_dev(parent, devclass, dom, vdev);
9255084Sjohnlev if (*childp == NULL)
9265084Sjohnlev *childp = xvdi_create_dev(parent, devclass, dom, vdev);
9275084Sjohnlev
9285084Sjohnlev ndi_devi_exit(parent, circ);
9295084Sjohnlev
9305084Sjohnlev if (*childp == NULL)
9315084Sjohnlev return (NDI_FAILURE);
9325084Sjohnlev else
9335084Sjohnlev return (ndi_busop_bus_config(parent, flag,
9345084Sjohnlev op, arg, childp, 0));
9355084Sjohnlev }
9365084Sjohnlev
9375084Sjohnlev case BUS_CONFIG_DRIVER: {
9385084Sjohnlev xendev_devclass_t devclass = XEN_INVAL;
9395084Sjohnlev
9405084Sjohnlev cname = ddi_major_to_name((major_t)(uintptr_t)arg);
9415084Sjohnlev if (cname != NULL)
9425084Sjohnlev devclass = xendev_nodename_to_devclass(cname);
9435084Sjohnlev
9445084Sjohnlev if (devclass == XEN_INVAL) {
9455084Sjohnlev ndi_devi_exit(parent, circ);
9465084Sjohnlev return (NDI_FAILURE);
9475084Sjohnlev } else {
9485084Sjohnlev xendev_enum_class(parent, devclass);
9495084Sjohnlev ndi_devi_exit(parent, circ);
9505084Sjohnlev return (ndi_busop_bus_config(parent, flag, op,
9515084Sjohnlev arg, childp, 0));
9525084Sjohnlev }
9535084Sjohnlev /* NOTREACHED */
9545084Sjohnlev }
9555084Sjohnlev
9565084Sjohnlev case BUS_CONFIG_ALL:
9575084Sjohnlev xendev_enum_all(parent, B_FALSE);
9585084Sjohnlev ndi_devi_exit(parent, circ);
9595084Sjohnlev
9605084Sjohnlev return (ndi_busop_bus_config(parent, flag, op,
9615084Sjohnlev arg, childp, 0));
9625084Sjohnlev
9635084Sjohnlev default:
9645084Sjohnlev ndi_devi_exit(parent, circ);
9655084Sjohnlev return (NDI_FAILURE);
9665084Sjohnlev }
9675084Sjohnlev }
9685084Sjohnlev
9695084Sjohnlev /*ARGSUSED*/
9705084Sjohnlev static int
xpvd_get_eventcookie(dev_info_t * dip,dev_info_t * rdip,char * eventname,ddi_eventcookie_t * cookie)9715084Sjohnlev xpvd_get_eventcookie(dev_info_t *dip, dev_info_t *rdip,
9725084Sjohnlev char *eventname, ddi_eventcookie_t *cookie)
9735084Sjohnlev {
9745084Sjohnlev return (ndi_event_retrieve_cookie(xpvd_ndi_event_handle,
9755084Sjohnlev rdip, eventname, cookie, NDI_EVENT_NOPASS));
9765084Sjohnlev }
9775084Sjohnlev
9785084Sjohnlev /*ARGSUSED*/
9795084Sjohnlev static int
xpvd_add_eventcall(dev_info_t * dip,dev_info_t * rdip,ddi_eventcookie_t cookie,void (* callback)(dev_info_t * dip,ddi_eventcookie_t cookie,void * arg,void * bus_impldata),void * arg,ddi_callback_id_t * cb_id)9805084Sjohnlev xpvd_add_eventcall(dev_info_t *dip, dev_info_t *rdip,
9815084Sjohnlev ddi_eventcookie_t cookie, void (*callback)(dev_info_t *dip,
9825084Sjohnlev ddi_eventcookie_t cookie, void *arg, void *bus_impldata),
9835084Sjohnlev void *arg, ddi_callback_id_t *cb_id)
9845084Sjohnlev {
9855084Sjohnlev return (ndi_event_add_callback(xpvd_ndi_event_handle,
9865084Sjohnlev rdip, cookie, callback, arg, NDI_SLEEP, cb_id));
9875084Sjohnlev }
9885084Sjohnlev
9895084Sjohnlev /*ARGSUSED*/
9905084Sjohnlev static int
xpvd_remove_eventcall(dev_info_t * dip,ddi_callback_id_t cb_id)9915084Sjohnlev xpvd_remove_eventcall(dev_info_t *dip, ddi_callback_id_t cb_id)
9925084Sjohnlev {
9935084Sjohnlev return (ndi_event_remove_callback(xpvd_ndi_event_handle,
9945084Sjohnlev cb_id));
9955084Sjohnlev }
9965084Sjohnlev
9975084Sjohnlev /*ARGSUSED*/
9985084Sjohnlev static int
xpvd_post_event(dev_info_t * dip,dev_info_t * rdip,ddi_eventcookie_t cookie,void * bus_impldata)9995084Sjohnlev xpvd_post_event(dev_info_t *dip, dev_info_t *rdip,
10005084Sjohnlev ddi_eventcookie_t cookie, void *bus_impldata)
10015084Sjohnlev {
10025084Sjohnlev return (ndi_event_run_callbacks(xpvd_ndi_event_handle, rdip,
10035084Sjohnlev cookie, bus_impldata));
10045084Sjohnlev }
1005