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