1 /*- 2 * Copyright (c) 2015 Landon Fuller <landon@landonf.org> 3 * All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions 7 * are met: 8 * 1. Redistributions of source code must retain the above copyright 9 * notice, this list of conditions and the following disclaimer, 10 * without modification. 11 * 2. Redistributions in binary form must reproduce at minimum a disclaimer 12 * similar to the "NO WARRANTY" disclaimer below ("Disclaimer") and any 13 * redistribution must be conditioned upon including a substantially 14 * similar Disclaimer requirement for further binary redistribution. 15 * 16 * NO WARRANTY 17 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 18 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 19 * LIMITED TO, THE IMPLIED WARRANTIES OF NONINFRINGEMENT, MERCHANTIBILITY 20 * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL 21 * THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, 22 * OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 23 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 24 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER 25 * IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 26 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF 27 * THE POSSIBILITY OF SUCH DAMAGES. 28 */ 29 30 #include <sys/cdefs.h> 31 __FBSDID("$FreeBSD$"); 32 33 #include <sys/param.h> 34 #include <sys/bus.h> 35 #include <sys/kernel.h> 36 #include <sys/malloc.h> 37 #include <sys/module.h> 38 #include <sys/systm.h> 39 40 #include <machine/bus.h> 41 42 #include <dev/bhnd/cores/chipc/chipcreg.h> 43 44 #include "sibareg.h" 45 #include "sibavar.h" 46 47 int 48 siba_probe(device_t dev) 49 { 50 device_set_desc(dev, "SIBA BHND bus"); 51 return (BUS_PROBE_DEFAULT); 52 } 53 54 int 55 siba_attach(device_t dev) 56 { 57 struct siba_devinfo *dinfo; 58 struct siba_softc *sc; 59 device_t *devs; 60 int ndevs; 61 int error; 62 63 sc = device_get_softc(dev); 64 sc->dev = dev; 65 66 /* Fetch references to the siba SIBA_CFG* blocks for all 67 * registered devices */ 68 if ((error = device_get_children(dev, &devs, &ndevs))) 69 return (error); 70 71 for (int i = 0; i < ndevs; i++) { 72 struct siba_addrspace *addrspace; 73 74 dinfo = device_get_ivars(devs[i]); 75 76 KASSERT(!device_is_suspended(devs[i]), 77 ("siba(4) stateful suspend handling requires that devices " 78 "not be suspended before siba_attach()")); 79 80 /* Fetch the core register address space */ 81 addrspace = siba_find_addrspace(dinfo, BHND_PORT_DEVICE, 0, 0); 82 if (addrspace == NULL) { 83 device_printf(dev, 84 "missing device registers for core %d\n", i); 85 error = ENXIO; 86 goto cleanup; 87 } 88 89 /* 90 * Map the per-core configuration blocks 91 */ 92 KASSERT(dinfo->core_id.num_cfg_blocks <= SIBA_MAX_CFG, 93 ("config block count %u out of range", 94 dinfo->core_id.num_cfg_blocks)); 95 96 for (u_int cfgidx = 0; cfgidx < dinfo->core_id.num_cfg_blocks; 97 cfgidx++) 98 { 99 rman_res_t r_start, r_count, r_end; 100 101 /* Determine the config block's address range; configuration 102 * blocks are allocated starting at SIBA_CFG0_OFFSET, 103 * growing downwards. */ 104 r_start = addrspace->sa_base + SIBA_CFG0_OFFSET; 105 r_start -= cfgidx * SIBA_CFG_SIZE; 106 107 r_count = SIBA_CFG_SIZE; 108 r_end = r_start + r_count - 1; 109 110 /* Allocate the config resource */ 111 dinfo->cfg_rid[cfgidx] = 0; 112 dinfo->cfg[cfgidx] = BHND_BUS_ALLOC_RESOURCE(dev, dev, 113 SYS_RES_MEMORY, &dinfo->cfg_rid[cfgidx], r_start, 114 r_end, r_count, RF_ACTIVE); 115 116 if (dinfo->cfg[cfgidx] == NULL) { 117 device_printf(dev, "failed allocating CFG_%u for " 118 "core %d\n", cfgidx, i); 119 error = ENXIO; 120 goto cleanup; 121 } 122 } 123 } 124 125 cleanup: 126 free(devs, M_BHND); 127 if (error) 128 return (error); 129 130 /* Delegate remainder to standard bhnd method implementation */ 131 return (bhnd_generic_attach(dev)); 132 } 133 134 int 135 siba_detach(device_t dev) 136 { 137 return (bhnd_generic_detach(dev)); 138 } 139 140 int 141 siba_resume(device_t dev) 142 { 143 return (bhnd_generic_resume(dev)); 144 } 145 146 int 147 siba_suspend(device_t dev) 148 { 149 return (bhnd_generic_suspend(dev)); 150 } 151 152 static int 153 siba_read_ivar(device_t dev, device_t child, int index, uintptr_t *result) 154 { 155 const struct siba_devinfo *dinfo; 156 const struct bhnd_core_info *cfg; 157 158 dinfo = device_get_ivars(child); 159 cfg = &dinfo->core_id.core_info; 160 161 switch (index) { 162 case BHND_IVAR_VENDOR: 163 *result = cfg->vendor; 164 return (0); 165 case BHND_IVAR_DEVICE: 166 *result = cfg->device; 167 return (0); 168 case BHND_IVAR_HWREV: 169 *result = cfg->hwrev; 170 return (0); 171 case BHND_IVAR_DEVICE_CLASS: 172 *result = bhnd_core_class(cfg); 173 return (0); 174 case BHND_IVAR_VENDOR_NAME: 175 *result = (uintptr_t) bhnd_vendor_name(cfg->vendor); 176 return (0); 177 case BHND_IVAR_DEVICE_NAME: 178 *result = (uintptr_t) bhnd_core_name(cfg); 179 return (0); 180 case BHND_IVAR_CORE_INDEX: 181 *result = cfg->core_idx; 182 return (0); 183 case BHND_IVAR_CORE_UNIT: 184 *result = cfg->unit; 185 return (0); 186 default: 187 return (ENOENT); 188 } 189 } 190 191 static int 192 siba_write_ivar(device_t dev, device_t child, int index, uintptr_t value) 193 { 194 switch (index) { 195 case BHND_IVAR_VENDOR: 196 case BHND_IVAR_DEVICE: 197 case BHND_IVAR_HWREV: 198 case BHND_IVAR_DEVICE_CLASS: 199 case BHND_IVAR_VENDOR_NAME: 200 case BHND_IVAR_DEVICE_NAME: 201 case BHND_IVAR_CORE_INDEX: 202 case BHND_IVAR_CORE_UNIT: 203 return (EINVAL); 204 default: 205 return (ENOENT); 206 } 207 } 208 209 static struct resource_list * 210 siba_get_resource_list(device_t dev, device_t child) 211 { 212 struct siba_devinfo *dinfo = device_get_ivars(child); 213 return (&dinfo->resources); 214 } 215 216 static device_t 217 siba_find_hostb_device(device_t dev) 218 { 219 struct siba_softc *sc = device_get_softc(dev); 220 221 /* This is set (or not) by the concrete siba driver subclass. */ 222 return (sc->hostb_dev); 223 } 224 225 static int 226 siba_reset_core(device_t dev, device_t child, uint16_t flags) 227 { 228 struct siba_devinfo *dinfo; 229 230 if (device_get_parent(child) != dev) 231 BHND_BUS_RESET_CORE(device_get_parent(dev), child, flags); 232 233 dinfo = device_get_ivars(child); 234 235 /* Can't reset the core without access to the CFG0 registers */ 236 if (dinfo->cfg[0] == NULL) 237 return (ENODEV); 238 239 // TODO - perform reset 240 241 return (ENXIO); 242 } 243 244 static int 245 siba_suspend_core(device_t dev, device_t child) 246 { 247 struct siba_devinfo *dinfo; 248 249 if (device_get_parent(child) != dev) 250 BHND_BUS_SUSPEND_CORE(device_get_parent(dev), child); 251 252 dinfo = device_get_ivars(child); 253 254 /* Can't suspend the core without access to the CFG0 registers */ 255 if (dinfo->cfg[0] == NULL) 256 return (ENODEV); 257 258 // TODO - perform suspend 259 260 return (ENXIO); 261 } 262 263 static uint32_t 264 siba_read_config(device_t dev, device_t child, bus_size_t offset, u_int width) 265 { 266 struct siba_devinfo *dinfo; 267 rman_res_t r_size; 268 269 /* Must be directly attached */ 270 if (device_get_parent(child) != dev) 271 return (UINT32_MAX); 272 273 /* CFG0 registers must be available */ 274 dinfo = device_get_ivars(child); 275 if (dinfo->cfg[0] == NULL) 276 return (UINT32_MAX); 277 278 /* Offset must fall within CFG0 */ 279 r_size = rman_get_size(dinfo->cfg[0]->res); 280 if (r_size < offset || r_size - offset < width) 281 return (UINT32_MAX); 282 283 switch (width) { 284 case 1: 285 return (bhnd_bus_read_1(dinfo->cfg[0], offset)); 286 case 2: 287 return (bhnd_bus_read_2(dinfo->cfg[0], offset)); 288 case 4: 289 return (bhnd_bus_read_4(dinfo->cfg[0], offset)); 290 } 291 292 /* Unsuported */ 293 return (UINT32_MAX); 294 } 295 296 static void 297 siba_write_config(device_t dev, device_t child, bus_size_t offset, uint32_t val, 298 u_int width) 299 { 300 struct siba_devinfo *dinfo; 301 rman_res_t r_size; 302 303 /* Must be directly attached */ 304 if (device_get_parent(child) != dev) 305 return; 306 307 /* CFG0 registers must be available */ 308 dinfo = device_get_ivars(child); 309 if (dinfo->cfg[0] == NULL) 310 return; 311 312 /* Offset must fall within CFG0 */ 313 r_size = rman_get_size(dinfo->cfg[0]->res); 314 if (r_size < offset || r_size - offset < width) 315 return; 316 317 switch (width) { 318 case 1: 319 bhnd_bus_write_1(dinfo->cfg[0], offset, val); 320 case 2: 321 bhnd_bus_write_2(dinfo->cfg[0], offset, val); 322 case 4: 323 bhnd_bus_write_4(dinfo->cfg[0], offset, val); 324 } 325 } 326 327 static u_int 328 siba_get_port_count(device_t dev, device_t child, bhnd_port_type type) 329 { 330 struct siba_devinfo *dinfo; 331 332 /* delegate non-bus-attached devices to our parent */ 333 if (device_get_parent(child) != dev) 334 return (BHND_BUS_GET_PORT_COUNT(device_get_parent(dev), child, 335 type)); 336 337 dinfo = device_get_ivars(child); 338 return (siba_addrspace_port_count(dinfo)); 339 } 340 341 static u_int 342 siba_get_region_count(device_t dev, device_t child, bhnd_port_type type, 343 u_int port) 344 { 345 struct siba_devinfo *dinfo; 346 347 /* delegate non-bus-attached devices to our parent */ 348 if (device_get_parent(child) != dev) 349 return (BHND_BUS_GET_REGION_COUNT(device_get_parent(dev), child, 350 type, port)); 351 352 dinfo = device_get_ivars(child); 353 if (!siba_is_port_valid(dinfo, type, port)) 354 return (0); 355 356 return (siba_addrspace_region_count(dinfo, port)); 357 } 358 359 static int 360 siba_get_port_rid(device_t dev, device_t child, bhnd_port_type port_type, 361 u_int port_num, u_int region_num) 362 { 363 struct siba_devinfo *dinfo; 364 struct siba_addrspace *addrspace; 365 366 /* delegate non-bus-attached devices to our parent */ 367 if (device_get_parent(child) != dev) 368 return (BHND_BUS_GET_PORT_RID(device_get_parent(dev), child, 369 port_type, port_num, region_num)); 370 371 dinfo = device_get_ivars(child); 372 addrspace = siba_find_addrspace(dinfo, port_type, port_num, region_num); 373 if (addrspace == NULL) 374 return (-1); 375 376 return (addrspace->sa_rid); 377 } 378 379 static int 380 siba_decode_port_rid(device_t dev, device_t child, int type, int rid, 381 bhnd_port_type *port_type, u_int *port_num, u_int *region_num) 382 { 383 struct siba_devinfo *dinfo; 384 385 /* delegate non-bus-attached devices to our parent */ 386 if (device_get_parent(child) != dev) 387 return (BHND_BUS_DECODE_PORT_RID(device_get_parent(dev), child, 388 type, rid, port_type, port_num, region_num)); 389 390 dinfo = device_get_ivars(child); 391 392 /* Ports are always memory mapped */ 393 if (type != SYS_RES_MEMORY) 394 return (EINVAL); 395 396 for (int i = 0; i < dinfo->core_id.num_addrspace; i++) { 397 if (dinfo->addrspace[i].sa_rid != rid) 398 continue; 399 400 *port_type = BHND_PORT_DEVICE; 401 *port_num = siba_addrspace_port(i); 402 *region_num = siba_addrspace_region(i); 403 return (0); 404 } 405 406 /* Not found */ 407 return (ENOENT); 408 } 409 410 static int 411 siba_get_region_addr(device_t dev, device_t child, bhnd_port_type port_type, 412 u_int port_num, u_int region_num, bhnd_addr_t *addr, bhnd_size_t *size) 413 { 414 struct siba_devinfo *dinfo; 415 struct siba_addrspace *addrspace; 416 417 /* delegate non-bus-attached devices to our parent */ 418 if (device_get_parent(child) != dev) { 419 return (BHND_BUS_GET_REGION_ADDR(device_get_parent(dev), child, 420 port_type, port_num, region_num, addr, size)); 421 } 422 423 dinfo = device_get_ivars(child); 424 addrspace = siba_find_addrspace(dinfo, port_type, port_num, region_num); 425 if (addrspace == NULL) 426 return (ENOENT); 427 428 *addr = addrspace->sa_base; 429 *size = addrspace->sa_size - addrspace->sa_bus_reserved; 430 return (0); 431 } 432 433 434 /** 435 * Register all address space mappings for @p di. 436 * 437 * @param dev The siba bus device. 438 * @param di The device info instance on which to register all address 439 * space entries. 440 * @param r A resource mapping the enumeration table block for @p di. 441 */ 442 static int 443 siba_register_addrspaces(device_t dev, struct siba_devinfo *di, 444 struct resource *r) 445 { 446 struct siba_core_id *cid; 447 uint32_t addr; 448 uint32_t size; 449 int error; 450 451 cid = &di->core_id; 452 453 454 /* Register the device address space entries */ 455 for (uint8_t i = 0; i < di->core_id.num_addrspace; i++) { 456 uint32_t adm; 457 u_int adm_offset; 458 uint32_t bus_reserved; 459 460 /* Determine the register offset */ 461 adm_offset = siba_admatch_offset(i); 462 if (adm_offset == 0) { 463 device_printf(dev, "addrspace %hhu is unsupported", i); 464 return (ENODEV); 465 } 466 467 /* Fetch the address match register value */ 468 adm = bus_read_4(r, adm_offset); 469 470 /* Parse the value */ 471 if ((error = siba_parse_admatch(adm, &addr, &size))) { 472 device_printf(dev, "failed to decode address " 473 " match register value 0x%x\n", adm); 474 return (error); 475 } 476 477 /* If this is the device's core/enumeration addrespace, 478 * reserve the Sonics configuration register blocks for the 479 * use of our bus. */ 480 bus_reserved = 0; 481 if (i == SIBA_CORE_ADDRSPACE) 482 bus_reserved = cid->num_cfg_blocks * SIBA_CFG_SIZE; 483 484 /* Append the region info */ 485 error = siba_append_dinfo_region(di, i, addr, size, 486 bus_reserved); 487 if (error) 488 return (error); 489 } 490 491 return (0); 492 } 493 494 static struct bhnd_devinfo * 495 siba_alloc_bhnd_dinfo(device_t dev) 496 { 497 struct siba_devinfo *dinfo = siba_alloc_dinfo(dev); 498 return ((struct bhnd_devinfo *)dinfo); 499 } 500 501 static void 502 siba_free_bhnd_dinfo(device_t dev, struct bhnd_devinfo *dinfo) 503 { 504 siba_free_dinfo(dev, (struct siba_devinfo *)dinfo); 505 } 506 507 508 static int 509 siba_get_core_table(device_t dev, device_t child, struct bhnd_core_info **cores, 510 u_int *num_cores) 511 { 512 const struct bhnd_chipid *chipid; 513 struct bhnd_core_info *table; 514 struct bhnd_resource *r; 515 int error; 516 int rid; 517 518 /* Fetch the core count from our chip identification */ 519 chipid = BHND_BUS_GET_CHIPID(dev, dev); 520 521 /* Allocate our local core table */ 522 table = malloc(sizeof(*table) * chipid->ncores, M_BHND, M_NOWAIT); 523 if (table == NULL) 524 return (ENOMEM); 525 526 /* Enumerate all cores. */ 527 for (u_int i = 0; i < chipid->ncores; i++) { 528 struct siba_core_id cid; 529 uint32_t idhigh, idlow; 530 531 /* Map the core's register block */ 532 rid = 0; 533 r = bhnd_alloc_resource(dev, SYS_RES_MEMORY, &rid, 534 SIBA_CORE_ADDR(i), SIBA_CORE_ADDR(i) + SIBA_CORE_SIZE - 1, 535 SIBA_CORE_SIZE, RF_ACTIVE); 536 if (r == NULL) { 537 error = ENXIO; 538 goto failed; 539 } 540 541 /* Read the core info */ 542 idhigh = bhnd_bus_read_4(r, SB0_REG_ABS(SIBA_CFG0_IDHIGH)); 543 idlow = bhnd_bus_read_4(r, SB0_REG_ABS(SIBA_CFG0_IDLOW)); 544 545 cid = siba_parse_core_id(idhigh, idlow, i, 0); 546 table[i] = cid.core_info; 547 548 /* Determine unit number */ 549 for (u_int j = 0; j < i; j++) { 550 if (table[j].vendor == table[i].vendor && 551 table[j].device == table[i].device) 552 table[i].unit++; 553 } 554 555 /* Release our resource */ 556 bhnd_release_resource(dev, SYS_RES_MEMORY, rid, r); 557 r = NULL; 558 } 559 560 /* Provide the result values (performed last to avoid modifying 561 * cores/num_cores if enumeration failed). */ 562 *cores = table; 563 *num_cores = chipid->ncores; 564 565 return (0); 566 567 failed: 568 if (table != NULL) 569 free(table, M_BHND); 570 571 if (r != NULL) 572 bhnd_release_resource(dev, SYS_RES_MEMORY, rid, r); 573 574 return (error); 575 } 576 577 /** 578 * Scan the core table and add all valid discovered cores to 579 * the bus. 580 * 581 * @param dev The siba bus device. 582 * @param chipid The chip identifier, if the device does not provide a 583 * ChipCommon core. Should o NULL otherwise. 584 */ 585 int 586 siba_add_children(device_t dev, const struct bhnd_chipid *chipid) 587 { 588 struct bhnd_chipid ccid; 589 struct bhnd_core_info *cores; 590 struct siba_devinfo *dinfo; 591 struct resource *r; 592 int rid; 593 int error; 594 595 dinfo = NULL; 596 cores = NULL; 597 r = NULL; 598 599 /* 600 * Try to determine the number of device cores via the ChipCommon 601 * identification registers. 602 * 603 * A small number of very early devices do not include a ChipCommon 604 * core, in which case our caller must supply the chip identification 605 * information via a non-NULL chipid parameter. 606 */ 607 if (chipid == NULL) { 608 uint32_t idhigh, ccreg; 609 uint16_t vendor, device; 610 uint8_t ccrev; 611 612 /* Map the first core's register block. If the ChipCommon core 613 * exists, it will always be the first core. */ 614 rid = 0; 615 r = bus_alloc_resource(dev, SYS_RES_MEMORY, &rid, 616 SIBA_CORE_ADDR(0), SIBA_CORE_SIZE, 617 SIBA_CORE_ADDR(0) + SIBA_CORE_SIZE - 1, 618 RF_ACTIVE); 619 620 /* Identify the core */ 621 idhigh = bus_read_4(r, SB0_REG_ABS(SIBA_CFG0_IDHIGH)); 622 vendor = SIBA_REG_GET(idhigh, IDH_VENDOR); 623 device = SIBA_REG_GET(idhigh, IDH_DEVICE); 624 ccrev = SIBA_IDH_CORE_REV(idhigh); 625 626 if (vendor != OCP_VENDOR_BCM || device != BHND_COREID_CC) { 627 device_printf(dev, 628 "cannot identify device: no chipcommon core " 629 "found\n"); 630 error = ENXIO; 631 goto cleanup; 632 } 633 634 /* Identify the chipset */ 635 ccreg = bus_read_4(r, CHIPC_ID); 636 ccid = bhnd_parse_chipid(ccreg, SIBA_ENUM_ADDR); 637 638 /* Fix up the core count */ 639 error = bhnd_chipid_fixed_ncores(&ccid, ccrev, &ccid.ncores); 640 if (error) { 641 device_printf(dev, "unable to determine core count for " 642 "chipset 0x%hx\n", ccid.chip_id); 643 goto cleanup; 644 } 645 646 chipid = &ccid; 647 bus_release_resource(dev, SYS_RES_MEMORY, rid, r); 648 } 649 650 /* Allocate our temporary core table and enumerate all cores */ 651 cores = malloc(sizeof(*cores) * chipid->ncores, M_BHND, M_NOWAIT); 652 if (cores == NULL) 653 return (ENOMEM); 654 655 /* Add all cores. */ 656 for (u_int i = 0; i < chipid->ncores; i++) { 657 struct siba_core_id cid; 658 device_t child; 659 uint32_t idhigh, idlow; 660 rman_res_t r_count, r_end, r_start; 661 662 /* Map the core's register block */ 663 rid = 0; 664 r_start = SIBA_CORE_ADDR(i); 665 r_count = SIBA_CORE_SIZE; 666 r_end = r_start + SIBA_CORE_SIZE - 1; 667 r = bus_alloc_resource(dev, SYS_RES_MEMORY, &rid, r_start, 668 r_end, r_count, RF_ACTIVE); 669 if (r == NULL) { 670 error = ENXIO; 671 goto cleanup; 672 } 673 674 /* Add the child device */ 675 child = BUS_ADD_CHILD(dev, 0, NULL, -1); 676 if (child == NULL) { 677 error = ENXIO; 678 goto cleanup; 679 } 680 681 /* Read the core info */ 682 idhigh = bus_read_4(r, SB0_REG_ABS(SIBA_CFG0_IDHIGH)); 683 idlow = bus_read_4(r, SB0_REG_ABS(SIBA_CFG0_IDLOW)); 684 685 cid = siba_parse_core_id(idhigh, idlow, i, 0); 686 cores[i] = cid.core_info; 687 688 /* Determine unit number */ 689 for (u_int j = 0; j < i; j++) { 690 if (cores[j].vendor == cores[i].vendor && 691 cores[j].device == cores[i].device) 692 cores[i].unit++; 693 } 694 695 /* Initialize per-device bus info */ 696 if ((dinfo = device_get_ivars(child)) == NULL) { 697 error = ENXIO; 698 goto cleanup; 699 } 700 701 if ((error = siba_init_dinfo(dev, dinfo, &cid))) 702 goto cleanup; 703 704 /* Register the core's address space(s). */ 705 if ((error = siba_register_addrspaces(dev, dinfo, r))) 706 goto cleanup; 707 708 /* If pins are floating or the hardware is otherwise 709 * unpopulated, the device shouldn't be used. */ 710 if (bhnd_is_hw_disabled(child)) 711 device_disable(child); 712 713 /* Release our resource */ 714 bus_release_resource(dev, SYS_RES_MEMORY, rid, r); 715 r = NULL; 716 717 /* Issue bus callback for fully initialized child. */ 718 BHND_BUS_CHILD_ADDED(dev, child); 719 } 720 721 cleanup: 722 if (cores != NULL) 723 free(cores, M_BHND); 724 725 if (r != NULL) 726 bus_release_resource(dev, SYS_RES_MEMORY, rid, r); 727 728 return (error); 729 } 730 731 static device_method_t siba_methods[] = { 732 /* Device interface */ 733 DEVMETHOD(device_probe, siba_probe), 734 DEVMETHOD(device_attach, siba_attach), 735 DEVMETHOD(device_detach, siba_detach), 736 DEVMETHOD(device_resume, siba_resume), 737 DEVMETHOD(device_suspend, siba_suspend), 738 739 /* Bus interface */ 740 DEVMETHOD(bus_read_ivar, siba_read_ivar), 741 DEVMETHOD(bus_write_ivar, siba_write_ivar), 742 DEVMETHOD(bus_get_resource_list, siba_get_resource_list), 743 744 /* BHND interface */ 745 DEVMETHOD(bhnd_bus_find_hostb_device, siba_find_hostb_device), 746 DEVMETHOD(bhnd_bus_get_core_table, siba_get_core_table), 747 DEVMETHOD(bhnd_bus_alloc_devinfo, siba_alloc_bhnd_dinfo), 748 DEVMETHOD(bhnd_bus_free_devinfo, siba_free_bhnd_dinfo), 749 DEVMETHOD(bhnd_bus_reset_core, siba_reset_core), 750 DEVMETHOD(bhnd_bus_suspend_core, siba_suspend_core), 751 DEVMETHOD(bhnd_bus_read_config, siba_read_config), 752 DEVMETHOD(bhnd_bus_write_config, siba_write_config), 753 DEVMETHOD(bhnd_bus_get_port_count, siba_get_port_count), 754 DEVMETHOD(bhnd_bus_get_region_count, siba_get_region_count), 755 DEVMETHOD(bhnd_bus_get_port_rid, siba_get_port_rid), 756 DEVMETHOD(bhnd_bus_decode_port_rid, siba_decode_port_rid), 757 DEVMETHOD(bhnd_bus_get_region_addr, siba_get_region_addr), 758 759 DEVMETHOD_END 760 }; 761 762 DEFINE_CLASS_1(bhnd, siba_driver, siba_methods, sizeof(struct siba_softc), bhnd_driver); 763 764 MODULE_VERSION(siba, 1); 765 MODULE_DEPEND(siba, bhnd, 1, 1, 1); 766