1 /* 2 * Copyright (c) 2000,2001 Jonathan Chen. 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, immediately at the beginning of the file. 11 * 2. Redistributions in binary form must reproduce the above copyright 12 * notice, this list of conditions and the following disclaimer in 13 * the documentation and/or other materials provided with the 14 * distribution. 15 * 16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 17 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 19 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR 20 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 21 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 22 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 23 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 24 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 25 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 26 * SUCH DAMAGE. 27 * 28 * $FreeBSD$ 29 */ 30 31 /* 32 * Driver for PCI to Cardbus Bridge chips 33 * 34 * References: 35 * TI Datasheets: 36 * http://www-s.ti.com/cgi-bin/sc/generic2.cgi?family=PCI+CARDBUS+CONTROLLERS 37 * Much of the 16-bit PC Card compatibility code stolen from dev/pcic/i82365.c 38 * XXX and should be cleaned up to share as much as possible. 39 * 40 * Written by Jonathan Chen <jon@freebsd.org> 41 * The author would like to acknowledge: 42 * * HAYAKAWA Koichi: Author of the NetBSD code for the same thing 43 * * Warner Losh: Newbus/newcard guru and author of the pccard side of things 44 * * YAMAMOTO Shigeru: Author of another FreeBSD cardbus driver 45 * * David Cross: Author of the initial ugly hack for a specific cardbus card 46 */ 47 48 #define CBB_DEBUG 49 50 #include <sys/param.h> 51 #include <sys/systm.h> 52 #include <sys/errno.h> 53 #include <sys/kernel.h> 54 #include <sys/kthread.h> 55 #include <sys/lock.h> 56 #include <sys/malloc.h> 57 #include <sys/mutex.h> 58 59 #include <sys/bus.h> 60 #include <machine/bus.h> 61 #include <sys/rman.h> 62 #include <machine/resource.h> 63 64 #include <pci/pcireg.h> 65 #include <pci/pcivar.h> 66 #include <machine/clock.h> 67 68 #include <dev/pccard/pccardreg.h> 69 #include <dev/pccard/pccardvar.h> 70 #include <dev/pcic/i82365reg.h> 71 72 #include <dev/pccbb/pccbbreg.h> 73 #include <dev/pccbb/pccbbvar.h> 74 75 #include "power_if.h" 76 #include "card_if.h" 77 #include "pcib_if.h" 78 79 #if defined CBB_DEBUG 80 #define DPRINTF(x) printf x 81 #define DEVPRINTF(x) device_printf x 82 #else 83 #define DPRINTF(x) 84 #define DEVPRINTF(x) 85 #endif 86 87 #define PCI_MASK_CONFIG(DEV,REG,MASK,SIZE) \ 88 pci_write_config(DEV, REG, pci_read_config(DEV, REG, SIZE) MASK, SIZE) 89 #define PCI_MASK2_CONFIG(DEV,REG,MASK1,MASK2,SIZE) \ 90 pci_write_config(DEV, REG, ( \ 91 pci_read_config(DEV, REG, SIZE) MASK1) MASK2, SIZE) 92 93 /* 94 * XXX all the pcic code really doesn't belong here and needs to be 95 * XXX migrated to its own file, shared with the 16-bit code 96 */ 97 #define PCIC_READ(SC,REG) \ 98 (((u_int8_t*)((SC)->sc_socketreg))[0x800+(REG)]) 99 #define PCIC_WRITE(SC,REG,val) \ 100 (((u_int8_t*)((SC)->sc_socketreg))[0x800+(REG)]) = (val) 101 #define PCIC_MASK(SC,REG,MASK) \ 102 PCIC_WRITE(SC,REG,PCIC_READ(SC,REG) MASK) 103 #define PCIC_MASK2(SC,REG,MASK,MASK2) \ 104 PCIC_WRITE(SC,REG,(PCIC_READ(SC,REG) MASK) MASK2) 105 106 struct pccbb_sclist { 107 struct pccbb_softc *sc; 108 STAILQ_ENTRY(pccbb_sclist) entries; 109 }; 110 111 static STAILQ_HEAD(, pccbb_sclist) softcs; 112 static int softcs_init = 0; 113 114 115 struct yenta_chipinfo { 116 u_int32_t yc_id; 117 const char *yc_name; 118 int yc_chiptype; 119 int yc_flags; 120 } yc_chipsets[] = { 121 /* Texas Instruments chips */ 122 {PCI_DEVICE_ID_PCIC_TI1130, "TI1130 PCI-CardBus Bridge", CB_TI113X, 123 PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32}, 124 {PCI_DEVICE_ID_PCIC_TI1131, "TI1131 PCI-CardBus Bridge", CB_TI113X, 125 PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32}, 126 127 {PCI_DEVICE_ID_PCIC_TI1211, "TI1211 PCI-CardBus Bridge", CB_TI12XX, 128 PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32}, 129 {PCI_DEVICE_ID_PCIC_TI1220, "TI1220 PCI-CardBus Bridge", CB_TI12XX, 130 PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32}, 131 {PCI_DEVICE_ID_PCIC_TI1221, "TI1221 PCI-CardBus Bridge", CB_TI12XX, 132 PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32}, 133 {PCI_DEVICE_ID_PCIC_TI1225, "TI1225 PCI-CardBus Bridge", CB_TI12XX, 134 PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32}, 135 {PCI_DEVICE_ID_PCIC_TI1250, "TI1250 PCI-CardBus Bridge", CB_TI12XX, 136 PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32}, 137 {PCI_DEVICE_ID_PCIC_TI1251, "TI1251 PCI-CardBus Bridge", CB_TI12XX, 138 PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32}, 139 {PCI_DEVICE_ID_PCIC_TI1251B,"TI1251B PCI-CardBus Bridge",CB_TI12XX, 140 PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32}, 141 {PCI_DEVICE_ID_PCIC_TI1410, "TI1410 PCI-CardBus Bridge", CB_TI12XX, 142 PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32}, 143 {PCI_DEVICE_ID_PCIC_TI1420, "TI1420 PCI-CardBus Bridge", CB_TI12XX, 144 PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32}, 145 {PCI_DEVICE_ID_PCIC_TI1450, "TI1450 PCI-CardBus Bridge", CB_TI12XX, 146 PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32}, 147 {PCI_DEVICE_ID_PCIC_TI1451, "TI1451 PCI-CardBus Bridge", CB_TI12XX, 148 PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32}, 149 {PCI_DEVICE_ID_PCIC_TI4451, "TI4451 PCI-CardBus Bridge", CB_TI12XX, 150 PCCBB_PCIC_IO_RELOC | PCCBB_PCIC_MEM_32}, 151 152 /* Ricoh chips */ 153 {PCI_DEVICE_ID_RICOH_RL5C465, "RF5C465 PCI-CardBus Bridge", 154 CB_RF5C46X, PCCBB_PCIC_MEM_32}, 155 {PCI_DEVICE_ID_RICOH_RL5C466, "RF5C466 PCI-CardBus Bridge", 156 CB_RF5C46X, PCCBB_PCIC_MEM_32}, 157 {PCI_DEVICE_ID_RICOH_RL5C475, "RF5C475 PCI-CardBus Bridge", 158 CB_RF5C47X, PCCBB_PCIC_MEM_32}, 159 {PCI_DEVICE_ID_RICOH_RL5C476, "RF5C476 PCI-CardBus Bridge", 160 CB_RF5C47X, PCCBB_PCIC_MEM_32}, 161 {PCI_DEVICE_ID_RICOH_RL5C478, "RF5C478 PCI-CardBus Bridge", 162 CB_RF5C47X, PCCBB_PCIC_MEM_32}, 163 164 /* Toshiba products */ 165 {PCI_DEVICE_ID_TOSHIBA_TOPIC95, "ToPIC95 PCI-CardBus Bridge", 166 CB_TOPIC95, PCCBB_PCIC_MEM_32}, 167 {PCI_DEVICE_ID_TOSHIBA_TOPIC95B, "ToPIC95B PCI-CardBus Bridge", 168 CB_TOPIC95B, PCCBB_PCIC_MEM_32}, 169 {PCI_DEVICE_ID_TOSHIBA_TOPIC97, "ToPIC97 PCI-CardBus Bridge", 170 CB_TOPIC97, PCCBB_PCIC_MEM_32}, 171 {PCI_DEVICE_ID_TOSHIBA_TOPIC100, "ToPIC100 PCI-CardBus Bridge", 172 CB_TOPIC97, PCCBB_PCIC_MEM_32}, 173 174 /* Cirrus Logic */ 175 {PCI_DEVICE_ID_PCIC_CLPD6832, "CLPD6832 PCI-CardBus Bridge", 176 CB_CIRRUS, PCCBB_PCIC_MEM_32}, 177 {PCI_DEVICE_ID_PCIC_CLPD6833, "CLPD6833 PCI-CardBus Bridge", 178 CB_CIRRUS, PCCBB_PCIC_MEM_32}, 179 180 /* 02Micro */ 181 {PCI_DEVICE_ID_PCIC_OZ6832, "O2Mirco OZ6832/6833 PCI-CardBus Bridge", 182 CB_CIRRUS, PCCBB_PCIC_MEM_32}, 183 {PCI_DEVICE_ID_PCIC_OZ6860, "O2Mirco OZ6836/6860 PCI-CardBus Bridge", 184 CB_CIRRUS, PCCBB_PCIC_MEM_32}, 185 {PCI_DEVICE_ID_PCIC_OZ6872, "O2Mirco OZ6812/6872 PCI-CardBus Bridge", 186 CB_CIRRUS, PCCBB_PCIC_MEM_32}, 187 188 /* sentinel */ 189 {0 /* null id */, "unknown", 190 CB_UNKNOWN, 0}, 191 }; 192 193 194 static int cb_chipset(u_int32_t pci_id, const char** namep, int* flagp); 195 static int pccbb_probe(device_t dev); 196 static void pccbb_chipinit(struct pccbb_softc* sc); 197 static int pccbb_attach(device_t dev); 198 static int pccbb_detach(device_t dev); 199 static void pccbb_driver_added(device_t dev, driver_t *driver); 200 static void pccbb_child_detached(device_t dev, device_t child); 201 static void pccbb_event_thread (void *arg); 202 static void pccbb_create_event_thread (struct pccbb_softc *sc); 203 static void pccbb_start_threads(void *arg); 204 static void pccbb_insert (struct pccbb_softc *sc); 205 static void pccbb_removal (struct pccbb_softc *sc); 206 static void pccbb_intr(void* arg); 207 static int pccbb_detect_voltage(device_t dev); 208 static int pccbb_power(device_t dev, int volts); 209 static void pccbb_cardbus_reset(device_t dev); 210 static int pccbb_cardbus_power_enable_socket(device_t self, device_t child); 211 static void pccbb_cardbus_power_disable_socket(device_t self, device_t child); 212 static int pccbb_cardbus_io_open(device_t dev, int win, 213 u_int32_t start, u_int32_t end); 214 static int pccbb_cardbus_mem_open(device_t dev, int win, 215 u_int32_t start, u_int32_t end); 216 static void pccbb_cardbus_auto_open(struct pccbb_softc *sc, int type); 217 static int pccbb_cardbus_activate_resource(device_t self, device_t child, 218 int type, int rid, 219 struct resource *r); 220 static int pccbb_cardbus_deactivate_resource(device_t self, device_t child, 221 int type, int rid, 222 struct resource *r); 223 static struct resource* pccbb_cardbus_alloc_resource(device_t self, 224 device_t child, int type, int* rid, 225 u_long start, u_long end, u_long count, 226 u_int flags); 227 static int pccbb_cardbus_release_resource(device_t self, device_t child, 228 int type,int rid, 229 struct resource *r); 230 static int pccbb_pcic_power_enable_socket(device_t self, device_t child); 231 static void pccbb_pcic_power_disable_socket(device_t self, device_t child); 232 static void pccbb_pcic_wait_ready(struct pccbb_softc *sc); 233 static void pccbb_pcic_do_mem_map(struct pccbb_softc *sc, int win); 234 static int pccbb_pcic_mem_map(struct pccbb_softc *sc, int kind, 235 struct resource *r, bus_addr_t card_addr, 236 int *win); 237 static void pccbb_pcic_mem_unmap(struct pccbb_softc *sc, int window); 238 static void pccbb_pcic_do_io_map(struct pccbb_softc *sc, int win); 239 static int pccbb_pcic_io_map(struct pccbb_softc *sc, int width, 240 struct resource *r, bus_addr_t card_addr, 241 int *win); 242 static void pccbb_pcic_io_unmap(struct pccbb_softc *sc, int window); 243 static int pccbb_pcic_activate_resource(device_t self, device_t child, 244 int type, int rid, struct resource *r); 245 static int pccbb_pcic_deactivate_resource(device_t self, device_t child, 246 int type,int rid, struct resource *r); 247 static struct resource* pccbb_pcic_alloc_resource(device_t self,device_t child, 248 int type, int* rid, u_long start, 249 u_long end, u_long count, u_int flags); 250 static int pccbb_pcic_release_resource(device_t self, device_t child, int type, 251 int rid, struct resource *res); 252 static int pccbb_pcic_set_res_flags(device_t self, device_t child, int type, 253 int rid, u_int32_t flags); 254 static int pccbb_pcic_set_memory_offset(device_t self, device_t child, int rid, 255 u_int32_t offset, u_int32_t *deltap); 256 static int pccbb_power_enable_socket(device_t self, device_t child); 257 static void pccbb_power_disable_socket(device_t self, device_t child); 258 static int pccbb_activate_resource(device_t self, device_t child, int type, 259 int rid, struct resource *r); 260 static int pccbb_deactivate_resource(device_t self, device_t child, int type, 261 int rid, struct resource *r); 262 static struct resource* pccbb_alloc_resource(device_t self, device_t child, 263 int type, int* rid, u_long start, 264 u_long end, u_long count, 265 u_int flags); 266 static int pccbb_release_resource(device_t self, device_t child, 267 int type, int rid, struct resource *r); 268 static int pccbb_maxslots(device_t dev); 269 static u_int32_t pccbb_read_config(device_t dev, int b, int s, int f, 270 int reg, int width); 271 static void pccbb_write_config(device_t dev, int b, int s, int f, int reg, 272 u_int32_t val, int width); 273 274 275 /************************************************************************/ 276 /* Probe/Attach */ 277 /************************************************************************/ 278 279 static int 280 cb_chipset(u_int32_t pci_id, const char** namep, int* flagp) 281 { 282 int loopend = sizeof(yc_chipsets)/sizeof(yc_chipsets[0]); 283 struct yenta_chipinfo *ycp, *ycend; 284 ycend = yc_chipsets + loopend; 285 286 for (ycp = yc_chipsets; ycp < ycend && pci_id != ycp->yc_id; ++ycp); 287 if (ycp == ycend) { 288 /* not found */ 289 ycp = yc_chipsets + loopend - 1; /* to point the sentinel */ 290 } 291 if (namep != NULL) { 292 *namep = ycp->yc_name; 293 } 294 if (flagp != NULL) { 295 *flagp = ycp->yc_flags; 296 } 297 return ycp->yc_chiptype; 298 } 299 300 static int 301 pccbb_probe(device_t dev) 302 { 303 const char *name; 304 305 if (cb_chipset(pci_get_devid(dev), &name, NULL) == CB_UNKNOWN) 306 return ENXIO; 307 device_set_desc(dev, name); 308 return 0; 309 } 310 311 static void 312 pccbb_chipinit(struct pccbb_softc* sc) 313 { 314 /* Set CardBus latency timer */ 315 if (pci_read_config(sc->sc_dev, PCIR_SECLAT_1, 1) < 0x20) 316 pci_write_config(sc->sc_dev, PCIR_SECLAT_1, 0x20, 1); 317 318 /* Set PCI latency timer */ 319 if (pci_read_config(sc->sc_dev, PCIR_LATTIMER, 1) < 0x20) 320 pci_write_config(sc->sc_dev, PCIR_LATTIMER, 0x20, 1); 321 322 /* Enable memory access */ 323 PCI_MASK_CONFIG(sc->sc_dev, PCIR_COMMAND, 324 | PCIM_CMD_MEMEN 325 | PCIM_CMD_PORTEN 326 | PCIM_CMD_BUSMASTEREN, 2); 327 328 /* disable Legacy IO */ 329 switch (sc->sc_chipset) { 330 case CB_RF5C46X: 331 case CB_RF5C47X: 332 PCI_MASK_CONFIG(sc->sc_dev, PCCBBR_BRIDGECTRL, 333 & ~(PCCBBM_BRIDGECTRL_RL_3E0_EN| 334 PCCBBM_BRIDGECTRL_RL_3E2_EN), 2); 335 break; 336 default: 337 pci_write_config(sc->sc_dev, PCCBBR_LEGACY, 0x0, 4); 338 break; 339 } 340 341 /* Use PCI interrupt for interrupt routing */ 342 PCI_MASK2_CONFIG(sc->sc_dev, PCCBBR_BRIDGECTRL, 343 & ~(PCCBBM_BRIDGECTRL_MASTER_ABORT | 344 PCCBBM_BRIDGECTRL_INTR_IREQ_EN), 345 | PCCBBM_BRIDGECTRL_WRITE_POST_EN, 346 2); 347 348 switch (sc->sc_chipset) { 349 case CB_TI113X: 350 PCI_MASK2_CONFIG(sc->sc_dev, PCCBBR_CBCTRL, 351 & ~PCCBBM_CBCTRL_113X_PCI_INTR, 352 | PCCBBM_CBCTRL_113X_PCI_CSC 353 | PCCBBM_CBCTRL_113X_PCI_IRQ_EN, 1); 354 PCI_MASK_CONFIG(sc->sc_dev, PCCBBR_DEVCTRL, 355 & ~(PCCBBM_DEVCTRL_INT_SERIAL| 356 PCCBBM_DEVCTRL_INT_PCI), 1); 357 PCIC_WRITE(sc, PCIC_INTR, PCIC_INTR_ENABLE); 358 PCIC_WRITE(sc, PCIC_CSC_INTR, 0); 359 break; 360 case CB_TI12XX: 361 PCIC_WRITE(sc, PCIC_INTR, PCIC_INTR_ENABLE); 362 PCIC_WRITE(sc, PCIC_CSC_INTR, 0); 363 break; 364 case CB_TOPIC95B: 365 PCI_MASK_CONFIG(sc->sc_dev, PCCBBR_TOPIC_SOCKETCTRL, 366 | PCCBBM_TOPIC_SOCKETCTRL_SCR_IRQSEL, 4); 367 PCI_MASK2_CONFIG(sc->sc_dev, PCCBBR_TOPIC_SLOTCTRL, 368 | PCCBBM_TOPIC_SLOTCTRL_SLOTON 369 | PCCBBM_TOPIC_SLOTCTRL_SLOTEN 370 | PCCBBM_TOPIC_SLOTCTRL_ID_LOCK 371 | PCCBBM_TOPIC_SLOTCTRL_CARDBUS, 372 & ~PCCBBM_TOPIC_SLOTCTRL_SWDETECT, 4); 373 break; 374 } 375 376 /* close all memory and io windows */ 377 pci_write_config(sc->sc_dev, PCCBBR_MEMBASE0, 0xffffffff, 4); 378 pci_write_config(sc->sc_dev, PCCBBR_MEMLIMIT0, 0, 4); 379 pci_write_config(sc->sc_dev, PCCBBR_MEMBASE1, 0xffffffff, 4); 380 pci_write_config(sc->sc_dev, PCCBBR_MEMLIMIT1, 0, 4); 381 pci_write_config(sc->sc_dev, PCCBBR_IOBASE0, 0xffffffff, 4); 382 pci_write_config(sc->sc_dev, PCCBBR_IOLIMIT0, 0, 4); 383 pci_write_config(sc->sc_dev, PCCBBR_IOBASE1, 0xffffffff, 4); 384 pci_write_config(sc->sc_dev, PCCBBR_IOLIMIT1, 0, 4); 385 } 386 387 static int 388 pccbb_attach(device_t dev) 389 { 390 struct pccbb_softc *sc = (struct pccbb_softc *)device_get_softc(dev); 391 int rid; 392 u_int32_t tmp; 393 394 if (!softcs_init) { 395 softcs_init = 1; 396 STAILQ_INIT(&softcs); 397 } 398 mtx_init(&sc->sc_mtx, device_get_nameunit(dev), MTX_DEF); 399 sc->sc_chipset = cb_chipset(pci_get_devid(dev), NULL, &sc->sc_flags); 400 sc->sc_dev = dev; 401 sc->sc_cbdev = NULL; 402 sc->sc_pccarddev = NULL; 403 sc->sc_secbus = pci_read_config(dev, PCIR_SECBUS_2, 1); 404 sc->sc_subbus = pci_read_config(dev, PCIR_SUBBUS_2, 1); 405 sc->memalloc = 0; 406 sc->ioalloc = 0; 407 SLIST_INIT(&sc->rl); 408 409 /* Ths PCI bus should have given me memory... right? */ 410 rid=PCCBBR_SOCKBASE; 411 sc->sc_base_res=bus_alloc_resource(dev, SYS_RES_MEMORY, &rid, 412 0,~0,1, RF_ACTIVE); 413 if (!sc->sc_base_res) { 414 /* 415 * XXX eVILE HACK BAD THING! XXX 416 * The pci bus device should do this for us. 417 * Some BIOSes doesn't assign a memory space properly. 418 * So we try to manually put one in... 419 */ 420 u_int32_t sockbase; 421 422 sockbase = pci_read_config(dev, rid, 4); 423 if (sockbase < 0x100000 || sockbase >= 0xfffffff0) { 424 pci_write_config(dev, rid, 0xffffffff, 4); 425 sockbase = pci_read_config(dev, rid, 4); 426 sockbase = (sockbase & 0xfffffff0) & 427 -(sockbase & 0xfffffff0); 428 sc->sc_base_res = bus_generic_alloc_resource( 429 device_get_parent(dev), dev, SYS_RES_MEMORY, 430 &rid, CARDBUS_SYS_RES_MEMORY_START, 431 CARDBUS_SYS_RES_MEMORY_END, sockbase, 432 RF_ACTIVE|rman_make_alignment_flags(sockbase)); 433 if (!sc->sc_base_res){ 434 device_printf(dev, 435 "Could not grab register memory\n"); 436 mtx_destroy(&sc->sc_mtx); 437 return ENOMEM; 438 } 439 pci_write_config(dev, PCCBBR_SOCKBASE, 440 rman_get_start(sc->sc_base_res), 4); 441 DEVPRINTF((dev, "PCI Memory allocated: %08lx\n", 442 rman_get_start(sc->sc_base_res))); 443 } else { 444 device_printf(dev, "Could not map register memory\n"); 445 mtx_destroy(&sc->sc_mtx); 446 return ENOMEM; 447 } 448 } 449 450 sc->sc_socketreg = 451 (struct pccbb_socketreg *)rman_get_virtual(sc->sc_base_res); 452 pccbb_chipinit(sc); 453 454 /* CSC Interrupt: Card detect interrupt on */ 455 sc->sc_socketreg->socket_mask |= PCCBB_SOCKET_MASK_CD; 456 457 /* reset interrupt */ 458 tmp = sc->sc_socketreg->socket_event; 459 sc->sc_socketreg->socket_event = tmp; 460 461 /* Map and establish the interrupt. */ 462 rid=0; 463 sc->sc_irq_res=bus_alloc_resource(dev, SYS_RES_IRQ, &rid, 0, ~0, 1, 464 RF_SHAREABLE | RF_ACTIVE); 465 if (sc->sc_irq_res == NULL) { 466 printf("pccbb: Unable to map IRQ...\n"); 467 bus_release_resource(dev, SYS_RES_MEMORY, PCCBBR_SOCKBASE, 468 sc->sc_base_res); 469 mtx_destroy(&sc->sc_mtx); 470 return ENOMEM; 471 } 472 473 if (bus_setup_intr(dev, sc->sc_irq_res, INTR_TYPE_BIO, pccbb_intr, sc, 474 &(sc->sc_intrhand))) { 475 device_printf(dev, "couldn't establish interrupt"); 476 bus_release_resource(dev, SYS_RES_IRQ, 0, sc->sc_irq_res); 477 bus_release_resource(dev, SYS_RES_MEMORY, PCCBBR_SOCKBASE, 478 sc->sc_base_res); 479 mtx_destroy(&sc->sc_mtx); 480 return ENOMEM; 481 } 482 483 /* attach children */ 484 sc->sc_cbdev = device_add_child(dev, "cardbus", -1); 485 if (sc->sc_cbdev == NULL) 486 DEVPRINTF((dev, "WARNING: cannot add cardbus bus.\n")); 487 else if (device_probe_and_attach(sc->sc_cbdev) != 0) { 488 DEVPRINTF((dev, "WARNING: cannot attach cardbus bus!\n")); 489 sc->sc_cbdev = NULL; 490 } 491 492 sc->sc_pccarddev = device_add_child(dev, "pccard", -1); 493 if (sc->sc_pccarddev == NULL) 494 DEVPRINTF((dev, "WARNING: cannot add pccard bus.\n")); 495 else if (device_probe_and_attach(sc->sc_pccarddev) != 0) { 496 DEVPRINTF((dev, "WARNING: cannot attach pccard bus.\n")); 497 sc->sc_pccarddev = NULL; 498 } 499 500 #ifndef KLD_MODULE 501 if (sc->sc_cbdev == NULL && sc->sc_pccarddev == NULL) { 502 device_printf(dev, "ERROR: Failed to attach cardbus/pccard bus!\n"); 503 bus_teardown_intr(dev, sc->sc_irq_res, sc->sc_intrhand); 504 bus_release_resource(dev, SYS_RES_IRQ, 0, sc->sc_irq_res); 505 bus_release_resource(dev, SYS_RES_MEMORY, PCCBBR_SOCKBASE, 506 sc->sc_base_res); 507 mtx_destroy(&sc->sc_mtx); 508 return ENOMEM; 509 } 510 #endif 511 512 { 513 struct pccbb_sclist *sclist; 514 sclist = malloc(sizeof(struct pccbb_sclist), M_DEVBUF, 515 M_WAITOK); 516 sclist->sc = sc; 517 STAILQ_INSERT_TAIL(&softcs, sclist, entries); 518 } 519 return 0; 520 } 521 522 static int 523 pccbb_detach(device_t dev) 524 { 525 struct pccbb_softc *sc = device_get_softc(dev); 526 int numdevs; 527 device_t *devlist; 528 int tmp; 529 int error; 530 531 device_get_children(dev, &devlist, &numdevs); 532 533 error = 0; 534 for (tmp = 0; tmp < numdevs; tmp++) { 535 if (device_detach(devlist[tmp]) == 0) 536 device_delete_child(dev, devlist[tmp]); 537 else 538 error++; 539 } 540 free(devlist, M_TEMP); 541 if (error > 0) 542 return ENXIO; 543 544 mtx_lock(&sc->sc_mtx); 545 bus_teardown_intr(dev, sc->sc_irq_res, sc->sc_intrhand); 546 547 sc->sc_flags |= PCCBB_KTHREAD_DONE; 548 if (sc->sc_flags & PCCBB_KTHREAD_RUNNING) { 549 wakeup(sc); 550 mtx_unlock(&sc->sc_mtx); 551 DEVPRINTF((dev, "waiting for kthread exit...")); 552 error = tsleep(sc, PWAIT, "pccbb-detach-wait", 60 * hz); 553 if (error) 554 DPRINTF(("timeout\n")); 555 else 556 DPRINTF(("done\n")); 557 } else 558 mtx_unlock(&sc->sc_mtx); 559 560 bus_release_resource(dev, SYS_RES_IRQ, 0, sc->sc_irq_res); 561 bus_release_resource(dev, SYS_RES_MEMORY, PCCBBR_SOCKBASE, 562 sc->sc_base_res); 563 mtx_destroy(&sc->sc_mtx); 564 return 0; 565 } 566 567 static void 568 pccbb_driver_added(device_t dev, driver_t *driver) 569 { 570 struct pccbb_softc *sc = device_get_softc(dev); 571 device_t *devlist; 572 int tmp; 573 int numdevs; 574 575 DEVICE_IDENTIFY(driver, dev); 576 device_get_children(dev, &devlist, &numdevs); 577 for (tmp = 0; tmp < numdevs; tmp++) { 578 if (device_get_state(devlist[tmp]) == DS_NOTPRESENT && 579 device_probe_and_attach(devlist[tmp]) == 0) { 580 if (devlist[tmp] == NULL) 581 /* NOTHING */; 582 else if (strcmp(driver->name, "cardbus") == 0) { 583 sc->sc_cbdev = devlist[tmp]; 584 if ((sc->sc_socketreg->socket_state 585 & PCCBB_SOCKET_STAT_CD) == 0) { 586 mtx_lock(&sc->sc_mtx); 587 wakeup(sc); 588 mtx_unlock(&sc->sc_mtx); 589 } 590 } else if (strcmp(driver->name, "pccard") == 0) { 591 sc->sc_pccarddev = devlist[tmp]; 592 if ((sc->sc_socketreg->socket_state 593 & PCCBB_SOCKET_STAT_CD) == 0) { 594 mtx_lock(&sc->sc_mtx); 595 wakeup(sc); 596 mtx_unlock(&sc->sc_mtx); 597 } 598 } else 599 device_printf(dev, 600 "Unsupported child bus: %s\n", 601 driver->name); 602 } 603 } 604 free(devlist, M_TEMP); 605 } 606 607 static void 608 pccbb_child_detached(device_t dev, device_t child) 609 { 610 struct pccbb_softc *sc = device_get_softc(dev); 611 if (child == sc->sc_cbdev) 612 sc->sc_cbdev = NULL; 613 else if (child == sc->sc_pccarddev) 614 sc->sc_pccarddev = NULL; 615 else 616 device_printf(dev, "Unknown child detached: %s %p/%p\n", 617 device_get_nameunit(child), sc->sc_cbdev, sc->sc_pccarddev); 618 } 619 620 /************************************************************************/ 621 /* Kthreads */ 622 /************************************************************************/ 623 624 static void 625 pccbb_event_thread (void *arg) 626 { 627 struct pccbb_softc *sc = arg; 628 u_int32_t status; 629 630 mtx_lock(&Giant); 631 for(;;) { 632 if (!(sc->sc_flags & PCCBB_KTHREAD_RUNNING)) 633 sc->sc_flags |= PCCBB_KTHREAD_RUNNING; 634 else { 635 tsleep (sc, PWAIT, "pccbbev", 0); 636 /* 637 * Delay 1 second, make sure the user is done with 638 * whatever he is doing. We tsleep on sc->sc_flags, 639 * which should never be woken up. 640 */ 641 tsleep (&sc->sc_flags, PWAIT, "pccbbev", 1*hz); 642 } 643 mtx_lock(&sc->sc_mtx); 644 if (sc->sc_flags & PCCBB_KTHREAD_DONE) 645 break; 646 647 status = sc->sc_socketreg->socket_state; 648 if ((status & PCCBB_SOCKET_STAT_CD) == 0) { 649 pccbb_insert(sc); 650 } else { 651 pccbb_removal(sc); 652 } 653 mtx_unlock(&sc->sc_mtx); 654 } 655 mtx_unlock(&sc->sc_mtx); 656 sc->sc_flags &= ~PCCBB_KTHREAD_RUNNING; 657 wakeup(sc); 658 kthread_exit(0); 659 } 660 661 static void 662 pccbb_create_event_thread (struct pccbb_softc *sc) 663 { 664 if (kthread_create(pccbb_event_thread, sc, &sc->event_thread, 665 0, "%s%d", device_get_name(sc->sc_dev), 666 device_get_unit(sc->sc_dev))) { 667 device_printf (sc->sc_dev, "unable to create event thread.\n"); 668 panic ("pccbb_create_event_thread"); 669 } 670 } 671 672 static void 673 pccbb_start_threads(void *arg) 674 { 675 struct pccbb_sclist *sclist; 676 677 STAILQ_FOREACH(sclist, &softcs, entries) { 678 pccbb_create_event_thread(sclist->sc); 679 } 680 } 681 682 /************************************************************************/ 683 /* Insert/removal */ 684 /************************************************************************/ 685 686 static void 687 pccbb_insert (struct pccbb_softc *sc) 688 { 689 u_int32_t sockevent, sockstate; 690 int timeout = 30; 691 692 do { 693 sockevent = sc->sc_socketreg->socket_event; 694 sockstate = sc->sc_socketreg->socket_state; 695 } while (sockstate & PCCBB_SOCKET_STAT_CD && --timeout > 0); 696 697 if (timeout < 0) { 698 device_printf (sc->sc_dev, "insert timeout"); 699 return; 700 } 701 702 DEVPRINTF((sc->sc_dev, "card inserted: event=0x%08x, state=%08x\n", 703 sockevent, sockstate)); 704 705 if (sockstate & PCCBB_SOCKET_STAT_16BIT && sc->sc_pccarddev != NULL) { 706 sc->sc_flags |= PCCBB_16BIT_CARD; 707 if (CARD_ATTACH_CARD(sc->sc_pccarddev) != 0) 708 device_printf(sc->sc_dev, "card activation failed\n"); 709 } else if (sockstate & PCCBB_SOCKET_STAT_CB && sc->sc_cbdev != NULL) { 710 sc->sc_flags &= ~PCCBB_16BIT_CARD; 711 if (CARD_ATTACH_CARD(sc->sc_cbdev) != 0) 712 device_printf(sc->sc_dev, "card activation failed\n"); 713 } else { 714 device_printf (sc->sc_dev, "Unsupported card type detected\n"); 715 } 716 } 717 718 static void 719 pccbb_removal (struct pccbb_softc *sc) 720 { 721 u_int32_t sockstate; 722 struct pccbb_reslist *rle; 723 724 sockstate = sc->sc_socketreg->socket_state; 725 726 if (sockstate & PCCBB_16BIT_CARD && sc->sc_pccarddev != NULL) 727 CARD_DETACH_CARD(sc->sc_pccarddev, DETACH_FORCE); 728 else if ((!(sockstate & PCCBB_16BIT_CARD)) && sc->sc_cbdev != NULL) 729 CARD_DETACH_CARD(sc->sc_cbdev, DETACH_FORCE); 730 731 while (NULL != (rle = SLIST_FIRST(&sc->rl))) { 732 device_printf(sc->sc_dev, "Danger Will Robinson: Resource " 733 "left allocated! This is a bug... " 734 "(rid=%x, type=%d, addr=%x)\n", rle->rid, rle->type, 735 rle->start); 736 SLIST_REMOVE_HEAD(&sc->rl, entries); 737 } 738 } 739 740 /************************************************************************/ 741 /* Interrupt Handler */ 742 /************************************************************************/ 743 744 static void 745 pccbb_intr(void* arg) 746 { 747 struct pccbb_softc *sc = arg; 748 u_int32_t sockevent; 749 750 if (!(sockevent = sc->sc_socketreg->socket_event)) { 751 /* not for me. */ 752 return; 753 } 754 755 /* reset bit */ 756 sc->sc_socketreg->socket_event = sockevent | 0x01; 757 758 if (sockevent & PCCBB_SOCKET_EVENT_CD) { 759 mtx_lock(&sc->sc_mtx); 760 wakeup(sc); 761 mtx_unlock(&sc->sc_mtx); 762 } else { 763 if (sockevent & PCCBB_SOCKET_EVENT_CSTS) { 764 DPRINTF((" cstsevent occures, 0x%08x\n", 765 sc->sc_socketreg->socket_state)); 766 } 767 if (sockevent & PCCBB_SOCKET_EVENT_POWER) { 768 DPRINTF((" pwrevent occures, 0x%08x\n", 769 sc->sc_socketreg->socket_state)); 770 } 771 } 772 773 return; 774 } 775 776 /************************************************************************/ 777 /* Generic Power functions */ 778 /************************************************************************/ 779 780 static int 781 pccbb_detect_voltage(device_t dev) 782 { 783 struct pccbb_softc *sc = device_get_softc(dev); 784 u_int32_t psr; 785 int vol = CARD_UKN_CARD; 786 787 psr = sc->sc_socketreg->socket_state; 788 789 if (psr & PCCBB_SOCKET_STAT_5VCARD) { 790 vol |= CARD_5V_CARD; 791 } 792 if (psr & PCCBB_SOCKET_STAT_3VCARD) { 793 vol |= CARD_3V_CARD; 794 } 795 if (psr & PCCBB_SOCKET_STAT_XVCARD) { 796 vol |= CARD_XV_CARD; 797 } 798 if (psr & PCCBB_SOCKET_STAT_YVCARD) { 799 vol |= CARD_YV_CARD; 800 } 801 802 return vol; 803 } 804 805 static int 806 pccbb_power(device_t dev, int volts) 807 { 808 u_int32_t status, sock_ctrl; 809 struct pccbb_softc *sc = device_get_softc(dev); 810 811 DEVPRINTF((sc->sc_dev, "pccbb_power: %s and %s [%x]\n", 812 (volts & CARD_VCCMASK) == CARD_VCC_UC ? "CARD_VCC_UC" : 813 (volts & CARD_VCCMASK) == CARD_VCC_5V ? "CARD_VCC_5V" : 814 (volts & CARD_VCCMASK) == CARD_VCC_3V ? "CARD_VCC_3V" : 815 (volts & CARD_VCCMASK) == CARD_VCC_XV ? "CARD_VCC_XV" : 816 (volts & CARD_VCCMASK) == CARD_VCC_YV ? "CARD_VCC_YV" : 817 (volts & CARD_VCCMASK) == CARD_VCC_0V ? "CARD_VCC_0V" : 818 "VCC-UNKNOWN", 819 (volts & CARD_VPPMASK) == CARD_VPP_UC ? "CARD_VPP_UC" : 820 (volts & CARD_VPPMASK) == CARD_VPP_12V ? "CARD_VPP_12V" : 821 (volts & CARD_VPPMASK) == CARD_VPP_VCC ? "CARD_VPP_VCC" : 822 (volts & CARD_VPPMASK) == CARD_VPP_0V ? "CARD_VPP_0V" : 823 "VPP-UNKNOWN", 824 volts)); 825 826 status = sc->sc_socketreg->socket_state; 827 sock_ctrl = sc->sc_socketreg->socket_control; 828 829 switch (volts & CARD_VCCMASK) { 830 case CARD_VCC_UC: 831 break; 832 case CARD_VCC_5V: 833 if (PCCBB_SOCKET_STAT_5VCARD & status) { /* check 5 V card */ 834 sock_ctrl &= ~PCCBB_SOCKET_CTRL_VCCMASK; 835 sock_ctrl |= PCCBB_SOCKET_CTRL_VCC_5V; 836 } else { 837 device_printf(sc->sc_dev, 838 "BAD voltage request: no 5 V card\n"); 839 } 840 break; 841 case CARD_VCC_3V: 842 if (PCCBB_SOCKET_STAT_3VCARD & status) { 843 sock_ctrl &= ~PCCBB_SOCKET_CTRL_VCCMASK; 844 sock_ctrl |= PCCBB_SOCKET_CTRL_VCC_3V; 845 } else { 846 device_printf(sc->sc_dev, 847 "BAD voltage request: no 3.3 V card\n"); 848 } 849 break; 850 case CARD_VCC_0V: 851 sock_ctrl &= ~PCCBB_SOCKET_CTRL_VCCMASK; 852 break; 853 default: 854 return 0; /* power NEVER changed */ 855 break; 856 } 857 858 switch (volts & CARD_VPPMASK) { 859 case CARD_VPP_UC: 860 break; 861 case CARD_VPP_0V: 862 sock_ctrl &= ~PCCBB_SOCKET_CTRL_VPPMASK; 863 break; 864 case CARD_VPP_VCC: 865 sock_ctrl &= ~PCCBB_SOCKET_CTRL_VPPMASK; 866 sock_ctrl |= ((sock_ctrl >> 4) & 0x07); 867 break; 868 case CARD_VPP_12V: 869 sock_ctrl &= ~PCCBB_SOCKET_CTRL_VPPMASK; 870 sock_ctrl |= PCCBB_SOCKET_CTRL_VPP_12V; 871 break; 872 } 873 874 if (sc->sc_socketreg->socket_control == sock_ctrl) 875 return 1; /* no change necessary */ 876 877 sc->sc_socketreg->socket_control = sock_ctrl; 878 status = sc->sc_socketreg->socket_state; 879 880 { 881 int timeout = 20; 882 u_int32_t sockevent; 883 do { 884 DELAY(20*1000); 885 sockevent = sc->sc_socketreg->socket_event; 886 } while (!(sockevent & PCCBB_SOCKET_EVENT_POWER) && 887 --timeout > 0); 888 /* reset event status */ 889 sc->sc_socketreg->socket_event = sockevent; 890 if ( timeout < 0 ) { 891 printf ("VCC supply failed.\n"); 892 return 0; 893 } 894 } 895 /* XXX 896 * delay 400 ms: thgough the standard defines that the Vcc set-up time 897 * is 20 ms, some PC-Card bridge requires longer duration. 898 */ 899 DELAY(400*1000); 900 901 if (status & PCCBB_SOCKET_STAT_BADVCC) { 902 device_printf(sc->sc_dev, 903 "bad Vcc request. ctrl=0x%x, status=0x%x\n", 904 sock_ctrl ,status); 905 printf("pccbb_power: %s and %s [%x]\n", 906 (volts & CARD_VCCMASK) == CARD_VCC_UC ? "CARD_VCC_UC" : 907 (volts & CARD_VCCMASK) == CARD_VCC_5V ? "CARD_VCC_5V" : 908 (volts & CARD_VCCMASK) == CARD_VCC_3V ? "CARD_VCC_3V" : 909 (volts & CARD_VCCMASK) == CARD_VCC_XV ? "CARD_VCC_XV" : 910 (volts & CARD_VCCMASK) == CARD_VCC_YV ? "CARD_VCC_YV" : 911 (volts & CARD_VCCMASK) == CARD_VCC_0V ? "CARD_VCC_0V" : 912 "VCC-UNKNOWN", 913 (volts & CARD_VPPMASK) == CARD_VPP_UC ? "CARD_VPP_UC" : 914 (volts & CARD_VPPMASK) == CARD_VPP_12V ? "CARD_VPP_12V": 915 (volts & CARD_VPPMASK) == CARD_VPP_VCC ? "CARD_VPP_VCC": 916 (volts & CARD_VPPMASK) == CARD_VPP_0V ? "CARD_VPP_0V" : 917 "VPP-UNKNOWN", 918 volts); 919 return 0; 920 } 921 return 1; /* power changed correctly */ 922 } 923 924 /************************************************************************/ 925 /* Cardbus power functions */ 926 /************************************************************************/ 927 928 static void 929 pccbb_cardbus_reset(device_t dev) 930 { 931 struct pccbb_softc *sc = device_get_softc(dev); 932 int delay_us; 933 934 delay_us = sc->sc_chipset == CB_RF5C47X ? 400*1000 : 20*1000; 935 936 PCI_MASK_CONFIG(dev, PCCBBR_BRIDGECTRL, |PCCBBM_BRIDGECTRL_RESET, 2); 937 938 DELAY(delay_us); 939 940 /* If a card exists, unreset it! */ 941 if ((sc->sc_socketreg->socket_state & PCCBB_SOCKET_STAT_CD) == 0) { 942 PCI_MASK_CONFIG(dev, PCCBBR_BRIDGECTRL, 943 &~PCCBBM_BRIDGECTRL_RESET, 2); 944 DELAY(delay_us); 945 } 946 } 947 948 static int 949 pccbb_cardbus_power_enable_socket(device_t self, device_t child) 950 { 951 struct pccbb_softc *sc = device_get_softc(self); 952 int voltage; 953 954 if ((sc->sc_socketreg->socket_state & PCCBB_SOCKET_STAT_CD) 955 == PCCBB_SOCKET_STAT_CD) 956 return ENODEV; 957 958 voltage = pccbb_detect_voltage(self); 959 960 pccbb_power(self, CARD_VCC_0V | CARD_VPP_0V); 961 if (voltage & CARD_5V_CARD) 962 pccbb_power(self, CARD_VCC_5V | CARD_VPP_VCC); 963 else if (voltage & CARD_3V_CARD) 964 pccbb_power(self, CARD_VCC_3V | CARD_VPP_VCC); 965 else { 966 device_printf(self, "Unknown card voltage\n"); 967 return ENXIO; 968 } 969 970 pccbb_cardbus_reset(self); 971 return 0; 972 } 973 974 static void 975 pccbb_cardbus_power_disable_socket(device_t self, device_t child) 976 { 977 pccbb_power(self, CARD_VCC_0V | CARD_VPP_0V); 978 pccbb_cardbus_reset(self); 979 } 980 981 /************************************************************************/ 982 /* Cardbus Resource */ 983 /************************************************************************/ 984 985 static int 986 pccbb_cardbus_io_open(device_t dev, int win, u_int32_t start, u_int32_t end) 987 { 988 int basereg; 989 int limitreg; 990 991 if ((win < 0) || (win > 1)) { 992 DEVPRINTF((dev, 993 "pccbb_cardbus_io_open: window out of range %d\n", 994 win)); 995 return EINVAL; 996 } 997 998 basereg = win*8 + PCCBBR_IOBASE0; 999 limitreg = win*8 + PCCBBR_IOLIMIT0; 1000 1001 pci_write_config(dev, basereg, start, 4); 1002 pci_write_config(dev, limitreg, end, 4); 1003 return 0; 1004 } 1005 1006 static int 1007 pccbb_cardbus_mem_open(device_t dev, int win, u_int32_t start, u_int32_t end) 1008 { 1009 int basereg; 1010 int limitreg; 1011 1012 if ((win < 0) || (win > 1)) { 1013 DEVPRINTF((dev, 1014 "pccbb_cardbus_mem_open: window out of range %d\n", 1015 win)); 1016 return EINVAL; 1017 } 1018 1019 basereg = win*8 + PCCBBR_MEMBASE0; 1020 limitreg = win*8 + PCCBBR_MEMLIMIT0; 1021 1022 pci_write_config(dev, basereg, start, 4); 1023 pci_write_config(dev, limitreg, end, 4); 1024 return 0; 1025 } 1026 1027 static void 1028 pccbb_cardbus_auto_open(struct pccbb_softc *sc, int type) 1029 { 1030 u_int32_t starts[2]; 1031 u_int32_t ends[2]; 1032 struct pccbb_reslist *rle; 1033 int align; 1034 1035 starts[0] = starts[1] = 0xffffffff; 1036 ends[0] = ends[1] = 0; 1037 1038 SLIST_FOREACH(rle, &sc->rl, entries) { 1039 if (rle->type != type) 1040 ; 1041 else if (starts[0] == 0xffffffff) { 1042 starts[0] = rle->start; 1043 ends[0] = rle->end; 1044 rle->win = 0; 1045 } else if (rle->end > ends[0] && 1046 rle->start - ends[0] < PCCBB_AUTO_OPEN_SMALLHOLE) { 1047 ends[0] = rle->end; 1048 rle->win = 0; 1049 } else if (rle->start < starts[0] && 1050 starts[0] - rle->end < PCCBB_AUTO_OPEN_SMALLHOLE) { 1051 starts[0] = rle->start; 1052 rle->win = 0; 1053 } else if (starts[1] == 0xffffffff) { 1054 starts[1] = rle->start; 1055 ends[1] = rle->end; 1056 rle->win = 1; 1057 } else if (rle->end > ends[1] && 1058 rle->start - ends[1] < PCCBB_AUTO_OPEN_SMALLHOLE) { 1059 ends[1] = rle->end; 1060 rle->win = 1; 1061 } else if (rle->start < starts[1] && 1062 starts[1] - rle->end < PCCBB_AUTO_OPEN_SMALLHOLE) { 1063 starts[1] = rle->start; 1064 rle->win = 1; 1065 } else { 1066 u_int32_t diffs[2]; 1067 1068 diffs[0] = diffs[1] = 0xffffffff; 1069 if (rle->start > ends[0]) 1070 diffs[0] = rle->start - ends[0]; 1071 else if (rle->end < starts[0]) 1072 diffs[0] = starts[0] - rle->end; 1073 if (rle->start > ends[1]) 1074 diffs[1] = rle->start - ends[1]; 1075 else if (rle->end < starts[1]) 1076 diffs[1] = starts[1] - rle->end; 1077 1078 rle->win = (diffs[0] <= diffs[1])?0:1; 1079 if (rle->start > ends[rle->win]) 1080 ends[rle->win] = rle->end; 1081 else if (rle->end < starts[rle->win]) 1082 starts[rle->win] = rle->start; 1083 } 1084 } 1085 if (type == SYS_RES_MEMORY) 1086 align = PCCBB_MEMALIGN; 1087 else if (type == SYS_RES_IOPORT) 1088 align = PCCBB_IOALIGN; 1089 else 1090 align = 1; 1091 1092 if (starts[0] != 0xffffffff) 1093 starts[0] -= starts[0] % align; 1094 if (starts[1] != 0xffffffff) 1095 starts[1] -= starts[1] % align; 1096 if (ends[0] % align != 0) 1097 ends[0] += align - ends[0]%align - 1; 1098 if (ends[1] % align != 0) 1099 ends[1] += align - ends[1]%align - 1; 1100 1101 if (type == SYS_RES_MEMORY) { 1102 pccbb_cardbus_mem_open(sc->sc_dev, 0, starts[0], ends[0]); 1103 pccbb_cardbus_mem_open(sc->sc_dev, 1, starts[1], ends[1]); 1104 } else if (type == SYS_RES_IOPORT) { 1105 pccbb_cardbus_io_open(sc->sc_dev, 0, starts[0], ends[0]); 1106 pccbb_cardbus_io_open(sc->sc_dev, 1, starts[1], ends[1]); 1107 } 1108 } 1109 1110 static int 1111 pccbb_cardbus_activate_resource(device_t self, device_t child, int type, 1112 int rid, struct resource *r) 1113 { 1114 struct pccbb_softc *sc = device_get_softc(self); 1115 struct pccbb_reslist *rle; 1116 1117 if (type == SYS_RES_MEMORY || type == SYS_RES_IOPORT) { 1118 SLIST_FOREACH(rle, &sc->rl, entries) { 1119 if (type == rle->type && rid == rle->rid && 1120 child == rle->odev) 1121 return bus_generic_activate_resource( 1122 self, child, type, rid, r); 1123 } 1124 rle = malloc(sizeof(struct pccbb_reslist), M_DEVBUF, M_WAITOK); 1125 rle->type = type; 1126 rle->rid = rid; 1127 rle->start = rman_get_start(r); 1128 rle->end = rman_get_end(r); 1129 rle->odev = child; 1130 rle->win = -1; 1131 SLIST_INSERT_HEAD(&sc->rl, rle, entries); 1132 1133 pccbb_cardbus_auto_open(sc, type); 1134 } 1135 return bus_generic_activate_resource(self, child, type, rid, r); 1136 } 1137 1138 static int 1139 pccbb_cardbus_deactivate_resource(device_t self, device_t child, int type, 1140 int rid, struct resource *r) 1141 { 1142 struct pccbb_softc *sc = device_get_softc(self); 1143 struct pccbb_reslist *rle; 1144 1145 SLIST_FOREACH(rle, &sc->rl, entries) { 1146 if (type == rle->type && rid == rle->rid && 1147 child == rle->odev) { 1148 SLIST_REMOVE(&sc->rl, rle, pccbb_reslist, entries); 1149 if (type == SYS_RES_IOPORT || 1150 type == SYS_RES_MEMORY) 1151 pccbb_cardbus_auto_open(sc, type); 1152 free(rle, M_DEVBUF); 1153 break; 1154 } 1155 } 1156 return bus_generic_deactivate_resource(self, child, type, rid, r); 1157 } 1158 1159 static struct resource* 1160 pccbb_cardbus_alloc_resource(device_t self, device_t child, int type, int* rid, 1161 u_long start, u_long end, u_long count, 1162 u_int flags) 1163 { 1164 if (type == SYS_RES_IRQ) { 1165 struct pccbb_softc *sc = device_get_softc(self); 1166 if (start == 0) { 1167 start = end = rman_get_start(sc->sc_irq_res); 1168 } 1169 return bus_generic_alloc_resource(self, child, type, rid, 1170 start, end, count, flags); 1171 } else { 1172 if (type == SYS_RES_MEMORY && start == 0 && end == ~0) { 1173 start = CARDBUS_SYS_RES_MEMORY_START; 1174 end = CARDBUS_SYS_RES_MEMORY_END; 1175 } else if (type == SYS_RES_IOPORT && start == 0 && end == ~0) { 1176 start = CARDBUS_SYS_RES_IOPORT_START; 1177 end = CARDBUS_SYS_RES_IOPORT_END; 1178 } 1179 return bus_generic_alloc_resource(self, child, type, rid, 1180 start, end, count, flags); 1181 } 1182 } 1183 1184 static int 1185 pccbb_cardbus_release_resource(device_t self, device_t child, int type, 1186 int rid, struct resource *r) 1187 { 1188 return bus_generic_release_resource(self, child, type, rid, r); 1189 } 1190 1191 /************************************************************************/ 1192 /* PC Card Power Functions */ 1193 /************************************************************************/ 1194 1195 static int 1196 pccbb_pcic_power_enable_socket(device_t self, device_t child) 1197 { 1198 struct pccbb_softc *sc = device_get_softc(self); 1199 1200 DPRINTF(("pccbb_pcic_socket_enable:\n")); 1201 1202 /* power down/up the socket to reset */ 1203 { 1204 int voltage = pccbb_detect_voltage(self); 1205 1206 pccbb_power(self, CARD_VCC_0V | CARD_VPP_0V); 1207 if (voltage & CARD_5V_CARD) 1208 pccbb_power(self, CARD_VCC_5V | CARD_VPP_VCC); 1209 else if (voltage & CARD_3V_CARD) 1210 pccbb_power(self, CARD_VCC_3V | CARD_VPP_VCC); 1211 else { 1212 device_printf(self, "Unknown card voltage\n"); 1213 return ENXIO; 1214 } 1215 } 1216 1217 /* enable socket i/o */ 1218 PCIC_MASK(sc, PCIC_PWRCTL, | PCIC_PWRCTL_OE); 1219 1220 PCIC_WRITE(sc, PCIC_INTR, PCIC_INTR_ENABLE); 1221 /* hold reset for 30ms */ 1222 DELAY(30*1000); 1223 /* clear the reset flag */ 1224 PCIC_MASK(sc, PCIC_INTR, | PCIC_INTR_RESET); 1225 /* wait 20ms as per pc card standard (r2.01) section 4.3.6 */ 1226 DELAY(20*1000); 1227 1228 pccbb_pcic_wait_ready(sc); 1229 1230 /* disable all address windows */ 1231 PCIC_WRITE(sc, PCIC_ADDRWIN_ENABLE, 0); 1232 1233 { 1234 int cardtype; 1235 CARD_GET_TYPE(child, &cardtype); 1236 PCIC_MASK(sc, PCIC_INTR, | ((cardtype == PCCARD_IFTYPE_IO) ? 1237 PCIC_INTR_CARDTYPE_IO : 1238 PCIC_INTR_CARDTYPE_MEM)); 1239 DEVPRINTF((sc->sc_dev, "card type is %s\n", 1240 (cardtype == PCCARD_IFTYPE_IO) ? "io" : "mem")); 1241 } 1242 1243 /* reinstall all the memory and io mappings */ 1244 { 1245 int win; 1246 1247 for (win = 0; win < PCIC_MEM_WINS; ++win) { 1248 if (sc->memalloc & (1 << win)) { 1249 pccbb_pcic_do_mem_map(sc, win); 1250 } 1251 } 1252 for (win = 0; win < PCIC_IO_WINS; ++win) { 1253 if (sc->ioalloc & (1 << win)) { 1254 pccbb_pcic_do_io_map(sc, win); 1255 } 1256 } 1257 } 1258 return 0; 1259 } 1260 1261 static void 1262 pccbb_pcic_power_disable_socket(device_t self, device_t child) 1263 { 1264 struct pccbb_softc *sc = device_get_softc(self); 1265 1266 DPRINTF(("pccbb_pcic_socket_disable\n")); 1267 1268 /* reset signal asserting... */ 1269 PCIC_MASK(sc, PCIC_INTR, & ~PCIC_INTR_RESET); 1270 DELAY(2*1000); 1271 1272 /* power down the socket */ 1273 PCIC_MASK(sc, PCIC_PWRCTL, &~PCIC_PWRCTL_OE); 1274 pccbb_power(self, CARD_VCC_0V | CARD_VPP_0V); 1275 1276 /* wait 300ms until power fails (Tpf). */ 1277 DELAY(300 * 1000); 1278 } 1279 1280 /************************************************************************/ 1281 /* PC Card Resource Functions */ 1282 /************************************************************************/ 1283 1284 static void 1285 pccbb_pcic_wait_ready(struct pccbb_softc *sc) 1286 { 1287 int i; 1288 DEVPRINTF((sc->sc_dev, "pccbb_pcic_wait_ready: status 0x%02x\n", 1289 PCIC_READ(sc, PCIC_IF_STATUS))); 1290 for (i = 0; i < 10000; i++) { 1291 if (PCIC_READ(sc, PCIC_IF_STATUS) & PCIC_IF_STATUS_READY) { 1292 return; 1293 } 1294 DELAY(500); 1295 } 1296 device_printf(sc->sc_dev, "ready never happened, status = %02x\n", 1297 PCIC_READ(sc, PCIC_IF_STATUS)); 1298 } 1299 1300 #define PCIC_MEMINFO(NUM) { \ 1301 PCIC_SYSMEM_ADDR ## NUM ## _START_LSB, \ 1302 PCIC_SYSMEM_ADDR ## NUM ## _START_MSB, \ 1303 PCIC_SYSMEM_ADDR ## NUM ## _STOP_LSB, \ 1304 PCIC_SYSMEM_ADDR ## NUM ## _STOP_MSB, \ 1305 PCIC_SYSMEM_ADDR ## NUM ## _WIN, \ 1306 PCIC_CARDMEM_ADDR ## NUM ## _LSB, \ 1307 PCIC_CARDMEM_ADDR ## NUM ## _MSB, \ 1308 PCIC_ADDRWIN_ENABLE_MEM ## NUM ## , \ 1309 } 1310 1311 static struct mem_map_index_st { 1312 int sysmem_start_lsb; 1313 int sysmem_start_msb; 1314 int sysmem_stop_lsb; 1315 int sysmem_stop_msb; 1316 int sysmem_win; 1317 int cardmem_lsb; 1318 int cardmem_msb; 1319 int memenable; 1320 } mem_map_index[] = { 1321 PCIC_MEMINFO(0), 1322 PCIC_MEMINFO(1), 1323 PCIC_MEMINFO(2), 1324 PCIC_MEMINFO(3), 1325 PCIC_MEMINFO(4), 1326 }; 1327 #undef PCIC_MEMINFO 1328 1329 static void 1330 pccbb_pcic_do_mem_map(struct pccbb_softc *sc, int win) 1331 { 1332 PCIC_WRITE(sc, mem_map_index[win].sysmem_start_lsb, 1333 (sc->mem[win].addr >> PCIC_SYSMEM_ADDRX_SHIFT) & 0xff); 1334 PCIC_WRITE(sc, mem_map_index[win].sysmem_start_msb, 1335 ((sc->mem[win].addr >> (PCIC_SYSMEM_ADDRX_SHIFT + 8)) & 1336 PCIC_SYSMEM_ADDRX_START_MSB_ADDR_MASK) | 0x80); 1337 1338 PCIC_WRITE(sc, mem_map_index[win].sysmem_stop_lsb, 1339 ((sc->mem[win].addr + sc->mem[win].realsize - 1) >> 1340 PCIC_SYSMEM_ADDRX_SHIFT) & 0xff); 1341 PCIC_WRITE(sc, mem_map_index[win].sysmem_stop_msb, 1342 (((sc->mem[win].addr + sc->mem[win].realsize - 1) >> 1343 (PCIC_SYSMEM_ADDRX_SHIFT + 8)) & 1344 PCIC_SYSMEM_ADDRX_STOP_MSB_ADDR_MASK) | 1345 PCIC_SYSMEM_ADDRX_STOP_MSB_WAIT2); 1346 1347 PCIC_WRITE(sc, mem_map_index[win].sysmem_win, 1348 (sc->mem[win].addr >> PCIC_MEMREG_WIN_SHIFT) & 0xff); 1349 1350 PCIC_WRITE(sc, mem_map_index[win].cardmem_lsb, 1351 (sc->mem[win].offset >> PCIC_CARDMEM_ADDRX_SHIFT) & 0xff); 1352 PCIC_WRITE(sc, mem_map_index[win].cardmem_msb, 1353 ((sc->mem[win].offset >> (PCIC_CARDMEM_ADDRX_SHIFT + 8)) & 1354 PCIC_CARDMEM_ADDRX_MSB_ADDR_MASK) | 1355 ((sc->mem[win].kind == PCCARD_MEM_ATTR) ? 1356 PCIC_CARDMEM_ADDRX_MSB_REGACTIVE_ATTR : 0)); 1357 1358 PCIC_MASK(sc, PCIC_ADDRWIN_ENABLE, | PCIC_ADDRWIN_ENABLE_MEMCS16 1359 | mem_map_index[win].memenable); 1360 1361 DELAY(100); 1362 1363 #ifdef CBB_DEBUG 1364 { 1365 int r1, r2, r3, r4, r5, r6, r7; 1366 r1 = PCIC_READ(sc, mem_map_index[win].sysmem_start_msb); 1367 r2 = PCIC_READ(sc, mem_map_index[win].sysmem_start_lsb); 1368 r3 = PCIC_READ(sc, mem_map_index[win].sysmem_stop_msb); 1369 r4 = PCIC_READ(sc, mem_map_index[win].sysmem_stop_lsb); 1370 r5 = PCIC_READ(sc, mem_map_index[win].cardmem_msb); 1371 r6 = PCIC_READ(sc, mem_map_index[win].cardmem_lsb); 1372 r7 = PCIC_READ(sc, mem_map_index[win].sysmem_win); 1373 DPRINTF(("pccbb_pcic_do_mem_map window %d: %02x%02x %02x%02x " 1374 "%02x%02x %02x (%08x+%08x.%08x*%08lx)\n", win, r1, r2, r3, r4, r5, r6, r7, 1375 sc->mem[win].addr, sc->mem[win].size, sc->mem[win].realsize, sc->mem[win].offset)); 1376 } 1377 #endif 1378 } 1379 1380 static int 1381 pccbb_pcic_mem_map(struct pccbb_softc *sc, int kind, 1382 struct resource *r, bus_addr_t card_addr, int *win) 1383 { 1384 int i; 1385 1386 *win = -1; 1387 for (i = 0; i < PCIC_MEM_WINS; i++) { 1388 if ((sc->memalloc & (1 << i)) == 0) { 1389 *win = i; 1390 sc->memalloc |= (1 << i); 1391 break; 1392 } 1393 } 1394 if (*win == -1) 1395 return (1); 1396 1397 card_addr = card_addr - card_addr % PCIC_MEM_PAGESIZE; 1398 sc->mem[*win].memt = rman_get_bustag(r); 1399 sc->mem[*win].memh = rman_get_bushandle(r); 1400 sc->mem[*win].addr = rman_get_start(r); 1401 sc->mem[*win].size = rman_get_end(r) - sc->mem[*win].addr + 1; 1402 sc->mem[*win].realsize = sc->mem[*win].size + PCIC_MEM_PAGESIZE - 1; 1403 sc->mem[*win].realsize = sc->mem[*win].realsize - 1404 (sc->mem[*win].realsize % PCIC_MEM_PAGESIZE); 1405 sc->mem[*win].offset = ((long)card_addr) - 1406 ((long)(sc->mem[*win].addr)); 1407 sc->mem[*win].kind = kind; 1408 1409 DPRINTF(("pccbb_pcic_mem_map window %d bus %x+%x+%lx card addr %x\n", 1410 *win, sc->mem[*win].addr, sc->mem[*win].size, 1411 sc->mem[*win].offset, card_addr)); 1412 1413 pccbb_pcic_do_mem_map(sc, *win); 1414 1415 return (0); 1416 } 1417 1418 static void 1419 pccbb_pcic_mem_unmap(struct pccbb_softc *sc, int window) 1420 { 1421 if (window >= PCIC_MEM_WINS) 1422 panic("pccbb_pcic_mem_unmap: window out of range"); 1423 1424 PCIC_MASK(sc, PCIC_ADDRWIN_ENABLE, & ~mem_map_index[window].memenable); 1425 sc->memalloc &= ~(1 << window); 1426 } 1427 1428 #define PCIC_IOINFO(NUM) { \ 1429 PCIC_IOADDR ## NUM ## _START_LSB, \ 1430 PCIC_IOADDR ## NUM ## _START_MSB, \ 1431 PCIC_IOADDR ## NUM ## _STOP_LSB, \ 1432 PCIC_IOADDR ## NUM ## _STOP_MSB, \ 1433 PCIC_ADDRWIN_ENABLE_IO ## NUM ## , \ 1434 PCIC_IOCTL_IO ## NUM ## _WAITSTATE \ 1435 | PCIC_IOCTL_IO ## NUM ## _ZEROWAIT \ 1436 | PCIC_IOCTL_IO ## NUM ## _IOCS16SRC_MASK \ 1437 | PCIC_IOCTL_IO ## NUM ## _DATASIZE_MASK, \ 1438 { \ 1439 PCIC_IOCTL_IO ## NUM ## _IOCS16SRC_CARD, \ 1440 PCIC_IOCTL_IO ## NUM ## _IOCS16SRC_DATASIZE \ 1441 | PCIC_IOCTL_IO ## NUM ## _DATASIZE_8BIT, \ 1442 PCIC_IOCTL_IO ## NUM ## _IOCS16SRC_DATASIZE \ 1443 | PCIC_IOCTL_IO ## NUM ## _DATASIZE_16BIT, \ 1444 } \ 1445 } 1446 1447 static struct io_map_index_st { 1448 int start_lsb; 1449 int start_msb; 1450 int stop_lsb; 1451 int stop_msb; 1452 int ioenable; 1453 int ioctlmask; 1454 int ioctlbits[3]; /* indexed by PCCARD_WIDTH_* */ 1455 } io_map_index[] = { 1456 PCIC_IOINFO(0), 1457 PCIC_IOINFO(1), 1458 }; 1459 #undef PCIC_IOINFO 1460 1461 static void pccbb_pcic_do_io_map(struct pccbb_softc *sc, int win) 1462 { 1463 PCIC_WRITE(sc, io_map_index[win].start_lsb, sc->io[win].addr & 0xff); 1464 PCIC_WRITE(sc, io_map_index[win].start_msb, 1465 (sc->io[win].addr >> 8) & 0xff); 1466 1467 PCIC_WRITE(sc, io_map_index[win].stop_lsb, 1468 (sc->io[win].addr + sc->io[win].size - 1) & 0xff); 1469 PCIC_WRITE(sc, io_map_index[win].stop_msb, 1470 ((sc->io[win].addr + sc->io[win].size - 1) >> 8) & 0xff); 1471 1472 PCIC_MASK2(sc, PCIC_IOCTL, 1473 & ~io_map_index[win].ioctlmask, 1474 | io_map_index[win].ioctlbits[sc->io[win].width]); 1475 1476 PCIC_MASK(sc, PCIC_ADDRWIN_ENABLE, | io_map_index[win].ioenable); 1477 #ifdef CBB_DEBUG 1478 { 1479 int r1, r2, r3, r4; 1480 r1 = PCIC_READ(sc, io_map_index[win].start_msb); 1481 r2 = PCIC_READ(sc, io_map_index[win].start_lsb); 1482 r3 = PCIC_READ(sc, io_map_index[win].stop_msb); 1483 r4 = PCIC_READ(sc, io_map_index[win].stop_lsb); 1484 DPRINTF(("pccbb_pcic_do_io_map window %d: %02x%02x %02x%02x " 1485 "(%08x+%08x)\n", win, r1, r2, r3, r4, 1486 sc->io[win].addr, sc->io[win].size)); 1487 } 1488 #endif 1489 } 1490 1491 static int 1492 pccbb_pcic_io_map(struct pccbb_softc *sc, int width, 1493 struct resource *r, bus_addr_t card_addr, int *win) 1494 { 1495 int i; 1496 #ifdef CBB_DEBUG 1497 static char *width_names[] = { "auto", "io8", "io16"}; 1498 #endif 1499 1500 *win = -1; 1501 for (i=0; i < PCIC_IO_WINS; i++) { 1502 if ((sc->ioalloc & (1 << i)) == 0) { 1503 *win = i; 1504 sc->ioalloc |= (1 << i); 1505 break; 1506 } 1507 } 1508 if (*win == -1) 1509 return (1); 1510 1511 sc->io[*win].iot = rman_get_bustag(r); 1512 sc->io[*win].ioh = rman_get_bushandle(r); 1513 sc->io[*win].addr = rman_get_start(r); 1514 sc->io[*win].size = rman_get_end(r) - sc->io[*win].addr + 1; 1515 sc->io[*win].flags = 0; 1516 sc->io[*win].width = width; 1517 1518 DPRINTF(("pccbb_pcic_io_map window %d %s port %x+%x\n", 1519 *win, width_names[width], sc->io[*win].addr, 1520 sc->io[*win].size)); 1521 1522 pccbb_pcic_do_io_map(sc, *win); 1523 1524 return (0); 1525 } 1526 1527 static void 1528 pccbb_pcic_io_unmap(struct pccbb_softc *sc, int window) 1529 { 1530 if (window >= PCIC_IO_WINS) 1531 panic("pccbb_pcic_io_unmap: window out of range"); 1532 1533 PCIC_MASK(sc, PCIC_ADDRWIN_ENABLE, & ~io_map_index[window].ioenable); 1534 1535 sc->ioalloc &= ~(1 << window); 1536 1537 sc->io[window].iot = 0; 1538 sc->io[window].ioh = 0; 1539 sc->io[window].addr = 0; 1540 sc->io[window].size = 0; 1541 sc->io[window].flags = 0; 1542 sc->io[window].width = 0; 1543 } 1544 1545 static int 1546 pccbb_pcic_activate_resource(device_t self, device_t child, int type, int rid, 1547 struct resource *r) 1548 { 1549 int err; 1550 int win; 1551 struct pccbb_reslist *rle; 1552 struct pccbb_softc *sc = device_get_softc(self); 1553 1554 if (rman_get_flags(r) & RF_ACTIVE) 1555 return 0; 1556 1557 switch (type) { 1558 case SYS_RES_IOPORT: 1559 err = pccbb_pcic_io_map(sc, 0, r, 0, &win); 1560 if (err) 1561 return err; 1562 break; 1563 case SYS_RES_MEMORY: 1564 err = pccbb_pcic_mem_map(sc, 0, r, 0, &win); 1565 if (err) 1566 return err; 1567 break; 1568 default: 1569 break; 1570 } 1571 SLIST_FOREACH(rle, &sc->rl, entries) { 1572 if (type == rle->type && rid == rle->rid && 1573 child == rle->odev) { 1574 rle->win = win; 1575 break; 1576 } 1577 } 1578 err = bus_generic_activate_resource(self, child, type, rid, r); 1579 return (err); 1580 } 1581 1582 static int 1583 pccbb_pcic_deactivate_resource(device_t self, device_t child, int type, 1584 int rid, struct resource *r) 1585 { 1586 struct pccbb_softc *sc = device_get_softc(self); 1587 int win; 1588 struct pccbb_reslist *rle; 1589 win = -1; 1590 SLIST_FOREACH(rle, &sc->rl, entries) { 1591 if (type == rle->type && rid == rle->rid && 1592 child == rle->odev) { 1593 win = rle->win; 1594 break; 1595 } 1596 } 1597 if (win == -1) { 1598 panic("pccbb_pcic: deactivating bogus resoure"); 1599 return 1; 1600 } 1601 1602 switch (type) { 1603 case SYS_RES_IOPORT: 1604 pccbb_pcic_io_unmap(sc, win); 1605 break; 1606 case SYS_RES_MEMORY: 1607 pccbb_pcic_mem_unmap(sc, win); 1608 break; 1609 default: 1610 break; 1611 } 1612 return bus_generic_deactivate_resource(self, child, type, rid, r); 1613 } 1614 1615 static struct resource* 1616 pccbb_pcic_alloc_resource(device_t self, device_t child, int type, int* rid, 1617 u_long start, u_long end, u_long count, u_int flags) 1618 { 1619 struct resource *r = NULL; 1620 struct pccbb_softc *sc = device_get_softc(self); 1621 struct pccbb_reslist *rle; 1622 1623 if ((sc->sc_flags & PCCBB_PCIC_MEM_32) == 0) { 1624 panic("PCCBB bridge cannot handle non MEM_32 bridges\n"); 1625 } 1626 1627 switch (type) { 1628 case SYS_RES_MEMORY: 1629 /* Nearly default */ 1630 if (start == 0 && end == ~0 && count != 1) { 1631 start = CARDBUS_SYS_RES_MEMORY_START; /* XXX -- should be tweakable*/ 1632 end = CARDBUS_SYS_RES_MEMORY_END; 1633 } 1634 flags = (flags & ~RF_ALIGNMENT_MASK) 1635 | rman_make_alignment_flags(PCCBB_MEMALIGN); 1636 break; 1637 case SYS_RES_IOPORT: 1638 if (start < 0x100) 1639 start = 0x100; /* XXX tweakable? */ 1640 if (end < start) 1641 end = start; 1642 break; 1643 case SYS_RES_IRQ: 1644 flags |= RF_SHAREABLE; 1645 start = end = rman_get_start(sc->sc_irq_res); 1646 break; 1647 } 1648 r = bus_generic_alloc_resource(self, child, type, rid, start, end, 1649 count, flags & ~RF_ACTIVE); 1650 if (r == NULL) 1651 return NULL; 1652 1653 rle = malloc(sizeof(struct pccbb_reslist), M_DEVBUF, M_WAITOK); 1654 rle->type = type; 1655 rle->rid = *rid; 1656 rle->start = rman_get_start(r); 1657 rle->end = rman_get_end(r); 1658 rle->odev = child; 1659 rle->win = -1; 1660 SLIST_INSERT_HEAD(&sc->rl, rle, entries); 1661 1662 if (flags & RF_ACTIVE) { 1663 if (bus_activate_resource(child, type, *rid, r) != 0) { 1664 BUS_RELEASE_RESOURCE(self, child, type, *rid, r); 1665 return NULL; 1666 } 1667 } 1668 1669 return r; 1670 } 1671 1672 static int 1673 pccbb_pcic_release_resource(device_t self, device_t child, int type, 1674 int rid, struct resource *res) 1675 { 1676 struct pccbb_softc *sc = device_get_softc(self); 1677 struct pccbb_reslist *rle; 1678 int count = 0; 1679 1680 if (rman_get_flags(res) & RF_ACTIVE) { 1681 int error; 1682 error = bus_deactivate_resource(child, type, rid, res); 1683 if (error != 0) 1684 return error; 1685 } 1686 1687 SLIST_FOREACH(rle, &sc->rl, entries) { 1688 if (type == rle->type && rid == rle->rid && 1689 child == rle->odev) { 1690 SLIST_REMOVE(&sc->rl, rle, pccbb_reslist, entries); 1691 free(rle, M_DEVBUF); 1692 count++; 1693 break; 1694 } 1695 } 1696 if (count == 0) { 1697 panic("pccbb_pcic: releasing bogus resource"); 1698 } 1699 1700 return bus_generic_release_resource(self, child, type, rid, res); 1701 } 1702 1703 /************************************************************************/ 1704 /* PC Card methods */ 1705 /************************************************************************/ 1706 1707 static int 1708 pccbb_pcic_set_res_flags(device_t self, device_t child, int type, int rid, 1709 u_int32_t flags) 1710 { 1711 struct pccbb_softc *sc = device_get_softc(self); 1712 1713 if (type != SYS_RES_MEMORY) 1714 return (EINVAL); 1715 sc->mem[rid].kind = flags; 1716 pccbb_pcic_do_mem_map(sc, rid); 1717 return 0; 1718 } 1719 1720 static int 1721 pccbb_pcic_set_memory_offset(device_t self, device_t child, int rid, 1722 u_int32_t cardaddr, u_int32_t *deltap) 1723 { 1724 struct pccbb_softc *sc = device_get_softc(self); 1725 int win; 1726 struct pccbb_reslist *rle; 1727 u_int32_t delta; 1728 1729 win = -1; 1730 1731 SLIST_FOREACH(rle, &sc->rl, entries) { 1732 if (SYS_RES_MEMORY == rle->type && rid == rle->rid && 1733 child == rle->odev) { 1734 win = rle->win; 1735 break; 1736 } 1737 } 1738 if (win == -1) { 1739 panic("pccbb_pcic: setting memory offset of bogus resource"); 1740 return 1; 1741 } 1742 1743 delta = cardaddr % PCIC_MEM_PAGESIZE; 1744 if (deltap) 1745 *deltap = delta; 1746 cardaddr -= delta; 1747 sc->mem[win].realsize = sc->mem[win].size + delta + 1748 PCIC_MEM_PAGESIZE - 1; 1749 sc->mem[win].realsize = sc->mem[win].realsize - 1750 (sc->mem[win].realsize % PCIC_MEM_PAGESIZE); 1751 sc->mem[win].offset = cardaddr - sc->mem[win].addr; 1752 pccbb_pcic_do_mem_map(sc, win); 1753 1754 return 0; 1755 } 1756 1757 /************************************************************************/ 1758 /* POWER methods */ 1759 /************************************************************************/ 1760 1761 static int 1762 pccbb_power_enable_socket(device_t self, device_t child) 1763 { 1764 struct pccbb_softc *sc = device_get_softc(self); 1765 1766 if (sc->sc_flags & PCCBB_16BIT_CARD) 1767 return pccbb_pcic_power_enable_socket(self, child); 1768 else 1769 return pccbb_cardbus_power_enable_socket(self, child); 1770 } 1771 1772 static void 1773 pccbb_power_disable_socket(device_t self, device_t child) 1774 { 1775 struct pccbb_softc *sc = device_get_softc(self); 1776 if (sc->sc_flags & PCCBB_16BIT_CARD) 1777 pccbb_pcic_power_disable_socket(self, child); 1778 else 1779 pccbb_cardbus_power_disable_socket(self, child); 1780 } 1781 /************************************************************************/ 1782 /* BUS Methods */ 1783 /************************************************************************/ 1784 1785 1786 static int 1787 pccbb_activate_resource(device_t self, device_t child, int type, int rid, 1788 struct resource *r) 1789 { 1790 struct pccbb_softc *sc = device_get_softc(self); 1791 1792 if (sc->sc_flags & PCCBB_16BIT_CARD) 1793 return pccbb_pcic_activate_resource(self, child, type, rid, r); 1794 else 1795 return pccbb_cardbus_activate_resource(self, child, type, rid, 1796 r); 1797 } 1798 1799 static int 1800 pccbb_deactivate_resource(device_t self, device_t child, int type, 1801 int rid, struct resource *r) 1802 { 1803 struct pccbb_softc *sc = device_get_softc(self); 1804 1805 if (sc->sc_flags & PCCBB_16BIT_CARD) 1806 return pccbb_pcic_deactivate_resource(self, child, type, 1807 rid, r); 1808 else 1809 return pccbb_cardbus_deactivate_resource(self, child, type, 1810 rid, r); 1811 } 1812 1813 static struct resource* 1814 pccbb_alloc_resource(device_t self, device_t child, int type, int* rid, 1815 u_long start, u_long end, u_long count, u_int flags) 1816 { 1817 struct pccbb_softc *sc = device_get_softc(self); 1818 1819 if (sc->sc_flags & PCCBB_16BIT_CARD) 1820 return pccbb_pcic_alloc_resource(self, child, type, rid, 1821 start, end, count, flags); 1822 else 1823 return pccbb_cardbus_alloc_resource(self, child, type, rid, 1824 start, end, count, flags); 1825 } 1826 1827 static int 1828 pccbb_release_resource(device_t self, device_t child, int type, int rid, 1829 struct resource *r) 1830 { 1831 struct pccbb_softc *sc = device_get_softc(self); 1832 1833 if (sc->sc_flags & PCCBB_16BIT_CARD) 1834 return pccbb_pcic_release_resource(self, child, type, 1835 rid, r); 1836 else 1837 return pccbb_cardbus_release_resource(self, child, type, 1838 rid, r); 1839 } 1840 1841 static int 1842 pccbb_read_ivar(device_t dev, device_t child, int which, uintptr_t *result) 1843 { 1844 struct pccbb_softc *sc = device_get_softc(dev); 1845 1846 switch (which) { 1847 case PCIB_IVAR_BUS: 1848 *result = sc->sc_secbus; 1849 return(0); 1850 } 1851 return(ENOENT); 1852 } 1853 1854 static int 1855 pccbb_write_ivar(device_t dev, device_t child, int which, uintptr_t value) 1856 { 1857 struct pccbb_softc *sc = device_get_softc(dev); 1858 1859 switch (which) { 1860 case PCIB_IVAR_BUS: 1861 sc->sc_secbus = value; 1862 break; 1863 } 1864 return(ENOENT); 1865 } 1866 1867 /************************************************************************/ 1868 /* PCI compat methods */ 1869 /************************************************************************/ 1870 1871 static int 1872 pccbb_maxslots(device_t dev) 1873 { 1874 return 0; 1875 } 1876 1877 static u_int32_t 1878 pccbb_read_config(device_t dev, int b, int s, int f, int reg, int width) 1879 { 1880 /* 1881 * Pass through to the next ppb up the chain (i.e. our grandparent). 1882 */ 1883 return PCIB_READ_CONFIG(device_get_parent(device_get_parent(dev)), 1884 b, s, f, reg, width); 1885 } 1886 1887 static void 1888 pccbb_write_config(device_t dev, int b, int s, int f, int reg, u_int32_t val, 1889 int width) 1890 { 1891 /* 1892 * Pass through to the next ppb up the chain (i.e. our grandparent). 1893 */ 1894 PCIB_WRITE_CONFIG(device_get_parent(device_get_parent(dev)), 1895 b, s, f, reg, val, width); 1896 } 1897 1898 static device_method_t pccbb_methods[] = { 1899 /* Device interface */ 1900 DEVMETHOD(device_probe, pccbb_probe), 1901 DEVMETHOD(device_attach, pccbb_attach), 1902 DEVMETHOD(device_detach, pccbb_detach), 1903 DEVMETHOD(device_suspend, bus_generic_suspend), 1904 DEVMETHOD(device_resume, bus_generic_resume), 1905 1906 /* bus methods */ 1907 DEVMETHOD(bus_print_child, bus_generic_print_child), 1908 DEVMETHOD(bus_read_ivar, pccbb_read_ivar), 1909 DEVMETHOD(bus_write_ivar, pccbb_write_ivar), 1910 DEVMETHOD(bus_alloc_resource, pccbb_alloc_resource), 1911 DEVMETHOD(bus_release_resource, pccbb_release_resource), 1912 DEVMETHOD(bus_activate_resource, pccbb_activate_resource), 1913 DEVMETHOD(bus_deactivate_resource, pccbb_deactivate_resource), 1914 DEVMETHOD(bus_driver_added, pccbb_driver_added), 1915 DEVMETHOD(bus_child_detached, pccbb_child_detached), 1916 DEVMETHOD(bus_setup_intr, bus_generic_setup_intr), 1917 DEVMETHOD(bus_teardown_intr, bus_generic_teardown_intr), 1918 1919 /* 16-bit card interface */ 1920 DEVMETHOD(card_set_res_flags, pccbb_pcic_set_res_flags), 1921 DEVMETHOD(card_set_memory_offset, pccbb_pcic_set_memory_offset), 1922 1923 /* power interface */ 1924 DEVMETHOD(power_enable_socket, pccbb_power_enable_socket), 1925 DEVMETHOD(power_disable_socket, pccbb_power_disable_socket), 1926 1927 /* pcib compatibility interface */ 1928 DEVMETHOD(pcib_maxslots, pccbb_maxslots), 1929 DEVMETHOD(pcib_read_config, pccbb_read_config), 1930 DEVMETHOD(pcib_write_config, pccbb_write_config), 1931 {0,0} 1932 }; 1933 1934 static driver_t pccbb_driver = { 1935 "pccbb", 1936 pccbb_methods, 1937 sizeof(struct pccbb_softc) 1938 }; 1939 1940 static devclass_t pccbb_devclass; 1941 1942 DRIVER_MODULE(pccbb, pci, pccbb_driver, pccbb_devclass, 0, 0); 1943 1944 SYSINIT(pccbb, SI_SUB_KTHREAD_IDLE, SI_ORDER_ANY, pccbb_start_threads, 0); 1945