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, Version 1.0 only 6 * (the "License"). You may not use this file except in compliance 7 * with the License. 8 * 9 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 10 * or http://www.opensolaris.org/os/licensing. 11 * See the License for the specific language governing permissions 12 * and limitations under the License. 13 * 14 * When distributing Covered Code, include this CDDL HEADER in each 15 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 16 * If applicable, add the following below this CDDL HEADER, with the 17 * fields enclosed by brackets "[]" replaced with your own identifying 18 * information: Portions Copyright [yyyy] [name of copyright owner] 19 * 20 * CDDL HEADER END 21 */ 22 /* 23 * Copyright 2005 Sun Microsystems, Inc. All rights reserved. 24 * Use is subject to license terms. 25 */ 26 27 #pragma ident "%Z%%M% %I% %E% SMI" 28 29 /* 30 * Intel 21554 PCI to PCI bus bridge nexus driver for sun4u platforms. 31 * Please note that 21554 is not a transparent bridge. 32 * This driver can be used when the 21554 bridge is used like a 33 * transparent bridge. The host OBP or the OS PCI Resource Allocator 34 * (during a hotplug/hotswap operation) must represent this device 35 * as a nexus and do the device tree representation of the child 36 * nodes underneath. 37 * Interrupt routing of the children must be done as per the PCI 38 * specifications recommendation similar to that of a transparent 39 * bridge. 40 * Address translations from secondary across primary can be 1:1 41 * or non 1:1. Currently only 1:1 translations are supported. 42 * Configuration cycles are indirect. Memory and IO cycles are direct. 43 */ 44 45 /* 46 * INCLUDES 47 */ 48 #include <sys/stat.h> 49 #include <sys/conf.h> 50 #include <sys/kmem.h> 51 #include <sys/debug.h> 52 #include <sys/modctl.h> 53 #include <sys/autoconf.h> 54 #include <sys/ddi_impldefs.h> 55 #include <sys/ddi_subrdefs.h> 56 #include <sys/pci.h> 57 #include <sys/pci/pci_nexus.h> 58 #include <sys/pci/pci_regs.h> 59 #include <sys/pci/db21554_config.h> /* 21554 configuration space registers */ 60 #include <sys/pci/db21554_csr.h> /* 21554 control status register layout */ 61 #include <sys/pci/db21554_ctrl.h> /* driver private control structure */ 62 #include <sys/pci/db21554_debug.h> /* driver debug declarations */ 63 #include <sys/ddi.h> 64 #include <sys/sunddi.h> 65 #include <sys/sunndi.h> 66 #include <sys/fm/protocol.h> 67 #include <sys/ddifm.h> 68 #include <sys/promif.h> 69 #include <sys/file.h> 70 #include <sys/hotplug/pci/pcihp.h> 71 72 /* 73 * DEFINES. 74 */ 75 #define DB_DEBUG 76 #define DB_MODINFO_DESCRIPTION "Intel/21554 pci-pci nexus:v%I%" 77 #define DB_DVMA_START 0xc0000000 78 #define DB_DVMA_LEN 0x20000000 79 80 #ifdef DB_DEBUG 81 /* ioctl definitions */ 82 #define DB_PCI_READ_CONF_HEADER 1 83 #define DEF_INVALID_REG_VAL -1 84 85 /* Default values for secondary cache line and latency timer */ 86 #define DB_SEC_LATENCY_TIMER_VAL 0x40 87 #define DB_SEC_CACHELN_SIZE_VAL 0x10 88 89 /* complete chip status information */ 90 typedef struct db_pci_data { 91 char name[256]; 92 uint32_t instance; 93 db_pci_header_t pri_hdr; 94 db_pci_header_t sec_hdr; 95 db_conf_regs_t conf_regs; 96 } db_pci_data_t; 97 #endif 98 99 /* 100 * LOCALS 101 */ 102 103 /* 104 * The next set of variables are control parameters for debug purposes only. 105 * Changing the default values as assigned below are not recommended. 106 * In some cases, the non-default values are mostly application specific and 107 * hence may not have been tested yet. 108 * 109 * db_conf_map_mode : specifies the access method used for generating 110 * configuration cycles. Default value indicates 111 * the indirect configuration method. 112 * db_io_map_mode : specifies the access method used for generating 113 * IO cycles. Default value indicates the direct 114 * method. 115 * db_pci_own_wait : For indirect cycles, indicates the wait period 116 * for acquiring the bus, when the bus is busy. 117 * db_pci_release_wait:For indirect cycles, indicates the wait period 118 * for releasing the bus when the bus is busy. 119 * db_pci_max_wait : max. wait time when bus is busy for indirect cycles 120 * db_set_latency_timer_register : 121 * when 1, the driver overwrites the OBP assigned 122 * latency timer register setting for every child 123 * device during child initialization. 124 * db_set_cache_line_size_register : 125 * when 1, the driver overwrites the OBP assigned 126 * cache line register setting for every child 127 * device during child initialization. 128 * db_use_config_own_bit: 129 * when 1, the driver will use the "config own bit" 130 * for accessing the configuration address and data 131 * registers. 132 */ 133 static uint32_t db_pci_own_wait = DB_PCI_WAIT_MS; 134 static uint32_t db_pci_release_wait = DB_PCI_WAIT_MS; 135 static uint32_t db_pci_max_wait = DB_PCI_TIMEOUT; 136 static uint32_t db_conf_map_mode = DB_CONF_MAP_INDIRECT_CONF; 137 static uint32_t db_io_map_mode = DB_IO_MAP_DIRECT; 138 static uint32_t db_set_latency_timer_register = 1; 139 static uint32_t db_set_cache_line_size_register = 1; 140 static uint32_t db_use_config_own_bit = 0; 141 142 /* 143 * Properties that can be set via .conf files. 144 */ 145 146 /* 147 * By default, we forward SERR# from secondary to primary. This behavior 148 * can be controlled via a property "serr-fwd-enable", type integer. 149 * Values are 0 or 1. 150 * 0 means 'do not forward SERR#'. 151 * 1 means forwards SERR# to the host. Should be the default. 152 */ 153 static uint32_t db_serr_fwd_enable = 1; 154 155 /* 156 * The next set of parameters are performance tuning parameters. 157 * These are in the form of properties settable through a .conf file. 158 * In case if the properties are absent the following defaults are assumed. 159 * These initial default values can be overwritten via /etc/system also. 160 * 161 * -1 means no setting is done ie. we either get OBP assigned value 162 * or reset values (at hotplug time for example). 163 */ 164 165 /* primary latency timer: property "p-latency-timer" : type integer */ 166 static int8_t p_latency_timer = DEF_INVALID_REG_VAL; 167 168 /* secondary latency timer: property "s-latency-timer": type integer */ 169 /* 170 * Currently on the secondary side the latency timer is not 171 * set by the serial PROM which causes performance degradation. 172 * Set the secondary latency timer register. 173 */ 174 static int8_t s_latency_timer = DB_SEC_LATENCY_TIMER_VAL; 175 176 /* primary cache line size: property "p-cache-line-size" : type integer */ 177 static int8_t p_cache_line_size = DEF_INVALID_REG_VAL; 178 179 /* secondary cache line size: property "s-cache-line-size" : type integer */ 180 /* 181 * Currently on the secondary side the cache line size is not 182 * set by the serial PROM which causes performance degradation. 183 * Set the secondary cache line size register. 184 */ 185 static int8_t s_cache_line_size = DB_SEC_CACHELN_SIZE_VAL; 186 187 /* 188 * control primary posted write queue threshold limit: 189 * property "p-pwrite-threshold" : type integer : values are 0 or 1. 190 * 1 enables control. 0 does not, and is the default reset value. 191 */ 192 static int8_t p_pwrite_threshold = DEF_INVALID_REG_VAL; 193 194 /* 195 * control secondary posted write queue threshold limit: 196 * property "s-pwrite-threshold" : type integer : values are 0 or 1. 197 * 1 enables control. 0 does not, and is the default reset value. 198 */ 199 static int8_t s_pwrite_threshold = DEF_INVALID_REG_VAL; 200 201 /* 202 * control read queue threshold for initiating delayed read transaction 203 * on primary bus. 204 * property "p-dread-threshold" : type integer: values are 205 * 206 * 0 : reset value, default behavior: at least 8DWords free for all MR 207 * 1 : reserved 208 * 2 : at least one cache line free for MRL and MRM, 8 DWords free for MR 209 * 3 : at least one cache line free for all MR 210 */ 211 static int8_t p_dread_threshold = DEF_INVALID_REG_VAL; 212 213 /* 214 * control read queue threshold for initiating delayed read transaction 215 * on secondary bus. 216 * property "s-dread-threshold" : type integer: values are 217 * 218 * 0 : reset value, default behavior: at least 8DWords free for all MR 219 * 1 : reserved 220 * 2 : at least one cache line free for MRL and MRM, 8 DWords free for MR 221 * 3 : at least one cache line free for all MR 222 */ 223 static int8_t s_dread_threshold = DEF_INVALID_REG_VAL; 224 225 /* 226 * control how 21554 issues delayed transactions on the target bus. 227 * property "delayed-trans-order" : type integer: values are 0 or 1. 228 * 1 means repeat transaction on same target on target retries. 229 * 0 is the reset/default value, and means enable round robin based 230 * reads on other targets in read queue on any target retries. 231 */ 232 static int8_t delayed_trans_order = DEF_INVALID_REG_VAL; 233 234 /* 235 * In case if the system DVMA information is not available, as it is 236 * prior to s28q1, the system dvma range can be set via these parameters. 237 */ 238 static uint32_t db_dvma_start = DB_DVMA_START; 239 static uint32_t db_dvma_len = DB_DVMA_LEN; 240 241 /* 242 * Default command register settings for all PCI nodes this nexus initializes. 243 */ 244 static uint16_t db_command_default = 245 PCI_COMM_SERR_ENABLE | 246 PCI_COMM_PARITY_DETECT | 247 PCI_COMM_ME | 248 PCI_COMM_MAE | 249 PCI_COMM_IO | 250 PCI_COMM_BACK2BACK_ENAB | 251 PCI_COMM_MEMWR_INVAL; 252 253 static int db_attach(dev_info_t *dip, ddi_attach_cmd_t cmd); 254 static int db_detach(dev_info_t *dip, ddi_detach_cmd_t cmd); 255 static void db_get_perf_parameters(db_ctrl_t *dbp); 256 static void db_set_perf_parameters(db_ctrl_t *dbp); 257 static void db_enable_io(db_ctrl_t *dbp); 258 static void db_orientation(db_ctrl_t *dbp); 259 static void db_set_dvma_range(db_ctrl_t *dbp); 260 static int db_getinfo(dev_info_t *dip, ddi_info_cmd_t infocmd, void *arg, 261 void **result); 262 static int db_pci_map(dev_info_t *, dev_info_t *, ddi_map_req_t *, 263 off_t, off_t, caddr_t *); 264 static int db_ctlops(dev_info_t *, dev_info_t *, ddi_ctl_enum_t, 265 void *, void *); 266 static int db_intr_ops(dev_info_t *dip, dev_info_t *rdip, 267 ddi_intr_op_t intr_op, ddi_intr_handle_impl_t *hdlp, 268 void *result); 269 static dev_info_t *db_get_my_childs_dip(dev_info_t *dip, dev_info_t *rdip); 270 static int db_fm_init_child(dev_info_t *dip, dev_info_t *tdip, int cap, 271 ddi_iblock_cookie_t *ibc); 272 static void db_bus_enter(dev_info_t *dip, ddi_acc_handle_t handle); 273 static void db_bus_exit(dev_info_t *dip, ddi_acc_handle_t handle); 274 275 struct bus_ops db_bus_ops = { 276 BUSO_REV, 277 db_pci_map, 278 0, 279 0, 280 0, 281 i_ddi_map_fault, 282 ddi_dma_map, 283 ddi_dma_allochdl, 284 ddi_dma_freehdl, 285 ddi_dma_bindhdl, 286 ddi_dma_unbindhdl, 287 ddi_dma_flush, 288 ddi_dma_win, 289 ddi_dma_mctl, 290 db_ctlops, 291 ddi_bus_prop_op, 292 ndi_busop_get_eventcookie, 293 ndi_busop_add_eventcall, 294 ndi_busop_remove_eventcall, 295 ndi_post_event, 296 0, 297 0, 298 0, 299 db_fm_init_child, 300 NULL, 301 db_bus_enter, 302 db_bus_exit, 303 0, 304 db_intr_ops 305 }; 306 307 static int db_open(dev_t *dev_p, int flag, int otyp, cred_t *cred_p); 308 static int db_close(dev_t dev, int flag, int otyp, cred_t *cred_p); 309 static int db_ioctl(dev_t dev, int cmd, intptr_t arg, int flag, 310 cred_t *cred_p, int *rval_p); 311 #ifdef DB_DEBUG 312 static dev_info_t *db_lookup_child_name(db_ctrl_t *dbp, char *name, 313 int instance); 314 static void db_pci_get_header(ddi_acc_handle_t config_handle, 315 db_pci_header_t *ph, off_t hdr_off); 316 static void db_pci_get_conf_regs(ddi_acc_handle_t config_handle, 317 db_conf_regs_t *cr); 318 #endif /* DB_DEBUG */ 319 320 #ifdef DEBUG 321 static void 322 db_debug(uint64_t func_id, dev_info_t *dip, char *fmt, 323 uintptr_t a1, uintptr_t a2, uintptr_t a3, uintptr_t a4, uintptr_t a5); 324 #endif 325 326 static int db_prop_op(dev_t dev, dev_info_t *dip, ddi_prop_op_t prop_op, 327 int flags, char *name, caddr_t valuep, int *lengthp); 328 329 static struct cb_ops db_cb_ops = { 330 db_open, /* open */ 331 db_close, /* close */ 332 nulldev, /* strategy */ 333 nulldev, /* print */ 334 nulldev, /* dump */ 335 nulldev, /* read */ 336 nulldev, /* write */ 337 db_ioctl, /* ioctl */ 338 nodev, /* devmap */ 339 nodev, /* mmap */ 340 nodev, /* segmap */ 341 nochpoll, /* poll */ 342 db_prop_op, /* cb_prop_op */ 343 NULL, /* streamtab */ 344 D_NEW | D_MP | D_HOTPLUG, /* Driver compatibility flag */ 345 CB_REV, /* rev */ 346 nodev, /* int (*cb_aread)() */ 347 nodev /* int (*cb_awrite)() */ 348 }; 349 350 static uint8_t db_ddi_get8(ddi_acc_impl_t *handle, uint8_t *addr); 351 static uint16_t db_ddi_get16(ddi_acc_impl_t *handle, uint16_t *addr); 352 static uint32_t db_ddi_get32(ddi_acc_impl_t *handle, uint32_t *addr); 353 static uint64_t db_ddi_get64(ddi_acc_impl_t *handle, uint64_t *addr); 354 static void db_ddi_put8(ddi_acc_impl_t *handle, uint8_t *addr, 355 uint8_t data); 356 static void db_ddi_put16(ddi_acc_impl_t *handle, uint16_t *addr, 357 uint16_t data); 358 static void db_ddi_put32(ddi_acc_impl_t *handle, uint32_t *addr, 359 uint32_t data); 360 static void db_ddi_put64(ddi_acc_impl_t *handle, uint64_t *addr, 361 uint64_t data); 362 static void db_ddi_rep_get8(ddi_acc_impl_t *handle, uint8_t *host_addr, 363 uint8_t *dev_addr, size_t repcount, uint_t flags); 364 static void db_ddi_rep_get16(ddi_acc_impl_t *handle, uint16_t *host_addr, 365 uint16_t *dev_addr, size_t repcount, uint_t flags); 366 static void db_ddi_rep_get32(ddi_acc_impl_t *handle, uint32_t *host_addr, 367 uint32_t *dev_addr, size_t repcount, uint_t flags); 368 static void db_ddi_rep_get64(ddi_acc_impl_t *handle, uint64_t *host_addr, 369 uint64_t *dev_addr, size_t repcount, uint_t flags); 370 static void db_ddi_rep_put8(ddi_acc_impl_t *handle, uint8_t *host_addr, 371 uint8_t *dev_addr, size_t repcount, uint_t flags); 372 static void db_ddi_rep_put16(ddi_acc_impl_t *handle, uint16_t *host_addr, 373 uint16_t *dev_addr, size_t repcount, uint_t flags); 374 static void db_ddi_rep_put32(ddi_acc_impl_t *handle, uint32_t *host_addr, 375 uint32_t *dev_addr, size_t repcount, uint_t flags); 376 static void db_ddi_rep_put64(ddi_acc_impl_t *handle, uint64_t *host_addr, 377 uint64_t *dev_addr, size_t repcount, uint_t flags); 378 379 static struct dev_ops db_dev_ops = { 380 DEVO_REV, /* devo_rev */ 381 0, /* refcnt */ 382 db_getinfo, /* info */ 383 nulldev, /* identify */ 384 nulldev, /* probe */ 385 db_attach, /* attach */ 386 db_detach, /* detach */ 387 nulldev, /* reset */ 388 &db_cb_ops, /* driver operations */ 389 &db_bus_ops, /* bus operations */ 390 ddi_power 391 }; 392 393 394 /* 395 * Module linkage information for the kernel. 396 */ 397 398 static struct modldrv modldrv = { 399 &mod_driverops, /* Type of module */ 400 DB_MODINFO_DESCRIPTION, 401 &db_dev_ops /* driver ops */ 402 }; 403 404 static struct modlinkage modlinkage = { 405 MODREV_1, 406 (void *)&modldrv, 407 NULL 408 }; 409 410 /* soft state pointer and structure template. */ 411 static void *db_state; 412 413 /* 414 * forward function declarations: 415 */ 416 static void db_uninitchild(dev_info_t *); 417 static int db_initchild(dev_info_t *child); 418 static int db_create_pci_prop(dev_info_t *child); 419 static int db_save_config_regs(db_ctrl_t *dbp); 420 static int db_restore_config_regs(db_ctrl_t *dbp); 421 422 /* 423 * FMA error callback 424 * Register error handling callback with our parent. We will just call 425 * our children's error callbacks and return their status. 426 */ 427 static int db_err_callback(dev_info_t *dip, ddi_fm_error_t *derr, 428 const void *impl_data); 429 430 /* 431 * init/fini routines to alloc/dealloc fm structures and 432 * register/unregister our callback. 433 */ 434 static void db_fm_init(db_ctrl_t *db_p); 435 static void db_fm_fini(db_ctrl_t *db_p); 436 437 int 438 _init(void) 439 { 440 int rc; 441 442 DB_DEBUG0(DB_INIT|DB_DONT_DISPLAY_DIP, NULL, "enter\n"); 443 if (((rc = ddi_soft_state_init(&db_state, 444 sizeof (db_ctrl_t), 1)) == 0) && 445 ((rc = mod_install(&modlinkage)) != 0)) 446 ddi_soft_state_fini(&db_state); 447 DB_DEBUG1(DB_INIT|DB_DONT_DISPLAY_DIP, NULL, "exit rc=%d\n", rc); 448 return (rc); 449 } 450 451 452 int 453 _fini(void) 454 { 455 int rc; 456 457 DB_DEBUG0(DB_FINI|DB_DONT_DISPLAY_DIP, NULL, "enter\n"); 458 if ((rc = mod_remove(&modlinkage)) == 0) 459 ddi_soft_state_fini(&db_state); 460 DB_DEBUG1(DB_FINI|DB_DONT_DISPLAY_DIP, NULL, "exit rc=%d\n", rc); 461 return (rc); 462 } 463 464 int 465 _info(struct modinfo *modinfop) 466 { 467 int rc; 468 rc = mod_info(&modlinkage, modinfop); 469 DB_DEBUG1(DB_INFO|DB_DONT_DISPLAY_DIP, NULL, "exit rc=%d\n", rc); 470 return (rc); 471 } 472 473 /*ARGSUSED*/ 474 static int 475 db_getinfo(dev_info_t *dip, ddi_info_cmd_t infocmd, void *arg, void **result) 476 { 477 db_ctrl_t *dbp; 478 int rc = DDI_FAILURE; 479 minor_t minor = getminor((dev_t)arg); 480 int instance = PCIHP_AP_MINOR_NUM_TO_INSTANCE(minor); 481 482 DB_DEBUG1(DB_GETINFO|DB_DONT_DISPLAY_DIP, dip, "enter:cmd=%d\n", 483 infocmd); 484 485 switch (infocmd) { 486 case DDI_INFO_DEVT2DEVINFO: 487 488 if ((dbp = ddi_get_soft_state(db_state, 489 instance)) != NULL) { 490 *result = dbp->dip; 491 rc = DDI_SUCCESS; 492 } else 493 *result = NULL; 494 break; 495 496 case DDI_INFO_DEVT2INSTANCE: 497 *result = (void *)instance; 498 rc = DDI_SUCCESS; 499 break; 500 501 default: 502 break; 503 } 504 DB_DEBUG2(DB_GETINFO|DB_DONT_DISPLAY_DIP, dip, 505 "exit: result=%x, rc=%d\n", *result, rc); 506 507 return (rc); 508 } 509 510 static int 511 db_attach(dev_info_t *dip, ddi_attach_cmd_t cmd) 512 { 513 int instance = ddi_get_instance(dip); 514 db_ctrl_t *dbp; 515 int rc = DDI_SUCCESS; 516 ddi_device_acc_attr_t db_csr_attr = { /* CSR map attributes */ 517 DDI_DEVICE_ATTR_V0, 518 DDI_STRUCTURE_LE_ACC, 519 DDI_STRICTORDER_ACC 520 }; 521 off_t bar_size; 522 int range_size; 523 char name[32]; 524 525 DB_DEBUG1(DB_ATTACH, dip, "enter: cmd=%d\n", cmd); 526 switch (cmd) { 527 528 case DDI_ATTACH: 529 if (ddi_soft_state_zalloc(db_state, instance) != DDI_SUCCESS) { 530 rc = DDI_FAILURE; 531 break; 532 } 533 534 dbp = (db_ctrl_t *)ddi_get_soft_state(db_state, instance); 535 536 dbp->dip = dip; 537 mutex_init(&dbp->db_mutex, NULL, MUTEX_DRIVER, NULL); 538 dbp->db_soft_state = DB_SOFT_STATE_CLOSED; 539 540 /* 541 * Cannot use pci_config_setup here as we'd need 542 * to get a pointer to the address map to be able 543 * to set the bus private handle during child map 544 * operation. 545 */ 546 if ((rc = ddi_regs_map_setup(dip, DB_PCI_CONF_RNUMBER, 547 (caddr_t *)&dbp->conf_io, DB_PCI_CONF_OFFSET, 548 PCI_CONF_HDR_SIZE, &db_csr_attr, &dbp->conf_handle)) 549 != DDI_SUCCESS) { 550 551 cmn_err(CE_WARN, 552 "%s#%d: cannot map configuration space", 553 ddi_driver_name(dip), ddi_get_instance(dip)); 554 mutex_destroy(&dbp->db_mutex); 555 ddi_soft_state_free(db_state, instance); 556 rc = DDI_FAILURE; 557 break; 558 } 559 560 db_get_perf_parameters(dbp); 561 562 if (ddi_dev_regsize(dip, DB_CSR_MEMBAR_RNUMBER, &bar_size) 563 != DDI_SUCCESS) { 564 cmn_err(CE_WARN, "%s#%d: cannot get memory CSR size", 565 ddi_driver_name(dbp->dip), 566 ddi_get_instance(dbp->dip)); 567 ddi_regs_map_free(&dbp->conf_handle); 568 mutex_destroy(&dbp->db_mutex); 569 ddi_soft_state_free(db_state, instance); 570 rc = DDI_FAILURE; 571 break; 572 } 573 574 /* map memory CSR space */ 575 if (ddi_regs_map_setup(dip, DB_CSR_MEMBAR_RNUMBER, 576 (caddr_t *)&dbp->csr_mem, DB_CSR_MEM_OFFSET, bar_size, 577 &db_csr_attr, &dbp->csr_mem_handle) != DDI_SUCCESS) { 578 579 cmn_err(CE_WARN, "%s#%d: cannot map memory CSR space", 580 ddi_driver_name(dbp->dip), 581 ddi_get_instance(dbp->dip)); 582 ddi_regs_map_free(&dbp->conf_handle); 583 mutex_destroy(&dbp->db_mutex); 584 ddi_soft_state_free(db_state, instance); 585 rc = DDI_FAILURE; 586 break; 587 } 588 589 if (ddi_dev_regsize(dip, DB_CSR_IOBAR_RNUMBER, &bar_size) 590 != DDI_SUCCESS) { 591 cmn_err(CE_WARN, "%s#%d: cannot get IO CSR size", 592 ddi_driver_name(dbp->dip), 593 ddi_get_instance(dbp->dip)); 594 ddi_regs_map_free(&dbp->csr_mem_handle); 595 ddi_regs_map_free(&dbp->conf_handle); 596 mutex_destroy(&dbp->db_mutex); 597 ddi_soft_state_free(db_state, instance); 598 rc = DDI_FAILURE; 599 break; 600 } 601 602 /* 603 * map IO CSR space. We need this map to initiate 604 * indirect configuration transactions as this is a better 605 * option than doing through configuration space map. 606 */ 607 if (ddi_regs_map_setup(dip, DB_CSR_IOBAR_RNUMBER, 608 (caddr_t *)&dbp->csr_io, DB_CSR_IO_OFFSET, bar_size, 609 &db_csr_attr, &dbp->csr_io_handle) != DDI_SUCCESS) { 610 611 cmn_err(CE_WARN, "%s#%d: cannot map IO CSR space", 612 ddi_driver_name(dbp->dip), 613 ddi_get_instance(dbp->dip)); 614 ddi_regs_map_free(&dbp->csr_mem_handle); 615 ddi_regs_map_free(&dbp->conf_handle); 616 mutex_destroy(&dbp->db_mutex); 617 ddi_soft_state_free(db_state, instance); 618 rc = DDI_FAILURE; 619 break; 620 } 621 622 db_orientation(dbp); 623 624 if (dbp->dev_state & DB_SECONDARY_NEXUS) { 625 if (pcihp_init(dip) != DDI_SUCCESS) 626 cmn_err(CE_WARN, 627 "%s#%d: could not register with hotplug", 628 ddi_driver_name(dbp->dip), 629 ddi_get_instance(dbp->dip)); 630 } else { 631 /* 632 * create minor node for devctl interfaces 633 */ 634 if (ddi_create_minor_node(dip, "devctl", S_IFCHR, 635 PCIHP_AP_MINOR_NUM(instance, PCIHP_DEVCTL_MINOR), 636 DDI_NT_NEXUS, 0) != DDI_SUCCESS) { 637 ddi_regs_map_free(&dbp->csr_io_handle); 638 ddi_regs_map_free(&dbp->csr_mem_handle); 639 ddi_regs_map_free(&dbp->conf_handle); 640 mutex_destroy(&dbp->db_mutex); 641 ddi_soft_state_free(db_state, instance); 642 rc = DDI_FAILURE; 643 break; 644 } 645 } 646 647 db_enable_io(dbp); 648 649 range_size = sizeof (dbp->range); 650 if (ddi_getlongprop_buf(DDI_DEV_T_ANY, dip, 651 DDI_PROP_DONTPASS, "bus-range", (caddr_t)&dbp->range, 652 &range_size) != DDI_SUCCESS) { 653 654 cmn_err(CE_WARN, 655 "%s#%d: cannot get bus-range property", 656 ddi_driver_name(dip), ddi_get_instance(dip)); 657 658 if (dbp->dev_state & DB_SECONDARY_NEXUS) 659 (void) pcihp_uninit(dip); 660 else 661 ddi_remove_minor_node(dip, "devctl"); 662 663 ddi_regs_map_free(&dbp->csr_mem_handle); 664 ddi_regs_map_free(&dbp->csr_io_handle); 665 ddi_regs_map_free(&dbp->conf_handle); 666 mutex_destroy(&dbp->db_mutex); 667 ddi_soft_state_free(db_state, instance); 668 rc = DDI_FAILURE; 669 break; 670 } 671 672 (void) sprintf(name, "%d", instance); 673 674 if (ddi_create_minor_node(dip, name, S_IFCHR, 675 PCIHP_AP_MINOR_NUM(instance, PCIHP_DEBUG_MINOR), 676 NULL, NULL) == DDI_FAILURE) { 677 cmn_err(CE_NOTE, "%s#%d: node creation failure", 678 ddi_driver_name(dbp->dip), instance); 679 } 680 681 mutex_init(&dbp->db_busown, NULL, MUTEX_DRIVER, NULL); 682 683 db_fm_init(dbp); 684 ddi_report_dev(dip); 685 dbp->dev_state |= DB_ATTACHED; 686 687 break; 688 689 case DDI_RESUME: 690 691 /* 692 * Get the soft state structure for the bridge. 693 */ 694 dbp = (db_ctrl_t *)ddi_get_soft_state(db_state, instance); 695 db_enable_io(dbp); 696 (void) db_restore_config_regs(dbp); 697 dbp->dev_state &= ~DB_SUSPENDED; 698 break; 699 700 default: 701 rc = DDI_FAILURE; /* not supported yet */ 702 break; 703 } 704 705 DB_DEBUG1(DB_ATTACH, dip, "exit: rc=%d\n", rc); 706 return (rc); 707 } 708 709 static int 710 db_detach(dev_info_t *dip, ddi_detach_cmd_t cmd) 711 { 712 int instance = ddi_get_instance(dip); 713 db_ctrl_t *dbp; 714 int rc = DDI_SUCCESS; 715 char name[32]; 716 717 dbp = (db_ctrl_t *)ddi_get_soft_state(db_state, instance); 718 719 DB_DEBUG1(DB_DETACH, dip, "enter: cmd=%d\n", cmd); 720 721 switch (cmd) { 722 723 case DDI_DETACH : 724 db_fm_fini(dbp); 725 if (dbp->dev_state & DB_SECONDARY_NEXUS) 726 if (pcihp_uninit(dip) == DDI_FAILURE) 727 return (DDI_FAILURE); 728 else 729 ddi_remove_minor_node(dip, "devctl"); 730 731 mutex_destroy(&dbp->db_busown); 732 ddi_regs_map_free(&dbp->csr_mem_handle); 733 ddi_regs_map_free(&dbp->csr_io_handle); 734 735 ddi_regs_map_free(&dbp->conf_handle); 736 dbp->dev_state &= ~DB_ATTACHED; 737 (void) sprintf(name, "%d", instance); 738 ddi_remove_minor_node(dip, name); 739 mutex_destroy(&dbp->db_mutex); 740 ddi_soft_state_free(db_state, instance); 741 break; 742 743 case DDI_SUSPEND : 744 if (db_save_config_regs(dbp) != DDI_SUCCESS) { 745 cmn_err(CE_WARN, 746 "%s#%d: Ignoring Child state Suspend Error", 747 ddi_driver_name(dbp->dip), 748 ddi_get_instance(dbp->dip)); 749 } 750 dbp->dev_state |= DB_SUSPENDED; 751 break; 752 753 default : 754 rc = DDI_FAILURE; 755 break; 756 } 757 758 DB_DEBUG1(DB_DETACH, dip, "exit: rc=%d\n", rc); 759 return (rc); 760 } 761 762 static void 763 db_get_perf_parameters(db_ctrl_t *dbp) 764 { 765 dbp->p_latency_timer = (int8_t)ddi_prop_get_int(DDI_DEV_T_ANY, 766 dbp->dip, 0, "p-latency-timer", p_latency_timer); 767 dbp->s_latency_timer = (int8_t)ddi_prop_get_int(DDI_DEV_T_ANY, 768 dbp->dip, 0, "s-latency-timer", s_latency_timer); 769 dbp->p_cache_line_size = (int8_t)ddi_prop_get_int(DDI_DEV_T_ANY, 770 dbp->dip, 0, "p-cache-line-size", p_cache_line_size); 771 dbp->s_cache_line_size = (int8_t)ddi_prop_get_int(DDI_DEV_T_ANY, 772 dbp->dip, 0, "s-cache-line-size", s_cache_line_size); 773 dbp->p_pwrite_threshold = (int8_t)ddi_prop_get_int(DDI_DEV_T_ANY, 774 dbp->dip, 0, "p-pwrite-threshold", p_pwrite_threshold); 775 dbp->s_pwrite_threshold = (int8_t)ddi_prop_get_int(DDI_DEV_T_ANY, 776 dbp->dip, 0, "s-pwrite-threshold", s_pwrite_threshold); 777 dbp->p_dread_threshold = (int8_t)ddi_prop_get_int(DDI_DEV_T_ANY, 778 dbp->dip, 0, "p-dread-threshold", p_dread_threshold); 779 dbp->s_dread_threshold = (int8_t)ddi_prop_get_int(DDI_DEV_T_ANY, 780 dbp->dip, 0, "s-dread-threshold", s_dread_threshold); 781 dbp->delayed_trans_order = (int8_t)ddi_prop_get_int(DDI_DEV_T_ANY, 782 dbp->dip, 0, "delayed-trans-order", delayed_trans_order); 783 } 784 785 static void 786 db_set_perf_parameters(db_ctrl_t *dbp) 787 { 788 uint_t poffset = 0, soffset = 0; 789 790 if (dbp->dev_state & DB_SECONDARY_NEXUS) 791 poffset = DB_SCONF_PRI_HDR_OFF; 792 else 793 soffset = DB_PCONF_SEC_HDR_OFF; 794 795 if ((dbp->p_latency_timer != (int8_t)DEF_INVALID_REG_VAL) && 796 (dbp->p_latency_timer != -1)) 797 ddi_put8(dbp->conf_handle, 798 (uint8_t *)dbp->conf_io+poffset+PCI_CONF_LATENCY_TIMER, 799 dbp->p_latency_timer); 800 if ((dbp->s_latency_timer != (int8_t)DEF_INVALID_REG_VAL) && 801 (dbp->s_latency_timer != -1)) 802 ddi_put8(dbp->conf_handle, 803 (uint8_t *)dbp->conf_io+soffset+PCI_CONF_LATENCY_TIMER, 804 dbp->s_latency_timer); 805 if ((dbp->p_cache_line_size != (int8_t)DEF_INVALID_REG_VAL) && 806 (dbp->p_cache_line_size != -1)) 807 ddi_put8(dbp->conf_handle, 808 (uint8_t *)dbp->conf_io+poffset+PCI_CONF_CACHE_LINESZ, 809 dbp->p_cache_line_size); 810 if ((dbp->s_cache_line_size != (int8_t)DEF_INVALID_REG_VAL) && 811 (dbp->s_cache_line_size != -1)) 812 ddi_put8(dbp->conf_handle, 813 (uint8_t *)dbp->conf_io+soffset+PCI_CONF_CACHE_LINESZ, 814 dbp->s_cache_line_size); 815 if ((dbp->p_pwrite_threshold != (int8_t)DEF_INVALID_REG_VAL) && 816 (dbp->p_pwrite_threshold != -1)) 817 ddi_put16(dbp->conf_handle, (uint16_t *) 818 ((uchar_t *)dbp->conf_io+DB_CONF_CHIP_CTRL1), 819 (ddi_get16(dbp->conf_handle, (uint16_t *) 820 ((uchar_t *)dbp->conf_io+DB_CONF_CHIP_CTRL1)) & 821 ~P_PW_THRESHOLD) | 822 (dbp->p_pwrite_threshold?P_PW_THRESHOLD:0)); 823 if ((dbp->s_pwrite_threshold != (int8_t)DEF_INVALID_REG_VAL) && 824 (dbp->s_pwrite_threshold != -1)) 825 ddi_put16(dbp->conf_handle, (uint16_t *) 826 ((uchar_t *)dbp->conf_io+DB_CONF_CHIP_CTRL1), 827 (ddi_get16(dbp->conf_handle, (uint16_t *) 828 ((uchar_t *)dbp->conf_io+DB_CONF_CHIP_CTRL1)) & 829 ~S_PW_THRESHOLD) | 830 (dbp->s_pwrite_threshold?S_PW_THRESHOLD:0)); 831 /* primary delayed read threshold. 0x01 is reserved ?. */ 832 if ((dbp->p_dread_threshold != (int8_t)DEF_INVALID_REG_VAL) && 833 (dbp->p_dread_threshold != -1)) 834 ddi_put16(dbp->conf_handle, (uint16_t *) 835 ((uchar_t *)dbp->conf_io+DB_CONF_CHIP_CTRL1), 836 ((ddi_get16(dbp->conf_handle, (uint16_t *) 837 ((uchar_t *)dbp->conf_io+DB_CONF_CHIP_CTRL1)) & 838 ~P_DREAD_THRESHOLD_MASK) | 839 ((dbp->p_dread_threshold & 840 DREAD_THRESHOLD_VALBITS)<<2))); 841 /* secondary delayed read threshold. 0x01 is reserved ?. */ 842 if ((dbp->s_dread_threshold != (int8_t)DEF_INVALID_REG_VAL) && 843 (dbp->s_dread_threshold != -1)) 844 ddi_put16(dbp->conf_handle, (uint16_t *) 845 ((uchar_t *)dbp->conf_io+DB_CONF_CHIP_CTRL1), 846 ((ddi_get16(dbp->conf_handle, (uint16_t *) 847 ((uchar_t *)dbp->conf_io+DB_CONF_CHIP_CTRL1)) & 848 ~S_DREAD_THRESHOLD_MASK) | 849 ((dbp->s_dread_threshold & 850 DREAD_THRESHOLD_VALBITS)<<4))); 851 if ((dbp->delayed_trans_order != (int8_t)DEF_INVALID_REG_VAL) && 852 (dbp->delayed_trans_order != -1)) 853 ddi_put16(dbp->conf_handle, (uint16_t *) 854 ((uchar_t *)dbp->conf_io+DB_CONF_CHIP_CTRL0), 855 (ddi_get16(dbp->conf_handle, (uint16_t *) 856 ((uchar_t *)dbp->conf_io+DB_CONF_CHIP_CTRL0)) & 857 ~DELAYED_TRANS_ORDER) | 858 (dbp->delayed_trans_order?DELAYED_TRANS_ORDER:0)); 859 } 860 861 static void 862 db_orientation(db_ctrl_t *dbp) 863 { 864 dev_info_t *dip = dbp->dip; 865 uint8_t pif; 866 uint32_t mem1; 867 uint32_t newval; 868 869 /* 870 * determine orientation of drawbridge and enable 871 * Upstream or Downstream path. 872 */ 873 874 /* 875 * if PIF is set correctly, use it to determine orientation 876 */ 877 pif = ddi_get8(dbp->conf_handle, (uchar_t *)dbp->conf_io + 878 PCI_CONF_PROGCLASS); 879 if (pif & 0xff) { 880 if (pif & DB_PIF_SECONDARY_TO_HOST) { 881 dbp->dev_state = DB_SECONDARY_NEXUS; 882 DB_DEBUG0(DB_ATTACH, dip, 883 "db_orientation: pif secondary\n"); 884 return; 885 } 886 if (pif & DB_PIF_PRIMARY_TO_HOST) { 887 dbp->dev_state = DB_PRIMARY_NEXUS; 888 DB_DEBUG0(DB_ATTACH, dip, 889 "db_orientation: pif primary\n"); 890 return; 891 } 892 /* otherwise, fall through */ 893 } 894 895 /* 896 * otherwise, test the chip directly by trying to write 897 * downstream mem1 setup register, only writeable from 898 * secondary. 899 */ 900 mem1 = ddi_get32(dbp->conf_handle, 901 (uint32_t *)((uchar_t *)dbp->conf_io + 902 DB_CONF_DS_IO_MEM1_SETUP)); 903 904 ddi_put32(dbp->conf_handle, 905 (uint32_t *)((uchar_t *)(dbp->conf_io + 906 DB_CONF_DS_IO_MEM1_SETUP)), ~mem1); 907 908 newval = ddi_get32(dbp->conf_handle, 909 (uint32_t *)((uchar_t *)dbp->conf_io + 910 DB_CONF_DS_IO_MEM1_SETUP)); 911 912 if (newval == mem1) 913 /* we couldn't write it, orientation is primary */ 914 dbp->dev_state = DB_PRIMARY_NEXUS; 915 else { 916 /* 917 * we could write it, therefore orientation secondary. 918 * restore mem1 value. 919 */ 920 dbp->dev_state = DB_SECONDARY_NEXUS; 921 ddi_put32(dbp->conf_handle, 922 (uint32_t *)((uchar_t *)(dbp->conf_io + 923 DB_CONF_DS_IO_MEM1_SETUP)), mem1); 924 } 925 926 927 if (dbp->dev_state & DB_PRIMARY_NEXUS) { 928 DB_DEBUG0(DB_ATTACH, dip, "db_orientation: chip primary\n"); 929 } else { 930 DB_DEBUG0(DB_ATTACH, dip, "db_orientation: chip secondary\n"); 931 } 932 } 933 934 static void 935 db_enable_io(db_ctrl_t *dbp) 936 { 937 dev_info_t *dip = dbp->dip; 938 pci_regspec_t *reg; 939 int rcount, length, i; 940 uint32_t offset; 941 uint32_t p_offset, s_offset; 942 uint16_t regval; 943 uint16_t enable; 944 945 /* 946 * Step 0: 947 * setup the primary and secondary offset and enable 948 * values based on the orientation of 21554. 949 */ 950 if (dbp->dev_state & DB_PRIMARY_NEXUS) { 951 DB_DEBUG0(DB_ATTACH, dip, "db_enable_io: primary\n"); 952 p_offset = 0; 953 s_offset = DB_SCONF_HDR_OFF; 954 enable = DS_ENABLE; 955 } else { 956 DB_DEBUG0(DB_ATTACH, dip, "db_enable_io: secondary\n"); 957 p_offset = DB_SCONF_HDR_OFF; 958 s_offset = 0; 959 enable = US_ENABLE; 960 } 961 962 db_set_perf_parameters(dbp); 963 db_set_dvma_range(dbp); 964 965 /* 966 * Step 1: 967 * setup latency timer and cache line size parameters 968 * which are used for child initialization. 969 */ 970 dbp->latency_timer = ddi_get8(dbp->conf_handle, (uint8_t *) 971 ((caddr_t)dbp->conf_io+PCI_CONF_LATENCY_TIMER)); 972 973 dbp->cache_line_size = ddi_get8(dbp->conf_handle, (uint8_t *) 974 ((caddr_t)dbp->conf_io+PCI_CONF_CACHE_LINESZ)); 975 976 DB_DEBUG2(DB_ATTACH, dip, 977 "db_enable_io: latency %d, cache line size %d\n", 978 dbp->latency_timer, dbp->cache_line_size); 979 980 /* 981 * Step 2: program command reg on both primary and secondary 982 * interfaces. 983 */ 984 ddi_put16(dbp->conf_handle, (uint16_t *)((caddr_t)dbp->conf_io + 985 (off_t)(p_offset + PCI_CONF_COMM)), db_command_default); 986 987 ddi_put16(dbp->conf_handle, (uint16_t *)((caddr_t)dbp->conf_io + 988 (off_t)(s_offset + PCI_CONF_COMM)), db_command_default); 989 990 /* 991 * Step 3: 992 * set up translated base registers, using the primary/ 993 * secondary interface pci configuration Base Address 994 * Registers (BAR's). 995 */ 996 997 /* mem0 translated base is setup for primary orientation only. */ 998 if (dbp->dev_state & DB_PRIMARY_NEXUS) { 999 /* 1000 * And only if the 21554 device node property indicates 1001 * the size of base0 register to be larger than csr map 1002 * space, DB_CSR_SIZE=4K. 1003 * 1004 * Note : Setting up 1:1 translations only (for now:), i.e. 1005 * no look up table. 1006 */ 1007 if (ddi_getlongprop(DDI_DEV_T_ANY, dip, 1008 DDI_PROP_DONTPASS, "reg", (caddr_t)®, 1009 &length) != DDI_PROP_SUCCESS) { 1010 DB_DEBUG0(DB_ATTACH, dip, 1011 "Failed to read reg property\n"); 1012 return; 1013 } 1014 1015 /* Find device node's base0 reg property and check its size */ 1016 rcount = length / sizeof (pci_regspec_t); 1017 for (i = 0; i < rcount; i++) { 1018 offset = PCI_REG_REG_G(reg[i].pci_phys_hi); 1019 if ((offset == PCI_CONF_BASE0) && 1020 (reg[i].pci_size_low > DB_CSR_SIZE)) 1021 break; 1022 } 1023 1024 /* 1025 * set up mem0 translated base, if base0 register was 1026 * found and its size was larger than csr map space. 1027 */ 1028 if (i != rcount) { 1029 DB_DEBUG0(DB_ATTACH, dip, 1030 "db_enable_io: setting up MEM0_TR_BASE\n"); 1031 DB_DEBUG1(DB_ATTACH, dip, "BASE0 register = %x\n", 1032 pci_config_get32(dbp->conf_handle, 1033 (off_t)(p_offset + PCI_CONF_BASE0))); 1034 1035 pci_config_put32(dbp->conf_handle, 1036 (off_t)DB_CONF_DS_MEM0_TR_BASE, 1037 pci_config_get32(dbp->conf_handle, 1038 (off_t)(p_offset + PCI_CONF_BASE0))); 1039 1040 DB_DEBUG1(DB_ATTACH, dip, 1041 "db_enable_io: MEM0_TR_BASE set value = %x\n", 1042 pci_config_get32(dbp->conf_handle, 1043 (off_t)DB_CONF_DS_MEM0_TR_BASE)); 1044 } 1045 kmem_free(reg, length); 1046 } 1047 1048 pci_config_put32(dbp->conf_handle, (off_t)DB_CONF_DS_IO_MEM1_TR_BASE, 1049 ((pci_config_get32(dbp->conf_handle, 1050 (off_t)(p_offset + PCI_CONF_BASE2))) & ~DB_IO_BIT)); 1051 1052 pci_config_put32(dbp->conf_handle, (off_t)DB_CONF_DS_MEM2_TR_BASE, 1053 ((pci_config_get32(dbp->conf_handle, 1054 (off_t)(p_offset + PCI_CONF_BASE3))) & ~DB_IO_BIT)); 1055 1056 pci_config_put32(dbp->conf_handle, (off_t)DB_CONF_DS_MEM3_TR_BASE, 1057 ((pci_config_get32(dbp->conf_handle, 1058 (off_t)(p_offset + PCI_CONF_BASE4))) & ~DB_IO_BIT)); 1059 1060 pci_config_put32(dbp->conf_handle, (off_t)DB_CONF_US_IO_MEM0_TR_BASE, 1061 ((pci_config_get32(dbp->conf_handle, 1062 (off_t)(s_offset + PCI_CONF_BASE2))) & ~DB_IO_BIT)); 1063 1064 pci_config_put32(dbp->conf_handle, (off_t)DB_CONF_US_MEM1_TR_BASE, 1065 ((pci_config_get32(dbp->conf_handle, 1066 (off_t)(s_offset + PCI_CONF_BASE3))) & ~DB_IO_BIT)); 1067 1068 /* 1069 * Step 4: enable downstream (for primary orientation) or upstream 1070 * (for secondary orientation) bits in Configuration Control 1071 * and Status register, if not already enabled. 1072 */ 1073 regval = pci_config_get16(dbp->conf_handle, (off_t)DB_CONF_CONF_CSR); 1074 1075 DB_DEBUG1(DB_ATTACH, dip, "db_enable_io: CSR value before: %x\n", 1076 regval); 1077 1078 if (!(regval & enable)) { 1079 /* enable down/upstream configuration transactions */ 1080 regval |= enable; 1081 pci_config_put16(dbp->conf_handle, (off_t)DB_CONF_CONF_CSR, 1082 regval); 1083 regval = pci_config_get16(dbp->conf_handle, 1084 (off_t)DB_CONF_CONF_CSR); 1085 } 1086 DB_DEBUG1(DB_ATTACH, dip, "db_enable_io: CSR value after: %x\n", 1087 regval); 1088 1089 /* 1090 * Step 5: enable downstream/upstream I/O (through CSR space) 1091 */ 1092 regval = ddi_get16(dbp->csr_mem_handle, 1093 (uint16_t *)((uchar_t *)dbp->csr_mem + DB_CSR_IO_CSR)); 1094 1095 DB_DEBUG1(DB_ATTACH, dip, "db_enable_io: IO_CSR value before: %x\n", 1096 regval); 1097 if (!(regval & enable)) { 1098 regval |= enable; 1099 ddi_put16(dbp->csr_mem_handle, 1100 (uint16_t *)((uchar_t *)dbp->csr_mem + 1101 DB_CSR_IO_CSR), regval); 1102 1103 regval = ddi_get16(dbp->csr_mem_handle, 1104 (uint16_t *)((uchar_t *)dbp->csr_mem + DB_CSR_IO_CSR)); 1105 } 1106 DB_DEBUG1(DB_ATTACH, dip, "db_enable_io: IO_CSR value after: %x\n", 1107 regval); 1108 1109 /* 1110 * Step 6: if 21554 orientation is primary to host, 1111 * forward SERR# to host. 1112 */ 1113 if (dbp->dev_state & DB_PRIMARY_NEXUS) { 1114 dbp->serr_fwd_enable = ddi_prop_get_int(DDI_DEV_T_ANY, 1115 dbp->dip, 0, "serr-fwd-enable", db_serr_fwd_enable); 1116 1117 regval = ddi_get16(dbp->conf_handle, 1118 (uint16_t *)((uchar_t *)dbp->conf_io + 1119 DB_CONF_CHIP_CTRL0)); 1120 1121 DB_DEBUG1(DB_ATTACH, dip, 1122 "db_enable_io: CHIP_CTRL0 value before: %x\n", regval); 1123 1124 ddi_put16(dbp->conf_handle, 1125 (uint16_t *)((uchar_t *)dbp->conf_io + 1126 DB_CONF_CHIP_CTRL0), 1127 (regval & ~SERR_FWD) | 1128 (dbp->serr_fwd_enable?SERR_FWD:0)); 1129 1130 regval = ddi_get16(dbp->conf_handle, 1131 (uint16_t *)((uchar_t *)dbp->conf_io + 1132 DB_CONF_CHIP_CTRL0)); 1133 1134 DB_DEBUG1(DB_ATTACH, dip, 1135 "db_enable_io: CHIP_CTRL0 value after: %x\n", regval); 1136 } 1137 1138 /* 1139 * Step 7: if orientation is secondary, make sure primary lockout 1140 * disable is reset. 1141 */ 1142 1143 if (dbp->dev_state & DB_SECONDARY_NEXUS) { 1144 regval = pci_config_get16(dbp->conf_handle, 1145 (off_t)DB_CONF_CHIP_CTRL0); 1146 DB_DEBUG1(DB_ATTACH, dip, 1147 "db_enable_io: chip ctrl (0x%x) before\n", regval); 1148 if (regval & PLOCKOUT) 1149 pci_config_put16(dbp->conf_handle, 1150 (off_t)DB_CONF_CHIP_CTRL0, 1151 (regval & ~PLOCKOUT)); 1152 regval = pci_config_get16(dbp->conf_handle, 1153 (off_t)DB_CONF_CHIP_CTRL0); 1154 DB_DEBUG1(DB_ATTACH, dip, 1155 "db_enable_io: chip ctrl (0x%x) after\n", regval); 1156 } 1157 } 1158 1159 /* 1160 * Set DVMA Address Range. 1161 * This code is common to both orientations of the nexus driver. 1162 */ 1163 static void 1164 db_set_dvma_range(db_ctrl_t *dbp) 1165 { 1166 uint32_t dvma_start = 0; 1167 uint32_t dvma_len = 0; 1168 uint64_t db_allocd = 0; 1169 uint32_t *dvma_prop; 1170 uint32_t dvma_size[2]; /* dvma size may span over 2 BARs */ 1171 uint32_t dvma_bar[2]; /* dvma range may span over 2 BARs */ 1172 int dvma_prop_len; 1173 uint64_t new_dvma_start, new_dvma_len, new_dvma_end; 1174 1175 /* 1176 * Need to traverse up the tree looking for a 1177 * "virtual-dma" property that specifies the 1178 * HPB DVMA range. 1179 */ 1180 if (ddi_getlongprop(DDI_DEV_T_ANY, ddi_get_parent(dbp->dip), 0, 1181 "virtual-dma", (caddr_t)&dvma_prop, &dvma_prop_len) 1182 == DDI_SUCCESS) { 1183 dvma_start = dvma_prop[0]; 1184 dvma_len = dvma_prop[1]; 1185 kmem_free((caddr_t)dvma_prop, dvma_prop_len); 1186 } else { 1187 /* 1188 * For initial implementation, lets avoid a warning since this 1189 * change has not been implemented in the host-pci nexus 1190 * driver. 1191 */ 1192 cmn_err(CE_WARN, 1193 "%s#%d: Could not get \"virtual-dma\" property", 1194 ddi_driver_name(dbp->dip), 1195 ddi_get_instance(dbp->dip)); 1196 dvma_start = db_dvma_start; 1197 dvma_len = db_dvma_len; 1198 } 1199 1200 DB_DEBUG2(DB_DVMA, dbp->dip, 1201 "DVMA Range is %lx,%lx\n", dvma_start, dvma_len); 1202 1203 dvma_size[0] = dvma_size[1] = 0; 1204 /* Validate DVMA size programming and system requirements. */ 1205 if (dbp->dev_state & DB_SECONDARY_NEXUS) { 1206 dvma_size[0] = pci_config_get32(dbp->conf_handle, 1207 DB_CONF_DS_IO_MEM1_SETUP); 1208 if (!(dvma_size[0] & 1)) /* make sure it is not a IO BAR */ 1209 dvma_size[0] = ((~dvma_size[0]) + 1) & 0xfffff000; 1210 else 1211 dvma_size[0] = 0; 1212 dvma_size[1] = db_dvma_len; 1213 } else { 1214 dvma_size[0] = pci_config_get32(dbp->conf_handle, 1215 DB_CONF_US_IO_MEM0_SETUP); 1216 if (!(dvma_size[0] & 1)) /* make sure it is not a IO BAR */ 1217 dvma_size[0] = ((~dvma_size[0]) + 1) & 0xfffff000; 1218 else 1219 dvma_size[0] = 0; 1220 dvma_size[1] = ((~(pci_config_get32(dbp->conf_handle, 1221 DB_CONF_US_MEM1_SETUP))) + 1) & 0xfffff000; 1222 } 1223 DB_DEBUG2(DB_DVMA, dbp->dip, "DVMA size register pair %lx, %lx\n", 1224 dvma_size[0], dvma_size[1]); 1225 1226 #ifdef DEBUG 1227 if ((dvma_size[0] + dvma_size[1]) < dvma_len) 1228 cmn_err(CE_WARN, "%s#%d: DVMA window (%u) does not coincide" 1229 " with system requirements", 1230 ddi_driver_name(dbp->dip), ddi_get_instance(dbp->dip), 1231 (dvma_size[0] + dvma_size[1])); 1232 #endif 1233 dvma_bar[0] = dvma_bar[1] = 0xFFFFFFFF; 1234 db_allocd = 0; 1235 new_dvma_start = dvma_start; 1236 new_dvma_len = dvma_len; 1237 1238 /* now, program the correct DVMA range over the 2 BARs. Max 4GB */ 1239 if (dvma_size[0]) { 1240 dvma_bar[0] = (uint32_t)(dvma_start & (~(dvma_size[0] - 1))); 1241 new_dvma_end = (uint64_t)((uint64_t)dvma_bar[0] + 1242 (uint64_t)dvma_size[0]); 1243 if (new_dvma_end > (new_dvma_start + new_dvma_len)) 1244 new_dvma_end = new_dvma_start + new_dvma_len; 1245 db_allocd += (new_dvma_end - new_dvma_start); 1246 new_dvma_start = new_dvma_end; 1247 new_dvma_len = dvma_len - db_allocd; 1248 } 1249 /* 1250 * It does not serve any purpose to set the other DVMA register 1251 * when we have already met the memory requirements so leave it 1252 * disabled. 1253 */ 1254 if ((db_allocd != dvma_len) && dvma_size[1]) { 1255 dvma_bar[1] = (uint32_t)((dvma_start + db_allocd) & 1256 (~(dvma_size[1] - 1))); 1257 new_dvma_end = (uint64_t)((uint64_t)dvma_bar[1] + 1258 (uint64_t)dvma_size[1]); 1259 if (new_dvma_end > (new_dvma_start + new_dvma_len)) 1260 new_dvma_end = new_dvma_start + new_dvma_len; 1261 db_allocd += (new_dvma_end - new_dvma_start); 1262 } 1263 1264 /* In case of secondary orientation, DVMA BAR0 is 0. */ 1265 if (dbp->dev_state & DB_SECONDARY_NEXUS) 1266 dvma_bar[0] = 0; 1267 1268 if (db_allocd != dvma_len) { 1269 cmn_err(CE_WARN, "%s#%d: dvma range error!", 1270 ddi_driver_name(dbp->dip), ddi_get_instance(dbp->dip)); 1271 } 1272 1273 DB_DEBUG2(DB_DVMA, dbp->dip, "DVMA BARs set as %x, %x\n", 1274 dvma_bar[0], dvma_bar[1]); 1275 1276 /* configure the setup register and DVMA BARs. */ 1277 if (dbp->dev_state & DB_SECONDARY_NEXUS) { 1278 if (dvma_bar[0] != 0xFFFFFFFF) { 1279 #ifdef DB_SEC_SETUP_WRITE 1280 /* 1281 * No need to program the setup register 1282 * as the PROM would have done it. 1283 */ 1284 pci_config_put32(dbp->conf_handle, 1285 DB_CONF_DS_MEM1_SETUP, 1286 (uint32_t)(((~(dvma_size[0] - 1)) | 1287 (pci_config_get32(dbp->conf_handle, 1288 DB_CONF_DS_MEM1_SETUP) & 0xF)) | 0x80000000)); 1289 #endif 1290 /* 1291 * when translations are to be provided, this will 1292 * change. 1293 */ 1294 pci_config_put32(dbp->conf_handle, 1295 DB_CONF_DS_IO_MEM1_TR_BASE, 1296 (uint32_t)dvma_bar[0]); 1297 pci_config_put32(dbp->conf_handle, 1298 DB_SCONF_DS_IO_MEM1, dvma_bar[0]); 1299 } 1300 if (dvma_bar[1] != 0xFFFFFFFF) { 1301 #ifdef DB_SEC_SETUP_WRITE 1302 /* 1303 * No need to program the setup register 1304 * as the PROM would have done it. 1305 */ 1306 pci_config_put32(dbp->conf_handle, 1307 DB_CONF_DS_MEM2_SETUP, 1308 (uint32_t)(((~(dvma_size[1] - 1)) | 1309 (pci_config_get32(dbp->conf_handle, 1310 DB_CONF_DS_MEM2_SETUP) & 0xF)) | 0x80000000)); 1311 #endif 1312 /* 1313 * when translations are to be provided, this will 1314 * change. 1315 */ 1316 pci_config_put32(dbp->conf_handle, 1317 DB_CONF_DS_MEM2_TR_BASE, (uint32_t)dvma_bar[1]); 1318 pci_config_put32(dbp->conf_handle, 1319 DB_SCONF_DS_MEM2, dvma_bar[1]); 1320 } 1321 1322 } else { 1323 if (dvma_bar[0] != 0xFFFFFFFF) { 1324 #ifdef DB_CONF_P2S_WRITE_ENABLED /* primary to secondary write enabled */ 1325 /* 1326 * We have a problem with this setup, because the 1327 * US_MEM1 setup register cannot be written from the 1328 * primary interface...!!! Hence in this configuration, 1329 * we cannot dynamically program the DVMA range! 1330 */ 1331 pci_config_put32(dbp->conf_handle, 1332 DB_CONF_US_IO_MEM0_SETUP, 1333 (uint32_t)(((~(dvma_size[0] - 1)) | 1334 (pci_config_get32(dbp->conf_handle, 1335 DB_CONF_US_IO_MEM0_SETUP) & 0xF)) | 1336 0x80000000)); 1337 #endif 1338 /* 1339 * when translations are to be provided, this will 1340 * change. 1341 */ 1342 pci_config_put32(dbp->conf_handle, 1343 DB_CONF_US_IO_MEM0_TR_BASE, 1344 (uint32_t)dvma_bar[0]); 1345 pci_config_put32(dbp->conf_handle, 1346 DB_PCONF_US_IO_MEM0, dvma_bar[0]); 1347 } 1348 if (dvma_bar[1] != 0xFFFFFFFF) { 1349 #ifdef DB_CONF_P2S_WRITE_ENABLED /* primary to secondary write enabled */ 1350 /* 1351 * We have a problem with this setup, because the 1352 * US_MEM1 setup register cannot be written from the 1353 * primary interface...!!! Hence in this configuration, 1354 * we cannot dynamically program the DVMA range! 1355 */ 1356 pci_config_put32(dbp->conf_handle, 1357 DB_CONF_US_MEM1_SETUP, 1358 (uint32_t)(((~(dvma_size[1] - 1)) | 1359 (pci_config_get32(dbp->conf_handle, 1360 DB_CONF_US_MEM1_SETUP) & 0xF)) | 0x80000000)); 1361 #endif 1362 /* 1363 * when translations are to be provided, this will 1364 * change. 1365 */ 1366 pci_config_put32(dbp->conf_handle, 1367 DB_CONF_US_MEM1_TR_BASE, (uint32_t)dvma_bar[1]); 1368 pci_config_put32(dbp->conf_handle, 1369 DB_PCONF_US_MEM1, dvma_bar[1]); 1370 } 1371 } 1372 } 1373 1374 /*ARGSUSED*/ 1375 static int 1376 db_open(dev_t *dev_p, int flag, int otyp, cred_t *cred_p) 1377 { 1378 minor_t minor = getminor(*dev_p); 1379 int instance = PCIHP_AP_MINOR_NUM_TO_INSTANCE(minor); 1380 db_ctrl_t *dbp = (db_ctrl_t *)ddi_get_soft_state(db_state, instance); 1381 1382 if (dbp == (db_ctrl_t *)NULL) 1383 return (ENXIO); 1384 1385 /* 1386 * check for debug node 1387 */ 1388 if ((minor & 0xff) == 0xfe) 1389 return (0); 1390 1391 if (dbp->dev_state & DB_SECONDARY_NEXUS) 1392 return ((pcihp_get_cb_ops())->cb_open(dev_p, flag, 1393 otyp, cred_p)); 1394 /* 1395 * Handle the open by tracking the device state. 1396 */ 1397 mutex_enter(&dbp->db_mutex); 1398 if (flag & FEXCL) { 1399 if (dbp->db_soft_state != DB_SOFT_STATE_CLOSED) { 1400 mutex_exit(&dbp->db_mutex); 1401 return (EBUSY); 1402 } 1403 dbp->db_soft_state = DB_SOFT_STATE_OPEN_EXCL; 1404 } else { 1405 if (dbp->db_soft_state == DB_SOFT_STATE_OPEN_EXCL) { 1406 mutex_exit(&dbp->db_mutex); 1407 return (EBUSY); 1408 } 1409 dbp->db_soft_state = DB_SOFT_STATE_OPEN; 1410 } 1411 mutex_exit(&dbp->db_mutex); 1412 return (0); 1413 } 1414 1415 /*ARGSUSED*/ 1416 static int 1417 db_close(dev_t dev, int flag, int otyp, cred_t *cred_p) 1418 { 1419 minor_t minor = getminor(dev); 1420 int instance = PCIHP_AP_MINOR_NUM_TO_INSTANCE(minor); 1421 db_ctrl_t *dbp = (db_ctrl_t *)ddi_get_soft_state(db_state, instance); 1422 1423 if (dbp == (db_ctrl_t *)NULL) 1424 return (ENXIO); 1425 1426 /* 1427 * check for debug node 1428 */ 1429 if ((minor & 0xff) == 0xfe) 1430 return (0); 1431 1432 if (dbp->dev_state & DB_SECONDARY_NEXUS) 1433 return ((pcihp_get_cb_ops())->cb_close(dev, flag, 1434 otyp, cred_p)); 1435 mutex_enter(&dbp->db_mutex); 1436 dbp->db_soft_state = DB_SOFT_STATE_CLOSED; 1437 mutex_exit(&dbp->db_mutex); 1438 return (0); 1439 } 1440 1441 /*ARGSUSED*/ 1442 static int 1443 db_ioctl(dev_t dev, int cmd, intptr_t arg, int mode, cred_t *cred_p, 1444 int *rval_p) 1445 { 1446 int rc = DDI_SUCCESS; 1447 #ifdef DB_DEBUG 1448 ddi_acc_handle_t config_handle; 1449 db_pci_data_t pci_data; 1450 dev_info_t *child_dip; 1451 #endif 1452 dev_info_t *self; 1453 minor_t minor = getminor(dev); 1454 int instance = PCIHP_AP_MINOR_NUM_TO_INSTANCE(minor); 1455 struct devctl_iocdata *dcp; 1456 uint_t bus_state; 1457 db_ctrl_t *dbp = (db_ctrl_t *)ddi_get_soft_state(db_state, instance); 1458 1459 #ifdef DB_DEBUG 1460 /* 1461 * try this first whether were SECONDARY_NEXUS or not 1462 */ 1463 if (cmd == DB_PCI_READ_CONF_HEADER) { 1464 if (ddi_copyin((caddr_t)arg, (caddr_t)&pci_data, 1465 sizeof (db_pci_data_t), mode)) { 1466 rc = EFAULT; 1467 return (rc); 1468 } 1469 1470 if (strcmp(pci_data.name, "") == 0) { 1471 child_dip = dbp->dip; 1472 (void) strcpy(pci_data.name, 1473 ddi_get_name(dbp->dip)); 1474 } else { 1475 1476 if ((child_dip = db_lookup_child_name(dbp, 1477 pci_data.name, pci_data.instance)) 1478 == (dev_info_t *)NULL) { 1479 rc = ENXIO; 1480 return (rc); 1481 } else { 1482 if (ddi_getprop(DDI_DEV_T_ANY, 1483 child_dip, DDI_PROP_DONTPASS, 1484 "vendor-id", DB_INVAL_VEND) 1485 == DB_INVAL_VEND) { 1486 /* non PCI device */ 1487 rc = EINVAL; 1488 return (rc); 1489 } 1490 } 1491 } 1492 pci_data.instance = ddi_get_instance(child_dip); 1493 (void) pci_config_setup(child_dip, &config_handle); 1494 db_pci_get_header(config_handle, &pci_data.pri_hdr, 0); 1495 1496 /* if it is the drawbridge itself, read sec header */ 1497 if (child_dip == dbp->dip) { 1498 db_pci_get_header(config_handle, 1499 &pci_data.sec_hdr, DB_PCONF_SEC_HDR_OFF); 1500 db_pci_get_conf_regs(config_handle, 1501 &pci_data.conf_regs); 1502 } 1503 pci_config_teardown(&config_handle); 1504 1505 if (ddi_copyout((caddr_t)&pci_data, (caddr_t)arg, 1506 sizeof (db_pci_data_t), mode)) { 1507 rc = EFAULT; 1508 return (rc); 1509 } 1510 1511 return (rc); 1512 } 1513 #endif /* DB_DEBUG */ 1514 1515 /* 1516 * if secondary nexus (hotplug), then use pcihp_ioctl to do everything 1517 */ 1518 if (dbp->dev_state & DB_SECONDARY_NEXUS) 1519 return ((pcihp_get_cb_ops())->cb_ioctl(dev, cmd, 1520 arg, mode, cred_p, rval_p)); 1521 1522 /* 1523 * if not secondary nexus, we do DEVCTL_DEVICE and DEVCTL_BUS ourselves 1524 */ 1525 self = dbp->dip; 1526 1527 /* 1528 * We can use the generic implementation for these ioctls 1529 */ 1530 switch (cmd) { 1531 case DEVCTL_DEVICE_GETSTATE: 1532 case DEVCTL_DEVICE_ONLINE: 1533 case DEVCTL_DEVICE_OFFLINE: 1534 case DEVCTL_BUS_GETSTATE: 1535 return (ndi_devctl_ioctl(self, cmd, arg, mode, 0)); 1536 } 1537 1538 /* 1539 * read devctl ioctl data 1540 */ 1541 if (ndi_dc_allochdl((void *)arg, &dcp) != NDI_SUCCESS) 1542 return (EFAULT); 1543 1544 switch (cmd) { 1545 1546 case DEVCTL_DEVICE_RESET: 1547 rc = ENOTSUP; 1548 break; 1549 1550 1551 case DEVCTL_BUS_QUIESCE: 1552 if (ndi_get_bus_state(self, &bus_state) == NDI_SUCCESS) 1553 if (bus_state == BUS_QUIESCED) 1554 break; 1555 (void) ndi_set_bus_state(self, BUS_QUIESCED); 1556 break; 1557 1558 case DEVCTL_BUS_UNQUIESCE: 1559 if (ndi_get_bus_state(self, &bus_state) == NDI_SUCCESS) 1560 if (bus_state == BUS_ACTIVE) 1561 break; 1562 (void) ndi_set_bus_state(self, BUS_ACTIVE); 1563 break; 1564 1565 case DEVCTL_BUS_RESET: 1566 rc = ENOTSUP; 1567 break; 1568 1569 case DEVCTL_BUS_RESETALL: 1570 rc = ENOTSUP; 1571 break; 1572 1573 default: 1574 rc = ENOTTY; 1575 } 1576 1577 ndi_dc_freehdl(dcp); 1578 return (rc); 1579 } 1580 1581 #ifdef DB_DEBUG 1582 static dev_info_t * 1583 db_lookup_child_name(db_ctrl_t *dbp, char *name, int instance) 1584 { 1585 dev_info_t *cdip, *pdip = dbp->dip; 1586 1587 for (cdip = ddi_get_child(pdip); cdip; 1588 cdip = ddi_get_next_sibling(pdip)) { 1589 1590 do { 1591 if (strcmp(ddi_node_name(cdip), name) == 0) { 1592 if (instance != -1) { 1593 if (ddi_get_instance(cdip) == instance) 1594 return (cdip); 1595 } else 1596 return (cdip); 1597 } 1598 pdip = cdip; 1599 } while ((cdip = ddi_get_child(pdip))); 1600 cdip = ddi_get_next_sibling(pdip); 1601 if (cdip == NULL) { 1602 pdip = ddi_get_parent(pdip); 1603 if (pdip == dbp->dip) 1604 break; 1605 } 1606 } 1607 return (NULL); 1608 } 1609 1610 static void 1611 db_pci_get_header(ddi_acc_handle_t config_handle, db_pci_header_t *ph, 1612 off_t hdr_off) 1613 { 1614 ph->venid = pci_config_get16(config_handle, hdr_off + PCI_CONF_VENID); 1615 ph->devid = pci_config_get16(config_handle, hdr_off + PCI_CONF_DEVID); 1616 ph->command = pci_config_get16(config_handle, hdr_off + PCI_CONF_COMM); 1617 ph->status = pci_config_get16(config_handle, hdr_off + PCI_CONF_STAT); 1618 ph->revid = pci_config_get8(config_handle, hdr_off + PCI_CONF_REVID); 1619 ph->pif = pci_config_get8(config_handle, hdr_off + PCI_CONF_PROGCLASS); 1620 ph->subclass = pci_config_get8(config_handle, 1621 hdr_off + PCI_CONF_SUBCLASS); 1622 ph->class = pci_config_get8(config_handle, 1623 hdr_off + PCI_CONF_BASCLASS); 1624 ph->cacheline = pci_config_get8(config_handle, 1625 hdr_off + PCI_CONF_CACHE_LINESZ); 1626 ph->lat = pci_config_get8(config_handle, 1627 hdr_off + PCI_CONF_LATENCY_TIMER); 1628 ph->hdr_type = pci_config_get8(config_handle, 1629 hdr_off + PCI_CONF_HEADER); 1630 ph->bist = pci_config_get8(config_handle, hdr_off + PCI_CONF_BIST); 1631 ph->bar0 = pci_config_get32(config_handle, hdr_off + PCI_CONF_BASE0); 1632 ph->bar1 = pci_config_get32(config_handle, hdr_off + PCI_CONF_BASE1); 1633 ph->bar2 = pci_config_get32(config_handle, hdr_off + PCI_CONF_BASE2); 1634 ph->bar3 = pci_config_get32(config_handle, hdr_off + PCI_CONF_BASE3); 1635 ph->bar4 = pci_config_get32(config_handle, hdr_off + PCI_CONF_BASE4); 1636 ph->bar5 = pci_config_get32(config_handle, hdr_off + PCI_CONF_BASE5); 1637 ph->cardbus_cisp = pci_config_get32(config_handle, 1638 hdr_off + PCI_CONF_CIS); 1639 ph->sub_venid = pci_config_get16(config_handle, 1640 hdr_off + PCI_CONF_SUBVENID); 1641 ph->sub_devid = pci_config_get16(config_handle, 1642 hdr_off + PCI_CONF_SUBSYSID); 1643 ph->exprom_bar = pci_config_get32(config_handle, 1644 hdr_off + PCI_CONF_ROM); 1645 ph->int_line = pci_config_get8(config_handle, hdr_off + PCI_CONF_ILINE); 1646 ph->int_pin = pci_config_get8(config_handle, hdr_off + PCI_CONF_IPIN); 1647 ph->min_gnt = pci_config_get8(config_handle, hdr_off + PCI_CONF_MIN_G); 1648 ph->max_lat = pci_config_get8(config_handle, hdr_off + PCI_CONF_MAX_L); 1649 } 1650 1651 static void 1652 db_pci_get_conf_regs(ddi_acc_handle_t config_handle, db_conf_regs_t *cr) 1653 { 1654 cr->ds_mem0_tr_base = pci_config_get32(config_handle, 1655 DB_CONF_DS_MEM0_TR_BASE); 1656 cr->ds_io_mem1_tr_base = pci_config_get32(config_handle, 1657 DB_CONF_DS_IO_MEM1_TR_BASE); 1658 cr->ds_mem2_tr_base = pci_config_get32(config_handle, 1659 DB_CONF_DS_MEM2_TR_BASE); 1660 cr->ds_mem3_tr_base = pci_config_get32(config_handle, 1661 DB_CONF_DS_MEM3_TR_BASE); 1662 cr->us_io_mem0_tr_base = pci_config_get32(config_handle, 1663 DB_CONF_US_IO_MEM0_TR_BASE); 1664 cr->us_mem1_tr_base = pci_config_get32(config_handle, 1665 DB_CONF_US_MEM1_TR_BASE); 1666 cr->ds_mem0_setup_reg = pci_config_get32(config_handle, 1667 DB_CONF_DS_MEM0_SETUP); 1668 cr->ds_io_mem1_setup_reg = pci_config_get32(config_handle, 1669 DB_CONF_DS_IO_MEM1_SETUP); 1670 cr->ds_mem2_setup_reg = pci_config_get32(config_handle, 1671 DB_CONF_DS_MEM2_SETUP); 1672 cr->ds_mem3_setup_reg = pci_config_get64(config_handle, 1673 DB_CONF_DS_MEM3_SETUP); 1674 cr->p_exp_rom_setup = pci_config_get32(config_handle, 1675 DB_CONF_PRIM_EXP_ROM_SETUP); 1676 cr->us_io_mem0_setup_reg = pci_config_get32(config_handle, 1677 DB_CONF_US_IO_MEM0_SETUP); 1678 cr->us_mem1_setup_reg = pci_config_get32(config_handle, 1679 DB_CONF_US_MEM1_SETUP); 1680 cr->chip_control0 = pci_config_get16(config_handle, DB_CONF_CHIP_CTRL0); 1681 cr->chip_control1 = pci_config_get16(config_handle, DB_CONF_CHIP_CTRL1); 1682 cr->chip_status = pci_config_get16(config_handle, DB_CONF_STATUS); 1683 cr->arb_control = pci_config_get16(config_handle, DB_CONF_ARBITER_CTRL); 1684 cr->p_serr_disables = pci_config_get8(config_handle, 1685 DB_CONF_PRIM_SERR_DISABLES); 1686 cr->s_serr_disables = pci_config_get8(config_handle, 1687 DB_CONF_PRIM_SERR_DISABLES); 1688 cr->config_csr = pci_config_get16(config_handle, DB_CONF_CONF_CSR); 1689 cr->reset_control = pci_config_get32(config_handle, DB_CONF_RESET_CTRL); 1690 cr->pm_cap = pci_config_get16(config_handle, DB_CONF_PM_CAP); 1691 cr->pm_csr = pci_config_get16(config_handle, DB_CONF_PM_CSR); 1692 cr->hs_csr = pci_config_get8(config_handle, DB_CONF_HS_CSR); 1693 } 1694 #endif /* DB_DEBUG */ 1695 1696 /* 1697 * Function: db_pci_map 1698 * 1699 * Note: Only memory accesses are direct. IO could be direct 1700 * or indirect. Config accesses are always indirect. 1701 * The question here is, does the "assigned-addresses" 1702 * property entry represents the addresses in the 1703 * local domain or the host domain itself. 1704 * Strictly speaking, the assumption should be that 1705 * it is in the local domain, as the transactions 1706 * upstream or downstream are automatically 1707 * translated by the bridge chip anyway. 1708 * 1709 * Return values: 1710 * DDI_SUCCESS: map call by child device success 1711 * DDI_FAILURE: map operation failed. 1712 */ 1713 1714 static int 1715 db_pci_map(dev_info_t *dip, dev_info_t *rdip, ddi_map_req_t *mp, 1716 off_t offset, off_t len, caddr_t *addrp) 1717 { 1718 register dev_info_t *pdip; 1719 int reg_proplen, num_regs, rnumber; 1720 uint_t addr_space_type; 1721 pci_regspec_t *pci_regsetp, pci_reg; 1722 db_ctrl_t *dbp; 1723 db_acc_pvt_t *db_pvt; 1724 ddi_acc_impl_t *ap; 1725 ddi_acc_hdl_t *hp; 1726 db_acc_cfg_addr_t *pci_addr; 1727 int instance = ddi_get_instance(dip); 1728 1729 DB_DEBUG0(DB_PCI_MAP, dip, "enter\n"); 1730 1731 /* get map type. check for config space */ 1732 switch (mp->map_type) { 1733 1734 case DDI_MT_RNUMBER : 1735 /* get the reg number */ 1736 rnumber = mp->map_obj.rnumber; 1737 1738 if (ddi_getlongprop(DDI_DEV_T_ANY, rdip, 1739 DDI_PROP_DONTPASS, "reg", 1740 (caddr_t)&pci_regsetp, ®_proplen) 1741 != DDI_SUCCESS) 1742 return (DDI_FAILURE); 1743 1744 num_regs = reg_proplen / (int)sizeof (pci_regspec_t); 1745 if (rnumber >= num_regs) { 1746 /* this is a DDI_ME_RNUMBER_RANGE error */ 1747 kmem_free(pci_regsetp, reg_proplen); 1748 return (DDI_FAILURE); 1749 } 1750 1751 pci_reg = pci_regsetp[rnumber]; 1752 kmem_free(pci_regsetp, reg_proplen); 1753 /* FALLTHROUGH */ 1754 case DDI_MT_REGSPEC : 1755 if (mp->map_type == DDI_MT_REGSPEC) 1756 pci_reg = *(pci_regspec_t *)mp->map_obj.rp; 1757 1758 /* 1759 * Intercept config space accesses only. All other 1760 * requests go to the parent. 1761 */ 1762 addr_space_type = pci_reg.pci_phys_hi & PCI_ADDR_MASK; 1763 1764 DB_DEBUG3(DB_PCI_MAP, dip, "rdip=%lx, rnum=%d(%d)\n", 1765 rdip, rnumber, num_regs); 1766 1767 /* if we do direct map IO, then lets break here */ 1768 if ((db_io_map_mode & DB_IO_MAP_DIRECT) && 1769 (addr_space_type == PCI_ADDR_IO)) 1770 break; 1771 1772 if ((addr_space_type != PCI_ADDR_CONFIG) && 1773 (addr_space_type != PCI_ADDR_IO)) 1774 break; 1775 1776 /* 1777 * User mapping requests not legal for indirect 1778 * IO/Config Space 1779 */ 1780 if (mp->map_op == DDI_MO_MAP_HANDLE) 1781 return (DDI_FAILURE); 1782 1783 dbp = (db_ctrl_t *)ddi_get_soft_state(db_state, 1784 instance); 1785 /* get our common access handle */ 1786 hp = (ddi_acc_hdl_t *)mp->map_handlep; 1787 1788 /* Check for unmap operation */ 1789 if ((mp->map_op == DDI_MO_UNMAP) || 1790 (mp->map_op == DDI_MO_UNLOCK)) { 1791 /* 1792 * free up memory allocated for our 1793 * private access handle. 1794 */ 1795 db_pvt = (db_acc_pvt_t *) 1796 hp->ah_bus_private; 1797 DB_DEBUG1(DB_PCI_MAP, dip, 1798 "unmap rdip=%lx\n", rdip); 1799 kmem_free((void *)db_pvt, 1800 sizeof (db_acc_pvt_t)); 1801 1802 /* 1803 * unmap operation of PCI IO/config 1804 * space. 1805 */ 1806 return (DDI_SUCCESS); 1807 } 1808 1809 if (addr_space_type == PCI_ADDR_CONFIG) { 1810 /* Config space access range check */ 1811 if ((offset >= PCI_CONF_HDR_SIZE) || 1812 (len > PCI_CONF_HDR_SIZE) || 1813 (offset + len > PCI_CONF_HDR_SIZE)) { 1814 1815 return (DDI_FAILURE); 1816 } 1817 } 1818 1819 /* define the complete access handle */ 1820 hp = (ddi_acc_hdl_t *)mp->map_handlep; 1821 1822 ap = (ddi_acc_impl_t *)hp->ah_platform_private; 1823 1824 ap->ahi_get8 = db_ddi_get8; 1825 ap->ahi_get16 = db_ddi_get16; 1826 ap->ahi_get32 = db_ddi_get32; 1827 ap->ahi_get64 = db_ddi_get64; 1828 ap->ahi_put8 = db_ddi_put8; 1829 ap->ahi_put16 = db_ddi_put16; 1830 ap->ahi_put32 = db_ddi_put32; 1831 ap->ahi_put64 = db_ddi_put64; 1832 ap->ahi_rep_get8 = db_ddi_rep_get8; 1833 ap->ahi_rep_get16 = db_ddi_rep_get16; 1834 ap->ahi_rep_get32 = db_ddi_rep_get32; 1835 ap->ahi_rep_get64 = db_ddi_rep_get64; 1836 ap->ahi_rep_put8 = db_ddi_rep_put8; 1837 ap->ahi_rep_put16 = db_ddi_rep_put16; 1838 ap->ahi_rep_put32 = db_ddi_rep_put32; 1839 ap->ahi_rep_put64 = db_ddi_rep_put64; 1840 1841 /* Initialize to default check/notify functions */ 1842 ap->ahi_fault = 0; 1843 ap->ahi_fault_check = i_ddi_acc_fault_check; 1844 ap->ahi_fault_notify = i_ddi_acc_fault_notify; 1845 1846 /* allocate memory for our private handle */ 1847 db_pvt = kmem_zalloc(sizeof (db_acc_pvt_t), KM_SLEEP); 1848 hp->ah_bus_private = (void *)db_pvt; 1849 db_pvt->dbp = dbp; 1850 1851 /* record the device address for future use */ 1852 pci_addr = &db_pvt->dev_addr; 1853 pci_addr->c_busnum = 1854 PCI_REG_BUS_G(pci_reg.pci_phys_hi); 1855 pci_addr->c_devnum = 1856 PCI_REG_DEV_G(pci_reg.pci_phys_hi); 1857 pci_addr->c_funcnum = 1858 PCI_REG_FUNC_G(pci_reg.pci_phys_hi); 1859 /* 1860 * We should keep the upstream or 1861 * downstream info in our own ah_bus_private 1862 * structure, so that we do not waste our 1863 * time in the actual IO routines, figuring out 1864 * if we should use upstream or downstream 1865 * configuration addr/data register. 1866 * So, check orientation and setup registers 1867 * right now. 1868 */ 1869 switch (addr_space_type) { 1870 1871 case PCI_ADDR_CONFIG : 1872 if (dbp->dev_state & DB_PRIMARY_NEXUS) { 1873 DB_DEBUG0(DB_PCI_MAP, dip, "primary\n"); 1874 db_pvt->mask = DS8_CONF_OWN; 1875 if (db_conf_map_mode & 1876 DB_CONF_MAP_INDIRECT_IO) { 1877 DB_DEBUG0(DB_PCI_MAP, dip, 1878 "INDIRECT_CONF\n"); 1879 1880 db_pvt->handle = 1881 dbp->csr_io_handle; 1882 db_pvt->addr = 1883 (uint32_t *) 1884 ((uchar_t *)dbp->csr_io 1885 + DB_CSR_DS_CONF_ADDR); 1886 db_pvt->data = 1887 (uint32_t *) 1888 ((uchar_t *)dbp->csr_io 1889 + DB_CSR_DS_CONF_DATA); 1890 db_pvt->bus_own = 1891 (uint8_t *) 1892 ((uchar_t *)dbp->csr_io 1893 + DB_CSR8_DS_CONF_OWN); 1894 db_pvt->bus_release = 1895 (uint8_t *) 1896 ((uchar_t *)dbp->csr_io 1897 + DB_CSR8_DS_CONF_CSR); 1898 } else { 1899 DB_DEBUG0(DB_PCI_MAP, dip, 1900 "DIRECT_CONF\n"); 1901 1902 db_pvt->handle = 1903 dbp->conf_handle; 1904 db_pvt->addr = 1905 (uint32_t *) 1906 ((uchar_t *)dbp->conf_io 1907 + DB_CONF_DS_CONF_ADDR); 1908 db_pvt->data = (uint32_t *) 1909 ((uchar_t *)dbp->conf_io 1910 + DB_CONF_DS_CONF_DATA); 1911 db_pvt->bus_own = 1912 (uint8_t *) 1913 ((uchar_t *)dbp->conf_io 1914 + DB_CONF8_DS_CONF_OWN); 1915 db_pvt->bus_release = 1916 (uint8_t *) 1917 ((uchar_t *)dbp->conf_io 1918 + DB_CONF8_DS_CONF_CSR); 1919 } 1920 } else { 1921 DB_DEBUG0(DB_PCI_MAP, dip, 1922 "secondary\n"); 1923 db_pvt->mask = US8_CONF_OWN; 1924 if (db_conf_map_mode & 1925 DB_CONF_MAP_INDIRECT_IO) { 1926 DB_DEBUG0(DB_PCI_MAP, dip, 1927 "INDIRECT_CONF\n"); 1928 1929 db_pvt->handle = 1930 dbp->csr_io_handle; 1931 db_pvt->addr = 1932 (uint32_t *) 1933 ((uchar_t *)dbp->csr_io 1934 + DB_CSR_US_CONF_ADDR); 1935 db_pvt->data = 1936 (uint32_t *) 1937 ((uchar_t *)dbp->csr_io 1938 + DB_CSR_US_CONF_DATA); 1939 db_pvt->bus_own = 1940 (uint8_t *) 1941 ((uchar_t *)dbp->csr_io 1942 + DB_CSR8_US_CONF_OWN); 1943 db_pvt->bus_release = 1944 (uint8_t *) 1945 ((uchar_t *)dbp->csr_io 1946 + DB_CSR8_US_CONF_CSR); 1947 } else { 1948 DB_DEBUG0(DB_PCI_MAP, dip, 1949 "DIRECT_CONF\n"); 1950 1951 db_pvt->handle = 1952 dbp->conf_handle; 1953 db_pvt->addr = 1954 (uint32_t *) 1955 ((uchar_t *)dbp->conf_io 1956 + DB_CONF_US_CONF_ADDR); 1957 db_pvt->data = 1958 (uint32_t *) 1959 ((uchar_t *)dbp->conf_io 1960 + DB_CONF_US_CONF_DATA); 1961 db_pvt->bus_own = 1962 (uint8_t *) 1963 ((uchar_t *)dbp->conf_io 1964 + DB_CONF8_US_CONF_OWN); 1965 db_pvt->bus_release = 1966 (uint8_t *) 1967 ((uchar_t *)dbp->conf_io 1968 + DB_CONF8_US_CONF_CSR); 1969 } 1970 } 1971 break; 1972 1973 case PCI_ADDR_IO : 1974 DB_DEBUG0(DB_PCI_MAP, dip, "PCI_ADDR_IO\n"); 1975 1976 /* ap->ahi_acc_attr |= DDI_ACCATTR_IO_SPACE; */ 1977 db_pvt->handle = dbp->csr_io_handle; 1978 if (dbp->dev_state & DB_PRIMARY_NEXUS) { 1979 DB_DEBUG0(DB_PCI_MAP, dip, "primary\n"); 1980 db_pvt->addr = (uint32_t *) 1981 ((uchar_t *)dbp->csr_io 1982 + DB_CSR_DS_IO_ADDR); 1983 db_pvt->data = (uint32_t *) 1984 ((uchar_t *)dbp->csr_io 1985 + DB_CSR_DS_IO_DATA); 1986 db_pvt->bus_own = (uint8_t *) 1987 ((uchar_t *)dbp->csr_io 1988 + DB_CSR8_DS_IO_OWN); 1989 db_pvt->bus_release = (uint8_t *) 1990 ((uchar_t *)dbp->csr_io 1991 + DB_CSR8_DS_IO_CSR); 1992 db_pvt->mask = DS8_IO_OWN; 1993 } else { 1994 DB_DEBUG0(DB_PCI_MAP, dip, 1995 "secondary\n"); 1996 db_pvt->addr = (uint32_t *) 1997 ((uchar_t *)dbp->csr_io 1998 + DB_CSR_US_IO_ADDR); 1999 db_pvt->data = (uint32_t *) 2000 ((uchar_t *)dbp->csr_io 2001 + DB_CSR_US_IO_DATA); 2002 db_pvt->bus_own = (uint8_t *) 2003 ((uchar_t *)dbp->csr_io 2004 + DB_CSR8_US_IO_OWN); 2005 db_pvt->bus_release = (uint8_t *) 2006 ((uchar_t *)dbp->csr_io 2007 + DB_CSR8_US_IO_CSR); 2008 db_pvt->mask = US8_IO_OWN; 2009 } 2010 break; 2011 2012 default : 2013 DB_DEBUG0(DB_PCI_MAP, dip, 2014 "PCI_ADDR unknown\n"); 2015 break; 2016 } 2017 2018 /* make and store a type 0/1 address in the *addrp */ 2019 if (pci_addr->c_busnum == dbp->range.lo) { 2020 *addrp = (caddr_t)DB_PCI_REG_ADDR_TYPE0( 2021 pci_addr->c_busnum, 2022 pci_addr->c_devnum, 2023 pci_addr->c_funcnum, 2024 offset); 2025 db_pvt->access_mode |= DB_PCI_CONF_CYCLE_TYPE0; 2026 DB_DEBUG0(DB_PCI_MAP, dip, 2027 "access mode type 0\n"); 2028 } else { 2029 *addrp = (caddr_t)DB_PCI_REG_ADDR_TYPE1( 2030 pci_addr->c_busnum, 2031 pci_addr->c_devnum, 2032 pci_addr->c_funcnum, 2033 offset); 2034 db_pvt->access_mode |= DB_PCI_CONF_CYCLE_TYPE1; 2035 DB_DEBUG0(DB_PCI_MAP, dip, 2036 "access mode type 1\n"); 2037 } 2038 DB_DEBUG4(DB_PCI_MAP, dip, "addrp<%x,%x,%x> = %lx\n", 2039 pci_addr->c_busnum, pci_addr->c_devnum, 2040 pci_addr->c_funcnum, *addrp); 2041 2042 return (DDI_SUCCESS); 2043 2044 default : 2045 DB_DEBUG1(DB_PCI_MAP, dip, "DDI other %x\n", 2046 mp->map_type); 2047 break; 2048 } 2049 DB_DEBUG0(DB_PCI_MAP, dip, "exit\n"); 2050 2051 pdip = (dev_info_t *)DEVI(dip)->devi_parent; 2052 return ((DEVI(pdip)->devi_ops->devo_bus_ops->bus_map) 2053 (pdip, rdip, mp, offset, len, addrp)); 2054 } 2055 2056 #ifdef DB_DEBUG 2057 char *db_ctlop_name[] = { 2058 "DDI_CTLOPS_DMAPMAPC", 2059 "DDI_CTLOPS_INITCHILD", 2060 "DDI_CTLOPS_UNINITCHILD", 2061 "DDI_CTLOPS_REPORTDEV", 2062 "DDI_CTLOPS_REPORTINT", 2063 "DDI_CTLOPS_REGSIZE", 2064 "DDI_CTLOPS_NREGS", 2065 "DDI_CTLOPS_RESERVED0", 2066 "DDI_CTLOPS_SIDDEV", 2067 "DDI_CTLOPS_SLAVEONLY", 2068 "DDI_CTLOPS_AFFINITY", 2069 "DDI_CTLOPS_IOMIN", 2070 "DDI_CTLOPS_PTOB", 2071 "DDI_CTLOPS_BTOP", 2072 "DDI_CTLOPS_BTOPR", 2073 "DDI_CTLOPS_RESERVED1", 2074 "DDI_CTLOPS_RESERVED2", 2075 "DDI_CTLOPS_RESERVED3", 2076 "DDI_CTLOPS_RESERVED4", 2077 "DDI_CTLOPS_RESERVED5", 2078 "DDI_CTLOPS_DVMAPAGESIZE", 2079 "DDI_CTLOPS_POWER", 2080 "DDI_CTLOPS_ATTACH", 2081 "DDI_CTLOPS_DETACH", 2082 "DDI_CTLOPS_POKE", 2083 "DDI_CTLOPS_PEEK" 2084 }; 2085 #endif 2086 2087 static int 2088 db_ctlops(dev_info_t *dip, dev_info_t *rdip, 2089 ddi_ctl_enum_t ctlop, void *arg, void *result) 2090 { 2091 2092 if ((ctlop >= DDI_CTLOPS_DMAPMAPC) && 2093 (ctlop <= DDI_CTLOPS_DETACH)) { 2094 DB_DEBUG1(DB_CTLOPS, dip, "ctlop=%s\n", db_ctlop_name[ctlop]); 2095 } else { 2096 DB_DEBUG1(DB_CTLOPS, dip, "ctlop=%d\n", ctlop); 2097 } 2098 2099 switch (ctlop) { 2100 case DDI_CTLOPS_REPORTDEV : 2101 if (rdip == (dev_info_t *)0) 2102 return (DDI_FAILURE); 2103 cmn_err(CE_CONT, "?PCI-device: %s@%s, %s#%d\n", 2104 ddi_node_name(rdip), ddi_get_name_addr(rdip), 2105 ddi_driver_name(rdip), 2106 ddi_get_instance(rdip)); 2107 return (DDI_SUCCESS); 2108 2109 case DDI_CTLOPS_INITCHILD : 2110 return (db_initchild((dev_info_t *)arg)); 2111 2112 case DDI_CTLOPS_UNINITCHILD : 2113 db_uninitchild((dev_info_t *)arg); 2114 return (DDI_SUCCESS); 2115 2116 case DDI_CTLOPS_SIDDEV : 2117 return (DDI_SUCCESS); 2118 2119 case DDI_CTLOPS_REGSIZE : 2120 case DDI_CTLOPS_NREGS : 2121 if (rdip == (dev_info_t *)0) 2122 return (DDI_FAILURE); 2123 /* fall through */ 2124 2125 default : 2126 return (ddi_ctlops(dip, rdip, ctlop, arg, result)); 2127 } 2128 2129 } 2130 2131 static dev_info_t * 2132 db_get_my_childs_dip(dev_info_t *dip, dev_info_t *rdip) 2133 { 2134 dev_info_t *cdip = rdip; 2135 2136 for (; ddi_get_parent(cdip) != dip; cdip = ddi_get_parent(cdip)) 2137 ; 2138 2139 return (cdip); 2140 } 2141 2142 static int 2143 db_intr_ops(dev_info_t *dip, dev_info_t *rdip, ddi_intr_op_t intr_op, 2144 ddi_intr_handle_impl_t *hdlp, void *result) 2145 { 2146 dev_info_t *cdip = rdip; 2147 pci_regspec_t *pci_rp; 2148 int reglen, len; 2149 uint32_t d, intr; 2150 2151 DB_DEBUG1(DB_INTR_OPS, dip, "intr_op=%d\n", intr_op); 2152 2153 if (hdlp->ih_type != DDI_INTR_TYPE_FIXED) 2154 goto done; 2155 2156 /* 2157 * If the interrupt-map property is defined at this 2158 * node, it will have performed the interrupt 2159 * translation as part of the property, so no 2160 * rotation needs to be done. 2161 */ 2162 2163 if (ddi_getproplen(DDI_DEV_T_ANY, dip, DDI_PROP_DONTPASS, 2164 "interrupt-map", &len) == DDI_PROP_SUCCESS) 2165 goto done; 2166 2167 cdip = db_get_my_childs_dip(dip, rdip); 2168 2169 /* 2170 * Use the devices reg property to determine it's 2171 * PCI bus number and device number. 2172 */ 2173 if (ddi_getlongprop(DDI_DEV_T_ANY, cdip, DDI_PROP_DONTPASS, 2174 "reg", (caddr_t)&pci_rp, ®len) != DDI_SUCCESS) 2175 return (DDI_FAILURE); 2176 2177 intr = hdlp->ih_vector; 2178 2179 /* Spin the interrupt */ 2180 d = PCI_REG_DEV_G(pci_rp[0].pci_phys_hi); 2181 2182 if ((intr >= PCI_INTA) && (intr <= PCI_INTD)) 2183 hdlp->ih_vector = ((intr - 1 + (d % 4)) % 4 + 1); 2184 else 2185 cmn_err(CE_WARN, "%s#%d: %s: PCI intr=%x out of range", 2186 ddi_driver_name(rdip), ddi_get_instance(rdip), 2187 ddi_driver_name(dip), intr); 2188 2189 DB_DEBUG3(DB_INTR_OPS, dip, "intr=%d, d=%d, is_intr=%d\n", 2190 intr, d, hdlp->ih_vector); 2191 2192 kmem_free(pci_rp, reglen); 2193 2194 done: 2195 /* Pass up the request to our parent. */ 2196 return (i_ddi_intr_ops(dip, rdip, intr_op, hdlp, result)); 2197 } 2198 2199 static int 2200 db_name_child(dev_info_t *child, char *name, int namelen) 2201 { 2202 uint_t n, slot, func; 2203 pci_regspec_t *pci_rp; 2204 2205 if (ndi_dev_is_persistent_node(child) == 0) { 2206 char **unit_addr; 2207 2208 /* name .conf nodes by "unit-address" property" */ 2209 if (ddi_prop_lookup_string_array(DDI_DEV_T_ANY, child, 2210 DDI_PROP_DONTPASS, "unit-address", &unit_addr, &n) != 2211 DDI_PROP_SUCCESS) { 2212 cmn_err(CE_WARN, "cannot name node from %s.conf", 2213 ddi_driver_name(child)); 2214 return (DDI_FAILURE); 2215 } 2216 if (n != 1 || *unit_addr == NULL || **unit_addr == 0) { 2217 cmn_err(CE_WARN, "unit-address property in %s.conf" 2218 " not well-formed", ddi_driver_name(child)); 2219 ddi_prop_free(unit_addr); 2220 return (DDI_FAILURE); 2221 } 2222 2223 (void) snprintf(name, namelen, "%s", *unit_addr); 2224 ddi_prop_free(unit_addr); 2225 return (DDI_SUCCESS); 2226 } 2227 2228 /* name hardware nodes by "reg" property */ 2229 if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY, child, 0, "reg", 2230 (int **)&pci_rp, &n) != DDI_SUCCESS) 2231 return (DDI_FAILURE); 2232 2233 /* get the device identifications */ 2234 slot = PCI_REG_DEV_G(pci_rp->pci_phys_hi); 2235 func = PCI_REG_FUNC_G(pci_rp->pci_phys_hi); 2236 2237 if (func != 0) 2238 (void) snprintf(name, namelen, "%x,%x", slot, func); 2239 else 2240 (void) snprintf(name, namelen, "%x", slot); 2241 2242 ddi_prop_free(pci_rp); 2243 return (DDI_SUCCESS); 2244 } 2245 2246 static int 2247 db_initchild(dev_info_t *child) 2248 { 2249 char name[MAXNAMELEN]; 2250 ddi_acc_handle_t config_handle; 2251 ushort_t command_preserve, command; 2252 uint_t n; 2253 ushort_t bcr; 2254 uchar_t header_type, min_gnt, latency_timer; 2255 db_ctrl_t *dbp; 2256 2257 if (db_name_child(child, name, MAXNAMELEN) != DDI_SUCCESS) 2258 return (DDI_FAILURE); 2259 2260 ddi_set_name_addr(child, name); 2261 ddi_set_parent_data(child, NULL); 2262 2263 /* 2264 * Pseudo nodes indicate a prototype node with per-instance 2265 * properties to be merged into the real h/w device node. 2266 * The interpretation of the unit-address is DD[,F] 2267 * where DD is the device id and F is the function. 2268 */ 2269 if (ndi_dev_is_persistent_node(child) == 0) { 2270 extern int pci_allow_pseudo_children; 2271 2272 /* 2273 * Try to merge the properties from this prototype 2274 * node into real h/w nodes. 2275 */ 2276 if (ndi_merge_node(child, db_name_child) == DDI_SUCCESS) { 2277 /* 2278 * Merged ok - return failure to remove the node. 2279 */ 2280 return (DDI_FAILURE); 2281 } 2282 2283 /* workaround for ddivs to run under PCI */ 2284 if (pci_allow_pseudo_children) { 2285 return (DDI_SUCCESS); 2286 } 2287 2288 /* 2289 * The child was not merged into a h/w node, 2290 * but there's not much we can do with it other 2291 * than return failure to cause the node to be removed. 2292 */ 2293 cmn_err(CE_WARN, "!%s@%s: %s.conf properties not merged", 2294 ddi_driver_name(child), ddi_get_name_addr(child), 2295 ddi_driver_name(child)); 2296 return (DDI_NOT_WELL_FORMED); 2297 } 2298 2299 2300 if ((db_create_pci_prop(child) != DDI_SUCCESS) || 2301 (pci_config_setup(child, &config_handle) != DDI_SUCCESS)) { 2302 db_uninitchild(child); 2303 return (DDI_FAILURE); 2304 } 2305 2306 /* 2307 * Determine the configuration header type. 2308 */ 2309 header_type = pci_config_get8(config_handle, PCI_CONF_HEADER); 2310 2311 /* 2312 * Support for the "command-preserve" property. 2313 */ 2314 command_preserve = ddi_prop_get_int(DDI_DEV_T_ANY, child, 2315 DDI_PROP_DONTPASS, "command-preserve", 0); 2316 command = pci_config_get16(config_handle, PCI_CONF_COMM); 2317 command &= (command_preserve | PCI_COMM_BACK2BACK_ENAB); 2318 command |= (db_command_default & ~command_preserve); 2319 pci_config_put16(config_handle, PCI_CONF_COMM, command); 2320 2321 DB_DEBUG2(DB_INITCHILD, ddi_get_parent(child), 2322 "initializing device vend=%x, devid=%x\n", 2323 pci_config_get16(config_handle, PCI_CONF_VENID), 2324 pci_config_get16(config_handle, PCI_CONF_DEVID)); 2325 /* 2326 * If the device has a bus control register then program it 2327 * based on the settings in the command register. 2328 */ 2329 if ((header_type & PCI_HEADER_TYPE_M) == PCI_HEADER_ONE) { 2330 bcr = pci_config_get8(config_handle, PCI_BCNF_BCNTRL); 2331 if (db_command_default & PCI_COMM_PARITY_DETECT) 2332 bcr |= PCI_BCNF_BCNTRL_PARITY_ENABLE; 2333 if (db_command_default & PCI_COMM_SERR_ENABLE) 2334 bcr |= PCI_BCNF_BCNTRL_SERR_ENABLE; 2335 bcr |= PCI_BCNF_BCNTRL_MAST_AB_MODE; 2336 pci_config_put8(config_handle, PCI_BCNF_BCNTRL, bcr); 2337 } 2338 2339 dbp = (db_ctrl_t *)ddi_get_soft_state(db_state, 2340 ddi_get_instance(ddi_get_parent(child))); 2341 2342 /* 2343 * Initialize cache-line-size configuration register if needed. 2344 */ 2345 if (db_set_cache_line_size_register && 2346 ddi_getprop(DDI_DEV_T_ANY, child, DDI_PROP_DONTPASS, 2347 "cache-line-size", 0) == 0) { 2348 pci_config_put8(config_handle, PCI_CONF_CACHE_LINESZ, 2349 dbp->cache_line_size); 2350 n = pci_config_get8(config_handle, PCI_CONF_CACHE_LINESZ); 2351 if (n != 0) { 2352 (void) ndi_prop_update_int(DDI_DEV_T_NONE, child, 2353 "cache-line-size", n); 2354 } 2355 DB_DEBUG1(DB_INITCHILD, ddi_get_parent(child), 2356 "\nChild Device Cache Size %x\n", dbp->cache_line_size); 2357 } 2358 2359 /* 2360 * Initialize latency timer configuration registers if needed. 2361 */ 2362 if (db_set_latency_timer_register && 2363 ddi_getprop(DDI_DEV_T_ANY, child, DDI_PROP_DONTPASS, 2364 "latency-timer", 0) == 0) { 2365 2366 if ((header_type & PCI_HEADER_TYPE_M) == PCI_HEADER_ONE) { 2367 latency_timer = dbp->p_latency_timer; 2368 pci_config_put8(config_handle, PCI_BCNF_LATENCY_TIMER, 2369 dbp->latency_timer); 2370 } else { 2371 min_gnt = pci_config_get8(config_handle, 2372 PCI_CONF_MIN_G); 2373 latency_timer = min_gnt * 8; 2374 } 2375 pci_config_put8(config_handle, PCI_CONF_LATENCY_TIMER, 2376 latency_timer); 2377 n = pci_config_get8(config_handle, PCI_CONF_LATENCY_TIMER); 2378 if (n != 0) { 2379 (void) ndi_prop_update_int(DDI_DEV_T_NONE, child, 2380 "latency-timer", n); 2381 } 2382 DB_DEBUG1(DB_INITCHILD, ddi_get_parent(child), 2383 "\nChild Device latency %x\n", latency_timer); 2384 } 2385 2386 pci_config_teardown(&config_handle); 2387 return (DDI_SUCCESS); 2388 } 2389 2390 static void 2391 db_uninitchild(dev_info_t *dip) 2392 { 2393 ddi_set_name_addr(dip, NULL); 2394 2395 /* 2396 * Strip the node to properly convert it back to prototype form 2397 */ 2398 impl_rem_dev_props(dip); 2399 } 2400 2401 static int 2402 db_create_pci_prop(dev_info_t *child) 2403 { 2404 pci_regspec_t *pci_rp; 2405 int length; 2406 int value; 2407 2408 /* get child "reg" property */ 2409 value = ddi_getlongprop(DDI_DEV_T_ANY, child, DDI_PROP_CANSLEEP, 2410 "reg", (caddr_t)&pci_rp, &length); 2411 if (value != DDI_SUCCESS) 2412 return (value); 2413 2414 (void) ndi_prop_update_byte_array(DDI_DEV_T_NONE, child, "reg", 2415 (uchar_t *)pci_rp, length); 2416 2417 /* 2418 * free the memory allocated by ddi_getlongprop (). 2419 */ 2420 kmem_free(pci_rp, length); 2421 2422 /* 2423 * No need to create any 1275 properties here, because either 2424 * the OBP creates them or the hotplug framework creates it 2425 * during a hotplug operation. So lets return here. 2426 */ 2427 return (DDI_SUCCESS); 2428 } 2429 2430 /* 2431 * db_save_config_regs 2432 * 2433 * This routine saves the state of the configuration registers of all 2434 * immediate child nodes. 2435 * 2436 * used by: db_detach() on suspends 2437 * 2438 * return value: DDI_SUCCESS: ALl children state saved. 2439 * DDI_FAILURE: Child device state could not be saved. 2440 */ 2441 static int 2442 db_save_config_regs(db_ctrl_t *dbp) 2443 { 2444 int i; 2445 dev_info_t *dip; 2446 ddi_acc_handle_t config_handle; 2447 db_cfg_state_t *statep; 2448 2449 for (i = 0, dip = ddi_get_child(dbp->dip); dip != NULL; 2450 dip = ddi_get_next_sibling(dip)) { 2451 if (i_ddi_node_state(dip) >= DS_ATTACHED) 2452 i++; 2453 } 2454 dbp->config_state_index = i; 2455 2456 if (!i) { 2457 /* no children */ 2458 dbp->db_config_state_p = NULL; 2459 return (DDI_SUCCESS); 2460 } 2461 2462 /* i now equals the total number of child devices */ 2463 dbp->db_config_state_p = 2464 kmem_zalloc(i * sizeof (db_cfg_state_t), KM_NOSLEEP); 2465 if (!dbp->db_config_state_p) { 2466 cmn_err(CE_WARN, 2467 "%s#%d: No memory to save state for child %s#%d\n", 2468 ddi_driver_name(dbp->dip), 2469 ddi_get_instance(dbp->dip), 2470 ddi_get_name(dip), ddi_get_instance(dip)); 2471 return (DDI_FAILURE); 2472 } 2473 2474 for (statep = dbp->db_config_state_p, 2475 dip = ddi_get_child(dbp->dip); 2476 dip != NULL; 2477 dip = ddi_get_next_sibling(dip)) { 2478 2479 if (i_ddi_node_state(dip) < DS_ATTACHED) 2480 continue; 2481 2482 if (pci_config_setup(dip, &config_handle) != DDI_SUCCESS) { 2483 cmn_err(CE_WARN, 2484 "%s#%d: can't config space for %s#%d", 2485 ddi_driver_name(dbp->dip), 2486 ddi_get_instance(dbp->dip), 2487 ddi_driver_name(dip), 2488 ddi_get_instance(dip)); 2489 continue; 2490 } 2491 2492 statep->dip = dip; 2493 statep->command = 2494 pci_config_get16(config_handle, PCI_CONF_COMM); 2495 statep->header_type = 2496 pci_config_get8(config_handle, PCI_CONF_HEADER); 2497 if ((statep->header_type & PCI_HEADER_TYPE_M) == PCI_HEADER_ONE) 2498 statep->bridge_control = 2499 pci_config_get16(config_handle, PCI_BCNF_BCNTRL); 2500 statep->cache_line_size = 2501 pci_config_get8(config_handle, PCI_CONF_CACHE_LINESZ); 2502 statep->latency_timer = 2503 pci_config_get8(config_handle, PCI_CONF_LATENCY_TIMER); 2504 if ((statep->header_type & PCI_HEADER_TYPE_M) == PCI_HEADER_ONE) 2505 statep->sec_latency_timer = 2506 pci_config_get8(config_handle, 2507 PCI_BCNF_LATENCY_TIMER); 2508 pci_config_teardown(&config_handle); 2509 statep++; 2510 } 2511 return (DDI_SUCCESS); 2512 } 2513 2514 2515 /* 2516 * db_restore_config_regs 2517 * 2518 * This routine restores the state of the configuration registers of 2519 * all immediate child nodes. 2520 * 2521 * used by: db_attach() on resume 2522 * 2523 * return value: none 2524 */ 2525 static int 2526 db_restore_config_regs(db_ctrl_t *dbp) 2527 { 2528 int i; 2529 dev_info_t *dip; 2530 ddi_acc_handle_t config_handle; 2531 db_cfg_state_t *statep = dbp->db_config_state_p; 2532 2533 for (i = 0; i < dbp->config_state_index; i++, statep++) { 2534 dip = statep->dip; 2535 if (!dip) { 2536 cmn_err(CE_WARN, 2537 "%s#%d: skipping bad dev info (index %d)", 2538 ddi_driver_name(dbp->dip), 2539 ddi_get_instance(dbp->dip), i); 2540 continue; 2541 } 2542 if (pci_config_setup(dip, &config_handle) != DDI_SUCCESS) { 2543 cmn_err(CE_WARN, 2544 "%s#%d: can't config space for %s#%d", 2545 ddi_driver_name(dbp->dip), 2546 ddi_get_instance(dbp->dip), 2547 ddi_driver_name(dip), 2548 ddi_get_instance(dip)); 2549 continue; 2550 } 2551 pci_config_put16(config_handle, PCI_CONF_COMM, statep->command); 2552 if ((statep->header_type & PCI_HEADER_TYPE_M) == PCI_HEADER_ONE) 2553 pci_config_put16(config_handle, PCI_BCNF_BCNTRL, 2554 statep->bridge_control); 2555 pci_config_put8(config_handle, PCI_CONF_CACHE_LINESZ, 2556 statep->cache_line_size); 2557 pci_config_put8(config_handle, PCI_CONF_LATENCY_TIMER, 2558 statep->latency_timer); 2559 if ((statep->header_type & PCI_HEADER_TYPE_M) == PCI_HEADER_ONE) 2560 pci_config_put8(config_handle, PCI_BCNF_LATENCY_TIMER, 2561 statep->sec_latency_timer); 2562 pci_config_teardown(&config_handle); 2563 } 2564 2565 kmem_free(dbp->db_config_state_p, 2566 dbp->config_state_index * sizeof (db_cfg_state_t)); 2567 dbp->db_config_state_p = NULL; 2568 dbp->config_state_index = 0; 2569 2570 return (DDI_SUCCESS); 2571 } 2572 2573 /* put a type 0/1 address on the bus */ 2574 static void 2575 db_put_reg_conf_addr(db_acc_pvt_t *db_pvt, uint32_t conf_addr) 2576 { 2577 if (db_pvt->access_mode & DB_PCI_CONF_CYCLE_TYPE0)\ 2578 ddi_put32(db_pvt->handle, db_pvt->addr, (uint32_t)\ 2579 DB_PCI_CONF_CYCLE_TYPE0_ADDR((conf_addr)));\ 2580 else /* type 1 cycle */\ 2581 ddi_put32(db_pvt->handle, db_pvt->addr, (uint32_t)\ 2582 DB_PCI_CONF_CYCLE_TYPE1_ADDR((conf_addr))); 2583 } 2584 2585 /* Get 8bits data off the 32bit data */ 2586 static uint8_t 2587 db_get_data8(uint32_t addr, uint32_t data) 2588 { 2589 return (((data) >> (((addr) & 3) * 8)) & 0xff); 2590 } 2591 2592 /* Get 16bits data off the 32bit data */ 2593 static uint16_t 2594 db_get_data16(uint32_t addr, uint32_t data) 2595 { 2596 return (((data) >> (((addr) & 3) * 8)) & 0xffff); 2597 } 2598 2599 /* merge 8bit data into the 32bit data */ 2600 static uint32_t 2601 db_put_data8(uint32_t addr, uint32_t rdata, uint8_t wdata) 2602 { 2603 return ((rdata & (~((0xff << ((((addr) & 3) * 8))) & 0xffffffff))) | 2604 (((wdata) & 0xff)<<((((addr) & 3))*8))); 2605 } 2606 2607 /* merge 16bit data into the 32bit data */ 2608 static uint32_t 2609 db_put_data16(uint32_t addr, uint32_t rdata, uint16_t wdata) 2610 { 2611 return ((rdata & (~((0xffff << ((((addr) & 3) * 8))) & 0xffffffff))) | 2612 (((wdata) & 0xffff) << ((((addr) & 3))*8))); 2613 } 2614 2615 2616 /* 2617 * For the next set of PCI configuration IO calls, we need 2618 * to make sure we own the bus before generating the config cycles, 2619 * using the drawbridge's semaphore method. 2620 */ 2621 2622 /* 2623 * Function to read 8 bit data off the PCI configuration space behind 2624 * the 21554's host interface. 2625 */ 2626 static uint8_t 2627 db_ddi_get8(ddi_acc_impl_t *handle, uint8_t *addr) 2628 { 2629 uint32_t data; 2630 2631 data = db_ddi_get32(handle, (uint32_t *)addr); 2632 return (db_get_data8((uint32_t)addr, data)); 2633 } 2634 2635 /* 2636 * Function to read 16 bit data off the PCI configuration space behind 2637 * the 21554's host interface. 2638 */ 2639 static uint16_t 2640 db_ddi_get16(ddi_acc_impl_t *handle, uint16_t *addr) 2641 { 2642 uint32_t data; 2643 2644 data = db_ddi_get32(handle, (uint32_t *)addr); 2645 return (db_get_data16((uint32_t)addr, data)); 2646 } 2647 2648 /* 2649 * Function to read 32 bit data off the PCI configuration space behind 2650 * the 21554's host interface. 2651 */ 2652 static uint32_t 2653 db_ddi_get32(ddi_acc_impl_t *handle, uint32_t *addr) 2654 { 2655 db_acc_pvt_t *db_pvt = (db_acc_pvt_t *) 2656 handle->ahi_common.ah_bus_private; 2657 uint32_t wait_count = 0; 2658 uint32_t data; 2659 db_ctrl_t *dbp; 2660 2661 dbp = db_pvt->dbp; 2662 2663 mutex_enter(&dbp->db_busown); 2664 2665 if (db_use_config_own_bit) { 2666 /* 2667 * check if (upstream/downstream)configuration address own 2668 * bit set. With this set, we cannot proceed. 2669 */ 2670 while (((ddi_get8(db_pvt->handle, db_pvt->bus_own)) & 2671 db_pvt->mask) == db_pvt->mask) { 2672 #ifdef DEBUG 2673 if (dbp->db_pci_max_wait_count < wait_count) 2674 dbp->db_pci_max_wait_count = wait_count; 2675 #endif 2676 drv_usecwait(db_pci_own_wait); 2677 if (++wait_count == db_pci_max_wait) { 2678 /* 2679 * the man page for pci_config_* routines do 2680 * Not specify any error condition values. 2681 */ 2682 cmn_err(CE_WARN, 2683 "%s#%d: pci config bus own error", 2684 ddi_driver_name(dbp->dip), 2685 ddi_get_instance(dbp->dip)); 2686 dbp->db_pci_err_count++; 2687 mutex_exit(&dbp->db_busown); 2688 return ((uint32_t)DB_CONF_FAILURE); 2689 } 2690 } 2691 wait_count = 0; 2692 } 2693 2694 db_put_reg_conf_addr(db_pvt, (uint32_t)addr); 2695 data = ddi_get32(db_pvt->handle, (uint32_t *)db_pvt->data); 2696 2697 if (db_use_config_own_bit) { 2698 while (((ddi_get8(db_pvt->handle, db_pvt->bus_release)) & 2699 db_pvt->mask) == db_pvt->mask) { 2700 #ifdef DEBUG 2701 if (dbp->db_pci_max_wait_count < wait_count) 2702 dbp->db_pci_max_wait_count = wait_count; 2703 #endif 2704 drv_usecwait(db_pci_release_wait); 2705 if (++wait_count == db_pci_max_wait) { 2706 /* 2707 * the man page for pci_config_* routines do 2708 * not specify any error condition values. 2709 */ 2710 cmn_err(CE_WARN, 2711 "%s#%d: pci config bus release error", 2712 ddi_driver_name(dbp->dip), 2713 ddi_get_instance(dbp->dip)); 2714 dbp->db_pci_err_count++; 2715 mutex_exit(&dbp->db_busown); 2716 return ((uint32_t)DB_CONF_FAILURE); 2717 } 2718 data = ddi_get32(db_pvt->handle, 2719 (uint32_t *)db_pvt->data); 2720 } 2721 } 2722 2723 mutex_exit(&dbp->db_busown); 2724 2725 return (data); 2726 } 2727 2728 /* 2729 * Function to read 64 bit data off the PCI configuration space behind 2730 * the 21554's host interface. 2731 */ 2732 static uint64_t 2733 db_ddi_get64(ddi_acc_impl_t *handle, uint64_t *addr) 2734 { 2735 uint64_t udata, ldata; 2736 2737 ldata = (uint32_t)db_ddi_get32(handle, (uint32_t *)addr); 2738 udata = (uint32_t)db_ddi_get32(handle, (uint32_t *)addr + 1); 2739 return (ldata | (udata << 32)); 2740 } 2741 2742 /* 2743 * Function to write 8 bit data into the PCI configuration space behind 2744 * the 21554's host interface. 2745 */ 2746 static void 2747 db_ddi_put8(ddi_acc_impl_t *handle, uint8_t *addr, uint8_t data) 2748 { 2749 uint32_t rdata; 2750 2751 rdata = db_ddi_get32(handle, (uint32_t *)addr); 2752 db_ddi_put32(handle, (uint32_t *)addr, 2753 db_put_data8((uint32_t)addr, rdata, data)); 2754 } 2755 2756 /* 2757 * Function to write 16 bit data into the PCI configuration space behind 2758 * the 21554's host interface. 2759 */ 2760 static void 2761 db_ddi_put16(ddi_acc_impl_t *handle, uint16_t *addr, uint16_t data) 2762 { 2763 uint32_t rdata; 2764 2765 rdata = db_ddi_get32(handle, (uint32_t *)addr); 2766 db_ddi_put32(handle, (uint32_t *)addr, 2767 db_put_data16((uint32_t)addr, rdata, data)); 2768 } 2769 2770 /* 2771 * Function to write 32 bit data into the PCI configuration space behind 2772 * the 21554's host interface. 2773 */ 2774 static void 2775 db_ddi_put32(ddi_acc_impl_t *handle, uint32_t *addr, uint32_t data) 2776 { 2777 db_acc_pvt_t *db_pvt = (db_acc_pvt_t *) 2778 handle->ahi_common.ah_bus_private; 2779 db_ctrl_t *dbp; 2780 uint32_t wait_count = 0; 2781 2782 dbp = db_pvt->dbp; 2783 2784 mutex_enter(&dbp->db_busown); 2785 2786 if (db_use_config_own_bit) { 2787 /* 2788 * check if (upstream/downstream)configuration address own 2789 * bit set. with this set, we cannot proceed. 2790 */ 2791 while (((ddi_get8(db_pvt->handle, db_pvt->bus_own)) & 2792 db_pvt->mask) == db_pvt->mask) { 2793 #ifdef DEBUG 2794 if (dbp->db_pci_max_wait_count < wait_count) 2795 dbp->db_pci_max_wait_count = wait_count; 2796 #endif 2797 drv_usecwait(db_pci_own_wait); 2798 if (++wait_count == db_pci_max_wait) { 2799 /* 2800 * Since the return value is void here, 2801 * we may need to print a message, as this 2802 * could be a serious situation. 2803 */ 2804 cmn_err(CE_WARN, 2805 "%s#%d: pci config bus own error", 2806 ddi_driver_name(dbp->dip), 2807 ddi_get_instance(dbp->dip)); 2808 dbp->db_pci_err_count++; 2809 mutex_exit(&dbp->db_busown); 2810 return; 2811 } 2812 } 2813 wait_count = 0; 2814 } 2815 2816 db_put_reg_conf_addr(db_pvt, (uint32_t)addr); 2817 ddi_put32(db_pvt->handle, (uint32_t *)db_pvt->data, data); 2818 2819 if (db_use_config_own_bit) { 2820 while (((ddi_get8(db_pvt->handle, db_pvt->bus_release)) & 2821 db_pvt->mask) == db_pvt->mask) { 2822 #ifdef DEBUG 2823 if (dbp->db_pci_max_wait_count < wait_count) 2824 dbp->db_pci_max_wait_count = wait_count; 2825 #endif 2826 drv_usecwait(db_pci_release_wait); 2827 if (++wait_count == db_pci_max_wait) { 2828 /* 2829 * the man page for pci_config_* routines do 2830 * Not specify any error condition values. 2831 */ 2832 cmn_err(CE_WARN, 2833 "%s#%d: pci config bus release error", 2834 ddi_driver_name(dbp->dip), 2835 ddi_get_instance(dbp->dip)); 2836 dbp->db_pci_err_count++; 2837 mutex_exit(&dbp->db_busown); 2838 return; 2839 } 2840 ddi_put32(db_pvt->handle, (uint32_t *)db_pvt->data, 2841 data); 2842 } 2843 } 2844 2845 mutex_exit(&dbp->db_busown); 2846 } 2847 2848 /* 2849 * Function to write 64 bit data into the PCI configuration space behind 2850 * the 21554's host interface. 2851 */ 2852 static void 2853 db_ddi_put64(ddi_acc_impl_t *handle, uint64_t *addr, uint64_t data) 2854 { 2855 db_ddi_put32(handle, (uint32_t *)addr, (uint32_t)(data & 0xffffffff)); 2856 db_ddi_put32(handle, (uint32_t *)addr + 1, (uint32_t)(data >> 32)); 2857 } 2858 2859 /* 2860 * Function to rep read 8 bit data off the PCI configuration space behind 2861 * the 21554's host interface. 2862 */ 2863 static void 2864 db_ddi_rep_get8(ddi_acc_impl_t *handle, uint8_t *host_addr, 2865 uint8_t *dev_addr, size_t repcount, uint_t flags) 2866 { 2867 if (flags == DDI_DEV_AUTOINCR) 2868 for (; repcount; repcount--) 2869 *host_addr++ = db_ddi_get8(handle, dev_addr++); 2870 else 2871 for (; repcount; repcount--) 2872 *host_addr++ = db_ddi_get8(handle, dev_addr); 2873 } 2874 2875 /* 2876 * Function to rep read 16 bit data off the PCI configuration space behind 2877 * the 21554's host interface. 2878 */ 2879 static void 2880 db_ddi_rep_get16(ddi_acc_impl_t *handle, uint16_t *host_addr, 2881 uint16_t *dev_addr, size_t repcount, uint_t flags) 2882 { 2883 if (flags == DDI_DEV_AUTOINCR) 2884 for (; repcount; repcount--) 2885 *host_addr++ = db_ddi_get16(handle, dev_addr++); 2886 else 2887 for (; repcount; repcount--) 2888 *host_addr++ = db_ddi_get16(handle, dev_addr); 2889 } 2890 2891 /* 2892 * Function to rep read 32 bit data off the PCI configuration space behind 2893 * the 21554's host interface. 2894 */ 2895 static void 2896 db_ddi_rep_get32(ddi_acc_impl_t *handle, uint32_t *host_addr, 2897 uint32_t *dev_addr, size_t repcount, uint_t flags) 2898 { 2899 if (flags == DDI_DEV_AUTOINCR) 2900 for (; repcount; repcount--) 2901 *host_addr++ = db_ddi_get32(handle, dev_addr++); 2902 else 2903 for (; repcount; repcount--) 2904 *host_addr++ = db_ddi_get32(handle, dev_addr); 2905 } 2906 2907 /* 2908 * Function to rep read 64 bit data off the PCI configuration space behind 2909 * the 21554's host interface. 2910 */ 2911 static void 2912 db_ddi_rep_get64(ddi_acc_impl_t *handle, uint64_t *host_addr, 2913 uint64_t *dev_addr, size_t repcount, uint_t flags) 2914 { 2915 if (flags == DDI_DEV_AUTOINCR) 2916 for (; repcount; repcount--) 2917 *host_addr++ = db_ddi_get64(handle, dev_addr++); 2918 else 2919 for (; repcount; repcount--) 2920 *host_addr++ = db_ddi_get64(handle, dev_addr); 2921 } 2922 2923 /* 2924 * Function to rep write 8 bit data into the PCI configuration space behind 2925 * the 21554's host interface. 2926 */ 2927 static void 2928 db_ddi_rep_put8(ddi_acc_impl_t *handle, uint8_t *host_addr, 2929 uint8_t *dev_addr, size_t repcount, uint_t flags) 2930 { 2931 if (flags == DDI_DEV_AUTOINCR) 2932 for (; repcount; repcount--) 2933 db_ddi_put8(handle, dev_addr++, *host_addr++); 2934 else 2935 for (; repcount; repcount--) 2936 db_ddi_put8(handle, dev_addr, *host_addr++); 2937 } 2938 2939 /* 2940 * Function to rep write 16 bit data into the PCI configuration space behind 2941 * the 21554's host interface. 2942 */ 2943 static void 2944 db_ddi_rep_put16(ddi_acc_impl_t *handle, uint16_t *host_addr, 2945 uint16_t *dev_addr, size_t repcount, uint_t flags) 2946 { 2947 if (flags == DDI_DEV_AUTOINCR) 2948 for (; repcount; repcount--) 2949 db_ddi_put16(handle, dev_addr++, *host_addr++); 2950 else 2951 for (; repcount; repcount--) 2952 db_ddi_put16(handle, dev_addr, *host_addr++); 2953 } 2954 2955 /* 2956 * Function to rep write 32 bit data into the PCI configuration space behind 2957 * the 21554's host interface. 2958 */ 2959 static void 2960 db_ddi_rep_put32(ddi_acc_impl_t *handle, uint32_t *host_addr, 2961 uint32_t *dev_addr, size_t repcount, uint_t flags) 2962 { 2963 if (flags == DDI_DEV_AUTOINCR) 2964 for (; repcount; repcount--) 2965 db_ddi_put32(handle, dev_addr++, *host_addr++); 2966 else 2967 for (; repcount; repcount--) 2968 db_ddi_put32(handle, dev_addr, *host_addr++); 2969 } 2970 2971 /* 2972 * Function to rep write 64 bit data into the PCI configuration space behind 2973 * the 21554's host interface. 2974 */ 2975 static void 2976 db_ddi_rep_put64(ddi_acc_impl_t *handle, uint64_t *host_addr, 2977 uint64_t *dev_addr, size_t repcount, uint_t flags) 2978 { 2979 if (flags == DDI_DEV_AUTOINCR) 2980 for (; repcount; repcount--) 2981 db_ddi_put64(handle, dev_addr++, *host_addr++); 2982 else 2983 for (; repcount; repcount--) 2984 db_ddi_put64(handle, dev_addr, *host_addr++); 2985 } 2986 2987 #ifdef DEBUG 2988 2989 static void 2990 db_debug(uint64_t func_id, dev_info_t *dip, char *fmt, 2991 uintptr_t a1, uintptr_t a2, uintptr_t a3, uintptr_t a4, uintptr_t a5) 2992 { 2993 char *s = NULL; 2994 uint_t dip_no_disp = 0; 2995 2996 if (func_id & DB_DONT_DISPLAY_DIP) { 2997 dip_no_disp = 1; 2998 } 2999 if (db_debug_funcs & func_id) { 3000 switch (func_id) { 3001 case DB_INIT: s = "_init"; break; 3002 case DB_FINI: s = "_fini"; break; 3003 case DB_INFO: s = "_info"; break; 3004 case DB_GETINFO: s = "getinfo"; break; 3005 case DB_ATTACH: s = "attach"; break; 3006 case DB_DETACH: s = "detach"; break; 3007 case DB_CTLOPS: s = "ctlops"; break; 3008 case DB_INITCHILD: s = "initchild"; break; 3009 case DB_REMOVECHILD: s = "removechild"; break; 3010 case DB_INTR_OPS: s = "intr_ops"; break; 3011 case DB_PCI_MAP: s = "map"; break; 3012 case DB_SAVE_CONF_REGS: s = "save_conf_regs"; break; 3013 case DB_REST_CONF_REGS: s = "restore_conf_regs"; break; 3014 case DB_INTR: s = "intr"; break; 3015 case DB_OPEN: s = "open"; break; 3016 case DB_CLOSE: s = "close"; break; 3017 case DB_IOCTL: s = "ioctl"; break; 3018 case DB_DVMA: s = "set_dvma_range"; break; 3019 3020 default: s = "PCI debug unknown"; break; 3021 } 3022 3023 if (s && !dip_no_disp) { 3024 prom_printf("%s(%d): %s: ", ddi_driver_name(dip), 3025 ddi_get_instance(dip), s); 3026 } 3027 prom_printf(fmt, a1, a2, a3, a4, a5); 3028 } 3029 } 3030 #endif 3031 3032 static int db_prop_op(dev_t dev, dev_info_t *dip, ddi_prop_op_t prop_op, 3033 int flags, char *name, caddr_t valuep, int *lengthp) 3034 { 3035 minor_t minor = getminor(dev); 3036 int instance = PCIHP_AP_MINOR_NUM_TO_INSTANCE(minor); 3037 3038 db_ctrl_t *dbp = (db_ctrl_t *)ddi_get_soft_state(db_state, instance); 3039 3040 3041 if (dbp == NULL) 3042 return (ENXIO); 3043 3044 if (dbp->dev_state & DB_SECONDARY_NEXUS) 3045 return ((pcihp_get_cb_ops())->cb_prop_op(dev, dip, 3046 prop_op, flags, name, valuep, lengthp)); 3047 3048 return (ddi_prop_op(dev, dip, prop_op, flags, name, valuep, lengthp)); 3049 } 3050 3051 /* 3052 * Initialize our FMA resources 3053 */ 3054 static void 3055 db_fm_init(db_ctrl_t *db_p) 3056 { 3057 ddi_fm_error_t derr; 3058 3059 db_p->fm_cap = DDI_FM_EREPORT_CAPABLE | DDI_FM_ERRCB_CAPABLE | 3060 DDI_FM_ACCCHK_CAPABLE | DDI_FM_DMACHK_CAPABLE; 3061 3062 /* 3063 * Request our capability level and get our parents capability 3064 * and ibc. 3065 */ 3066 ddi_fm_init(db_p->dip, &db_p->fm_cap, &db_p->fm_ibc); 3067 ASSERT((db_p->fm_cap & DDI_FM_EREPORT_CAPABLE) && 3068 (db_p->fm_cap & DDI_FM_ERRCB_CAPABLE)); 3069 3070 pci_ereport_setup(db_p->dip); 3071 3072 /* 3073 * clear any outstanding error bits 3074 */ 3075 bzero(&derr, sizeof (ddi_fm_error_t)); 3076 derr.fme_version = DDI_FME_VERSION; 3077 derr.fme_flag = DDI_FM_ERR_EXPECTED; 3078 pci_ereport_post(db_p->dip, &derr, NULL); 3079 pci_bdg_ereport_post(db_p->dip, &derr, NULL); 3080 3081 /* 3082 * Register error callback with our parent. 3083 */ 3084 ddi_fm_handler_register(db_p->dip, db_err_callback, NULL); 3085 } 3086 3087 /* 3088 * Breakdown our FMA resources 3089 */ 3090 static void 3091 db_fm_fini(db_ctrl_t *db_p) 3092 { 3093 /* 3094 * Clean up allocated fm structures 3095 */ 3096 ddi_fm_handler_unregister(db_p->dip); 3097 pci_ereport_teardown(db_p->dip); 3098 ddi_fm_fini(db_p->dip); 3099 } 3100 3101 /* 3102 * Initialize FMA resources for children devices. Called when 3103 * child calls ddi_fm_init(). 3104 */ 3105 /*ARGSUSED*/ 3106 static int 3107 db_fm_init_child(dev_info_t *dip, dev_info_t *tdip, int cap, 3108 ddi_iblock_cookie_t *ibc) 3109 { 3110 db_ctrl_t *db_p = (db_ctrl_t *)ddi_get_soft_state(db_state, 3111 ddi_get_instance(dip)); 3112 *ibc = db_p->fm_ibc; 3113 return (db_p->fm_cap); 3114 } 3115 3116 /* 3117 * FMA registered error callback 3118 */ 3119 static int 3120 db_err_callback(dev_info_t *dip, ddi_fm_error_t *derr, const void *impl_data) 3121 { 3122 uint16_t pci_cfg_stat, pci_cfg_sec_stat; 3123 3124 ASSERT(impl_data == NULL); 3125 pci_ereport_post(dip, derr, &pci_cfg_stat); 3126 pci_bdg_ereport_post(dip, derr, &pci_cfg_sec_stat); 3127 return (pci_bdg_check_status(dip, derr, pci_cfg_stat, 3128 pci_cfg_sec_stat)); 3129 } 3130 3131 static void 3132 db_bus_enter(dev_info_t *dip, ddi_acc_handle_t handle) 3133 { 3134 i_ndi_busop_access_enter(dip, handle); 3135 } 3136 3137 /* ARGSUSED */ 3138 static void 3139 db_bus_exit(dev_info_t *dip, ddi_acc_handle_t handle) 3140 { 3141 i_ndi_busop_access_exit(dip, handle); 3142 } 3143