diff options
Diffstat (limited to 'sys/dev/ic/bcw.c')
-rw-r--r-- | sys/dev/ic/bcw.c | 7160 |
1 files changed, 0 insertions, 7160 deletions
diff --git a/sys/dev/ic/bcw.c b/sys/dev/ic/bcw.c deleted file mode 100644 index b329cb730da..00000000000 --- a/sys/dev/ic/bcw.c +++ /dev/null @@ -1,7160 +0,0 @@ -/* $OpenBSD: bcw.c,v 1.88 2007/04/04 20:48:12 mglocker Exp $ */ - -/* - * Copyright (c) 2007 Marcus Glocker <mglocker@openbsd.org> - * Copyright (c) 2006 Jon Simola <jsimola@gmail.com> - * - * Permission to use, copy, modify, and distribute this software for any - * purpose with or without fee is hereby granted, provided that the above - * copyright notice and this permission notice appear in all copies. - * - * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES - * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF - * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR - * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES - * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN - * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF - * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. - */ - -/* - * Broadcom BCM43xx Wireless network chipsets (broadcom.com) - * SiliconBackplane is technology from Sonics, Inc. (sonicsinc.com) - */ - -#include "bpfilter.h" - -#include <sys/param.h> -#include <sys/systm.h> -#include <sys/timeout.h> -#include <sys/sockio.h> -#include <sys/mbuf.h> -#include <sys/malloc.h> -#include <sys/kernel.h> -#include <sys/device.h> -#include <sys/socket.h> - -#include <net/if.h> -#include <net/if_dl.h> -#include <net/if_media.h> - -#ifdef INET -#include <netinet/in.h> -#include <netinet/in_systm.h> -#include <netinet/in_var.h> -#include <netinet/ip.h> -#include <netinet/if_ether.h> -#endif -#if NBPFILTER > 0 -#include <net/bpf.h> -#endif - -#include <net80211/ieee80211_var.h> -#include <net80211/ieee80211_radiotap.h> - -#include <dev/pci/pcireg.h> -#include <dev/pci/pcivar.h> -#include <dev/pci/pcidevs.h> - -#include <dev/ic/bcwreg.h> -#include <dev/ic/bcwvar.h> - -#include <uvm/uvm_extern.h> - -/* - * Helper Routines - */ -void bcw_shm_ctl_word(struct bcw_softc *, uint16_t, - uint16_t); -uint16_t bcw_shm_read16(struct bcw_softc *, uint16_t, uint16_t); -void bcw_shm_write16(struct bcw_softc *, uint16_t, uint16_t, - uint16_t); -uint32_t bcw_shm_read32(struct bcw_softc *, uint16_t, uint16_t); -void bcw_shm_write32(struct bcw_softc *, uint16_t, uint16_t, - uint32_t); -void bcw_radio_write16(struct bcw_softc *, uint16_t, - uint16_t); -int bcw_radio_read16(struct bcw_softc *, uint16_t); -void bcw_phy_write16(struct bcw_softc *, uint16_t, uint16_t); -int bcw_phy_read16(struct bcw_softc *, uint16_t); -void bcw_ram_write(struct bcw_softc *, uint16_t, uint32_t); -int bcw_lv(int, int, int); -void bcw_dummy_transmission(struct bcw_softc *); -struct bcw_lopair * bcw_get_lopair(struct bcw_softc *, uint16_t, - uint16_t); -void bcw_stack_save(uint32_t *, size_t *, uint8_t, uint16_t, - uint16_t); -uint16_t bcw_stack_restore(uint32_t *, uint8_t, uint16_t); -int bcw_using_pio(struct bcw_softc *); -void bcw_rate_memory_write(struct bcw_softc *, uint16_t, - int); -void bcw_rate_memory_init(struct bcw_softc *); -uint16_t bcw_flip_4bit(uint16_t); - -/* - * 80211 - */ -int bcw_init(struct ifnet *); -void bcw_start(struct ifnet *); -void bcw_stop(struct ifnet *, int); -void bcw_watchdog(struct ifnet *); -void bcw_set_opmode(struct ifnet *); -void bcw_macfilter_set(struct bcw_softc *, uint16_t, - const uint8_t *); -void bcw_macfilter_clear(struct bcw_softc *, uint16_t); -void bcw_templateram_set(struct bcw_softc *); -void bcw_mac_enable(struct bcw_softc *); -void bcw_mac_disable(struct bcw_softc *); -uint32_t bcw_intr_enable(struct bcw_softc *, uint32_t); -uint32_t bcw_intr_disable(struct bcw_softc *, uint32_t); -void bcw_rxintr(struct bcw_softc *); -void bcw_txintr(struct bcw_softc *); -int bcw_add_rxbuf(struct bcw_softc *, int); -void bcw_rxdrain(struct bcw_softc *); -void bcw_tick(void *); -int bcw_ioctl(struct ifnet *, u_long, caddr_t); -int bcw_alloc_rx_ring(struct bcw_softc *, - struct bcw_rx_ring *, int); -void bcw_reset_rx_ring(struct bcw_softc *, - struct bcw_rx_ring *); -void bcw_free_rx_ring(struct bcw_softc *, - struct bcw_rx_ring *); -int bcw_alloc_tx_ring(struct bcw_softc *, - struct bcw_tx_ring *, int); -void bcw_reset_tx_ring(struct bcw_softc *, - struct bcw_tx_ring *); -void bcw_free_tx_ring(struct bcw_softc *, - struct bcw_tx_ring *); -int bcw_newstate(struct ieee80211com *, - enum ieee80211_state, int); -int bcw_media_change(struct ifnet *); -void bcw_media_status(struct ifnet *, struct ifmediareq *); -int bcw_validate_chip_access(struct bcw_softc *); -int bcw_change_core(struct bcw_softc *, int); -int bcw_core_enable(struct bcw_softc *, uint32_t); -int bcw_core_disable(struct bcw_softc *, uint32_t); -void bcw_80211_core_reset(struct bcw_softc *, int); -void bcw_80211_core_disable(struct bcw_softc *); -int bcw_iocore_enable(struct bcw_softc *, uint32_t); -int bcw_get_firmware(const char *, const uint8_t *, size_t, - size_t *, size_t *); -int bcw_load_firmware(struct bcw_softc *); -int bcw_write_initvals(struct bcw_softc *, - const struct bcw_initval *, const unsigned int); -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 *); -int bcw_bs_init(struct bcw_softc *); -int bcw_80211_core_init(struct bcw_softc *, int); -uint8_t bcw_sprom_crc8(uint8_t, uint8_t); -uint8_t bcw_sprom_crc(const uint16_t *); -int bcw_sprom_read(struct bcw_softc *, uint16_t *); -int bcw_sprom_get(struct bcw_softc *); -void bcw_microcode_init_fbf(struct bcw_softc *); - -/* - * PHY - */ -int bcw_phy_get(struct bcw_softc *); -int bcw_phy_init(struct bcw_softc *); -void bcw_phy_initg(struct bcw_softc *); -void bcw_phy_initb2(struct bcw_softc *); -void bcw_phy_initb4(struct bcw_softc *); -void bcw_phy_initb5(struct bcw_softc *); -void bcw_phy_initb6(struct bcw_softc *); -void bcw_phy_inita(struct bcw_softc *); -void bcw_phy_setupa(struct bcw_softc *); -void bcw_phy_setupg(struct bcw_softc *); -void bcw_phy_calc_loopback_gain(struct bcw_softc *); -void bcw_phy_agcsetup(struct bcw_softc *); -void bcw_phy_init_pctl(struct bcw_softc *); -void bcw_phy_init_noisescaletbl(struct bcw_softc *); -void bcw_phy_set_baseband_atten(struct bcw_softc *, - uint16_t); -int8_t bcw_phy_estimate_powerout(struct bcw_softc *, int8_t); -void bcw_phy_xmitpower(struct bcw_softc *); -uint16_t bcw_phy_lo_b_r15_loop(struct bcw_softc *); -void bcw_phy_lo_b_measure(struct bcw_softc *); -void bcw_phy_lo_g_state(struct bcw_softc *, - struct bcw_lopair *, struct bcw_lopair *, uint16_t); -void bcw_phy_lo_g_measure(struct bcw_softc *); -void bcw_phy_lo_g_measure_txctl2(struct bcw_softc *); -uint32_t bcw_phy_lo_g_singledeviation(struct bcw_softc *, - uint16_t); -uint16_t bcw_phy_lo_g_deviation_subval(struct bcw_softc *, - uint16_t); -void bcw_phy_lo_adjust(struct bcw_softc *, int); -void bcw_phy_lo_mark_current_used(struct bcw_softc *); -void bcw_phy_lo_write(struct bcw_softc *, - struct bcw_lopair *); -struct bcw_lopair * bcw_phy_find_lopair(struct bcw_softc *, uint16_t, - uint16_t, uint16_t); -struct bcw_lopair * bcw_phy_current_lopair(struct bcw_softc *); -void bcw_phy_set_antenna_diversity(struct bcw_softc *); -void bcw_phy_calibrate(struct bcw_softc *); -int bcw_phy_connect(struct bcw_softc *, int); -void bcw_phy_lock(struct bcw_softc *); -void bcw_phy_unlock(struct bcw_softc *); - -/* - * Radio - */ -int bcw_radio_get(struct bcw_softc *); -void bcw_radio_off(struct bcw_softc *); -void bcw_radio_on(struct bcw_softc *); -void bcw_radio_nrssi_hw_write(struct bcw_softc *, - uint16_t, int16_t); -int16_t bcw_radio_nrssi_hw_read(struct bcw_softc *, uint16_t); -void bcw_radio_nrssi_hw_update(struct bcw_softc *, uint16_t); -void bcw_radio_calc_nrssi_threshold(struct bcw_softc *); -void bcw_radio_calc_nrssi_slope(struct bcw_softc *); -void bcw_radio_calc_nrssi_offset(struct bcw_softc *); -void bcw_radio_set_all_gains(struct bcw_softc *, int16_t, - int16_t, int16_t); -void bcw_radio_set_original_gains(struct bcw_softc *); -uint16_t bcw_radio_calibrationvalue(struct bcw_softc *); -void bcw_radio_set_txpower_a(struct bcw_softc *, uint16_t); -void bcw_radio_set_txpower_bg(struct bcw_softc *, uint16_t, - uint16_t, uint16_t); -uint16_t bcw_radio_init2050(struct bcw_softc *); -void bcw_radio_init2060(struct bcw_softc *); -void bcw_radio_spw(struct bcw_softc *, uint8_t); -int bcw_radio_select_channel(struct bcw_softc *, uint8_t, - int); -uint16_t bcw_radio_chan2freq_a(uint8_t); -uint16_t bcw_radio_chan2freq_bg(uint8_t); -uint16_t bcw_radio_default_baseband_atten(struct bcw_softc *); -uint16_t bcw_radio_default_radio_atten(struct bcw_softc *); -uint16_t bcw_radio_default_txctl1(struct bcw_softc *); -void bcw_radio_clear_tssi(struct bcw_softc *); -void bcw_radio_set_tx_iq(struct bcw_softc *); -uint16_t bcw_radio_get_txgain_baseband(uint16_t); -uint16_t bcw_radio_get_txgain_freq_power_amp(uint16_t); -uint16_t bcw_radio_get_txgain_dac(uint16_t); -uint16_t bcw_radio_freq_r3a_value(uint16_t); -int bcw_radio_set_interf_mitigation(struct bcw_softc *, - int); -int bcw_radio_interf_mitigation_enable(struct bcw_softc *, - int); -void bcw_radio_set_txantenna(struct bcw_softc *, uint32_t); -void bcw_radio_lock(struct bcw_softc *); -void bcw_radio_unlock(struct bcw_softc *); - -/* - * ILT - */ -void bcw_ilt_write(struct bcw_softc *, uint16_t, uint16_t); -uint16_t bcw_ilt_read(struct bcw_softc *, uint16_t); - -/* - * Power Control - */ -void bcw_pc_crystal_on(struct bcw_softc *); -void bcw_pc_crystal_off(struct bcw_softc *); -int bcw_pc_init(struct bcw_softc *); -int bcw_pc_set_clock(struct bcw_softc *, uint16_t); -void bcw_pc_saving_ctl_bits(struct bcw_softc *, int, int); -uint16_t bcw_pc_powerup_delay(struct bcw_softc *); -int bcw_pc_clock_freqlimit(struct bcw_softc *, int); -int bcw_pc_get_slowclocksrc(struct bcw_softc *); - -/* - * XMIT - */ -uint8_t bcw_xmit_plcp_get_ratecode_cck(const uint8_t); -uint8_t bcw_xmit_plcp_get_ratecode_ofdm(const uint8_t); - -struct cfdriver bcw_cd = { - NULL, "bcw", DV_IFNET -}; - -#define BCW_PHY_STACKSAVE(offset) \ - do { \ - bcw_stack_save(stack, &stackidx, 0x1, (offset), \ - bcw_phy_read16(sc, (offset))); \ - } while (0) -#define BCW_ILT_STACKSAVE(offset) \ - do { \ - bcw_stack_save(stack, &stackidx, 0x3, (offset), \ - bcw_ilt_read(sc, (offset))); \ - } while (0) -#define BCW_RADIO_STACKSAVE(offset) \ - do { \ - bcw_stack_save(stack, &stackidx, 0x2, (offset), \ - bcw_radio_read16(sc, (offset))); \ - } while (0) - -/* - * Table for bcw_radio_calibrationvalue() - */ -const uint16_t rcc_table[16] = { - 0x0002, 0x0003, 0x0001, 0x000F, - 0x0006, 0x0007, 0x0005, 0x000F, - 0x000A, 0x000B, 0x0009, 0x000F, - 0x000E, 0x000F, 0x000D, 0x000F, -}; - -/* - * ILT (Internal Lookup Table) - */ -const uint32_t bcw_ilt_rotor[BCW_ILT_ROTOR_SIZE] = { - 0xFEB93FFD, 0xFEC63FFD, /* 0 */ - 0xFED23FFD, 0xFEDF3FFD, - 0xFEEC3FFE, 0xFEF83FFE, - 0xFF053FFE, 0xFF113FFE, - 0xFF1E3FFE, 0xFF2A3FFF, /* 8 */ - 0xFF373FFF, 0xFF443FFF, - 0xFF503FFF, 0xFF5D3FFF, - 0xFF693FFF, 0xFF763FFF, - 0xFF824000, 0xFF8F4000, /* 16 */ - 0xFF9B4000, 0xFFA84000, - 0xFFB54000, 0xFFC14000, - 0xFFCE4000, 0xFFDA4000, - 0xFFE74000, 0xFFF34000, /* 24 */ - 0x00004000, 0x000D4000, - 0x00194000, 0x00264000, - 0x00324000, 0x003F4000, - 0x004B4000, 0x00584000, /* 32 */ - 0x00654000, 0x00714000, - 0x007E4000, 0x008A3FFF, - 0x00973FFF, 0x00A33FFF, - 0x00B03FFF, 0x00BC3FFF, /* 40 */ - 0x00C93FFF, 0x00D63FFF, - 0x00E23FFE, 0x00EF3FFE, - 0x00FB3FFE, 0x01083FFE, - 0x01143FFE, 0x01213FFD, /* 48 */ - 0x012E3FFD, 0x013A3FFD, - 0x01473FFD, -}; - -const uint16_t bcw_ilt_sigmasqr1[BCW_ILT_SIGMASQR_SIZE] = { - 0x007A, 0x0075, 0x0071, 0x006C, /* 0 */ - 0x0067, 0x0063, 0x005E, 0x0059, - 0x0054, 0x0050, 0x004B, 0x0046, - 0x0042, 0x003D, 0x003D, 0x003D, - 0x003D, 0x003D, 0x003D, 0x003D, /* 16 */ - 0x003D, 0x003D, 0x003D, 0x003D, - 0x003D, 0x003D, 0x0000, 0x003D, - 0x003D, 0x003D, 0x003D, 0x003D, - 0x003D, 0x003D, 0x003D, 0x003D, /* 32 */ - 0x003D, 0x003D, 0x003D, 0x003D, - 0x0042, 0x0046, 0x004B, 0x0050, - 0x0054, 0x0059, 0x005E, 0x0063, - 0x0067, 0x006C, 0x0071, 0x0075, /* 48 */ - 0x007A, -}; - -const uint16_t bcw_ilt_sigmasqr2[BCW_ILT_SIGMASQR_SIZE] = { - 0x00DE, 0x00DC, 0x00DA, 0x00D8, /* 0 */ - 0x00D6, 0x00D4, 0x00D2, 0x00CF, - 0x00CD, 0x00CA, 0x00C7, 0x00C4, - 0x00C1, 0x00BE, 0x00BE, 0x00BE, - 0x00BE, 0x00BE, 0x00BE, 0x00BE, /* 16 */ - 0x00BE, 0x00BE, 0x00BE, 0x00BE, - 0x00BE, 0x00BE, 0x0000, 0x00BE, - 0x00BE, 0x00BE, 0x00BE, 0x00BE, - 0x00BE, 0x00BE, 0x00BE, 0x00BE, /* 32 */ - 0x00BE, 0x00BE, 0x00BE, 0x00BE, - 0x00C1, 0x00C4, 0x00C7, 0x00CA, - 0x00CD, 0x00CF, 0x00D2, 0x00D4, - 0x00D6, 0x00D8, 0x00DA, 0x00DC, /* 48 */ - 0x00DE, -}; - -const uint32_t bcw_ilt_retard[BCW_ILT_RETARD_SIZE] = { - 0xDB93CB87, 0xD666CF64, /* 0 */ - 0xD1FDD358, 0xCDA6D826, - 0xCA38DD9F, 0xC729E2B4, - 0xC469E88E, 0xC26AEE2B, - 0xC0DEF46C, 0xC073FA62, /* 8 */ - 0xC01D00D5, 0xC0760743, - 0xC1560D1E, 0xC2E51369, - 0xC4ED18FF, 0xC7AC1ED7, - 0xCB2823B2, 0xCEFA28D9, /* 16 */ - 0xD2F62D3F, 0xD7BB3197, - 0xDCE53568, 0xE1FE3875, - 0xE7D13B35, 0xED663D35, - 0xF39B3EC4, 0xF98E3FA7, /* 24 */ - 0x00004000, 0x06723FA7, - 0x0C653EC4, 0x129A3D35, - 0x182F3B35, 0x1E023875, - 0x231B3568, 0x28453197, /* 32 */ - 0x2D0A2D3F, 0x310628D9, - 0x34D823B2, 0x38541ED7, - 0x3B1318FF, 0x3D1B1369, - 0x3EAA0D1E, 0x3F8A0743, /* 40 */ - 0x3FE300D5, 0x3F8DFA62, - 0x3F22F46C, 0x3D96EE2B, - 0x3B97E88E, 0x38D7E2B4, - 0x35C8DD9F, 0x325AD826, /* 48 */ - 0x2E03D358, 0x299ACF64, - 0x246DCB87, -}; - -const uint16_t bcw_ilt_finefreqa[BCW_ILT_FINEFREQA_SIZE] = { - 0x0082, 0x0082, 0x0102, 0x0182, /* 0 */ - 0x0202, 0x0282, 0x0302, 0x0382, - 0x0402, 0x0482, 0x0502, 0x0582, - 0x05E2, 0x0662, 0x06E2, 0x0762, - 0x07E2, 0x0842, 0x08C2, 0x0942, /* 16 */ - 0x09C2, 0x0A22, 0x0AA2, 0x0B02, - 0x0B82, 0x0BE2, 0x0C62, 0x0CC2, - 0x0D42, 0x0DA2, 0x0E02, 0x0E62, - 0x0EE2, 0x0F42, 0x0FA2, 0x1002, /* 32 */ - 0x1062, 0x10C2, 0x1122, 0x1182, - 0x11E2, 0x1242, 0x12A2, 0x12E2, - 0x1342, 0x13A2, 0x1402, 0x1442, - 0x14A2, 0x14E2, 0x1542, 0x1582, /* 48 */ - 0x15E2, 0x1622, 0x1662, 0x16C1, - 0x1701, 0x1741, 0x1781, 0x17E1, - 0x1821, 0x1861, 0x18A1, 0x18E1, - 0x1921, 0x1961, 0x19A1, 0x19E1, /* 64 */ - 0x1A21, 0x1A61, 0x1AA1, 0x1AC1, - 0x1B01, 0x1B41, 0x1B81, 0x1BA1, - 0x1BE1, 0x1C21, 0x1C41, 0x1C81, - 0x1CA1, 0x1CE1, 0x1D01, 0x1D41, /* 80 */ - 0x1D61, 0x1DA1, 0x1DC1, 0x1E01, - 0x1E21, 0x1E61, 0x1E81, 0x1EA1, - 0x1EE1, 0x1F01, 0x1F21, 0x1F41, - 0x1F81, 0x1FA1, 0x1FC1, 0x1FE1, /* 96 */ - 0x2001, 0x2041, 0x2061, 0x2081, - 0x20A1, 0x20C1, 0x20E1, 0x2101, - 0x2121, 0x2141, 0x2161, 0x2181, - 0x21A1, 0x21C1, 0x21E1, 0x2201, /* 112 */ - 0x2221, 0x2241, 0x2261, 0x2281, - 0x22A1, 0x22C1, 0x22C1, 0x22E1, - 0x2301, 0x2321, 0x2341, 0x2361, - 0x2361, 0x2381, 0x23A1, 0x23C1, /* 128 */ - 0x23E1, 0x23E1, 0x2401, 0x2421, - 0x2441, 0x2441, 0x2461, 0x2481, - 0x2481, 0x24A1, 0x24C1, 0x24C1, - 0x24E1, 0x2501, 0x2501, 0x2521, /* 144 */ - 0x2541, 0x2541, 0x2561, 0x2561, - 0x2581, 0x25A1, 0x25A1, 0x25C1, - 0x25C1, 0x25E1, 0x2601, 0x2601, - 0x2621, 0x2621, 0x2641, 0x2641, /* 160 */ - 0x2661, 0x2661, 0x2681, 0x2681, - 0x26A1, 0x26A1, 0x26C1, 0x26C1, - 0x26E1, 0x26E1, 0x2701, 0x2701, - 0x2721, 0x2721, 0x2740, 0x2740, /* 176 */ - 0x2760, 0x2760, 0x2780, 0x2780, - 0x2780, 0x27A0, 0x27A0, 0x27C0, - 0x27C0, 0x27E0, 0x27E0, 0x27E0, - 0x2800, 0x2800, 0x2820, 0x2820, /* 192 */ - 0x2820, 0x2840, 0x2840, 0x2840, - 0x2860, 0x2860, 0x2880, 0x2880, - 0x2880, 0x28A0, 0x28A0, 0x28A0, - 0x28C0, 0x28C0, 0x28C0, 0x28E0, /* 208 */ - 0x28E0, 0x28E0, 0x2900, 0x2900, - 0x2900, 0x2920, 0x2920, 0x2920, - 0x2940, 0x2940, 0x2940, 0x2960, - 0x2960, 0x2960, 0x2960, 0x2980, /* 224 */ - 0x2980, 0x2980, 0x29A0, 0x29A0, - 0x29A0, 0x29A0, 0x29C0, 0x29C0, - 0x29C0, 0x29E0, 0x29E0, 0x29E0, - 0x29E0, 0x2A00, 0x2A00, 0x2A00, /* 240 */ - 0x2A00, 0x2A20, 0x2A20, 0x2A20, - 0x2A20, 0x2A40, 0x2A40, 0x2A40, - 0x2A40, 0x2A60, 0x2A60, 0x2A60, -}; - -const uint16_t bcw_ilt_noisea2[BCW_ILT_NOISEA2_SIZE] = { - 0x0001, 0x0001, 0x0001, 0xFFFE, - 0xFFFE, 0x3FFF, 0x1000, 0x0393, -}; - -const uint16_t bcw_ilt_noisea3[BCW_ILT_NOISEA3_SIZE] = { - 0x4C4C, 0x4C4C, 0x4C4C, 0x2D36, - 0x4C4C, 0x4C4C, 0x4C4C, 0x2D36, -}; - -const uint16_t bcw_ilt_finefreqg[BCW_ILT_FINEFREQG_SIZE] = { - 0x0089, 0x02E9, 0x0409, 0x04E9, /* 0 */ - 0x05A9, 0x0669, 0x0709, 0x0789, - 0x0829, 0x08A9, 0x0929, 0x0989, - 0x0A09, 0x0A69, 0x0AC9, 0x0B29, - 0x0BA9, 0x0BE9, 0x0C49, 0x0CA9, /* 16 */ - 0x0D09, 0x0D69, 0x0DA9, 0x0E09, - 0x0E69, 0x0EA9, 0x0F09, 0x0F49, - 0x0FA9, 0x0FE9, 0x1029, 0x1089, - 0x10C9, 0x1109, 0x1169, 0x11A9, /* 32 */ - 0x11E9, 0x1229, 0x1289, 0x12C9, - 0x1309, 0x1349, 0x1389, 0x13C9, - 0x1409, 0x1449, 0x14A9, 0x14E9, - 0x1529, 0x1569, 0x15A9, 0x15E9, /* 48 */ - 0x1629, 0x1669, 0x16A9, 0x16E8, - 0x1728, 0x1768, 0x17A8, 0x17E8, - 0x1828, 0x1868, 0x18A8, 0x18E8, - 0x1928, 0x1968, 0x19A8, 0x19E8, /* 64 */ - 0x1A28, 0x1A68, 0x1AA8, 0x1AE8, - 0x1B28, 0x1B68, 0x1BA8, 0x1BE8, - 0x1C28, 0x1C68, 0x1CA8, 0x1CE8, - 0x1D28, 0x1D68, 0x1DC8, 0x1E08, /* 80 */ - 0x1E48, 0x1E88, 0x1EC8, 0x1F08, - 0x1F48, 0x1F88, 0x1FE8, 0x2028, - 0x2068, 0x20A8, 0x2108, 0x2148, - 0x2188, 0x21C8, 0x2228, 0x2268, /* 96 */ - 0x22C8, 0x2308, 0x2348, 0x23A8, - 0x23E8, 0x2448, 0x24A8, 0x24E8, - 0x2548, 0x25A8, 0x2608, 0x2668, - 0x26C8, 0x2728, 0x2787, 0x27E7, /* 112 */ - 0x2847, 0x28C7, 0x2947, 0x29A7, - 0x2A27, 0x2AC7, 0x2B47, 0x2BE7, - 0x2CA7, 0x2D67, 0x2E47, 0x2F67, - 0x3247, 0x3526, 0x3646, 0x3726, /* 128 */ - 0x3806, 0x38A6, 0x3946, 0x39E6, - 0x3A66, 0x3AE6, 0x3B66, 0x3BC6, - 0x3C45, 0x3CA5, 0x3D05, 0x3D85, - 0x3DE5, 0x3E45, 0x3EA5, 0x3EE5, /* 144 */ - 0x3F45, 0x3FA5, 0x4005, 0x4045, - 0x40A5, 0x40E5, 0x4145, 0x4185, - 0x41E5, 0x4225, 0x4265, 0x42C5, - 0x4305, 0x4345, 0x43A5, 0x43E5, /* 160 */ - 0x4424, 0x4464, 0x44C4, 0x4504, - 0x4544, 0x4584, 0x45C4, 0x4604, - 0x4644, 0x46A4, 0x46E4, 0x4724, - 0x4764, 0x47A4, 0x47E4, 0x4824, /* 176 */ - 0x4864, 0x48A4, 0x48E4, 0x4924, - 0x4964, 0x49A4, 0x49E4, 0x4A24, - 0x4A64, 0x4AA4, 0x4AE4, 0x4B23, - 0x4B63, 0x4BA3, 0x4BE3, 0x4C23, /* 192 */ - 0x4C63, 0x4CA3, 0x4CE3, 0x4D23, - 0x4D63, 0x4DA3, 0x4DE3, 0x4E23, - 0x4E63, 0x4EA3, 0x4EE3, 0x4F23, - 0x4F63, 0x4FC3, 0x5003, 0x5043, /* 208 */ - 0x5083, 0x50C3, 0x5103, 0x5143, - 0x5183, 0x51E2, 0x5222, 0x5262, - 0x52A2, 0x52E2, 0x5342, 0x5382, - 0x53C2, 0x5402, 0x5462, 0x54A2, /* 224 */ - 0x5502, 0x5542, 0x55A2, 0x55E2, - 0x5642, 0x5682, 0x56E2, 0x5722, - 0x5782, 0x57E1, 0x5841, 0x58A1, - 0x5901, 0x5961, 0x59C1, 0x5A21, /* 240 */ - 0x5AA1, 0x5B01, 0x5B81, 0x5BE1, - 0x5C61, 0x5D01, 0x5D80, 0x5E20, - 0x5EE0, 0x5FA0, 0x6080, 0x61C0, -}; - -const uint16_t bcw_ilt_noiseg1[BCW_ILT_NOISEG1_SIZE] = { - 0x013C, 0x01F5, 0x031A, 0x0631, - 0x0001, 0x0001, 0x0001, 0x0001, -}; - -const uint16_t bcw_ilt_noiseg2[BCW_ILT_NOISEG2_SIZE] = { - 0x5484, 0x3C40, 0x0000, 0x0000, - 0x0000, 0x0000, 0x0000, 0x0000, -}; - -const uint16_t bcw_ilt_noisescaleg1[BCW_ILT_NOISESCALEG_SIZE] = { - 0x6C77, 0x5162, 0x3B40, 0x3335, /* 0 */ - 0x2F2D, 0x2A2A, 0x2527, 0x1F21, - 0x1A1D, 0x1719, 0x1616, 0x1414, - 0x1414, 0x1400, 0x1414, 0x1614, - 0x1716, 0x1A19, 0x1F1D, 0x2521, /* 16 */ - 0x2A27, 0x2F2A, 0x332D, 0x3B35, - 0x5140, 0x6C62, 0x0077, -}; - -const uint16_t bcw_ilt_noisescaleg3[BCW_ILT_NOISESCALEG_SIZE] = { - 0xA4A4, 0xA4A4, 0xA4A4, 0xA4A4, /* 0 */ - 0xA4A4, 0xA4A4, 0xA4A4, 0xA4A4, - 0xA4A4, 0xA4A4, 0xA4A4, 0xA4A4, - 0xA4A4, 0xA400, 0xA4A4, 0xA4A4, - 0xA4A4, 0xA4A4, 0xA4A4, 0xA4A4, /* 16 */ - 0xA4A4, 0xA4A4, 0xA4A4, 0xA4A4, - 0xA4A4, 0xA4A4, 0x00A4, -}; - -const uint16_t bcw_ilt_noisescaleg2[BCW_ILT_NOISESCALEG_SIZE] = { - 0xD8DD, 0xCBD4, 0xBCC0, 0XB6B7, /* 0 */ - 0xB2B0, 0xADAD, 0xA7A9, 0x9FA1, - 0x969B, 0x9195, 0x8F8F, 0x8A8A, - 0x8A8A, 0x8A00, 0x8A8A, 0x8F8A, - 0x918F, 0x9695, 0x9F9B, 0xA7A1, /* 16 */ - 0xADA9, 0xB2AD, 0xB6B0, 0xBCB7, - 0xCBC0, 0xD8D4, 0x00DD, -}; - -/* - * Helper routines - */ -void -bcw_shm_ctl_word(struct bcw_softc *sc, uint16_t routing, uint16_t offset) -{ - uint32_t control; - - control = routing; - control <<= 16; - control |= offset; - - BCW_WRITE(sc, BCW_MMIO_SHM_CONTROL, control); -} - -uint16_t -bcw_shm_read16(struct bcw_softc *sc, uint16_t routing, uint16_t offset) -{ - if (routing == BCW_SHM_SHARED) { - if (offset & 0x0003) { - bcw_shm_ctl_word(sc, routing, offset >> 2); - - return (BCW_READ16(sc, BCW_MMIO_SHM_DATA_UNALIGNED)); - } - offset >>= 2; - } - bcw_shm_ctl_word(sc, routing, offset); - - return (BCW_READ16(sc, BCW_MMIO_SHM_DATA)); -} - -void -bcw_shm_write16(struct bcw_softc *sc, uint16_t routing, uint16_t offset, - uint16_t val) -{ - if (routing == BCW_SHM_SHARED) { - if (offset & 0x0003) { - bcw_shm_ctl_word(sc, routing, offset >> 2); - BCW_WRITE16(sc, BCW_MMIO_SHM_DATA_UNALIGNED, val); - return; - } - offset >>= 2; - } - bcw_shm_ctl_word(sc, routing, offset); - BCW_WRITE16(sc, BCW_MMIO_SHM_DATA, val); -} - -uint32_t -bcw_shm_read32(struct bcw_softc *sc, uint16_t routing, uint16_t offset) -{ - uint32_t r; - - if (routing == BCW_SHM_SHARED) { - if (offset & 0x003) { - bcw_shm_ctl_word(sc, routing, offset >> 2); - r = BCW_READ16(sc, BCW_MMIO_SHM_DATA_UNALIGNED); - r <<= 16; - bcw_shm_ctl_word(sc, routing, (offset >> 2) + 1); - r |= BCW_READ16(sc, BCW_MMIO_SHM_DATA); - return (r); - } - offset >>= 2; - } - bcw_shm_ctl_word(sc, routing, offset); - r = BCW_READ(sc, BCW_MMIO_SHM_DATA); - - return (r); -} - -void -bcw_shm_write32(struct bcw_softc *sc, uint16_t routing, uint16_t offset, - uint32_t val) -{ - if (routing == BCW_SHM_SHARED) { - if (offset & 0x0003) { - bcw_shm_ctl_word(sc, routing, offset >> 2); - BCW_WRITE16(sc, BCW_MMIO_SHM_DATA_UNALIGNED, - (val >> 16) & 0xffff); - bcw_shm_ctl_word(sc, routing, (offset >> 2) + 1); - BCW_WRITE16(sc, BCW_MMIO_SHM_DATA, val & 0xffff); - return; - } - offset >>= 2; - } - bcw_shm_ctl_word(sc, routing, offset); - BCW_WRITE(sc, BCW_MMIO_SHM_DATA, val); -} - -void -bcw_radio_write16(struct bcw_softc *sc, uint16_t offset, uint16_t val) -{ - BCW_WRITE16(sc, BCW_MMIO_RADIO_CONTROL, offset); - BCW_WRITE16(sc, BCW_MMIO_RADIO_DATA_LOW, val); -} - -int -bcw_radio_read16(struct bcw_softc *sc, uint16_t offset) -{ - switch (sc->sc_phy_type) { - case BCW_PHY_TYPEA: - offset |= 0x0040; - break; - case BCW_PHY_TYPEB: - if (sc->sc_radio_ver == 0x2053) { - if (offset < 0x70) - offset += 0x80; - else if (offset < 0x80) - offset += 0x70; - } else if (sc->sc_radio_ver == 0x2050) - offset |= 0x80; - else - return (0); - break; - case BCW_PHY_TYPEG: - offset |= 0x80; - break; - } - - BCW_WRITE16(sc, BCW_MMIO_RADIO_CONTROL, offset); - - return (BCW_READ16(sc, BCW_MMIO_RADIO_DATA_LOW)); -} - -void -bcw_phy_write16(struct bcw_softc *sc, uint16_t offset, uint16_t val) -{ - BCW_WRITE16(sc, BCW_MMIO_PHY_CONTROL, offset); - BCW_WRITE16(sc, BCW_MMIO_PHY_DATA, val); -} - -int -bcw_phy_read16(struct bcw_softc *sc, uint16_t offset) -{ - BCW_WRITE16(sc, BCW_MMIO_PHY_CONTROL, offset); - - return (BCW_READ16(sc, BCW_MMIO_PHY_DATA)); -} - -void -bcw_ram_write(struct bcw_softc *sc, uint16_t offset, uint32_t val) -{ - uint32_t status; - - status = BCW_READ(sc, BCW_MMIO_SBF); - if (!(status & BCW_SBF_REGISTER_BYTESWAP)) - val = swap32(val); - - BCW_WRITE(sc, BCW_MMIO_RAM_CONTROL, offset); - BCW_WRITE(sc, BCW_MMIO_RAM_DATA, val); -} - -int -bcw_lv(int number, int min, int max) -{ - if (number < min) - return (min); - else if (number > max) - return (max); - else - return (number); -} - -uint16_t -bcw_flip_4bit(uint16_t val) -{ - uint16_t flip = 0; - - flip |= (val & 0x0001) << 3; - flip |= (val & 0x0002) << 1; - flip |= (val & 0x0004) >> 1; - flip |= (val & 0x0008) >> 3; - - return (flip); -} - -/* - * Dummy Transmission - * - * http://bcm-specs.sipsolutions.net/DummyTransmission - */ -void -bcw_dummy_transmission(struct bcw_softc *sc) -{ - unsigned int i, max_loop; - uint16_t val = 0; - uint32_t buffer[5] = { - 0x00000000, - 0x00000400, - 0x00000000, - 0x00000001, - 0x00000000 - }; - - switch (sc->sc_phy_type) { - case BCW_PHY_TYPEA: - max_loop = 0x1e; - buffer[0] = 0xcc010200; - break; - case BCW_PHY_TYPEB: - case BCW_PHY_TYPEG: - max_loop = 0xfa; - buffer[0] = 0x6e840b00; - break; - default: - /* XXX panic()? */ - return; - } - - for (i = 0; i < 5; i++) - bcw_ram_write(sc, i * 4, buffer[i]); - - BCW_READ(sc, BCW_MMIO_SBF); - - BCW_WRITE16(sc, 0x0568, 0); - BCW_WRITE16(sc, 0x07c0, 0); - BCW_WRITE16(sc, 0x050c, sc->sc_phy_type == BCW_PHY_TYPEA ? 1 : 0); - - BCW_WRITE16(sc, 0x0508, 0); - BCW_WRITE16(sc, 0x050a, 0); - BCW_WRITE16(sc, 0x054c, 0); - BCW_WRITE16(sc, 0x056a, 0x0014); - BCW_WRITE16(sc, 0x0568, 0x0826); - BCW_WRITE16(sc, 0x0500, 0); - BCW_WRITE16(sc, 0x0502, 0x0030); - - if (sc->sc_radio_ver == 0x2050 && sc->sc_radio_rev <= 0x5) - bcw_radio_write16(sc, 0x0051, 0x0017); - for (i = 0; i < max_loop; i++) { - val = BCW_READ16(sc, 0x050e); - if (val & 0x0080) - break; - delay(10); - } - for (i = 0; i < 0x0a; i++) { - val = BCW_READ16(sc, 0x050e); - if (val & 0x0400) - break; - delay(10); - } - for (i = 0; i < 0x0a; i++) { - val = BCW_READ16(sc, 0x0690); - if (!(val & 0x0100)) - break; - delay(10); - } - if (sc->sc_radio_ver == 0x2050 && sc->sc_radio_rev <= 0x5) - bcw_radio_write16(sc, 0x0051, 0x0037); -} - -struct bcw_lopair * -bcw_get_lopair(struct bcw_softc *sc, uint16_t radio_atten, - uint16_t baseband_atten) -{ - return (sc->sc_phy_lopairs + (radio_atten + 14 * - (baseband_atten / 2))); -} - -void -bcw_stack_save(uint32_t *_stackptr, size_t *stackidx, uint8_t id, - uint16_t offset, uint16_t val) -{ - uint32_t *stackptr = &(_stackptr[*stackidx]); - - *stackptr = offset; - *stackptr |= ((uint32_t)id) << 12; - *stackptr |= ((uint32_t)val) << 16; - (*stackidx)++; -} - -uint16_t -bcw_stack_restore(uint32_t *stackptr, uint8_t id, uint16_t offset) -{ - size_t i; - - for (i = 0; i < BCW_INTERFSTACK_SIZE; i++, stackptr++) { - if ((*stackptr & 0x00000fff) != offset) - continue; - if (((*stackptr & 0x0000f000) >> 12) != id) - continue; - return (((*stackptr & 0xffff0000) >> 16)); - } - - return (0); -} - -int -bcw_using_pio(struct bcw_softc *sc) -{ - return (sc->sc_using_pio); -} - -void -bcw_rate_memory_write(struct bcw_softc *sc, uint16_t rate, int is_ofdm) -{ - uint16_t offset; - - if (is_ofdm) { - offset = 0x480; - offset += (bcw_xmit_plcp_get_ratecode_ofdm(rate) & 0x000f) * 2; - } else { - offset = 0x4c0; - offset += (bcw_xmit_plcp_get_ratecode_cck(rate) & 0x000f) * 2; - } - bcw_shm_write16(sc, BCW_SHM_SHARED, offset + 0x20, - bcw_shm_read16(sc, BCW_SHM_SHARED, offset)); -} - -void -bcw_rate_memory_init(struct bcw_softc *sc) -{ - switch (sc->sc_phy_type) { - case BCW_PHY_TYPEA: - case BCW_PHY_TYPEG: - bcw_rate_memory_write(sc, - ieee80211_std_rateset_11g.rs_rates[0], 1); - bcw_rate_memory_write(sc, - ieee80211_std_rateset_11g.rs_rates[1], 1); - bcw_rate_memory_write(sc, - ieee80211_std_rateset_11g.rs_rates[2], 1); - bcw_rate_memory_write(sc, - ieee80211_std_rateset_11g.rs_rates[3], 1); - bcw_rate_memory_write(sc, - ieee80211_std_rateset_11g.rs_rates[4], 1); - bcw_rate_memory_write(sc, - ieee80211_std_rateset_11g.rs_rates[5], 1); - bcw_rate_memory_write(sc, - ieee80211_std_rateset_11g.rs_rates[6], 1); - bcw_rate_memory_write(sc, - ieee80211_std_rateset_11g.rs_rates[7], 1); - case BCW_PHY_TYPEB: - bcw_rate_memory_write(sc, - ieee80211_std_rateset_11b.rs_rates[0], 0); - bcw_rate_memory_write(sc, - ieee80211_std_rateset_11b.rs_rates[1], 0); - bcw_rate_memory_write(sc, - ieee80211_std_rateset_11b.rs_rates[2], 0); - bcw_rate_memory_write(sc, - ieee80211_std_rateset_11b.rs_rates[3], 0); - default: - /* XXX panic()? */ - break; - } -} - -/* - * Attach device - * - * http://bcm-specs.sipsolutions.net/DeviceAttach - */ -void -bcw_attach(struct bcw_softc *sc) -{ - struct ieee80211com *ic = &sc->sc_ic; - struct ifnet *ifp = &ic->ic_if; - int i; - uint32_t sbval; - uint32_t core_id, core_rev, core_vendor; - - /* power on cardbus socket */ - if (sc->sc_enable) - sc->sc_enable(sc); - - DPRINTF(("\n%s: BoardVendor=0x%x, BoardType=0x%x, BoardRev=0x%x\n", - sc->sc_dev.dv_xname, - sc->sc_board_vendor, sc->sc_board_type, sc->sc_board_rev)); - - /* - * Attach chipset to backplane - */ - bcw_pc_crystal_on(sc); - (sc->sc_conf_write)(sc, PCI_COMMAND_STATUS_REG, - (sc->sc_conf_read)(sc, PCI_COMMAND_STATUS_REG) & - ~PCI_STATUS_TARGET_TARGET_ABORT); - - /* - * Try and change to the ChipCommon Core - */ - if (bcw_change_core(sc, 0) == 0) { - DPRINTF(("%s: Selected ChipCommon Core\n", - sc->sc_dev.dv_xname)); - sc->sc_chip_common_capa = - BCW_READ(sc, BCW_CHIPCOMMON_CAPABILITIES); /* XXX */ - } - - /* - * Core ID REG, this is either the default wireless core (0x812) or - * a ChipCommon core that was successfully selected above - */ - sbval = BCW_READ(sc, BCW_CIR_SBID_HI); - core_id = (sbval & 0x8ff0) >> 4; - core_rev = (sbval & 0x7000) >> 8; - core_rev |= (sbval & 0xf); - core_vendor = (sbval & 0xffff0000) >> 16; - DPRINTF(("%s: CoreID=0x%x, CoreRev=0x%x, CoreVendor=0x%x\n", - sc->sc_dev.dv_xname, core_id, core_rev, core_vendor)); - - /* - * If we successfully got a commoncore, and the corerev=4 or >=6 - * get the number of cores from the chipid reg - */ - if (core_id == BCW_CORE_COMMON) { - sc->sc_havecommon = 1; - - /* powercontrol init is done if a common core exists */ - bcw_pc_init(sc); - - sbval = BCW_READ(sc, BCW_CORE_COMMON_CHIPID); - sc->sc_chip_id = (sbval & 0x0000ffff); - sc->sc_chip_rev = (sbval & 0x000f0000) >> 16; - sc->sc_chip_pkg = (sbval & 0x00f00000) >> 20; - - if (core_rev >= 4) - sc->sc_numcores = (sbval & 0x0f000000) >> 24; - else { - switch (sc->sc_chip_id) { - case 0x4710: - case 0x4610: - case 0x4704: - sc->sc_numcores = 9; - break; - case 0x4310: - sc->sc_numcores = 8; - break; - case 0x5365: - sc->sc_numcores = 7; - break; - case 0x4306: - sc->sc_numcores = 6; - break; - case 0x4307: - case 0x4301: - sc->sc_numcores = 5; - break; - case 0x4402: - sc->sc_numcores = 3; - break; - default: - /* set to max */ - sc->sc_numcores = BCW_MAX_CORES; - } - } - } else { /* No CommonCore, set chipid, cores, rev based on product id */ - sc->sc_core_common = NULL; - sc->sc_havecommon = 0; - switch (sc->sc_prodid) { - case 0x4710: - case 0x4711: - case 0x4712: - case 0x4713: - case 0x4714: - case 0x4715: - sc->sc_chip_id = 0x4710; - sc->sc_numcores = 9; - break; - case 0x4610: - case 0x4611: - case 0x4612: - case 0x4613: - case 0x4614: - case 0x4615: - sc->sc_chip_id = 0x4610; - sc->sc_numcores = 9; - break; - case 0x4402: - case 0x4403: - sc->sc_chip_id = 0x4402; - sc->sc_numcores = 3; - break; - case 0x4305: - case 0x4306: - case 0x4307: - sc->sc_chip_id = 0x4307; - sc->sc_numcores = 5; - break; - case 0x4301: - sc->sc_chip_id = 0x4301; - sc->sc_numcores = 5; - break; - default: - sc->sc_chip_id = sc->sc_prodid; - /* Set to max */ - sc->sc_numcores = BCW_MAX_CORES; - } - } - - DPRINTF(("%s: ChipID=0x%x, ChipRev=0x%x, ChipPkg=0x%x, NumCores=%d\n", - sc->sc_dev.dv_xname, - sc->sc_chip_id, sc->sc_chip_rev, sc->sc_chip_pkg, - sc->sc_numcores)); - - bcw_pc_set_clock(sc, BCW_PCTL_CLK_FAST); - - bcw_phy_connect(sc, 0); - - /* - * Get SPROM values and save them where they belong - */ - bcw_sprom_get(sc); - - /* Identify each core and reset the 802.11 cores */ - for (i = 0; i < sc->sc_numcores; i++) { - if (bcw_change_core(sc, i)) { - printf("%s: Failed change to core %d\n", - sc->sc_dev.dv_xname, i); - continue; - } - - sbval = BCW_READ(sc, BCW_CIR_SBID_HI); - sc->sc_core[i].id = (sbval & 0x00008ff0) >> 4; - sc->sc_core[i].rev = ((sbval & 0x00007000) >> 8 | - (sbval & 0x0000000f)); - - switch (sc->sc_core[i].id) { - case BCW_CORE_COMMON: - sc->sc_core_common = &sc->sc_core[i]; - break; -#if 0 - case BCW_CORE_PCI: - (sc->sc_ca == NULL) - sc->sc_core_bus = &sc->sc_core[i]; - break; - case BCW_CORE_PCMCIA: - if (sc->sc_pa == NULL) - sc->sc_core_bus = &sc->sc_core[i]; - break; -#endif - case BCW_CORE_80211: - sc->sc_core_80211 = &sc->sc_core[i]; - bcw_80211_core_reset(sc, 1); - if (bcw_phy_get(sc)) - return; - if (bcw_radio_get(sc)) - return; - if (bcw_validate_chip_access(sc)) - return; - bcw_radio_off(sc); - bcw_microcode_init_fbf(sc); - break; - case BCW_CORE_NONEXIST: - sc->sc_numcores = i + 1; - break; - default: - /* ignore all other core types */ - break; - } - - sc->sc_core[i].index = i; - - DPRINTF(("%s: core %d is type 0x%x rev %d\n", - sc->sc_dev.dv_xname, sc->sc_core[i].index, - sc->sc_core[i].id, sc->sc_core[i].rev)); - } - - /* turn crystal off */ - bcw_pc_crystal_off(sc); - - /* - * Switch to the 80211 core - */ - bcw_change_core(sc, sc->sc_core_80211->index); - - /* - * Set MAC address - */ - if (!IEEE80211_IS_MULTICAST(sc->sc_sprom.et1macaddr)) - memcpy(ic->ic_myaddr, sc->sc_sprom.et1macaddr, ETHER_ADDR_LEN); - else - memcpy(ic->ic_myaddr, sc->sc_sprom.il0macaddr, ETHER_ADDR_LEN); - printf(", address %s\n", ether_sprintf(ic->ic_myaddr)); - - /* - * Initialize the TSSI to DBM table - * The method is described at - * http://bcm-specs.sipsolutions.net/TSSI_to_DBM_Table - * but I suspect there's a standard way to do it in the 80211 stuff - */ - - /* - * Allocate DMA-safe memory for ring descriptors. - * The receive and transmit rings are 4k aligned. - */ - bcw_alloc_rx_ring(sc, &sc->sc_rxring, BCW_RX_RING_COUNT); - bcw_alloc_tx_ring(sc, &sc->sc_txring, BCW_TX_RING_COUNT); - - ic->ic_phytype = IEEE80211_T_OFDM; - ic->ic_opmode = IEEE80211_M_STA; /* default to BSS mode */ - ic->ic_state = IEEE80211_S_INIT; - - /* Set device capabilities - keep it simple */ - ic->ic_caps = IEEE80211_C_IBSS; /* IBSS mode supported */ - - /* Set supported rates */ - ic->ic_sup_rates[IEEE80211_MODE_11B] = ieee80211_std_rateset_11b; - ic->ic_sup_rates[IEEE80211_MODE_11G] = ieee80211_std_rateset_11g; - - /* Set supported channels */ - for (i = 1; i <= 14; i++) { - ic->ic_channels[i].ic_freq = - ieee80211_ieee2mhz(i, IEEE80211_CHAN_2GHZ); - ic->ic_channels[i].ic_flags = - IEEE80211_CHAN_CCK | IEEE80211_CHAN_OFDM | - IEEE80211_CHAN_DYN | IEEE80211_CHAN_2GHZ; - } - - /* IBSS channel undefined for now */ - ic->ic_ibss_chan = &ic->ic_channels[0]; - - ifp->if_softc = sc; - ifp->if_init = bcw_init; - ifp->if_ioctl = bcw_ioctl; - ifp->if_start = bcw_start; - ifp->if_watchdog = bcw_watchdog; - ifp->if_flags = IFF_SIMPLEX | IFF_BROADCAST | IFF_MULTICAST; - strlcpy(ifp->if_xname, sc->sc_dev.dv_xname, IFNAMSIZ); - IFQ_SET_READY(&ifp->if_snd); - - /* Attach the interface */ - if_attach(ifp); - ieee80211_ifattach(ifp); - - /* override state transition machine */ - sc->sc_newstate = ic->ic_newstate; - ic->ic_newstate = bcw_newstate; - - ieee80211_media_init(ifp, bcw_media_change, bcw_media_status); - - timeout_set(&sc->sc_timeout, bcw_tick, sc); -} - -/* handle media, and ethernet requests */ -int -bcw_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data) -{ - struct bcw_softc *sc = ifp->if_softc; - struct ieee80211com *ic = &sc->sc_ic; - struct ifreq *ifr = (struct ifreq *) data; - struct ifaddr *ifa = (struct ifaddr *)data; - int s, error = 0; - - s = splnet(); - - switch (cmd) { - case SIOCSIFADDR: - ifp->if_flags |= IFF_UP; - - switch (ifa->ifa_addr->sa_family) { -#ifdef INET - case AF_INET: - //bcw_init(ifp); - //arp_ifinit(&sc->bcw_ac, ifa); /* XXX */ - break; -#endif /* INET */ - default: - //bcw_init(ifp); - break; - } - break; - case SIOCSIFFLAGS: - if ((ifp->if_flags & IFF_UP) && - (!(ifp->if_flags & IFF_RUNNING))) - bcw_init(ifp); - else if (ifp->if_flags & IFF_RUNNING) - bcw_stop(ifp, 1); - break; - case SIOCADDMULTI: - case SIOCDELMULTI: - error = (cmd == SIOCADDMULTI) ? - ether_addmulti(ifr, &ic->ic_ac) : - ether_delmulti(ifr, &ic->ic_ac); - - if (error == ENETRESET) - error = 0; - break; - - case SIOCG80211TXPOWER: - /* - * If the hardware radio transmitter switch is off, report a - * tx power of IEEE80211_TXPOWER_MIN to indicate that radio - * transmitter is killed. - */ - break; - - default: - error = ieee80211_ioctl(ifp, cmd, data); - break; - } - - if (error == ENETRESET) { - if ((ifp->if_flags & (IFF_UP | IFF_RUNNING)) == - (IFF_UP | IFF_RUNNING)) - bcw_init(ifp); - error = 0; - } - - splx(s); - return error; -} - -/* - * Start packet transmission on the interface - */ -void -bcw_start(struct ifnet *ifp) -{ -#if 0 - struct malo_softc *sc = ifp->if_softc; - struct ieee80211com *ic = &sc->sc_ic; - struct ieee80211_node *ni; - struct mbuf *m; -#endif - - /* if device is not up return directly */ - if ((ifp->if_flags & (IFF_RUNNING | IFF_OACTIVE)) != IFF_RUNNING) - return; -} - -/* Watchdog timer handler. */ -void -bcw_watchdog(struct ifnet *ifp) -{ - struct bcw_softc *sc = ifp->if_softc; - - printf("%s: device timeout\n", sc->sc_dev.dv_xname); - ifp->if_oerrors++; - - (void) bcw_init(ifp); - - /* Try to get more packets going. */ - bcw_start(ifp); -} - -void -bcw_set_opmode(struct ifnet *ifp) -{ - struct bcw_softc *sc = ifp->if_softc; - struct ieee80211com *ic = &sc->sc_ic; - uint32_t status; - uint16_t val; - - /* TODO */ - - status = BCW_READ(sc, BCW_MMIO_SBF); - /* reset status to infrastructure mode */ - status &= ~(BCW_SBF_AP | BCW_SBF_MONITOR); - status &= ~BCW_SBF_PROMISC; - status |= BCW_SBF_ADHOC; - status |= BCW_SBF_PROMISC; /* XXX */ - - switch (ic->ic_opmode) { - case IEEE80211_M_MONITOR: - status |= BCW_SBF_MONITOR; - status |= BCW_SBF_PROMISC; - break; - case IEEE80211_M_IBSS: - status &= ~BCW_SBF_ADHOC; - break; - case IEEE80211_M_HOSTAP: - status |= BCW_SBF_AP; - break; - case IEEE80211_M_STA: - /* nothing todo here */ - break; - default: - printf("%s: unknown mode in bcw_set_opmode()\n", - sc->sc_dev.dv_xname); - break; - } - if (ifp->if_flags & IFF_PROMISC) - status |= BCW_SBF_PROMISC; - - val = 0x0002; - if (ic->ic_opmode != IEEE80211_M_IBSS && - ic->ic_opmode != IEEE80211_M_HOSTAP) { - if (sc->sc_chip_id == 0x4306 && sc->sc_chip_rev == 3) - val = 0x0064; - else - val = 0x0032; - } - BCW_WRITE16(sc, 0x0612, val); -} - -void -bcw_macfilter_set(struct bcw_softc *sc, uint16_t offset, const uint8_t *mac) -{ - uint16_t data; - - offset |= 0x0020; - BCW_WRITE16(sc, BCW_MMIO_MACFILTER_CONTROL, offset); - - data = mac[0]; - data |= mac[1] << 8; - BCW_WRITE16(sc, BCW_MMIO_MACFILTER_DATA, data); - data = mac[2]; - data |= mac[3] << 8; - BCW_WRITE16(sc, BCW_MMIO_MACFILTER_DATA, data); - data = mac[4]; - data |= mac[5] << 8; - BCW_WRITE16(sc, BCW_MMIO_MACFILTER_DATA, data); -} - -void -bcw_macfilter_clear(struct bcw_softc *sc, uint16_t offset) -{ - const uint8_t zero_addr[ETHER_ADDR_LEN] = { 0 }; - - bcw_macfilter_set(sc, offset, zero_addr); -} - -/* - * Write MAC and BSSID into template RAM - * - * http://bcm-specs.sipsolutions.net/TemplateRam - */ -void -bcw_templateram_set(struct bcw_softc *sc) -{ - struct ieee80211com *ic = &sc->sc_ic; - uint8_t mac_bssid[ETHER_ADDR_LEN * 2]; - int i; - - bcopy(ic->ic_myaddr, mac_bssid, ETHER_ADDR_LEN); - bcopy(ic->ic_bss->ni_bssid, mac_bssid + ETHER_ADDR_LEN, ETHER_ADDR_LEN); - - /* - * We write our MAC and BSSID at three different addresses in the - * template RAM, in 4 byte blocks. - */ - for (i = 0; i < sizeof(mac_bssid); i += sizeof(uint32_t)) - bcw_ram_write(sc, 0x20 + i, *((int32_t *)(mac_bssid + i))); - for (i = 0; i < sizeof(mac_bssid); i += sizeof(uint32_t)) - bcw_ram_write(sc, 0x78 + i, *((int32_t *)(mac_bssid + i))); - for (i = 0; i < sizeof(mac_bssid); i += sizeof(uint32_t)) - bcw_ram_write(sc, 0x478 + i, *((int32_t *)(mac_bssid + i))); -} - -/* - * Enable MAC on a PHY - * - * http://bcm-specs.sipsolutions.net/EnableMAC - */ -void -bcw_mac_enable(struct bcw_softc *sc) -{ - BCW_WRITE(sc, BCW_MMIO_SBF, - BCW_READ(sc, BCW_MMIO_SBF) | BCW_SBF_MAC_ENABLED); - BCW_WRITE(sc, BCW_MMIO_GIR, BCW_INTR_READY); - BCW_READ(sc, BCW_MMIO_SBF); /* dummy read */ - BCW_READ(sc, BCW_MMIO_GIR); /* dummy read */ - bcw_pc_saving_ctl_bits(sc, -1, -1); -} - -/* - * Disable MAC on a PHY - * - * http://bcm-specs.sipsolutions.net/SuspendMAC - */ -void -bcw_mac_disable(struct bcw_softc *sc) -{ - int i; - uint32_t tmp; - - bcw_pc_saving_ctl_bits(sc, -1, -1); - BCW_WRITE(sc, BCW_MMIO_SBF, - BCW_READ(sc, BCW_MMIO_SBF) & ~BCW_SBF_MAC_ENABLED); - BCW_READ(sc, BCW_MMIO_GIR); /* dummy read */ - for (i = 10000; i; i--) { - tmp = BCW_READ(sc, BCW_MMIO_GIR); - if (tmp & BCW_INTR_READY) - return; - delay(1); - } - DPRINTF(("%s: MAC disable failed!\n", sc->sc_dev.dv_xname)); -} - -uint32_t -bcw_intr_enable(struct bcw_softc *sc, uint32_t mask) -{ - uint32_t old_mask; - - old_mask = BCW_READ(sc, BCW_MMIO_GIM); - BCW_WRITE(sc, BCW_MMIO_GIM, old_mask | mask); - - DPRINTF(("%s: interrupts enabled\n", sc->sc_dev.dv_xname)); - - return (old_mask); -} - -uint32_t -bcw_intr_disable(struct bcw_softc *sc, uint32_t mask) -{ - uint32_t old_mask; - - old_mask = BCW_READ(sc, BCW_MMIO_GIM); - BCW_WRITE(sc, BCW_MMIO_GIM, old_mask & ~mask); - - DPRINTF(("%s: interrupts disabled\n", sc->sc_dev.dv_xname)); - - return (old_mask); -} - -int -bcw_intr(void *arg) -{ - struct bcw_softc *sc = arg; - uint32_t reason; - uint32_t dma_reason[6]; - - if (!(sc->sc_flags & BCW_FLAGS_INITIALIZED)) - return (0); - - reason = BCW_READ(sc, BCW_MMIO_GIR); - if (reason == 0xffffffff) - return (0); - - reason &= BCW_READ(sc, BCW_MMIO_GIM); - if (!reason) - return (0); - - dma_reason[0] = BCW_READ(sc, BCW_MMIO_DMA0_REASON) & 0x0001dc00; - dma_reason[1] = BCW_READ(sc, BCW_MMIO_DMA1_REASON) & 0x0000dc00; - dma_reason[2] = BCW_READ(sc, BCW_MMIO_DMA2_REASON) & 0x0000dc00; - dma_reason[3] = BCW_READ(sc, BCW_MMIO_DMA3_REASON) & 0x0001dc00; - dma_reason[4] = BCW_READ(sc, BCW_MMIO_DMA4_REASON) & 0x0000dc00; - dma_reason[5] = BCW_READ(sc, BCW_MMIO_DMA5_REASON) & 0x0000dc00; - - /* ACK interrupt */ - BCW_WRITE(sc, BCW_MMIO_GIR, reason); - BCW_WRITE(sc, BCW_MMIO_GIM, dma_reason[0]); - BCW_WRITE(sc, BCW_MMIO_GIM, dma_reason[1]); - BCW_WRITE(sc, BCW_MMIO_GIM, dma_reason[2]); - BCW_WRITE(sc, BCW_MMIO_GIM, dma_reason[3]); - BCW_WRITE(sc, BCW_MMIO_GIM, dma_reason[4]); - BCW_WRITE(sc, BCW_MMIO_GIM, dma_reason[5]); - - bcw_intr_disable(sc, BCW_INTR_ALL); - - if (reason & BCW_INTR_PS) { - printf("handle PS intr\n"); - bcw_pc_saving_ctl_bits(sc, -1, -1); - } else - printf("INTR ALERT!!!\n"); - - bcw_intr_enable(sc, BCW_INTR_INITIAL); - - return (1); -} - -/* - * Receive interrupt handler - */ -void -bcw_rxintr(struct bcw_softc *sc) -{ -#if 0 - struct ieee80211com *ic = &sc->sc_ic; - struct ifnet *ifp = &ic->ic_if; -#endif -} - -/* - * Transmit interrupt handler - */ -void -bcw_txintr(struct bcw_softc *sc) -{ -#if 0 - struct ieee80211com *ic = &sc->sc_ic; - struct ifnet *ifp = &ic->ic_if; -#endif - -} - -/* - * Initialize the interface - * - * http://bcm-specs.sipsolutions.net/DeviceUp - */ -int -bcw_init(struct ifnet *ifp) -{ - struct bcw_softc *sc = ifp->if_softc; - //struct ieee80211com *ic = &sc->sc_ic; - int i, error; - uint32_t coremask = 0; - - /* initialize PHY values */ - sc->sc_phy_antenna_diversity = 0xffff; - memset(sc->sc_phy_minlowsig, 0xff, sizeof(sc->sc_phy_minlowsig)); - memset(sc->sc_phy_minlowsigpos, 0, sizeof(sc->sc_phy_minlowsigpos)); - sc->sc_phy_calibrated = 0; - sc->sc_phy_is_locked = 0; - sc->sc_phy_lopairs = malloc(sizeof(struct bcw_lopair) * BCW_LO_COUNT, - M_DEVBUF, M_NOWAIT); - if (sc->sc_phy_lopairs) - memset(sc->sc_phy_lopairs, 0, sizeof(struct bcw_lopair) * - BCW_LO_COUNT); - memset(sc->sc_phy_loopback_gain, 0, sizeof(sc->sc_phy_loopback_gain)); - - /* initialize Radio values */ - sc->sc_radio_baseband_atten = - bcw_radio_default_baseband_atten(sc); - sc->sc_radio_radio_atten = - bcw_radio_default_radio_atten(sc); - sc->sc_radio_txctl1 = bcw_radio_default_txctl1(sc); - sc->sc_radio_txctl2 = 0xffff; - sc->sc_radio_txpwr_offset = 0; - sc->sc_radio_nrssislope = 0; - for (i = 0; i < BCW_ARRAY_SIZE(sc->sc_radio_nrssi); i++) - sc->sc_radio_nrssi[i] = -1000; - for (i = 0; i < BCW_ARRAY_SIZE(sc->sc_radio_nrssi_lt); i++) - sc->sc_radio_nrssi_lt[i] = i; - sc->sc_radio_lofcal = 0xffff; - sc->sc_radio_initval = 0xffff; - sc->sc_radio_aci_enable = 0; - sc->sc_radio_aci_wlan_automatic = 0; - sc->sc_radio_aci_hw_rssi = 0; - - bcw_change_core(sc, sc->sc_core_80211->index); - - bcw_pc_crystal_on(sc); - - bcw_pc_init(sc); - - bcw_pc_set_clock(sc, BCW_PCTL_CLK_FAST); - - coremask |= (1 << 1); - bcw_iocore_enable(sc, coremask); - - bcw_80211_core_reset(sc, 1); - - /* initialize 80211 core */ - if ((error = bcw_80211_core_init(sc, 1))) - return (error); - - bcw_pc_set_clock(sc, BCW_PCTL_CLK_DYNAMIC); - - bcw_mac_enable(sc); - bcw_intr_enable(sc, BCW_INTR_INITIAL); - bcw_pc_set_clock(sc, BCW_PCTL_CLK_DYNAMIC); - - /* start timer */ - timeout_add(&sc->sc_timeout, hz); - - /* mark as running, and no outputs active */ - ifp->if_flags |= IFF_RUNNING; - ifp->if_flags &= ~IFF_OACTIVE; - - sc->sc_flags |= BCW_FLAGS_INITIALIZED; - - return (0); -} - -/* - * Add a receive buffer to the indicated descriptor - */ -int -bcw_add_rxbuf(struct bcw_softc *sc, int idx) -{ - return (0); -} - -/* - * Drain the receive queue - */ -void -bcw_rxdrain(struct bcw_softc *sc) -{ - -} - -/* - * Stop transmission on the interface - * - * http://bcm-specs.sipsolutions.net/DeviceDown - */ -void -bcw_stop(struct ifnet *ifp, int disable) -{ - struct bcw_softc *sc = ifp->if_softc; - - /* disable all interrupts */ - bcw_intr_disable(sc, BCW_INTR_ALL); - - /* disable the 80211 core */ - bcw_80211_core_disable(sc); - - /* turn off crystal */ - bcw_pc_crystal_off(sc); - - /* Stop the 1 second timer */ - timeout_del(&sc->sc_timeout); - - /* Mark the interface down and cancel the watchdog timer. */ - ifp->if_flags &= ~(IFF_RUNNING | IFF_OACTIVE); - ifp->if_timer = 0; - - /* Disable interrupts. */ - BCW_WRITE(sc, BCW_MMIO_DMA0_INT_MASK, 0); - sc->sc_intmask = 0; - delay(10); - - /* Stop the DMA */ - BCW_WRITE(sc, BCW_DMA_RXCONTROL(0), 0); - BCW_WRITE(sc, BCW_DMA_TXCONTROL(0), 0); - delay(10); - - /* drain receive queue */ - if (disable) - bcw_rxdrain(sc); -} - -int -bcw_newstate(struct ieee80211com *ic, enum ieee80211_state nstate, int arg) -{ - return (0); -} - -int -bcw_media_change(struct ifnet *ifp) -{ - int error; - - error = ieee80211_media_change(ifp); - if (error != ENETRESET) - return error; - - if ((ifp->if_flags & (IFF_UP | IFF_RUNNING)) == (IFF_UP | IFF_RUNNING)) - bcw_init(ifp); - - return (0); -} - - void -bcw_media_status(struct ifnet *ifp, struct ifmediareq *imr) -{ - struct bcw_softc *sc = ifp->if_softc; - struct ieee80211com *ic = &sc->sc_ic; - int rate; - - imr->ifm_status = IFM_AVALID; - imr->ifm_active = IFM_IEEE80211; - if (ic->ic_state == IEEE80211_S_RUN) - imr->ifm_status |= IFM_ACTIVE; - - /* - * XXX Read current transmission rate from the adapter. - */ - //val = CSR_READ_4(sc, IWI_CSR_CURRENT_TX_RATE); - /* convert PLCP signal to 802.11 rate */ - //rate = bcw_rate(val); - rate = 0; - - imr->ifm_active |= ieee80211_rate2media(ic, rate, ic->ic_curmode); - switch (ic->ic_opmode) { - case IEEE80211_M_STA: - break; - case IEEE80211_M_IBSS: - imr->ifm_active |= IFM_IEEE80211_ADHOC; - break; - case IEEE80211_M_MONITOR: - imr->ifm_active |= IFM_IEEE80211_MONITOR; - break; - case IEEE80211_M_AHDEMO: - case IEEE80211_M_HOSTAP: - /* should not get there */ - break; - } -} - -/* - * One second timer, checks link status - */ -void -bcw_tick(void *v) -{ -#if 0 - struct bcw_softc *sc = v; - - /* http://bcm-specs.sipsolutions.net/PeriodicTasks */ - timeout_add(&sc->bcw_timeout, hz); -#endif -} - -/* - * Validate Chip Access - * - * This function ensures that the 80211 core is setup correctly and - * and is read for use. - * - * http://bcm-specs.sipsolutions.net/ValidateChipAccess - */ -int -bcw_validate_chip_access(struct bcw_softc *sc) -{ - uint32_t save, val; - - /* test 1 (this test is just for the 80211 core) */ - if (sc->sc_core[sc->sc_currentcore].id != BCW_CORE_80211) - goto fail; - - /* save value */ - save = bcw_shm_read32(sc, BCW_SHM_SHARED, 0); - - /* test 2 */ - bcw_shm_write32(sc, BCW_SHM_SHARED, 0, 0xaa5555aa); - val = bcw_shm_read32(sc, BCW_SHM_SHARED, 0); - if (val != 0xaa5555aa) - goto fail; - - /* test 3 */ - bcw_shm_write32(sc, BCW_SHM_SHARED, 0, 0x55aaaa55); - val = bcw_shm_read32(sc, BCW_SHM_SHARED, 0); - if (val != 0x55aaaa55) - goto fail; - - /* restore value */ - bcw_shm_write32(sc, BCW_SHM_SHARED, 0, save); - - if (sc->sc_core_80211->rev >= 3) { - /* test 4 */ - BCW_WRITE16(sc, 0x18c, 0xaaaa); - BCW_WRITE(sc, 0x18c, 0xccccbbbb); - val = BCW_READ16(sc, 0x604); - if (val != 0xbbbb) - goto fail; - /* test 5 */ - val = BCW_READ16(sc, 0x606); - if (val != 0xcccc) - goto fail; - BCW_WRITE(sc, 0x18c, 0); - } - - /* test 6 */ - val = BCW_READ(sc, BCW_MMIO_SBF); - if ((val | 0x80000000) != 0x80000400) - goto fail; - - /* test 7 */ - val = BCW_READ(sc, BCW_MMIO_GIR); - if (val != 0) - goto fail; - - /* test 8 */ - if (sc->sc_phy_type > BCW_PHY_TYPEG) - goto fail; - - return (0); - -fail: - printf("%s: Chip Access Validation failed!\n", sc->sc_dev.dv_xname); - return (1); -} - -int -bcw_detach(void *arg) -{ - struct bcw_softc *sc = arg; - struct ieee80211com *ic = &sc->sc_ic; - struct ifnet *ifp = &ic->ic_if; - - timeout_del(&sc->sc_scan_to); - - bcw_stop(ifp, 1); - ieee80211_ifdetach(ifp); - if_detach(ifp); - bcw_free_rx_ring(sc, &sc->sc_rxring); - bcw_free_tx_ring(sc, &sc->sc_txring); - - /* power off cardbus socket */ - if (sc->sc_disable) - sc->sc_disable(sc); - - return (0); -} - -int -bcw_alloc_rx_ring(struct bcw_softc *sc, struct bcw_rx_ring *ring, int count) -{ - struct bcw_desc *desc; - struct bcw_rx_data *data; - int i, nsegs, error; - - ring->count = count; - ring->cur = ring->next = 0; - - error = bus_dmamap_create(sc->sc_dmat, - count * sizeof(struct bcw_desc), 1, - count * sizeof(struct bcw_desc), 0, - BUS_DMA_NOWAIT, &ring->map); - if (error != 0) { - printf("%s: could not create desc DMA map\n", - sc->sc_dev.dv_xname); - goto fail; - } - - error = bus_dmamem_alloc(sc->sc_dmat, - count * sizeof(struct bcw_desc), - PAGE_SIZE, 0, &ring->seg, 1, &nsegs, BUS_DMA_NOWAIT); - if (error != 0) { - printf("%s: could not allocate DMA memory\n", - sc->sc_dev.dv_xname); - goto fail; - } - - error = bus_dmamem_map(sc->sc_dmat, &ring->seg, nsegs, - count * sizeof(struct bcw_desc), (caddr_t *)&ring->desc, - BUS_DMA_NOWAIT); - if (error != 0) { - printf("%s: could not map desc DMA memory\n", - sc->sc_dev.dv_xname); - goto fail; - } - - error = bus_dmamap_load(sc->sc_dmat, ring->map, ring->desc, - count * sizeof(struct bcw_desc), NULL, BUS_DMA_NOWAIT); - if (error != 0) { - printf("%s: could not load desc DMA map\n", - sc->sc_dev.dv_xname); - goto fail; - } - - bzero(ring->desc, count * sizeof(struct bcw_desc)); - ring->physaddr = ring->map->dm_segs->ds_addr; - - ring->data = malloc(count * sizeof (struct bcw_rx_data), M_DEVBUF, - M_NOWAIT); - if (ring->data == NULL) { - printf("%s: could not allocate soft data\n", - sc->sc_dev.dv_xname); - error = ENOMEM; - goto fail; - } - - BCW_WRITE(sc, BCW_DMA_RXADDR, ring->physaddr + 0x40000000); - - /* - * Pre-allocate Rx buffers and populate Rx ring. - */ - bzero(ring->data, count * sizeof (struct bcw_rx_data)); - for (i = 0; i < count; i++) { - desc = &ring->desc[i]; - data = &ring->data[i]; - - error = bus_dmamap_create(sc->sc_dmat, MCLBYTES, 1, MCLBYTES, - 0, BUS_DMA_NOWAIT, &data->map); - if (error != 0) { - printf("%s: could not create DMA map\n", - sc->sc_dev.dv_xname); - goto fail; - } - - MGETHDR(data->m, M_DONTWAIT, MT_DATA); - if (data->m == NULL) { - printf("%s: could not allocate rx mbuf\n", - sc->sc_dev.dv_xname); - error = ENOMEM; - goto fail; - } - - MCLGET(data->m, M_DONTWAIT); - if (!(data->m->m_flags & M_EXT)) { - printf("%s: could not allocate rx mbuf cluster\n", - sc->sc_dev.dv_xname); - error = ENOMEM; - goto fail; - } - - error = bus_dmamap_load(sc->sc_dmat, data->map, - mtod(data->m, void *), MCLBYTES, NULL, BUS_DMA_NOWAIT); - if (error != 0) { - printf("%s: could not load rx buf DMA map", - sc->sc_dev.dv_xname); - goto fail; - } - - desc->addr = htole32(data->map->dm_segs->ds_addr); - - if (i != (count - 1)) - desc->ctrl = htole32(BCW_RXBUF_LEN); - else - desc->ctrl = htole32(BCW_RXBUF_LEN | CTRL_EOT); - } - - bus_dmamap_sync(sc->sc_dmat, ring->map, 0, ring->map->dm_mapsize, - BUS_DMASYNC_PREWRITE); - - return (0); - -fail: bcw_free_rx_ring(sc, ring); - return (error); -} - -void -bcw_reset_rx_ring(struct bcw_softc *sc, struct bcw_rx_ring *ring) -{ - bus_dmamap_sync(sc->sc_dmat, ring->map, 0, ring->map->dm_mapsize, - BUS_DMASYNC_PREWRITE); - - ring->cur = ring->next = 0; -} - -void -bcw_free_rx_ring(struct bcw_softc *sc, struct bcw_rx_ring *ring) -{ - struct bcw_rx_data *data; - int i; - - if (ring->desc != NULL) { - bus_dmamap_sync(sc->sc_dmat, ring->map, 0, - ring->map->dm_mapsize, BUS_DMASYNC_POSTWRITE); - bus_dmamap_unload(sc->sc_dmat, ring->map); - bus_dmamem_unmap(sc->sc_dmat, (caddr_t)ring->desc, - ring->count * sizeof(struct bcw_desc)); - bus_dmamem_free(sc->sc_dmat, &ring->seg, 1); - } - - if (ring->data != NULL) { - for (i = 0; i < ring->count; i++) { - data = &ring->data[i]; - - if (data->m != NULL) { - bus_dmamap_sync(sc->sc_dmat, data->map, 0, - data->map->dm_mapsize, - BUS_DMASYNC_POSTREAD); - bus_dmamap_unload(sc->sc_dmat, data->map); - m_freem(data->m); - } - - if (data->map != NULL) - bus_dmamap_destroy(sc->sc_dmat, data->map); - } - free(ring->data, M_DEVBUF); - } -} - -int -bcw_alloc_tx_ring(struct bcw_softc *sc, struct bcw_tx_ring *ring, - int count) -{ - int i, nsegs, error; - - ring->count = count; - ring->queued = 0; - ring->cur = ring->next = ring->stat = 0; - - error = bus_dmamap_create(sc->sc_dmat, - count * sizeof(struct bcw_desc), 1, - count * sizeof(struct bcw_desc), 0, BUS_DMA_NOWAIT, &ring->map); - if (error != 0) { - printf("%s: could not create desc DMA map\n", - sc->sc_dev.dv_xname); - goto fail; - } - - error = bus_dmamem_alloc(sc->sc_dmat, - count * sizeof(struct bcw_desc), - PAGE_SIZE, 0, &ring->seg, 1, &nsegs, BUS_DMA_NOWAIT); - if (error != 0) { - printf("%s: could not allocate DMA memory\n", - sc->sc_dev.dv_xname); - goto fail; - } - - error = bus_dmamem_map(sc->sc_dmat, &ring->seg, nsegs, - count * sizeof(struct bcw_desc), (caddr_t *)&ring->desc, - BUS_DMA_NOWAIT); - if (error != 0) { - printf("%s: could not map desc DMA memory\n", - sc->sc_dev.dv_xname); - goto fail; - } - - error = bus_dmamap_load(sc->sc_dmat, ring->map, ring->desc, - count * sizeof(struct bcw_desc), NULL, BUS_DMA_NOWAIT); - if (error != 0) { - printf("%s: could not load desc DMA map\n", - sc->sc_dev.dv_xname); - goto fail; - } - - memset(ring->desc, 0, count * sizeof(struct bcw_desc)); - ring->physaddr = ring->map->dm_segs->ds_addr; - - /* MAGIC */ - BCW_WRITE(sc, BCW_DMA_TXADDR, ring->physaddr + 0x40000000); - - ring->data = malloc(count * sizeof(struct bcw_tx_data), M_DEVBUF, - M_NOWAIT); - if (ring->data == NULL) { - printf("%s: could not allocate soft data\n", - sc->sc_dev.dv_xname); - error = ENOMEM; - goto fail; - } - - memset(ring->data, 0, count * sizeof(struct bcw_tx_data)); - for (i = 0; i < count; i++) { - error = bus_dmamap_create(sc->sc_dmat, MCLBYTES, - BCW_MAX_SCATTER, MCLBYTES, 0, BUS_DMA_NOWAIT, - &ring->data[i].map); - if (error != 0) { - printf("%s: could not create DMA map\n", - sc->sc_dev.dv_xname); - goto fail; - } - } - - return (0); - -fail: bcw_free_tx_ring(sc, ring); - return (error); -} - -void -bcw_reset_tx_ring(struct bcw_softc *sc, struct bcw_tx_ring *ring) -{ - struct bcw_desc *desc; - struct bcw_tx_data *data; - int i; - - for (i = 0; i < ring->count; i++) { - desc = &ring->desc[i]; - data = &ring->data[i]; - - if (data->m != NULL) { - bus_dmamap_sync(sc->sc_dmat, data->map, 0, - data->map->dm_mapsize, BUS_DMASYNC_POSTWRITE); - bus_dmamap_unload(sc->sc_dmat, data->map); - m_freem(data->m); - data->m = NULL; - } - - /* - * The node has already been freed at that point so don't call - * ieee80211_release_node() here. - */ - data->ni = NULL; - } - - bus_dmamap_sync(sc->sc_dmat, ring->map, 0, ring->map->dm_mapsize, - BUS_DMASYNC_PREWRITE); - - ring->queued = 0; - ring->cur = ring->next = ring->stat = 0; -} - -void -bcw_free_tx_ring(struct bcw_softc *sc, struct bcw_tx_ring *ring) -{ - struct bcw_tx_data *data; - int i; - - if (ring->desc != NULL) { - bus_dmamap_sync(sc->sc_dmat, ring->map, 0, - ring->map->dm_mapsize, BUS_DMASYNC_POSTWRITE); - bus_dmamap_unload(sc->sc_dmat, ring->map); - bus_dmamem_unmap(sc->sc_dmat, (caddr_t)ring->desc, - ring->count * sizeof(struct bcw_desc)); - bus_dmamem_free(sc->sc_dmat, &ring->seg, 1); - } - - if (ring->data != NULL) { - for (i = 0; i < ring->count; i++) { - data = &ring->data[i]; - - if (data->m != NULL) { - bus_dmamap_sync(sc->sc_dmat, data->map, 0, - data->map->dm_mapsize, - BUS_DMASYNC_POSTWRITE); - bus_dmamap_unload(sc->sc_dmat, data->map); - m_freem(data->m); - } - - /* - * The node has already been freed at that point so - * don't call ieee80211_release_node() here. - */ - data->ni = NULL; - - if (data->map != NULL) - bus_dmamap_destroy(sc->sc_dmat, data->map); - } - free(ring->data, M_DEVBUF); - } -} - -int -bcw_change_core(struct bcw_softc *sc, int changeto) -{ - int i; - uint32_t sbval; - - (sc->sc_conf_write)(sc, BCW_ADDR_SPACE0, BCW_CORE_SELECT(changeto)); - - /* loop to see if the selected core shows up */ - for (i = 0; i < 10; i++) { - sbval = (sc->sc_conf_read)(sc, BCW_ADDR_SPACE0); - if (sbval == BCW_CORE_SELECT(changeto)) - break; - delay(10); - } - if (i == 10) { - DPRINTF(("%s: can not change to core %d!\n", - sc->sc_dev.dv_xname, changeto)); - return (1); - } - - /* core changed */ - sc->sc_lastcore = sc->sc_currentcore; - sc->sc_currentcore = changeto; - - return (0); -} - -/* - * Enable current core - */ -int -bcw_core_enable(struct bcw_softc *sc, uint32_t core_flags) -{ - uint32_t sbtmstatelow; - uint32_t sbtmstatehigh; - uint32_t sbimstate; - int error; - - if ((error = bcw_core_disable(sc, core_flags))) - return (error); - - sbtmstatelow = - BCW_SBTMSTATELOW_CLOCK | - BCW_SBTMSTATELOW_RESET | - BCW_SBTMSTATELOW_FGCLOCK | - core_flags; - BCW_WRITE(sc, BCW_CIR_SBTMSTATELOW, sbtmstatelow); - delay(1); - - sbtmstatehigh = BCW_READ(sc, BCW_CIR_SBTMSTATEHIGH); - if (sbtmstatehigh & BCW_SBTMSTATEHIGH_SERROR) { - sbtmstatehigh = 0; - BCW_WRITE(sc, BCW_CIR_SBTMSTATEHIGH, sbtmstatehigh); - } - - sbimstate = BCW_READ(sc, BCW_CIR_SBIMSTATE); - if (sbimstate & (BCW_SBIMSTATE_IB_ERROR | BCW_SBIMSTATE_TIMEOUT)) { - sbimstate &= ~(BCW_SBIMSTATE_IB_ERROR | BCW_SBIMSTATE_TIMEOUT); - BCW_WRITE(sc, BCW_CIR_SBIMSTATE, sbimstate); - } - - sbtmstatelow = - BCW_SBTMSTATELOW_CLOCK | - BCW_SBTMSTATELOW_FGCLOCK | - core_flags; - BCW_WRITE(sc, BCW_CIR_SBTMSTATELOW, sbtmstatelow); - delay(1); - - sbtmstatelow = BCW_SBTMSTATELOW_CLOCK | core_flags; - BCW_WRITE(sc, BCW_CIR_SBTMSTATELOW, sbtmstatelow); - delay(1); - - return (0); -} - -/* - * Disable current core - */ -int -bcw_core_disable(struct bcw_softc *sc, uint32_t core_flags) -{ - uint32_t sbtmstatelow; - uint32_t sbtmstatehigh; - int i; - - /* fetch sbtmstatelow from core information registers */ - sbtmstatelow = BCW_READ(sc, BCW_CIR_SBTMSTATELOW); - - /* core is already in reset */ - if (sbtmstatelow & BCW_SBTMSTATELOW_RESET) - goto out; - - if (sbtmstatelow & BCW_SBTMSTATELOW_CLOCK) { - sbtmstatelow = - BCW_SBTMSTATELOW_CLOCK | - BCW_SBTMSTATELOW_REJECT; - BCW_WRITE(sc, BCW_CIR_SBTMSTATELOW, sbtmstatelow); - - for (i = 0; i < 1000; i++) { - sbtmstatelow = BCW_READ(sc, BCW_CIR_SBTMSTATELOW); - if (sbtmstatelow & BCW_SBTMSTATELOW_REJECT) { - i = -1; - break; - } - delay(10); - } - if (i != -1) { - DPRINTF(("%s: can not disable core, REJECT timeout!\n", - sc->sc_dev.dv_xname)); - return (EBUSY); - } - - for (i = 0; i < 1000; i++) { - sbtmstatehigh = BCW_READ(sc, BCW_CIR_SBTMSTATEHIGH); - if (!(sbtmstatehigh & BCW_SBTMSTATEHIGH_BUSY)) { - i = -1; - break; - } - delay(10); - } - if (i != -1) { - DPRINTF(("%s: can not disable, core, BUSY timeout!\n", - sc->sc_dev.dv_xname)); - return (EBUSY); - } - - sbtmstatelow = - BCW_SBTMSTATELOW_FGCLOCK | - BCW_SBTMSTATELOW_REJECT | - BCW_SBTMSTATELOW_RESET | - BCW_SBTMSTATELOW_CLOCK | - core_flags; - BCW_WRITE(sc, BCW_CIR_SBTMSTATELOW, sbtmstatelow); - delay(10); - } - - sbtmstatelow = - BCW_SBTMSTATELOW_RESET | - BCW_SBTMSTATELOW_REJECT | - core_flags; - BCW_WRITE(sc, BCW_CIR_SBTMSTATELOW, sbtmstatelow); - -out: - /* XXX sc_current_core_enabled = 0 */ - - return (0); -} - -/* - * Reset the 80211 core - * - * http://bcm-specs.sipsolutions.net/80211CoreReset - */ -void -bcw_80211_core_reset(struct bcw_softc *sc, int connect_phy) -{ - uint32_t flags = 0x00040000; - - if (connect_phy) - flags |= 0x20000000; - bcw_phy_connect(sc, connect_phy); - bcw_core_enable(sc, flags); - BCW_WRITE16(sc, 0x03e6, 0); - BCW_WRITE(sc, BCW_MMIO_SBF, BCW_READ(sc, BCW_MMIO_SBF) | - BCW_SBF_400_MAGIC); -} - -/* - * Disable the 80211 core - * - * http://bcm-specs.sipsolutions.net/80211CoreDisable - */ -void -bcw_80211_core_disable(struct bcw_softc *sc) -{ - bcw_radio_off(sc); - BCW_WRITE16(sc, 0x03e6, 0x00f4); - bcw_core_disable(sc, 0); -} - -int -bcw_iocore_enable(struct bcw_softc *sc, uint32_t coremask) -{ - uint32_t val; - uint32_t bp_flag_nr; - int error; - - val = BCW_READ(sc, BCW_CIR_SBTPSFLAG); - bp_flag_nr = val & SBTPS_BACKPLANEFLAGMASK; - - if ((error = bcw_change_core(sc, 2))) - return (error); - - if (sc->sc_core[sc->sc_currentcore].rev < 6 && - sc->sc_core[sc->sc_currentcore].id == BCW_CORE_PCI) { - val = BCW_READ(sc, BCW_CIR_SBINTVEC); - val |= (1 << bp_flag_nr); - BCW_WRITE(sc, BCW_CIR_SBINTVEC, val); - } else { - val = (sc->sc_conf_read)(sc, BCW_PCICFG_ICR); - val |= coremask << 8; - (sc->sc_conf_write)(sc, BCW_PCICFG_ICR, val); - } - - if (sc->sc_core[sc->sc_currentcore].id == BCW_CORE_PCI) { - val = BCW_READ(sc, BCW_PCICORE_SBTOPCI2); - val |= BCW_SBTOPCI2_PREFETCH | BCW_SBTOPCI2_BURST; - BCW_WRITE(sc, BCW_PCICORE_SBTOPCI2, val); - - if (sc->sc_core[sc->sc_currentcore].rev < 5) { - val = BCW_READ(sc, BCW_CIR_SBIMCONFIGLOW); - val |= (2 << BCW_SBIMCONFIGLOW_STS) & - BCW_SBIMCONFIGLOW_STM; - val |= (3 << BCW_SBIMCONFIGLOW_STS) & - BCW_SBIMCONFIGLOW_STM; - BCW_WRITE(sc, BCW_CIR_SBIMCONFIGLOW, val); - /* TODO commit PCI settings */ - } else if (sc->sc_core[sc->sc_currentcore].rev >= 11) { - val = BCW_READ(sc, BCW_PCICORE_SBTOPCI2); - val |= BCW_SBTOPCI2_MEMREAD_MULTI; - BCW_WRITE(sc, BCW_PCICORE_SBTOPCI2, val); - } - } - - if ((error = bcw_change_core(sc, sc->sc_lastcore))) - return (error); - - return (0); -} - -int -bcw_get_firmware(const char *name, const uint8_t *ucode, size_t size_ucode, - size_t *size, size_t *offset) -{ - int i, nfiles, off = 0, ret = 1; - struct fwheader *h; - - if ((h = malloc(sizeof(struct fwheader), M_DEVBUF, M_NOWAIT)) == NULL) - return (ret); - - /* get number of firmware files */ - bcopy(ucode, &nfiles, sizeof(nfiles)); - nfiles = ntohl(nfiles); - off += sizeof(nfiles); - - /* parse header and search the firmware */ - for (i = 0; i < nfiles && off < size_ucode; i++) { - bzero(h, sizeof(struct fwheader)); - bcopy(ucode + off, h, sizeof(struct fwheader)); - off += sizeof(struct fwheader); - - if (strcmp(name, h->filename) == 0) { - ret = 0; - *size = ntohl(h->filesize); - *offset = ntohl(h->fileoffset); - break; - } - } - - free(h, M_DEVBUF); - - return (ret); -} - -int -bcw_load_firmware(struct bcw_softc *sc) -{ - int rev = sc->sc_core[sc->sc_currentcore].rev; - int error, len, i; - uint32_t *data; - uint8_t *ucode; - size_t size_ucode, size_micro, size_pcm, off_micro, off_pcm; - char *name = "bcw-bcm43xx"; - char filename[64]; - - /* load firmware */ - if ((error = loadfirmware(name, &ucode, &size_ucode)) != 0) { - printf("%s: error %d, could not read microcode %s!\n", - sc->sc_dev.dv_xname, error, name); - return (EIO); - } - DPRINTF(("%s: successfully read %s\n", sc->sc_dev.dv_xname, name)); - - /* get microcode file offset */ - snprintf(filename, sizeof(filename), "bcm43xx_microcode%d.fw", - rev >= 5 ? 5 : rev); - - if (bcw_get_firmware(filename, ucode, size_ucode, &size_micro, - &off_micro) != 0) { - printf("%s: get offset for firmware file %s failed!\n", - sc->sc_dev.dv_xname, filename); - goto fail; - } - - /* get pcm file offset */ - snprintf(filename, sizeof(filename), "bcm43xx_pcm%d.fw", - rev < 5 ? 4 : 5); - - if (bcw_get_firmware(filename, ucode, size_ucode, &size_pcm, - &off_pcm) != 0) { - printf("%s: get offset for firmware file %s failed!\n", - sc->sc_dev.dv_xname, filename); - goto fail; - } - - /* upload microcode */ - data = (uint32_t *)(ucode + off_micro); - len = size_micro / sizeof(uint32_t); - bcw_shm_ctl_word(sc, BCW_SHM_UCODE, 0); - for (i = 0; i < len; i++) { - BCW_WRITE(sc, BCW_MMIO_SHM_DATA, betoh32(data[i])); - delay(10); - } - DPRINTF(("%s: uploaded microcode\n", sc->sc_dev.dv_xname)); - - /* upload pcm */ - data = (uint32_t *)(ucode + off_pcm); - len = size_pcm / sizeof(uint32_t); - bcw_shm_ctl_word(sc, BCW_SHM_PCM, 0x01ea); - BCW_WRITE(sc, BCW_MMIO_SHM_DATA, 0x00004000); - bcw_shm_ctl_word(sc, BCW_SHM_PCM, 0x01eb); - for (i = 0; i < len; i++) { - BCW_WRITE(sc, BCW_MMIO_SHM_DATA, betoh32(data[i])); - delay(10); - } - DPRINTF(("%s: uploaded pcm\n", sc->sc_dev.dv_xname)); - - free(ucode, M_DEVBUF); - - return (0); - -fail: free(ucode, M_DEVBUF); - return (EIO); -} - -int -bcw_write_initvals(struct bcw_softc *sc, const struct bcw_initval *data, - const unsigned int len) -{ - int i; - uint16_t offset, size; - uint32_t value; - - for (i = 0; i < len; i++) { - offset = betoh16(data[i].offset); - size = betoh16(data[i].size); - value = betoh32(data[i].value); - - if (offset >= 0x1000) - goto bad_format; - if (size == 2) { - if (value & 0xffff0000) - goto bad_format; - BCW_WRITE16(sc, offset, (uint16_t)value); - } else if (size == 4) - BCW_WRITE(sc, offset, value); - else - goto bad_format; - } - - return (0); - -bad_format: - printf("%s: initvals file-format error!\n", sc->sc_dev.dv_xname); - return (EIO); -} - -int -bcw_load_initvals(struct bcw_softc *sc) -{ - int rev = sc->sc_core[sc->sc_currentcore].rev; - int error, nr; - uint32_t val; - uint8_t *ucode; - size_t size_ucode, size_ival0, size_ival1, off_ival0, off_ival1; - char *name = "bcw-bcm43xx"; - char filename[64]; - - /* load firmware */ - if ((error = loadfirmware(name, &ucode, &size_ucode)) != 0) { - printf("%s: error %d, could not read microcode %s!\n", - sc->sc_dev.dv_xname, error, name); - return (EIO); - } - DPRINTF(("%s: successfully read %s\n", sc->sc_dev.dv_xname, name)); - - /* get initval0 file offset */ - if (rev == 2 || rev == 4) { - switch (sc->sc_phy_type) { - case BCW_PHY_TYPEA: - nr = 3; - break; - case BCW_PHY_TYPEB: - case BCW_PHY_TYPEG: - nr = 1; - break; - default: - printf("%s: no initvals available!\n", - sc->sc_dev.dv_xname); - goto fail; - } - } else if (rev >= 5) { - switch (sc->sc_phy_type) { - case BCW_PHY_TYPEA: - nr = 7; - break; - case BCW_PHY_TYPEB: - case BCW_PHY_TYPEG: - nr = 5; - break; - default: - printf("%s: no initvals available!\n", - sc->sc_dev.dv_xname); - goto fail; - } - } else { - printf("%s: no initvals available!\n", sc->sc_dev.dv_xname); - goto fail; - } - - snprintf(filename, sizeof(filename), "bcm43xx_initval%02d.fw", nr); - - if (bcw_get_firmware(filename, ucode, size_ucode, &size_ival0, - &off_ival0) != 0) { - printf("%s: get offset for initval0 file %s failed\n", - sc->sc_dev.dv_xname, filename); - goto fail; - } - - /* get initval1 file offset */ - if (rev >= 5) { - switch (sc->sc_phy_type) { - case BCW_PHY_TYPEA: - val = BCW_READ(sc, BCW_SBTMSTATEHI); - if (val & 0x00010000) - nr = 9; - else - nr = 10; - break; - case BCW_PHY_TYPEB: - case BCW_PHY_TYPEG: - nr = 6; - break; - default: - printf("%s: no initvals available!\n", - sc->sc_dev.dv_xname); - goto fail; - } - - snprintf(filename, sizeof(filename), "bcm43xx_initval%02d.fw", - nr); - - if (bcw_get_firmware(filename, ucode, size_ucode, &size_ival1, - &off_ival1) != 0) { - printf("%s: get offset for initval1 file %s failed\n", - sc->sc_dev.dv_xname, filename); - goto fail; - } - } - - /* upload initval0 */ - if (bcw_write_initvals(sc, (struct bcw_initval *)(ucode + off_ival0), - size_ival0 / sizeof(struct bcw_initval))) - goto fail; - DPRINTF(("%s: uploaded initval0\n", sc->sc_dev.dv_xname)); - - /* upload initval1 */ - if (off_ival1 != 0) { - if (bcw_write_initvals(sc, - (struct bcw_initval *)(ucode + off_ival1), - size_ival1 / sizeof(struct bcw_initval))) - goto fail; - DPRINTF(("%s: uploaded initval1\n", sc->sc_dev.dv_xname)); - } - - free(ucode, M_DEVBUF); - - return (0); - -fail: free(ucode, M_DEVBUF); - return (EIO); -} - -void -bcw_leds_switch_all(struct bcw_softc *sc, int on) -{ - struct bcw_led *led; - uint16_t ledctl; - int i, bit_on; - - ledctl = BCW_READ16(sc, BCW_MMIO_GPIO_CONTROL); - - for (i = 0; i < BCW_NR_LEDS; i++) { - led = &(sc->leds[i]); - if (led->behaviour == BCW_LED_INACTIVE) - continue; - if (on) - bit_on = led->activelow ? 0 : 1; - else - bit_on = led->activelow ? 0 : 1; - if (bit_on) - ledctl |= (1 << i); - else - ledctl &= ~(1 << i); - } - - BCW_WRITE16(sc, BCW_MMIO_GPIO_CONTROL, ledctl); -} - -int -bcw_gpio_init(struct bcw_softc *sc) -{ - uint32_t mask, set; - int error = 0; - - BCW_WRITE(sc, BCW_MMIO_SBF, BCW_READ(sc, BCW_MMIO_SBF) & 0xffff3fff); - - bcw_leds_switch_all(sc, 0); - - BCW_WRITE16(sc, BCW_MMIO_GPIO_MASK, - BCW_READ16(sc, BCW_MMIO_GPIO_MASK) | 0x000f); - - mask = 0x0000001f; - set = 0x0000000f; - - if (sc->sc_chip_id == 0x4301) { - mask |= 0x0060; - set |= 0x0060; - } - if (0) { /* XXX conditional unknown */ - BCW_WRITE16(sc, BCW_MMIO_GPIO_MASK, - BCW_READ16(sc, BCW_MMIO_GPIO_MASK) | 0x0100); - mask |= 0x0180; - set |= 0x0180; - } - if (sc->sc_sprom.boardflags & BCW_BF_PACTRL) { - BCW_WRITE16(sc, BCW_MMIO_GPIO_MASK, - BCW_READ16(sc, BCW_MMIO_GPIO_MASK) | 0x0200); - mask |= 0x0200; - set |= 0x0200; - } - if (sc->sc_chip_rev >= 2) - mask |= 0x0010; /* XXX this is redundant */ - - /* - * TODO bcw_change_core_to_gpio() - * - * Where to find the GPIO register depends on the chipset. - * If it has a ChipCommon, its register at offset 0x6c is the GPIO - * control register. Otherwise the register at offset 0x6c in the - * PCI core is the GPIO control register. - */ - if ((error = bcw_change_core(sc, 0))) - return (error); - - BCW_WRITE(sc, BCW_GPIO_CTRL, (BCW_READ(sc, BCW_GPIO_CTRL) & mask) | - set); - - error = bcw_change_core(sc, sc->sc_lastcore); - - return (error); -} - -/* - * Initialize the chip - * - * http://bcm-specs.sipsolutions.net/ChipInit - */ -int -bcw_chip_init(struct bcw_softc *sc) -{ - struct ieee80211com *ic = &sc->sc_ic; - struct ifnet *ifp = &sc->sc_ic.ic_if; - uint8_t limit; - uint16_t val16; - uint32_t val32; - int error, i; - - 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); - - if (sc->sc_core[sc->sc_currentcore].rev > 9) { - BCW_WRITE(sc, 0x540, 0x540); - BCW_WRITE(sc, 0x520, 0x520); - BCW_WRITE(sc, 0x540, 0x540); - BCW_WRITE(sc, 0x418, 0xe0a); - BCW_WRITE(sc, 0x41a, 0x90b); - BCW_WRITE(sc, 0x41c, 0x20e); - BCW_WRITE(sc, 0x41e, 0); - } - - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x5c, 0xa); - - 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_AP; - 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); - - 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); - - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x400, 0x8); - - /* TODO slot timing */ - - 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); - - if (sc->sc_core[sc->sc_currentcore].rev >= 5) - BCW_WRITE16(sc, BCW_MMIO_POWERUP_DELAY, - bcw_pc_powerup_delay(sc)); - - val16 = sc->sc_core[sc->sc_currentcore].rev; - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x16, val16); - - /* short/long retry limit */ - limit = bcw_lv(BCW_DEFAULT_SHORT_RETRY_LIMIT, 0, 0xf); - bcw_shm_write32(sc, BCW_SHM_80211, 0x0006, limit); - limit = bcw_lv(BCW_DEFAULT_LONG_RETRY_LIMIT, 0, 0xf); - bcw_shm_write32(sc, BCW_SHM_80211, 0x0007, limit); - - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x0044, 3); - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x0046, 2); - - bcw_macfilter_clear(sc, BCW_MACFILTER_SELF); - bcw_macfilter_set(sc, BCW_MACFILTER_SELF, ic->ic_myaddr); - - bcw_macfilter_clear(sc, BCW_MACFILTER_ASSOC); - bcw_macfilter_set(sc, BCW_MACFILTER_ASSOC, ic->ic_bss->ni_bssid); - - bcw_templateram_set(sc); - - DPRINTF(("%s: Chip initialized\n", sc->sc_dev.dv_xname)); - - return (0); -} - -/* - * Initialize the BS - * - * http://bcm-specs.sipsolutions.net/BSInit - */ -int -bcw_bs_init(struct bcw_softc *sc) -{ - struct ieee80211com *ic = &sc->sc_ic; - uint32_t ucodeflags; - - BCW_WRITE16(sc, 0x03e6, 0); - - bcw_radio_on(sc); - - bcw_phy_init(sc); - - bcw_radio_set_interf_mitigation(sc, sc->sc_radio_interfmode); - - bcw_phy_set_antenna_diversity(sc); - - bcw_radio_set_txantenna(sc, BCW_RADIO_TXANTENNA_DEFAULT); - - /* minimum contention window */ - if (sc->sc_phy_type == BCW_PHY_TYPEB) - bcw_shm_write32(sc, BCW_SHM_80211, 0x0003, 0x0000001f); - else - bcw_shm_write32(sc, BCW_SHM_80211, 0x0003, 0x0000000f); - - /* maximum contention window */ - bcw_shm_write32(sc, BCW_SHM_80211, 0x0004, 0x000003ff); - - bcw_rate_memory_init(sc); - - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x52, sc->sc_phy_type); - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x50, sc->sc_phy_rev); - - ucodeflags = bcw_shm_read32(sc, BCW_SHM_SHARED, BCW_UCODEFLAGS_OFFSET); - ucodeflags |= 0x100000; - bcw_shm_write32(sc, BCW_SHM_SHARED, BCW_UCODEFLAGS_OFFSET, ucodeflags); - - if (sc->sc_phy_type == BCW_PHY_TYPEA && - sc->sc_core[sc->sc_currentcore].rev <= 4) - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x3c, 0x1d); - else - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x3c, 0x1e); - - if (ic->ic_opmode != IEEE80211_M_IBSS && - ic->ic_opmode == IEEE80211_M_HOSTAP) { - BCW_WRITE16(sc, 0x612, 0x2); - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x416, 0x2); - } else if (sc->sc_phy_type == BCW_PHY_TYPEA) { - BCW_WRITE16(sc, 0x612, 0x78); - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x416, 0x78); - } else { - BCW_WRITE16(sc, 0x612, 0xfa); - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x416, 0xfa); - } - - return (0); -} - -/* - * Initialize the 80211 core - * - * http://bcm-specs.sipsolutions.net/80211Init - */ -int -bcw_80211_core_init(struct bcw_softc *sc, int active_80211_core) -{ - uint32_t ucodeflags; - //uint32_t sbimconfiglow; - int error; -#if 0 - if (sc->sc_core_bus->rev <= 5 && sc->sc_core_bus->id != BCW_CORE_PCIE) { - sbimconfiglow = BCW_READ(sc, BCW_CIR_SBIMCONFIGLOW); - sbimconfiglow &= ~BCW_SBIMCONFIGLOW_RTM; - sbimconfiglow &= ~BCW_SBIMCONFIGLOW_STM; - sbimconfiglow |= 0x32; - BCW_WRITE(sc, BCW_CIR_SBIMCONFIGLOW, sbimconfiglow); - } -#endif - - bcw_phy_calibrate(sc); - - if ((error = bcw_chip_init(sc))) - return (error); - - if ((error = bcw_bs_init(sc))) - return (error); - - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x0016, 0); - - /* - * Set microcode flags if necessery - */ - ucodeflags = bcw_shm_read32(sc, BCW_SHM_SHARED, BCW_UCODEFLAGS_OFFSET); - //ucodeflags |= 0x00000010; /* XXX */ - if (sc->sc_phy_type == BCW_PHY_TYPEB || - sc->sc_phy_type == BCW_PHY_TYPEG) - ucodeflags |= BCW_UCODEFLAG_UNKBGPHY; - if (sc->sc_phy_type == BCW_PHY_TYPEG && sc->sc_phy_rev == 1) - ucodeflags |= BCW_UCODEFLAG_UNKGPHY; - if (sc->sc_phy_type == BCW_PHY_TYPEB && sc->sc_phy_rev >= 2 && - sc->sc_radio_ver == 0x2050) - ucodeflags &= ~BCW_UCODEFLAG_UNKGPHY; - if (sc->sc_sprom.boardflags & BCW_BF_PACTRL) - ucodeflags |= BCW_UCODEFLAG_UNKPACTRL; - if (ucodeflags != - bcw_shm_read32(sc, BCW_SHM_SHARED, BCW_UCODEFLAGS_OFFSET)) - bcw_shm_write32(sc, BCW_SHM_SHARED, BCW_UCODEFLAGS_OFFSET, - ucodeflags); - - if (sc->sc_core[sc->sc_currentcore].rev >= 5) - BCW_WRITE16(sc, 0x043c, 0x000c); - - if (active_80211_core) { - if (bcw_using_pio(sc)) { - /* TODO bcw_pio_init() */ - } else { - /* TODO bcw_dma_init() */ - } - } - - BCW_WRITE16(sc, 0x0612, 0x0050); - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x0416, 0x0050); - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x0414, 0x01f4); -#if 0 - bcw_mac_enable(sc); - bcw_intr_enable(sc, 0xb007a864); -#endif - - return (0); -} - -uint8_t -bcw_sprom_crc8(uint8_t crc, uint8_t data) -{ - static const uint8_t t[] = { - 0x00, 0xF7, 0xB9, 0x4E, 0x25, 0xD2, 0x9C, 0x6B, - 0x4A, 0xBD, 0xF3, 0x04, 0x6F, 0x98, 0xD6, 0x21, - 0x94, 0x63, 0x2D, 0xDA, 0xB1, 0x46, 0x08, 0xFF, - 0xDE, 0x29, 0x67, 0x90, 0xFB, 0x0C, 0x42, 0xB5, - 0x7F, 0x88, 0xC6, 0x31, 0x5A, 0xAD, 0xE3, 0x14, - 0x35, 0xC2, 0x8C, 0x7B, 0x10, 0xE7, 0xA9, 0x5E, - 0xEB, 0x1C, 0x52, 0xA5, 0xCE, 0x39, 0x77, 0x80, - 0xA1, 0x56, 0x18, 0xEF, 0x84, 0x73, 0x3D, 0xCA, - 0xFE, 0x09, 0x47, 0xB0, 0xDB, 0x2C, 0x62, 0x95, - 0xB4, 0x43, 0x0D, 0xFA, 0x91, 0x66, 0x28, 0xDF, - 0x6A, 0x9D, 0xD3, 0x24, 0x4F, 0xB8, 0xF6, 0x01, - 0x20, 0xD7, 0x99, 0x6E, 0x05, 0xF2, 0xBC, 0x4B, - 0x81, 0x76, 0x38, 0xCF, 0xA4, 0x53, 0x1D, 0xEA, - 0xCB, 0x3C, 0x72, 0x85, 0xEE, 0x19, 0x57, 0xA0, - 0x15, 0xE2, 0xAC, 0x5B, 0x30, 0xC7, 0x89, 0x7E, - 0x5F, 0xA8, 0xE6, 0x11, 0x7A, 0x8D, 0xC3, 0x34, - 0xAB, 0x5C, 0x12, 0xE5, 0x8E, 0x79, 0x37, 0xC0, - 0xE1, 0x16, 0x58, 0xAF, 0xC4, 0x33, 0x7D, 0x8A, - 0x3F, 0xC8, 0x86, 0x71, 0x1A, 0xED, 0xA3, 0x54, - 0x75, 0x82, 0xCC, 0x3B, 0x50, 0xA7, 0xE9, 0x1E, - 0xD4, 0x23, 0x6D, 0x9A, 0xF1, 0x06, 0x48, 0xBF, - 0x9E, 0x69, 0x27, 0xD0, 0xBB, 0x4C, 0x02, 0xF5, - 0x40, 0xB7, 0xF9, 0x0E, 0x65, 0x92, 0xDC, 0x2B, - 0x0A, 0xFD, 0xB3, 0x44, 0x2F, 0xD8, 0x96, 0x61, - 0x55, 0xA2, 0xEC, 0x1B, 0x70, 0x87, 0xC9, 0x3E, - 0x1F, 0xE8, 0xA6, 0x51, 0x3A, 0xCD, 0x83, 0x74, - 0xC1, 0x36, 0x78, 0x8F, 0xE4, 0x13, 0x5D, 0xAA, - 0x8B, 0x7C, 0x32, 0xC5, 0xAE, 0x59, 0x17, 0xE0, - 0x2A, 0xDD, 0x93, 0x64, 0x0F, 0xF8, 0xB6, 0x41, - 0x60, 0x97, 0xD9, 0x2E, 0x45, 0xB2, 0xFC, 0x0B, - 0xBE, 0x49, 0x07, 0xF0, 0x9B, 0x6C, 0x22, 0xD5, - 0xF4, 0x03, 0x4D, 0xBA, 0xD1, 0x26, 0x68, 0x9F, - }; - - return (t[crc ^ data]); -} - -uint8_t -bcw_sprom_crc(const uint16_t *sprom) -{ - int word; - uint8_t crc = 0xff; - - for (word = 0; word < BCW_SPROM_SIZE - 1; word++) { - crc = bcw_sprom_crc8(crc, sprom[word] & 0x00ff); - crc = bcw_sprom_crc8(crc, (sprom[word] & 0xff00) >> 8); - } - crc = bcw_sprom_crc8(crc, sprom[BCW_SPROM_VERSION] & 0x00ff); - crc ^= 0xff; - - return (crc); -} - -int -bcw_sprom_read(struct bcw_softc *sc, uint16_t *sprom) -{ - int i; - uint8_t crc, expected_crc; - - for (i = 0; i < BCW_SPROM_SIZE; i++) - sprom[i] = BCW_READ16(sc, BCW_SPROM_BASE + (i * 2)); - - /* CRC-8 check */ - crc = bcw_sprom_crc(sprom); - expected_crc = (sprom[BCW_SPROM_VERSION] & 0xff00) >> 8; - if (crc != expected_crc) { - printf("%s: invalid SPROM checksum! (0x%02x, expected 0x%02x\n", - sc->sc_dev.dv_xname, crc, expected_crc); - return (EINVAL); - } - - return (0); -} - -/* - * Get whole SPROM content - * - * http://bcm-specs.sipsolutions.net/SPROM - */ -int -bcw_sprom_get(struct bcw_softc *sc) -{ - uint16_t val; - uint16_t *sprom; - - sprom = malloc(BCW_SPROM_SIZE * sizeof(uint16_t), M_DEVBUF, M_NOWAIT); - if (sprom == NULL) { - printf("%s: malloc in SPROM extract failed!\n", - sc->sc_dev.dv_xname); - return (ENOMEM); - } - - bcw_sprom_read(sc, sprom); - - /* boardflags */ - val = sprom[BCW_SPROM_BOARDFLAGS]; - sc->sc_sprom.boardflags2 = val; - - /* il0macaddr */ - val = sprom[BCW_SPROM_IL0MACADDR]; - *((uint16_t *)sc->sc_sprom.il0macaddr) = htobe16(val); - val = sprom[BCW_SPROM_IL0MACADDR + 1]; - *(((uint16_t *)sc->sc_sprom.il0macaddr) + 1) = htobe16(val); - val = sprom[BCW_SPROM_IL0MACADDR + 2]; - *(((uint16_t *)sc->sc_sprom.il0macaddr) + 2) = htobe16(val); - - /* et0macaddr */ - val = sprom[BCW_SPROM_ET0MACADDR]; - *((uint16_t *)sc->sc_sprom.et0macaddr) = htobe16(val); - val = sprom[BCW_SPROM_ET0MACADDR + 1]; - *(((uint16_t *)sc->sc_sprom.et0macaddr) + 1) = htobe16(val); - val = sprom[BCW_SPROM_ET0MACADDR + 2]; - *(((uint16_t *)sc->sc_sprom.et0macaddr) + 2) = htobe16(val); - - /* et1macaddr */ - val = sprom[BCW_SPROM_ET1MACADDR]; - *((uint16_t *)sc->sc_sprom.et1macaddr) = htobe16(val); - val = sprom[BCW_SPROM_ET1MACADDR + 1]; - *(((uint16_t *)sc->sc_sprom.et1macaddr) + 1) = htobe16(val); - val = sprom[BCW_SPROM_ET1MACADDR + 2]; - *(((uint16_t *)sc->sc_sprom.et1macaddr) + 2) = htobe16(val); - - /* ethernet phy settigns */ - val = sprom[BCW_SPROM_ETHPHY]; - sc->sc_sprom.et0phyaddr = (val & 0x001f); - sc->sc_sprom.et1phyaddr = (val & 0x03e0); - sc->sc_sprom.et0mdcport = (val & (1 << 14)) >> 14; - sc->sc_sprom.et1mdcport = (val & (1 << 15)) >> 15; - - /* boardrev, antennas, locale */ - val = sprom[BCW_SPROM_BOARDREV]; - sc->sc_sprom.boardrev = (val & 0x00ff); - sc->sc_sprom.locale = (val & 0x0f00); - sc->sc_sprom.antennas_aphy = (val & 0x3000) >> 12; - sc->sc_sprom.antennas_bgphy = (val & 0xc000) >> 14; - - /* pa0b */ - sc->sc_sprom.pa0b0 = sprom[BCW_SPROM_PA0B0]; - sc->sc_sprom.pa0b1 = sprom[BCW_SPROM_PA0B1]; - sc->sc_sprom.pa0b2 = sprom[BCW_SPROM_PA0B2]; - - /* wl0gpio */ - val = sprom[BCW_SPROM_WL0GPIO0]; - if (val == 0) - val = 0xffff; - sc->sc_sprom.wl0gpio0 = val & 0x00ff; - sc->sc_sprom.wl0gpio1 = (val & 0xff00) >> 8; - val = sprom[BCW_SPROM_WL0GPIO2]; - if (val == 0) - val = 0xffff; - sc->sc_sprom.wl0gpio2 = val & 0x00ff; - sc->sc_sprom.wl0gpio3 = (val & 0xff00) >> 8; - - /* maxpower */ - val = sprom[BCW_SPROM_MAXPWR]; - sc->sc_sprom.maxpower_aphy = (val & 0xff00) >> 8; - sc->sc_sprom.maxpower_bgphy = val & 0xff00; - - /* pa1b */ - sc->sc_sprom.pa1b0 = sprom[BCW_SPROM_PA1B0]; - sc->sc_sprom.pa1b1 = sprom[BCW_SPROM_PA1B1]; - sc->sc_sprom.pa1b2 = sprom[BCW_SPROM_PA1B2]; - - /* idle tssi target */ - val = sprom[BCW_SPROM_IDL_TSSI_TGT]; - sc->sc_sprom.idle_tssi_tgt_aphy = val & 0x00ff; - sc->sc_sprom.idle_tssi_tgt_bgphy = (val & 0xff00) >> 8; - - /* boardflags */ - val = sprom[BCW_SPROM_BOARDFLAGS]; - if (val == 0xffff) - val = 0; - sc->sc_sprom.boardflags = val; - /* TODO boardflags workarounds */ - - /* antenna gain */ - val = sprom[BCW_SPROM_ANTENNA_GAIN]; - if (val == 0 || val == 0xffff) - val = 0x0202; - sc->sc_sprom.antennagain_aphy = ((val & 0xff00) >> 8) * 4; - sc->sc_sprom.antennagain_bgphy = (val & 0x00ff) * 4; - - return (0); -} - -/* - * Init the Microcode Flags Bitfield - * - * http://bcm-specs.sipsolutions.net/MicrocodeFlagsBitfield - */ -void -bcw_microcode_init_fbf(struct bcw_softc *sc) -{ - uint32_t val; - - val = 0; - if (sc->sc_phy_type == BCW_PHY_TYPEA || - sc->sc_phy_type == BCW_PHY_TYPEB || - sc->sc_phy_type == BCW_PHY_TYPEG) - val |= 2; - if (sc->sc_phy_type == BCW_PHY_TYPEG && sc->sc_phy_rev == 1) - val |= 0x20; - if (sc->sc_phy_type == BCW_PHY_TYPEG && - sc->sc_sprom.boardflags & BCW_BF_PACTRL) - val |= 0x40; - if (sc->sc_phy_type == BCW_PHY_TYPEG && sc->sc_phy_rev < 3) - val |= 0x8; - if (sc->sc_sprom.boardflags & BCW_BF_XTAL) - val |= 0x400; - if (sc->sc_phy_type == BCW_PHY_TYPEB) - val |= 0x4; - if (sc->sc_radio_ver == 0x2050 && sc->sc_radio_rev <= 5) - val |= 0x40000; - /* XXX device not up and PCI bus with rev =< 10 set 0x80000 */ - - bcw_shm_write32(sc, BCW_SHM_SHARED, BCW_UCODEFLAGS_OFFSET, val); -} - -/* - * PHY - */ -int -bcw_phy_get(struct bcw_softc *sc) -{ - uint32_t val; - - val = BCW_READ16(sc, 0x3E0); - sc->sc_phy_ver = (val & 0xf000) >> 12; - sc->sc_phy_rev = val & 0xf; - sc->sc_phy_type = (val & 0xf00) >> 8; - - switch (sc->sc_phy_type) { - case BCW_PHY_TYPEA: - DPRINTF(("%s: PHY %d (A) ", - sc->sc_dev.dv_xname, sc->sc_phy_type)); - if (sc->sc_phy_rev != 2 && - sc->sc_phy_rev != 3 && - sc->sc_phy_rev != 5 && - sc->sc_phy_rev != 6 && - sc->sc_phy_rev != 7) { - printf("has invalid revision %d!\n", sc->sc_phy_rev); - return (1); - } - break; - case BCW_PHY_TYPEB: - DPRINTF(("%s: PHY %d (B) ", - sc->sc_dev.dv_xname, sc->sc_phy_type)); - if (sc->sc_phy_rev != 2 && - sc->sc_phy_rev != 4 && - sc->sc_phy_rev != 7) { - printf("has invalid revision %d!\n", sc->sc_phy_rev); - return (1); - } - break; - case BCW_PHY_TYPEG: - DPRINTF(("%s: PHY %d (G) ", - sc->sc_dev.dv_xname, sc->sc_phy_type)); - if (sc->sc_phy_rev != 1 && - sc->sc_phy_rev != 2 && - sc->sc_phy_rev != 4 && - sc->sc_phy_rev != 6 && - sc->sc_phy_rev != 7 && - sc->sc_phy_rev != 8) { - printf("has invalid revision %d!\n", sc->sc_phy_rev); - return (1); - } - break; - default: - DPRINTF(("Unknown PHY type %d!\n", sc->sc_phy_type)); - return (1); - } - - DPRINTF(("ver %d rev %d\n", sc->sc_phy_ver, sc->sc_phy_rev)); - - return (0); -} - -int -bcw_phy_init(struct bcw_softc *sc) -{ - int error = ENODEV; - - switch (sc->sc_phy_type) { - case BCW_PHY_TYPEA: - if (sc->sc_phy_rev == 2 || sc->sc_phy_rev == 3) { - error = 0; - } - break; - case BCW_PHY_TYPEB: - switch (sc->sc_phy_rev) { - case 2: - bcw_phy_initb2(sc); - error = 0; - break; - case 4: - bcw_phy_initb4(sc); - error = 0; - break; - case 5: - bcw_phy_initb5(sc); - error = 0; - break; - case 6: - bcw_phy_initb6(sc); - error = 0; - break; - } - break; - case BCW_PHY_TYPEG: - bcw_phy_initg(sc); - error = 0; - DPRINTF(("%s: PHY type G initialized\n", sc->sc_dev.dv_xname)); - break; - } - - if (error) - printf("%s: PHY type unknown!\n", sc->sc_dev.dv_xname); - - return (error); -} - -void -bcw_phy_initg(struct bcw_softc *sc) -{ - uint16_t tmp; - - if (sc->sc_phy_rev == 1) - bcw_phy_initb5(sc); - else - bcw_phy_initb6(sc); - if (sc->sc_phy_rev >= 2 || sc->sc_phy_connected) - bcw_phy_inita(sc); - - if (sc->sc_phy_rev >= 2) { - bcw_phy_write16(sc, 0x0814, 0); - bcw_phy_write16(sc, 0x0815, 0); - if (sc->sc_phy_rev == 2) - bcw_phy_write16(sc, 0x0811, 0); - else if (sc->sc_phy_rev >= 3) - bcw_phy_write16(sc, 0x0811, 0x0400); - bcw_phy_write16(sc, 0x0015, 0x00c0); - if (sc->sc_phy_connected) { - tmp = bcw_phy_read16(sc, 0x0400) & 0xff; - if (tmp < 6) { - bcw_phy_write16(sc, 0x04c2, 0x1816); - bcw_phy_write16(sc, 0x04c3, 0x8006); - if (tmp != 3) - bcw_phy_write16(sc, 0x04cc, - (bcw_phy_read16(sc, 0x04cc) & - 0x00ff) | 0x1f00); - } - } - } - - if (sc->sc_phy_rev < 3 && sc->sc_phy_connected) - bcw_phy_write16(sc, 0x047e, 0x0078); - if (sc->sc_phy_rev >= 6 && sc->sc_phy_rev <= 8) { - bcw_phy_write16(sc, 0x0801, bcw_phy_read16(sc, 0x0801) | - 0x0080); - bcw_phy_write16(sc, 0x043e, bcw_phy_read16(sc, 0x043e) | - 0x0004); - } - if (sc->sc_phy_rev >= 2 && sc->sc_phy_connected) - bcw_phy_calc_loopback_gain(sc); - if (sc->sc_radio_rev != 8) { - if (sc->sc_radio_initval == 0xffff) - sc->sc_radio_initval = bcw_radio_init2050(sc); - else - bcw_radio_write16(sc, 0x0078, sc->sc_radio_initval); - } - if (sc->sc_radio_txctl2 == 0xffff) - bcw_phy_lo_g_measure(sc); - else { - if (sc->sc_radio_ver == 0x2050 && sc->sc_radio_rev == 8) - bcw_radio_write16(sc, 0x0052, - (sc->sc_radio_txctl1 << 4) | sc->sc_radio_txctl2); - else - bcw_radio_write16(sc, 0x0052, - (bcw_radio_read16(sc, 0x0052) & 0xfff0) | - sc->sc_radio_txctl1); - if (sc->sc_phy_rev >= 6) - bcw_phy_write16(sc, 0x0036, - (bcw_phy_read16(sc, 0x0036) & 0xf000) | - (sc->sc_radio_txctl2 << 12)); - if (sc->sc_sprom.boardflags & BCW_BF_PACTRL) - bcw_phy_write16(sc, 0x002e, 0x8075); - else - bcw_phy_write16(sc, 0x002e, 0x807f); - if (sc->sc_phy_rev < 2) - bcw_phy_write16(sc, 0x002f, 0x0101); - else - bcw_phy_write16(sc, 0x002f, 0x0202); - } - if (sc->sc_phy_connected) { - bcw_phy_lo_adjust(sc, 0); - bcw_phy_write16(sc, 0x080f, 0x8078); - } - - if (!(sc->sc_sprom.boardflags & BCW_BF_RSSI)) { - bcw_radio_nrssi_hw_update(sc, 0xffff); - bcw_radio_calc_nrssi_threshold(sc); - } else if (sc->sc_phy_connected) { - if (sc->sc_radio_nrssi[0] == -1000) { - bcw_radio_calc_nrssi_slope(sc); - } else { - bcw_radio_calc_nrssi_threshold(sc); - } - } - - if (sc->sc_radio_rev == 8) - bcw_phy_write16(sc, 0x0805, 0x3230); - bcw_phy_init_pctl(sc); - if (sc->sc_chip_id == 0x4306 && sc->sc_chip_pkg == 2) { - bcw_phy_write16(sc, 0x0429, bcw_phy_read16(sc, 0x0429) & - 0xbfff); - bcw_phy_write16(sc, 0x04c3, bcw_phy_read16(sc, 0x04c3) & - 0x7fff); - } -} - -void -bcw_phy_initb5(struct bcw_softc *sc) -{ - uint16_t offset; - - if (sc->sc_phy_rev == 1 && sc->sc_radio_rev == 0x2050) - bcw_radio_write16(sc, 0x007a, bcw_radio_read16(sc, 0x007a) | - 0x0050); - - if (sc->sc_board_vendor != PCI_VENDOR_BROADCOM && - sc->sc_board_type != 0x0416) { - for (offset = 0x00a8; offset < 0x00c7; offset++) - bcw_phy_write16(sc, offset, - (bcw_phy_read16(sc, offset) + 0x02020) & 0x3f3f); - } - - bcw_phy_write16(sc, 0x0035, (bcw_phy_read16(sc, 0x0035) & 0xf0ff) | - 0x0700); - - if (sc->sc_radio_rev == 0x2050) - bcw_phy_write16(sc, 0x0038, 0x0667); - - if (sc->sc_phy_connected) { - if (sc->sc_radio_rev == 0x2050) { - bcw_radio_write16(sc, 0x007a, - bcw_radio_read16(sc, 0x007a) | 0x0020); - bcw_radio_write16(sc, 0x0051, - bcw_radio_read16(sc, 0x0051) | 0x0004); - } - BCW_WRITE16(sc, BCW_MMIO_PHY_RADIO, 0); - bcw_phy_write16(sc, 0x0802, bcw_phy_read16(sc, 0x0802) | - 0x0100); - bcw_phy_write16(sc, 0x042b, bcw_phy_read16(sc, 0x042b) | - 0x2000); - bcw_phy_write16(sc, 0x001c, 0x186a); - bcw_phy_write16(sc, 0x0013, - (bcw_phy_read16(sc, 0x0013) & 0x00ff) | 0x1900); - bcw_phy_write16(sc, 0x0035, - (bcw_phy_read16(sc, 0x0035) & 0xffc0) | 0x0064); - bcw_phy_write16(sc, 0x005d, - (bcw_phy_read16(sc, 0x005d) & 0xff80) | 0x000a); - } - - if (sc->sc_phy_rev == 1 && sc->sc_radio_rev == 0x2050) { - bcw_phy_write16(sc, 0x0026, 0xce00); - bcw_phy_write16(sc, 0x0021, 0x3763); - bcw_phy_write16(sc, 0x0022, 0x1bc3); - bcw_phy_write16(sc, 0x0023, 0x06f9); - bcw_phy_write16(sc, 0x0024, 0x037e); - } else - bcw_phy_write16(sc, 0x0026, 0xcc00); - bcw_phy_write16(sc, 0x0030, 0x00c6); - BCW_WRITE16(sc, 0x3ec, 0x3f22); - - if (sc->sc_phy_rev == 1 && sc->sc_radio_rev == 0x2050) - bcw_phy_write16(sc, 0x0020, 0x3e1c); - else - bcw_phy_write16(sc, 0x0020, 0x301c); - - if (sc->sc_phy_rev == 0) - BCW_WRITE16(sc, 0x03e4, 0x3000); - - /* force to channel 7, even if not supported */ - bcw_radio_select_channel(sc, 7, 0); - - if (sc->sc_radio_rev != 0x2050) { - bcw_radio_write16(sc, 0x0075, 0x0080); - bcw_radio_write16(sc, 0x0079, 0x0081); - } - - bcw_radio_write16(sc, 0x0050, 0x0020); - bcw_radio_write16(sc, 0x0050, 0x0023); - - if (sc->sc_radio_rev == 0x2050) { - bcw_radio_write16(sc, 0x0050, 0x0020); - bcw_radio_write16(sc, 0x005a, 0x0070); - } - - bcw_radio_write16(sc, 0x005b, 0x007b); - bcw_radio_write16(sc, 0x005c, 0x00b0); - - bcw_radio_write16(sc, 0x007a, bcw_radio_read16(sc, 0x007a) | 0x0007); - - bcw_radio_select_channel(sc, BCW_RADIO_DEFAULT_CHANNEL_BG, 0); - - bcw_phy_write16(sc, 0x0014, 0x0080); - bcw_phy_write16(sc, 0x0032, 0x00ca); - bcw_phy_write16(sc, 0x88a3, 0x002a); - - bcw_radio_set_txpower_bg(sc, 0xffff, 0xffff, 0xffff); - - if (sc->sc_radio_rev == 0x2050) - bcw_radio_write16(sc, 0x005d, 0x000d); - - BCW_WRITE16(sc, 0x03e4, (BCW_READ16(sc, 0x03e4) & 0xffc0) | 0x0004); -} - -void -bcw_phy_initb2(struct bcw_softc *sc) -{ - uint16_t offset, val; - - BCW_WRITE16(sc, 0x03ec, 0x3f22); - bcw_phy_write16(sc, 0x0020, 0x301c); - bcw_phy_write16(sc, 0x0026, 0x0000); - bcw_phy_write16(sc, 0x0030, 0x00c6); - bcw_phy_write16(sc, 0x0088, 0x3e00); - val = 0x3c3d; - for (offset = 0x0089; offset < 0x00a7; offset++) { - bcw_phy_write16(sc, offset, val); - val -= 0x0202; - } - bcw_phy_write16(sc, 0x03e4, 0x3000); - if (sc->sc_radio_channel == 0xff) - bcw_radio_select_channel(sc, BCW_RADIO_DEFAULT_CHANNEL_BG, 0); - else - bcw_radio_select_channel(sc, sc->sc_radio_channel, 0); - if (sc->sc_radio_ver != 0x2050) { - bcw_radio_write16(sc, 0x0075, 0x0080); - bcw_radio_write16(sc, 0x0079, 0x0081); - } - bcw_radio_write16(sc, 0x0050, 0x0020); - bcw_radio_write16(sc, 0x0050, 0x0023); - if (sc->sc_radio_ver == 0x2050) { - bcw_radio_write16(sc, 0x0050, 0x0020); - bcw_radio_write16(sc, 0x005a, 0x0070); - bcw_radio_write16(sc, 0x005b, 0x007b); - bcw_radio_write16(sc, 0x005c, 0x00b0); - bcw_radio_write16(sc, 0x007a, 0x000f); - bcw_phy_write16(sc, 0x0038, 0x0677); - bcw_radio_init2050(sc); - } - bcw_phy_write16(sc, 0x0014, 0x0080); - bcw_phy_write16(sc, 0x0032, 0x00ca); - bcw_phy_write16(sc, 0x0032, 0x00cc); - bcw_phy_write16(sc, 0x0035, 0x07c2); - bcw_phy_lo_b_measure(sc); - bcw_phy_write16(sc, 0x0026, 0xcc00); - if (sc->sc_radio_ver != 0x2050) - bcw_phy_write16(sc, 0x0026, 0xce00); - BCW_WRITE16(sc, BCW_MMIO_CHANNEL_EXT, 0x1000); - bcw_phy_write16(sc, 0x002a, 0x88a3); - bcw_radio_set_txpower_bg(sc, 0xffff, 0xffff, 0xffff); - bcw_phy_init_pctl(sc); -} - -void -bcw_phy_initb4(struct bcw_softc *sc) -{ - uint16_t offset, val; - - BCW_WRITE16(sc, 0x03ec, 0x3f22); - bcw_phy_write16(sc, 0x0020, 0x301c); - bcw_phy_write16(sc, 0x0026, 0x0000); - bcw_phy_write16(sc, 0x0030, 0x00c6); - bcw_phy_write16(sc, 0x0088, 0x3e00); - val = 0x3c3d; - for (offset = 0x0089; offset < 0x00a7; offset++) { - bcw_phy_write16(sc, offset, val); - val -= 0x0202; - } - bcw_phy_write16(sc, 0x3e4, 0x3000); - if (sc->sc_radio_channel == 0xff) - bcw_radio_select_channel(sc, BCW_RADIO_DEFAULT_CHANNEL_BG, 0); - else - bcw_radio_select_channel(sc, sc->sc_radio_channel, 0); - if (sc->sc_radio_ver == 0x2050) { - bcw_radio_write16(sc, 0x0050, 0x0020); - bcw_radio_write16(sc, 0x005a, 0x0070); - bcw_radio_write16(sc, 0x005b, 0x007b); - bcw_radio_write16(sc, 0x005c, 0x00b0); - bcw_radio_write16(sc, 0x007a, 0x000f); - bcw_phy_write16(sc, 0x0038, 0x0677); - bcw_radio_init2050(sc); - } - bcw_phy_write16(sc, 0x0014, 0x0080); - bcw_phy_write16(sc, 0x0032, 0x00ca); - if (sc->sc_radio_ver == 0x2050) - bcw_phy_write16(sc, 0x0032, 0x00e0); - bcw_phy_write16(sc, 0x0035, 0x07c2); - - bcw_phy_lo_b_measure(sc); - - bcw_phy_write16(sc, 0x0026, 0xcc00); - if (sc->sc_radio_ver == 0x2050) - bcw_phy_write16(sc, 0x0026, 0xce00); - BCW_WRITE16(sc, BCW_MMIO_CHANNEL_EXT, 0x1100); - bcw_phy_write16(sc, 0x002a, 0x88c2); - if (sc->sc_radio_ver == 0x2050) - bcw_phy_write16(sc, 0x002a, 0x88c2); - bcw_radio_set_txpower_bg(sc, 0xffff, 0xffff, 0xffff); - if (sc->sc_sprom.boardflags & BCW_BF_RSSI) { - bcw_radio_calc_nrssi_slope(sc); - bcw_radio_calc_nrssi_threshold(sc); - } - bcw_phy_init_pctl(sc); -} - -void -bcw_phy_initb6(struct bcw_softc *sc) -{ - uint16_t offset, val; - - bcw_phy_write16(sc, 0x003e, 0x817a); - bcw_radio_write16(sc, 0x007a, (bcw_radio_read16(sc, 0x007a) | 0x0058)); - - if (sc->sc_radio_mnf == 0x17f && sc->sc_radio_ver == 0x2050 && - (sc->sc_radio_rev == 3 || sc->sc_radio_rev == 4 || - sc->sc_radio_rev == 5)) { - bcw_radio_write16(sc, 0x0051, 0x001f); - bcw_radio_write16(sc, 0x0052, 0x0040); - bcw_radio_write16(sc, 0x0053, 0x005b); - bcw_radio_write16(sc, 0x0054, 0x0098); - bcw_radio_write16(sc, 0x005a, 0x0088); - bcw_radio_write16(sc, 0x005b, 0x0088); - bcw_radio_write16(sc, 0x005d, 0x0088); - bcw_radio_write16(sc, 0x005e, 0x0088); - bcw_radio_write16(sc, 0x007d, 0x0088); - } - - if (sc->sc_radio_mnf == 0x17f && sc->sc_radio_ver == 0x2050 && - sc->sc_radio_rev == 6) { - bcw_radio_write16(sc, 0x0051, 0); - bcw_radio_write16(sc, 0x0052, 0x0040); - bcw_radio_write16(sc, 0x0053, 0x00b7); - bcw_radio_write16(sc, 0x0054, 0x0098); - bcw_radio_write16(sc, 0x005a, 0x0088); - bcw_radio_write16(sc, 0x005b, 0x008b); - bcw_radio_write16(sc, 0x005c, 0x00b5); - bcw_radio_write16(sc, 0x005d, 0x0088); - bcw_radio_write16(sc, 0x005e, 0x0088); - bcw_radio_write16(sc, 0x007d, 0x0088); - bcw_radio_write16(sc, 0x007c, 0x0001); - bcw_radio_write16(sc, 0x007e, 0x0008); - } - - if (sc->sc_radio_mnf == 0x017f && sc->sc_radio_ver == 0x2050 && - sc->sc_radio_rev == 7) { - bcw_radio_write16(sc, 0x0051, 0); - bcw_radio_write16(sc, 0x0052, 0x0040); - bcw_radio_write16(sc, 0x0053, 0x00b7); - bcw_radio_write16(sc, 0x0054, 0x0098); - bcw_radio_write16(sc, 0x005a, 0x0088); - bcw_radio_write16(sc, 0x005b, 0x00a8); - bcw_radio_write16(sc, 0x005c, 0x0075); - bcw_radio_write16(sc, 0x005d, 0x00f5); - bcw_radio_write16(sc, 0x005e, 0x00b8); - bcw_radio_write16(sc, 0x007d, 0x00e8); - bcw_radio_write16(sc, 0x007c, 0x0001); - bcw_radio_write16(sc, 0x007e, 0x0008); - bcw_radio_write16(sc, 0x007b, 0); - } - - if (sc->sc_radio_mnf == 0x17f && sc->sc_radio_ver == 0x2050 && - sc->sc_radio_rev == 8) { - bcw_radio_write16(sc, 0x0051, 0); - bcw_radio_write16(sc, 0x0052, 0x0040); - bcw_radio_write16(sc, 0x0053, 0x00b7); - bcw_radio_write16(sc, 0x0054, 0x0098); - bcw_radio_write16(sc, 0x005a, 0x0088); - bcw_radio_write16(sc, 0x005b, 0x006b); - bcw_radio_write16(sc, 0x005c, 0x000f); - if (sc->sc_sprom.boardflags & 0x8000) { - bcw_radio_write16(sc, 0x005d, 0x00fa); - bcw_radio_write16(sc, 0x005e, 0x00d8); - } else { - bcw_radio_write16(sc, 0x005d, 0x00f5); - bcw_radio_write16(sc, 0x005e, 0x00b8); - } - bcw_radio_write16(sc, 0x0073, 0x0003); - bcw_radio_write16(sc, 0x007d, 0x00a8); - bcw_radio_write16(sc, 0x007c, 0x0001); - bcw_radio_write16(sc, 0x007e, 0x0008); - } - - val = 0x1e1f; - for (offset = 0x0088; offset < 0x0098; offset++) { - bcw_phy_write16(sc, offset, val); - val -= 0x0202; - } - val = 0x3e3f; - for (offset = 0x0098; offset < 0x00a8; offset++) { - bcw_phy_write16(sc, offset, val); - val -= 0x0202; - } - val = 0x2120; - for (offset = 0x00a8; offset < 0x00c8; offset++) { - bcw_phy_write16(sc, offset, (val & 0x3f3f)); - val += 0x0202; - } - - if (sc->sc_phy_type == BCW_PHY_TYPEG) { - bcw_radio_write16(sc, 0x007a, bcw_radio_read16(sc, 0x007a) | - 0x0020); - bcw_radio_write16(sc, 0x0051, bcw_radio_read16(sc, 0x0051) | - 0x0004); - bcw_phy_write16(sc, 0x0802, bcw_phy_read16(sc, 0x0802) | - 0x0100); - bcw_phy_write16(sc, 0x042b, bcw_phy_read16(sc, 0x042b) | - 0x2000); - } - - /* force to channel 7, even if not supported */ - bcw_radio_select_channel(sc, 7, 0); - - bcw_radio_write16(sc, 0x0050, 0x0020); - bcw_radio_write16(sc, 0x0050, 0x0023); - delay(40); - bcw_radio_write16(sc, 0x007c, (bcw_radio_read16(sc, 0x007c) | - 0x0002)); - bcw_radio_write16(sc, 0x0050, 0x0020); - if (sc->sc_radio_mnf == 0x17f && sc->sc_radio_ver == 0x2050 && - sc->sc_radio_rev <= 2) { - bcw_radio_write16(sc, 0x0050, 0x0020); - bcw_radio_write16(sc, 0x005a, 0x0070); - bcw_radio_write16(sc, 0x005b, 0x007b); - bcw_radio_write16(sc, 0x005c, 0x00b0); - } - bcw_radio_write16(sc, 0x007a, (bcw_radio_read16(sc, 0x007a) & 0x00f8) | - 0x0007); - - bcw_radio_select_channel(sc, BCW_RADIO_DEFAULT_CHANNEL_BG, 0); - - bcw_phy_write16(sc, 0x0014, 0x0200); - if (sc->sc_radio_ver == 0x2050) { - if (sc->sc_radio_rev == 3 || sc->sc_radio_rev == 4 || - sc->sc_radio_rev == 5) - bcw_phy_write16(sc, 0x002a, 0x8ac0); - else - bcw_phy_write16(sc, 0x002a, 0x88c2); - } - bcw_phy_write16(sc, 0x0038, 0x0668); - bcw_radio_set_txpower_bg(sc, 0xffff, 0xffff, 0xffff); - if (sc->sc_radio_ver == 0x2050) { - if (sc->sc_radio_rev == 3 || sc->sc_radio_rev == 4 || - sc->sc_radio_rev == 5) - bcw_phy_write16(sc, 0x005d, - bcw_phy_read16(sc, 0x005d) | 0x0003); - else if (sc->sc_radio_rev <= 2) - bcw_phy_write16(sc, 0x005d, 0x000d); - } - - if (sc->sc_phy_rev == 4) - bcw_phy_write16(sc, 0x0002, (bcw_phy_read16(sc, 0x0002) & - 0xffc0) | 0x0004); - else - BCW_WRITE16(sc, 0x03e4, 0x0009); - if (sc->sc_phy_type == BCW_PHY_TYPEB) { - BCW_WRITE16(sc, 0x03e6, 0x8140); - bcw_phy_write16(sc, 0x0016, 0x0410); - bcw_phy_write16(sc, 0x0017, 0x0820); - bcw_phy_write16(sc, 0x0062, 0x0007); - (void)bcw_radio_calibrationvalue(sc); - bcw_phy_lo_b_measure(sc); - if (sc->sc_sprom.boardflags & BCW_BF_RSSI) { - bcw_radio_calc_nrssi_slope(sc); - bcw_radio_calc_nrssi_threshold(sc); - } - bcw_phy_init_pctl(sc); - } else - BCW_WRITE16(sc, 0x03e6, 0); -} - -void -bcw_phy_inita(struct bcw_softc *sc) -{ - uint16_t tval; - - if (sc->sc_phy_type == BCW_PHY_TYPEA) - bcw_phy_setupa(sc); - else { - bcw_phy_setupg(sc); - if (sc->sc_sprom.boardflags & BCW_BF_PACTRL) - bcw_phy_write16(sc, 0x046e, 0x03cf); - return; - } - - bcw_phy_write16(sc, BCW_PHY_A_CRS, (bcw_phy_read16(sc, BCW_PHY_A_CRS) & - 0xf83c) | 0x0340); - bcw_phy_write16(sc, 0x0034, 0x0001); - - bcw_phy_write16(sc, BCW_PHY_A_CRS, bcw_phy_read16(sc, BCW_PHY_A_CRS) | - (1 << 14)); - bcw_radio_init2060(sc); - - if (sc->sc_board_vendor == PCI_VENDOR_BROADCOM && - (sc->sc_board_type == 0x0416 || sc->sc_board_type == 0x040a)) { - if (sc->sc_radio_lofcal == 0xffff) - bcw_radio_set_tx_iq(sc); - else - bcw_radio_write16(sc, 0x001e, sc->sc_radio_lofcal); - } - - bcw_phy_write16(sc, 0x007a, 0xf111); - - if (sc->sc_phy_savedpctlreg == 0xffff) { - bcw_radio_write16(sc, 0x0019, 0); - bcw_radio_write16(sc, 0x0017, 0x0020); - - tval = bcw_ilt_read(sc, 0x3001); - if (sc->sc_phy_rev == 1) - bcw_ilt_write(sc, 0x3001, (bcw_ilt_read(sc, 0x3001) & - 0xff87) | 0x0058); - else - bcw_ilt_write(sc, 0x3001, (bcw_ilt_read(sc, 0x3001) & - 0xffc3) | 0x002c); - bcw_dummy_transmission(sc); - sc->sc_phy_savedpctlreg = bcw_phy_read16(sc, BCW_PHY_A_PCTL); - bcw_ilt_write(sc, 0x3001, tval); - - bcw_radio_set_txpower_a(sc, 0x0018); - } - bcw_radio_clear_tssi(sc); -} - -void -bcw_phy_setupa(struct bcw_softc *sc) -{ - uint16_t i; - - switch (sc->sc_phy_rev) { - case 2: - bcw_phy_write16(sc, 0x008e, 0x3800); - bcw_phy_write16(sc, 0x0035, 0x03ff); - bcw_phy_write16(sc, 0x0036, 0x0400); - - bcw_ilt_write(sc, 0x3807, 0x0051); - - bcw_phy_write16(sc, 0x001c, 0x0ff9); - bcw_phy_write16(sc, 0x0020, bcw_phy_read16(sc, 0x0020) & - 0xff0f); - bcw_ilt_write(sc, 0x3c0c, 0x07bf); - bcw_radio_write16(sc, 0x0002, 0x07bf); - - bcw_phy_write16(sc, 0x0024, 0x4680); - bcw_phy_write16(sc, 0x0020, 0x0003); - bcw_phy_write16(sc, 0x001d, 0x0f40); - bcw_phy_write16(sc, 0x001f, 0x1c00); - - bcw_phy_write16(sc, 0x002a, (bcw_phy_read16(sc, 0x002a) & - 0x00ff) | 0x0400); - bcw_phy_write16(sc, 0x002b, bcw_phy_read16(sc, 0x002b) & - 0xfbff); - bcw_phy_write16(sc, 0x008e, 0x58c1); - - bcw_ilt_write(sc, 0x0803, 0x000f); - bcw_ilt_write(sc, 0x0804, 0x001f); - bcw_ilt_write(sc, 0x0805, 0x002a); - bcw_ilt_write(sc, 0x0805, 0x0030); - bcw_ilt_write(sc, 0x0807, 0x003a); - - bcw_ilt_write(sc, 0x0000, 0x0013); - bcw_ilt_write(sc, 0x0001, 0x0013); - bcw_ilt_write(sc, 0x0002, 0x0013); - bcw_ilt_write(sc, 0x0003, 0x0013); - bcw_ilt_write(sc, 0x0004, 0x0015); - bcw_ilt_write(sc, 0x0005, 0x0015); - bcw_ilt_write(sc, 0x0006, 0x0019); - - bcw_ilt_write(sc, 0x0404, 0x0003); - bcw_ilt_write(sc, 0x0405, 0x0003); - bcw_ilt_write(sc, 0x0406, 0x0007); - - for (i = 0; i < 16; i++) - bcw_ilt_write(sc, 0x4000 + i, (0x8 + i) & 0x000f); - - bcw_ilt_write(sc, 0x3003, 0x1044); - bcw_ilt_write(sc, 0x3004, 0x7201); - bcw_ilt_write(sc, 0x3006, 0x0040); - bcw_ilt_write(sc, 0x3001, (bcw_ilt_read(sc, 0x3001) & 0x0010) | - 0x0008); - - for (i = 0; i < BCW_ILT_FINEFREQA_SIZE; i++) - bcw_ilt_write(sc, 0x5800 + i, bcw_ilt_finefreqa[i]); - - for (i = 0; i < BCW_ILT_NOISEA2_SIZE; i++) - bcw_ilt_write(sc, 0x1800 + i, bcw_ilt_noisea2[i]); - - for (i = 0; i < BCW_ILT_ROTOR_SIZE; i++) - bcw_ilt_write(sc, 0x2000 + i, bcw_ilt_rotor[i]); - - bcw_phy_init_noisescaletbl(sc); - for (i = 0; i < BCW_ILT_RETARD_SIZE; i++) - bcw_ilt_write(sc, 0x2400 + i, bcw_ilt_retard[i]); - - break; - case 3: - for (i = 0; i< 64; i++) - bcw_ilt_write(sc, 0x4000 + i, i); - - bcw_ilt_write(sc, 0x3807, 0x0051); - - bcw_phy_write16(sc, 0x001c, 0x0ff9); - bcw_phy_write16(sc, 0x0020, bcw_phy_read16(sc, 0x0020) & - 0x0ff0f); - bcw_radio_write16(sc, 0x0002, 0x07bf); - - bcw_phy_write16(sc, 0x0024, 0x4680); - bcw_phy_write16(sc, 0x0020, 0x0003); - bcw_phy_write16(sc, 0x001d, 0x0f40); - bcw_phy_write16(sc, 0x001f, 0x1c00); - bcw_phy_write16(sc, 0x002a, (bcw_phy_read16(sc, 0x002a) & - 0x00ff) | 0x0400); - bcw_ilt_write(sc, 0x3001, (bcw_ilt_read(sc, 0x3001) & - 0x0010) | 0x0008); - for (i = 0; i < BCW_ILT_NOISEA3_SIZE; i++) - bcw_ilt_write(sc, 0x1800 + i, bcw_ilt_noisea3[i]); - - bcw_phy_init_noisescaletbl(sc); - for (i = 0; i < BCW_ILT_SIGMASQR_SIZE; i++) - bcw_ilt_write(sc, 0x5000 + i, bcw_ilt_sigmasqr1[i]); - - bcw_phy_write16(sc, 0x0003, 0x1808); - - bcw_ilt_write(sc, 0x0803, 0x000f); - bcw_ilt_write(sc, 0x0804, 0x001f); - bcw_ilt_write(sc, 0x0805, 0x002a); - bcw_ilt_write(sc, 0x0805, 0x0030); - bcw_ilt_write(sc, 0x0807, 0x003a); - - bcw_ilt_write(sc, 0x0000, 0x0013); - bcw_ilt_write(sc, 0x0001, 0x0013); - bcw_ilt_write(sc, 0x0002, 0x0013); - bcw_ilt_write(sc, 0x0003, 0x0013); - bcw_ilt_write(sc, 0x0004, 0x0015); - bcw_ilt_write(sc, 0x0005, 0x0015); - bcw_ilt_write(sc, 0x0006, 0x0019); - - bcw_ilt_write(sc, 0x0404, 0x0003); - bcw_ilt_write(sc, 0x0405, 0x0003); - bcw_ilt_write(sc, 0x0406, 0x0007); - - bcw_ilt_write(sc, 0x3c02, 0x000f); - bcw_ilt_write(sc, 0x3c03, 0x0014); - break; - default: - /* XXX panic()? */ - break; - } -} - -void -bcw_phy_setupg(struct bcw_softc *sc) -{ - uint16_t i; - - if (sc->sc_phy_rev == 1) { - bcw_phy_write16(sc, 0x0406, 0x4f19); - bcw_phy_write16(sc, BCW_PHY_G_CRS, (bcw_phy_read16(sc, - BCW_PHY_G_CRS) & 0xfc3f) | 0x0340); - bcw_phy_write16(sc, 0x042c, 0x005a); - bcw_phy_write16(sc, 0x0427, 0x001a); - - for (i = 0; i < BCW_ILT_FINEFREQG_SIZE; i++) - bcw_ilt_write(sc, 0x5800 + i, bcw_ilt_finefreqg[i]); - - for (i = 0; i < BCW_ILT_NOISEG1_SIZE; i++) - bcw_ilt_write(sc, 0x1800 + i, bcw_ilt_noiseg1[i]); - - for (i = 0; i < BCW_ILT_ROTOR_SIZE; i++) - bcw_ilt_write(sc, 0x2000 + i, bcw_ilt_rotor[i]); - } else { - bcw_radio_nrssi_hw_write(sc, 0xba98, (int16_t) 0x7654); - - if (sc->sc_phy_type == 2) { - bcw_phy_write16(sc, 0x04c0, 0x1861); - bcw_phy_write16(sc, 0x04c1, 0x0271); - } else { - bcw_phy_write16(sc, 0x04c0, 0x0098); - bcw_phy_write16(sc, 0x04c1, 0x0070); - bcw_phy_write16(sc, 0x04c9, 0x0080); - } - bcw_phy_write16(sc, 0x042b, bcw_phy_read16(sc, 0x042b) | - 0x0800); - - for (i = 0; i < 64; i++) - bcw_ilt_write(sc, 0x4000 + i, i); - for (i = 0; i < BCW_ILT_NOISEG2_SIZE; i++) - bcw_ilt_write(sc, 0x1800 + i, bcw_ilt_noiseg2[i]); - } - - if (sc->sc_phy_rev <= 2) - for (i = 0; i < BCW_ILT_NOISESCALEG_SIZE; i++) - bcw_ilt_write(sc, 0x1400 + i, bcw_ilt_noisescaleg1[i]); - else if ((sc->sc_phy_rev >= 7) && (bcw_phy_read16(sc, 0x0449) & 0x0200)) - for (i = 0; i < BCW_ILT_NOISESCALEG_SIZE; i++) - bcw_ilt_write(sc, 0x1400 + i, bcw_ilt_noisescaleg3[i]); - else - for (i = 0; i < BCW_ILT_NOISESCALEG_SIZE; i++) - bcw_ilt_write(sc, 0x1400 + i, bcw_ilt_noisescaleg2[i]); - - if (sc->sc_phy_rev == 2) - for (i = 0; i < BCW_ILT_SIGMASQR_SIZE; i++) - bcw_ilt_write(sc, 0x5000 + i, bcw_ilt_sigmasqr1[i]); - else if ((sc->sc_phy_rev > 2) && (sc->sc_phy_rev <= 8)) - for (i = 0; i < BCW_ILT_SIGMASQR_SIZE; i++) - bcw_ilt_write(sc, 0x5000 + i, bcw_ilt_sigmasqr2[i]); - - if (sc->sc_phy_rev == 1) { - for (i = 0; i < BCW_ILT_RETARD_SIZE; i++) - bcw_ilt_write(sc, 0x2400 + i, bcw_ilt_retard[i]); - for (i = 0; i < 4; i++) { - bcw_ilt_write(sc, 0x5404 + i, 0x0020); - bcw_ilt_write(sc, 0x5408 + i, 0x0020); - bcw_ilt_write(sc, 0x540c + i, 0x0020); - bcw_ilt_write(sc, 0x5410 + i, 0x0020); - } - bcw_phy_agcsetup(sc); - - if (sc->sc_board_vendor == PCI_VENDOR_BROADCOM && - sc->sc_board_type == 0x0416 && - sc->sc_board_rev == 0x0017) - return; - - bcw_ilt_write(sc, 0x5001, 0x0002); - bcw_ilt_write(sc, 0x5002, 0x0001); - } else { - for (i = 0; i <= 0x2f; i++) - bcw_ilt_write(sc, 0x1000 + i, 0x0820); - bcw_phy_agcsetup(sc); - bcw_phy_read16(sc, 0x0400); - bcw_phy_write16(sc, 0x0403, 0x1000); - bcw_ilt_write(sc, 0x3c02, 0x000f); - bcw_ilt_write(sc, 0x3c03, 0x0014); - - if (sc->sc_board_vendor == PCI_VENDOR_BROADCOM && - sc->sc_board_type == 0x0416 && - sc->sc_board_rev == 0x0017) - return; - - bcw_ilt_write(sc, 0x0401, 0x0002); - bcw_ilt_write(sc, 0x0402, 0x0001); - } -} - -void -bcw_phy_calc_loopback_gain(struct bcw_softc *sc) -{ - uint16_t backup_phy[15]; - uint16_t backup_radio[3]; - uint16_t backup_bband; - uint16_t i; - uint16_t loop1_cnt, loop1_done, loop1_omitted; - uint16_t loop2_done; - - backup_phy[0] = bcw_phy_read16(sc, 0x0429); - backup_phy[1] = bcw_phy_read16(sc, 0x0001); - backup_phy[2] = bcw_phy_read16(sc, 0x0811); - backup_phy[3] = bcw_phy_read16(sc, 0x0812); - backup_phy[4] = bcw_phy_read16(sc, 0x0814); - backup_phy[5] = bcw_phy_read16(sc, 0x0815); - backup_phy[6] = bcw_phy_read16(sc, 0x005a); - backup_phy[7] = bcw_phy_read16(sc, 0x0059); - backup_phy[8] = bcw_phy_read16(sc, 0x0058); - backup_phy[9] = bcw_phy_read16(sc, 0x000a); - backup_phy[10] = bcw_phy_read16(sc, 0x0003); - backup_phy[11] = bcw_phy_read16(sc, 0x080f); - backup_phy[12] = bcw_phy_read16(sc, 0x0810); - backup_phy[13] = bcw_phy_read16(sc, 0x002b); - backup_phy[14] = bcw_phy_read16(sc, 0x0015); - bcw_phy_read16(sc, 0x002d); - backup_bband = sc->sc_radio_baseband_atten; - backup_radio[0] = bcw_radio_read16(sc, 0x0052); - backup_radio[1] = bcw_radio_read16(sc, 0x0043); - backup_radio[2] = bcw_radio_read16(sc, 0x007a); - - bcw_phy_write16(sc, 0x0429, bcw_phy_read16(sc, 0x0429) & 0x3fff); - bcw_phy_write16(sc, 0x0001, bcw_phy_read16(sc, 0x0001) & 0x8000); - bcw_phy_write16(sc, 0x0811, bcw_phy_read16(sc, 0x0811) & 0x0002); - bcw_phy_write16(sc, 0x0812, bcw_phy_read16(sc, 0x0812) & 0xfffd); - bcw_phy_write16(sc, 0x0811, bcw_phy_read16(sc, 0x0811) & 0x0001); - bcw_phy_write16(sc, 0x0812, bcw_phy_read16(sc, 0x0812) & 0xfffe); - bcw_phy_write16(sc, 0x0814, bcw_phy_read16(sc, 0x0814) & 0x0001); - bcw_phy_write16(sc, 0x0815, bcw_phy_read16(sc, 0x0815) & 0xfffe); - bcw_phy_write16(sc, 0x0814, bcw_phy_read16(sc, 0x0814) & 0x0002); - bcw_phy_write16(sc, 0x0815, bcw_phy_read16(sc, 0x0815) & 0xfffd); - bcw_phy_write16(sc, 0x0811, bcw_phy_read16(sc, 0x0811) & 0x000c); - bcw_phy_write16(sc, 0x0812, bcw_phy_read16(sc, 0x0812) & 0x000c); - - bcw_phy_write16(sc, 0x0811, (bcw_phy_read16(sc, 0x0811) & 0xffcf) | - 0x0030); - bcw_phy_write16(sc, 0x0812, (bcw_phy_read16(sc, 0x0812) & 0xffcf) | - 0x0010); - - bcw_phy_write16(sc, 0x005a, 0x0780); - bcw_phy_write16(sc, 0x0059, 0xc810); - bcw_phy_write16(sc, 0x0058, 0x000d); - if (sc->sc_phy_ver == 9) - bcw_phy_write16(sc, 0x0003, 0x0122); - else - bcw_phy_write16(sc, 0x000a, bcw_phy_read16(sc, 0x000a) | - 0x2000); - bcw_phy_write16(sc, 0x0814, bcw_phy_read16(sc, 0x0814) | 0x0004); - bcw_phy_write16(sc, 0x0815, bcw_phy_read16(sc, 0x0815) & 0xfffb); - bcw_phy_write16(sc, 0x0003, (bcw_phy_read16(sc, 0x0003) & 0xff9f) | - 0x0040); - if (sc->sc_radio_ver == 0x2050 && sc->sc_radio_rev == 2) { - bcw_radio_write16(sc, 0x0052, 0x0000); - bcw_radio_write16(sc, 0x0043, (bcw_radio_read16(sc, 0x0043) & - 0xfff0) | 0x0009); - loop1_cnt = 9; - } else if (sc->sc_radio_rev == 8) { - bcw_radio_write16(sc, 0x0043, 0x000f); - loop1_cnt = 15; - } else - loop1_cnt = 0; - - bcw_phy_set_baseband_atten(sc, 11); - - if (sc->sc_phy_rev >= 3) - bcw_phy_write16(sc, 0x080f, 0xc020); - else - bcw_phy_write16(sc, 0x080f, 0x8020); - bcw_phy_write16(sc, 0x0810, 0x0000); - - bcw_phy_write16(sc, 0x002b, (bcw_phy_read16(sc, 0x002b) & 0xffc0) | - 0x0001); - bcw_phy_write16(sc, 0x002b, (bcw_phy_read16(sc, 0x002b) & 0xc0ff) | - 0x0800); - bcw_phy_write16(sc, 0x0811, bcw_phy_read16(sc, 0x0811) | 0x0100); - bcw_phy_write16(sc, 0x0812, bcw_phy_read16(sc, 0x0812) | 0xcfff); - if (sc->sc_sprom.boardflags & BCW_BF_EXTLNA) { - if (sc->sc_phy_rev >= 7) { - bcw_phy_write16(sc, 0x0811, bcw_phy_read16(sc, 0x0811) | - 0x0800); - bcw_phy_write16(sc, 0x0812, bcw_phy_read16(sc, 0x0812) | - 0x8000); - } - } - bcw_radio_write16(sc, 0x007a, bcw_radio_read16(sc, 0x007a) & 0x00f7); - - for (i = 0; i < loop1_cnt; i++) { - bcw_radio_write16(sc, 0x0043, loop1_cnt); - bcw_phy_write16(sc, 0x0812, (bcw_phy_read16(sc, 0x0812) & - 0xf0ff) | (i << 8)); - bcw_phy_write16(sc, 0x0015, (bcw_phy_read16(sc, 0x0015) & - 0x0fff) | 0xa000); - bcw_phy_write16(sc, 0x0015, (bcw_phy_read16(sc, 0x0015) & - 0x0fff) | 0xf000); - delay(20); - if (bcw_phy_read16(sc, 0x0002d) >= 0x0dfc) - break; - } - loop1_done = i; - loop1_omitted = loop1_cnt - loop1_done; - - loop2_done = 0; - if (loop1_done >= 8) { - bcw_phy_write16(sc, 0x0812, bcw_phy_read16(sc, 0x0812) | - 0x0030); - for (i = loop1_done - 8; i < 16; i++) { - bcw_phy_write16(sc, 0x0812, - (bcw_phy_read16(sc, 0x0812) & 0xf0ff) | (i << 8)); - bcw_phy_write16(sc, 0x0015, - (bcw_phy_read16(sc, 0x0015) & 0x0fff) | 0xa000); - bcw_phy_write16(sc, 0x0015, - (bcw_phy_read16(sc, 0x0015) & 0x0fff) | 0xf000); - delay(20); - if (bcw_phy_read16(sc, 0x002d) >= 0x0dfc) - break; - } - } - - bcw_phy_write16(sc, 0x0814, backup_phy[4]); - bcw_phy_write16(sc, 0x0815, backup_phy[5]); - bcw_phy_write16(sc, 0x005a, backup_phy[6]); - bcw_phy_write16(sc, 0x0059, backup_phy[7]); - bcw_phy_write16(sc, 0x0058, backup_phy[8]); - bcw_phy_write16(sc, 0x000a, backup_phy[9]); - bcw_phy_write16(sc, 0x0003, backup_phy[10]); - bcw_phy_write16(sc, 0x080f, backup_phy[11]); - bcw_phy_write16(sc, 0x0810, backup_phy[12]); - bcw_phy_write16(sc, 0x002b, backup_phy[13]); - bcw_phy_write16(sc, 0x0015, backup_phy[14]); - - bcw_phy_set_baseband_atten(sc, backup_bband); - - bcw_radio_write16(sc, 0x0052, backup_radio[0]); - bcw_radio_write16(sc, 0x0043, backup_radio[1]); - bcw_radio_write16(sc, 0x007a, backup_radio[2]); - - bcw_phy_write16(sc, 0x0811, backup_phy[2] | 0x0003); - delay(10); - bcw_phy_write16(sc, 0x0811, backup_phy[2]); - bcw_phy_write16(sc, 0x0812, backup_phy[3]); - bcw_phy_write16(sc, 0x0429, backup_phy[0]); - bcw_phy_write16(sc, 0x0001, backup_phy[1]); - - sc->sc_phy_loopback_gain[0] = ((loop1_done * 6) - (loop1_omitted * 4)) - - 11; - sc->sc_phy_loopback_gain[1] = (24 - (3 * loop2_done)) * 2; -} - -void -bcw_phy_agcsetup(struct bcw_softc *sc) -{ - uint16_t offset = 0; - - if (sc->sc_phy_rev == 1) - offset = 0x4c00; - - bcw_ilt_write(sc, offset, 0x00fe); - bcw_ilt_write(sc, offset + 1, 0x000d); - bcw_ilt_write(sc, offset + 2, 0x0013); - bcw_ilt_write(sc, offset + 3, 0x0019); - - if (sc->sc_phy_rev == 1) { - bcw_ilt_write(sc, 0x1800, 0x2710); - bcw_ilt_write(sc, 0x1801, 0x9b83); - bcw_ilt_write(sc, 0x1802, 0x9b83); - bcw_ilt_write(sc, 0x1803, 0x0f8d); - bcw_phy_write16(sc, 0x0455, 0x0004); - } - - bcw_phy_write16(sc, 0x04a5, (bcw_phy_read16(sc, 0x04a5) & 0x00ff) | - 0x5700); - bcw_phy_write16(sc, 0x041a, (bcw_phy_read16(sc, 0x041a) & 0xff80) | - 0x000f); - bcw_phy_write16(sc, 0x041a, (bcw_phy_read16(sc, 0x041a) & 0xc07f) | - 0x2b80); - bcw_phy_write16(sc, 0x048c, (bcw_phy_read16(sc, 0x048c) & 0xf0ff) | - 0x0300); - - bcw_radio_write16(sc, 0x007a, bcw_radio_read16(sc, 0x007a) | 0x0008); - - bcw_phy_write16(sc, 0x04a0, (bcw_phy_read16(sc, 0x04a0) & 0xfff0) | - 0x0008); - bcw_phy_write16(sc, 0x04a1, (bcw_phy_read16(sc, 0x04a1) & 0xf0ff) | - 0x0600); - bcw_phy_write16(sc, 0x04a2, (bcw_phy_read16(sc, 0x04a2) & 0xf0ff) | - 0x0700); - bcw_phy_write16(sc, 0x04a0, (bcw_phy_read16(sc, 0x04a0) & 0xf0ff) | - 0x0100); - - if (sc->sc_phy_rev == 1) - bcw_phy_write16(sc, 0x04a2, (bcw_phy_read16(sc, 0x4a2) & - 0xfff0) | 0x0007); - - bcw_phy_write16(sc, 0x0488, (bcw_phy_read16(sc, 0x0488) & 0xff00) | - 0x001c); - bcw_phy_write16(sc, 0x0488, (bcw_phy_read16(sc, 0x0488) & 0xc0ff) | - 0x0200); - bcw_phy_write16(sc, 0x0496, (bcw_phy_read16(sc, 0x0496) & 0xff00) | - 0x001c); - bcw_phy_write16(sc, 0x0489, (bcw_phy_read16(sc, 0x0489) & 0xff00) | - 0x0020); - bcw_phy_write16(sc, 0x0489, (bcw_phy_read16(sc, 0x0489) & 0xc0ff) | - 0x0200); - bcw_phy_write16(sc, 0x0482, (bcw_phy_read16(sc, 0x0482) & 0xff00) | - 0x001c); - bcw_phy_write16(sc, 0x0496, (bcw_phy_read16(sc, 0x0496) & 0x00ff) | - 0x1a00); - bcw_phy_write16(sc, 0x0481, (bcw_phy_read16(sc, 0x0481) & 0xff00) | - 0x0028); - bcw_phy_write16(sc, 0x0481, (bcw_phy_read16(sc, 0x0481) & 0x00ff) | - 0x2c00); - - if (sc->sc_phy_rev == 1) { - bcw_phy_write16(sc, 0x0430, 0x092b); - bcw_phy_write16(sc, 0x041b, (bcw_phy_read16(sc, 0x041b) & - 0xffe1) | 0x0002); - } else { - bcw_phy_write16(sc, 0x041b, bcw_phy_read16(sc, 0x41b) & - 0xffe1); - bcw_phy_write16(sc, 0x041f, 0x287a); - bcw_phy_write16(sc, 0x0420, (bcw_phy_read16(sc, 0x0420) & - 0xfff0) | 0x0004); - } - - if (sc->sc_phy_rev > 2) { - bcw_phy_write16(sc, 0x0422, 0x287a); - bcw_phy_write16(sc, 0x0420, (bcw_phy_read16(sc, 0x0420) & - 0x0fff) | 0x3000); - } - - bcw_phy_write16(sc, 0x04a8, (bcw_phy_read16(sc, 0x04a8) & 0x8080) | - 0x7874); - bcw_phy_write16(sc, 0x048e, 0x1c00); - - if (sc->sc_phy_rev == 1) { - bcw_phy_write16(sc, 0x04ab, (bcw_phy_read16(sc, 0x04ab) & - 0xf0ff) | 0x0600); - bcw_phy_write16(sc, 0x048b, 0x005e); - bcw_phy_write16(sc, 0x048c, (bcw_phy_read16(sc, 0x048c) & - 0xff00) | 0x001e); - bcw_phy_write16(sc, 0x048d, 0x0002); - } - - bcw_ilt_write(sc, offset + 0x0800, 0); - bcw_ilt_write(sc, offset + 0x0801, 7); - bcw_ilt_write(sc, offset + 0x0802, 16); - bcw_ilt_write(sc, offset + 0x0803, 28); -} - -void -bcw_phy_init_pctl(struct bcw_softc *sc) -{ - uint16_t saved_batt = 0, saved_ratt = 0, saved_txctl1 = 0; - int must_reset_txpower = 0; - - if (sc->sc_board_vendor == PCI_VENDOR_BROADCOM && - sc->sc_board_type == 0x0416) - return; - - BCW_WRITE16(sc, 0x03e6, BCW_READ16(sc, 0x03e6) & 0xffdf); - bcw_phy_write16(sc, 0x0028, 0x8018); - - if (sc->sc_phy_type == BCW_PHY_TYPEG) { - if (!sc->sc_phy_connected) - return; - bcw_phy_write16(sc, 0x047a, 0xc111); - } - if (sc->sc_phy_savedpctlreg != 0xffff) - return; - - if (sc->sc_phy_type == BCW_PHY_TYPEB && - sc->sc_phy_rev >= 2 && - sc->sc_radio_ver == 0x2050) { - bcw_radio_write16(sc, 0x0076, bcw_radio_read16(sc, 0x0076) | - 0x0084); - } else { - saved_batt = sc->sc_radio_baseband_atten; - saved_ratt = sc->sc_radio_radio_atten; - saved_txctl1 = sc->sc_radio_txctl1; - if ((sc->sc_radio_rev >= 6) && (sc->sc_radio_rev <= 8)) - bcw_radio_set_txpower_bg(sc, 0xb, 0x1f, 0); - else - bcw_radio_set_txpower_bg(sc, 0xb, 9, 0); - must_reset_txpower = 1; - } - bcw_dummy_transmission(sc); - - sc->sc_phy_savedpctlreg = bcw_phy_read16(sc, BCW_PHY_G_PCTL); - - if (must_reset_txpower) - bcw_radio_set_txpower_bg(sc, saved_batt, saved_ratt, - saved_txctl1); - else - bcw_radio_write16(sc, 0x0076, bcw_radio_read16(sc, 0x0076) & - 0xff7b); - bcw_radio_clear_tssi(sc); -} - -void -bcw_phy_init_noisescaletbl(struct bcw_softc *sc) -{ - int i; - - bcw_phy_write16(sc, BCW_PHY_ILT_A_CTRL, 0x1400); - for (i = 0; i < 12; i++) { - if (sc->sc_phy_rev == 2) - bcw_phy_write16(sc, BCW_PHY_ILT_A_DATA1, 0x6767); - else - bcw_phy_write16(sc, BCW_PHY_ILT_A_DATA1, 0x2323); - } - - if (sc->sc_phy_rev == 2) - bcw_phy_write16(sc, BCW_PHY_ILT_A_DATA1, 0x6700); - else - bcw_phy_write16(sc, BCW_PHY_ILT_A_DATA1, 0x2300); - - for (i = 0; i < 11; i++) { - if (sc->sc_phy_rev == 2) - bcw_phy_write16(sc, BCW_PHY_ILT_A_DATA1, 0x6767); - else - bcw_phy_write16(sc, BCW_PHY_ILT_A_DATA1, 0x0023); - } - - if (sc->sc_phy_rev == 2) - bcw_phy_write16(sc, BCW_PHY_ILT_A_DATA1, 0x0067); - else - bcw_phy_write16(sc, BCW_PHY_ILT_A_DATA1, 0x0023); -} - -void -bcw_phy_set_baseband_atten(struct bcw_softc *sc, - uint16_t baseband_atten) -{ - uint16_t val; - - if (sc->sc_phy_ver == 0) { - val = (BCW_READ16(sc, 0x03e6) & 0xfff0); - val |= (baseband_atten & 0x000f); - BCW_WRITE16(sc, 0x03e6, val); - return; - } - - if (sc->sc_phy_ver > 1) { - val = bcw_phy_read16(sc, 0x0060) & ~0x003c; - val |= (baseband_atten << 2) & 0x003c; - } else { - val = bcw_phy_read16(sc, 0x0060) & ~0x0078; - val |= (baseband_atten << 3) & 0x0078; - } - bcw_phy_write16(sc, 0x0060, val); -} - -int8_t -bcw_phy_estimate_powerout(struct bcw_softc *sc, int8_t tssi) -{ - int8_t dbm = 0; - int32_t tmp; - - tmp = sc->sc_phy_idle_tssi; - tmp += tssi; - tmp -= sc->sc_phy_savedpctlreg; - - switch (sc->sc_phy_type) { - case BCW_PHY_TYPEA: - tmp += 0x80; - tmp = bcw_lv(tmp, 0x00, 0xff); - //dbm = sc->sc_phy_tssi2dbm[tmp]; /* XXX */ - break; - case BCW_PHY_TYPEB: - case BCW_PHY_TYPEG: - tmp = bcw_lv(tmp, 0x00, 0x3f); - //dbm = sc->sc_phy_tssi2dbm[tmp]; /* XXX "/ - break; - default: - /* XXX panic()? */ - break; - } - - return (dbm); -} - -void -bcw_phy_xmitpower(struct bcw_softc *sc) -{ - if (sc->sc_phy_savedpctlreg == 0xffff) - return; - if (sc->sc_board_type == 0x0416 && - sc->sc_board_vendor == PCI_VENDOR_BROADCOM) - return; - - switch (sc->sc_phy_type) { - case BCW_PHY_TYPEA: - /* nohting todo for A PHYs yet */ - break; - case BCW_PHY_TYPEB: - case BCW_PHY_TYPEG: { - uint16_t tmp; - uint16_t txpower; - int8_t v0, v1, v2, v3; - int8_t average; - uint8_t max_pwr; - int16_t desired_pwr, estimated_pwr, pwr_adjust; - int16_t radio_att_delta, baseband_att_delta; - int16_t radio_attenuation, baseband_attenuation; - - tmp = bcw_shm_read16(sc, BCW_SHM_SHARED, 0x0058); - v0 = (int8_t)(tmp & 0x00ff); - v1 = (int8_t)((tmp & 0xff00) >> 8); - tmp = bcw_shm_read16(sc, BCW_SHM_SHARED, 0x005a); - v2 = (int8_t)(tmp & 0x00ff); - v3 = (int8_t)((tmp & 0xff00) >> 8); - tmp = 0; - - if (v0 == 0x7f || v1 == 0x7f || v2 == 0x7f || v3 == 0x07) { - tmp = bcw_shm_read16(sc, BCW_SHM_SHARED, - 0x0070); - v0 = (int8_t)(tmp & 0x00ff); - v1 = (int8_t)((tmp & 0xff00) >> 8); - tmp = bcw_shm_read16(sc, BCW_SHM_SHARED, - 0x0072); - - v2 = (int8_t)(tmp & 0x00ff); - v3 = (int8_t)((tmp & 0xff00) >> 8); - if (v0 == 0x7f || v1 == 0x7f || v2 == 0x7f || - v3 == 0x7f) - return; - v0 = (v0 + 0x20) & 0x3f; - v1 = (v1 + 0x20) & 0x3f; - v2 = (v2 + 0x20) & 0x3f; - v3 = (v3 + 0x20) & 0x3f; - tmp = 1; - } - bcw_radio_clear_tssi(sc); - - average = (v0 + v1 + v2 + v3 + 2) / 4; - - if (tmp && (bcw_shm_read16(sc, BCW_SHM_SHARED, 0x005e) & - 0x8)) - average -= 13; - - bcw_phy_estimate_powerout(sc, average); - - if ((sc->sc_sprom.boardflags & BCW_BF_PACTRL) && - (sc->sc_phy_type == BCW_PHY_TYPEG)) - max_pwr -= 0x3; - - desired_pwr = bcw_lv(sc->sc_radio_txpower_desired, 0, max_pwr); - /* check if we need to adjust the current power */ - pwr_adjust = desired_pwr - estimated_pwr; - radio_att_delta = -(pwr_adjust + 7) >> 3; - baseband_att_delta = -(pwr_adjust >> 1) - (4 * radio_att_delta); - if ((radio_att_delta == 0) && (baseband_att_delta == 0)) { - bcw_phy_lo_mark_current_used(sc); - return; - } - - /* calculate the new attenuation values */ - baseband_attenuation = sc->sc_radio_baseband_atten; - baseband_attenuation += baseband_att_delta; - radio_attenuation = sc->sc_radio_radio_atten; - radio_attenuation += radio_att_delta; - - if (radio_attenuation < 0) { - baseband_attenuation -= (4 * -radio_attenuation); - radio_attenuation = 0; - } else if (radio_attenuation > 9) { - baseband_attenuation += (4 * (radio_attenuation - 9)); - radio_attenuation = 9; - } else { - while (baseband_attenuation < 0 && - radio_attenuation > 0) { - baseband_attenuation += 4; - radio_attenuation--; - } - while (baseband_attenuation > 11 && - radio_attenuation < 9) { - baseband_attenuation -= 4; - radio_attenuation++; - } - } - baseband_attenuation = bcw_lv(baseband_attenuation, 0, 11); - - txpower = sc->sc_radio_txctl1; - if ((sc->sc_radio_ver == 0x02050) && (sc->sc_radio_rev == 2)) { - if (radio_attenuation <= 1) { - if (txpower == 0) { - txpower = 3; - radio_attenuation += 2; - baseband_attenuation += 2; - } else if (sc->sc_sprom.boardflags & - BCW_BF_PACTRL) { - baseband_attenuation += 4 * - (radio_attenuation - 2); - radio_attenuation = 2; - } - } else if (radio_attenuation > 4 && txpower != 0) { - txpower = 0; - if (baseband_attenuation < 3) { - radio_attenuation -= 3; - baseband_attenuation += 2; - } else { - radio_attenuation -= 2; - baseband_attenuation -= 2; - } - } - } - sc->sc_radio_txctl1 = txpower; - baseband_attenuation = bcw_lv(baseband_attenuation, 0, 11); - radio_attenuation = bcw_lv(radio_attenuation, 0, 9); - - bcw_phy_lock(sc); - bcw_radio_lock(sc); - bcw_radio_set_txpower_bg(sc, baseband_attenuation, - radio_attenuation, txpower); - bcw_phy_lo_mark_current_used(sc); - bcw_radio_unlock(sc); - bcw_phy_unlock(sc); - break; - } - default: - /* XXX panic()? */ - break; - } -} - -uint16_t -bcw_phy_lo_b_r15_loop(struct bcw_softc *sc) -{ - int i; - uint16_t r = 0; - - /* XXX splnet()? */ - for (i = 0; i < 10; i++) { - bcw_phy_write16(sc, 0x0015, 0xafa0); - delay(1); - bcw_phy_write16(sc, 0x0015, 0xefa0); - delay(10); - bcw_phy_write16(sc, 0x0015, 0xffa0); - delay(40); - r += bcw_phy_read16(sc, 0x002c); - } - /* XXX splnet()? */ - - return (r); -} - -void -bcw_phy_lo_b_measure(struct bcw_softc *sc) -{ - uint16_t regstack[12] = { 0 }; - uint16_t mls; - uint16_t fval; - int i, j; - - regstack[0] = bcw_phy_read16(sc, 0x0015); - regstack[1] = bcw_radio_read16(sc, 0x0052) & 0xfff0; - - if (sc->sc_radio_ver == 0x2053) { - regstack[2] = bcw_phy_read16(sc, 0x000a); - regstack[3] = bcw_phy_read16(sc, 0x002a); - regstack[4] = bcw_phy_read16(sc, 0x0035); - regstack[5] = bcw_phy_read16(sc, 0x0003); - regstack[6] = bcw_phy_read16(sc, 0x0001); - regstack[7] = bcw_phy_read16(sc, 0x0030); - - regstack[8] = bcw_radio_read16(sc, 0x0043); - regstack[9] = bcw_radio_read16(sc, 0x007a); - regstack[10] = BCW_READ16(sc, 0x03ec); - regstack[11] = bcw_radio_read16(sc, 0x0052) & 0x00f0; - - bcw_phy_write16(sc, 0x0030, 0x00ff); - BCW_WRITE16(sc, 0x3ec, 0x3f3f); - bcw_phy_write16(sc, 0x0035, regstack[4] & 0xff7f); - bcw_radio_write16(sc, 0x007a, regstack[9] & 0xfff0); - } - bcw_phy_write16(sc, 0x0015, 0xb000); - bcw_phy_write16(sc, 0x002b, 0x0004); - - if (sc->sc_radio_ver == 0x2053) { - bcw_phy_write16(sc, 0x002b, 0x0203); - bcw_phy_write16(sc, 0x002a, 0x08a3); - } - - sc->sc_phy_minlowsig[0] = 0xffff; - - for (i = 0; i < 4; i++) { - bcw_radio_write16(sc, 0x0052, regstack[1] | i); - bcw_phy_lo_b_r15_loop(sc); - } - for (i = 0; i < 10; i++) { - bcw_radio_write16(sc, 0x0052, regstack[1] | i); - mls = bcw_phy_lo_b_r15_loop(sc) / 10; - if (mls < sc->sc_phy_minlowsig[0]) { - sc->sc_phy_minlowsig[0] = mls; - sc->sc_phy_minlowsigpos[0] = i; - } - } - bcw_radio_write16(sc, 0x0052, regstack[1] | sc->sc_phy_minlowsigpos[0]); - - sc->sc_phy_minlowsig[1] = 0xffff; - - for (i = -4; i < 5; i += 2) { - for (j = -4; j < 5; j += 2) { - if (j < 0) - fval = (0x0100 * i) + j + 0x0100; - else - fval = (0x0100 * i) + j; - bcw_phy_write16(sc, 0x002f, fval); - mls = bcw_phy_lo_b_r15_loop(sc) / 10; - if (mls < sc->sc_phy_minlowsig[1]) { - sc->sc_phy_minlowsig[1] = mls; - sc->sc_phy_minlowsigpos[1] = fval; - } - } - } - sc->sc_phy_minlowsigpos[1] += 0x0101; - - bcw_phy_write16(sc, 0x002f, sc->sc_phy_minlowsigpos[1]); - if (sc->sc_radio_ver == 0x2053) { - bcw_phy_write16(sc, 0x000a, regstack[2]); - bcw_phy_write16(sc, 0x002a, regstack[3]); - bcw_phy_write16(sc, 0x0035, regstack[4]); - bcw_phy_write16(sc, 0x0003, regstack[5]); - bcw_phy_write16(sc, 0x0001, regstack[6]); - bcw_phy_write16(sc, 0x0030, regstack[7]); - - bcw_radio_write16(sc, 0x0043, regstack[8]); - bcw_radio_write16(sc, 0x007a, regstack[9]); - - bcw_radio_write16(sc, 0x0052, (bcw_radio_read16(sc, 0x0052) & - 0x000f) | regstack[11]); - BCW_WRITE16(sc, 0x03ec, regstack[10]); - } - bcw_phy_write16(sc, 0x0015, regstack[0]); -} - -void -bcw_phy_lo_g_state(struct bcw_softc *sc, struct bcw_lopair *in_pair, - struct bcw_lopair *out_pair, uint16_t r27) -{ - const struct bcw_lopair transitions[8] = { - { .high = 1, .low = 1, }, - { .high = 1, .low = 0, }, - { .high = 1, .low = -1, }, - { .high = 0, .low = -1, }, - { .high = -1, .low = -1, }, - { .high = -1, .low = 0, }, - { .high = -1, .low = 1, }, - { .high = 0, .low = 1, } }; - struct bcw_lopair lowest_transition = { - .high = in_pair->high, - .low = in_pair->low }; - struct bcw_lopair tmp_pair; - struct bcw_lopair transition; - int i = 12; - int state = 0; - int found_lower; - int j, begin, end; - uint32_t lowest_deviation; - uint32_t tmp; - - bcw_phy_lo_write(sc, &lowest_transition); - lowest_deviation = bcw_phy_lo_g_singledeviation(sc, r27); - do { - found_lower = 0; - if (state == 0) { - begin = 1; - end = 8; - } else if (state % 2 == 0) { - begin = state - 1; - end = state + 1; - } else { - begin = state - 2; - end = state + 2; - } - if (begin < 1) - begin += 8; - if (end > 8) - end -= 8; - - j = begin; - tmp_pair.high = lowest_transition.high; - tmp_pair.low = lowest_transition.low; - while (1) { - transition.high = tmp_pair.high + - transitions[j - 1].high; - transition.low = tmp_pair.low + - transitions[j - 1].low; - if ((abs(transition.low) < 9) && (abs(transition.high) - < 9)) { - bcw_phy_lo_write(sc, &transition); - tmp = bcw_phy_lo_g_singledeviation(sc, r27); - if (tmp < lowest_deviation) { - lowest_deviation = tmp; - state = j; - found_lower = 1; - lowest_transition.high = - transition.high; - lowest_transition.low = - transition.low; - } - } - if (j == end) - break; - if (j == 8) - j = 1; - else - j++; - } - } while (i-- && found_lower); - - out_pair->high = lowest_transition.high; - out_pair->low = lowest_transition.low; -} - -void -bcw_phy_lo_g_measure(struct bcw_softc *sc) -{ - const uint8_t pairorder[10] = { 3, 1, 5, 7, 9, 2, 0, 4, 6, 8 }; - const int is_initializing = 0; /* XXX */ - uint16_t h, i, oldi = 0, j; - struct bcw_lopair control; - struct bcw_lopair *tmp_control; - uint16_t tmp; - uint16_t regstack[16] = { 0 }; - uint16_t oldchannel; - uint8_t r27 = 0, r31; - - oldchannel = sc->sc_radio_channel; - /* setup */ - if (sc->sc_phy_connected) { - regstack[0] = bcw_phy_read16(sc, BCW_PHY_G_CRS); - regstack[1] = bcw_phy_read16(sc, 0x0802); - bcw_phy_write16(sc, BCW_PHY_G_CRS, regstack[0] & 0x7fff); - bcw_phy_write16(sc, 0x0802, regstack[1] & 0xfffc); - } - regstack[3] = BCW_READ16(sc, 0x03e2); - BCW_WRITE16(sc, 0x03e2, regstack[3] | 0x8000); - regstack[4] = BCW_READ16(sc, BCW_MMIO_CHANNEL_EXT); - regstack[5] = bcw_phy_read16(sc, 0x15); - regstack[6] = bcw_phy_read16(sc, 0x2a); - regstack[7] = bcw_phy_read16(sc, 0x35); - regstack[8] = bcw_phy_read16(sc, 0x60); - regstack[9] = bcw_radio_read16(sc, 0x43); - regstack[10] = bcw_radio_read16(sc, 0x7a); - regstack[11] = bcw_radio_read16(sc, 0x52); - if (sc->sc_phy_connected) { - regstack[12] = bcw_phy_read16(sc, 0x0811); - regstack[13] = bcw_phy_read16(sc, 0x0812); - regstack[14] = bcw_phy_read16(sc, 0x0814); - regstack[15] = bcw_phy_read16(sc, 0x0815); - } - bcw_radio_select_channel(sc, 6, 0); - if (sc->sc_phy_connected) { - bcw_phy_write16(sc, BCW_PHY_G_CRS, regstack[0] & 0x7fff); - bcw_phy_write16(sc, 0x0802, regstack[1] & 0xfffc); - bcw_dummy_transmission(sc); - } - bcw_radio_write16(sc, 0x0043, 0x0006); - - bcw_phy_set_baseband_atten(sc, 2); - - BCW_WRITE16(sc, BCW_MMIO_CHANNEL_EXT, 0); - bcw_phy_write16(sc, 0x002e, 0x007f); - bcw_phy_write16(sc, 0x080f, 0x0078); - bcw_phy_write16(sc, 0x0035, regstack[7] & ~(1 << 7)); - bcw_radio_write16(sc, 0x007a, regstack[10] & 0xfff0); - bcw_phy_write16(sc, 0x002b, 0x0203); - bcw_phy_write16(sc, 0x002a, 0x08a3); - if (sc->sc_phy_type) { - bcw_phy_write16(sc, 0x0814, regstack[14] | 0x0003); - bcw_phy_write16(sc, 0x0815, regstack[15] & 0xfffc); - bcw_phy_write16(sc, 0x0811, 0x01b3); - bcw_phy_write16(sc, 0x0812, 0x00b2); - } - if (is_initializing) - bcw_phy_lo_g_measure_txctl2(sc); - - bcw_phy_write16(sc, 0x080f, 0x8078); - - /* measure */ - control.low = 0; - control.high = 0; - for (h = 0; h < 10; h++) { - /* loop over each possible RadioAttenuation (0 - 9) */ - i = pairorder[h]; - if (is_initializing) { - if (i == 3) { - control.low = 0; - control.high = 0; - } else if (((i % 2 == 1) && (oldi % 2 == 1)) || - ((i % 2 == 0) && (oldi % 2 == 0))) { - tmp_control = bcw_get_lopair(sc, oldi, 0); - memcpy(&control, tmp_control, sizeof(control)); - } else { - tmp_control = bcw_get_lopair(sc, 3, 0); - memcpy(&control, tmp_control, sizeof(control)); - } - } - /* loop over each possible BasebandAttenuation / 2 */ - for (j = 0; j < 4; j++) { - if (is_initializing) { - tmp = i * 2 + j; - r27 = 0; - r31 = 0; - if (tmp > 14) { - r31 = 1; - if (tmp > 17) - r27 = 1; - if (tmp > 19) - r27 = 2; - } - } else { - tmp_control = bcw_get_lopair(sc, i, j * 2); - if (!tmp_control->used) - continue; - memcpy(&control, tmp_control, sizeof(control)); - r27 = 3; - r31 = 0; - } - bcw_radio_write16(sc, 0x43, i); - bcw_radio_write16(sc, 0x52, sc->sc_radio_txctl2); - delay(10); - - bcw_phy_set_baseband_atten(sc, j * 2); - - tmp = (regstack[10] & 0xfff0); - if (r31) - tmp |= 0x0008; - bcw_radio_write16(sc, 0x007a, tmp); - - tmp_control = bcw_get_lopair(sc, i, j * 2); - bcw_phy_lo_g_state(sc, &control, tmp_control, r27); - } - oldi = i; - } - /* loop over each possible RadioAttenuation (10 - 13) */ - for (i = 10; i < 14; i++) { - /* loop over each possible BasebandAttenuation / 2 */ - for (j = 0; j < 4; j++) { - if (is_initializing) { - tmp_control = bcw_get_lopair(sc, i - 9, j * 2); - memcpy(&control, tmp_control, sizeof(control)); - tmp = (i - 9) * 2 + j - 5; /* XXX */ - r27 = 0; - r31 = 0; - if (tmp > 14) { - r31 = 1; - if (tmp > 17) - r27 = 1; - if (tmp > 19) - r27 = 2; - } - } else { - tmp_control = bcw_get_lopair(sc, i - 9, j * 2); - if (!tmp_control->used) - continue; - memcpy(&control, tmp_control, sizeof(control)); - r27 = 3; - r31 = 0; - } - bcw_radio_write16(sc, 0x043, i - 9); - bcw_radio_write16(sc, 0x052, sc->sc_radio_txctl2 | - (3 << 4)); /* XXX */ - delay(10); - - bcw_phy_set_baseband_atten(sc, j * 2); - - tmp = (regstack[10] & 0xfff0); - if (r31) - tmp |= 0x0008; - bcw_radio_write16(sc, 0x7a, tmp); - - tmp_control = bcw_get_lopair(sc, i, j * 2); - bcw_phy_lo_g_state(sc, &control, tmp_control, r27); - } - - } - - /* restoration */ - if (sc->sc_phy_connected) { - bcw_phy_write16(sc, 0x0015, 0xe300); - bcw_phy_write16(sc, 0x0812, (r27 << 8) | 0xa0); - delay(5); - bcw_phy_write16(sc, 0x0812, (r27 << 8) | 0xa2); - delay(2); - bcw_phy_write16(sc, 0x0812, (r27 << 8) | 0xa3); - } else - bcw_phy_write16(sc, 0x0015, r27 | 0xefa0); - bcw_phy_lo_adjust(sc, is_initializing); - bcw_phy_write16(sc, 0x002e, 0x807f); - if (sc->sc_phy_connected) - bcw_phy_write16(sc, 0x002f, 0x0202); - else - bcw_phy_write16(sc, 0x002f, 0x0101); - bcw_phy_write16(sc, BCW_MMIO_CHANNEL_EXT, regstack[4]); - bcw_phy_write16(sc, 0x0015, regstack[5]); - bcw_phy_write16(sc, 0x0015, regstack[6]); - bcw_phy_write16(sc, 0x0015, regstack[7]); - bcw_phy_write16(sc, 0x0015, regstack[8]); - bcw_radio_write16(sc, 0x0043, regstack[9]); - bcw_radio_write16(sc, 0x007a, regstack[10]); - regstack[11] &= 0x00f0; - regstack[11] |= (bcw_radio_read16(sc, 0x52) & 0x000f); - bcw_radio_write16(sc, 0x52, regstack[11]); - BCW_WRITE16(sc, 0x03e2, regstack[3]); - if (sc->sc_phy_connected) { - bcw_phy_write16(sc, 0x0811, regstack[12]); - bcw_phy_write16(sc, 0x0812, regstack[13]); - bcw_phy_write16(sc, 0x0814, regstack[14]); - bcw_phy_write16(sc, 0x0815, regstack[15]); - bcw_phy_write16(sc, BCW_PHY_G_CRS, regstack[0]); - bcw_phy_write16(sc, 0x0802, regstack[1]); - } - bcw_radio_select_channel(sc, oldchannel, 1); -} - -void -bcw_phy_lo_g_measure_txctl2(struct bcw_softc *sc) -{ - uint16_t txctl2 = 0, i; - uint32_t smallest, tmp; - - bcw_radio_write16(sc, 0x0052, 0); - delay(10); - smallest = bcw_phy_lo_g_singledeviation(sc, 0); - for (i = 0; i < 16; i++) { - bcw_radio_write16(sc, 0x0052, i); - delay(10); - tmp = bcw_phy_lo_g_singledeviation(sc, 0); - if (tmp < smallest) { - smallest = tmp; - txctl2 = i; - } - } - sc->sc_radio_txctl2 = txctl2; -} - -uint32_t -bcw_phy_lo_g_singledeviation(struct bcw_softc *sc, uint16_t control) -{ - int i; - uint32_t r = 0; - - for (i = 0; i < 8; i++) - r += bcw_phy_lo_g_deviation_subval(sc, control); - - return (r); -} - -uint16_t -bcw_phy_lo_g_deviation_subval(struct bcw_softc *sc, uint16_t control) -{ - uint16_t r; - //unsigned long flags; - - /* XXX splnet()? */ - - if (sc->sc_phy_connected) { - bcw_phy_write16(sc, 0x15, 0xe300); - control <<= 8; - bcw_phy_write16(sc, 0x0812, control | 0x00b0); - delay(5); - bcw_phy_write16(sc, 0x0812, control | 0x00b2); - delay(2); - bcw_phy_write16(sc, 0x0812, control | 0x00b3); - delay(4); - bcw_phy_write16(sc, 0x0015, 0xf300); - delay(8); - } else { - bcw_phy_write16(sc, 0x0015, control | 0xefa0); - delay(2); - bcw_phy_write16(sc, 0x0015, control | 0xefe0); - delay(4); - bcw_phy_write16(sc, 0x0015, control | 0xffe0); - delay(8); - } - r = bcw_phy_read16(sc, 0x002d); - - /* XXX splnet()? */ - - return (r); -} - -void -bcw_phy_lo_adjust(struct bcw_softc *sc, int fixed) -{ - struct bcw_lopair *pair; - - if (fixed) - pair = bcw_phy_find_lopair(sc, 2, 3, 0); - else - pair = bcw_phy_current_lopair(sc); - - bcw_phy_lo_write(sc, pair); -} - -void -bcw_phy_lo_mark_current_used(struct bcw_softc *sc) -{ - struct bcw_lopair *pair; - - pair = bcw_phy_current_lopair(sc); - pair->used = 1; -} - -void -bcw_phy_lo_write(struct bcw_softc *sc, struct bcw_lopair *pair) -{ - uint16_t val; - - val = (uint8_t)(pair->low); - val |= ((uint8_t)(pair->high)) << 8; - -#ifdef BCW_DEBUG - if (pair->low < -8 || pair->low > 8 || - pair->high < -8 || pair->high > 8) - printf("%s: writing invalid LO pair " - "low: %d, high: %d, index: %lu)\n", - pair->low, pair->high, - (unsigned long)(pair - sc->sc_phy_lopairs)); -#endif - - bcw_phy_write16(sc, BCW_PHY_G_LO_CONTROL, val); -} - -struct bcw_lopair * -bcw_phy_find_lopair(struct bcw_softc *sc, uint16_t baseband_atten, - uint16_t radio_atten, uint16_t tx) -{ - static const uint8_t dict[10] = - { 11, 10, 11, 12, 13, 12, 13, 12, 13, 12 }; - - if (baseband_atten > 6) - baseband_atten = 6; - - if (tx == 3) - return (bcw_get_lopair(sc, radio_atten, baseband_atten)); - - return (bcw_get_lopair(sc, dict[radio_atten], baseband_atten)); -} - -struct bcw_lopair * -bcw_phy_current_lopair(struct bcw_softc *sc) -{ - return (bcw_phy_find_lopair(sc, sc->sc_radio_baseband_atten, - sc->sc_radio_radio_atten, sc->sc_radio_txctl1)); -} - -void -bcw_phy_set_antenna_diversity(struct bcw_softc *sc) -{ - uint16_t antennadiv; - uint16_t offset; - uint16_t val; - uint32_t ucodeflags; - - if (antennadiv == 0xffff) - antennadiv = 3; - - ucodeflags = bcw_shm_read32(sc, BCW_SHM_SHARED, BCW_UCODEFLAGS_OFFSET); - bcw_shm_write32(sc, BCW_SHM_SHARED, BCW_UCODEFLAGS_OFFSET, - ucodeflags & ~BCW_UCODEFLAG_AUTODIV); - - switch (sc->sc_phy_type) { - case BCW_PHY_TYPEA: - case BCW_PHY_TYPEG: - if (sc->sc_phy_type == BCW_PHY_TYPEA) - offset = 0; - else - offset = 0x0400; - - if (antennadiv == 2) - val = (3 << 7); - else - val = (antennadiv << 7); - bcw_phy_write16(sc, offset + 1, - (bcw_phy_read16(sc, offset + 1) & 0x7e7f) | val); - - if (antennadiv >= 2) { - if (antennadiv == 2) - val = (antennadiv << 7); - else - val = (0 << 7); - bcw_phy_write16(sc, offset + 0x2b, - (bcw_phy_read16(sc, offset + 0x2b) & 0xfeff) | val); - } - - if (sc->sc_phy_type == BCW_PHY_TYPEG) { - if (antennadiv >= 2) - bcw_phy_write16(sc, 0x048c, - bcw_phy_read16(sc, 0x048c) | 0x2000); - else - bcw_phy_write16(sc, 0x048c, - bcw_phy_read16(sc, 0x048c) | ~0x2000); - if (sc->sc_phy_rev >= 2) { - bcw_phy_write16(sc, 0x0461, - bcw_phy_read16(sc, 0x0461) | 0x0010); - bcw_phy_write16(sc, 0x04ad, - (bcw_phy_read16(sc, 0x04ad) & 0x00ff) | - 0x0015); - if (sc->sc_phy_rev == 2) - bcw_phy_write16(sc, 0x0427, 0x0008); - else - bcw_phy_write16(sc, 0x0427, - (bcw_phy_read16(sc, 0x0427) & - 0x00ff) | 0x0008); - } else if (sc->sc_phy_rev >= 6) - bcw_phy_write16(sc, 0x049b, 0x00dc); - } else { - if (sc->sc_phy_rev < 3) - bcw_phy_write16(sc, 0x002b, - (bcw_phy_read16(sc, 0x002b) & 0x00ff) | - 0x0024); - else { - bcw_phy_write16(sc, 0x0061, - bcw_phy_read16(sc, 0x0061) | 0x0010); - if (sc->sc_phy_rev == 3) { - bcw_phy_write16(sc, 0x0093, 0x001d); - bcw_phy_write16(sc, 0x0027, 0x0008); - } else { - bcw_phy_write16(sc, 0x0093, 0x003a); - bcw_phy_write16(sc, 0x0027, - (bcw_phy_read16(sc, 0x0027) & - 0x00ff) | 0x0008); - } - } - } - break; - case BCW_PHY_TYPEB: - if (sc->sc_core[sc->sc_currentcore].rev == 2) - val = (3 << 7); - else - val = (antennadiv << 7); - bcw_phy_write16(sc, 0x03e2, (bcw_phy_read16(sc, 0x03e2) & - 0xfe7f) | val); - break; - default: - /* XXX panic()? */ - break; - } - - if (antennadiv >= 2) { - ucodeflags = bcw_shm_read32(sc, BCW_SHM_SHARED, - BCW_UCODEFLAGS_OFFSET); - bcw_shm_write32(sc, BCW_SHM_SHARED, - BCW_UCODEFLAGS_OFFSET, ucodeflags | - BCW_UCODEFLAG_AUTODIV); - } - - sc->sc_phy_antenna_diversity = antennadiv; -} - -void -bcw_phy_calibrate(struct bcw_softc *sc) -{ - BCW_READ(sc, BCW_MMIO_SBF); - - if (sc->sc_phy_calibrated) - return; - if (sc->sc_phy_type == BCW_PHY_TYPEG && sc->sc_phy_rev == 1) { - bcw_80211_core_reset(sc, 0); - bcw_phy_initg(sc); - bcw_80211_core_reset(sc, 1); - } - sc->sc_phy_calibrated = 1; -} - -/* - * Connect the PHY - * - * http://bcm-specs.sipsolutions.net/SetPHY - */ -int -bcw_phy_connect(struct bcw_softc *sc, int connect) -{ - uint32_t flags; - - if (sc->sc_core[sc->sc_currentcore].rev < 5) - goto out; - - flags = BCW_READ(sc, BCW_CIR_SBTMSTATEHIGH); - if (connect) { - if (!(flags & 0x00010000)) - return (ENODEV); - flags = BCW_READ(sc, BCW_CIR_SBTMSTATELOW); - flags |= (0x800 << 18); - BCW_WRITE(sc, BCW_CIR_SBTMSTATELOW, flags); - } else { - if (!(flags & 0x00020000)) - return (ENODEV); - flags = BCW_READ(sc, BCW_CIR_SBTMSTATELOW); - flags &= ~(0x800 << 18); - BCW_WRITE(sc, BCW_CIR_SBTMSTATELOW, flags); - } - -out: - sc->sc_phy_connected = connect; - if (connect) - DPRINTF(("%s: PHY connected\n", sc->sc_dev.dv_xname)); - else - DPRINTF(("%s: PHY disconnected\n", sc->sc_dev.dv_xname)); - - return (0); -} - -void -bcw_phy_lock(struct bcw_softc *sc) -{ - struct ieee80211com *ic = &sc->sc_ic; - - if (BCW_READ(sc, BCW_MMIO_SBF) == 0) { - sc->sc_phy_is_locked = 0; - return; - } - - if (sc->sc_core[sc->sc_currentcore].rev < 3) - bcw_mac_disable(sc); - else { - if (ic->ic_opmode != IEEE80211_M_HOSTAP) - bcw_pc_saving_ctl_bits(sc, -1, -1); - } - - sc->sc_phy_is_locked = 1; -} - -void -bcw_phy_unlock(struct bcw_softc *sc) -{ - struct ieee80211com *ic = &sc->sc_ic; - - if (sc->sc_core[sc->sc_currentcore].rev < 3) { - if (sc->sc_phy_is_locked) - bcw_mac_enable(sc); - } else { - if (ic->ic_opmode != IEEE80211_M_HOSTAP) - bcw_pc_saving_ctl_bits(sc, -1, -1); - } - - sc->sc_phy_is_locked = 0; -} - -/* - * Radio - */ -int -bcw_radio_get(struct bcw_softc *sc) -{ - uint32_t val; - - /* - * XXX - * Query the RadioID register, on a 4317 use a lookup instead - * Different PHYs have different radio register layouts, so - * a wrapper func should be written. - * Getting the RadioID is the only 32bit operation done with the - * Radio registers, and requires seperate 16bit reads from the low - * and the high data addresses. - */ - if (sc->sc_chip_id != 0x4317) { - BCW_WRITE16(sc, BCW_MMIO_RADIO_CONTROL, BCW_RADIO_ID); - val = BCW_READ16(sc, BCW_MMIO_RADIO_DATA_HIGH); - val <<= 16; - BCW_WRITE16(sc, BCW_MMIO_RADIO_CONTROL, BCW_RADIO_ID); - sc->sc_radio_mnf = - val | BCW_READ16(sc, BCW_MMIO_RADIO_DATA_LOW); - } else { - switch (sc->sc_chip_rev) { - case 0: - sc->sc_radio_mnf = 0x3205017F; - break; - case 1: - sc->sc_radio_mnf = 0x4205017f; - break; - default: - sc->sc_radio_mnf = 0x5205017f; - } - } - - sc->sc_radio_rev = (sc->sc_radio_mnf & 0xf0000000) >> 28; - sc->sc_radio_ver = (sc->sc_radio_mnf & 0x0ffff000) >> 12; - - switch (sc->sc_phy_type) { - case BCW_PHY_TYPEA: - if (sc->sc_radio_ver != 0x2060) { - printf("%s: invalid PHY A radio 0x%x!\n", - sc->sc_dev.dv_xname, sc->sc_radio_ver); - return (1); - } - break; - case BCW_PHY_TYPEB: - if ((sc->sc_radio_ver & 0xfff0) != 0x2050) { - printf("%s: invalid PHY B radio 0x%x!\n", - sc->sc_dev.dv_xname, sc->sc_radio_ver); - return (1); - } - break; - case BCW_PHY_TYPEG: - if (sc->sc_radio_ver != 0x2050) { - printf("%s: invalid PHY G radio 0x%x!\n", - sc->sc_dev.dv_xname, sc->sc_radio_ver); - return (1); - } - break; - } - - DPRINTF(("%s: Radio rev %d ver 0x%x mnf 0x%x\n", - sc->sc_dev.dv_xname, sc->sc_radio_rev, sc->sc_radio_ver, - sc->sc_radio_mnf & 0xfff)); - - return (0); -} - -void -bcw_radio_off(struct bcw_softc *sc) -{ - /* Magic unexplained values */ - if (sc->sc_phy_type == BCW_PHY_TYPEA) { - bcw_radio_write16(sc, 0x0004, 0x00ff); - bcw_radio_write16(sc, 0x0005, 0x00fb); - bcw_phy_write16(sc, 0x0010, bcw_phy_read16(sc, 0x0010) | - 0x0008); - bcw_phy_write16(sc, 0x0011, bcw_phy_read16(sc, 0x0011) | - 0x0008); - } - if (sc->sc_phy_type == BCW_PHY_TYPEB && sc->sc_core_80211->rev >= 5) { - bcw_phy_write16(sc, 0x0811, bcw_phy_read16(sc, 0x0811) | - 0x008c); - bcw_phy_write16(sc, 0x0812, bcw_phy_read16(sc, 0x0812) & - 0xff73); - } else - bcw_phy_write16(sc, 0x0015, 0xaa00); - - DPRINTF(("%s: Radio turned off\n", sc->sc_dev.dv_xname)); -} - -void -bcw_radio_on(struct bcw_softc *sc) -{ - switch (sc->sc_phy_type) { - case BCW_PHY_TYPEA: - bcw_radio_write16(sc, 0x0004, 0x00c0); - bcw_radio_write16(sc, 0x0005, 0x0008); - bcw_phy_write16(sc, 0x0010, bcw_phy_read16(sc, 0x0010) & - 0xfff7); - bcw_phy_write16(sc, 0x0011, bcw_phy_read16(sc, 0x0011) & - 0xfff7); - bcw_radio_init2060(sc); - break; - case BCW_PHY_TYPEB: - case BCW_PHY_TYPEG: - bcw_phy_write16(sc, 0x0015, 0x8000); - bcw_phy_write16(sc, 0x0015, 0xcc00); - bcw_phy_write16(sc, 0x0015, sc->sc_phy_connected ? 0x00c0 : 0); - if (bcw_radio_select_channel(sc, BCW_RADIO_DEFAULT_CHANNEL_BG, - 1)) - return; - break; - default: - return; - } - - DPRINTF(("%s: Radio turned on\n", sc->sc_dev.dv_xname)); -} - -void -bcw_radio_nrssi_hw_write(struct bcw_softc *sc, uint16_t offset, int16_t val) -{ - bcw_phy_write16(sc, BCW_PHY_NRSSILT_CTRL, offset); - bcw_phy_write16(sc, BCW_PHY_NRSSILT_DATA, (uint16_t)val); -} - -int16_t -bcw_radio_nrssi_hw_read(struct bcw_softc *sc, uint16_t offset) -{ - uint16_t val; - - bcw_phy_write16(sc, BCW_PHY_NRSSILT_CTRL, offset); - val = bcw_phy_read16(sc, BCW_PHY_NRSSILT_DATA); - - return ((int16_t)val); -} - -void -bcw_radio_nrssi_hw_update(struct bcw_softc *sc, uint16_t val) -{ - uint16_t i; - int16_t tmp; - - for (i = 0; i < 64; i++) { - tmp = bcw_radio_nrssi_hw_read(sc, i); - tmp -= val; - tmp = bcw_lv(tmp, 32, 31); - bcw_radio_nrssi_hw_write(sc, i, tmp); - } -} - -void -bcw_radio_calc_nrssi_threshold(struct bcw_softc *sc) -{ - int32_t threshold; - int32_t a, b; - int16_t tmp16; - uint16_t tmp_u16; - - switch (sc->sc_phy_type) { - case BCW_PHY_TYPEB: - if (sc->sc_radio_ver != 0x2050) - return; - if (!(sc->sc_sprom.boardflags & BCW_BF_RSSI)) - return; - - if (sc->sc_radio_rev >= 6) { - threshold = (sc->sc_radio_nrssi[1] - - sc->sc_radio_nrssi[0]) * 32; - threshold += 20 * (sc->sc_radio_nrssi[0] + 1); - threshold /= 40; - } else - threshold = sc->sc_radio_nrssi[1] - 5; - - threshold = bcw_lv(threshold, 0, 0x3e); - bcw_phy_read16(sc, 0x0020); - bcw_phy_write16(sc, 0x0020, (((uint16_t)threshold) << 8) | - 0x001c); - - if (sc->sc_radio_rev >= 6) { - bcw_phy_write16(sc, 0x0087, 0x0e0d); - bcw_phy_write16(sc, 0x0086, 0x0c0d); - bcw_phy_write16(sc, 0x0085, 0x0a09); - bcw_phy_write16(sc, 0x0084, 0x0808); - bcw_phy_write16(sc, 0x0083, 0x0808); - bcw_phy_write16(sc, 0x0082, 0x0604); - bcw_phy_write16(sc, 0x0081, 0x0302); - bcw_phy_write16(sc, 0x0080, 0x0100); - } - break; - case BCW_PHY_TYPEG: - if (!sc->sc_phy_connected || - !(sc->sc_sprom.boardflags & BCW_BF_RSSI)) { - tmp16 = bcw_radio_nrssi_hw_read(sc, 0x20); - if (tmp16 >= 0x20) - tmp16 -= 0x40; - if (tmp16 < 3) - bcw_phy_write16(sc, 0x048a, - (bcw_phy_read16(sc, 0x048a) & 0xf000) | - 0x09eb); - else - bcw_phy_write16(sc, 0x048a, - (bcw_phy_read16(sc, 0x048a) & 0xf000) | - 0x0aed); - } else { - if (sc->sc_radio_interfmode == - BCW_RADIO_INTERFMODE_NONWLAN) { - a = 0xe; - b = 0xa; - } else if (sc->sc_radio_aci_wlan_automatic && - sc->sc_radio_aci_enable) { - a = 0x13; - b = 0x12; - } else { - a = 0xe; - b = 0x11; - } - - a = a * (sc->sc_radio_nrssi[1] - sc->sc_radio_nrssi[0]); - a += (sc->sc_radio_nrssi[0] << 6); - if (a < 32) - a += 31; - else - a += 32; - a = a >> 6; - a = bcw_lv(a, -31, 31); - - b = b * (sc->sc_radio_nrssi[1] - sc->sc_radio_nrssi[0]); - b += (sc->sc_radio_nrssi[0] << 6); - if (b < 32) - b += 31; - else - b += 32; - b = b >> 6; - b = bcw_lv(b, -31, 31); - - tmp_u16 = bcw_phy_read16(sc, 0x048a) & 0xf000; - tmp_u16 |= ((uint32_t)b & 0x0000003f); - tmp_u16 |= (((uint32_t)a & 0x0000003f) << 6); - bcw_phy_write16(sc, 0x048a, tmp_u16); - } - break; - default: - /* XXX panic()? */ - return; - } -} - -void -bcw_radio_calc_nrssi_slope(struct bcw_softc *sc) -{ - uint16_t backup[18] = { 0 }; - uint16_t tmp; - int16_t nrssi0, nrssi1; - - switch (sc->sc_phy_type) { - case BCW_PHY_TYPEB: - backup[0] = bcw_radio_read16(sc, 0x007a); - backup[1] = bcw_radio_read16(sc, 0x0052); - backup[2] = bcw_radio_read16(sc, 0x0043); - backup[3] = bcw_phy_read16(sc, 0x0030); - backup[4] = bcw_phy_read16(sc, 0x0026); - backup[5] = bcw_phy_read16(sc, 0x0015); - backup[6] = bcw_phy_read16(sc, 0x0020); - backup[7] = bcw_phy_read16(sc, 0x005a); - backup[8] = bcw_phy_read16(sc, 0x0059); - backup[9] = bcw_phy_read16(sc, 0x0058); - backup[10] = bcw_phy_read16(sc, 0x03e2); - backup[11] = BCW_READ16(sc, 0x03e6); - backup[12] = BCW_READ16(sc, 0x007a); - backup[13] = BCW_READ16(sc, BCW_MMIO_CHANNEL_EXT); - - tmp = bcw_radio_read16(sc, 0x007a); - tmp &= ((sc->sc_phy_rev >= 5) ? 0x007f : 0x000f); - bcw_radio_write16(sc, 0x007a, tmp); - bcw_phy_write16(sc, 0x0030, 0x00f); - BCW_WRITE16(sc, 0x03ec, 0x7f7f); - bcw_phy_write16(sc, 0x0026, 0); - bcw_phy_write16(sc, 0x0015, bcw_phy_read16(sc, 0x0015) | - 0x0020); - bcw_phy_write16(sc, 0x002a, 0x08a3); - bcw_radio_write16(sc, 0x007a, bcw_radio_read16(sc, 0x007a) | - 0x0080); - - nrssi0 = (uint16_t)bcw_phy_read16(sc, 0x0027); - bcw_radio_write16(sc, 0x007a, bcw_radio_read16(sc, 0x007a) & - 0x007f); - if (sc->sc_phy_rev >= 2) - BCW_WRITE16(sc, 0x03e6, 0x0040); - else if (sc->sc_phy_rev == 0) - BCW_WRITE16(sc, 0x03e6, 0x0122); - else - BCW_WRITE16(sc, BCW_MMIO_CHANNEL_EXT, - BCW_READ16(sc, BCW_MMIO_CHANNEL_EXT) & 0x2000); - bcw_phy_write16(sc, 0x0020, 0x3f3f); - bcw_phy_write16(sc, 0x0014, 0xf330); - bcw_radio_write16(sc, 0x005a, 0xf330); - bcw_radio_write16(sc, 0x0043, - bcw_radio_read16(sc, 0x0043) & 0x0f0); - bcw_phy_write16(sc, 0x005a, 0x0480); - bcw_phy_write16(sc, 0x0059, 0x0810); - bcw_phy_write16(sc, 0x0058, 0x000d); - delay(20); - - nrssi1 = (int16_t)bcw_phy_read16(sc, 0x0027); - bcw_phy_write16(sc, 0x0030, backup[3]); - bcw_radio_write16(sc, 0x007a, backup[0]); - BCW_WRITE16(sc, 0x03e2, backup[11]); - bcw_phy_write16(sc, 0x0026, backup[4]); - bcw_phy_write16(sc, 0x0015, backup[5]); - bcw_phy_write16(sc, 0x002a, backup[6]); - bcw_radio_spw(sc, sc->sc_radio_channel); - if (sc->sc_phy_rev != 0) - BCW_WRITE16(sc, 0x3f4, backup[13]); - - bcw_phy_write16(sc, 0x0020, backup[7]); - bcw_phy_write16(sc, 0x0020, backup[8]); - bcw_phy_write16(sc, 0x0020, backup[9]); - bcw_phy_write16(sc, 0x0020, backup[10]); - bcw_radio_write16(sc, 0x0052, backup[1]); - bcw_radio_write16(sc, 0x0043, backup[2]); - - if (nrssi0 == nrssi1) - sc->sc_radio_nrssislope = 0x00010000; - else - sc->sc_radio_nrssislope = 0x00400000 / - (nrssi0 - nrssi1); - - if (nrssi0 <= -4) { - sc->sc_radio_nrssi[0] = nrssi0; - sc->sc_radio_nrssi[1] = nrssi1; - } - break; - case BCW_PHY_TYPEG: - if (sc->sc_radio_rev >= 9) - return; - if (sc->sc_radio_rev == 8) - bcw_radio_calc_nrssi_offset(sc); - break; - } -} - -void -bcw_radio_calc_nrssi_offset(struct bcw_softc *sc) -{ - uint16_t backup[20] = { 0 }; - int16_t v47f; - uint16_t i; - uint16_t saved = 0xffff; - - backup[0] = bcw_phy_read16(sc, 0x0001); - backup[1] = bcw_phy_read16(sc, 0x0811); - backup[2] = bcw_phy_read16(sc, 0x0812); - backup[3] = bcw_phy_read16(sc, 0x0814); - backup[4] = bcw_phy_read16(sc, 0x0815); - backup[5] = bcw_phy_read16(sc, 0x005a); - backup[6] = bcw_phy_read16(sc, 0x0059); - backup[7] = bcw_phy_read16(sc, 0x0058); - backup[8] = bcw_phy_read16(sc, 0x000a); - backup[9] = bcw_phy_read16(sc, 0x0003); - backup[10] = bcw_radio_read16(sc, 0x007a); - backup[11] = bcw_radio_read16(sc, 0x0043); - - bcw_phy_write16(sc, 0x0429, bcw_phy_read16(sc, 0x0429) & 0x7fff); - bcw_phy_write16(sc, 0x0001, (bcw_phy_read16(sc, 0x0001) & 0x3fff) | - 0x4000); - bcw_phy_write16(sc, 0x0811, bcw_phy_read16(sc, 0x0811) | 0x000c); - bcw_phy_write16(sc, 0x0812, (bcw_phy_read16(sc, 0x0812) & 0xfff3) | - 0x0004); - bcw_phy_write16(sc, 0x0802, bcw_phy_read16(sc, 0x0802) & ~(0x1 | 0x2)); - - if (sc->sc_phy_rev >= 6) { - backup[12] = bcw_phy_read16(sc, 0x002e); - backup[13] = bcw_phy_read16(sc, 0x002f); - backup[14] = bcw_phy_read16(sc, 0x080f); - backup[15] = bcw_phy_read16(sc, 0x0810); - backup[16] = bcw_phy_read16(sc, 0x0801); - backup[17] = bcw_phy_read16(sc, 0x0060); - backup[18] = bcw_phy_read16(sc, 0x0014); - backup[19] = bcw_phy_read16(sc, 0x0478); - - bcw_phy_write16(sc, 0x002e, 0); - bcw_phy_write16(sc, 0x002e, 0); - bcw_phy_write16(sc, 0x002e, 0); - bcw_phy_write16(sc, 0x002e, 0); - bcw_phy_write16(sc, 0x0478, bcw_phy_read16(sc, 0x0478) | - 0x0100); - bcw_phy_write16(sc, 0x0801, bcw_phy_read16(sc, 0x0801) | - 0x0040); - bcw_phy_write16(sc, 0x0060, bcw_phy_read16(sc, 0x0060) | - 0x0040); - bcw_phy_write16(sc, 0x0014, bcw_phy_read16(sc, 0x0014) | - 0x0200); - } - bcw_radio_write16(sc, 0x007a, bcw_radio_read16(sc, 0x007a) | 0x0070); - bcw_radio_write16(sc, 0x007a, bcw_radio_read16(sc, 0x007a) | 0x0080); - delay(30); - - v47f = (int16_t)((bcw_phy_read16(sc, 0x047f) >> 8) & 0x00ef); - if (v47f >= 0x20) - v47f -= 0x40; - if (v47f == 31) { - for (i = 7; i >= 4; i--) { - bcw_radio_write16(sc, 0x007b, i); - delay(20); - v47f = (int16_t)((bcw_phy_read16(sc, 0x047f) >> 8) & - 0x003f); - if (v47f >= 0x20) - v47f -= 0x40; - if (v47f < 31 && saved == 0xffff) - saved = i; - } - if (saved == 0xffff) - saved = 4; - } else { - bcw_radio_write16(sc, 0x007a, bcw_radio_read16(sc, 0x007a) & - 0x007f); - - bcw_phy_write16(sc, 0x0814, bcw_phy_read16(sc, 0x0814) | - 0x0001); - bcw_phy_write16(sc, 0x0815, bcw_phy_read16(sc, 0x0815) & - 0xfffe); - bcw_phy_write16(sc, 0x0811, bcw_phy_read16(sc, 0x0811) | - 0x000c); - bcw_phy_write16(sc, 0x0812, bcw_phy_read16(sc, 0x0812) | - 0x000c); - bcw_phy_write16(sc, 0x0811, bcw_phy_read16(sc, 0x0811) | - 0x0030); - bcw_phy_write16(sc, 0x0812, bcw_phy_read16(sc, 0x0812) | - 0x0030); - bcw_phy_write16(sc, 0x005a, 0x0480); - bcw_phy_write16(sc, 0x0059, 0x0810); - bcw_phy_write16(sc, 0x0058, 0x000d); - if (sc->sc_phy_rev == 0) - bcw_phy_write16(sc, 0x0003, 0x0122); - else - bcw_phy_write16(sc, 0x000a, bcw_phy_read16(sc, 0x000a) | - 0x2000); - bcw_phy_write16(sc, 0x0814, bcw_phy_read16(sc, 0x0814) | - 0x0004); - bcw_phy_write16(sc, 0x0815, bcw_phy_read16(sc, 0x0815) & - 0xfffb); - bcw_phy_write16(sc, 0x0003, (bcw_phy_read16(sc, 0x0003) & - 0xff9f) | 0x0040); - bcw_radio_write16(sc, 0x007a, bcw_radio_read16(sc, 0x007a) | - 0x000f); - bcw_radio_set_all_gains(sc, 3, 0, 1); - bcw_radio_write16(sc, 0x0043, (bcw_radio_read16(sc, 0x0043) & - 0x00f0) | 0x000f); - delay(30); - v47f = (int16_t)((bcw_phy_read16(sc, 0x047f) >> 8) & 0x003f); - if (v47f >= 0x20) - v47f -= 0x40; - if (v47f == -32) { - for (i = 0; i < 4; i++) { - bcw_radio_write16(sc, 0x007b, i); - delay(20); - v47f = (int16_t)((bcw_phy_read16(sc, 0x047f) >> - 8) & 0x003f); - if (v47f >= 0x20) - v47f -= 0x40; - if (v47f > -31 && saved == 0xffff) - saved = i; - } - if (saved == 0xfff) - saved = 3; - } else - saved = 0; - } - bcw_radio_write16(sc, 0x007b, saved); - - if (sc->sc_phy_rev >= 6) { - bcw_phy_write16(sc, 0x002e, backup[12]); - bcw_phy_write16(sc, 0x002f, backup[13]); - bcw_phy_write16(sc, 0x080f, backup[14]); - bcw_phy_write16(sc, 0x0810, backup[15]); - } - bcw_phy_write16(sc, 0x0814, backup[3]); - bcw_phy_write16(sc, 0x0815, backup[4]); - bcw_phy_write16(sc, 0x005a, backup[5]); - bcw_phy_write16(sc, 0x0059, backup[6]); - bcw_phy_write16(sc, 0x0058, backup[7]); - bcw_phy_write16(sc, 0x000a, backup[8]); - bcw_phy_write16(sc, 0x0003, backup[9]); - bcw_radio_write16(sc, 0x0043, backup[11]); - bcw_radio_write16(sc, 0x007a, backup[10]); - bcw_phy_write16(sc, 0x0802, bcw_phy_read16(sc, 0x0802) | 0x1 | 0x2); - bcw_phy_write16(sc, 0x0429, bcw_phy_read16(sc, 0x0429) | 0x8000); - bcw_radio_set_original_gains(sc); - if (sc->sc_phy_rev >= 6) { - bcw_phy_write16(sc, 0x0801, backup[16]); - bcw_phy_write16(sc, 0x0060, backup[17]); - bcw_phy_write16(sc, 0x0014, backup[18]); - bcw_phy_write16(sc, 0x0478, backup[19]); - } - bcw_phy_write16(sc, 0x0001, backup[0]); - bcw_phy_write16(sc, 0x0812, backup[2]); - bcw_phy_write16(sc, 0x0001, backup[1]); -} - -void -bcw_radio_set_all_gains(struct bcw_softc *sc, int16_t first, int16_t second, - int16_t third) -{ - uint16_t i; - uint16_t start = 0x08, end = 0x18; - uint16_t offset = 0x0400; - uint16_t tmp; - - if (sc->sc_phy_rev <= 1) { - offset = 0x5000; - start = 0x10; - end = 0x20; - } - - for (i = 0; i < 4; i++) - bcw_ilt_write(sc, offset + i, first); - - for (i = start; i < end; i++) - bcw_ilt_write(sc, offset + i, second); - - if (third != -1) { - bcw_phy_write16(sc, 0x04a0, (bcw_phy_read16(sc, 0x04a0) & - 0xbfbf) | tmp); - bcw_phy_write16(sc, 0x04a1, (bcw_phy_read16(sc, 0x04a1) & - 0xbfbf) | tmp); - bcw_phy_write16(sc, 0x04a2, (bcw_phy_read16(sc, 0x04a2) & - 0xbfbf) | tmp); - } - bcw_dummy_transmission(sc); -} - -void -bcw_radio_set_original_gains(struct bcw_softc *sc) -{ - uint16_t i, tmp; - uint16_t offset = 0x0400; - uint16_t start = 0x0008, end = 0x0018; - - if (sc->sc_phy_rev <= 1) { - offset = 0x5000; - start = 0x0010; - end = 0x0020; - } - - for (i = 0; i < 4; i++) { - tmp = (i & 0xfffc); - tmp |= (i & 0x0001) << 1; - tmp |= (i & 0x0002) >> 1; - - bcw_ilt_write(sc, offset + i, tmp); - } - - for (i = start; i < end; i++) - bcw_ilt_write(sc, offset + i, i - start); - - bcw_phy_write16(sc, 0x04a0, (bcw_phy_read16(sc, 0x04a0) & 0xbfbf) | - 0x4040); - bcw_phy_write16(sc, 0x04a1, (bcw_phy_read16(sc, 0x04a1) & 0xbfbf) | - 0x4040); - bcw_phy_write16(sc, 0x04a2, (bcw_phy_read16(sc, 0x04a2) & 0xbfbf) | - 0x4000); - bcw_dummy_transmission(sc); -} - -uint16_t -bcw_radio_calibrationvalue(struct bcw_softc *sc) -{ - uint16_t reg, index, r; - - reg = bcw_radio_read16(sc, 0x0060); - index = (reg & 0x001e) >> 1; - r = rcc_table[index] << 1; - r |= (reg & 0x0001); - r |= 0x0020; - - return (r); -} - -void -bcw_radio_set_txpower_a(struct bcw_softc *sc, uint16_t txpower) -{ - uint16_t pamp, base, dac, ilt; - - txpower = bcw_lv(txpower, 0, 63); - - pamp = bcw_radio_get_txgain_freq_power_amp(txpower); - pamp <<= 5; - pamp &= 0x00e0; - bcw_phy_write16(sc, 0x0019, pamp); - - base = bcw_radio_get_txgain_baseband(txpower); - base &= 0x000f; - bcw_phy_write16(sc, 0x0017, base | 0x0020); - - ilt = bcw_ilt_read(sc, 0x3001); - ilt &= 0x0007; - - dac = bcw_radio_get_txgain_dac(txpower); - dac <<= 3; - dac |= ilt; - - bcw_ilt_write(sc, 0x3001, dac); - - sc->sc_radio_txpwr_offset = txpower; -} - -void -bcw_radio_set_txpower_bg(struct bcw_softc *sc, uint16_t baseband_atten, - uint16_t radio_atten, uint16_t txpower) -{ - if (baseband_atten == 0xffff) - baseband_atten = sc->sc_radio_baseband_atten; - if (radio_atten == 0xffff) - radio_atten = sc->sc_radio_radio_atten; - if (txpower == 0xffff) - txpower = sc->sc_radio_txctl1; - - sc->sc_radio_baseband_atten = baseband_atten; - sc->sc_radio_radio_atten = radio_atten; - sc->sc_radio_txctl1 = txpower; - - bcw_phy_set_baseband_atten(sc, baseband_atten); - bcw_radio_write16(sc, 0x0043, radio_atten); - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x0064, radio_atten); - if (sc->sc_radio_ver == 0x2050) - bcw_radio_write16(sc, 0x0052, (bcw_radio_read16(sc, 0x0052) & - ~0x0070) | ((txpower << 4) & 0x0070)); - - /* XXX unclear specs */ - if (sc->sc_phy_type == BCW_PHY_TYPEG) - bcw_phy_lo_adjust(sc, 0); -} - -uint16_t -bcw_radio_init2050(struct bcw_softc *sc) -{ - uint16_t backup[19] = { 0 }; - uint16_t r; - uint16_t i, j; - uint32_t tmp1 = 0, tmp2 = 0; - - backup[0] = bcw_radio_read16(sc, 0x0043); - backup[14] = bcw_radio_read16(sc, 0x0051); - backup[15] = bcw_radio_read16(sc, 0x0052); - backup[1] = bcw_phy_read16(sc, 0x0015); - backup[16] = bcw_phy_read16(sc, 0x005a); - backup[17] = bcw_phy_read16(sc, 0x0059); - backup[18] = bcw_phy_read16(sc, 0x0058); - if (sc->sc_phy_type == BCW_PHY_TYPEB) { - backup[2] = bcw_phy_read16(sc, 0x0030); - backup[3] = BCW_READ16(sc, 0x03ec); - bcw_phy_write16(sc, 0x0030, 0x00ff); - BCW_WRITE16(sc, 0x03ec, 0x3f3f); - } else { - if (sc->sc_phy_connected) { - backup[4] = bcw_phy_read16(sc, 0x0811); - backup[5] = bcw_phy_read16(sc, 0x0812); - backup[6] = bcw_phy_read16(sc, 0x0814); - backup[7] = bcw_phy_read16(sc, 0x0815); - backup[8] = bcw_phy_read16(sc, BCW_PHY_G_CRS); - backup[9] = bcw_phy_read16(sc, 0x0802); - bcw_phy_write16(sc, 0x0814, - (bcw_phy_read16(sc, 0x0814) | 0x0003)); - bcw_phy_write16(sc, 0x0815, - (bcw_phy_read16(sc, 0x0815) & 0xfffc)); - bcw_phy_write16(sc, BCW_PHY_G_CRS, - (bcw_phy_read16(sc, BCW_PHY_G_CRS) & 0x7fff)); - bcw_phy_write16(sc, 0x0802, - (bcw_phy_read16(sc, 0x0802) & 0xfffc)); - bcw_phy_write16(sc, 0x0811, 0x01b3); - bcw_phy_write16(sc, 0x0812, 0x0fb2); - } - BCW_WRITE16(sc, BCW_MMIO_PHY_RADIO, - (BCW_READ16(sc, BCW_MMIO_PHY_RADIO) | 0x8000)); - } - backup[10] = bcw_phy_read16(sc, 0x0035); - bcw_phy_write16(sc, 0x0035, (bcw_phy_read16(sc, 0x0035) & 0xff7f)); - backup[11] = BCW_READ16(sc, 0x03e6); - backup[12] = BCW_READ16(sc, BCW_MMIO_CHANNEL_EXT); - - /* initialization */ - if (sc->sc_phy_ver == 0) - BCW_WRITE16(sc, 0x03e6, 0x0122); - else { - if (sc->sc_phy_ver >= 2) - BCW_WRITE16(sc, 0x03e6, 0x0040); - BCW_WRITE16(sc, BCW_MMIO_CHANNEL_EXT, - (BCW_READ16(sc, BCW_MMIO_CHANNEL_EXT) | 0x2000)); - } - - r = bcw_radio_calibrationvalue(sc); - - if (sc->sc_phy_type == BCW_PHY_TYPEB) - bcw_radio_write16(sc, 0x0078, 0x0003); - - bcw_phy_write16(sc, 0x0015, 0xbfaf); - bcw_phy_write16(sc, 0x002b, 0x1403); - if (sc->sc_phy_connected) - bcw_phy_write16(sc, 0x0812, 0x00b2); - bcw_phy_write16(sc, 0x0015, 0xbfa0); - bcw_radio_write16(sc, 0x0051, (bcw_radio_read16(sc, 0x0051) | - 0x0004)); - bcw_radio_write16(sc, 0x0052, 0); - bcw_radio_write16(sc, 0x0043, bcw_radio_read16(sc, 0x0043) | 0x0009); - bcw_phy_write16(sc, 0x0058, 0); - - for (i = 0; i < 16; i++) { - bcw_phy_write16(sc, 0x005a, 0x0480); - bcw_phy_write16(sc, 0x0059, 0xc810); - bcw_phy_write16(sc, 0x0058, 0x000d); - if (sc->sc_phy_connected) - bcw_phy_write16(sc, 0x0812, 0x30b2); - bcw_phy_write16(sc, 0x0015, 0xafb0); - delay(10); - if (sc->sc_phy_connected) - bcw_phy_write16(sc, 0x0812, 0x30b2); - bcw_phy_write16(sc, 0x0015, 0xefb0); - delay(10); - if (sc->sc_phy_connected) - bcw_phy_write16(sc, 0x0812, 0x30b2); - bcw_phy_write16(sc, 0x0015, 0xfff0); - delay(10); - tmp1 += bcw_phy_read16(sc, 0x002d); - bcw_phy_write16(sc, 0x0058, 0); - if (sc->sc_phy_connected) - bcw_phy_write16(sc, 0x0812, 0x30b2); - bcw_phy_write16(sc, 0x0015, 0xafb0); - } - - tmp1++; - tmp1 >>= 9; - delay(10); - bcw_phy_write16(sc, 0x0058, 0); - - for (i = 0; i < 16; i++) { - bcw_radio_write16(sc, 0x0078, (bcw_flip_4bit(i) << 1) | 0x0020); - - backup[13] = bcw_radio_read16(sc, 0x0078); - delay(10); - for (j = 0; j < 16; j++) { - bcw_phy_write16(sc, 0x005a, 0x0d80); - bcw_phy_write16(sc, 0x0059, 0xc810); - bcw_phy_write16(sc, 0x0058, 0x000d); - if (sc->sc_phy_connected) - bcw_phy_write16(sc, 0x0812, 0x30b2); - bcw_phy_write16(sc, 0x0015, 0xafb0); - delay(10); - if (sc->sc_phy_connected) - bcw_phy_write16(sc, 0x0812, 0x30b2); - bcw_phy_write16(sc, 0x0015, 0xefb0); - delay(10); - if (sc->sc_phy_connected) - bcw_phy_write16(sc, 0x812, 0x30b3); - bcw_phy_write16(sc, 0x0015, 0xfff0); - delay(10); - tmp2 += bcw_phy_read16(sc, 0x002d); - bcw_phy_write16(sc, 0x0058, 0); - if (sc->sc_phy_connected) - bcw_phy_write16(sc, 0x0812, 0x30b2); - bcw_phy_write16(sc, 0x0015, 0xafb0); - } - tmp2++; - tmp2 >>= 8; - if (tmp1 < tmp2) - break; - } - - /* restore the registers */ - bcw_phy_write16(sc, 0x0015, backup[1]); - bcw_radio_write16(sc, 0x0051, backup[14]); - bcw_radio_write16(sc, 0x0052, backup[15]); - bcw_radio_write16(sc, 0x0043, backup[0]); - bcw_phy_write16(sc, 0x005a, backup[16]); - bcw_phy_write16(sc, 0x0059, backup[17]); - bcw_phy_write16(sc, 0x0058, backup[18]); - BCW_WRITE16(sc, 0x03e6, backup[11]); - if (sc->sc_phy_ver != 0) - BCW_WRITE16(sc, BCW_MMIO_CHANNEL_EXT, backup[12]); - bcw_phy_write16(sc, 0x0035, backup[10]); - bcw_radio_select_channel(sc, sc->sc_radio_channel, 1); - if (sc->sc_phy_type == BCW_PHY_TYPEB) { - bcw_phy_write16(sc, 0x0030, backup[2]); - BCW_WRITE16(sc, 0x03ec, backup[3]); - } else { - BCW_WRITE16(sc, BCW_MMIO_PHY_RADIO, - (BCW_READ16(sc, BCW_MMIO_PHY_RADIO) & 0x7fff)); - if (sc->sc_phy_connected) { - bcw_phy_write16(sc, 0x0811, backup[4]); - bcw_phy_write16(sc, 0x0812, backup[5]); - bcw_phy_write16(sc, 0x0814, backup[6]); - bcw_phy_write16(sc, 0x0815, backup[7]); - bcw_phy_write16(sc, BCW_PHY_G_CRS, backup[8]); - bcw_phy_write16(sc, 0x0802, backup[9]); - } - } - if (i >= 15) - r = backup[13]; - - return (r); -} - -void -bcw_radio_init2060(struct bcw_softc *sc) -{ - int error; - - bcw_radio_write16(sc, 0x0004, 0x00c0); - bcw_radio_write16(sc, 0x0005, 0x0008); - bcw_radio_write16(sc, 0x0009, 0x0040); - bcw_radio_write16(sc, 0x0005, 0x00aa); - bcw_radio_write16(sc, 0x0032, 0x008f); - bcw_radio_write16(sc, 0x0006, 0x008f); - bcw_radio_write16(sc, 0x0034, 0x008f); - bcw_radio_write16(sc, 0x002c, 0x0007); - bcw_radio_write16(sc, 0x0082, 0x0080); - bcw_radio_write16(sc, 0x0080, 0x0000); - bcw_radio_write16(sc, 0x003f, 0x00d4); - bcw_radio_write16(sc, 0x0005, bcw_radio_read16(sc, 0x0005) & ~0x0008); - bcw_radio_write16(sc, 0x0081, bcw_radio_read16(sc, 0x0081) & ~0x0010); - bcw_radio_write16(sc, 0x0081, bcw_radio_read16(sc, 0x0081) & ~0x0020); - bcw_radio_write16(sc, 0x0081, bcw_radio_read16(sc, 0x0081) & ~0x0020); - delay(400); - bcw_radio_write16(sc, 0x0081, (bcw_radio_read16(sc, 0x0081) & ~0x0020) | - 0x0010); - delay(400); - bcw_radio_write16(sc, 0x0005, (bcw_radio_read16(sc, 0x0005) & ~0x0020) | - 0x0008); - bcw_radio_write16(sc, 0x0085, bcw_radio_read16(sc, 0x0085) & ~0x0010); - bcw_radio_write16(sc, 0x0005, bcw_radio_read16(sc, 0x0005) & ~0x0008); - bcw_radio_write16(sc, 0x0081, bcw_radio_read16(sc, 0x0081) & ~0x0040); - bcw_radio_write16(sc, 0x0081, (bcw_radio_read16(sc, 0x0081) & ~0x0040) | - 0x0040); - bcw_radio_write16(sc, 0x0005, (bcw_radio_read16(sc, 0x0081) & ~0x0008) | - 0x0008); - bcw_phy_write16(sc, 0x0063, 0xddc6); - bcw_phy_write16(sc, 0x0069, 0x07be); - bcw_phy_write16(sc, 0x006a, 0x0000); - - error = bcw_radio_select_channel(sc, BCW_RADIO_DEFAULT_CHANNEL_A, 0); - - delay(1000); -} - -void -bcw_radio_spw(struct bcw_softc *sc, uint8_t channel) -{ - if (sc->sc_radio_ver != 0x2050 || sc->sc_radio_rev >= 6) - /* we do not need the workaround */ - return; - - if (channel <= 10) - BCW_WRITE16(sc, BCW_MMIO_CHANNEL, - bcw_radio_chan2freq_bg(channel + 4)); - else - BCW_WRITE16(sc, BCW_MMIO_CHANNEL, bcw_radio_chan2freq_bg(1)); - - delay(100); - - BCW_WRITE16(sc, BCW_MMIO_CHANNEL, bcw_radio_chan2freq_bg(channel)); -} - -int -bcw_radio_select_channel(struct bcw_softc *sc, uint8_t channel, int spw) -{ - uint16_t freq, tmp, r8; - - if (sc->sc_radio_mnf == 0x17f && sc->sc_radio_ver == 0x2060 && - sc->sc_radio_rev == 1) { - freq = bcw_radio_chan2freq_a(channel); - - r8 = bcw_radio_read16(sc, 0x0008); - BCW_WRITE16(sc, 0x03f0, freq); - bcw_radio_write16(sc, 0x0008, r8); - - tmp = bcw_radio_read16(sc, 0x002e); - tmp &= 0x0080; - - bcw_radio_write16(sc, 0x002e, tmp); - - if (freq >= 4920 && freq <= 5500) - r8 = 3 * freq / 116; - bcw_radio_write16(sc, 0x0007, (r8 << 4) | r8); - bcw_radio_write16(sc, 0x0020, (r8 << 4) | r8); - bcw_radio_write16(sc, 0x0021, (r8 << 4) | r8); - bcw_radio_write16(sc, 0x0022, (bcw_radio_read16(sc, 0x0022) & - 0x000f) | (r8 << 4)); - bcw_radio_write16(sc, 0x002a, (r8 << 4)); - bcw_radio_write16(sc, 0x002b, (r8 << 4)); - bcw_radio_write16(sc, 0x0008, (bcw_radio_read16(sc, 0x0008) & - 0x00f0) | (r8 << 4)); - bcw_radio_write16(sc, 0x0029, (bcw_radio_read16(sc, 0x0029) & - 0xff0f) | 0x00b0); - bcw_radio_write16(sc, 0x0035, 0x00aa); - bcw_radio_write16(sc, 0x0036, 0x0085); - bcw_radio_write16(sc, 0x003a, (bcw_radio_read16(sc, 0x003a) & - 0xff20) | bcw_radio_freq_r3a_value(freq)); - bcw_radio_write16(sc, 0x003d, bcw_radio_read16(sc, 0x003d) & - 0x00ff); - bcw_radio_write16(sc, 0x0081, (bcw_radio_read16(sc, 0x0081) & - 0xff7f) | 0x0080); - bcw_radio_write16(sc, 0x0035, bcw_radio_read16(sc, 0x0035) & - 0xffef); - bcw_radio_write16(sc, 0x0035, (bcw_radio_read16(sc, 0x0035) & - 0xffef) | 0x0010); - bcw_radio_set_tx_iq(sc); - bcw_phy_xmitpower(sc); - } else { - if (spw) - bcw_radio_spw(sc, channel); - - BCW_WRITE16(sc, BCW_MMIO_CHANNEL, - bcw_radio_chan2freq_bg(channel)); - - if (channel == 14) { - /* TODO */ - } else { - BCW_WRITE16(sc, BCW_MMIO_CHANNEL_EXT, - BCW_READ16(sc, BCW_MMIO_CHANNEL_EXT) & 0xf7bf); - } - } - - sc->sc_radio_channel = channel; - - delay(8000); - - return (0); -} - -uint16_t -bcw_radio_chan2freq_a(uint8_t channel) -{ - return (5000 + 5 * channel); -} - -uint16_t -bcw_radio_chan2freq_bg(uint8_t channel) -{ - static const uint16_t freqs_bg[14] = { - 12, 17, 22, 27, - 32, 37, 42, 47, - 52, 57, 62, 67, - 72, 84 }; - - if (channel < 1 && channel > 14) - return (0); - - return (freqs_bg[channel - 1]); -} - -uint16_t -bcw_radio_default_baseband_atten(struct bcw_softc *sc) -{ - if (sc->sc_radio_ver == 0x2050 && sc->sc_radio_rev < 6) - return (0); - - return (2); -} - -uint16_t -bcw_radio_default_radio_atten(struct bcw_softc *sc) -{ - uint16_t att = 0xffff; - - if (sc->sc_phy_type == BCW_PHY_TYPEA) - return (0x60); - - switch (sc->sc_radio_ver) { - case 0x2053: - switch (sc->sc_radio_rev) { - case 1: - att = 6; - break; - } - break; - case 0x2050: - switch (sc->sc_radio_rev) { - case 0: - att = 5; - break; - case 1: - if (sc->sc_phy_type == BCW_PHY_TYPEG) { - if (sc->sc_board_vendor == - PCI_VENDOR_BROADCOM && - sc->sc_board_type == 0x0421 && - sc->sc_board_rev >= 30) - att = 3; - else if (sc->sc_board_vendor == - PCI_VENDOR_BROADCOM && - sc->sc_board_type == 0x0416) - att = 3; - else - att = 1; - } else { - if (sc->sc_board_vendor == - PCI_VENDOR_BROADCOM && - sc->sc_board_type == 0x0421 && - sc->sc_board_rev >= 30) - att = 7; - else - att = 6; - } - break; - case 2: - if (sc->sc_phy_type == BCW_PHY_TYPEG) { - if (sc->sc_board_vendor == - PCI_VENDOR_BROADCOM && - sc->sc_board_type == 0x0421 && - sc->sc_board_rev >= 30) - att = 3; - else if (sc->sc_board_vendor == - PCI_VENDOR_BROADCOM && - sc->sc_board_type == 0x0416) - att = 5; - else if (sc->sc_chip_id == 0x4320) - att = 4; - else - att = 3; - } else - att = 6; - break; - case 3: - att = 5; - break; - case 4: - case 5: - att = 1; - break; - case 6: - case 7: - att = 5; - break; - case 8: - att = 0x1a; - break; - case 9: - default: - att = 5; - } - } - if (sc->sc_board_vendor == PCI_VENDOR_BROADCOM && - sc->sc_board_type == 0x0421) { - if (sc->sc_board_rev < 0x43) - att = 2; - else if (sc->sc_board_rev < 0x51) - att = 3; - } - if (att == 0x0ffff) - att = 5; - - return (att); -} - -uint16_t -bcw_radio_default_txctl1(struct bcw_softc *sc) -{ - if (sc->sc_radio_ver != 0x2050) - return (0); - if (sc->sc_radio_rev == 1) - return (3); - if (sc->sc_radio_rev < 6) - return (2); - if (sc->sc_radio_rev == 8) - return (1); - - return (0); -} - -void -bcw_radio_clear_tssi(struct bcw_softc *sc) -{ - switch (sc->sc_phy_type) { - case BCW_PHY_TYPEA: - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x0068, 0x7f7f); - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x006a, 0x7f7f); - break; - case BCW_PHY_TYPEB: - case BCW_PHY_TYPEG: - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x0058, 0x7f7f); - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x005a, 0x7f7f); - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x0070, 0x7f7f); - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x0072, 0x7f7f); - break; - } -} - -void -bcw_radio_set_tx_iq(struct bcw_softc *sc) -{ - const uint8_t data_high[5] = { 0x00, 0x40, 0x80, 0x90, 0xD0 }; - const uint8_t data_low[5] = { 0x00, 0x01, 0x05, 0x06, 0x0A }; - uint16_t tmp = bcw_radio_read16(sc, 0x001e); - int i, j; - - for (i = 0; i < 5; i++) { - for (j = 0; j < 5; j++) { - if (tmp == (data_high[i] << 4 | data_low[j])) { - bcw_phy_write16(sc, 0x0069, (i - j) << 8 | - 0x00c0); - return; - } - } - } -} - -uint16_t -bcw_radio_get_txgain_baseband(uint16_t txpower) -{ - uint16_t r; - - if (txpower >= 54) - r = 2; - else if (txpower >= 49) - r = 4; - else if (txpower >= 44) - r = 5; - else - r = 6; - - return (r); -} - -uint16_t -bcw_radio_get_txgain_freq_power_amp(uint16_t txpower) -{ - uint16_t r; - - if (txpower >= 32) - r = 0; - else if (txpower >= 25) - r = 1; - else if (txpower >= 20) - r = 2; - else if (txpower >= 12) - r = 3; - else - r = 4; - - return (r); -} - -uint16_t -bcw_radio_get_txgain_dac(uint16_t txpower) -{ - uint16_t r; - - if (txpower >= 54) - r = txpower - 53; - else if (txpower >= 49) - r = txpower - 42; - else if (txpower >= 44) - r = txpower - 37; - else if (txpower >= 32) - r = txpower - 32; - else if (txpower >= 25) - r = txpower - 20; - else if (txpower >= 20) - r = txpower - 13; - else if (txpower >= 12) - r = txpower - 8; - else - r = txpower; - - return (r); -} - -uint16_t -bcw_radio_freq_r3a_value(uint16_t frequency) -{ - uint16_t val; - - if (frequency < 5091) - val = 0x0040; - else if (frequency < 5321) - val = 0; - else if (frequency < 5806) - val = 0x0080; - else - val = 0x0040; - - return (val); -} - -int -bcw_radio_set_interf_mitigation(struct bcw_softc *sc, int mode) -{ - int currentmode; - - if (sc->sc_phy_type != BCW_PHY_TYPEG || sc->sc_phy_type == 0 || - sc->sc_phy_connected == 0) - return (ENODEV); - - sc->sc_radio_aci_wlan_automatic = 0; - switch (mode) { - case BCW_RADIO_INTERFMODE_AUTOWLAN: - sc->sc_radio_aci_wlan_automatic = 1; - if (sc->sc_radio_aci_enable) - mode = BCW_RADIO_INTERFMODE_MANUALWLAN; - else - mode = BCW_RADIO_INTERFMODE_NONE; - break; - case BCW_RADIO_INTERFMODE_NONE: - case BCW_RADIO_INTERFMODE_NONWLAN: - case BCW_RADIO_INTERFMODE_MANUALWLAN: - break; - default: - return (EINVAL); - } - - currentmode = sc->sc_radio_interfmode; - if (currentmode == mode) - return (0); - if (currentmode != BCW_RADIO_INTERFMODE_NONE) { - sc->sc_radio_aci_enable = 0; - sc->sc_radio_aci_hw_rssi = 0; - } else - bcw_radio_interf_mitigation_enable(sc, mode); - sc->sc_radio_interfmode = mode; - - return (0); -} - -int -bcw_radio_interf_mitigation_enable(struct bcw_softc *sc, int mode) -{ - uint16_t tmp, flipped; - uint32_t tmp32; - size_t stackidx = 0; - uint32_t *stack = sc->sc_radio_interfstack; - - switch (mode) { - case BCW_RADIO_INTERFMODE_NONWLAN: - if (sc->sc_phy_rev != 1) { - bcw_phy_write16(sc, 0x042b, - bcw_phy_read16(sc, 0x042b) | 0x0800); - bcw_phy_write16(sc, BCW_PHY_G_CRS, - bcw_phy_read16(sc, BCW_PHY_G_CRS) & ~0x4000); - break; - } - BCW_RADIO_STACKSAVE(0x0078); - tmp = (bcw_radio_read16(sc, 0x0078) & 0x001e); - flipped = bcw_flip_4bit(tmp); - if (flipped < 10 && flipped >= 8) - flipped = 7; - else if (flipped >= 10) - flipped -= 3; - flipped = bcw_flip_4bit(flipped); - flipped = (flipped << 1) | 0x0020; - bcw_radio_write16(sc, 0x0078, flipped); - - bcw_radio_calc_nrssi_threshold(sc); - - BCW_PHY_STACKSAVE(0x0406); - bcw_phy_write16(sc, 0x0406, 0x7e28); - - bcw_phy_write16(sc, 0x042b, - bcw_phy_read16(sc, 0x042b) | 0x08000); - bcw_phy_write16(sc, BCW_PHY_RADIO_BITFIELD, - bcw_phy_read16(sc, BCW_PHY_RADIO_BITFIELD) | 0x1000); - BCW_PHY_STACKSAVE(0x04a0); - bcw_phy_write16(sc, 0x04a0, - (bcw_phy_read16(sc, 0x04a0) & 0xc0c0) | 0x0008); - BCW_PHY_STACKSAVE(0x04a1); - bcw_phy_write16(sc, 0x04a1, - (bcw_phy_read16(sc, 0x04a1) & 0xc0c0) | 0x0605); - BCW_PHY_STACKSAVE(0x04a2); - bcw_phy_write16(sc, 0x04a2, - (bcw_phy_read16(sc, 0x04a2) & 0xc0c0) | 0x0204); - BCW_PHY_STACKSAVE(0x04a8); - bcw_phy_write16(sc, 0x04a8, - (bcw_phy_read16(sc, 0x04a8) & 0xc0c0) | 0x0803); - BCW_PHY_STACKSAVE(0x04ab); - bcw_phy_write16(sc, 0x04ab, - (bcw_phy_read16(sc, 0x04ab) & 0xc0c0) | 0x0605); - - BCW_PHY_STACKSAVE(0x04a7); - bcw_phy_write16(sc, 0x04a7, 0x0002); - BCW_PHY_STACKSAVE(0x04a3); - bcw_phy_write16(sc, 0x04a3, 0x287a); - BCW_PHY_STACKSAVE(0x04a9); - bcw_phy_write16(sc, 0x04a9, 0x2027); - BCW_PHY_STACKSAVE(0x0493); - bcw_phy_write16(sc, 0x0493, 0x32f5); - BCW_PHY_STACKSAVE(0x04aa); - bcw_phy_write16(sc, 0x04aa, 0x2027); - BCW_PHY_STACKSAVE(0x04ac); - bcw_phy_write16(sc, 0x04ac, 0x32f5); - break; - case BCW_RADIO_INTERFMODE_MANUALWLAN: - if (bcw_phy_read16(sc, 0x0033) & 0x0800) - break; - - sc->sc_radio_aci_enable = 1; - - BCW_PHY_STACKSAVE(BCW_PHY_RADIO_BITFIELD); - BCW_PHY_STACKSAVE(BCW_PHY_G_CRS); - if (sc->sc_phy_rev < 2) - BCW_PHY_STACKSAVE(0x0406); - else { - BCW_PHY_STACKSAVE(0x04c0); - BCW_PHY_STACKSAVE(0x04c1); - } - BCW_PHY_STACKSAVE(0x0033); - BCW_PHY_STACKSAVE(0x04a7); - BCW_PHY_STACKSAVE(0x04a3); - BCW_PHY_STACKSAVE(0x04a9); - BCW_PHY_STACKSAVE(0x04aa); - BCW_PHY_STACKSAVE(0x04ac); - BCW_PHY_STACKSAVE(0x0493); - BCW_PHY_STACKSAVE(0x04a1); - BCW_PHY_STACKSAVE(0x04a0); - BCW_PHY_STACKSAVE(0x04a2); - BCW_PHY_STACKSAVE(0x048a); - BCW_PHY_STACKSAVE(0x04a8); - BCW_PHY_STACKSAVE(0x04ab); - if (sc->sc_phy_rev == 2) { - BCW_PHY_STACKSAVE(0x04ad); - BCW_PHY_STACKSAVE(0x04ae); - } else if (sc->sc_phy_rev >= 3) { - BCW_PHY_STACKSAVE(0x04ad); - BCW_PHY_STACKSAVE(0x0415); - BCW_PHY_STACKSAVE(0x0416); - BCW_PHY_STACKSAVE(0x0417); - BCW_ILT_STACKSAVE(0x1a00 + 0x2); - BCW_ILT_STACKSAVE(0x1a00 + 0x3); - } - BCW_PHY_STACKSAVE(0x042b); - BCW_PHY_STACKSAVE(0x048c); - - bcw_phy_write16(sc, BCW_PHY_RADIO_BITFIELD, - bcw_phy_read16(sc, BCW_PHY_RADIO_BITFIELD) & ~0x1000); - bcw_phy_write16(sc, BCW_PHY_G_CRS, - (bcw_phy_read16(sc, BCW_PHY_G_CRS) & 0xfffc) | 0x0002); - - bcw_phy_write16(sc, 0x0033, 0x0800); - bcw_phy_write16(sc, 0x04a3, 0x2027); - bcw_phy_write16(sc, 0x04a9, 0x1ca8); - bcw_phy_write16(sc, 0x0493, 0x287a); - bcw_phy_write16(sc, 0x04aa, 0x1ca8); - bcw_phy_write16(sc, 0x04ac, 0x287a); - - bcw_phy_write16(sc, 0x04a0, - (bcw_phy_read16(sc, 0x04a0) & 0xffc0) | 0x001a); - bcw_phy_write16(sc, 0x04a7, 0x000d); - - if (sc->sc_phy_rev < 2) { - bcw_phy_write16(sc, 0x0406, 0xff0d); - } else if (sc->sc_phy_rev == 2) { - bcw_phy_write16(sc, 0x04c0, 0xffff); - bcw_phy_write16(sc, 0x04c1, 0x00a9); - } else { - bcw_phy_write16(sc, 0x04c0, 0x00c1); - bcw_phy_write16(sc, 0x04c1, 0x0059); - } - - bcw_phy_write16(sc, 0x04a1, - (bcw_phy_read16(sc, 0x04a1) & 0xc0ff) | 0x1800); - bcw_phy_write16(sc, 0x04a1, - (bcw_phy_read16(sc, 0x04a1) & 0xc0ff) | 0x0015); - bcw_phy_write16(sc, 0x04a8, - (bcw_phy_read16(sc, 0x04a8) & 0xcfff) | 0x1000); - bcw_phy_write16(sc, 0x04a8, - (bcw_phy_read16(sc, 0x04a8) & 0xf0ff) | 0x0a00); - bcw_phy_write16(sc, 0x04ab, - (bcw_phy_read16(sc, 0x04a8) & 0xcfff) | 0x1000); - bcw_phy_write16(sc, 0x04ab, - (bcw_phy_read16(sc, 0x04ab) & 0xf0ff) | 0x0800); - bcw_phy_write16(sc, 0x04ab, - (bcw_phy_read16(sc, 0x04ab) & 0xf0ff) | 0x0010); - bcw_phy_write16(sc, 0x04ab, - (bcw_phy_read16(sc, 0x04ab) & 0xfff0) | 0x0005); - bcw_phy_write16(sc, 0x04a8, - (bcw_phy_read16(sc, 0x04a8) & 0xffcf) | 0x0010); - bcw_phy_write16(sc, 0x04a8, - (bcw_phy_read16(sc, 0x04a8) & 0xfff0) | 0x0006); - bcw_phy_write16(sc, 0x04a2, - (bcw_phy_read16(sc, 0x04a2) & 0xf0ff) | 0x0800); - bcw_phy_write16(sc, 0x04a0, - (bcw_phy_read16(sc, 0x04a0) & 0xf0ff) | 0x0500); - bcw_phy_write16(sc, 0x04a2, - (bcw_phy_read16(sc, 0x04a2) & 0xfff0) | 0x000b); - - if (sc->sc_phy_rev >= 3) { - bcw_phy_write16(sc, 0x048a, - bcw_phy_read16(sc, 0x048a) & ~0x8000); - bcw_phy_write16(sc, 0x0415, - (bcw_phy_read16(sc, 0x0415) & 0x8000) | 0x36d8); - bcw_phy_write16(sc, 0x0416, - (bcw_phy_read16(sc, 0x0416) & 0x8000) | 0x36d8); - bcw_phy_write16(sc, 0x0417, - (bcw_phy_read16(sc, 0x0417) & 0xfe00) | 0x016d); - } else { - bcw_phy_write16(sc, 0x048a, - bcw_phy_read16(sc, 0x048a) | 0x1000); - bcw_phy_write16(sc, 0x048a, - (bcw_phy_read16(sc, 0x048a) & 0x9fff) | 0x2000); - tmp32 = bcw_shm_read16(sc, BCW_SHM_SHARED, - BCW_UCODEFLAGS_OFFSET); - if (!(tmp32 & 0x800)) { - tmp32 |= 0x800; - bcw_shm_write16(sc, BCW_SHM_SHARED, - BCW_UCODEFLAGS_OFFSET, tmp32); - } - } - if (sc->sc_phy_rev >= 2) { - bcw_phy_write16(sc, 0x042b, bcw_phy_read16(sc, 0x042b) | - 0x0800); - } - bcw_phy_write16(sc, 0x048c, (bcw_phy_read16(sc, 0x048c) & - 0xff00) | 0x007f); - if (sc->sc_phy_rev == 2) { - bcw_phy_write16(sc, 0x04ae, - (bcw_phy_read16(sc, 0x04ae) & 0xff00) | 0x007f); - bcw_phy_write16(sc, 0x04ad, - (bcw_phy_read16(sc, 0x04ad) & 0x00ff) | 0x1300); - } else if (sc->sc_phy_rev >= 6) { - bcw_ilt_write(sc, 0x1a00 + 0x3, 0x007f); - bcw_ilt_write(sc, 0x1a00 + 0x2, 0x007f); - bcw_phy_write16(sc, 0x04ad, - bcw_phy_read16(sc, 0x04ad) & 0x00ff); - } - bcw_radio_calc_nrssi_slope(sc); - break; - default: - /* XXX panic()? */ - break; - } - - return (0); -} - -void -bcw_radio_set_txantenna(struct bcw_softc *sc, uint32_t val) -{ - uint16_t tmp; - - val <<= 8; - tmp = bcw_shm_read16(sc, BCW_SHM_SHARED, 0x0022) & 0xfcff; - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x0022, tmp | val); - tmp = bcw_shm_read16(sc, BCW_SHM_SHARED, 0x03a8) & 0xfcff; - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x03a8, tmp | val); - tmp = bcw_shm_read16(sc, BCW_SHM_SHARED, 0x0054) & 0xfcff; - bcw_shm_write16(sc, BCW_SHM_SHARED, 0x0054, tmp | val); -} - -void -bcw_radio_lock(struct bcw_softc *sc) -{ - uint32_t status; - - status = BCW_READ(sc, BCW_MMIO_SBF); - status |= BCW_SBF_RADIOREG_LOCK; - BCW_WRITE(sc, BCW_MMIO_SBF, status); - delay(10); -} - -void -bcw_radio_unlock(struct bcw_softc *sc) -{ - uint32_t status; - - BCW_READ16(sc, 0x3e0); - status = BCW_READ(sc, BCW_MMIO_SBF); - status &= ~BCW_SBF_RADIOREG_LOCK; - BCW_WRITE(sc, BCW_MMIO_SBF, status); -} - -/* - * ILT - */ -void -bcw_ilt_write(struct bcw_softc *sc, uint16_t offset, uint16_t val) -{ - if (sc->sc_phy_type == BCW_PHY_TYPEA) { - bcw_phy_write16(sc, BCW_PHY_ILT_A_CTRL, offset); - bcw_phy_write16(sc, BCW_PHY_ILT_A_DATA1, val); - } else { - bcw_phy_write16(sc, BCW_PHY_ILT_G_CTRL, offset); - bcw_phy_write16(sc, BCW_PHY_ILT_G_DATA1, val); - } -} - -uint16_t -bcw_ilt_read(struct bcw_softc *sc, uint16_t offset) -{ - if (sc->sc_phy_type == BCW_PHY_TYPEA) { - bcw_phy_write16(sc, BCW_PHY_ILT_A_CTRL, offset); - return (bcw_phy_read16(sc, BCW_PHY_ILT_A_DATA1)); - } else { - bcw_phy_write16(sc, BCW_PHY_ILT_G_CTRL, offset); - return (bcw_phy_read16(sc, BCW_PHY_ILT_G_DATA1)); - } -} - -/* - * Power Control - */ -void -bcw_pc_crystal_on(struct bcw_softc *sc) -{ - uint32_t val; - - val = (sc->sc_conf_read)(sc->sc_dev_softc, BCW_GPIOI); - if (val & BCW_PCTL_XTAL_POWERUP) - return; /* crystal is already on */ - - val = (sc->sc_conf_read)(sc->sc_dev_softc, BCW_GPIOO); - val |= (BCW_PCTL_XTAL_POWERUP | BCW_PCTL_PLL_POWERDOWN); - (sc->sc_conf_write)(sc->sc_dev_softc, BCW_GPIOO, val); - (sc->sc_conf_write)(sc->sc_dev_softc, BCW_GPIOE, val); - - delay(1000); - - val = (sc->sc_conf_read)(sc->sc_dev_softc, BCW_GPIOO); - val &= ~BCW_PCTL_PLL_POWERDOWN; - (sc->sc_conf_write)(sc->sc_dev_softc, BCW_GPIOO, val); - - delay(5000); -} - -void -bcw_pc_crystal_off(struct bcw_softc *sc) -{ - uint32_t val; - - /* TODO return if radio is hardware disabled */ - - if (sc->sc_chip_rev < 5) - return; - if (sc->sc_sprom.boardflags & BCW_BF_XTAL) - return; - - bcw_pc_set_clock(sc, BCW_PCTL_CLK_SLOW); - - val = (sc->sc_conf_read)(sc->sc_dev_softc, BCW_GPIOO); - val |= BCW_PCTL_PLL_POWERDOWN; - val &= ~BCW_PCTL_XTAL_POWERUP; - (sc->sc_conf_write)(sc->sc_dev_softc, BCW_GPIOO, val); - - val = (sc->sc_conf_read)(sc->sc_dev_softc, BCW_GPIOE); - val |= (BCW_PCTL_PLL_POWERDOWN | BCW_PCTL_XTAL_POWERUP); - (sc->sc_conf_write)(sc->sc_dev_softc, BCW_GPIOE, val); -} - -int -bcw_pc_init(struct bcw_softc *sc) -{ - int maxfreq; - int error; - - if ((error = bcw_change_core(sc, 0))) - return (error); - - if (sc->sc_chip_id == 0x4321) { - if (sc->sc_chip_rev == 1) - BCW_WRITE(sc, BCW_CHIPCOMMON_CTL, 0x00a4); - if (sc->sc_chip_rev == 0) - BCW_WRITE(sc, BCW_CHIPCOMMON_CTL, 0x03a4); - } - - if (!(sc->sc_chip_common_capa & BCW_CAPABILITIES_PCTL)) - return (ENODEV); /* no power control */ - - if (sc->sc_core[sc->sc_currentcore].rev >= 10) { - BCW_WRITE(sc, BCW_CHIPCOMMON_SYSCLKCTL, - (BCW_READ(sc, BCW_CHIPCOMMON_SYSCLKCTL) & 0x0000ffff) | - 0x40000); - } else { - maxfreq = bcw_pc_clock_freqlimit(sc, 1); - BCW_WRITE(sc, BCW_CHIPCOMMON_PLLONDELAY, - (maxfreq * 150 + 99999) / 1000000); - BCW_WRITE(sc, BCW_CHIPCOMMON_FREFSELDELAY, - (maxfreq * 15 + 99999) / 100000); - } - - if ((error = bcw_change_core(sc, sc->sc_lastcore))) - return (error); - - return (0); -} - -int -bcw_pc_set_clock(struct bcw_softc *sc, uint16_t mode) -{ - int error; - uint32_t tmp; - - if ((error = bcw_change_core(sc, 0))) - return (error); - - if (sc->sc_core[sc->sc_currentcore].rev < 6) { - if (mode == BCW_PCTL_CLK_FAST) - bcw_pc_crystal_on(sc); - } else if (sc->sc_chip_common_capa & BCW_CAPABILITIES_PCTL && - sc->sc_core[sc->sc_currentcore].rev < 10) { - switch (mode) { - case BCW_PCTL_CLK_FAST: - tmp = BCW_READ(sc, BCW_CHIPCOMMON_SLOWCLKCTL); - tmp = (tmp & ~BCW_PCTL_FORCE_SLOW) | - BCW_PCTL_FORCE_PLL; - BCW_WRITE(sc, BCW_CHIPCOMMON_SLOWCLKCTL, tmp); - break; - case BCW_PCTL_CLK_SLOW: - tmp = BCW_READ(sc, BCW_CHIPCOMMON_SLOWCLKCTL); - tmp |= BCW_PCTL_FORCE_SLOW; - BCW_WRITE(sc, BCW_CHIPCOMMON_SLOWCLKCTL, tmp); - break; - case BCW_PCTL_CLK_DYNAMIC: - tmp = BCW_READ(sc, BCW_CHIPCOMMON_SLOWCLKCTL); - tmp &= ~BCW_PCTL_FORCE_SLOW; - tmp |= BCW_PCTL_FORCE_PLL; - tmp &= ~BCW_PCTL_DYN_XTAL; - BCW_WRITE(sc, BCW_CHIPCOMMON_SLOWCLKCTL, tmp); - break; - } - } - - if ((error = bcw_change_core(sc, sc->sc_lastcore))) - return (error); - - return (0); -} - -void -bcw_pc_saving_ctl_bits(struct bcw_softc *sc, int bit25, int bit26) -{ - int i; - uint32_t status; - - bit25 = 0; - bit26 = 1; - - if (bit25 == -1) { - /* TODO */ - } - if (bit26 == -1) { - /* TODO */ - } - - status = BCW_READ(sc, BCW_MMIO_SBF); - if (bit25) - status |= BCW_SBF_PS1; - else - status &= ~BCW_SBF_PS1; - if (bit26) - status |= BCW_SBF_PS2; - else - status &= ~BCW_SBF_PS2; - BCW_WRITE(sc, BCW_MMIO_SBF, status); - if (bit26 && sc->sc_core[sc->sc_currentcore].rev >= 5) { - for (i = 0; i < 100; i++) { - if (bcw_shm_read32(sc, BCW_SHM_SHARED, 0x0040) != 4) - break; - delay(10); - } - } -} - -uint16_t -bcw_pc_powerup_delay(struct bcw_softc *sc) -{ - uint16_t delay = 0; - uint32_t pllondelay; - int minfreq; - - /* TODO just for PCI bus type */ - - if (!(sc->sc_chip_common_capa & BCW_CAPABILITIES_PCTL)) - return (0); - - bcw_change_core(sc, 0); - - minfreq = bcw_pc_clock_freqlimit(sc, 0); - pllondelay = BCW_READ(sc, BCW_CHIPCOMMON_PLLONDELAY); - delay = (((pllondelay + 2) * 1000000) + (minfreq - 1)) / minfreq; - - bcw_change_core(sc, sc->sc_lastcore); - - return (delay); -} - -int -bcw_pc_clock_freqlimit(struct bcw_softc *sc, int getmax) -{ - int limit, clocksrc, divisor; - uint32_t tmp; - - clocksrc = bcw_pc_get_slowclocksrc(sc); - - if (sc->sc_core[sc->sc_currentcore].rev < 6) { - switch (clocksrc) { - case 0: - divisor = 64; - break; - case 1: - divisor = 32; - break; - default: - /* XXX panic()? */ - divisor = 1; - } - } else if (sc->sc_core[sc->sc_currentcore].rev < 10) { - switch (clocksrc) { - case 2: - divisor = 1; - break; - case 1: - case 0: - tmp = BCW_READ(sc, 0xb8); - divisor = ((tmp & 0xffff0000) >> 16) + 1; - divisor *= 4; - break; - default: - /* XXX panic()? */ - divisor = 1; - } - } else { - tmp = BCW_READ(sc, 0xb8); - divisor = ((tmp & 0xffff0000) >> 16) + 1; - divisor *= 4; - } - - switch (clocksrc) { - case 2: - if (getmax) - limit = 43000; - else - limit = 25000; - break; - case 1: - if (getmax) - limit = 20200000; - else - limit = 19800000; - break; - case 0: - if (getmax) - limit = 34000000; - else - limit = 25000000; - break; - default: - /* panic()? */ - limit = 0; - } - - limit /= divisor; - - return (limit); - -} - -int -bcw_pc_get_slowclocksrc(struct bcw_softc *sc) -{ - uint32_t tmp; - - if (sc->sc_core[sc->sc_currentcore].rev < 6) { - /* TODO check bus type */ - tmp = (sc->sc_conf_read)(sc, BCW_PCTL_OUT); - if (tmp & 0x10) - return (0); - return (1); - } - - if (sc->sc_core[sc->sc_currentcore].rev < 10) { - tmp = BCW_READ(sc, 0xb8); /* XXX */ - tmp &= 0x7; - if (tmp == 0) - return (2); - if (tmp == 1) - return (1); - if (tmp == 2) - return (0); - } - - return (1); -} - -/* - * XMIT - */ -uint8_t -bcw_xmit_plcp_get_ratecode_cck(const uint8_t bitrate) -{ - if (ieee80211_std_rateset_11b.rs_rates[0] == bitrate) - return (0x0a); - else if (ieee80211_std_rateset_11b.rs_rates[1] == bitrate) - return (0x14); - else if (ieee80211_std_rateset_11b.rs_rates[2] == bitrate) - return (0x37); - else if (ieee80211_std_rateset_11b.rs_rates[3] == bitrate) - return (0x6e); - - return (0); -} - -uint8_t -bcw_xmit_plcp_get_ratecode_ofdm(const uint8_t bitrate) -{ - if (ieee80211_std_rateset_11g.rs_rates[0] == bitrate) - return (0xb); - else if (ieee80211_std_rateset_11g.rs_rates[1] == bitrate) - return (0xf); - else if (ieee80211_std_rateset_11g.rs_rates[2] == bitrate) - return (0xa); - else if (ieee80211_std_rateset_11g.rs_rates[3] == bitrate) - return (0xe); - else if (ieee80211_std_rateset_11g.rs_rates[4] == bitrate) - return (0x9); - else if (ieee80211_std_rateset_11g.rs_rates[5] == bitrate) - return (0xd); - else if (ieee80211_std_rateset_11g.rs_rates[6] == bitrate) - return (0x8); - else if (ieee80211_std_rateset_11g.rs_rates[7] == bitrate) - return (0xc); - - return (0); -} |