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