1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 22 /* 23 * Copyright 2009 Sun Microsystems, Inc. All rights reserved. 24 * Use is subject to license terms. 25 */ 26 /* 27 * Copyright 2012 Garrett D'Amore <garrett@damore.org>. All rights reserved. 28 */ 29 30 /* 31 * Host to hypervisor virtual devices nexus driver 32 * 33 * TODO: 34 * - Add watchpoints on vbd/vif and enumerate/offline on watch callback 35 * - Add DR IOCTLs 36 * - Filter/restrict property lookups into xenstore 37 */ 38 39 #include <sys/conf.h> 40 #include <sys/kmem.h> 41 #include <sys/debug.h> 42 #include <sys/modctl.h> 43 #include <sys/autoconf.h> 44 #include <sys/ddi_impldefs.h> 45 #include <sys/ddi_subrdefs.h> 46 #include <sys/ddi.h> 47 #include <sys/sunddi.h> 48 #include <sys/sunndi.h> 49 #include <sys/avintr.h> 50 #include <sys/psm.h> 51 #include <sys/spl.h> 52 #include <sys/promif.h> 53 #include <sys/list.h> 54 #include <sys/bootconf.h> 55 #include <sys/bootsvcs.h> 56 #include <util/sscanf.h> 57 #include <sys/mach_intr.h> 58 #include <sys/bootinfo.h> 59 #ifdef XPV_HVM_DRIVER 60 #include <sys/xpv_support.h> 61 #include <sys/hypervisor.h> 62 #include <sys/archsystm.h> 63 #include <sys/cpu.h> 64 #include <public/xen.h> 65 #include <public/event_channel.h> 66 #include <public/io/xenbus.h> 67 #else 68 #include <sys/hypervisor.h> 69 #include <sys/evtchn_impl.h> 70 #include <sys/xen_mmu.h> 71 #endif 72 #include <xen/sys/xenbus_impl.h> 73 #include <xen/sys/xendev.h> 74 75 /* 76 * DDI dev_ops entrypoints 77 */ 78 static int xpvd_info(dev_info_t *, ddi_info_cmd_t, void *, void **); 79 static int xpvd_attach(dev_info_t *devi, ddi_attach_cmd_t cmd); 80 static int xpvd_detach(dev_info_t *devi, ddi_detach_cmd_t cmd); 81 82 83 /* 84 * NDI bus_ops entrypoints 85 */ 86 static int xpvd_ctlops(dev_info_t *, dev_info_t *, ddi_ctl_enum_t, void *, 87 void *); 88 static int xpvd_intr_ops(dev_info_t *, dev_info_t *, ddi_intr_op_t, 89 ddi_intr_handle_impl_t *, void *); 90 static int xpvd_prop_op(dev_t, dev_info_t *, dev_info_t *, ddi_prop_op_t, 91 int, char *, caddr_t, int *); 92 static int xpvd_bus_config(dev_info_t *, uint_t, ddi_bus_config_op_t, 93 void *, dev_info_t **); 94 static int xpvd_bus_unconfig(dev_info_t *, uint_t, ddi_bus_config_op_t, 95 void *); 96 static int xpvd_get_eventcookie(dev_info_t *, dev_info_t *, 97 char *, ddi_eventcookie_t *); 98 static int xpvd_add_eventcall(dev_info_t *, dev_info_t *, 99 ddi_eventcookie_t, void (*)(dev_info_t *, 100 ddi_eventcookie_t, void *, void *), 101 void *, ddi_callback_id_t *); 102 static int xpvd_remove_eventcall(dev_info_t *, ddi_callback_id_t); 103 static int xpvd_post_event(dev_info_t *, dev_info_t *, 104 ddi_eventcookie_t, void *); 105 106 /* 107 * misc functions 108 */ 109 static int xpvd_enable_intr(dev_info_t *, ddi_intr_handle_impl_t *, int); 110 static void xpvd_disable_intr(dev_info_t *, ddi_intr_handle_impl_t *, int); 111 static int xpvd_removechild(dev_info_t *); 112 static int xpvd_initchild(dev_info_t *); 113 static int xpvd_name_child(dev_info_t *, char *, int); 114 static boolean_t i_xpvd_parse_devname(char *, xendev_devclass_t *, 115 domid_t *, int *); 116 117 118 /* Extern declarations */ 119 extern int (*psm_intr_ops)(dev_info_t *, ddi_intr_handle_impl_t *, 120 psm_intr_op_t, int *); 121 122 struct bus_ops xpvd_bus_ops = { 123 BUSO_REV, 124 i_ddi_bus_map, 125 NULL, 126 NULL, 127 NULL, 128 i_ddi_map_fault, 129 NULL, 130 ddi_dma_allochdl, 131 ddi_dma_freehdl, 132 ddi_dma_bindhdl, 133 ddi_dma_unbindhdl, 134 ddi_dma_flush, 135 ddi_dma_win, 136 ddi_dma_mctl, 137 xpvd_ctlops, 138 xpvd_prop_op, 139 xpvd_get_eventcookie, 140 xpvd_add_eventcall, 141 xpvd_remove_eventcall, 142 xpvd_post_event, 143 0, /* (*bus_intr_ctl)(); */ 144 xpvd_bus_config, 145 xpvd_bus_unconfig, 146 NULL, /* (*bus_fm_init)(); */ 147 NULL, /* (*bus_fm_fini)(); */ 148 NULL, /* (*bus_fm_access_enter)(); */ 149 NULL, /* (*bus_fm_access_exit)(); */ 150 NULL, /* (*bus_power)(); */ 151 xpvd_intr_ops /* (*bus_intr_op)(); */ 152 }; 153 154 struct dev_ops xpvd_ops = { 155 DEVO_REV, /* devo_rev */ 156 0, /* refcnt */ 157 xpvd_info, /* info */ 158 nulldev, /* identify */ 159 nulldev, /* probe */ 160 xpvd_attach, /* attach */ 161 xpvd_detach, /* detach */ 162 nulldev, /* reset */ 163 (struct cb_ops *)0, /* driver operations */ 164 &xpvd_bus_ops, /* bus operations */ 165 NULL, /* power */ 166 ddi_quiesce_not_needed, /* quiesce */ 167 }; 168 169 170 dev_info_t *xpvd_dip; 171 172 #define CF_DBG 0x1 173 #define ALL_DBG 0xff 174 175 static ndi_event_definition_t xpvd_ndi_event_defs[] = { 176 { 0, XS_OE_STATE, EPL_KERNEL, NDI_EVENT_POST_TO_TGT }, 177 { 1, XS_HP_STATE, EPL_KERNEL, NDI_EVENT_POST_TO_TGT }, 178 }; 179 180 #define XENDEV_N_NDI_EVENTS \ 181 (sizeof (xpvd_ndi_event_defs) / sizeof (xpvd_ndi_event_defs[0])) 182 183 static ndi_event_set_t xpvd_ndi_events = { 184 NDI_EVENTS_REV1, XENDEV_N_NDI_EVENTS, xpvd_ndi_event_defs 185 }; 186 187 static ndi_event_hdl_t xpvd_ndi_event_handle; 188 189 /* 190 * Hypervisor interrupt capabilities 191 */ 192 #define XENDEV_INTR_CAPABILITIES \ 193 (DDI_INTR_FLAG_EDGE | DDI_INTR_FLAG_MASKABLE | DDI_INTR_FLAG_PENDING) 194 195 /* 196 * Module linkage information for the kernel. 197 */ 198 199 static struct modldrv modldrv = { 200 &mod_driverops, /* Type of module */ 201 "virtual device nexus driver", 202 &xpvd_ops, /* driver ops */ 203 }; 204 205 static struct modlinkage modlinkage = { 206 MODREV_1, 207 (void *)&modldrv, 208 NULL 209 }; 210 211 int 212 _init(void) 213 { 214 return (mod_install(&modlinkage)); 215 } 216 217 int 218 _fini(void) 219 { 220 return (mod_remove(&modlinkage)); 221 } 222 223 int 224 _info(struct modinfo *modinfop) 225 { 226 return (mod_info(&modlinkage, modinfop)); 227 } 228 229 /* ARGSUSED */ 230 static int 231 xpvd_info(dev_info_t *dip, ddi_info_cmd_t cmd, void *arg, void **result) 232 { 233 switch (cmd) { 234 default: 235 return (DDI_FAILURE); 236 237 case DDI_INFO_DEVT2INSTANCE: 238 *result = (void *)0; 239 return (DDI_SUCCESS); 240 241 case DDI_INFO_DEVT2DEVINFO: 242 *result = (void *)xpvd_dip; 243 return (DDI_SUCCESS); 244 } 245 } 246 247 /*ARGSUSED*/ 248 static int 249 xpvd_attach(dev_info_t *devi, ddi_attach_cmd_t cmd) 250 { 251 extern void xvdi_watch_devices(int); 252 #ifdef XPV_HVM_DRIVER 253 extern dev_info_t *xpv_dip; 254 255 if (xpv_dip == NULL) { 256 if (ddi_hold_installed_driver(ddi_name_to_major("xpv")) == 257 NULL) { 258 cmn_err(CE_WARN, "Couldn't initialize xpv framework"); 259 return (DDI_FAILURE); 260 } 261 } 262 #endif /* XPV_HVM_DRIVER */ 263 264 if (ndi_event_alloc_hdl(devi, 0, &xpvd_ndi_event_handle, 265 NDI_SLEEP) != NDI_SUCCESS) { 266 xpvd_dip = NULL; 267 return (DDI_FAILURE); 268 } 269 if (ndi_event_bind_set(xpvd_ndi_event_handle, &xpvd_ndi_events, 270 NDI_SLEEP) != NDI_SUCCESS) { 271 (void) ndi_event_free_hdl(xpvd_ndi_event_handle); 272 xpvd_dip = NULL; 273 return (DDI_FAILURE); 274 } 275 276 #ifdef XPV_HVM_DRIVER 277 (void) ddi_prop_update_int(DDI_DEV_T_NONE, devi, DDI_NO_AUTODETACH, 1); 278 279 /* 280 * Report our version to dom0. 281 */ 282 if (xenbus_printf(XBT_NULL, "guest/xpvd", "version", "%d", 283 HVMPV_XPVD_VERS)) 284 cmn_err(CE_WARN, "xpvd: couldn't write version\n"); 285 #endif /* XPV_HVM_DRIVER */ 286 287 /* watch both frontend and backend for new devices */ 288 if (DOMAIN_IS_INITDOMAIN(xen_info)) 289 (void) xs_register_xenbus_callback(xvdi_watch_devices); 290 else 291 xvdi_watch_devices(XENSTORE_UP); 292 293 xpvd_dip = devi; 294 ddi_report_dev(devi); 295 296 return (DDI_SUCCESS); 297 } 298 299 /*ARGSUSED*/ 300 static int 301 xpvd_detach(dev_info_t *devi, ddi_detach_cmd_t cmd) 302 { 303 return (DDI_FAILURE); 304 } 305 306 /* 307 * xpvd_prop_op() 308 * 309 * Query xenstore for the value of properties if DDI_PROP_NOTPROM 310 * is not set. Xenstore property values are represented as ascii strings. 311 */ 312 static int 313 xpvd_prop_op(dev_t dev, dev_info_t *dip, dev_info_t *ch_dip, 314 ddi_prop_op_t prop_op, int mod_flags, char *name, caddr_t valuep, 315 int *lengthp) 316 { 317 caddr_t buff; 318 struct xendev_ppd *pdp; 319 void *prop_str; 320 size_t prop_len; 321 unsigned int len; 322 int rv; 323 324 pdp = (struct xendev_ppd *)ddi_get_parent_data(ch_dip); 325 326 if ((pdp == NULL) || !(mod_flags & (DDI_PROP_CANSLEEP)) || 327 (mod_flags & DDI_PROP_NOTPROM) || (pdp->xd_xsdev.nodename == NULL)) 328 goto toss_off; 329 /* 330 * First try reading the property off the the frontend. if that 331 * fails, try and read it from the backend node. If that 332 * also fails, pass the request on the DDI framework 333 */ 334 prop_str = NULL; 335 if ((xenbus_read(XBT_NULL, pdp->xd_xsdev.nodename, name, &prop_str, 336 &len) == 0) && (prop_str != NULL) && (strlen(prop_str) != 0)) 337 goto got_xs_prop; 338 339 prop_str = NULL; 340 if ((pdp->xd_xsdev.otherend != NULL) && 341 (xenbus_read(XBT_NULL, pdp->xd_xsdev.otherend, name, &prop_str, 342 &len) == 0) && (prop_str != NULL) && (strlen(prop_str) != 0)) 343 goto got_xs_prop; 344 345 toss_off: 346 return (ddi_bus_prop_op(dev, dip, ch_dip, prop_op, 347 mod_flags | DDI_PROP_NOTPROM, name, valuep, lengthp)); 348 349 got_xs_prop: 350 prop_len = strlen(prop_str) + 1; 351 rv = DDI_PROP_SUCCESS; 352 353 switch (prop_op) { 354 case PROP_LEN: 355 *lengthp = prop_len; 356 break; 357 358 case PROP_LEN_AND_VAL_ALLOC: 359 buff = kmem_alloc((size_t)prop_len, KM_SLEEP); 360 *(caddr_t *)valuep = (caddr_t)buff; 361 break; 362 case PROP_LEN_AND_VAL_BUF: 363 buff = (caddr_t)valuep; 364 if (*lengthp < prop_len) 365 rv = DDI_PROP_BUF_TOO_SMALL; 366 break; 367 default: 368 rv = DDI_PROP_INVAL_ARG; 369 break; 370 } 371 372 if ((rv == DDI_PROP_SUCCESS) && (prop_len > 0)) { 373 bcopy(prop_str, buff, prop_len); 374 *lengthp = prop_len; 375 } 376 kmem_free(prop_str, len); 377 return (rv); 378 } 379 380 381 /* 382 * return address of the device's interrupt spec structure. 383 */ 384 /*ARGSUSED*/ 385 struct intrspec * 386 xpvd_get_ispec(dev_info_t *rdip, uint_t inumber) 387 { 388 struct xendev_ppd *pdp; 389 390 ASSERT(inumber == 0); 391 392 if ((pdp = ddi_get_parent_data(rdip)) == NULL) 393 return (NULL); 394 395 return (&pdp->xd_ispec); 396 } 397 398 /* 399 * return (and determine) the interrupt priority of the device. 400 */ 401 /*ARGSUSED*/ 402 static int 403 xpvd_get_priority(dev_info_t *dip, int inum, int *pri) 404 { 405 struct xendev_ppd *pdp; 406 struct intrspec *ispec; 407 int *intpriorities; 408 uint_t num_intpriorities; 409 410 DDI_INTR_NEXDBG((CE_CONT, "xpvd_get_priority: dip = 0x%p\n", 411 (void *)dip)); 412 413 ASSERT(inum == 0); 414 415 if ((pdp = ddi_get_parent_data(dip)) == NULL) 416 return (DDI_FAILURE); 417 418 ispec = &pdp->xd_ispec; 419 420 /* 421 * Set the default priority based on the device class. The 422 * "interrupt-priorities" property can be used to override 423 * the default. 424 */ 425 if (ispec->intrspec_pri == 0) { 426 ispec->intrspec_pri = xendev_devclass_ipl(pdp->xd_devclass); 427 if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY, dip, 428 DDI_PROP_NOTPROM | DDI_PROP_DONTPASS, 429 "interrupt-priorities", &intpriorities, 430 &num_intpriorities) == DDI_PROP_SUCCESS) { 431 ispec->intrspec_pri = intpriorities[0]; 432 ddi_prop_free(intpriorities); 433 } 434 } 435 *pri = ispec->intrspec_pri; 436 return (DDI_SUCCESS); 437 } 438 439 440 /* 441 * xpvd_intr_ops: bus_intr_op() function for interrupt support 442 */ 443 /* ARGSUSED */ 444 static int 445 xpvd_intr_ops(dev_info_t *pdip, dev_info_t *rdip, ddi_intr_op_t intr_op, 446 ddi_intr_handle_impl_t *hdlp, void *result) 447 { 448 int priority = 0; 449 struct intrspec *ispec; 450 struct xendev_ppd *pdp; 451 452 DDI_INTR_NEXDBG((CE_CONT, 453 "xpvd_intr_ops: pdip 0x%p, rdip 0x%p, op %x handle 0x%p\n", 454 (void *)pdip, (void *)rdip, intr_op, (void *)hdlp)); 455 456 /* Process the request */ 457 switch (intr_op) { 458 case DDI_INTROP_SUPPORTED_TYPES: 459 /* Fixed supported by default */ 460 *(int *)result = DDI_INTR_TYPE_FIXED; 461 break; 462 463 case DDI_INTROP_NINTRS: 464 *(int *)result = 1; 465 break; 466 467 case DDI_INTROP_ALLOC: 468 /* 469 * FIXED interrupts: just return available interrupts 470 */ 471 if (hdlp->ih_type == DDI_INTR_TYPE_FIXED) { 472 /* 473 * event channels are edge-triggered, maskable, 474 * and support int pending. 475 */ 476 hdlp->ih_cap |= XENDEV_INTR_CAPABILITIES; 477 *(int *)result = 1; /* DDI_INTR_TYPE_FIXED */ 478 } else { 479 return (DDI_FAILURE); 480 } 481 break; 482 483 case DDI_INTROP_FREE: 484 ispec = xpvd_get_ispec(rdip, (int)hdlp->ih_inum); 485 if (ispec == NULL) 486 return (DDI_FAILURE); 487 ispec->intrspec_pri = 0; /* mark as un-initialized */ 488 break; 489 490 case DDI_INTROP_GETPRI: 491 if (xpvd_get_priority(rdip, hdlp->ih_inum, &priority) != 492 DDI_SUCCESS) 493 return (DDI_FAILURE); 494 DDI_INTR_NEXDBG((CE_CONT, "xpvd_intr_ops: priority = 0x%x\n", 495 priority)); 496 *(int *)result = priority; 497 break; 498 499 case DDI_INTROP_SETPRI: 500 /* Validate the interrupt priority passed */ 501 if (*(int *)result > LOCK_LEVEL) 502 return (DDI_FAILURE); 503 504 /* Ensure that PSM is all initialized */ 505 if (psm_intr_ops == NULL) 506 return (DDI_FAILURE); 507 508 /* Change the priority */ 509 if ((*psm_intr_ops)(rdip, hdlp, PSM_INTR_OP_SET_PRI, result) == 510 PSM_FAILURE) 511 return (DDI_FAILURE); 512 513 ispec = xpvd_get_ispec(rdip, (int)hdlp->ih_inum); 514 if (ispec == NULL) 515 return (DDI_FAILURE); 516 ispec->intrspec_pri = *(int *)result; 517 break; 518 519 case DDI_INTROP_ADDISR: 520 /* update ispec */ 521 ispec = xpvd_get_ispec(rdip, (int)hdlp->ih_inum); 522 if (ispec == NULL) 523 return (DDI_FAILURE); 524 ispec->intrspec_func = hdlp->ih_cb_func; 525 526 break; 527 528 case DDI_INTROP_REMISR: 529 ispec = xpvd_get_ispec(rdip, (int)hdlp->ih_inum); 530 pdp = (struct xendev_ppd *)ddi_get_parent_data(rdip); 531 532 ASSERT(pdp != NULL); 533 ASSERT(pdp->xd_evtchn != INVALID_EVTCHN); 534 535 if (ispec) { 536 ispec->intrspec_vec = 0; 537 ispec->intrspec_func = (uint_t (*)()) 0; 538 } 539 pdp->xd_evtchn = INVALID_EVTCHN; 540 break; 541 542 case DDI_INTROP_GETCAP: 543 if (hdlp->ih_type == DDI_INTR_TYPE_FIXED) { 544 /* 545 * event channels are edge-triggered, maskable, 546 * and support int pending. 547 */ 548 *(int *)result = XENDEV_INTR_CAPABILITIES; 549 } else { 550 *(int *)result = 0; 551 return (DDI_FAILURE); 552 } 553 DDI_INTR_NEXDBG((CE_CONT, "xpvd: GETCAP returned = %x\n", 554 *(int *)result)); 555 break; 556 case DDI_INTROP_SETCAP: 557 DDI_INTR_NEXDBG((CE_CONT, "xpvd_intr_ops: SETCAP cap=0x%x\n", 558 *(int *)result)); 559 if (psm_intr_ops == NULL) 560 return (DDI_FAILURE); 561 562 if ((*psm_intr_ops)(rdip, hdlp, PSM_INTR_OP_SET_CAP, result)) { 563 DDI_INTR_NEXDBG((CE_CONT, "GETCAP: psm_intr_ops" 564 " returned failure\n")); 565 return (DDI_FAILURE); 566 } 567 break; 568 569 case DDI_INTROP_ENABLE: 570 if (psm_intr_ops == NULL) 571 return (DDI_FAILURE); 572 573 if (xpvd_enable_intr(rdip, hdlp, (int)hdlp->ih_inum) != 574 DDI_SUCCESS) 575 return (DDI_FAILURE); 576 577 DDI_INTR_NEXDBG((CE_CONT, "xpvd_intr_ops: ENABLE vec=0x%x\n", 578 hdlp->ih_vector)); 579 break; 580 581 case DDI_INTROP_DISABLE: 582 if (psm_intr_ops == NULL) 583 return (DDI_FAILURE); 584 xpvd_disable_intr(rdip, hdlp, hdlp->ih_inum); 585 DDI_INTR_NEXDBG((CE_CONT, "xpvd_intr_ops: DISABLE vec = %x\n", 586 hdlp->ih_vector)); 587 break; 588 589 case DDI_INTROP_BLOCKENABLE: 590 case DDI_INTROP_BLOCKDISABLE: 591 return (DDI_FAILURE); 592 593 case DDI_INTROP_SETMASK: 594 case DDI_INTROP_CLRMASK: 595 #ifdef XPV_HVM_DRIVER 596 return (DDI_ENOTSUP); 597 #else 598 /* 599 * Handle this here 600 */ 601 if (hdlp->ih_type != DDI_INTR_TYPE_FIXED) 602 return (DDI_FAILURE); 603 if (intr_op == DDI_INTROP_SETMASK) { 604 ec_disable_irq(hdlp->ih_vector); 605 } else { 606 ec_enable_irq(hdlp->ih_vector); 607 } 608 break; 609 #endif 610 case DDI_INTROP_GETPENDING: 611 #ifdef XPV_HVM_DRIVER 612 return (DDI_ENOTSUP); 613 #else 614 if (hdlp->ih_type != DDI_INTR_TYPE_FIXED) 615 return (DDI_FAILURE); 616 *(int *)result = ec_pending_irq(hdlp->ih_vector); 617 DDI_INTR_NEXDBG((CE_CONT, "xpvd: GETPENDING returned = %x\n", 618 *(int *)result)); 619 break; 620 #endif 621 622 case DDI_INTROP_NAVAIL: 623 *(int *)result = 1; 624 DDI_INTR_NEXDBG((CE_CONT, "xpvd: NAVAIL returned = %x\n", 625 *(int *)result)); 626 break; 627 628 default: 629 return (i_ddi_intr_ops(pdip, rdip, intr_op, hdlp, result)); 630 } 631 632 return (DDI_SUCCESS); 633 } 634 635 636 static int 637 xpvd_enable_intr(dev_info_t *rdip, ddi_intr_handle_impl_t *hdlp, int inum) 638 { 639 int vector; 640 ihdl_plat_t *ihdl_plat_datap = (ihdl_plat_t *)hdlp->ih_private; 641 642 DDI_INTR_NEXDBG((CE_CONT, "xpvd_enable_intr: hdlp %p inum %x\n", 643 (void *)hdlp, inum)); 644 645 ihdl_plat_datap->ip_ispecp = xpvd_get_ispec(rdip, inum); 646 if (ihdl_plat_datap->ip_ispecp == NULL) 647 return (DDI_FAILURE); 648 649 /* translate the interrupt if needed */ 650 (void) (*psm_intr_ops)(rdip, hdlp, PSM_INTR_OP_XLATE_VECTOR, &vector); 651 DDI_INTR_NEXDBG((CE_CONT, "xpvd_enable_intr: priority=%x vector=%x\n", 652 hdlp->ih_pri, vector)); 653 654 /* Add the interrupt handler */ 655 if (!add_avintr((void *)hdlp, hdlp->ih_pri, hdlp->ih_cb_func, 656 DEVI(rdip)->devi_name, vector, hdlp->ih_cb_arg1, 657 hdlp->ih_cb_arg2, NULL, rdip)) 658 return (DDI_FAILURE); 659 660 /* Note this really is an irq. */ 661 hdlp->ih_vector = (ushort_t)vector; 662 663 return (DDI_SUCCESS); 664 } 665 666 667 static void 668 xpvd_disable_intr(dev_info_t *rdip, ddi_intr_handle_impl_t *hdlp, int inum) 669 { 670 int vector; 671 ihdl_plat_t *ihdl_plat_datap = (ihdl_plat_t *)hdlp->ih_private; 672 673 DDI_INTR_NEXDBG((CE_CONT, "xpvd_disable_intr: \n")); 674 ihdl_plat_datap->ip_ispecp = xpvd_get_ispec(rdip, inum); 675 if (ihdl_plat_datap->ip_ispecp == NULL) 676 return; 677 678 /* translate the interrupt if needed */ 679 (void) (*psm_intr_ops)(rdip, hdlp, PSM_INTR_OP_XLATE_VECTOR, &vector); 680 681 /* Disable the interrupt handler */ 682 rem_avintr((void *)hdlp, hdlp->ih_pri, hdlp->ih_cb_func, vector); 683 ihdl_plat_datap->ip_ispecp = NULL; 684 } 685 686 /*ARGSUSED*/ 687 static int 688 xpvd_ctlops(dev_info_t *dip, dev_info_t *rdip, 689 ddi_ctl_enum_t ctlop, void *arg, void *result) 690 { 691 switch (ctlop) { 692 case DDI_CTLOPS_REPORTDEV: 693 if (rdip == (dev_info_t *)0) 694 return (DDI_FAILURE); 695 cmn_err(CE_CONT, "?%s@%s, %s%d\n", ddi_node_name(rdip), 696 ddi_get_name_addr(rdip), ddi_driver_name(rdip), 697 ddi_get_instance(rdip)); 698 return (DDI_SUCCESS); 699 700 case DDI_CTLOPS_INITCHILD: 701 return (xpvd_initchild((dev_info_t *)arg)); 702 703 case DDI_CTLOPS_UNINITCHILD: 704 return (xpvd_removechild((dev_info_t *)arg)); 705 706 case DDI_CTLOPS_SIDDEV: 707 return (DDI_SUCCESS); 708 709 case DDI_CTLOPS_REGSIZE: 710 case DDI_CTLOPS_NREGS: 711 return (DDI_FAILURE); 712 713 case DDI_CTLOPS_POWER: { 714 return (ddi_ctlops(dip, rdip, ctlop, arg, result)); 715 } 716 717 default: 718 return (ddi_ctlops(dip, rdip, ctlop, arg, result)); 719 } 720 721 /* NOTREACHED */ 722 723 } 724 725 /* 726 * Assign the address portion of the node name 727 */ 728 static int 729 xpvd_name_child(dev_info_t *child, char *addr, int addrlen) 730 { 731 int *domain, *vdev; 732 uint_t ndomain, nvdev; 733 char *prop_str; 734 735 /* 736 * i_xpvd_parse_devname() knows the formats used by this 737 * routine. If this code changes, so must that. 738 */ 739 740 if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY, child, DDI_PROP_DONTPASS, 741 "domain", &domain, &ndomain) != DDI_PROP_SUCCESS) 742 return (DDI_FAILURE); 743 ASSERT(ndomain == 1); 744 745 /* 746 * Use "domain" and "vdev" properties (backend drivers). 747 */ 748 if (*domain != DOMID_SELF) { 749 if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY, child, 750 DDI_PROP_DONTPASS, "vdev", &vdev, &nvdev) 751 != DDI_PROP_SUCCESS) { 752 ddi_prop_free(domain); 753 return (DDI_FAILURE); 754 } 755 ASSERT(nvdev == 1); 756 757 (void) snprintf(addr, addrlen, "%d,%d", domain[0], vdev[0]); 758 ddi_prop_free(vdev); 759 ddi_prop_free(domain); 760 return (DDI_SUCCESS); 761 } 762 ddi_prop_free(domain); 763 764 /* 765 * Use "unit-address" property (frontend/softdev drivers). 766 */ 767 if (ddi_prop_lookup_string(DDI_DEV_T_ANY, child, DDI_PROP_DONTPASS, 768 "unit-address", &prop_str) != DDI_PROP_SUCCESS) 769 return (DDI_FAILURE); 770 (void) strlcpy(addr, prop_str, addrlen); 771 ddi_prop_free(prop_str); 772 return (DDI_SUCCESS); 773 } 774 775 static int 776 xpvd_initchild(dev_info_t *child) 777 { 778 char addr[80]; 779 780 /* 781 * Pseudo nodes indicate a prototype node with per-instance 782 * properties to be merged into the real h/w device node. 783 */ 784 if (ndi_dev_is_persistent_node(child) == 0) { 785 ddi_set_parent_data(child, NULL); 786 787 /* 788 * Try to merge the properties from this prototype 789 * node into real h/w nodes. 790 */ 791 if (ndi_merge_node(child, xpvd_name_child) == DDI_SUCCESS) { 792 /* 793 * Merged ok - return failure to remove the node. 794 */ 795 ddi_set_name_addr(child, NULL); 796 return (DDI_FAILURE); 797 } 798 799 /* 800 * The child was not merged into a h/w node, 801 * but there's not much we can do with it other 802 * than return failure to cause the node to be removed. 803 */ 804 cmn_err(CE_WARN, "!%s@%s: %s.conf properties not merged", 805 ddi_get_name(child), ddi_get_name_addr(child), 806 ddi_get_name(child)); 807 ddi_set_name_addr(child, NULL); 808 return (DDI_NOT_WELL_FORMED); 809 } 810 811 if (xvdi_init_dev(child) != DDI_SUCCESS) 812 return (DDI_FAILURE); 813 814 if (xpvd_name_child(child, addr, sizeof (addr)) != DDI_SUCCESS) { 815 xvdi_uninit_dev(child); 816 return (DDI_FAILURE); 817 } 818 ddi_set_name_addr(child, addr); 819 820 return (DDI_SUCCESS); 821 } 822 823 static int 824 xpvd_removechild(dev_info_t *dip) 825 { 826 xvdi_uninit_dev(dip); 827 828 ddi_set_name_addr(dip, NULL); 829 830 /* 831 * Strip the node to properly convert it back to prototype 832 * form. 833 */ 834 ddi_remove_minor_node(dip, NULL); 835 836 return (DDI_SUCCESS); 837 } 838 839 static int 840 xpvd_bus_unconfig(dev_info_t *parent, uint_t flag, ddi_bus_config_op_t op, 841 void *device_name) 842 { 843 return (ndi_busop_bus_unconfig(parent, flag, op, device_name)); 844 } 845 846 /* 847 * Given the name of a child of xpvd, determine the device class, 848 * domain and vdevnum to which it refers. 849 */ 850 static boolean_t 851 i_xpvd_parse_devname(char *name, xendev_devclass_t *devclassp, 852 domid_t *domp, int *vdevp) 853 { 854 int len = strlen(name) + 1; 855 char *device_name = i_ddi_strdup(name, KM_SLEEP); 856 char *cname = NULL, *caddr = NULL; 857 boolean_t ret; 858 859 i_ddi_parse_name(device_name, &cname, &caddr, NULL); 860 861 if ((cname == NULL) || (strlen(cname) == 0) || 862 (caddr == NULL) || (strlen(caddr) == 0)) { 863 ret = B_FALSE; 864 goto done; 865 } 866 867 *devclassp = xendev_nodename_to_devclass(cname); 868 if (*devclassp < 0) { 869 ret = B_FALSE; 870 goto done; 871 } 872 873 /* 874 * Parsing the address component requires knowledge of how 875 * xpvd_name_child() works. If that code changes, so must 876 * this. 877 */ 878 879 /* Backend format is "<domain>,<vdev>". */ 880 if (sscanf(caddr, "%hu,%d", domp, vdevp) == 2) { 881 ret = B_TRUE; 882 goto done; 883 } 884 885 /* Frontend format is "<vdev>". */ 886 *domp = DOMID_SELF; 887 if (sscanf(caddr, "%d", vdevp) == 1) 888 ret = B_TRUE; 889 done: 890 kmem_free(device_name, len); 891 return (ret); 892 } 893 894 /* 895 * xpvd_bus_config() 896 * 897 * BUS_CONFIG_ONE: 898 * Enumerate the exact instance of a driver. 899 * 900 * BUS_CONFIG_ALL: 901 * Enumerate all the instances of all the possible children (seen before 902 * and never seen before). 903 * 904 * BUS_CONFIG_DRIVER: 905 * Enumerate all the instances of a particular driver. 906 */ 907 static int 908 xpvd_bus_config(dev_info_t *parent, uint_t flag, ddi_bus_config_op_t op, 909 void *arg, dev_info_t **childp) 910 { 911 int circ; 912 char *cname = NULL; 913 914 ndi_devi_enter(parent, &circ); 915 916 switch (op) { 917 case BUS_CONFIG_ONE: { 918 xendev_devclass_t devclass; 919 domid_t dom; 920 int vdev; 921 922 if (!i_xpvd_parse_devname(arg, &devclass, &dom, &vdev)) { 923 ndi_devi_exit(parent, circ); 924 return (NDI_FAILURE); 925 } 926 927 *childp = xvdi_find_dev(parent, devclass, dom, vdev); 928 if (*childp == NULL) 929 *childp = xvdi_create_dev(parent, devclass, dom, vdev); 930 931 ndi_devi_exit(parent, circ); 932 933 if (*childp == NULL) 934 return (NDI_FAILURE); 935 else 936 return (ndi_busop_bus_config(parent, flag, 937 op, arg, childp, 0)); 938 } 939 940 case BUS_CONFIG_DRIVER: { 941 xendev_devclass_t devclass = XEN_INVAL; 942 943 cname = ddi_major_to_name((major_t)(uintptr_t)arg); 944 if (cname != NULL) 945 devclass = xendev_nodename_to_devclass(cname); 946 947 if (devclass == XEN_INVAL) { 948 ndi_devi_exit(parent, circ); 949 return (NDI_FAILURE); 950 } else { 951 xendev_enum_class(parent, devclass); 952 ndi_devi_exit(parent, circ); 953 return (ndi_busop_bus_config(parent, flag, op, 954 arg, childp, 0)); 955 } 956 /* NOTREACHED */ 957 } 958 959 case BUS_CONFIG_ALL: 960 xendev_enum_all(parent, B_FALSE); 961 ndi_devi_exit(parent, circ); 962 963 return (ndi_busop_bus_config(parent, flag, op, 964 arg, childp, 0)); 965 966 default: 967 ndi_devi_exit(parent, circ); 968 return (NDI_FAILURE); 969 } 970 } 971 972 /*ARGSUSED*/ 973 static int 974 xpvd_get_eventcookie(dev_info_t *dip, dev_info_t *rdip, 975 char *eventname, ddi_eventcookie_t *cookie) 976 { 977 return (ndi_event_retrieve_cookie(xpvd_ndi_event_handle, 978 rdip, eventname, cookie, NDI_EVENT_NOPASS)); 979 } 980 981 /*ARGSUSED*/ 982 static int 983 xpvd_add_eventcall(dev_info_t *dip, dev_info_t *rdip, 984 ddi_eventcookie_t cookie, void (*callback)(dev_info_t *dip, 985 ddi_eventcookie_t cookie, void *arg, void *bus_impldata), 986 void *arg, ddi_callback_id_t *cb_id) 987 { 988 return (ndi_event_add_callback(xpvd_ndi_event_handle, 989 rdip, cookie, callback, arg, NDI_SLEEP, cb_id)); 990 } 991 992 /*ARGSUSED*/ 993 static int 994 xpvd_remove_eventcall(dev_info_t *dip, ddi_callback_id_t cb_id) 995 { 996 return (ndi_event_remove_callback(xpvd_ndi_event_handle, 997 cb_id)); 998 } 999 1000 /*ARGSUSED*/ 1001 static int 1002 xpvd_post_event(dev_info_t *dip, dev_info_t *rdip, 1003 ddi_eventcookie_t cookie, void *bus_impldata) 1004 { 1005 return (ndi_event_run_callbacks(xpvd_ndi_event_handle, rdip, 1006 cookie, bus_impldata)); 1007 }