diff options
-rw-r--r-- | sys/dev/ic/bcw.c | 283 |
1 files changed, 152 insertions, 131 deletions
diff --git a/sys/dev/ic/bcw.c b/sys/dev/ic/bcw.c index 0479eebec80..107ed019ec4 100644 --- a/sys/dev/ic/bcw.c +++ b/sys/dev/ic/bcw.c @@ -1,4 +1,4 @@ -/* $OpenBSD: bcw.c,v 1.74 2007/03/16 15:28:54 mglocker Exp $ */ +/* $OpenBSD: bcw.c,v 1.75 2007/03/16 22:22:24 mglocker Exp $ */ /* * Copyright (c) 2006 Jon Simola <jsimola@gmail.com> @@ -130,6 +130,7 @@ int bcw_write_initvals(struct bcw_softc *, int bcw_load_initvals(struct bcw_softc *); void bcw_leds_switch_all(struct bcw_softc *, int); int bcw_gpio_init(struct bcw_softc *); +int bcw_chip_init(struct bcw_softc *); /* * PHY @@ -1555,142 +1556,19 @@ bcw_txintr(struct bcw_softc *sc) } -/* initialize the interface */ +/* + * Initialize the interface + */ int bcw_init(struct ifnet *ifp) { - struct bcw_softc *sc = ifp->if_softc; - uint16_t val16; - uint32_t val32; - int error, i, tmp; - - BCW_WRITE(sc, BCW_MMIO_SBF, BCW_SBF_CORE_READY | BCW_SBF_400_MAGIC); - - /* load firmware */ - if ((error = bcw_load_firmware(sc))) - return (error); - - /* - * verify firmware revision - */ - BCW_WRITE(sc, BCW_MMIO_GIR, 0xffffffff); - BCW_WRITE(sc, BCW_MMIO_SBF, 0x00020402); - for (i = 0; i < 50; i++) { - if (BCW_READ(sc, BCW_MMIO_GIR) == BCW_INTR_READY) - break; - delay(10); - } - if (i == 50) { - printf("%s: interrupt-ready timeout!\n", sc->sc_dev.dv_xname); - return (1); - } - BCW_READ(sc, BCW_MMIO_GIR); /* dummy read */ - - val16 = bcw_shm_read16(sc, BCW_SHM_SHARED, BCW_UCODE_REVISION); - - DPRINTF(("%s: Firmware revision 0x%x, patchlevel 0x%x " - "(20%.2i-%.2i-%.2i %.2i:%.2i:%.2i)\n", - sc->sc_dev.dv_xname, val16, - bcw_shm_read16(sc, BCW_SHM_SHARED, BCW_UCODE_PATCHLEVEL), - (bcw_shm_read16(sc, BCW_SHM_SHARED, BCW_UCODE_DATE) >> 12) - & 0xf, - (bcw_shm_read16(sc, BCW_SHM_SHARED, BCW_UCODE_DATE) >> 8) - & 0xf, - bcw_shm_read16(sc, BCW_SHM_SHARED, BCW_UCODE_DATE) - & 0xff, - (bcw_shm_read16(sc, BCW_SHM_SHARED, BCW_UCODE_TIME) >> 11) - & 0x1f, - (bcw_shm_read16(sc, BCW_SHM_SHARED, BCW_UCODE_TIME) >> 5) - & 0x3f, - bcw_shm_read16(sc, BCW_SHM_SHARED, BCW_UCODE_TIME) - & 0x1f)); - - if (val16 > 0x128) { - printf("%s: no support for this firmware revision!\n", - sc->sc_dev.dv_xname); - return (1); - } + struct bcw_softc *sc = ifp->if_softc; + int error; - /* initialize GPIO */ - if ((error = bcw_gpio_init(sc))) + /* initialize chip */ + if ((error = bcw_chip_init(sc))) return (error); - /* load init values */ - if ((error = bcw_load_initvals(sc))) - return (error); - - /* turn radio on */ - bcw_radio_on(sc); - - BCW_WRITE16(sc, 0x03e6, 0); - if ((error = bcw_phy_init(sc))) - return (error); - - /* select initial interference mitigation */ - tmp = sc->sc_radio_interfmode; - sc->sc_radio_interfmode = BCW_RADIO_INTERFMODE_NONE; - bcw_radio_set_interf_mitigation(sc, tmp); - - bcw_phy_set_antenna_diversity(sc); - bcw_radio_set_txantenna(sc, BCW_RADIO_TXANTENNA_DEFAULT); - if (sc->sc_phy_type == BCW_PHY_TYPEB) { - val16 = BCW_READ16(sc, 0x005e); - val16 |= 0x0004; - BCW_WRITE16(sc, 0x005e, val16); - } - BCW_WRITE(sc, 0x0100, 0x01000000); - if (sc->sc_core[sc->sc_currentcore].rev < 5) - BCW_WRITE(sc, 0x010c, 0x01000000); - - val32 = BCW_READ(sc, BCW_MMIO_SBF); - val32 &= ~BCW_SBF_ADHOC; - BCW_WRITE(sc, BCW_MMIO_SBF, val32); - val32 = BCW_READ(sc, BCW_MMIO_SBF); - val32 |= BCW_SBF_ADHOC; - BCW_WRITE(sc, BCW_MMIO_SBF, val32); - - val32 = BCW_READ(sc, BCW_MMIO_SBF); - val32 |= 0x100000; - BCW_WRITE(sc, BCW_MMIO_SBF, val32); - - if (bcw_using_pio(sc)) { - BCW_WRITE(sc, 0x0210, 0x00000100); - BCW_WRITE(sc, 0x0230, 0x00000100); - BCW_WRITE(sc, 0x0250, 0x00000100); - BCW_WRITE(sc, 0x0270, 0x00000100); - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x0034, 0); - } - - /* probe response timeout value */ - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x0074, 0); - - /* initially set the wireless operation mode */ - bcw_set_opmode(ifp); - - if (sc->sc_core[sc->sc_currentcore].rev < 3) { - BCW_WRITE16(sc, 0x060e, 0); - BCW_WRITE16(sc, 0x0610, 0x8000); - BCW_WRITE16(sc, 0x0604, 0); - BCW_WRITE16(sc, 0x0606, 0x0200); - } else { - BCW_WRITE(sc, 0x0188, 0x80000000); - BCW_WRITE(sc, 0x018c, 0x02000000); - } - BCW_WRITE(sc, BCW_MMIO_GIR, 0x00004000); - BCW_WRITE(sc, BCW_MMIO_DMA0_INT_MASK, 0x0001dc00); - BCW_WRITE(sc, BCW_MMIO_DMA1_INT_MASK, 0x0000dc00); - BCW_WRITE(sc, BCW_MMIO_DMA2_INT_MASK, 0x0000dc00); - BCW_WRITE(sc, BCW_MMIO_DMA3_INT_MASK, 0x0001dc00); - BCW_WRITE(sc, BCW_MMIO_DMA4_INT_MASK, 0x0000dc00); - BCW_WRITE(sc, BCW_MMIO_DMA5_INT_MASK, 0x0000dc00); - - val32 = BCW_READ(sc, BCW_CIR_SBTMSTATELOW); - val32 |= 0x00100000; - BCW_WRITE(sc, BCW_CIR_SBTMSTATELOW, val32); - /* TODO bcw_pctl_powerup_delay(sc) */ - - DPRINTF(("%s: Chip initialized\n", sc->sc_dev.dv_xname)); - /* start timer */ timeout_add(&sc->sc_timeout, hz); @@ -2820,6 +2698,149 @@ bcw_gpio_init(struct bcw_softc *sc) } /* + * Initialize the chip + * + * http://bcm-specs.sipsolutions.net/ChipInit + */ +int +bcw_chip_init(struct bcw_softc *sc) +{ + struct ifnet *ifp = &sc->sc_ic.ic_if; + uint16_t val16; + uint32_t val32; + int error, i, tmp; + + BCW_WRITE(sc, BCW_MMIO_SBF, BCW_SBF_CORE_READY | BCW_SBF_400_MAGIC); + + /* load firmware */ + if ((error = bcw_load_firmware(sc))) + return (error); + + /* + * verify firmware revision + */ + BCW_WRITE(sc, BCW_MMIO_GIR, 0xffffffff); + BCW_WRITE(sc, BCW_MMIO_SBF, 0x00020402); + for (i = 0; i < 50; i++) { + if (BCW_READ(sc, BCW_MMIO_GIR) == BCW_INTR_READY) + break; + delay(10); + } + if (i == 50) { + printf("%s: interrupt-ready timeout!\n", sc->sc_dev.dv_xname); + return (1); + } + BCW_READ(sc, BCW_MMIO_GIR); /* dummy read */ + + val16 = bcw_shm_read16(sc, BCW_SHM_SHARED, BCW_UCODE_REVISION); + + DPRINTF(("%s: Firmware revision 0x%x, patchlevel 0x%x " + "(20%.2i-%.2i-%.2i %.2i:%.2i:%.2i)\n", + sc->sc_dev.dv_xname, val16, + bcw_shm_read16(sc, BCW_SHM_SHARED, BCW_UCODE_PATCHLEVEL), + (bcw_shm_read16(sc, BCW_SHM_SHARED, BCW_UCODE_DATE) >> 12) + & 0xf, + (bcw_shm_read16(sc, BCW_SHM_SHARED, BCW_UCODE_DATE) >> 8) + & 0xf, + bcw_shm_read16(sc, BCW_SHM_SHARED, BCW_UCODE_DATE) + & 0xff, + (bcw_shm_read16(sc, BCW_SHM_SHARED, BCW_UCODE_TIME) >> 11) + & 0x1f, + (bcw_shm_read16(sc, BCW_SHM_SHARED, BCW_UCODE_TIME) >> 5) + & 0x3f, + bcw_shm_read16(sc, BCW_SHM_SHARED, BCW_UCODE_TIME) + & 0x1f)); + + if (val16 > 0x128) { + printf("%s: no support for this firmware revision!\n", + sc->sc_dev.dv_xname); + return (1); + } + + /* initialize GPIO */ + if ((error = bcw_gpio_init(sc))) + return (error); + + /* load init values */ + if ((error = bcw_load_initvals(sc))) + return (error); + + /* turn radio on */ + bcw_radio_on(sc); + + BCW_WRITE16(sc, 0x03e6, 0); + if ((error = bcw_phy_init(sc))) + return (error); + + /* select initial interference mitigation */ + tmp = sc->sc_radio_interfmode; + sc->sc_radio_interfmode = BCW_RADIO_INTERFMODE_NONE; + bcw_radio_set_interf_mitigation(sc, tmp); + + bcw_phy_set_antenna_diversity(sc); + bcw_radio_set_txantenna(sc, BCW_RADIO_TXANTENNA_DEFAULT); + if (sc->sc_phy_type == BCW_PHY_TYPEB) { + val16 = BCW_READ16(sc, 0x005e); + val16 |= 0x0004; + BCW_WRITE16(sc, 0x005e, val16); + } + BCW_WRITE(sc, 0x0100, 0x01000000); + if (sc->sc_core[sc->sc_currentcore].rev < 5) + BCW_WRITE(sc, 0x010c, 0x01000000); + + val32 = BCW_READ(sc, BCW_MMIO_SBF); + val32 &= ~BCW_SBF_ADHOC; + BCW_WRITE(sc, BCW_MMIO_SBF, val32); + val32 = BCW_READ(sc, BCW_MMIO_SBF); + val32 |= BCW_SBF_ADHOC; + BCW_WRITE(sc, BCW_MMIO_SBF, val32); + + val32 = BCW_READ(sc, BCW_MMIO_SBF); + val32 |= 0x100000; + BCW_WRITE(sc, BCW_MMIO_SBF, val32); + + if (bcw_using_pio(sc)) { + BCW_WRITE(sc, 0x0210, 0x00000100); + BCW_WRITE(sc, 0x0230, 0x00000100); + BCW_WRITE(sc, 0x0250, 0x00000100); + BCW_WRITE(sc, 0x0270, 0x00000100); + bcw_shm_write16(sc, BCW_SHM_SHARED, 0x0034, 0); + } + + /* probe response timeout value */ + bcw_shm_write16(sc, BCW_SHM_SHARED, 0x0074, 0); + + /* initially set the wireless operation mode */ + bcw_set_opmode(ifp); + + if (sc->sc_core[sc->sc_currentcore].rev < 3) { + BCW_WRITE16(sc, 0x060e, 0); + BCW_WRITE16(sc, 0x0610, 0x8000); + BCW_WRITE16(sc, 0x0604, 0); + BCW_WRITE16(sc, 0x0606, 0x0200); + } else { + BCW_WRITE(sc, 0x0188, 0x80000000); + BCW_WRITE(sc, 0x018c, 0x02000000); + } + BCW_WRITE(sc, BCW_MMIO_GIR, 0x00004000); + BCW_WRITE(sc, BCW_MMIO_DMA0_INT_MASK, 0x0001dc00); + BCW_WRITE(sc, BCW_MMIO_DMA1_INT_MASK, 0x0000dc00); + BCW_WRITE(sc, BCW_MMIO_DMA2_INT_MASK, 0x0000dc00); + BCW_WRITE(sc, BCW_MMIO_DMA3_INT_MASK, 0x0001dc00); + BCW_WRITE(sc, BCW_MMIO_DMA4_INT_MASK, 0x0000dc00); + BCW_WRITE(sc, BCW_MMIO_DMA5_INT_MASK, 0x0000dc00); + + val32 = BCW_READ(sc, BCW_CIR_SBTMSTATELOW); + val32 |= 0x00100000; + BCW_WRITE(sc, BCW_CIR_SBTMSTATELOW, val32); + /* TODO bcw_pctl_powerup_delay(sc) */ + + DPRINTF(("%s: Chip initialized\n", sc->sc_dev.dv_xname)); + + return (0); +} + +/* * PHY */ int |