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 * Copyright 2010 Sun Microsystems, Inc. All rights reserved. 23 * Use is subject to license terms. 24 * Copyright 2019 Joyent, Inc. 25 */ 26 27 /* 28 * This file contains PCI HotPlug functionality that is compatible with the 29 * PCI SHPC specification 1.x. 30 * 31 * NOTE: This file is compiled and delivered through misc/pcie module. 32 */ 33 34 #include <sys/note.h> 35 #include <sys/conf.h> 36 #include <sys/kmem.h> 37 #include <sys/kstat.h> 38 #include <sys/debug.h> 39 #include <sys/vtrace.h> 40 #include <sys/autoconf.h> 41 #include <sys/varargs.h> 42 #include <sys/hwconf.h> 43 #include <sys/ddi_impldefs.h> 44 #include <sys/callb.h> 45 #include <sys/ddi.h> 46 #include <sys/sunddi.h> 47 #include <sys/sunndi.h> 48 #include <sys/sysevent/dr.h> 49 #include <sys/ndi_impldefs.h> 50 #include <sys/pci_impl.h> 51 #include <sys/hotplug/pci/pcie_hp.h> 52 #include <sys/hotplug/pci/pcishpc.h> 53 54 typedef struct pcishpc_prop { 55 char *prop_name; 56 char *prop_value; 57 } pcishpc_prop_t; 58 59 static pcishpc_prop_t pcishpc_props[] = { 60 { PCIEHPC_PROP_LED_FAULT, PCIEHPC_PROP_VALUE_LED }, 61 { PCIEHPC_PROP_LED_POWER, PCIEHPC_PROP_VALUE_LED }, 62 { PCIEHPC_PROP_LED_ATTN, PCIEHPC_PROP_VALUE_LED }, 63 { PCIEHPC_PROP_LED_ACTIVE, PCIEHPC_PROP_VALUE_LED }, 64 { PCIEHPC_PROP_CARD_TYPE, PCIEHPC_PROP_VALUE_TYPE }, 65 { PCIEHPC_PROP_BOARD_TYPE, PCIEHPC_PROP_VALUE_TYPE }, 66 { PCIEHPC_PROP_SLOT_CONDITION, PCIEHPC_PROP_VALUE_TYPE } 67 }; 68 69 /* reset delay to 1 sec. */ 70 static int pcishpc_reset_delay = 1000000; 71 72 /* Local function prototype */ 73 static pcie_hp_ctrl_t *pcishpc_create_controller(dev_info_t *dip); 74 static int pcishpc_setup_controller(pcie_hp_ctrl_t *ctrl_p); 75 static int pcishpc_destroy_controller(dev_info_t *dip); 76 static pcie_hp_slot_t *pcishpc_create_slot(pcie_hp_ctrl_t *ctrl_p); 77 static int pcishpc_register_slot(pcie_hp_ctrl_t *ctrl_p, int slot); 78 static int pcishpc_destroy_slots(pcie_hp_ctrl_t *ctrl_p); 79 static int pcishpc_slot_get_property(pcie_hp_slot_t *slot_p, 80 ddi_hp_property_t *arg, ddi_hp_property_t *rval); 81 static int pcishpc_slot_set_property(pcie_hp_slot_t *slot_p, 82 ddi_hp_property_t *arg, ddi_hp_property_t *rval); 83 static int pcishpc_issue_command(pcie_hp_ctrl_t *ctrl_p, 84 uint32_t cmd_code); 85 static int pcishpc_wait_busy(pcie_hp_ctrl_t *ctrl_p); 86 static void pcishpc_attn_btn_handler(pcie_hp_slot_t *slot_p); 87 static void pcishpc_get_slot_state(pcie_hp_slot_t *slot_p); 88 static int pcishpc_set_slot_state(pcie_hp_slot_t *slot_p, 89 ddi_hp_cn_state_t new_slot_state); 90 static void pcishpc_set_slot_name(pcie_hp_ctrl_t *ctrl_p, int slot); 91 static int pcishpc_set_bus_speed(pcie_hp_slot_t *slot_p); 92 static int pcishpc_setled(pcie_hp_slot_t *slot_p, pcie_hp_led_t led, 93 pcie_hp_led_state_t state); 94 static int pcishpc_led_shpc_to_hpc(int state); 95 static int pcishpc_led_hpc_to_shpc(int state); 96 static int pcishpc_slot_shpc_to_hpc(int shpc_state); 97 static int pcishpc_slot_hpc_to_shpc(int state); 98 static char *pcishpc_slot_textslotstate(ddi_hp_cn_state_t state); 99 static char *pcishpc_slot_textledstate(pcie_hp_led_state_t state); 100 101 static uint32_t pcishpc_read_reg(pcie_hp_ctrl_t *ctrl_p, int reg); 102 static void pcishpc_write_reg(pcie_hp_ctrl_t *ctrl_p, int reg, 103 uint32_t data); 104 105 static int pcishpc_upgrade_slot_state(pcie_hp_slot_t *slot_p, 106 ddi_hp_cn_state_t target_state); 107 static int pcishpc_downgrade_slot_state(pcie_hp_slot_t *slot_p, 108 ddi_hp_cn_state_t target_state); 109 static int pcishpc_change_slot_state(pcie_hp_slot_t *slot_p, 110 ddi_hp_cn_state_t target_state); 111 112 static int pcishpc_slot_poweron(pcie_hp_slot_t *slot_p, 113 ddi_hp_cn_state_t *result_state); 114 static int pcishpc_slot_poweroff(pcie_hp_slot_t *slot_p, 115 ddi_hp_cn_state_t *result_state); 116 static int pcishpc_slot_probe(pcie_hp_slot_t *slot_p); 117 static int pcishpc_slot_unprobe(pcie_hp_slot_t *slot_p); 118 #ifdef DEBUG 119 static void pcishpc_dump_regs(pcie_hp_ctrl_t *ctrl_p); 120 #endif /* DEBUG */ 121 122 123 /* 124 * Global functions (called by other drivers/modules) 125 */ 126 127 /* 128 * pcishpc_init() 129 * 130 * Install and configure an SHPC controller and register the HotPlug slots 131 * with the Solaris HotPlug framework. This function is usually called by 132 * a PCI bridge Nexus driver that has a built in SHPC controller. 133 */ 134 int 135 pcishpc_init(dev_info_t *dip) 136 { 137 pcie_bus_t *bus_p = PCIE_DIP2BUS(dip); 138 pcie_hp_ctrl_t *ctrl_p; 139 int i; 140 141 PCIE_DBG("pcishpc_init() called from %s#%d\n", 142 ddi_driver_name(dip), ddi_get_instance(dip)); 143 144 if ((ctrl_p = PCIE_GET_HP_CTRL(dip)) != NULL) { 145 PCIE_DBG("pcishpc_init() shpc instance already " 146 "initialized!\n"); 147 return (DDI_SUCCESS); 148 } 149 150 /* Initialize soft state structure for the SHPC instance. */ 151 ctrl_p = pcishpc_create_controller(dip); 152 153 if (ctrl_p == NULL) { 154 PCIE_DBG("pcishpc_init() failed to create shpc softstate\n"); 155 return (DDI_FAILURE); 156 } 157 158 if (pcishpc_setup_controller(ctrl_p) != DDI_SUCCESS) { 159 PCIE_DBG("pcishpc_init() failed to setup controller\n"); 160 goto cleanup; 161 } 162 163 /* 164 * Setup resource maps for this bus node. 165 */ 166 (void) pci_resource_setup(dip); 167 168 #ifdef DEBUG 169 PCIE_DBG("%s%d: P2P bridge register dump:\n", 170 ddi_driver_name(dip), ddi_get_instance(dip)); 171 172 for (i = 0; i < 0x100; i += 4) { 173 PCIE_DBG("SHPC Cfg reg 0x%02x: %08x\n", i, 174 pci_config_get32(bus_p->bus_cfg_hdl, i)); 175 } 176 #endif /* DEBUG */ 177 178 /* Setup each HotPlug slot on this SHPC controller. */ 179 for (i = 0; i < ctrl_p->hc_num_slots_impl; i++) { 180 if (pcishpc_register_slot(ctrl_p, i) != DDI_SUCCESS) { 181 PCIE_DBG("pcishpc_init() failed to register " 182 "slot %d\n", i); 183 goto cleanup1; 184 } 185 if (pcie_create_minor_node(ctrl_p, i) != DDI_SUCCESS) { 186 PCIE_DBG("pcishpc_init() failed to create " 187 "minor node for slot %d\n", i); 188 goto cleanup1; 189 } 190 } 191 192 #ifdef DEBUG 193 /* Dump out the SHPC registers. */ 194 pcishpc_dump_regs(ctrl_p); 195 #endif /* DEBUG */ 196 197 PCIE_DBG("pcishpc_init() success(dip=%p)\n", dip); 198 return (DDI_SUCCESS); 199 200 cleanup1: 201 for (i = 0; i < ctrl_p->hc_num_slots_impl; i++) { 202 if (ctrl_p->hc_slots[i] == NULL) 203 continue; 204 205 pcie_remove_minor_node(ctrl_p, i); 206 } 207 (void) pci_resource_destroy(dip); 208 cleanup: 209 (void) pcishpc_destroy_controller(dip); 210 return (DDI_FAILURE); 211 } 212 213 /* 214 * pcishpc_uninit() 215 * Unload the HogPlug controller driver and deallocate all resources. 216 */ 217 int 218 pcishpc_uninit(dev_info_t *dip) 219 { 220 pcie_hp_ctrl_t *ctrl_p; 221 int i; 222 223 PCIE_DBG("pcishpc_uninit() called(dip=%p)\n", dip); 224 225 ctrl_p = PCIE_GET_HP_CTRL(dip); 226 227 if (!ctrl_p) { 228 PCIE_DBG("pcishpc_uninit() Unable to find softstate\n"); 229 return (DDI_FAILURE); 230 } 231 232 for (i = 0; i < PCIE_HP_MAX_SLOTS; i++) { 233 if (ctrl_p->hc_slots[i] == NULL) 234 continue; 235 236 pcie_remove_minor_node(ctrl_p, i); 237 } 238 239 ctrl_p->hc_flags = 0; 240 241 /* 242 * Destroy resource maps for this bus node. 243 */ 244 (void) pci_resource_destroy(dip); 245 246 (void) pcishpc_destroy_controller(dip); 247 248 PCIE_DBG("pcishpc_uninit() success(dip=%p)\n", dip); 249 250 return (DDI_SUCCESS); 251 } 252 253 /* 254 * pcishpc_intr() 255 * 256 * This is the SHPC controller interrupt handler. 257 */ 258 int 259 pcishpc_intr(dev_info_t *dip) 260 { 261 pcie_hp_ctrl_t *ctrl_p; 262 uint32_t irq_locator, irq_serr_locator, reg; 263 int slot; 264 265 PCIE_DBG("pcishpc_intr() called\n"); 266 267 /* get the soft state structure for this dip */ 268 if ((ctrl_p = PCIE_GET_HP_CTRL(dip)) == NULL) 269 return (DDI_INTR_UNCLAIMED); 270 271 mutex_enter(&ctrl_p->hc_mutex); 272 273 if (!(ctrl_p->hc_flags & PCIE_HP_INITIALIZED_FLAG)) { 274 PCIE_DBG("pcishpc_intr() unclaimed\n"); 275 mutex_exit(&ctrl_p->hc_mutex); 276 return (DDI_INTR_UNCLAIMED); 277 } 278 279 PCIE_DBG("pcishpc_intr() interrupt received\n"); 280 281 reg = pcishpc_read_reg(ctrl_p, PCI_HP_CTRL_SERR_INT_REG); 282 283 if (reg & PCI_HP_SERR_INT_CMD_COMPLETE_IRQ) { 284 PCIE_DBG("pcishpc_intr() " 285 "PCI_HP_SERR_INT_CMD_COMPLETE_IRQ detected\n"); 286 ctrl_p->hc_cmd_pending = B_FALSE; 287 cv_signal(&ctrl_p->hc_cmd_comp_cv); 288 } 289 290 if (reg & PCI_HP_SERR_INT_ARBITER_IRQ) { 291 PCIE_DBG("pcishpc_intr() PCI_HP_SERR_INT_ARBITER_IRQ " 292 "detected\n"); 293 ctrl_p->hc_arbiter_timeout = B_TRUE; 294 } 295 296 /* Write back the SERR INT register to acknowledge the IRQs. */ 297 pcishpc_write_reg(ctrl_p, PCI_HP_CTRL_SERR_INT_REG, reg); 298 299 irq_locator = pcishpc_read_reg(ctrl_p, PCI_HP_IRQ_LOCATOR_REG); 300 irq_serr_locator = pcishpc_read_reg(ctrl_p, PCI_HP_SERR_LOCATOR_REG); 301 302 /* Check for slot events that might have occured. */ 303 for (slot = 0; slot < ctrl_p->hc_num_slots_impl; slot++) { 304 if ((irq_locator & (PCI_HP_IRQ_SLOT_N_PENDING<<slot)) || 305 (irq_serr_locator & 306 (PCI_HP_IRQ_SERR_SLOT_N_PENDING<<slot))) { 307 PCIE_DBG("pcishpc_intr() slot %d and " 308 "pending IRQ\n", slot+1); 309 310 reg = pcishpc_read_reg(ctrl_p, 311 PCI_HP_LOGICAL_SLOT_REGS+slot); 312 313 if (reg & PCI_HP_SLOT_PRESENCE_DETECTED) 314 PCIE_DBG("slot %d: " 315 "PCI_HP_SLOT_PRESENCE_DETECTED\n", 316 slot+1); 317 318 if (reg & PCI_HP_SLOT_ISO_PWR_DETECTED) 319 PCIE_DBG("slot %d: " 320 "PCI_HP_SLOT_ISO_PWR_DETECTED\n", 321 slot+1); 322 323 if (reg & PCI_HP_SLOT_ATTN_DETECTED) { 324 PCIE_DBG("slot %d: " 325 "PCI_HP_SLOT_ATTN_DETECTED\n", slot+1); 326 327 /* 328 * if ATTN button event is still pending 329 * then cancel it 330 */ 331 if (ctrl_p->hc_slots[slot]-> 332 hs_attn_btn_pending == B_TRUE) 333 ctrl_p->hc_slots[slot]-> 334 hs_attn_btn_pending = B_FALSE; 335 336 /* wake up the ATTN event handler */ 337 cv_signal(&ctrl_p->hc_slots[slot]-> 338 hs_attn_btn_cv); 339 } 340 341 if (reg & PCI_HP_SLOT_MRL_DETECTED) 342 PCIE_DBG("slot %d: " 343 "PCI_HP_SLOT_MRL_DETECTED\n", slot+1); 344 345 if (reg & PCI_HP_SLOT_POWER_DETECTED) 346 PCIE_DBG("slot %d: " 347 "PCI_HP_SLOT_POWER_DETECTED\n", slot+1); 348 349 /* Acknoledge any slot interrupts */ 350 pcishpc_write_reg(ctrl_p, PCI_HP_LOGICAL_SLOT_REGS+slot, 351 reg); 352 } 353 } 354 355 mutex_exit(&ctrl_p->hc_mutex); 356 357 PCIE_DBG("pcishpc_intr() claimed\n"); 358 359 return (DDI_INTR_CLAIMED); 360 } 361 362 int 363 pcishpc_slot_get_property(pcie_hp_slot_t *slot_p, ddi_hp_property_t *arg, 364 ddi_hp_property_t *rval) 365 { 366 ddi_hp_property_t request, result; 367 #ifdef _SYSCALL32_IMPL 368 ddi_hp_property32_t request32, result32; 369 #endif 370 pcie_hp_ctrl_t *ctrl_p = slot_p->hs_ctrl; 371 nvlist_t *prop_list; 372 nvlist_t *prop_rlist; /* nvlist for return values */ 373 nvpair_t *prop_pair; 374 char *name, *value; 375 int ret = DDI_SUCCESS; 376 int i, n; 377 boolean_t get_all_prop = B_FALSE; 378 379 if (get_udatamodel() == DATAMODEL_NATIVE) { 380 if (copyin(arg, &request, sizeof (ddi_hp_property_t)) || 381 copyin(rval, &result, sizeof (ddi_hp_property_t))) 382 return (DDI_FAILURE); 383 } 384 #ifdef _SYSCALL32_IMPL 385 else { 386 bzero(&request, sizeof (request)); 387 bzero(&result, sizeof (result)); 388 if (copyin(arg, &request32, sizeof (ddi_hp_property32_t)) || 389 copyin(rval, &result32, sizeof (ddi_hp_property32_t))) 390 return (DDI_FAILURE); 391 request.nvlist_buf = (char *)(uintptr_t)request32.nvlist_buf; 392 request.buf_size = request32.buf_size; 393 result.nvlist_buf = (char *)(uintptr_t)result32.nvlist_buf; 394 result.buf_size = result32.buf_size; 395 } 396 #endif 397 398 if ((ret = pcie_copyin_nvlist(request.nvlist_buf, request.buf_size, 399 &prop_list)) != DDI_SUCCESS) 400 return (ret); 401 402 if (nvlist_alloc(&prop_rlist, NV_UNIQUE_NAME, 0)) { 403 ret = DDI_ENOMEM; 404 goto get_prop_cleanup; 405 } 406 407 /* check whether the requested property is "all" or "help" */ 408 prop_pair = nvlist_next_nvpair(prop_list, NULL); 409 if (prop_pair && !nvlist_next_nvpair(prop_list, prop_pair)) { 410 name = nvpair_name(prop_pair); 411 n = sizeof (pcishpc_props) / sizeof (pcishpc_prop_t); 412 413 if (strcmp(name, PCIEHPC_PROP_ALL) == 0) { 414 (void) nvlist_remove_all(prop_list, PCIEHPC_PROP_ALL); 415 416 /* 417 * Add all properties into the request list, so that we 418 * will get the values in the following for loop. 419 */ 420 for (i = 0; i < n; i++) { 421 if (nvlist_add_string(prop_list, 422 pcishpc_props[i].prop_name, "") != 0) { 423 ret = DDI_FAILURE; 424 goto get_prop_cleanup1; 425 } 426 } 427 get_all_prop = B_TRUE; 428 } else if (strcmp(name, PCIEHPC_PROP_HELP) == 0) { 429 /* 430 * Empty the request list, and add help strings into the 431 * return list. We will pass the following for loop. 432 */ 433 (void) nvlist_remove_all(prop_list, PCIEHPC_PROP_HELP); 434 435 for (i = 0; i < n; i++) { 436 if (nvlist_add_string(prop_rlist, 437 pcishpc_props[i].prop_name, 438 pcishpc_props[i].prop_value) != 0) { 439 ret = DDI_FAILURE; 440 goto get_prop_cleanup1; 441 } 442 } 443 } 444 } 445 446 mutex_enter(&ctrl_p->hc_mutex); 447 448 /* get the current slot state */ 449 pcishpc_get_slot_state(slot_p); 450 451 /* for each requested property, get the value and add it to nvlist */ 452 prop_pair = NULL; 453 while ((prop_pair = nvlist_next_nvpair(prop_list, prop_pair)) != NULL) { 454 name = nvpair_name(prop_pair); 455 value = NULL; 456 457 if (strcmp(name, PCIEHPC_PROP_LED_FAULT) == 0) { 458 value = pcie_led_state_text( 459 slot_p->hs_fault_led_state); 460 } else if (strcmp(name, PCIEHPC_PROP_LED_POWER) == 0) { 461 value = pcie_led_state_text( 462 slot_p->hs_power_led_state); 463 } else if (strcmp(name, PCIEHPC_PROP_LED_ATTN) == 0) { 464 value = pcie_led_state_text( 465 slot_p->hs_attn_led_state); 466 } else if (strcmp(name, PCIEHPC_PROP_LED_ACTIVE) == 0) { 467 value = pcie_led_state_text( 468 slot_p->hs_active_led_state); 469 } else if (strcmp(name, PCIEHPC_PROP_CARD_TYPE) == 0) { 470 ddi_acc_handle_t handle; 471 dev_info_t *cdip; 472 uint8_t prog_class, base_class, sub_class; 473 int i; 474 475 mutex_exit(&ctrl_p->hc_mutex); 476 cdip = pcie_hp_devi_find( 477 ctrl_p->hc_dip, slot_p->hs_device_num, 0); 478 mutex_enter(&ctrl_p->hc_mutex); 479 480 if ((slot_p->hs_info.cn_state != 481 DDI_HP_CN_STATE_ENABLED) || (cdip == NULL)) { 482 /* 483 * When getting all properties, just ignore the 484 * one that's not available under certain state. 485 */ 486 if (get_all_prop) 487 continue; 488 489 ret = DDI_ENOTSUP; 490 goto get_prop_cleanup2; 491 } 492 493 if (pci_config_setup(cdip, &handle) != DDI_SUCCESS) { 494 ret = DDI_FAILURE; 495 goto get_prop_cleanup2; 496 } 497 498 prog_class = pci_config_get8(handle, 499 PCI_CONF_PROGCLASS); 500 base_class = pci_config_get8(handle, PCI_CONF_BASCLASS); 501 sub_class = pci_config_get8(handle, PCI_CONF_SUBCLASS); 502 pci_config_teardown(&handle); 503 504 for (i = 0; i < class_pci_items; i++) { 505 if ((base_class == class_pci[i].base_class) && 506 (sub_class == class_pci[i].sub_class) && 507 (prog_class == class_pci[i].prog_class)) { 508 value = class_pci[i].short_desc; 509 break; 510 } 511 } 512 if (i == class_pci_items) 513 value = PCIEHPC_PROP_VALUE_UNKNOWN; 514 } else if (strcmp(name, PCIEHPC_PROP_BOARD_TYPE) == 0) { 515 if (slot_p->hs_info.cn_state <= DDI_HP_CN_STATE_EMPTY) 516 value = PCIEHPC_PROP_VALUE_UNKNOWN; 517 else 518 value = PCIEHPC_PROP_VALUE_PCIHOTPLUG; 519 } else if (strcmp(name, PCIEHPC_PROP_SLOT_CONDITION) == 0) { 520 value = pcie_slot_condition_text(slot_p->hs_condition); 521 } else { 522 /* unsupported property */ 523 PCIE_DBG("Unsupported property: %s\n", name); 524 525 ret = DDI_ENOTSUP; 526 goto get_prop_cleanup2; 527 } 528 if (nvlist_add_string(prop_rlist, name, value) != 0) { 529 ret = DDI_FAILURE; 530 goto get_prop_cleanup2; 531 } 532 } 533 534 // pack nvlist and copyout 535 if ((ret = pcie_copyout_nvlist(prop_rlist, result.nvlist_buf, 536 &result.buf_size)) != DDI_SUCCESS) { 537 goto get_prop_cleanup2; 538 } 539 if (get_udatamodel() == DATAMODEL_NATIVE) { 540 if (copyout(&result, rval, sizeof (ddi_hp_property_t))) { 541 ret = DDI_FAILURE; 542 goto get_prop_cleanup2; 543 } 544 } 545 #ifdef _SYSCALL32_IMPL 546 else { 547 if (result.buf_size > UINT32_MAX) { 548 ret = DDI_FAILURE; 549 } else { 550 result32.buf_size = (uint32_t)result.buf_size; 551 if (copyout(&result32, rval, 552 sizeof (ddi_hp_property32_t))) 553 ret = DDI_FAILURE; 554 } 555 } 556 #endif 557 558 get_prop_cleanup2: 559 mutex_exit(&ctrl_p->hc_mutex); 560 get_prop_cleanup1: 561 nvlist_free(prop_rlist); 562 get_prop_cleanup: 563 nvlist_free(prop_list); 564 return (ret); 565 } 566 567 int 568 pcishpc_slot_set_property(pcie_hp_slot_t *slot_p, ddi_hp_property_t *arg, 569 ddi_hp_property_t *rval) 570 { 571 ddi_hp_property_t request, result; 572 #ifdef _SYSCALL32_IMPL 573 ddi_hp_property32_t request32, result32; 574 #endif 575 pcie_hp_ctrl_t *ctrl_p = slot_p->hs_ctrl; 576 nvlist_t *prop_list; 577 nvlist_t *prop_rlist; 578 nvpair_t *prop_pair; 579 char *name, *value; 580 pcie_hp_led_state_t led_state; 581 int ret = DDI_SUCCESS; 582 583 if (get_udatamodel() == DATAMODEL_NATIVE) { 584 if (copyin(arg, &request, sizeof (ddi_hp_property_t))) 585 return (DDI_FAILURE); 586 if (rval && 587 copyin(rval, &result, sizeof (ddi_hp_property_t))) 588 return (DDI_FAILURE); 589 } 590 #ifdef _SYSCALL32_IMPL 591 else { 592 bzero(&request, sizeof (request)); 593 bzero(&result, sizeof (result)); 594 if (copyin(arg, &request32, sizeof (ddi_hp_property32_t))) 595 return (DDI_FAILURE); 596 if (rval && 597 copyin(rval, &result32, sizeof (ddi_hp_property32_t))) 598 return (DDI_FAILURE); 599 request.nvlist_buf = (char *)(uintptr_t)request32.nvlist_buf; 600 request.buf_size = request32.buf_size; 601 if (rval) { 602 result.nvlist_buf = 603 (char *)(uintptr_t)result32.nvlist_buf; 604 result.buf_size = result32.buf_size; 605 } 606 } 607 #endif 608 609 if ((ret = pcie_copyin_nvlist(request.nvlist_buf, request.buf_size, 610 &prop_list)) != DDI_SUCCESS) 611 return (ret); 612 613 /* check whether the requested property is "help" */ 614 prop_pair = nvlist_next_nvpair(prop_list, NULL); 615 if (prop_pair && !nvlist_next_nvpair(prop_list, prop_pair) && 616 (strcmp(nvpair_name(prop_pair), PCIEHPC_PROP_HELP) == 0)) { 617 if (!rval) { 618 ret = DDI_ENOTSUP; 619 goto set_prop_cleanup; 620 } 621 622 if (nvlist_alloc(&prop_rlist, NV_UNIQUE_NAME, 0)) { 623 ret = DDI_ENOMEM; 624 goto set_prop_cleanup; 625 } 626 if (nvlist_add_string(prop_rlist, PCIEHPC_PROP_LED_ATTN, 627 PCIEHPC_PROP_VALUE_LED) != 0) { 628 ret = DDI_FAILURE; 629 goto set_prop_cleanup1; 630 } 631 632 if ((ret = pcie_copyout_nvlist(prop_rlist, result.nvlist_buf, 633 &result.buf_size)) != DDI_SUCCESS) { 634 goto set_prop_cleanup1; 635 } 636 if (get_udatamodel() == DATAMODEL_NATIVE) { 637 if (copyout(&result, rval, 638 sizeof (ddi_hp_property_t))) { 639 ret = DDI_FAILURE; 640 goto set_prop_cleanup1; 641 } 642 } 643 #ifdef _SYSCALL32_IMPL 644 else { 645 if (result.buf_size > UINT32_MAX) { 646 ret = DDI_FAILURE; 647 goto set_prop_cleanup1; 648 } else { 649 result32.buf_size = (uint32_t)result.buf_size; 650 if (copyout(&result32, rval, 651 sizeof (ddi_hp_property32_t))) { 652 ret = DDI_FAILURE; 653 goto set_prop_cleanup1; 654 } 655 } 656 } 657 #endif 658 set_prop_cleanup1: 659 nvlist_free(prop_rlist); 660 nvlist_free(prop_list); 661 return (ret); 662 } 663 664 /* Validate the request */ 665 prop_pair = NULL; 666 while ((prop_pair = nvlist_next_nvpair(prop_list, prop_pair)) != NULL) { 667 name = nvpair_name(prop_pair); 668 if (nvpair_type(prop_pair) != DATA_TYPE_STRING) { 669 PCIE_DBG("Unexpected data type of setting " 670 "property %s.\n", name); 671 ret = DDI_EINVAL; 672 goto set_prop_cleanup; 673 } 674 if (nvpair_value_string(prop_pair, &value)) { 675 PCIE_DBG("Get string value failed for property %s.\n", 676 name); 677 ret = DDI_FAILURE; 678 goto set_prop_cleanup; 679 } 680 681 if (strcmp(name, PCIEHPC_PROP_LED_ATTN) == 0) { 682 if ((strcmp(value, PCIEHPC_PROP_VALUE_ON) != 0) && 683 (strcmp(value, PCIEHPC_PROP_VALUE_OFF) != 0) && 684 (strcmp(value, PCIEHPC_PROP_VALUE_BLINK) != 0)) { 685 PCIE_DBG("Unsupported value of setting " 686 "property %s\n", name); 687 ret = DDI_ENOTSUP; 688 goto set_prop_cleanup; 689 } 690 } else { 691 PCIE_DBG("Unsupported property: %s\n", name); 692 ret = DDI_ENOTSUP; 693 goto set_prop_cleanup; 694 } 695 } 696 697 mutex_enter(&ctrl_p->hc_mutex); 698 699 /* get the current slot state */ 700 pcishpc_get_slot_state(slot_p); 701 702 // set each property 703 prop_pair = NULL; 704 while ((prop_pair = nvlist_next_nvpair(prop_list, prop_pair)) != NULL) { 705 name = nvpair_name(prop_pair); 706 707 if (strcmp(name, PCIEHPC_PROP_LED_ATTN) == 0) { 708 if (strcmp(value, PCIEHPC_PROP_VALUE_ON) == 0) 709 led_state = PCIE_HP_LED_ON; 710 else if (strcmp(value, PCIEHPC_PROP_VALUE_OFF) == 0) 711 led_state = PCIE_HP_LED_OFF; 712 else if (strcmp(value, PCIEHPC_PROP_VALUE_BLINK) == 0) 713 led_state = PCIE_HP_LED_BLINK; 714 else 715 continue; 716 717 (void) pcishpc_setled(slot_p, PCIE_HP_ATTN_LED, 718 led_state); 719 } 720 } 721 if (rval) { 722 if (get_udatamodel() == DATAMODEL_NATIVE) { 723 result.buf_size = 0; 724 if (copyout(&result, rval, sizeof (ddi_hp_property_t))) 725 ret = DDI_FAILURE; 726 } 727 #ifdef _SYSCALL32_IMPL 728 else { 729 result32.buf_size = 0; 730 if (copyout(&result32, rval, 731 sizeof (ddi_hp_property32_t))) 732 ret = DDI_FAILURE; 733 } 734 #endif 735 } 736 737 mutex_exit(&ctrl_p->hc_mutex); 738 set_prop_cleanup: 739 nvlist_free(prop_list); 740 return (ret); 741 } 742 743 /* 744 * pcishpc_hp_ops() 745 * 746 * Handle hotplug commands 747 * 748 * Note: This function is called by DDI HP framework at kernel context only 749 */ 750 /* ARGSUSED */ 751 int 752 pcishpc_hp_ops(dev_info_t *dip, char *cn_name, ddi_hp_op_t op, 753 void *arg, void *result) 754 { 755 pcie_hp_slot_t *slot_p = NULL; 756 pcie_hp_ctrl_t *ctrl_p; 757 int ret = DDI_SUCCESS, i; 758 759 PCIE_DBG("pcishpc_hp_ops: dip=%p cn_name=%s op=%x arg=%p\n", 760 dip, cn_name, op, arg); 761 762 if ((ctrl_p = PCIE_GET_HP_CTRL(dip)) == NULL) 763 return (DDI_FAILURE); 764 765 for (i = 0; i < PCIE_HP_MAX_SLOTS && ctrl_p->hc_slots[i]; i++) { 766 if (strcmp(ctrl_p->hc_slots[i]->hs_info.cn_name, cn_name) 767 == 0) { 768 /* Match with a physical slot, found */ 769 slot_p = ctrl_p->hc_slots[i]; 770 break; 771 } 772 } 773 if (!slot_p) { 774 PCIE_DBG("pcishpc_hp_ops: Failed to find the slot under" 775 "dip %p with name: %s; op=%x arg=%p\n", 776 dip, cn_name, op, arg); 777 return (DDI_EINVAL); 778 } 779 switch (op) { 780 case DDI_HPOP_CN_GET_STATE: 781 { 782 mutex_enter(&ctrl_p->hc_mutex); 783 784 /* get the current slot state */ 785 pcishpc_get_slot_state(slot_p); 786 787 *((ddi_hp_cn_state_t *)result) = slot_p->hs_info.cn_state; 788 789 mutex_exit(&ctrl_p->hc_mutex); 790 break; 791 } 792 case DDI_HPOP_CN_CHANGE_STATE: 793 { 794 ddi_hp_cn_state_t target_state = *(ddi_hp_cn_state_t *)arg; 795 796 mutex_enter(&slot_p->hs_ctrl->hc_mutex); 797 798 ret = pcishpc_change_slot_state(slot_p, target_state); 799 *((ddi_hp_cn_state_t *)result) = slot_p->hs_info.cn_state; 800 801 mutex_exit(&slot_p->hs_ctrl->hc_mutex); 802 break; 803 } 804 case DDI_HPOP_CN_PROBE: 805 ret = pcishpc_slot_probe(slot_p); 806 807 break; 808 case DDI_HPOP_CN_UNPROBE: 809 ret = pcishpc_slot_unprobe(slot_p); 810 811 break; 812 case DDI_HPOP_CN_GET_PROPERTY: 813 ret = pcishpc_slot_get_property(slot_p, 814 (ddi_hp_property_t *)arg, (ddi_hp_property_t *)result); 815 break; 816 case DDI_HPOP_CN_SET_PROPERTY: 817 ret = pcishpc_slot_set_property(slot_p, 818 (ddi_hp_property_t *)arg, (ddi_hp_property_t *)result); 819 break; 820 default: 821 ret = DDI_ENOTSUP; 822 break; 823 } 824 825 return (ret); 826 } 827 828 /* 829 * Local functions (called within this file) 830 */ 831 832 /* 833 * pcishpc_create_controller() 834 * 835 * This function allocates and creates an SHPC controller state structure 836 * and adds it to the linked list of controllers. 837 */ 838 static pcie_hp_ctrl_t * 839 pcishpc_create_controller(dev_info_t *dip) 840 { 841 pcie_bus_t *bus_p = PCIE_DIP2BUS(dip); 842 pcie_hp_ctrl_t *ctrl_p; 843 844 PCIE_DBG("pcishpc: create controller for %s#%d\n", 845 ddi_driver_name(dip), ddi_get_instance(dip)); 846 847 ctrl_p = kmem_zalloc(sizeof (pcie_hp_ctrl_t), KM_SLEEP); 848 ctrl_p->hc_dip = dip; 849 850 cv_init(&ctrl_p->hc_cmd_comp_cv, NULL, CV_DRIVER, NULL); 851 852 /* Init the shpc controller's mutex. */ 853 mutex_init(&ctrl_p->hc_mutex, NULL, MUTEX_DRIVER, NULL); 854 855 /* HPC initialization is complete now */ 856 ctrl_p->hc_flags = PCIE_HP_INITIALIZED_FLAG; 857 bus_p->bus_hp_curr_mode = PCIE_PCI_HP_MODE; 858 859 PCIE_SET_HP_CTRL(dip, ctrl_p); 860 861 PCIE_DBG("pcishpc_create_controller() success\n"); 862 863 return (ctrl_p); 864 } 865 866 867 /* 868 * pcishpc_setup_controller() 869 * 870 * Get the number of HotPlug Slots, and the PCI device information 871 * for this HotPlug controller. 872 */ 873 static int 874 pcishpc_setup_controller(pcie_hp_ctrl_t *ctrl_p) 875 { 876 uint32_t config; 877 dev_info_t *ppdip; 878 879 config = pcishpc_read_reg(ctrl_p, PCI_HP_SLOT_CONFIGURATION_REG); 880 881 /* Get the number of HotPlug slots implemented */ 882 ctrl_p->hc_num_slots_impl = ((config)&31); 883 884 /* 885 * Initilize the current bus speed and number of hotplug slots 886 * currently connected. 887 */ 888 ctrl_p->hc_curr_bus_speed = -1; 889 ctrl_p->hc_num_slots_connected = 0; 890 891 /* 892 * Get the first PCI device Number used. 893 * 894 * PCI-X I/O boat workaround. 895 * The register doesn't set up the correct value. 896 */ 897 ppdip = ddi_get_parent(ddi_get_parent(ctrl_p->hc_dip)); 898 if ((ddi_prop_get_int(DDI_DEV_T_ANY, ppdip, DDI_PROP_DONTPASS, 899 "vendor-id", -1) == 0x108e) && 900 (ddi_prop_get_int(DDI_DEV_T_ANY, ppdip, DDI_PROP_DONTPASS, 901 "device-id", -1) == 0x9010)) 902 ctrl_p->hc_device_start = 4; 903 else 904 ctrl_p->hc_device_start = ((config>>8)&31); 905 906 /* Get the first Physical device number. */ 907 ctrl_p->hc_phys_start = ((config>>16)&0x7ff); 908 909 /* Check if the device numbers increase or decrease. */ 910 ctrl_p->hc_device_increases = ((config>>29)&0x1); 911 912 ctrl_p->hc_has_attn = 913 (config & PCI_HP_SLOT_CONFIG_ATTN_BUTTON) ? B_TRUE : B_FALSE; 914 ctrl_p->hc_has_mrl = 915 (config & PCI_HP_SLOT_CONFIG_MRL_SENSOR) ? B_TRUE : B_FALSE; 916 917 ctrl_p->hc_cmd_pending = B_FALSE; 918 ctrl_p->hc_arbiter_timeout = B_FALSE; 919 920 if (ctrl_p->hc_num_slots_impl > PCIE_HP_MAX_SLOTS) { 921 PCIE_DBG("pcishpc_setup_controller() too many SHPC " 922 "slots error\n"); 923 return (DDI_FAILURE); 924 } 925 926 return (DDI_SUCCESS); 927 } 928 929 930 /* 931 * pcishpc_destroy_controller() 932 * 933 * This function deallocates all of the SHPC controller resources. 934 */ 935 static int 936 pcishpc_destroy_controller(dev_info_t *dip) 937 { 938 pcie_hp_ctrl_t *ctrl_p; 939 pcie_bus_t *bus_p = PCIE_DIP2BUS(dip); 940 941 PCIE_DBG("pcishpc_destroy_controller() called(dip=%p)\n", dip); 942 943 /* get the soft state structure for this dip */ 944 if ((ctrl_p = PCIE_GET_HP_CTRL(dip)) == NULL) { 945 PCIE_DBG("pcishpc_destroy_controller() not found\n"); 946 return (DDI_FAILURE); 947 } 948 949 /* 950 * Deallocate the slot state structures for this controller. 951 */ 952 PCIE_SET_HP_CTRL(dip, NULL); 953 bus_p->bus_hp_curr_mode = PCIE_NONE_HP_MODE; 954 955 (void) pcishpc_destroy_slots(ctrl_p); 956 cv_destroy(&ctrl_p->hc_cmd_comp_cv); 957 mutex_destroy(&ctrl_p->hc_mutex); 958 kmem_free(ctrl_p, sizeof (pcie_hp_ctrl_t)); 959 960 PCIE_DBG("pcishpc_destroy_controller() success\n"); 961 return (DDI_SUCCESS); 962 } 963 964 /* 965 * pcishpc_create_slot() 966 * 967 * Allocate and add a new HotPlug slot state structure to the linked list. 968 */ 969 static pcie_hp_slot_t * 970 pcishpc_create_slot(pcie_hp_ctrl_t *ctrl_p) 971 { 972 pcie_hp_slot_t *slot_p; 973 974 PCIE_DBG("pcishpc_create_slot() called(ctrl_p=%x)\n", ctrl_p); 975 976 /* Allocate a new slot structure. */ 977 slot_p = kmem_zalloc(sizeof (pcie_hp_slot_t), KM_SLEEP); 978 slot_p->hs_ctrl = ctrl_p; 979 980 /* Assign an initial value */ 981 slot_p->hs_info.cn_state = DDI_HP_CN_STATE_EMPTY; 982 983 PCIE_DBG("pcishpc_create_slot() success\n"); 984 return (slot_p); 985 } 986 987 /* 988 * pcishpc_register_slot() 989 * 990 * Create and register a slot with the Solaris HotPlug framework. 991 */ 992 static int 993 pcishpc_register_slot(pcie_hp_ctrl_t *ctrl_p, int slot) 994 { 995 dev_info_t *dip = ctrl_p->hc_dip; 996 pcie_hp_slot_t *slot_p; 997 998 slot_p = pcishpc_create_slot(ctrl_p); 999 ctrl_p->hc_slots[slot] = slot_p; 1000 slot_p->hs_num = slot; 1001 1002 /* Setup the PCI device # for this SHPC slot. */ 1003 if (ctrl_p->hc_device_increases) 1004 slot_p->hs_device_num = ctrl_p->hc_device_start + 1005 slot_p->hs_num; 1006 else 1007 slot_p->hs_device_num = ctrl_p->hc_device_start - 1008 slot_p->hs_num; 1009 1010 /* Setup the DDI HP framework slot information. */ 1011 slot_p->hs_info.cn_type = DDI_HP_CN_TYPE_PCI; 1012 slot_p->hs_info.cn_type_str = PCIE_PCI_HP_TYPE; 1013 slot_p->hs_info.cn_child = NULL; 1014 1015 slot_p->hs_minor = PCI_MINOR_NUM( 1016 ddi_get_instance(dip), slot_p->hs_device_num); 1017 slot_p->hs_condition = AP_COND_UNKNOWN; 1018 1019 /* setup thread for handling ATTN button events */ 1020 if (ctrl_p->hc_has_attn) { 1021 PCIE_DBG("pcishpc_register_slot: " 1022 "setting up ATTN button event " 1023 "handler thread for slot %d\n", slot); 1024 1025 cv_init(&slot_p->hs_attn_btn_cv, NULL, CV_DRIVER, NULL); 1026 slot_p->hs_attn_btn_pending = B_FALSE; 1027 slot_p->hs_attn_btn_threadp = thread_create(NULL, 0, 1028 pcishpc_attn_btn_handler, 1029 (void *)slot_p, 0, &p0, TS_RUN, minclsyspri); 1030 slot_p->hs_attn_btn_thread_exit = B_FALSE; 1031 } 1032 1033 /* setup the slot name (used for ap-id) */ 1034 pcishpc_set_slot_name(ctrl_p, slot); 1035 1036 pcishpc_get_slot_state(slot_p); 1037 if (slot_p->hs_info.cn_state >= DDI_HP_CN_STATE_ENABLED) 1038 slot_p->hs_condition = AP_COND_OK; 1039 1040 /* register the slot with DDI HP framework */ 1041 if (ndi_hp_register(dip, &slot_p->hs_info) != NDI_SUCCESS) { 1042 PCIE_DBG("pciehpc_register_slot() failed to register slot %d\n", 1043 slot_p->hs_phy_slot_num); 1044 return (DDI_FAILURE); 1045 } 1046 1047 pcie_hp_create_occupant_props(dip, makedevice(ddi_driver_major(dip), 1048 slot_p->hs_minor), slot_p->hs_device_num); 1049 1050 PCIE_DBG("pcishpc_register_slot() success for slot %d\n", slot); 1051 1052 return (DDI_SUCCESS); 1053 } 1054 1055 /* 1056 * pcishpc_destroy_slots() 1057 * 1058 * Free up all of the slot resources for this controller. 1059 */ 1060 static int 1061 pcishpc_destroy_slots(pcie_hp_ctrl_t *ctrl_p) 1062 { 1063 dev_info_t *dip = ctrl_p->hc_dip; 1064 pcie_hp_slot_t *slot_p; 1065 int i; 1066 1067 PCIE_DBG("pcishpc_destroy_slots() called(ctrl_p=%p)\n", ctrl_p); 1068 1069 for (i = 0; i < PCIE_HP_MAX_SLOTS; i++) { 1070 if ((slot_p = ctrl_p->hc_slots[i]) == NULL) 1071 continue; 1072 1073 if (slot_p->hs_attn_btn_threadp != NULL) { 1074 mutex_enter(&ctrl_p->hc_mutex); 1075 slot_p->hs_attn_btn_thread_exit = B_TRUE; 1076 cv_signal(&slot_p->hs_attn_btn_cv); 1077 PCIE_DBG("pcishpc_destroy_slots: " 1078 "waiting for ATTN thread exit\n"); 1079 cv_wait(&slot_p->hs_attn_btn_cv, &ctrl_p->hc_mutex); 1080 PCIE_DBG("pcishpc_destroy_slots: " 1081 "ATTN thread exit\n"); 1082 cv_destroy(&slot_p->hs_attn_btn_cv); 1083 slot_p->hs_attn_btn_threadp = NULL; 1084 mutex_exit(&ctrl_p->hc_mutex); 1085 } 1086 1087 PCIE_DBG("pcishpc_destroy_slots() (shpc_p=%p)\n" 1088 "destroyed", slot_p); 1089 1090 pcie_hp_delete_occupant_props(dip, 1091 makedevice(ddi_driver_major(dip), 1092 slot_p->hs_minor)); 1093 1094 /* unregister the slot with DDI HP framework */ 1095 if (ndi_hp_unregister(dip, slot_p->hs_info.cn_name) != 1096 NDI_SUCCESS) { 1097 PCIE_DBG("pcishpc_destroy_slots() " 1098 "failed to unregister slot %d\n", 1099 slot_p->hs_phy_slot_num); 1100 return (DDI_FAILURE); 1101 } 1102 kmem_free(slot_p->hs_info.cn_name, 1103 strlen(slot_p->hs_info.cn_name) + 1); 1104 kmem_free(slot_p, sizeof (pcie_hp_slot_t)); 1105 } 1106 1107 return (DDI_SUCCESS); 1108 } 1109 1110 /* 1111 * pcishpc_enable_irqs() 1112 * 1113 * Enable/unmask the different IRQ's we support from the SHPC controller. 1114 */ 1115 int 1116 pcishpc_enable_irqs(pcie_hp_ctrl_t *ctrl_p) 1117 { 1118 uint32_t reg; 1119 int slot; 1120 1121 reg = pcishpc_read_reg(ctrl_p, PCI_HP_CTRL_SERR_INT_REG); 1122 1123 /* Enable all interrupts. */ 1124 reg &= ~PCI_HP_SERR_INT_MASK_ALL; 1125 1126 pcishpc_write_reg(ctrl_p, PCI_HP_CTRL_SERR_INT_REG, reg); 1127 1128 /* Unmask the interrupts for each slot. */ 1129 for (slot = 0; slot < ctrl_p->hc_num_slots_impl; slot++) { 1130 reg = pcishpc_read_reg(ctrl_p, PCI_HP_LOGICAL_SLOT_REGS+slot); 1131 if ((reg & PCI_HP_SLOT_STATE_MASK) == PCI_HP_SLOT_ENABLED) { 1132 reg &= ~(PCI_HP_SLOT_MASK_ALL | 1133 PCI_HP_SLOT_MRL_SERR_MASK); 1134 ctrl_p->hc_num_slots_connected++; 1135 if (ctrl_p->hc_curr_bus_speed == -1) 1136 ctrl_p->hc_curr_bus_speed = 1137 pcishpc_read_reg(ctrl_p, 1138 PCI_HP_PROF_IF_SBCR_REG) & 1139 PCI_HP_SBCR_SPEED_MASK; 1140 } else { 1141 reg &= ~(PCI_HP_SLOT_MASK_ALL); 1142 } 1143 1144 /* Enable/Unmask all slot interrupts. */ 1145 pcishpc_write_reg(ctrl_p, PCI_HP_LOGICAL_SLOT_REGS+slot, reg); 1146 } 1147 1148 PCIE_DBG("pcishpc_enable_irqs: ctrl_p 0x%p, " 1149 "current bus speed 0x%x, slots connected 0x%x\n", ctrl_p, 1150 ctrl_p->hc_curr_bus_speed, ctrl_p->hc_num_slots_connected); 1151 1152 return (DDI_SUCCESS); 1153 } 1154 1155 1156 /* 1157 * pcishpc_disable_irqs() 1158 * 1159 * Disable/Mask the different IRQ's we support from the SHPC controller. 1160 */ 1161 int 1162 pcishpc_disable_irqs(pcie_hp_ctrl_t *ctrl_p) 1163 { 1164 uint32_t reg; 1165 int slot; 1166 1167 reg = pcishpc_read_reg(ctrl_p, PCI_HP_CTRL_SERR_INT_REG); 1168 1169 /* Mask all interrupts. */ 1170 reg |= PCI_HP_SERR_INT_MASK_ALL; 1171 1172 pcishpc_write_reg(ctrl_p, PCI_HP_CTRL_SERR_INT_REG, reg); 1173 1174 /* Unmask the interrupts for each slot. */ 1175 for (slot = 0; slot < ctrl_p->hc_num_slots_impl; slot++) { 1176 reg = pcishpc_read_reg(ctrl_p, PCI_HP_LOGICAL_SLOT_REGS+slot); 1177 1178 /* Disable/Mask all slot interrupts. */ 1179 reg |= PCI_HP_SLOT_MASK_ALL; 1180 1181 pcishpc_write_reg(ctrl_p, PCI_HP_LOGICAL_SLOT_REGS+slot, reg); 1182 } 1183 1184 PCIE_DBG("pcishpc_disable_irqs: ctrl_p 0x%p, " 1185 "current bus speed 0x%x, slots connected 0x%x\n", ctrl_p, 1186 ctrl_p->hc_curr_bus_speed, ctrl_p->hc_num_slots_connected); 1187 1188 return (DDI_SUCCESS); 1189 } 1190 1191 /* 1192 * pcishpc_slot_poweron() 1193 * 1194 * Poweron/Enable the slot. 1195 * 1196 * Note: This function is called by DDI HP framework at kernel context only 1197 */ 1198 /*ARGSUSED*/ 1199 static int 1200 pcishpc_slot_poweron(pcie_hp_slot_t *slot_p, ddi_hp_cn_state_t *result_state) 1201 { 1202 uint32_t status; 1203 1204 PCIE_DBG("pcishpc_slot_poweron called()\n"); 1205 1206 ASSERT(MUTEX_HELD(&slot_p->hs_ctrl->hc_mutex)); 1207 1208 /* get the current slot state */ 1209 pcishpc_get_slot_state(slot_p); 1210 1211 /* check if the slot is already in the 'enabled' state */ 1212 if (slot_p->hs_info.cn_state >= DDI_HP_CN_STATE_POWERED) { 1213 /* slot is already in the 'enabled' state */ 1214 PCIE_DBG("pcishpc_slot_poweron() slot %d already enabled\n", 1215 slot_p->hs_phy_slot_num); 1216 1217 *result_state = slot_p->hs_info.cn_state; 1218 return (DDI_SUCCESS); 1219 } 1220 1221 if (slot_p->hs_info.cn_state == DDI_HP_CN_STATE_EMPTY) { 1222 PCIE_DBG("pcishpc_slot_poweron() slot in empty state\n"); 1223 goto cleanup; 1224 } 1225 1226 /* make sure the MRL sensor is closed */ 1227 status = pcishpc_read_reg(slot_p->hs_ctrl, 1228 PCI_HP_LOGICAL_SLOT_REGS+slot_p->hs_num); 1229 1230 if (status & PCI_HP_SLOT_MRL_STATE_MASK) { 1231 PCIE_DBG("pcishpc_slot_poweron() failed: MRL open\n"); 1232 goto cleanup; 1233 } 1234 1235 /* Set the Power LED to blink */ 1236 (void) pcishpc_setled(slot_p, PCIE_HP_POWER_LED, PCIE_HP_LED_BLINK); 1237 1238 /* Turn all other LEDS off */ 1239 (void) pcishpc_setled(slot_p, PCIE_HP_FAULT_LED, PCIE_HP_LED_OFF); 1240 (void) pcishpc_setled(slot_p, PCIE_HP_ATTN_LED, PCIE_HP_LED_OFF); 1241 (void) pcishpc_setled(slot_p, PCIE_HP_ACTIVE_LED, PCIE_HP_LED_OFF); 1242 1243 /* Set the bus speed only if the bus segment is not running */ 1244 if (pcishpc_set_bus_speed(slot_p) != DDI_SUCCESS) { 1245 PCIE_DBG("pcishpc_slot_poweron() setting speed failed\n"); 1246 goto cleanup; 1247 } 1248 1249 slot_p->hs_ctrl->hc_num_slots_connected++; 1250 1251 PCIE_DBG("pcishpc_slot_poweron(): slot_p 0x%p, slot state 0x%x, " 1252 "current bus speed 0x%x, slots connected 0x%x\n", slot_p, 1253 slot_p->hs_info.cn_state, slot_p->hs_ctrl->hc_curr_bus_speed, 1254 slot_p->hs_ctrl->hc_num_slots_connected); 1255 1256 /* Mask or Unmask MRL Sensor SEER bit based on new slot state */ 1257 if (slot_p->hs_ctrl->hc_has_mrl == B_TRUE) { 1258 uint32_t reg; 1259 1260 reg = pcishpc_read_reg(slot_p->hs_ctrl, 1261 PCI_HP_LOGICAL_SLOT_REGS+slot_p->hs_num); 1262 1263 pcishpc_write_reg(slot_p->hs_ctrl, 1264 PCI_HP_LOGICAL_SLOT_REGS+slot_p->hs_num, 1265 reg & ~PCI_HP_SLOT_MRL_SERR_MASK); 1266 } 1267 1268 /* Update the hardware slot state. */ 1269 if (pcishpc_set_slot_state(slot_p, 1270 DDI_HP_CN_STATE_ENABLED) != DDI_SUCCESS) { 1271 PCIE_DBG("pcishpc_slot_poweron() failed\n"); 1272 1273 pcishpc_get_slot_state(slot_p); 1274 goto cleanup; 1275 } 1276 /* Update the current state. It will be used in pcishpc_setled() */ 1277 slot_p->hs_info.cn_state = DDI_HP_CN_STATE_ENABLED; 1278 1279 /* Turn the Power LED ON for a enabled slot. */ 1280 (void) pcishpc_setled(slot_p, PCIE_HP_POWER_LED, PCIE_HP_LED_ON); 1281 1282 /* Turn all other LEDS off. */ 1283 (void) pcishpc_setled(slot_p, PCIE_HP_FAULT_LED, PCIE_HP_LED_OFF); 1284 (void) pcishpc_setled(slot_p, PCIE_HP_ATTN_LED, PCIE_HP_LED_OFF); 1285 (void) pcishpc_setled(slot_p, PCIE_HP_ACTIVE_LED, PCIE_HP_LED_OFF); 1286 1287 /* delay after powerON to let the device initialize itself */ 1288 delay(drv_usectohz(pcishpc_reset_delay)); 1289 1290 PCIE_DBG("pcishpc_slot_poweron() success!\n"); 1291 1292 /* 1293 * Want to show up as POWERED state for now. It will be updated to 1294 * ENABLED state when user explicitly enable the slot. 1295 */ 1296 slot_p->hs_info.cn_state = DDI_HP_CN_STATE_POWERED; 1297 1298 /* get the current slot state */ 1299 pcishpc_get_slot_state(slot_p); 1300 /* 1301 * It should be poweron'ed now. Have a check here in case any 1302 * hardware problems. 1303 */ 1304 if (slot_p->hs_info.cn_state < DDI_HP_CN_STATE_POWERED) { 1305 PCIE_DBG("pcishpc_slot_poweron() failed after hardware" 1306 " registers all programmed.\n"); 1307 1308 goto cleanup; 1309 } 1310 1311 *result_state = slot_p->hs_info.cn_state; 1312 1313 return (DDI_SUCCESS); 1314 1315 cleanup: 1316 (void) pcishpc_setled(slot_p, PCIE_HP_POWER_LED, PCIE_HP_LED_OFF); 1317 return (DDI_FAILURE); 1318 } 1319 1320 /*ARGSUSED*/ 1321 static int 1322 pcishpc_slot_poweroff(pcie_hp_slot_t *slot_p, ddi_hp_cn_state_t *result_state) 1323 { 1324 PCIE_DBG("pcishpc_slot_poweroff called()\n"); 1325 1326 ASSERT(MUTEX_HELD(&slot_p->hs_ctrl->hc_mutex)); 1327 1328 /* get the current slot state */ 1329 pcishpc_get_slot_state(slot_p); 1330 1331 /* check if the slot is not in the "enabled" or "powered" state */ 1332 if (slot_p->hs_info.cn_state < DDI_HP_CN_STATE_POWERED) { 1333 /* slot is in the 'disabled' state */ 1334 PCIE_DBG("pcishpc_slot_poweroff(): " 1335 "slot %d already disabled\n", slot_p->hs_phy_slot_num); 1336 1337 *result_state = slot_p->hs_info.cn_state; 1338 return (DDI_SUCCESS); 1339 } 1340 1341 /* Set the Power LED to blink */ 1342 (void) pcishpc_setled(slot_p, PCIE_HP_POWER_LED, PCIE_HP_LED_BLINK); 1343 1344 /* Turn all other LEDS off */ 1345 (void) pcishpc_setled(slot_p, PCIE_HP_FAULT_LED, PCIE_HP_LED_OFF); 1346 (void) pcishpc_setled(slot_p, PCIE_HP_ATTN_LED, PCIE_HP_LED_OFF); 1347 (void) pcishpc_setled(slot_p, PCIE_HP_ACTIVE_LED, PCIE_HP_LED_OFF); 1348 1349 if (--slot_p->hs_ctrl->hc_num_slots_connected == 0) 1350 slot_p->hs_ctrl->hc_curr_bus_speed = -1; 1351 1352 PCIE_DBG("pcishpc_slot_poweroff(): slot_p 0x%p, slot state 0x%x, " 1353 "current bus speed 0x%x, slots connected 0x%x\n", slot_p, 1354 slot_p->hs_info.cn_state, slot_p->hs_ctrl->hc_curr_bus_speed, 1355 slot_p->hs_ctrl->hc_num_slots_connected); 1356 1357 /* Mask or Unmask MRL Sensor SEER bit based on new slot state */ 1358 if (slot_p->hs_ctrl->hc_has_mrl == B_TRUE) { 1359 uint32_t reg; 1360 1361 reg = pcishpc_read_reg(slot_p->hs_ctrl, 1362 PCI_HP_LOGICAL_SLOT_REGS+slot_p->hs_num); 1363 1364 pcishpc_write_reg(slot_p->hs_ctrl, 1365 PCI_HP_LOGICAL_SLOT_REGS+slot_p->hs_num, 1366 reg | PCI_HP_SLOT_MRL_SERR_MASK); 1367 } 1368 1369 /* Update the hardware slot state. */ 1370 if (pcishpc_set_slot_state(slot_p, DDI_HP_CN_STATE_PRESENT) != 1371 DDI_SUCCESS) { 1372 PCIE_DBG("pcishpc_slot_poweroff() failed\n"); 1373 1374 pcishpc_get_slot_state(slot_p); 1375 goto cleanup; 1376 } 1377 1378 /* Update the current state. It will be used in pcishpc_setled() */ 1379 slot_p->hs_info.cn_state = DDI_HP_CN_STATE_PRESENT; 1380 1381 /* Turn the Power LED OFF for a disabled slot. */ 1382 (void) pcishpc_setled(slot_p, PCIE_HP_POWER_LED, PCIE_HP_LED_OFF); 1383 1384 /* Turn all other LEDS off. */ 1385 (void) pcishpc_setled(slot_p, PCIE_HP_FAULT_LED, PCIE_HP_LED_OFF); 1386 (void) pcishpc_setled(slot_p, PCIE_HP_ATTN_LED, PCIE_HP_LED_OFF); 1387 (void) pcishpc_setled(slot_p, PCIE_HP_ACTIVE_LED, PCIE_HP_LED_OFF); 1388 1389 /* delay after powerON to let the device initialize itself */ 1390 delay(drv_usectohz(pcishpc_reset_delay)); 1391 1392 pcishpc_get_slot_state(slot_p); 1393 /* 1394 * It should be poweroff'ed now. Have a check here in case any 1395 * hardware problems. 1396 */ 1397 if (slot_p->hs_info.cn_state > DDI_HP_CN_STATE_PRESENT) { 1398 PCIE_DBG("pcishpc_slot_poweroff() failed after hardware" 1399 " registers all programmed.\n"); 1400 1401 goto cleanup; 1402 } 1403 1404 PCIE_DBG("pcishpc_slot_poweroff() success!\n"); 1405 1406 *result_state = slot_p->hs_info.cn_state; 1407 return (DDI_SUCCESS); 1408 1409 cleanup: 1410 (void) pcishpc_setled(slot_p, PCIE_HP_POWER_LED, PCIE_HP_LED_OFF); 1411 return (DDI_FAILURE); 1412 } 1413 1414 /* 1415 * pcishpc_slot_probe() 1416 * 1417 * Probe the slot. 1418 * 1419 * Note: This function is called by DDI HP framework at kernel context only 1420 */ 1421 /*ARGSUSED*/ 1422 static int 1423 pcishpc_slot_probe(pcie_hp_slot_t *slot_p) 1424 { 1425 mutex_enter(&slot_p->hs_ctrl->hc_mutex); 1426 1427 PCIE_DBG("pcishpc_slot_probe called()\n"); 1428 1429 /* get the current slot state */ 1430 pcishpc_get_slot_state(slot_p); 1431 1432 /* 1433 * Probe a given PCI Hotplug Connection (CN). 1434 */ 1435 if (pcie_hp_probe(slot_p) != DDI_SUCCESS) { 1436 (void) pcishpc_setled(slot_p, PCIE_HP_ATTN_LED, 1437 PCIE_HP_LED_BLINK); 1438 1439 PCIE_DBG("pcishpc_slot_probe() failed\n"); 1440 1441 mutex_exit(&slot_p->hs_ctrl->hc_mutex); 1442 return (DDI_FAILURE); 1443 } 1444 1445 PCIE_DBG("pcishpc_slot_probe() success!\n"); 1446 1447 /* get the current slot state */ 1448 pcishpc_get_slot_state(slot_p); 1449 1450 mutex_exit(&slot_p->hs_ctrl->hc_mutex); 1451 return (DDI_SUCCESS); 1452 } 1453 1454 /* 1455 * pcishpc_slot_unprobe() 1456 * 1457 * Unprobe the slot. 1458 * 1459 * Note: This function is called by DDI HP framework at kernel context only 1460 */ 1461 /*ARGSUSED*/ 1462 static int 1463 pcishpc_slot_unprobe(pcie_hp_slot_t *slot_p) 1464 { 1465 mutex_enter(&slot_p->hs_ctrl->hc_mutex); 1466 1467 PCIE_DBG("pcishpc_slot_unprobe called()\n"); 1468 1469 /* get the current slot state */ 1470 pcishpc_get_slot_state(slot_p); 1471 1472 /* 1473 * Unprobe a given PCI Hotplug Connection (CN). 1474 */ 1475 if (pcie_hp_unprobe(slot_p) != DDI_SUCCESS) { 1476 (void) pcishpc_setled(slot_p, PCIE_HP_ATTN_LED, 1477 PCIE_HP_LED_BLINK); 1478 1479 PCIE_DBG("pcishpc_slot_unprobe() failed\n"); 1480 1481 mutex_exit(&slot_p->hs_ctrl->hc_mutex); 1482 return (DDI_FAILURE); 1483 } 1484 1485 PCIE_DBG("pcishpc_slot_unprobe() success!\n"); 1486 1487 /* get the current slot state */ 1488 pcishpc_get_slot_state(slot_p); 1489 1490 mutex_exit(&slot_p->hs_ctrl->hc_mutex); 1491 return (DDI_SUCCESS); 1492 } 1493 1494 static int 1495 pcishpc_upgrade_slot_state(pcie_hp_slot_t *slot_p, 1496 ddi_hp_cn_state_t target_state) 1497 { 1498 ddi_hp_cn_state_t curr_state; 1499 int rv = DDI_SUCCESS; 1500 1501 if (target_state > DDI_HP_CN_STATE_ENABLED) { 1502 return (DDI_EINVAL); 1503 } 1504 1505 curr_state = slot_p->hs_info.cn_state; 1506 while ((curr_state < target_state) && (rv == DDI_SUCCESS)) { 1507 1508 switch (curr_state) { 1509 case DDI_HP_CN_STATE_EMPTY: 1510 /* 1511 * From EMPTY to PRESENT, just check the hardware 1512 * slot state. 1513 */ 1514 pcishpc_get_slot_state(slot_p); 1515 curr_state = slot_p->hs_info.cn_state; 1516 if (curr_state < DDI_HP_CN_STATE_PRESENT) 1517 rv = DDI_FAILURE; 1518 break; 1519 case DDI_HP_CN_STATE_PRESENT: 1520 rv = pcishpc_slot_poweron(slot_p, &curr_state); 1521 break; 1522 case DDI_HP_CN_STATE_POWERED: 1523 curr_state = slot_p->hs_info.cn_state = 1524 DDI_HP_CN_STATE_ENABLED; 1525 break; 1526 default: 1527 /* should never reach here */ 1528 ASSERT("unknown devinfo state"); 1529 } 1530 } 1531 1532 return (rv); 1533 } 1534 1535 static int 1536 pcishpc_downgrade_slot_state(pcie_hp_slot_t *slot_p, 1537 ddi_hp_cn_state_t target_state) 1538 { 1539 ddi_hp_cn_state_t curr_state; 1540 int rv = DDI_SUCCESS; 1541 1542 1543 curr_state = slot_p->hs_info.cn_state; 1544 while ((curr_state > target_state) && (rv == DDI_SUCCESS)) { 1545 1546 switch (curr_state) { 1547 case DDI_HP_CN_STATE_PRESENT: 1548 /* 1549 * From PRESENT to EMPTY, just check hardware 1550 * slot state. 1551 */ 1552 pcishpc_get_slot_state(slot_p); 1553 curr_state = slot_p->hs_info.cn_state; 1554 if (curr_state >= DDI_HP_CN_STATE_PRESENT) 1555 rv = DDI_FAILURE; 1556 break; 1557 case DDI_HP_CN_STATE_POWERED: 1558 rv = pcishpc_slot_poweroff(slot_p, &curr_state); 1559 1560 break; 1561 case DDI_HP_CN_STATE_ENABLED: 1562 curr_state = slot_p->hs_info.cn_state = 1563 DDI_HP_CN_STATE_POWERED; 1564 1565 break; 1566 default: 1567 /* should never reach here */ 1568 ASSERT("unknown devinfo state"); 1569 } 1570 } 1571 1572 return (rv); 1573 } 1574 1575 /* Change slot state to a target state */ 1576 static int 1577 pcishpc_change_slot_state(pcie_hp_slot_t *slot_p, 1578 ddi_hp_cn_state_t target_state) 1579 { 1580 ddi_hp_cn_state_t curr_state; 1581 int rv; 1582 1583 pcishpc_get_slot_state(slot_p); 1584 curr_state = slot_p->hs_info.cn_state; 1585 1586 if (curr_state == target_state) { 1587 return (DDI_SUCCESS); 1588 } 1589 if (curr_state < target_state) { 1590 1591 rv = pcishpc_upgrade_slot_state(slot_p, target_state); 1592 } else { 1593 rv = pcishpc_downgrade_slot_state(slot_p, target_state); 1594 } 1595 1596 return (rv); 1597 } 1598 1599 /* 1600 * pcishpc_issue_command() 1601 * 1602 * Sends a command to the SHPC controller. 1603 */ 1604 static int 1605 pcishpc_issue_command(pcie_hp_ctrl_t *ctrl_p, uint32_t cmd_code) 1606 { 1607 int retCode; 1608 1609 ASSERT(MUTEX_HELD(&ctrl_p->hc_mutex)); 1610 1611 PCIE_DBG("pcishpc_issue_command() cmd_code=%02x\n", cmd_code); 1612 1613 ctrl_p->hc_cmd_pending = B_TRUE; 1614 1615 /* Write the command to the SHPC controller. */ 1616 pcishpc_write_reg(ctrl_p, PCI_HP_COMMAND_STATUS_REG, cmd_code); 1617 1618 while (ctrl_p->hc_cmd_pending == B_TRUE) 1619 cv_wait(&ctrl_p->hc_cmd_comp_cv, &ctrl_p->hc_mutex); 1620 1621 /* Wait until the SHPC controller processes the command. */ 1622 retCode = pcishpc_wait_busy(ctrl_p); 1623 1624 /* Make sure the command completed. */ 1625 if (retCode == DDI_SUCCESS) { 1626 /* Did the command fail to generate the command complete IRQ? */ 1627 if (ctrl_p->hc_cmd_pending != B_FALSE) { 1628 PCIE_DBG("pcishpc_issue_command() Failed on " 1629 "generate cmd complete IRQ\n"); 1630 retCode = DDI_FAILURE; 1631 } 1632 } 1633 1634 if (retCode == DDI_FAILURE) 1635 PCIE_DBG("pcishpc_issue_command() Failed on cmd_code=%02x\n", 1636 cmd_code); 1637 else 1638 PCIE_DBG("pcishpc_issue_command() Success on " 1639 "cmd_code=%02x\n", cmd_code); 1640 1641 return (retCode); 1642 } 1643 1644 /* 1645 * pcishpc_wait_busy() 1646 * 1647 * Wait until the SHPC controller is not busy. 1648 */ 1649 static int 1650 pcishpc_wait_busy(pcie_hp_ctrl_t *ctrl_p) 1651 { 1652 uint32_t status; 1653 1654 /* Wait until SHPC controller is NOT busy */ 1655 for (;;) { 1656 status = pcishpc_read_reg(ctrl_p, PCI_HP_COMMAND_STATUS_REG); 1657 1658 /* Is there an MRL Sensor error? */ 1659 if ((status & PCI_HP_COMM_STS_ERR_MASK) == 1660 PCI_HP_COMM_STS_ERR_MRL_OPEN) { 1661 PCIE_DBG("pcishpc_wait_busy() ERROR: " 1662 "MRL Sensor error\n"); 1663 break; 1664 } 1665 1666 /* Is there an Invalid command error? */ 1667 if ((status & PCI_HP_COMM_STS_ERR_MASK) == 1668 PCI_HP_COMM_STS_ERR_INVALID_COMMAND) { 1669 PCIE_DBG("pcishpc_wait_busy() ERROR: Invalid " 1670 "command error\n"); 1671 break; 1672 } 1673 1674 /* Is there an Invalid Speed/Mode error? */ 1675 if ((status & PCI_HP_COMM_STS_ERR_MASK) == 1676 PCI_HP_COMM_STS_ERR_INVALID_SPEED) { 1677 PCIE_DBG("pcishpc_wait_busy() ERROR: Invalid " 1678 "Speed/Mode error\n"); 1679 break; 1680 } 1681 1682 /* Is the SHPC controller not BUSY? */ 1683 if (!(status & PCI_HP_COMM_STS_CTRL_BUSY)) { 1684 /* Return Success. */ 1685 return (DDI_SUCCESS); 1686 } 1687 1688 PCIE_DBG("pcishpc_wait_busy() SHPC controller busy. Waiting\n"); 1689 1690 /* Wait before polling the status register again. */ 1691 delay(drv_usectohz(PCIE_HP_CMD_WAIT_TIME)); 1692 } 1693 1694 return (DDI_FAILURE); 1695 } 1696 1697 static void 1698 pcishpc_attn_btn_handler(pcie_hp_slot_t *slot_p) 1699 { 1700 pcie_hp_led_state_t hs_power_led_state; 1701 callb_cpr_t cprinfo; 1702 1703 PCIE_DBG("pcishpc_attn_btn_handler: thread started\n"); 1704 1705 CALLB_CPR_INIT(&cprinfo, &slot_p->hs_ctrl->hc_mutex, 1706 callb_generic_cpr, "pcishpc_attn_btn_handler"); 1707 1708 mutex_enter(&slot_p->hs_ctrl->hc_mutex); 1709 1710 /* wait for ATTN button event */ 1711 cv_wait(&slot_p->hs_attn_btn_cv, &slot_p->hs_ctrl->hc_mutex); 1712 1713 while (slot_p->hs_attn_btn_thread_exit == B_FALSE) { 1714 if (slot_p->hs_attn_btn_pending == B_TRUE) { 1715 /* get the current state of power LED */ 1716 hs_power_led_state = slot_p->hs_power_led_state; 1717 1718 /* Blink the Power LED while we wait for 5 seconds */ 1719 (void) pcishpc_setled(slot_p, PCIE_HP_POWER_LED, 1720 PCIE_HP_LED_BLINK); 1721 1722 /* wait for 5 seconds before taking any action */ 1723 if (cv_reltimedwait(&slot_p->hs_attn_btn_cv, 1724 &slot_p->hs_ctrl->hc_mutex, 1725 SEC_TO_TICK(5), TR_CLOCK_TICK) == -1) { 1726 /* 1727 * It is a time out; 1728 * make sure the ATTN pending flag is 1729 * still ON before sending the event 1730 * to DDI HP framework. 1731 */ 1732 if (slot_p->hs_attn_btn_pending == B_TRUE) { 1733 int hint; 1734 1735 /* restore the power LED state */ 1736 (void) pcishpc_setled(slot_p, 1737 PCIE_HP_POWER_LED, 1738 hs_power_led_state); 1739 /* 1740 * send the ATTN button event 1741 * to DDI HP framework 1742 */ 1743 slot_p->hs_attn_btn_pending = B_FALSE; 1744 1745 pcishpc_get_slot_state(slot_p); 1746 1747 if (slot_p->hs_info.cn_state <= 1748 DDI_HP_CN_STATE_PRESENT) { 1749 /* 1750 * Insertion. 1751 */ 1752 hint = SE_INCOMING_RES; 1753 } else { 1754 /* 1755 * Want to remove; 1756 */ 1757 hint = SE_OUTGOING_RES; 1758 } 1759 pcie_hp_gen_sysevent_req( 1760 slot_p->hs_info.cn_name, 1761 hint, 1762 slot_p->hs_ctrl->hc_dip, 1763 KM_SLEEP); 1764 1765 continue; 1766 } 1767 } 1768 1769 /* restore the power LED state */ 1770 (void) pcishpc_setled(slot_p, PCIE_HP_POWER_LED, 1771 hs_power_led_state); 1772 continue; 1773 } 1774 1775 /* wait for another ATTN button event */ 1776 cv_wait(&slot_p->hs_attn_btn_cv, &slot_p->hs_ctrl->hc_mutex); 1777 } 1778 1779 PCIE_DBG("pcishpc_attn_btn_handler: thread exit\n"); 1780 cv_signal(&slot_p->hs_attn_btn_cv); 1781 CALLB_CPR_EXIT(&cprinfo); 1782 thread_exit(); 1783 } 1784 1785 /* 1786 * pcishpc_get_slot_state() 1787 * 1788 * Get the state of the slot. 1789 * The slot state should have been initialized before this function gets called. 1790 */ 1791 static void 1792 pcishpc_get_slot_state(pcie_hp_slot_t *slot_p) 1793 { 1794 uint32_t reg; 1795 ddi_hp_cn_state_t curr_state = slot_p->hs_info.cn_state; 1796 1797 /* Read the logical slot register for this Slot. */ 1798 reg = pcishpc_read_reg(slot_p->hs_ctrl, 1799 PCI_HP_LOGICAL_SLOT_REGS+slot_p->hs_num); 1800 1801 /* Convert from the SHPC slot state to the HPC slot state. */ 1802 slot_p->hs_info.cn_state = pcishpc_slot_shpc_to_hpc(reg); 1803 if (curr_state == DDI_HP_CN_STATE_POWERED && 1804 slot_p->hs_info.cn_state > DDI_HP_CN_STATE_POWERED) { 1805 /* 1806 * Keep POWERED state if it is currently POWERED state because 1807 * this driver does not really implement enable/disable 1808 * slot operations. That is, when poweron, it actually enables 1809 * the slot also. 1810 * So, from hardware view, POWERED == ENABLED. 1811 * But, when user explicitly change to POWERED state, it should 1812 * be kept until user explicitly change to other states later. 1813 */ 1814 slot_p->hs_info.cn_state = DDI_HP_CN_STATE_POWERED; 1815 } 1816 1817 /* Convert from the SHPC Power LED state to the HPC Power LED state. */ 1818 slot_p->hs_power_led_state = pcishpc_led_shpc_to_hpc((reg>>2)&3); 1819 1820 /* Convert from the SHPC Attn LED state to the HPC Attn LED state. */ 1821 slot_p->hs_attn_led_state = pcishpc_led_shpc_to_hpc((reg>>4)&3); 1822 1823 /* We don't have a fault LED so just default it to OFF. */ 1824 slot_p->hs_fault_led_state = PCIE_HP_LED_OFF; 1825 1826 /* We don't have an active LED so just default it to OFF. */ 1827 slot_p->hs_active_led_state = PCIE_HP_LED_OFF; 1828 } 1829 1830 /* 1831 * pcishpc_set_slot_state() 1832 * 1833 * Updates the slot's state and leds. 1834 */ 1835 static int 1836 pcishpc_set_slot_state(pcie_hp_slot_t *slot_p, 1837 ddi_hp_cn_state_t new_slot_state) 1838 { 1839 uint32_t reg, cmd_code; 1840 ddi_hp_cn_state_t curr_state; 1841 1842 ASSERT(MUTEX_HELD(&slot_p->hs_ctrl->hc_mutex)); 1843 1844 reg = pcishpc_read_reg(slot_p->hs_ctrl, 1845 PCI_HP_LOGICAL_SLOT_REGS+slot_p->hs_num); 1846 1847 /* Default all states to unchanged. */ 1848 cmd_code = ((1 + slot_p->hs_num) << 8); 1849 1850 /* Has the slot state changed? */ 1851 curr_state = pcishpc_slot_shpc_to_hpc(reg); 1852 if (curr_state != new_slot_state) { 1853 PCIE_DBG("pcishpc_set_slot_state() Slot State changed"); 1854 1855 /* Set the new slot state in the Slot operation command. */ 1856 cmd_code |= pcishpc_slot_hpc_to_shpc(new_slot_state); 1857 } 1858 1859 /* Has the Power LED state changed? */ 1860 if (slot_p->hs_power_led_state != pcishpc_led_shpc_to_hpc((reg>>2)&3)) { 1861 PCIE_DBG("pcishpc_set_slot_state() Power LED State changed\n"); 1862 1863 /* Set the new power led state in the Slot operation command. */ 1864 cmd_code |= 1865 (pcishpc_led_hpc_to_shpc(slot_p->hs_power_led_state) << 2); 1866 } 1867 1868 /* Has the Attn LED state changed? */ 1869 if (slot_p->hs_attn_led_state != pcishpc_led_shpc_to_hpc((reg>>4)&3)) { 1870 PCIE_DBG("pcishpc_set_slot_state() Attn LED State changed\n"); 1871 1872 /* Set the new attn led state in the Slot operation command. */ 1873 cmd_code |= 1874 (pcishpc_led_hpc_to_shpc(slot_p->hs_attn_led_state) << 4); 1875 } 1876 1877 return (pcishpc_issue_command(slot_p->hs_ctrl, cmd_code)); 1878 } 1879 1880 /* 1881 * setup slot name/slot-number info. 1882 */ 1883 static void 1884 pcishpc_set_slot_name(pcie_hp_ctrl_t *ctrl_p, int slot) 1885 { 1886 pcie_hp_slot_t *slot_p = ctrl_p->hc_slots[slot]; 1887 pcie_bus_t *bus_p = PCIE_DIP2BUS(ctrl_p->hc_dip); 1888 uchar_t *slotname_data; 1889 int *slotnum; 1890 uint_t count; 1891 int len; 1892 uchar_t *s; 1893 uint32_t bit_mask; 1894 int pci_id_cnt, pci_id_bit; 1895 int slots_before, found; 1896 int invalid_slotnum = 0; 1897 1898 if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY, ctrl_p->hc_dip, 1899 DDI_PROP_DONTPASS, "physical-slot#", &slotnum, &count) == 1900 DDI_PROP_SUCCESS) { 1901 slot_p->hs_phy_slot_num = slotnum[0]; 1902 ddi_prop_free(slotnum); 1903 } else { 1904 if (ctrl_p->hc_device_increases) 1905 slot_p->hs_phy_slot_num = ctrl_p->hc_phys_start + slot; 1906 else 1907 slot_p->hs_phy_slot_num = ctrl_p->hc_phys_start - slot; 1908 1909 if ((ndi_prop_update_int(DDI_DEV_T_NONE, ctrl_p->hc_dip, 1910 "physical-slot#", slot_p->hs_phy_slot_num)) != DDI_SUCCESS) 1911 PCIE_DBG("pcishpc_set_slot_name(): failed to " 1912 "create phyical-slot#%d\n", 1913 slot_p->hs_phy_slot_num); 1914 } 1915 1916 /* Platform may not have initialized it */ 1917 if (!slot_p->hs_phy_slot_num) { 1918 slot_p->hs_phy_slot_num = pci_config_get8(bus_p->bus_cfg_hdl, 1919 PCI_BCNF_SECBUS); 1920 invalid_slotnum = 1; 1921 } 1922 slot_p->hs_info.cn_num = slot_p->hs_phy_slot_num; 1923 slot_p->hs_info.cn_num_dpd_on = DDI_HP_CN_NUM_NONE; 1924 1925 /* 1926 * construct the slot_name: 1927 * if "slot-names" property exists then use that name 1928 * else if valid slot number exists then it is "pci<slot-num>". 1929 * else it will be "pci<sec-bus-number>dev<dev-number>" 1930 */ 1931 if (ddi_getlongprop(DDI_DEV_T_ANY, ctrl_p->hc_dip, DDI_PROP_DONTPASS, 1932 "slot-names", (caddr_t)&slotname_data, &len) == DDI_PROP_SUCCESS) { 1933 bit_mask = slotname_data[3] | (slotname_data[2] << 8) | 1934 (slotname_data[1] << 16) | (slotname_data[0] << 24); 1935 1936 pci_id_bit = 1; 1937 pci_id_cnt = slots_before = found = 0; 1938 1939 /* 1940 * Walk the bit mask until we find the bit that corresponds 1941 * to our slots device number. We count how many bits 1942 * we find before we find our slot's bit. 1943 */ 1944 while (!found && (pci_id_cnt < 32)) { 1945 while (slot_p->hs_device_num != pci_id_cnt) { 1946 1947 /* 1948 * Find the next bit set. 1949 */ 1950 while (!(bit_mask & pci_id_bit) && 1951 (pci_id_cnt < 32)) { 1952 pci_id_bit = pci_id_bit << 1; 1953 pci_id_cnt++; 1954 } 1955 1956 if (slot_p->hs_device_num != pci_id_cnt) 1957 slots_before++; 1958 else 1959 found = 1; 1960 } 1961 } 1962 1963 if (pci_id_cnt < 32) { 1964 1965 /* 1966 * Set ptr to first string. 1967 */ 1968 s = slotname_data + 4; 1969 1970 /* 1971 * Increment past all the strings for the slots 1972 * before ours. 1973 */ 1974 while (slots_before) { 1975 while (*s != '\0') 1976 s++; 1977 s++; 1978 slots_before--; 1979 } 1980 1981 slot_p->hs_info.cn_name = i_ddi_strdup((char *)s, 1982 KM_SLEEP); 1983 kmem_free(slotname_data, len); 1984 return; 1985 } 1986 1987 /* slot-names entry not found */ 1988 PCIE_DBG("pcishpc_set_slot_name(): " 1989 "No slot-names entry found for slot #%d\n", 1990 slot_p->hs_phy_slot_num); 1991 kmem_free(slotname_data, len); 1992 } 1993 1994 if (invalid_slotnum) { 1995 char tmp_name[256]; 1996 1997 (void) snprintf(tmp_name, sizeof (tmp_name), "pci%d", 1998 slot_p->hs_device_num); 1999 slot_p->hs_info.cn_name = i_ddi_strdup(tmp_name, KM_SLEEP); 2000 } else { 2001 char tmp_name[256]; 2002 2003 (void) snprintf(tmp_name, sizeof (tmp_name), "pci%d", 2004 slot_p->hs_phy_slot_num); 2005 slot_p->hs_info.cn_name = i_ddi_strdup(tmp_name, KM_SLEEP); 2006 } 2007 } 2008 2009 /* 2010 * pcishpc_set_bus_speed() 2011 * 2012 * Set the bus speed and mode. 2013 */ 2014 static int 2015 pcishpc_set_bus_speed(pcie_hp_slot_t *slot_p) 2016 { 2017 pcie_hp_ctrl_t *ctrl_p = slot_p->hs_ctrl; 2018 int curr_speed = ctrl_p->hc_curr_bus_speed; 2019 int speed = -1; 2020 int avail_slots; 2021 uint32_t status, slots_avail1_reg, slots_avail2_reg; 2022 2023 ASSERT(MUTEX_HELD(&slot_p->hs_ctrl->hc_mutex)); 2024 2025 /* Make sure that the slot is in a correct state */ 2026 status = pcishpc_read_reg(ctrl_p, 2027 PCI_HP_LOGICAL_SLOT_REGS+slot_p->hs_num); 2028 2029 /* Return failure if the slot is empty */ 2030 if ((status & PCI_HP_SLOT_CARD_EMPTY_MASK) == 2031 PCI_HP_SLOT_CARD_EMPTY_MASK) { 2032 PCIE_DBG("pcishpc_set_bus_speed() failed: " 2033 "the slot is empty\n"); 2034 return (DDI_FAILURE); 2035 } 2036 2037 /* Return failure if the slot is not in disabled state */ 2038 if ((status & PCI_HP_SLOT_STATE_MASK) != PCI_HP_SLOT_DISABLED) { 2039 PCIE_DBG("pcishpc_set_bus_speed() failed: " 2040 "incorrect slot state\n"); 2041 return (DDI_FAILURE); 2042 } 2043 2044 /* Set the "power-only" mode for the slot */ 2045 if (pcishpc_issue_command(ctrl_p, ((1+slot_p->hs_num)<<8) | 2046 PCI_HP_SLOT_POWER_ONLY) != DDI_SUCCESS) { 2047 PCIE_DBG("pcishpc_set_bus_speed() failed to set " 2048 "the slot %d in the power-only mode\n", slot_p->hs_num); 2049 return (DDI_FAILURE); 2050 } 2051 2052 /* Wait for power good */ 2053 delay(drv_usectohz(PCIE_HP_POWER_GOOD_WAIT_TIME)); 2054 2055 /* Make sure that the slot is in "power-only" state */ 2056 status = pcishpc_read_reg(ctrl_p, 2057 PCI_HP_LOGICAL_SLOT_REGS+slot_p->hs_num); 2058 2059 if ((status & PCI_HP_SLOT_STATE_MASK) != PCI_HP_SLOT_POWER_ONLY) { 2060 PCIE_DBG("pcishpc_set_bus_speed() " 2061 "power-only failed: incorrect slot state\n"); 2062 return (DDI_FAILURE); 2063 } 2064 2065 slots_avail1_reg = pcishpc_read_reg(ctrl_p, 2066 PCI_HP_SLOTS_AVAIL_I_REG); 2067 slots_avail2_reg = pcishpc_read_reg(ctrl_p, 2068 PCI_HP_SLOTS_AVAIL_II_REG); 2069 2070 /* 2071 * Check if SHPC has available slots and select the highest 2072 * available bus speed for the slot. 2073 * 2074 * The bus speed codes are: 2075 * 100 - 133Mhz; <--+ 2076 * 011 - 100Mhz; <--+ PCI-X 2077 * 010 - 66Mhz; <--+ 2078 * 2079 * 001 - 66Mhz; <--+ 2080 * 000 - 33Mhz <--+ Conv PCI 2081 */ 2082 switch (status & PCI_HP_SLOT_PCIX_CAPABLE_MASK) { 2083 case PCI_HP_SLOT_133MHZ_PCIX_CAPABLE: 2084 avail_slots = (slots_avail1_reg >> 2085 PCI_HP_AVAIL_133MHZ_PCIX_SPEED_SHIFT) & 2086 PCI_HP_AVAIL_SPEED_MASK; 2087 2088 if (((curr_speed == -1) && avail_slots) || 2089 (curr_speed == PCI_HP_SBCR_133MHZ_PCIX_SPEED)) { 2090 speed = PCI_HP_SBCR_133MHZ_PCIX_SPEED; 2091 break; 2092 } 2093 /* FALLTHROUGH */ 2094 case PCI_HP_SLOT_100MHZ_PCIX_CAPABLE: 2095 avail_slots = (slots_avail1_reg >> 2096 PCI_HP_AVAIL_100MHZ_PCIX_SPEED_SHIFT) & 2097 PCI_HP_AVAIL_SPEED_MASK; 2098 2099 if (((curr_speed == -1) && avail_slots) || 2100 (curr_speed == PCI_HP_SBCR_100MHZ_PCIX_SPEED)) { 2101 speed = PCI_HP_SBCR_100MHZ_PCIX_SPEED; 2102 break; 2103 } 2104 /* FALLTHROUGH */ 2105 case PCI_HP_SLOT_66MHZ_PCIX_CAPABLE: 2106 avail_slots = (slots_avail1_reg >> 2107 PCI_HP_AVAIL_66MHZ_PCIX_SPEED_SHIFT) & 2108 PCI_HP_AVAIL_SPEED_MASK; 2109 2110 if (((curr_speed == -1) && avail_slots) || 2111 (curr_speed == PCI_HP_SBCR_66MHZ_PCIX_SPEED)) { 2112 speed = PCI_HP_SBCR_66MHZ_PCIX_SPEED; 2113 break; 2114 } 2115 /* FALLTHROUGH */ 2116 default: 2117 avail_slots = (slots_avail2_reg >> 2118 PCI_HP_AVAIL_66MHZ_CONV_SPEED_SHIFT) & 2119 PCI_HP_AVAIL_SPEED_MASK; 2120 2121 if ((status & PCI_HP_SLOT_66MHZ_CONV_CAPABLE) && 2122 (((curr_speed == -1) && avail_slots) || 2123 (curr_speed == PCI_HP_SBCR_66MHZ_CONV_SPEED))) { 2124 speed = PCI_HP_SBCR_66MHZ_CONV_SPEED; 2125 } else { 2126 avail_slots = (slots_avail1_reg >> 2127 PCI_HP_AVAIL_33MHZ_CONV_SPEED_SHIFT) & 2128 PCI_HP_AVAIL_SPEED_MASK; 2129 2130 if (((curr_speed == -1) && (avail_slots)) || 2131 (curr_speed == PCI_HP_SBCR_33MHZ_CONV_SPEED)) { 2132 speed = PCI_HP_SBCR_33MHZ_CONV_SPEED; 2133 } else { 2134 PCIE_DBG("pcishpc_set_bus_speed() " 2135 " failed to set the bus speed, slot# %d\n", 2136 slot_p->hs_num); 2137 return (DDI_FAILURE); 2138 } 2139 } 2140 break; 2141 } 2142 2143 /* 2144 * If the bus segment is already running, check to see the card 2145 * in the slot can support the current bus speed. 2146 */ 2147 if (curr_speed == speed) { 2148 /* 2149 * Check to see there is any slot available for the current 2150 * bus speed. Otherwise, we need fail the current slot connect 2151 * request. 2152 */ 2153 return ((avail_slots <= ctrl_p->hc_num_slots_connected) ? 2154 DDI_FAILURE : DDI_SUCCESS); 2155 } 2156 2157 /* Set the bus speed */ 2158 if (pcishpc_issue_command(ctrl_p, PCI_HP_COMM_STS_SET_SPEED | 2159 speed) == DDI_FAILURE) { 2160 PCIE_DBG("pcishpc_set_bus_speed() failed " 2161 "to set bus %d speed\n", slot_p->hs_num); 2162 return (DDI_FAILURE); 2163 } 2164 2165 /* Check the current bus speed */ 2166 status = pcishpc_read_reg(ctrl_p, PCI_HP_PROF_IF_SBCR_REG) & 2167 PCI_HP_SBCR_SPEED_MASK; 2168 if ((status & PCI_HP_SBCR_SPEED_MASK) != speed) { 2169 PCIE_DBG("pcishpc_set_bus_speed() an incorrect " 2170 "bus speed, slot = 0x%x, speed = 0x%x\n", 2171 slot_p->hs_num, status & PCI_HP_SBCR_SPEED_MASK); 2172 return (DDI_FAILURE); 2173 } 2174 2175 2176 /* Save the current bus speed */ 2177 ctrl_p->hc_curr_bus_speed = speed; 2178 2179 return (DDI_SUCCESS); 2180 } 2181 2182 /* 2183 * pcishpc_setled() 2184 * 2185 * Change the state of a slot's LED. 2186 */ 2187 static int 2188 pcishpc_setled(pcie_hp_slot_t *slot_p, pcie_hp_led_t led, 2189 pcie_hp_led_state_t state) 2190 { 2191 ASSERT(MUTEX_HELD(&slot_p->hs_ctrl->hc_mutex)); 2192 2193 switch (led) { 2194 case PCIE_HP_FAULT_LED: 2195 PCIE_DBG("pcishpc_setled() - PCIE_HP_FAULT_LED " 2196 "(set %s)\n", pcishpc_slot_textledstate(state)); 2197 slot_p->hs_fault_led_state = state; 2198 break; 2199 2200 case PCIE_HP_POWER_LED: 2201 PCIE_DBG("pcishpc_setled() - PCIE_HP_POWER_LED " 2202 "(set %s)\n", pcishpc_slot_textledstate(state)); 2203 slot_p->hs_power_led_state = state; 2204 break; 2205 2206 case PCIE_HP_ATTN_LED: 2207 PCIE_DBG("pcishpc_setled() - PCIE_HP_ATTN_LED " 2208 "(set %s)\n", pcishpc_slot_textledstate(state)); 2209 slot_p->hs_attn_led_state = state; 2210 break; 2211 2212 case PCIE_HP_ACTIVE_LED: 2213 PCIE_DBG("pcishpc_setled() - PCIE_HP_ACTIVE_LED " 2214 "(set %s)\n", pcishpc_slot_textledstate(state)); 2215 slot_p->hs_active_led_state = state; 2216 break; 2217 } 2218 2219 return (pcishpc_set_slot_state(slot_p, slot_p->hs_info.cn_state)); 2220 } 2221 2222 /* 2223 * pcishpc_led_shpc_to_hpc() 2224 * 2225 * Convert from SHPC indicator status to HPC indicator status. 2226 */ 2227 static int 2228 pcishpc_led_shpc_to_hpc(int state) 2229 { 2230 switch (state) { 2231 case 1: /* SHPC On bits b01 */ 2232 return (PCIE_HP_LED_ON); 2233 case 2: /* SHPC Blink bits b10 */ 2234 return (PCIE_HP_LED_BLINK); 2235 case 3: /* SHPC Off bits b11 */ 2236 return (PCIE_HP_LED_OFF); 2237 } 2238 2239 return (PCIE_HP_LED_OFF); 2240 } 2241 2242 2243 /* 2244 * pcishpc_led_hpc_to_shpc() 2245 * 2246 * Convert from HPC indicator status to SHPC indicator status. 2247 */ 2248 static int 2249 pcishpc_led_hpc_to_shpc(int state) 2250 { 2251 switch (state) { 2252 case PCIE_HP_LED_ON: 2253 return (1); /* SHPC On bits b01 */ 2254 case PCIE_HP_LED_BLINK: 2255 return (2); /* SHPC Blink bits b10 */ 2256 case PCIE_HP_LED_OFF: 2257 return (3); /* SHPC Off bits b11 */ 2258 } 2259 2260 return (3); /* SHPC Off bits b11 */ 2261 } 2262 2263 /* 2264 * pcishpc_slot_shpc_to_hpc() 2265 * 2266 * Convert from SHPC slot state to HPC slot state. 2267 * The argument shpc_state is expected to be read from the slot register. 2268 */ 2269 static int 2270 pcishpc_slot_shpc_to_hpc(int shpc_state) 2271 { 2272 if ((shpc_state & PCI_HP_SLOT_CARD_EMPTY_MASK) == 2273 PCI_HP_SLOT_CARD_EMPTY_MASK) 2274 return (DDI_HP_CN_STATE_EMPTY); 2275 2276 switch (shpc_state & PCI_HP_SLOT_STATE_MASK) { 2277 case PCI_HP_SLOT_POWER_ONLY: /* SHPC Powered Only */ 2278 return (DDI_HP_CN_STATE_POWERED); 2279 2280 case PCI_HP_SLOT_ENABLED: /* SHPC Enabled */ 2281 return (DDI_HP_CN_STATE_ENABLED); 2282 2283 case PCI_HP_SLOT_DISABLED: /* SHPC Disabled */ 2284 default : /* SHPC Reserved */ 2285 return (DDI_HP_CN_STATE_PRESENT); 2286 } 2287 } 2288 2289 /* 2290 * pcishpc_slot_hpc_to_shpc() 2291 * 2292 * Convert from HPC slot state to SHPC slot state. 2293 */ 2294 static int 2295 pcishpc_slot_hpc_to_shpc(int state) 2296 { 2297 switch (state) { 2298 case DDI_HP_CN_STATE_EMPTY: 2299 return (0); 2300 2301 case DDI_HP_CN_STATE_POWERED: 2302 return (PCI_HP_SLOT_POWER_ONLY); 2303 2304 case DDI_HP_CN_STATE_ENABLED: 2305 return (PCI_HP_SLOT_ENABLED); 2306 2307 default: 2308 return (PCI_HP_SLOT_DISABLED); 2309 } 2310 } 2311 2312 /* 2313 * pcishpc_slot_textslotstate() 2314 * 2315 * Convert the request into a text message. 2316 */ 2317 static char * 2318 pcishpc_slot_textslotstate(ddi_hp_cn_state_t state) 2319 { 2320 /* Convert an HPC slot state into a textual string. */ 2321 if (state == DDI_HP_CN_STATE_EMPTY) 2322 return ("HPC_SLOT_EMPTY"); 2323 else if (state == DDI_HP_CN_STATE_ENABLED) 2324 return ("HPC_SLOT_ENABLED"); 2325 else if (state == DDI_HP_CN_STATE_POWERED) 2326 return ("HPC_SLOT_POWERED_ONLY"); 2327 else 2328 return ("HPC_SLOT_DISABLED"); 2329 } 2330 2331 2332 /* 2333 * pcishpc_slot_textledstate() 2334 * 2335 * Convert the led state into a text message. 2336 */ 2337 static char * 2338 pcishpc_slot_textledstate(pcie_hp_led_state_t state) 2339 { 2340 /* Convert an HPC led state into a textual string. */ 2341 switch (state) { 2342 case PCIE_HP_LED_OFF: 2343 return ("off"); 2344 2345 case PCIE_HP_LED_ON: 2346 return ("on"); 2347 2348 case PCIE_HP_LED_BLINK: 2349 return ("blink"); 2350 } 2351 return ("unknown"); 2352 } 2353 2354 2355 /* 2356 * pcishpc_read_reg() 2357 * 2358 * Read from a SHPC controller register. 2359 */ 2360 static uint32_t 2361 pcishpc_read_reg(pcie_hp_ctrl_t *ctrl_p, int reg) 2362 { 2363 pcie_bus_t *bus_p = PCIE_DIP2BUS(ctrl_p->hc_dip); 2364 2365 /* Setup the SHPC dword select register. */ 2366 pci_config_put8(bus_p->bus_cfg_hdl, 2367 bus_p->bus_pci_hp_off + PCI_HP_DWORD_SELECT_OFF, (uint8_t)reg); 2368 2369 /* Read back the SHPC dword select register and verify. */ 2370 if (pci_config_get8(bus_p->bus_cfg_hdl, bus_p->bus_pci_hp_off + 2371 PCI_HP_DWORD_SELECT_OFF) != (uint8_t)reg) { 2372 PCIE_DBG("pcishpc_read_reg() - Failed writing DWORD " 2373 "select reg\n"); 2374 return (0xFFFFFFFF); 2375 } 2376 2377 /* Read from the SHPC dword data register. */ 2378 return (pci_config_get32(bus_p->bus_cfg_hdl, 2379 bus_p->bus_pci_hp_off + PCI_HP_DWORD_DATA_OFF)); 2380 } 2381 2382 2383 /* 2384 * pcishpc_write_reg() 2385 * 2386 * Write to a SHPC controller register. 2387 */ 2388 static void 2389 pcishpc_write_reg(pcie_hp_ctrl_t *ctrl_p, int reg, uint32_t data) 2390 { 2391 pcie_bus_t *bus_p = PCIE_DIP2BUS(ctrl_p->hc_dip); 2392 2393 /* Setup the SHPC dword select register. */ 2394 pci_config_put8(bus_p->bus_cfg_hdl, 2395 bus_p->bus_pci_hp_off + PCI_HP_DWORD_SELECT_OFF, (uint8_t)reg); 2396 2397 /* Read back the SHPC dword select register and verify. */ 2398 if (pci_config_get8(bus_p->bus_cfg_hdl, bus_p->bus_pci_hp_off + 2399 PCI_HP_DWORD_SELECT_OFF) != (uint8_t)reg) { 2400 PCIE_DBG("pcishpc_write_reg() - Failed writing " 2401 "DWORD select reg\n"); 2402 return; 2403 } 2404 2405 /* Write to the SHPC dword data register. */ 2406 pci_config_put32(bus_p->bus_cfg_hdl, 2407 bus_p->bus_pci_hp_off + PCI_HP_DWORD_DATA_OFF, data); 2408 2409 /* 2410 * Issue a read of the VendorID/DeviceID just to force the previous 2411 * write to complete. This is probably not necessary, but it does 2412 * help enforce ordering if there is an issue. 2413 */ 2414 (void) pci_config_get16(bus_p->bus_cfg_hdl, PCI_CONF_VENID); 2415 } 2416 2417 2418 #ifdef DEBUG 2419 /* 2420 * pcishpc_dump_regs() 2421 * 2422 * Dumps all of the SHPC controller registers. 2423 */ 2424 static void 2425 pcishpc_dump_regs(pcie_hp_ctrl_t *ctrl_p) 2426 { 2427 int slot, numSlots; 2428 uint32_t reg; 2429 char *state; 2430 2431 if (!pcie_debug_flags) 2432 return; 2433 2434 PCIE_DBG("pcishpc_dump_regs() called:\n"); 2435 PCIE_DBG("=========================================================="); 2436 2437 PCIE_DBG("SHPC Base Offset " 2438 ": 0x%08x\n", pcishpc_read_reg(ctrl_p, PCI_HP_BASE_OFFSET_REG)); 2439 2440 reg = pcishpc_read_reg(ctrl_p, PCI_HP_SLOTS_AVAIL_I_REG); 2441 2442 PCIE_DBG("Number of PCIX slots avail (33 Mhz) : %d\n", 2443 (reg & 31)); 2444 2445 PCIE_DBG("Number of PCIX slots avail (66 Mhz) : %d\n", 2446 ((reg>>8) & 31)); 2447 2448 PCIE_DBG("Number of PCIX slots avail (100 Mhz) : %d\n", 2449 ((reg>>16) & 31)); 2450 2451 PCIE_DBG("Number of PCIX slots avail (133 Mhz) : %d\n", 2452 ((reg>>24) & 31)); 2453 2454 reg = pcishpc_read_reg(ctrl_p, PCI_HP_SLOTS_AVAIL_II_REG); 2455 2456 PCIE_DBG("Number of conventional PCI slots (66 Mhz) : %d\n", 2457 (reg & 31)); 2458 2459 reg = pcishpc_read_reg(ctrl_p, PCI_HP_SLOT_CONFIGURATION_REG); 2460 2461 numSlots = (reg & 31); 2462 2463 PCIE_DBG("Number of Slots connected to this port : %d\n", 2464 numSlots); 2465 2466 PCIE_DBG("PCI Device # for First HotPlug Slot : %d\n", 2467 ((reg>>8) & 31)); 2468 2469 PCIE_DBG("Physical Slot # for First PCI Device # : %d\n", 2470 ((reg>>16) & 0x7ff)); 2471 2472 PCIE_DBG("Physical Slot Number Up/Down : %d\n", 2473 ((reg>>29) & 0x1)); 2474 2475 PCIE_DBG("MRL Sensor Implemented : %s\n", 2476 (reg & PCI_HP_SLOT_CONFIG_MRL_SENSOR) ? "Yes" : "No"); 2477 2478 PCIE_DBG("Attention Button Implemented : %s\n", 2479 (reg & PCI_HP_SLOT_CONFIG_ATTN_BUTTON) ? "Yes" : "No"); 2480 2481 reg = pcishpc_read_reg(ctrl_p, PCI_HP_PROF_IF_SBCR_REG); 2482 2483 switch (reg & 7) { 2484 case 0: 2485 state = "33Mhz Conventional PCI"; 2486 break; 2487 case 1: 2488 state = "66Mhz Conventional PCI"; 2489 break; 2490 case 2: 2491 state = "66Mhz PCI-X"; 2492 break; 2493 case 3: 2494 state = "100Mhz PCI-X"; 2495 break; 2496 case 4: 2497 state = "133Mhz PCI-X"; 2498 break; 2499 default: 2500 state = "Reserved (Error)"; 2501 break; 2502 } 2503 2504 PCIE_DBG("Current Port Operation Mode : %s\n", state); 2505 2506 PCIE_DBG("SHPC Interrupt Message Number : %d\n", 2507 ((reg>>16) &31)); 2508 2509 PCIE_DBG("SHPC Programming Interface : %d\n", 2510 ((reg>>24) & 0xff)); 2511 2512 reg = pcishpc_read_reg(ctrl_p, PCI_HP_COMMAND_STATUS_REG); 2513 2514 PCIE_DBG("SHPC Command Code : %d\n", 2515 (reg & 0xff)); 2516 2517 PCIE_DBG("SHPC Target Slot : %d\n", 2518 ((reg>>8) & 31)); 2519 2520 PCIE_DBG("SHPC Controller Busy : %s\n", 2521 ((reg>>16) & 1) ? "Yes" : "No"); 2522 2523 PCIE_DBG("SHPC Controller Err: MRL Sensor : %s\n", 2524 ((reg>>17) & 1) ? "Yes" : "No"); 2525 2526 PCIE_DBG("SHPC Controller Err: Invalid Command : %s\n", 2527 ((reg>>18) & 1) ? "Yes" : "No"); 2528 2529 PCIE_DBG("SHPC Controller Err: Invalid Speed/Mode : %s\n", 2530 ((reg>>19) & 1) ? "Yes" : "No"); 2531 2532 reg = pcishpc_read_reg(ctrl_p, PCI_HP_IRQ_LOCATOR_REG); 2533 2534 PCIE_DBG("Command Completion Interrupt Pending : %s\n", 2535 (reg & PCI_HP_IRQ_CMD_COMPLETE) ? "Yes" : "No"); 2536 2537 for (slot = 0; slot < numSlots; slot++) { 2538 PCIE_DBG("Slot %d Interrupt Pending : %s\n", slot+1, 2539 (reg & (PCI_HP_IRQ_SLOT_N_PENDING<<slot)) ? "Yes" : "No"); 2540 } 2541 2542 reg = pcishpc_read_reg(ctrl_p, PCI_HP_SERR_LOCATOR_REG); 2543 2544 PCIE_DBG("Arbiter SERR Pending : %s\n", 2545 (reg & PCI_HP_IRQ_SERR_ARBITER_PENDING) ? "Yes" : "No"); 2546 2547 for (slot = 0; slot < numSlots; slot++) { 2548 PCIE_DBG("Slot %d SERR Pending : %s\n", 2549 slot+1, (reg & 2550 (PCI_HP_IRQ_SERR_SLOT_N_PENDING<<slot)) ? "Yes" : "No"); 2551 } 2552 2553 reg = pcishpc_read_reg(ctrl_p, PCI_HP_CTRL_SERR_INT_REG); 2554 2555 PCIE_DBG("Global Interrupt Mask : %s\n", 2556 (reg & PCI_HP_SERR_INT_GLOBAL_IRQ_MASK) ? "Yes" : "No"); 2557 2558 PCIE_DBG("Global SERR Mask : %s\n", 2559 (reg & PCI_HP_SERR_INT_GLOBAL_SERR_MASK) ? "Yes" : "No"); 2560 2561 PCIE_DBG("Command Completion Interrupt Mask : %s\n", 2562 (reg & PCI_HP_SERR_INT_CMD_COMPLETE_MASK) ? "Yes" : "No"); 2563 2564 PCIE_DBG("Arbiter SERR Mask : %s\n", 2565 (reg & PCI_HP_SERR_INT_ARBITER_SERR_MASK) ? "Yes" : "No"); 2566 2567 PCIE_DBG("Command Completion Detected : %s\n", 2568 (reg & PCI_HP_SERR_INT_CMD_COMPLETE_IRQ) ? "Yes" : "No"); 2569 2570 PCIE_DBG("Arbiter Timeout Detected : %s\n", 2571 (reg & PCI_HP_SERR_INT_ARBITER_IRQ) ? "Yes" : "No"); 2572 2573 for (slot = 0; slot < numSlots; slot++) { 2574 PCIE_DBG("Logical Slot %d Registers:\n", slot+1); 2575 PCIE_DBG("------------------------------------\n"); 2576 2577 reg = pcishpc_read_reg(ctrl_p, PCI_HP_LOGICAL_SLOT_REGS+slot); 2578 2579 PCIE_DBG("Slot %d state : %s\n", slot+1, 2580 pcishpc_slot_textslotstate(pcishpc_slot_shpc_to_hpc(reg))); 2581 2582 PCIE_DBG("Slot %d Power Indicator State : %s\n", slot+1, 2583 pcishpc_slot_textledstate(pcishpc_led_shpc_to_hpc( 2584 (reg>>2) &3))); 2585 2586 PCIE_DBG("Slot %d Attention Indicator State : %s\n", slot+1, 2587 pcishpc_slot_textledstate(pcishpc_led_shpc_to_hpc( 2588 (reg>>4)&3))); 2589 2590 PCIE_DBG("Slot %d Power Fault : %s\n", slot+1, 2591 ((reg>>6)&1) ? "Fault Detected" : "No Fault"); 2592 PCIE_DBG("Slot %d Attention Button : %s\n", slot+1, 2593 ((reg>>7)&1) ? "Depressed" : "Not Depressed"); 2594 PCIE_DBG("Slot %d MRL Sensor : %s\n", slot+1, 2595 ((reg>>8)&1) ? "Not Closed" : "Closed"); 2596 PCIE_DBG("Slot %d 66mhz Capable : %s\n", slot+1, 2597 ((reg>>9)&1) ? "66mhz" : "33mgz"); 2598 2599 switch ((reg>>10)&3) { 2600 case 0: 2601 state = "Card Present 7.5W"; 2602 break; 2603 case 1: 2604 state = "Card Present 15W"; 2605 break; 2606 case 2: 2607 state = "Card Present 25W"; 2608 break; 2609 case 3: 2610 state = "Slot Empty"; 2611 break; 2612 } 2613 2614 PCIE_DBG("Slot %d PRSNT1#/PRSNT2# : %s\n", slot+1, 2615 state); 2616 2617 switch ((reg>>12)&3) { 2618 case 0: 2619 state = "Non PCI-X"; 2620 break; 2621 case 1: 2622 state = "66mhz PCI-X"; 2623 break; 2624 case 2: 2625 state = "Reserved"; 2626 break; 2627 case 3: 2628 state = "133mhz PCI-X"; 2629 break; 2630 } 2631 2632 PCIE_DBG("Slot %d Card Presence Change Detected : %s\n", 2633 slot+1, (reg & PCI_HP_SLOT_PRESENCE_DETECTED) ? "Yes" : 2634 "No"); 2635 PCIE_DBG("Slot %d Isolated Power Fault Detected : %s\n", 2636 slot+1, (reg & PCI_HP_SLOT_ISO_PWR_DETECTED) ? "Yes" : 2637 "No"); 2638 PCIE_DBG("Slot %d Attention Button Press Detected : %s\n", 2639 slot+1, (reg & PCI_HP_SLOT_ATTN_DETECTED) ? "Yes" : "No"); 2640 PCIE_DBG("Slot %d MRL Sensor Change Detected : %s\n", 2641 slot+1, (reg & PCI_HP_SLOT_MRL_DETECTED) ? "Yes" : "No"); 2642 PCIE_DBG("Slot %d Connected Power Fault Detected : %s\n", 2643 slot+1, (reg & PCI_HP_SLOT_POWER_DETECTED) ? "Yes" : "No"); 2644 2645 PCIE_DBG("Slot %d Card Presence IRQ Masked : %s\n", 2646 slot+1, (reg & PCI_HP_SLOT_PRESENCE_MASK) ? "Yes" : "No"); 2647 PCIE_DBG("Slot %d Isolated Power Fault IRQ Masked : %s\n", 2648 slot+1, (reg & PCI_HP_SLOT_ISO_PWR_MASK) ? "Yes" : "No"); 2649 PCIE_DBG("Slot %d Attention Button IRQ Masked : %s\n", 2650 slot+1, (reg & PCI_HP_SLOT_ATTN_MASK) ? "Yes" : "No"); 2651 PCIE_DBG("Slot %d MRL Sensor IRQ Masked : %s\n", 2652 slot+1, (reg & PCI_HP_SLOT_MRL_MASK) ? "Yes" : "No"); 2653 PCIE_DBG("Slot %d Connected Power Fault IRQ Masked : %s\n", 2654 slot+1, (reg & PCI_HP_SLOT_POWER_MASK) ? "Yes" : "No"); 2655 PCIE_DBG("Slot %d MRL Sensor SERR Masked : %s\n", 2656 slot+1, (reg & PCI_HP_SLOT_MRL_SERR_MASK) ? "Yes" : "No"); 2657 PCIE_DBG("Slot %d Connected Power Fault SERR Masked : %s\n", 2658 slot+1, (reg & PCI_HP_SLOT_POWER_SERR_MASK) ? "Yes" : "No"); 2659 } 2660 } 2661 #endif /* DEBUG */ 2662