[BACK]Return to ar5212.c CVS log [TXT][DIR] Up to [local] / sys / dev / ic

Annotation of sys/dev/ic/ar5212.c, Revision 1.1

1.1     ! nbrk        1: /*     $OpenBSD: ar5212.c,v 1.40 2007/04/10 17:47:55 miod Exp $        */
        !             2:
        !             3: /*
        !             4:  * Copyright (c) 2004, 2005, 2006, 2007 Reyk Floeter <reyk@openbsd.org>
        !             5:  *
        !             6:  * Permission to use, copy, modify, and distribute this software for any
        !             7:  * purpose with or without fee is hereby granted, provided that the above
        !             8:  * copyright notice and this permission notice appear in all copies.
        !             9:  *
        !            10:  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
        !            11:  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
        !            12:  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
        !            13:  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
        !            14:  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
        !            15:  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
        !            16:  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
        !            17:  */
        !            18:
        !            19: /*
        !            20:  * HAL interface for the Atheros AR5001 Wireless LAN chipset
        !            21:  * (AR5212 + AR5111/AR5112).
        !            22:  */
        !            23:
        !            24: #include <dev/ic/ar5xxx.h>
        !            25: #include <dev/ic/ar5212reg.h>
        !            26: #include <dev/ic/ar5212var.h>
        !            27:
        !            28: HAL_BOOL        ar5k_ar5212_nic_reset(struct ath_hal *, u_int32_t);
        !            29: HAL_BOOL        ar5k_ar5212_nic_wakeup(struct ath_hal *, u_int16_t);
        !            30: u_int16_t       ar5k_ar5212_radio_revision(struct ath_hal *, HAL_CHIP);
        !            31: const void      ar5k_ar5212_fill(struct ath_hal *);
        !            32: HAL_BOOL        ar5k_ar5212_txpower(struct ath_hal *, HAL_CHANNEL *, u_int);
        !            33:
        !            34: /*
        !            35:  * Initial register setting for the AR5212
        !            36:  */
        !            37: static const struct ar5k_ar5212_ini ar5212_ini[] =
        !            38:     AR5K_AR5212_INI;
        !            39: static const struct ar5k_ar5212_ini_mode ar5212_mode[] =
        !            40:     AR5K_AR5212_INI_MODE;
        !            41:
        !            42: AR5K_HAL_FUNCTIONS(extern, ar5k_ar5212,);
        !            43:
        !            44: const void
        !            45: ar5k_ar5212_fill(struct ath_hal *hal)
        !            46: {
        !            47:        hal->ah_magic = AR5K_AR5212_MAGIC;
        !            48:
        !            49:        /*
        !            50:         * Init/Exit functions
        !            51:         */
        !            52:        AR5K_HAL_FUNCTION(hal, ar5212, get_rate_table);
        !            53:        AR5K_HAL_FUNCTION(hal, ar5212, detach);
        !            54:
        !            55:        /*
        !            56:         * Reset functions
        !            57:         */
        !            58:        AR5K_HAL_FUNCTION(hal, ar5212, reset);
        !            59:        AR5K_HAL_FUNCTION(hal, ar5212, set_opmode);
        !            60:        AR5K_HAL_FUNCTION(hal, ar5212, calibrate);
        !            61:
        !            62:        /*
        !            63:         * TX functions
        !            64:         */
        !            65:        AR5K_HAL_FUNCTION(hal, ar5212, update_tx_triglevel);
        !            66:        AR5K_HAL_FUNCTION(hal, ar5212, setup_tx_queue);
        !            67:        AR5K_HAL_FUNCTION(hal, ar5212, setup_tx_queueprops);
        !            68:        AR5K_HAL_FUNCTION(hal, ar5212, release_tx_queue);
        !            69:        AR5K_HAL_FUNCTION(hal, ar5212, reset_tx_queue);
        !            70:        AR5K_HAL_FUNCTION(hal, ar5212, get_tx_buf);
        !            71:        AR5K_HAL_FUNCTION(hal, ar5212, put_tx_buf);
        !            72:        AR5K_HAL_FUNCTION(hal, ar5212, tx_start);
        !            73:        AR5K_HAL_FUNCTION(hal, ar5212, stop_tx_dma);
        !            74:        AR5K_HAL_FUNCTION(hal, ar5212, setup_tx_desc);
        !            75:        AR5K_HAL_FUNCTION(hal, ar5212, setup_xtx_desc);
        !            76:        AR5K_HAL_FUNCTION(hal, ar5212, fill_tx_desc);
        !            77:        AR5K_HAL_FUNCTION(hal, ar5212, proc_tx_desc);
        !            78:        AR5K_HAL_FUNCTION(hal, ar5212, has_veol);
        !            79:
        !            80:        /*
        !            81:         * RX functions
        !            82:         */
        !            83:        AR5K_HAL_FUNCTION(hal, ar5212, get_rx_buf);
        !            84:        AR5K_HAL_FUNCTION(hal, ar5212, put_rx_buf);
        !            85:        AR5K_HAL_FUNCTION(hal, ar5212, start_rx);
        !            86:        AR5K_HAL_FUNCTION(hal, ar5212, stop_rx_dma);
        !            87:        AR5K_HAL_FUNCTION(hal, ar5212, start_rx_pcu);
        !            88:        AR5K_HAL_FUNCTION(hal, ar5212, stop_pcu_recv);
        !            89:        AR5K_HAL_FUNCTION(hal, ar5212, set_mcast_filter);
        !            90:        AR5K_HAL_FUNCTION(hal, ar5212, set_mcast_filterindex);
        !            91:        AR5K_HAL_FUNCTION(hal, ar5212, clear_mcast_filter_idx);
        !            92:        AR5K_HAL_FUNCTION(hal, ar5212, get_rx_filter);
        !            93:        AR5K_HAL_FUNCTION(hal, ar5212, set_rx_filter);
        !            94:        AR5K_HAL_FUNCTION(hal, ar5212, setup_rx_desc);
        !            95:        AR5K_HAL_FUNCTION(hal, ar5212, proc_rx_desc);
        !            96:        AR5K_HAL_FUNCTION(hal, ar5212, set_rx_signal);
        !            97:
        !            98:        /*
        !            99:         * Misc functions
        !           100:         */
        !           101:        AR5K_HAL_FUNCTION(hal, ar5212, dump_state);
        !           102:        AR5K_HAL_FUNCTION(hal, ar5212, get_diag_state);
        !           103:        AR5K_HAL_FUNCTION(hal, ar5212, get_lladdr);
        !           104:        AR5K_HAL_FUNCTION(hal, ar5212, set_lladdr);
        !           105:        AR5K_HAL_FUNCTION(hal, ar5212, set_regdomain);
        !           106:        AR5K_HAL_FUNCTION(hal, ar5212, set_ledstate);
        !           107:        AR5K_HAL_FUNCTION(hal, ar5212, set_associd);
        !           108:        AR5K_HAL_FUNCTION(hal, ar5212, set_gpio_input);
        !           109:        AR5K_HAL_FUNCTION(hal, ar5212, set_gpio_output);
        !           110:        AR5K_HAL_FUNCTION(hal, ar5212, get_gpio);
        !           111:        AR5K_HAL_FUNCTION(hal, ar5212, set_gpio);
        !           112:        AR5K_HAL_FUNCTION(hal, ar5212, set_gpio_intr);
        !           113:        AR5K_HAL_FUNCTION(hal, ar5212, get_tsf32);
        !           114:        AR5K_HAL_FUNCTION(hal, ar5212, get_tsf64);
        !           115:        AR5K_HAL_FUNCTION(hal, ar5212, reset_tsf);
        !           116:        AR5K_HAL_FUNCTION(hal, ar5212, get_regdomain);
        !           117:        AR5K_HAL_FUNCTION(hal, ar5212, detect_card_present);
        !           118:        AR5K_HAL_FUNCTION(hal, ar5212, update_mib_counters);
        !           119:        AR5K_HAL_FUNCTION(hal, ar5212, get_rf_gain);
        !           120:        AR5K_HAL_FUNCTION(hal, ar5212, set_slot_time);
        !           121:        AR5K_HAL_FUNCTION(hal, ar5212, get_slot_time);
        !           122:        AR5K_HAL_FUNCTION(hal, ar5212, set_ack_timeout);
        !           123:        AR5K_HAL_FUNCTION(hal, ar5212, get_ack_timeout);
        !           124:        AR5K_HAL_FUNCTION(hal, ar5212, set_cts_timeout);
        !           125:        AR5K_HAL_FUNCTION(hal, ar5212, get_cts_timeout);
        !           126:
        !           127:        /*
        !           128:         * Key table (WEP) functions
        !           129:         */
        !           130:        AR5K_HAL_FUNCTION(hal, ar5212, is_cipher_supported);
        !           131:        AR5K_HAL_FUNCTION(hal, ar5212, get_keycache_size);
        !           132:        AR5K_HAL_FUNCTION(hal, ar5212, reset_key);
        !           133:        AR5K_HAL_FUNCTION(hal, ar5212, is_key_valid);
        !           134:        AR5K_HAL_FUNCTION(hal, ar5212, set_key);
        !           135:        AR5K_HAL_FUNCTION(hal, ar5212, set_key_lladdr);
        !           136:
        !           137:        /*
        !           138:         * Power management functions
        !           139:         */
        !           140:        AR5K_HAL_FUNCTION(hal, ar5212, set_power);
        !           141:        AR5K_HAL_FUNCTION(hal, ar5212, get_power_mode);
        !           142:        AR5K_HAL_FUNCTION(hal, ar5212, query_pspoll_support);
        !           143:        AR5K_HAL_FUNCTION(hal, ar5212, init_pspoll);
        !           144:        AR5K_HAL_FUNCTION(hal, ar5212, enable_pspoll);
        !           145:        AR5K_HAL_FUNCTION(hal, ar5212, disable_pspoll);
        !           146:
        !           147:        /*
        !           148:         * Beacon functions
        !           149:         */
        !           150:        AR5K_HAL_FUNCTION(hal, ar5212, init_beacon);
        !           151:        AR5K_HAL_FUNCTION(hal, ar5212, set_beacon_timers);
        !           152:        AR5K_HAL_FUNCTION(hal, ar5212, reset_beacon);
        !           153:        AR5K_HAL_FUNCTION(hal, ar5212, wait_for_beacon);
        !           154:
        !           155:        /*
        !           156:         * Interrupt functions
        !           157:         */
        !           158:        AR5K_HAL_FUNCTION(hal, ar5212, is_intr_pending);
        !           159:        AR5K_HAL_FUNCTION(hal, ar5212, get_isr);
        !           160:        AR5K_HAL_FUNCTION(hal, ar5212, get_intr);
        !           161:        AR5K_HAL_FUNCTION(hal, ar5212, set_intr);
        !           162:
        !           163:        /*
        !           164:         * Chipset functions (ar5k-specific, non-HAL)
        !           165:         */
        !           166:        AR5K_HAL_FUNCTION(hal, ar5212, get_capabilities);
        !           167:        AR5K_HAL_FUNCTION(hal, ar5212, radar_alert);
        !           168:
        !           169:        /*
        !           170:         * EEPROM access
        !           171:         */
        !           172:        AR5K_HAL_FUNCTION(hal, ar5212, eeprom_is_busy);
        !           173:        AR5K_HAL_FUNCTION(hal, ar5212, eeprom_read);
        !           174:        AR5K_HAL_FUNCTION(hal, ar5212, eeprom_write);
        !           175:
        !           176:        /*
        !           177:         * Unused functions or functions not implemented
        !           178:         */
        !           179:        AR5K_HAL_FUNCTION(hal, ar5212, set_bssid_mask);
        !           180:        AR5K_HAL_FUNCTION(hal, ar5212, get_tx_queueprops);
        !           181:        AR5K_HAL_FUNCTION(hal, ar5212, num_tx_pending);
        !           182:        AR5K_HAL_FUNCTION(hal, ar5212, phy_disable);
        !           183:        AR5K_HAL_FUNCTION(hal, ar5212, set_txpower_limit);
        !           184:        AR5K_HAL_FUNCTION(hal, ar5212, set_def_antenna);
        !           185:        AR5K_HAL_FUNCTION(hal, ar5212, get_def_antenna);
        !           186: #ifdef notyet
        !           187:        AR5K_HAL_FUNCTION(hal, ar5212, set_capability);
        !           188:        AR5K_HAL_FUNCTION(hal, ar5212, proc_mib_event);
        !           189:        AR5K_HAL_FUNCTION(hal, ar5212, get_tx_inter_queue);
        !           190: #endif
        !           191: }
        !           192:
        !           193: struct ath_hal *
        !           194: ar5k_ar5212_attach(u_int16_t device, void *sc, bus_space_tag_t st,
        !           195:     bus_space_handle_t sh, int *status)
        !           196: {
        !           197:        struct ath_hal *hal = (struct ath_hal*) sc;
        !           198:        u_int8_t mac[IEEE80211_ADDR_LEN];
        !           199:        u_int32_t srev;
        !           200:
        !           201:        ar5k_ar5212_fill(hal);
        !           202:
        !           203:        /* Bring device out of sleep and reset its units */
        !           204:        if (ar5k_ar5212_nic_wakeup(hal, AR5K_INIT_MODE) != AH_TRUE)
        !           205:                return (NULL);
        !           206:
        !           207:        /* Get MAC, PHY and RADIO revisions */
        !           208:        srev = AR5K_REG_READ(AR5K_AR5212_SREV);
        !           209:        hal->ah_mac_srev = srev;
        !           210:        hal->ah_mac_version = AR5K_REG_MS(srev, AR5K_AR5212_SREV_VER);
        !           211:        hal->ah_mac_revision = AR5K_REG_MS(srev, AR5K_AR5212_SREV_REV);
        !           212:        hal->ah_phy_revision = AR5K_REG_READ(AR5K_AR5212_PHY_CHIP_ID) &
        !           213:            0x00ffffffff;
        !           214:        hal->ah_radio_5ghz_revision =
        !           215:            ar5k_ar5212_radio_revision(hal, HAL_CHIP_5GHZ);
        !           216:        hal->ah_radio_2ghz_revision =
        !           217:            ar5k_ar5212_radio_revision(hal, HAL_CHIP_2GHZ);
        !           218:
        !           219:        /* Single chip radio */
        !           220:        if (hal->ah_radio_2ghz_revision == hal->ah_radio_5ghz_revision)
        !           221:                hal->ah_radio_2ghz_revision = 0;
        !           222:
        !           223:        /* Identify the chipset (this has to be done in an early step) */
        !           224:        hal->ah_version = AR5K_AR5212;
        !           225:        hal->ah_radio = hal->ah_radio_5ghz_revision < AR5K_SREV_RAD_5112 ?
        !           226:            AR5K_AR5111 : AR5K_AR5112;
        !           227:        hal->ah_phy = AR5K_AR5212_PHY(0);
        !           228:
        !           229:        bcopy(etherbroadcastaddr, mac, IEEE80211_ADDR_LEN);
        !           230:        ar5k_ar5212_set_associd(hal, mac, 0, 0);
        !           231:        ar5k_ar5212_get_lladdr(hal, mac);
        !           232:        ar5k_ar5212_set_opmode(hal);
        !           233:
        !           234:        return (hal);
        !           235: }
        !           236:
        !           237: HAL_BOOL
        !           238: ar5k_ar5212_nic_reset(struct ath_hal *hal, u_int32_t val)
        !           239: {
        !           240:        HAL_BOOL ret = AH_FALSE;
        !           241:        u_int32_t mask = val ? val : ~0;
        !           242:
        !           243:        /* Read-and-clear */
        !           244:        AR5K_REG_READ(AR5K_AR5212_RXDP);
        !           245:
        !           246:        /*
        !           247:         * Reset the device and wait until success
        !           248:         */
        !           249:        AR5K_REG_WRITE(AR5K_AR5212_RC, val);
        !           250:
        !           251:        /* Wait at least 128 PCI clocks */
        !           252:        AR5K_DELAY(15);
        !           253:
        !           254:        val &=
        !           255:            AR5K_AR5212_RC_PCU | AR5K_AR5212_RC_BB;
        !           256:
        !           257:        mask &=
        !           258:            AR5K_AR5212_RC_PCU | AR5K_AR5212_RC_BB;
        !           259:
        !           260:        ret = ar5k_register_timeout(hal, AR5K_AR5212_RC, mask, val, AH_FALSE);
        !           261:
        !           262:        /*
        !           263:         * Reset configuration register
        !           264:         */
        !           265:        if ((val & AR5K_AR5212_RC_PCU) == 0)
        !           266:                AR5K_REG_WRITE(AR5K_AR5212_CFG, AR5K_AR5212_INIT_CFG);
        !           267:
        !           268:        return (ret);
        !           269: }
        !           270:
        !           271: HAL_BOOL
        !           272: ar5k_ar5212_nic_wakeup(struct ath_hal *hal, u_int16_t flags)
        !           273: {
        !           274:        u_int32_t turbo, mode, clock;
        !           275:
        !           276:        turbo = 0;
        !           277:        mode = 0;
        !           278:        clock = 0;
        !           279:
        !           280:        /*
        !           281:         * Get channel mode flags
        !           282:         */
        !           283:
        !           284:        if (hal->ah_radio >= AR5K_AR5112) {
        !           285:                mode = AR5K_AR5212_PHY_MODE_RAD_AR5112;
        !           286:                clock = AR5K_AR5212_PHY_PLL_AR5112;
        !           287:        } else {
        !           288:                mode = AR5K_AR5212_PHY_MODE_RAD_AR5111;
        !           289:                clock = AR5K_AR5212_PHY_PLL_AR5111;
        !           290:        }
        !           291:
        !           292:        if (flags & IEEE80211_CHAN_2GHZ) {
        !           293:                mode |= AR5K_AR5212_PHY_MODE_FREQ_2GHZ;
        !           294:                clock |= AR5K_AR5212_PHY_PLL_44MHZ;
        !           295:        } else if (flags & IEEE80211_CHAN_5GHZ) {
        !           296:                mode |= AR5K_AR5212_PHY_MODE_FREQ_5GHZ;
        !           297:                clock |= AR5K_AR5212_PHY_PLL_40MHZ;
        !           298:        } else {
        !           299:                AR5K_PRINT("invalid radio frequency mode\n");
        !           300:                return (AH_FALSE);
        !           301:        }
        !           302:
        !           303:        if (flags & IEEE80211_CHAN_CCK) {
        !           304:                mode |= AR5K_AR5212_PHY_MODE_MOD_CCK;
        !           305:        } else if (flags & IEEE80211_CHAN_OFDM) {
        !           306:                mode |= AR5K_AR5212_PHY_MODE_MOD_OFDM;
        !           307:        } else if (flags & IEEE80211_CHAN_DYN) {
        !           308:                mode |= AR5K_AR5212_PHY_MODE_MOD_DYN;
        !           309:        } else {
        !           310:                AR5K_PRINT("invalid radio frequency mode\n");
        !           311:                return (AH_FALSE);
        !           312:        }
        !           313:
        !           314:        if (flags & IEEE80211_CHAN_TURBO) {
        !           315:                turbo = AR5K_AR5212_PHY_TURBO_MODE |
        !           316:                    AR5K_AR5212_PHY_TURBO_SHORT;
        !           317:        }
        !           318:
        !           319:        /*
        !           320:         * Reset and wakeup the device
        !           321:         */
        !           322:
        !           323:        /* ...reset chipset and PCI device */
        !           324:        if (hal->ah_single_chip == AH_FALSE &&
        !           325:            ar5k_ar5212_nic_reset(hal,
        !           326:            AR5K_AR5212_RC_CHIP | AR5K_AR5212_RC_PCI) == AH_FALSE) {
        !           327:                AR5K_PRINT("failed to reset the AR5212 + PCI chipset\n");
        !           328:                return (AH_FALSE);
        !           329:        }
        !           330:
        !           331:        /* ...wakeup */
        !           332:        if (ar5k_ar5212_set_power(hal,
        !           333:                HAL_PM_AWAKE, AH_TRUE, 0) == AH_FALSE) {
        !           334:                AR5K_PRINT("failed to resume the AR5212 (again)\n");
        !           335:                return (AH_FALSE);
        !           336:        }
        !           337:
        !           338:        /* ...final warm reset */
        !           339:        if (ar5k_ar5212_nic_reset(hal, 0) == AH_FALSE) {
        !           340:                AR5K_PRINT("failed to warm reset the AR5212\n");
        !           341:                return (AH_FALSE);
        !           342:        }
        !           343:
        !           344:        /* ...set the PHY operating mode */
        !           345:        AR5K_REG_WRITE(AR5K_AR5212_PHY_PLL, clock);
        !           346:        AR5K_DELAY(300);
        !           347:
        !           348:        AR5K_REG_WRITE(AR5K_AR5212_PHY_MODE, mode);
        !           349:        AR5K_REG_WRITE(AR5K_AR5212_PHY_TURBO, turbo);
        !           350:
        !           351:        return (AH_TRUE);
        !           352: }
        !           353:
        !           354: u_int16_t
        !           355: ar5k_ar5212_radio_revision(struct ath_hal *hal, HAL_CHIP chip)
        !           356: {
        !           357:        int i;
        !           358:        u_int32_t srev;
        !           359:        u_int16_t ret;
        !           360:
        !           361:        /*
        !           362:         * Set the radio chip access register
        !           363:         */
        !           364:        switch (chip) {
        !           365:        case HAL_CHIP_2GHZ:
        !           366:                AR5K_REG_WRITE(AR5K_AR5212_PHY(0), AR5K_AR5212_PHY_SHIFT_2GHZ);
        !           367:                break;
        !           368:        case HAL_CHIP_5GHZ:
        !           369:                AR5K_REG_WRITE(AR5K_AR5212_PHY(0), AR5K_AR5212_PHY_SHIFT_5GHZ);
        !           370:                break;
        !           371:        default:
        !           372:                return (0);
        !           373:        }
        !           374:
        !           375:        AR5K_DELAY(2000);
        !           376:
        !           377:        /* ...wait until PHY is ready and read the selected radio revision */
        !           378:        AR5K_REG_WRITE(AR5K_AR5212_PHY(0x34), 0x00001c16);
        !           379:
        !           380:        for (i = 0; i < 8; i++)
        !           381:                AR5K_REG_WRITE(AR5K_AR5212_PHY(0x20), 0x00010000);
        !           382:        srev = (AR5K_REG_READ(AR5K_AR5212_PHY(0x100)) >> 24) & 0xff;
        !           383:
        !           384:        ret = ar5k_bitswap(((srev & 0xf0) >> 4) | ((srev & 0x0f) << 4), 8);
        !           385:
        !           386:        /* Reset to the 5GHz mode */
        !           387:        AR5K_REG_WRITE(AR5K_AR5212_PHY(0), AR5K_AR5212_PHY_SHIFT_5GHZ);
        !           388:
        !           389:        return (ret);
        !           390: }
        !           391:
        !           392: const HAL_RATE_TABLE *
        !           393: ar5k_ar5212_get_rate_table(struct ath_hal *hal, u_int mode)
        !           394: {
        !           395:        switch (mode) {
        !           396:        case HAL_MODE_11A:
        !           397:                return (&hal->ah_rt_11a);
        !           398:        case HAL_MODE_TURBO:
        !           399:                return (&hal->ah_rt_turbo);
        !           400:        case HAL_MODE_11B:
        !           401:                return (&hal->ah_rt_11b);
        !           402:        case HAL_MODE_11G:
        !           403:        case HAL_MODE_PUREG:
        !           404:                return (&hal->ah_rt_11g);
        !           405:        case HAL_MODE_XR:
        !           406:                return (&hal->ah_rt_xr);
        !           407:        default:
        !           408:                return (NULL);
        !           409:        }
        !           410:
        !           411:        return (NULL);
        !           412: }
        !           413:
        !           414: void
        !           415: ar5k_ar5212_detach(struct ath_hal *hal)
        !           416: {
        !           417:        if (hal->ah_rf_banks != NULL)
        !           418:                free(hal->ah_rf_banks, M_DEVBUF);
        !           419:
        !           420:        /*
        !           421:         * Free HAL structure, assume interrupts are down
        !           422:         */
        !           423:        free(hal, M_DEVBUF);
        !           424: }
        !           425:
        !           426: HAL_BOOL
        !           427: ar5k_ar5212_phy_disable(struct ath_hal *hal)
        !           428: {
        !           429:        AR5K_REG_WRITE(AR5K_AR5212_PHY_ACTIVE, AR5K_AR5212_PHY_DISABLE);
        !           430:        return (AH_TRUE);
        !           431: }
        !           432:
        !           433: HAL_BOOL
        !           434: ar5k_ar5212_reset(struct ath_hal *hal, HAL_OPMODE op_mode, HAL_CHANNEL *channel,
        !           435:     HAL_BOOL change_channel, HAL_STATUS *status)
        !           436: {
        !           437:        struct ar5k_eeprom_info *ee = &hal->ah_capabilities.cap_eeprom;
        !           438:        u_int8_t mac[IEEE80211_ADDR_LEN];
        !           439:        u_int32_t data, s_seq, s_ant, s_led[3];
        !           440:        u_int i, phy, mode, freq, off, ee_mode, ant[2];
        !           441:        const HAL_RATE_TABLE *rt;
        !           442:
        !           443:        /*
        !           444:         * Save some registers before a reset
        !           445:         */
        !           446:        if (change_channel == AH_TRUE) {
        !           447:                s_seq = AR5K_REG_READ(AR5K_AR5212_DCU_SEQNUM(0));
        !           448:                s_ant = AR5K_REG_READ(AR5K_AR5212_DEFAULT_ANTENNA);
        !           449:        } else {
        !           450:                s_seq = 0;
        !           451:                s_ant = 1;
        !           452:        }
        !           453:
        !           454:        s_led[0] = AR5K_REG_READ(AR5K_AR5212_PCICFG) &
        !           455:            AR5K_AR5212_PCICFG_LEDSTATE;
        !           456:        s_led[1] = AR5K_REG_READ(AR5K_AR5212_GPIOCR);
        !           457:        s_led[2] = AR5K_REG_READ(AR5K_AR5212_GPIODO);
        !           458:
        !           459:        if (change_channel == AH_TRUE && hal->ah_rf_banks != NULL)
        !           460:                ar5k_ar5212_get_rf_gain(hal);
        !           461:
        !           462:        if (ar5k_ar5212_nic_wakeup(hal, channel->c_channel_flags) == AH_FALSE)
        !           463:                return (AH_FALSE);
        !           464:
        !           465:        /*
        !           466:         * Initialize operating mode
        !           467:         */
        !           468:        hal->ah_op_mode = op_mode;
        !           469:
        !           470:        if (hal->ah_radio == AR5K_AR5111) {
        !           471:                phy = AR5K_INI_PHY_5111;
        !           472:        } else if (hal->ah_radio == AR5K_AR5112) {
        !           473:                phy = AR5K_INI_PHY_5112;
        !           474:        } else {
        !           475:                AR5K_PRINTF("invalid phy radio: %u\n", hal->ah_radio);
        !           476:                return (AH_FALSE);
        !           477:        }
        !           478:
        !           479:        switch (channel->c_channel_flags & CHANNEL_MODES) {
        !           480:        case CHANNEL_A:
        !           481:                mode = AR5K_INI_VAL_11A;
        !           482:                freq = AR5K_INI_RFGAIN_5GHZ;
        !           483:                ee_mode = AR5K_EEPROM_MODE_11A;
        !           484:                break;
        !           485:        case CHANNEL_B:
        !           486:                mode = AR5K_INI_VAL_11B;
        !           487:                freq = AR5K_INI_RFGAIN_2GHZ;
        !           488:                ee_mode = AR5K_EEPROM_MODE_11B;
        !           489:                break;
        !           490:        case CHANNEL_G:
        !           491:        case CHANNEL_PUREG:
        !           492:                mode = AR5K_INI_VAL_11G;
        !           493:                freq = AR5K_INI_RFGAIN_2GHZ;
        !           494:                ee_mode = AR5K_EEPROM_MODE_11G;
        !           495:                break;
        !           496:        case CHANNEL_T:
        !           497:                mode = AR5K_INI_VAL_11A_TURBO;
        !           498:                freq = AR5K_INI_RFGAIN_5GHZ;
        !           499:                ee_mode = AR5K_EEPROM_MODE_11A;
        !           500:                break;
        !           501:        case CHANNEL_TG:
        !           502:                mode = AR5K_INI_VAL_11G_TURBO;
        !           503:                freq = AR5K_INI_RFGAIN_2GHZ;
        !           504:                ee_mode = AR5K_EEPROM_MODE_11G;
        !           505:                break;
        !           506:        case CHANNEL_XR:
        !           507:                mode = AR5K_INI_VAL_XR;
        !           508:                freq = AR5K_INI_RFGAIN_5GHZ;
        !           509:                ee_mode = AR5K_EEPROM_MODE_11A;
        !           510:                break;
        !           511:        default:
        !           512:                AR5K_PRINTF("invalid channel: %d\n", channel->c_channel);
        !           513:                return (AH_FALSE);
        !           514:        }
        !           515:
        !           516:        /* PHY access enable */
        !           517:        AR5K_REG_WRITE(AR5K_AR5212_PHY(0), AR5K_AR5212_PHY_SHIFT_5GHZ);
        !           518:
        !           519:        /*
        !           520:         * Write initial mode settings
        !           521:         */
        !           522:        for (i = 0; i < AR5K_ELEMENTS(ar5212_mode); i++) {
        !           523:                if (ar5212_mode[i].mode_flags == AR5K_INI_FLAG_511X)
        !           524:                        off = AR5K_INI_PHY_511X;
        !           525:                else if (ar5212_mode[i].mode_flags & AR5K_INI_FLAG_5111 &&
        !           526:                    hal->ah_radio == AR5K_AR5111)
        !           527:                        off = AR5K_INI_PHY_5111;
        !           528:                else if (ar5212_mode[i].mode_flags & AR5K_INI_FLAG_5112 &&
        !           529:                    hal->ah_radio == AR5K_AR5112)
        !           530:                        off = AR5K_INI_PHY_5112;
        !           531:                else
        !           532:                        continue;
        !           533:
        !           534:                AR5K_REG_WAIT(i);
        !           535:                AR5K_REG_WRITE((u_int32_t)ar5212_mode[i].mode_register,
        !           536:                    ar5212_mode[i].mode_value[off][mode]);
        !           537:        }
        !           538:
        !           539:        /*
        !           540:         * Write initial register settings
        !           541:         */
        !           542:        for (i = 0; i < AR5K_ELEMENTS(ar5212_ini); i++) {
        !           543:                if (change_channel == AH_TRUE &&
        !           544:                    ar5212_ini[i].ini_register >= AR5K_AR5212_PCU_MIN &&
        !           545:                    ar5212_ini[i].ini_register <= AR5K_AR5212_PCU_MAX)
        !           546:                        continue;
        !           547:
        !           548:                if ((hal->ah_radio == AR5K_AR5111 &&
        !           549:                    ar5212_ini[i].ini_flags & AR5K_INI_FLAG_5111) ||
        !           550:                    (hal->ah_radio == AR5K_AR5112 &&
        !           551:                    ar5212_ini[i].ini_flags & AR5K_INI_FLAG_5112)) {
        !           552:                        AR5K_REG_WAIT(i);
        !           553:                        AR5K_REG_WRITE((u_int32_t)ar5212_ini[i].ini_register,
        !           554:                            ar5212_ini[i].ini_value);
        !           555:                }
        !           556:        }
        !           557:
        !           558:        /*
        !           559:         * Write initial RF gain settings
        !           560:         */
        !           561:        if (ar5k_rfgain(hal, phy, freq) == AH_FALSE)
        !           562:                return (AH_FALSE);
        !           563:
        !           564:        AR5K_DELAY(1000);
        !           565:
        !           566:        /*
        !           567:         * Set rate duration table
        !           568:         */
        !           569:        rt = ar5k_ar5212_get_rate_table(hal,
        !           570:            channel->c_channel_flags & IEEE80211_CHAN_TURBO ?
        !           571:            HAL_MODE_TURBO : HAL_MODE_XR);
        !           572:
        !           573:        for (i = 0; i < rt->rt_rate_count; i++) {
        !           574:                AR5K_REG_WRITE(AR5K_AR5212_RATE_DUR(rt->rt_info[i].r_rate_code),
        !           575:                    ath_hal_computetxtime(hal, rt, 14,
        !           576:                    rt->rt_info[i].r_control_rate, AH_FALSE));
        !           577:        }
        !           578:
        !           579:        if ((channel->c_channel_flags & IEEE80211_CHAN_TURBO) == 0) {
        !           580:                rt = ar5k_ar5212_get_rate_table(hal, HAL_MODE_11B);
        !           581:                for (i = 0; i < rt->rt_rate_count; i++) {
        !           582:                        data = AR5K_AR5212_RATE_DUR(rt->rt_info[i].r_rate_code);
        !           583:                        AR5K_REG_WRITE(data,
        !           584:                            ath_hal_computetxtime(hal, rt, 14,
        !           585:                            rt->rt_info[i].r_control_rate, AH_FALSE));
        !           586:                        if (rt->rt_info[i].r_short_preamble) {
        !           587:                                AR5K_REG_WRITE(data +
        !           588:                                    (rt->rt_info[i].r_short_preamble << 2),
        !           589:                                    ath_hal_computetxtime(hal, rt, 14,
        !           590:                                    rt->rt_info[i].r_control_rate, AH_FALSE));
        !           591:                        }
        !           592:                }
        !           593:        }
        !           594:
        !           595:        /* Fix for first revision of the AR5112 RF chipset */
        !           596:        if (hal->ah_radio >= AR5K_AR5112 &&
        !           597:            hal->ah_radio_5ghz_revision < AR5K_SREV_RAD_5112A) {
        !           598:                AR5K_REG_WRITE(AR5K_AR5212_PHY_CCKTXCTL,
        !           599:                    AR5K_AR5212_PHY_CCKTXCTL_WORLD);
        !           600:                if (channel->c_channel_flags & IEEE80211_CHAN_OFDM)
        !           601:                        data = 0xffb81020;
        !           602:                else
        !           603:                        data = 0xffb80d20;
        !           604:                AR5K_REG_WRITE(AR5K_AR5212_PHY_FC, data);
        !           605:        }
        !           606:
        !           607:        /*
        !           608:         * Set TX power (XXX use txpower from net80211)
        !           609:         */
        !           610:        if (ar5k_ar5212_txpower(hal, channel,
        !           611:                AR5K_TUNE_DEFAULT_TXPOWER) == AH_FALSE)
        !           612:                return (AH_FALSE);
        !           613:
        !           614:        /*
        !           615:         * Write RF registers
        !           616:         */
        !           617:        if (ar5k_rfregs(hal, channel, mode) == AH_FALSE)
        !           618:                return (AH_FALSE);
        !           619:
        !           620:        /*
        !           621:         * Configure additional registers
        !           622:         */
        !           623:
        !           624:        /* OFDM timings */
        !           625:        if (channel->c_channel_flags & IEEE80211_CHAN_OFDM) {
        !           626:                u_int32_t coef_scaled, coef_exp, coef_man, ds_coef_exp,
        !           627:                    ds_coef_man, clock;
        !           628:
        !           629:                clock = channel->c_channel_flags & IEEE80211_CHAN_T ? 80 : 40;
        !           630:                coef_scaled = ((5 * (clock << 24)) / 2) / channel->c_channel;
        !           631:
        !           632:                for (coef_exp = 31; coef_exp > 0; coef_exp--)
        !           633:                        if ((coef_scaled >> coef_exp) & 0x1)
        !           634:                                break;
        !           635:
        !           636:                if (!coef_exp)
        !           637:                        return (AH_FALSE);
        !           638:
        !           639:                coef_exp = 14 - (coef_exp - 24);
        !           640:                coef_man = coef_scaled + (1 << (24 - coef_exp - 1));
        !           641:                ds_coef_man = coef_man >> (24 - coef_exp);
        !           642:                ds_coef_exp = coef_exp - 16;
        !           643:
        !           644:                AR5K_REG_WRITE_BITS(AR5K_AR5212_PHY_TIMING_3,
        !           645:                    AR5K_AR5212_PHY_TIMING_3_DSC_MAN, ds_coef_man);
        !           646:                AR5K_REG_WRITE_BITS(AR5K_AR5212_PHY_TIMING_3,
        !           647:                    AR5K_AR5212_PHY_TIMING_3_DSC_EXP, ds_coef_exp);
        !           648:        }
        !           649:
        !           650:        if (hal->ah_radio == AR5K_AR5111) {
        !           651:                if (channel->c_channel_flags & IEEE80211_CHAN_B)
        !           652:                        AR5K_REG_ENABLE_BITS(AR5K_AR5212_TXCFG,
        !           653:                            AR5K_AR5212_TXCFG_B_MODE);
        !           654:                else
        !           655:                        AR5K_REG_DISABLE_BITS(AR5K_AR5212_TXCFG,
        !           656:                            AR5K_AR5212_TXCFG_B_MODE);
        !           657:        }
        !           658:
        !           659:        /* Set antenna mode */
        !           660:        AR5K_REG_MASKED_BITS(AR5K_AR5212_PHY(0x44),
        !           661:            hal->ah_antenna[ee_mode][0], 0xfffffc06);
        !           662:
        !           663:        if (freq == AR5K_INI_RFGAIN_2GHZ)
        !           664:                ant[0] = ant[1] = HAL_ANT_FIXED_B;
        !           665:        else
        !           666:                ant[0] = ant[1] = HAL_ANT_FIXED_A;
        !           667:
        !           668:        AR5K_REG_WRITE(AR5K_AR5212_PHY_ANT_SWITCH_TABLE_0,
        !           669:            hal->ah_antenna[ee_mode][ant[0]]);
        !           670:        AR5K_REG_WRITE(AR5K_AR5212_PHY_ANT_SWITCH_TABLE_1,
        !           671:            hal->ah_antenna[ee_mode][ant[1]]);
        !           672:
        !           673:        /* Commit values from EEPROM */
        !           674:        if (hal->ah_radio == AR5K_AR5111)
        !           675:                AR5K_REG_WRITE_BITS(AR5K_AR5212_PHY_FC,
        !           676:                    AR5K_AR5212_PHY_FC_TX_CLIP, ee->ee_tx_clip);
        !           677:
        !           678:        AR5K_REG_WRITE(AR5K_AR5212_PHY(0x5a),
        !           679:            AR5K_AR5212_PHY_NF_SVAL(ee->ee_noise_floor_thr[ee_mode]));
        !           680:
        !           681:        AR5K_REG_MASKED_BITS(AR5K_AR5212_PHY(0x11),
        !           682:            (ee->ee_switch_settling[ee_mode] << 7) & 0x3f80, 0xffffc07f);
        !           683:        AR5K_REG_MASKED_BITS(AR5K_AR5212_PHY(0x12),
        !           684:            (ee->ee_ant_tx_rx[ee_mode] << 12) & 0x3f000, 0xfffc0fff);
        !           685:        AR5K_REG_MASKED_BITS(AR5K_AR5212_PHY(0x14),
        !           686:            (ee->ee_adc_desired_size[ee_mode] & 0x00ff) |
        !           687:            ((ee->ee_pga_desired_size[ee_mode] << 8) & 0xff00), 0xffff0000);
        !           688:
        !           689:        AR5K_REG_WRITE(AR5K_AR5212_PHY(0x0d),
        !           690:            (ee->ee_tx_end2xpa_disable[ee_mode] << 24) |
        !           691:            (ee->ee_tx_end2xpa_disable[ee_mode] << 16) |
        !           692:            (ee->ee_tx_frm2xpa_enable[ee_mode] << 8) |
        !           693:            (ee->ee_tx_frm2xpa_enable[ee_mode]));
        !           694:
        !           695:        AR5K_REG_MASKED_BITS(AR5K_AR5212_PHY(0x0a),
        !           696:            ee->ee_tx_end2xlna_enable[ee_mode] << 8, 0xffff00ff);
        !           697:        AR5K_REG_MASKED_BITS(AR5K_AR5212_PHY(0x19),
        !           698:            (ee->ee_thr_62[ee_mode] << 12) & 0x7f000, 0xfff80fff);
        !           699:        AR5K_REG_MASKED_BITS(AR5K_AR5212_PHY(0x49), 4, 0xffffff01);
        !           700:
        !           701:        AR5K_REG_ENABLE_BITS(AR5K_AR5212_PHY_IQ,
        !           702:            AR5K_AR5212_PHY_IQ_CORR_ENABLE |
        !           703:            (ee->ee_i_cal[ee_mode] << AR5K_AR5212_PHY_IQ_CORR_Q_I_COFF_S) |
        !           704:            ee->ee_q_cal[ee_mode]);
        !           705:
        !           706:        if (hal->ah_ee_version >= AR5K_EEPROM_VERSION_4_1) {
        !           707:                AR5K_REG_WRITE_BITS(AR5K_AR5212_PHY_GAIN_2GHZ,
        !           708:                    AR5K_AR5212_PHY_GAIN_2GHZ_MARGIN_TXRX,
        !           709:                    ee->ee_margin_tx_rx[ee_mode]);
        !           710:        }
        !           711:
        !           712:        /*
        !           713:         * Restore saved values
        !           714:         */
        !           715:        AR5K_REG_WRITE(AR5K_AR5212_DCU_SEQNUM(0), s_seq);
        !           716:        AR5K_REG_WRITE(AR5K_AR5212_DEFAULT_ANTENNA, s_ant);
        !           717:        AR5K_REG_ENABLE_BITS(AR5K_AR5212_PCICFG, s_led[0]);
        !           718:        AR5K_REG_WRITE(AR5K_AR5212_GPIOCR, s_led[1]);
        !           719:        AR5K_REG_WRITE(AR5K_AR5212_GPIODO, s_led[2]);
        !           720:
        !           721:        /*
        !           722:         * Misc
        !           723:         */
        !           724:        bcopy(etherbroadcastaddr, mac, IEEE80211_ADDR_LEN);
        !           725:        ar5k_ar5212_set_associd(hal, mac, 0, 0);
        !           726:        ar5k_ar5212_set_opmode(hal);
        !           727:        AR5K_REG_WRITE(AR5K_AR5212_PISR, 0xffffffff);
        !           728:        AR5K_REG_WRITE(AR5K_AR5212_RSSI_THR, AR5K_TUNE_RSSI_THRES);
        !           729:
        !           730:        /*
        !           731:         * Set Rx/Tx DMA Configuration
        !           732:         */
        !           733:        AR5K_REG_WRITE_BITS(AR5K_AR5212_TXCFG, AR5K_AR5212_TXCFG_SDMAMR,
        !           734:            AR5K_AR5212_DMASIZE_512B | AR5K_AR5212_TXCFG_DMASIZE);
        !           735:        AR5K_REG_WRITE_BITS(AR5K_AR5212_RXCFG, AR5K_AR5212_RXCFG_SDMAMW,
        !           736:            AR5K_AR5212_DMASIZE_512B);
        !           737:
        !           738:        /*
        !           739:         * Set channel and calibrate the PHY
        !           740:         */
        !           741:        if (ar5k_channel(hal, channel) == AH_FALSE)
        !           742:                return (AH_FALSE);
        !           743:
        !           744:        /*
        !           745:         * Enable the PHY and wait until completion
        !           746:         */
        !           747:        AR5K_REG_WRITE(AR5K_AR5212_PHY_ACTIVE, AR5K_AR5212_PHY_ENABLE);
        !           748:
        !           749:        data = AR5K_REG_READ(AR5K_AR5212_PHY_RX_DELAY) &
        !           750:            AR5K_AR5212_PHY_RX_DELAY_M;
        !           751:        data = (channel->c_channel_flags & IEEE80211_CHAN_CCK) ?
        !           752:            ((data << 2) / 22) : (data / 10);
        !           753:
        !           754:        AR5K_DELAY(100 + data);
        !           755:
        !           756:        /*
        !           757:         * Start calibration
        !           758:         */
        !           759:        AR5K_REG_ENABLE_BITS(AR5K_AR5212_PHY_AGCCTL,
        !           760:            AR5K_AR5212_PHY_AGCCTL_NF |
        !           761:            AR5K_AR5212_PHY_AGCCTL_CAL);
        !           762:
        !           763:        hal->ah_calibration = AH_FALSE;
        !           764:        if ((channel->c_channel_flags & IEEE80211_CHAN_B) == 0) {
        !           765:                hal->ah_calibration = AH_TRUE;
        !           766:                AR5K_REG_WRITE_BITS(AR5K_AR5212_PHY_IQ,
        !           767:                    AR5K_AR5212_PHY_IQ_CAL_NUM_LOG_MAX, 15);
        !           768:                AR5K_REG_ENABLE_BITS(AR5K_AR5212_PHY_IQ,
        !           769:                    AR5K_AR5212_PHY_IQ_RUN);
        !           770:        }
        !           771:
        !           772:        /*
        !           773:         * Reset queues and start beacon timers at the end of the reset routine
        !           774:         */
        !           775:        for (i = 0; i < hal->ah_capabilities.cap_queues.q_tx_num; i++) {
        !           776:                AR5K_REG_WRITE_Q(AR5K_AR5212_DCU_QCUMASK(i), i);
        !           777:                if (ar5k_ar5212_reset_tx_queue(hal, i) == AH_FALSE) {
        !           778:                        AR5K_PRINTF("failed to reset TX queue #%d\n", i);
        !           779:                        return (AH_FALSE);
        !           780:                }
        !           781:        }
        !           782:
        !           783:        /* Pre-enable interrupts */
        !           784:        ar5k_ar5212_set_intr(hal, HAL_INT_RX | HAL_INT_TX | HAL_INT_FATAL);
        !           785:
        !           786:        /*
        !           787:         * Set RF kill flags if supported by the device (read from the EEPROM)
        !           788:         */
        !           789:        if (AR5K_EEPROM_HDR_RFKILL(hal->ah_capabilities.cap_eeprom.ee_header)) {
        !           790:                ar5k_ar5212_set_gpio_input(hal, 0);
        !           791:                if ((hal->ah_gpio[0] = ar5k_ar5212_get_gpio(hal, 0)) == 0)
        !           792:                        ar5k_ar5212_set_gpio_intr(hal, 0, 1);
        !           793:                else
        !           794:                        ar5k_ar5212_set_gpio_intr(hal, 0, 0);
        !           795:        }
        !           796:
        !           797:        /*
        !           798:         * Set the 32MHz reference clock
        !           799:         */
        !           800:        AR5K_REG_WRITE(AR5K_AR5212_PHY_SCR, AR5K_AR5212_PHY_SCR_32MHZ);
        !           801:        AR5K_REG_WRITE(AR5K_AR5212_PHY_SLMT, AR5K_AR5212_PHY_SLMT_32MHZ);
        !           802:        AR5K_REG_WRITE(AR5K_AR5212_PHY_SCAL, AR5K_AR5212_PHY_SCAL_32MHZ);
        !           803:        AR5K_REG_WRITE(AR5K_AR5212_PHY_SCLOCK, AR5K_AR5212_PHY_SCLOCK_32MHZ);
        !           804:        AR5K_REG_WRITE(AR5K_AR5212_PHY_SDELAY, AR5K_AR5212_PHY_SDELAY_32MHZ);
        !           805:        AR5K_REG_WRITE(AR5K_AR5212_PHY_SPENDING, hal->ah_radio == AR5K_AR5111 ?
        !           806:            AR5K_AR5212_PHY_SPENDING_AR5111 : AR5K_AR5212_PHY_SPENDING_AR5112);
        !           807:
        !           808:        /*
        !           809:         * Disable beacons and reset the register
        !           810:         */
        !           811:        AR5K_REG_DISABLE_BITS(AR5K_AR5212_BEACON,
        !           812:            AR5K_AR5212_BEACON_ENABLE | AR5K_AR5212_BEACON_RESET_TSF);
        !           813:
        !           814:        return (AH_TRUE);
        !           815: }
        !           816:
        !           817: void
        !           818: ar5k_ar5212_set_def_antenna(struct ath_hal *hal, u_int ant)
        !           819: {
        !           820:        AR5K_REG_WRITE(AR5K_AR5212_DEFAULT_ANTENNA, ant);
        !           821: }
        !           822:
        !           823: u_int
        !           824: ar5k_ar5212_get_def_antenna(struct ath_hal *hal)
        !           825: {
        !           826:        return AR5K_REG_READ(AR5K_AR5212_DEFAULT_ANTENNA);
        !           827: }
        !           828:
        !           829: void
        !           830: ar5k_ar5212_set_opmode(struct ath_hal *hal)
        !           831: {
        !           832:        u_int32_t pcu_reg, low_id, high_id;
        !           833:
        !           834:        pcu_reg = 0;
        !           835:
        !           836:        switch (hal->ah_op_mode) {
        !           837:        case IEEE80211_M_IBSS:
        !           838:                pcu_reg |= AR5K_AR5212_STA_ID1_ADHOC |
        !           839:                    AR5K_AR5212_STA_ID1_DESC_ANTENNA;
        !           840:                break;
        !           841:
        !           842:        case IEEE80211_M_HOSTAP:
        !           843:                pcu_reg |= AR5K_AR5212_STA_ID1_AP |
        !           844:                    AR5K_AR5212_STA_ID1_RTS_DEFAULT_ANTENNA;
        !           845:                break;
        !           846:
        !           847:        case IEEE80211_M_STA:
        !           848:        case IEEE80211_M_MONITOR:
        !           849:                pcu_reg |= AR5K_AR5212_STA_ID1_DEFAULT_ANTENNA;
        !           850:                break;
        !           851:
        !           852:        default:
        !           853:                return;
        !           854:        }
        !           855:
        !           856:        /*
        !           857:         * Set PCU registers
        !           858:         */
        !           859:        low_id = AR5K_LOW_ID(hal->ah_sta_id);
        !           860:        high_id = AR5K_HIGH_ID(hal->ah_sta_id);
        !           861:        AR5K_REG_WRITE(AR5K_AR5212_STA_ID0, low_id);
        !           862:        AR5K_REG_WRITE(AR5K_AR5212_STA_ID1, pcu_reg | high_id);
        !           863:
        !           864:        return;
        !           865: }
        !           866:
        !           867: HAL_BOOL
        !           868: ar5k_ar5212_calibrate(struct ath_hal *hal, HAL_CHANNEL *channel)
        !           869: {
        !           870:        u_int32_t i_pwr, q_pwr;
        !           871:        int32_t iq_corr, i_coff, i_coffd, q_coff, q_coffd;
        !           872:
        !           873:        if (hal->ah_calibration == AH_FALSE ||
        !           874:            AR5K_REG_READ(AR5K_AR5212_PHY_IQ) & AR5K_AR5212_PHY_IQ_RUN)
        !           875:                goto done;
        !           876:
        !           877:        hal->ah_calibration = AH_FALSE;
        !           878:
        !           879:        iq_corr = AR5K_REG_READ(AR5K_AR5212_PHY_IQRES_CAL_CORR);
        !           880:        i_pwr = AR5K_REG_READ(AR5K_AR5212_PHY_IQRES_CAL_PWR_I);
        !           881:        q_pwr = AR5K_REG_READ(AR5K_AR5212_PHY_IQRES_CAL_PWR_Q);
        !           882:        i_coffd = ((i_pwr >> 1) + (q_pwr >> 1)) >> 7;
        !           883:        q_coffd = q_pwr >> 6;
        !           884:
        !           885:        if (i_coffd == 0 || q_coffd == 0)
        !           886:                goto done;
        !           887:
        !           888:        i_coff = ((-iq_corr) / i_coffd) & 0x3f;
        !           889:        q_coff = (((int32_t)i_pwr / q_coffd) - 64) & 0x1f;
        !           890:
        !           891:        /* Commit new IQ value */
        !           892:        AR5K_REG_ENABLE_BITS(AR5K_AR5212_PHY_IQ,
        !           893:            AR5K_AR5212_PHY_IQ_CORR_ENABLE |
        !           894:            ((u_int32_t)q_coff) |
        !           895:            ((u_int32_t)i_coff << AR5K_AR5212_PHY_IQ_CORR_Q_I_COFF_S));
        !           896:
        !           897:  done:
        !           898:        /* Start noise floor calibration */
        !           899:        AR5K_REG_ENABLE_BITS(AR5K_AR5212_PHY_AGCCTL,
        !           900:            AR5K_AR5212_PHY_AGCCTL_NF);
        !           901:
        !           902:        /* Request RF gain */
        !           903:        if (channel->c_channel_flags & IEEE80211_CHAN_5GHZ) {
        !           904:                AR5K_REG_WRITE(AR5K_AR5212_PHY_PAPD_PROBE,
        !           905:                    AR5K_REG_SM(hal->ah_txpower.txp_max,
        !           906:                    AR5K_AR5212_PHY_PAPD_PROBE_TXPOWER) |
        !           907:                    AR5K_AR5212_PHY_PAPD_PROBE_TX_NEXT);
        !           908:                hal->ah_rf_gain = HAL_RFGAIN_READ_REQUESTED;
        !           909:        }
        !           910:
        !           911:        return (AH_TRUE);
        !           912: }
        !           913:
        !           914: /*
        !           915:  * Transmit functions
        !           916:  */
        !           917:
        !           918: HAL_BOOL
        !           919: ar5k_ar5212_update_tx_triglevel(struct ath_hal *hal, HAL_BOOL increase)
        !           920: {
        !           921:        u_int32_t trigger_level, imr;
        !           922:        HAL_BOOL status = AH_FALSE;
        !           923:
        !           924:        /*
        !           925:         * Disable interrupts by setting the mask
        !           926:         */
        !           927:        imr = ar5k_ar5212_set_intr(hal, hal->ah_imr & ~HAL_INT_GLOBAL);
        !           928:
        !           929:        trigger_level = AR5K_REG_MS(AR5K_REG_READ(AR5K_AR5212_TXCFG),
        !           930:            AR5K_AR5212_TXCFG_TXFULL);
        !           931:
        !           932:        if (increase == AH_FALSE) {
        !           933:                if (--trigger_level < AR5K_TUNE_MIN_TX_FIFO_THRES)
        !           934:                        goto done;
        !           935:        } else
        !           936:                trigger_level +=
        !           937:                    ((AR5K_TUNE_MAX_TX_FIFO_THRES - trigger_level) / 2);
        !           938:
        !           939:        /*
        !           940:         * Update trigger level on success
        !           941:         */
        !           942:        AR5K_REG_WRITE_BITS(AR5K_AR5212_TXCFG,
        !           943:            AR5K_AR5212_TXCFG_TXFULL, trigger_level);
        !           944:        status = AH_TRUE;
        !           945:
        !           946:  done:
        !           947:        /*
        !           948:         * Restore interrupt mask
        !           949:         */
        !           950:        ar5k_ar5212_set_intr(hal, imr);
        !           951:
        !           952:        return (status);
        !           953: }
        !           954:
        !           955: int
        !           956: ar5k_ar5212_setup_tx_queue(struct ath_hal *hal, HAL_TX_QUEUE queue_type,
        !           957:     const HAL_TXQ_INFO *queue_info)
        !           958: {
        !           959:        u_int queue;
        !           960:
        !           961:        /*
        !           962:         * Get queue by type
        !           963:         */
        !           964:        if (queue_type == HAL_TX_QUEUE_DATA) {
        !           965:                for (queue = HAL_TX_QUEUE_ID_DATA_MIN;
        !           966:                     hal->ah_txq[queue].tqi_type != HAL_TX_QUEUE_INACTIVE;
        !           967:                     queue++)
        !           968:                        if (queue > HAL_TX_QUEUE_ID_DATA_MAX)
        !           969:                                return (-1);
        !           970:        } else if (queue_type == HAL_TX_QUEUE_PSPOLL) {
        !           971:                queue = HAL_TX_QUEUE_ID_PSPOLL;
        !           972:        } else if (queue_type == HAL_TX_QUEUE_BEACON) {
        !           973:                queue = HAL_TX_QUEUE_ID_BEACON;
        !           974:        } else if (queue_type == HAL_TX_QUEUE_CAB) {
        !           975:                queue = HAL_TX_QUEUE_ID_CAB;
        !           976:        } else
        !           977:                return (-1);
        !           978:
        !           979:        /*
        !           980:         * Setup internal queue structure
        !           981:         */
        !           982:        bzero(&hal->ah_txq[queue], sizeof(HAL_TXQ_INFO));
        !           983:        if (queue_info != NULL) {
        !           984:                if (ar5k_ar5212_setup_tx_queueprops(hal, queue, queue_info)
        !           985:                    != AH_TRUE)
        !           986:                        return (-1);
        !           987:        }
        !           988:        hal->ah_txq[queue].tqi_type = queue_type;
        !           989:
        !           990:        AR5K_Q_ENABLE_BITS(hal->ah_txq_interrupts, queue);
        !           991:
        !           992:        return (queue);
        !           993: }
        !           994:
        !           995: HAL_BOOL
        !           996: ar5k_ar5212_setup_tx_queueprops(struct ath_hal *hal, int queue,
        !           997:     const HAL_TXQ_INFO *queue_info)
        !           998: {
        !           999:        AR5K_ASSERT_ENTRY(queue, hal->ah_capabilities.cap_queues.q_tx_num);
        !          1000:
        !          1001:        if (hal->ah_txq[queue].tqi_type != HAL_TX_QUEUE_INACTIVE)
        !          1002:                return (AH_FALSE);
        !          1003:
        !          1004:        bcopy(queue_info, &hal->ah_txq[queue], sizeof(HAL_TXQ_INFO));
        !          1005:
        !          1006:        if (queue_info->tqi_type == HAL_TX_QUEUE_DATA &&
        !          1007:            (queue_info->tqi_subtype >= HAL_WME_AC_VI) &&
        !          1008:            (queue_info->tqi_subtype <= HAL_WME_UPSD))
        !          1009:                hal->ah_txq[queue].tqi_flags |=
        !          1010:                    AR5K_TXQ_FLAG_POST_FR_BKOFF_DIS;
        !          1011:
        !          1012:        return (AH_TRUE);
        !          1013: }
        !          1014:
        !          1015: HAL_BOOL
        !          1016: ar5k_ar5212_get_tx_queueprops(struct ath_hal *hal, int queue,
        !          1017:     HAL_TXQ_INFO *queue_info)
        !          1018: {
        !          1019:        AR5K_ASSERT_ENTRY(queue, hal->ah_capabilities.cap_queues.q_tx_num);
        !          1020:        bcopy(&hal->ah_txq[queue], queue_info, sizeof(HAL_TXQ_INFO));
        !          1021:        return (AH_TRUE);
        !          1022: }
        !          1023:
        !          1024: HAL_BOOL
        !          1025: ar5k_ar5212_release_tx_queue(struct ath_hal *hal, u_int queue)
        !          1026: {
        !          1027:        AR5K_ASSERT_ENTRY(queue, hal->ah_capabilities.cap_queues.q_tx_num);
        !          1028:
        !          1029:        /* This queue will be skipped in further operations */
        !          1030:        hal->ah_txq[queue].tqi_type = HAL_TX_QUEUE_INACTIVE;
        !          1031:        AR5K_Q_DISABLE_BITS(hal->ah_txq_interrupts, queue);
        !          1032:
        !          1033:        return (AH_FALSE);
        !          1034: }
        !          1035:
        !          1036: HAL_BOOL
        !          1037: ar5k_ar5212_reset_tx_queue(struct ath_hal *hal, u_int queue)
        !          1038: {
        !          1039:        u_int32_t cw_min, cw_max, retry_lg, retry_sh;
        !          1040:        struct ieee80211_channel *channel = (struct ieee80211_channel*)
        !          1041:            &hal->ah_current_channel;
        !          1042:        HAL_TXQ_INFO *tq;
        !          1043:
        !          1044:        AR5K_ASSERT_ENTRY(queue, hal->ah_capabilities.cap_queues.q_tx_num);
        !          1045:
        !          1046:        tq = &hal->ah_txq[queue];
        !          1047:
        !          1048:        if (tq->tqi_type == HAL_TX_QUEUE_INACTIVE)
        !          1049:                return (AH_TRUE);
        !          1050:
        !          1051:        /*
        !          1052:         * Set registers by channel mode
        !          1053:         */
        !          1054:        cw_min = hal->ah_cw_min = AR5K_TUNE_CWMIN;
        !          1055:        cw_max = hal->ah_cw_max = AR5K_TUNE_CWMAX;
        !          1056:        hal->ah_aifs = AR5K_TUNE_AIFS;
        !          1057:        if (IEEE80211_IS_CHAN_XR(channel)) {
        !          1058:                cw_min = hal->ah_cw_min = AR5K_TUNE_CWMIN_XR;
        !          1059:                cw_max = hal->ah_cw_max = AR5K_TUNE_CWMAX_XR;
        !          1060:                hal->ah_aifs = AR5K_TUNE_AIFS_XR;
        !          1061:        } else if (IEEE80211_IS_CHAN_B(channel)) {
        !          1062:                cw_min = hal->ah_cw_min = AR5K_TUNE_CWMIN_11B;
        !          1063:                cw_max = hal->ah_cw_max = AR5K_TUNE_CWMAX_11B;
        !          1064:                hal->ah_aifs = AR5K_TUNE_AIFS_11B;
        !          1065:        }
        !          1066:
        !          1067:        /*
        !          1068:         * Set retry limits
        !          1069:         */
        !          1070:        if (hal->ah_software_retry == AH_TRUE) {
        !          1071:                /* XXX Need to test this */
        !          1072:                retry_lg = hal->ah_limit_tx_retries;
        !          1073:                retry_sh = retry_lg =
        !          1074:                    retry_lg > AR5K_AR5212_DCU_RETRY_LMT_SH_RETRY ?
        !          1075:                    AR5K_AR5212_DCU_RETRY_LMT_SH_RETRY : retry_lg;
        !          1076:        } else {
        !          1077:                retry_lg = AR5K_INIT_LG_RETRY;
        !          1078:                retry_sh = AR5K_INIT_SH_RETRY;
        !          1079:        }
        !          1080:
        !          1081:        AR5K_REG_WRITE(AR5K_AR5212_DCU_RETRY_LMT(queue),
        !          1082:            AR5K_REG_SM(AR5K_INIT_SLG_RETRY,
        !          1083:            AR5K_AR5212_DCU_RETRY_LMT_SLG_RETRY) |
        !          1084:            AR5K_REG_SM(AR5K_INIT_SSH_RETRY,
        !          1085:            AR5K_AR5212_DCU_RETRY_LMT_SSH_RETRY) |
        !          1086:            AR5K_REG_SM(retry_lg, AR5K_AR5212_DCU_RETRY_LMT_LG_RETRY) |
        !          1087:            AR5K_REG_SM(retry_sh, AR5K_AR5212_DCU_RETRY_LMT_SH_RETRY));
        !          1088:
        !          1089:        /*
        !          1090:         * Set initial content window (cw_min/cw_max)
        !          1091:         */
        !          1092:        cw_min = 1;
        !          1093:        while (cw_min < hal->ah_cw_min)
        !          1094:                cw_min = (cw_min << 1) | 1;
        !          1095:
        !          1096:        cw_min = tq->tqi_cw_min < 0 ?
        !          1097:            (cw_min >> (-tq->tqi_cw_min)) :
        !          1098:            ((cw_min << tq->tqi_cw_min) + (1 << tq->tqi_cw_min) - 1);
        !          1099:        cw_max = tq->tqi_cw_max < 0 ?
        !          1100:            (cw_max >> (-tq->tqi_cw_max)) :
        !          1101:            ((cw_max << tq->tqi_cw_max) + (1 << tq->tqi_cw_max) - 1);
        !          1102:
        !          1103:        AR5K_REG_WRITE(AR5K_AR5212_DCU_LCL_IFS(queue),
        !          1104:            AR5K_REG_SM(cw_min, AR5K_AR5212_DCU_LCL_IFS_CW_MIN) |
        !          1105:            AR5K_REG_SM(cw_max, AR5K_AR5212_DCU_LCL_IFS_CW_MAX) |
        !          1106:            AR5K_REG_SM(hal->ah_aifs + tq->tqi_aifs,
        !          1107:            AR5K_AR5212_DCU_LCL_IFS_AIFS));
        !          1108:
        !          1109:        /*
        !          1110:         * Set misc registers
        !          1111:         */
        !          1112:        AR5K_REG_WRITE(AR5K_AR5212_QCU_MISC(queue),
        !          1113:            AR5K_AR5212_QCU_MISC_DCU_EARLY);
        !          1114:
        !          1115:        if (tq->tqi_cbr_period) {
        !          1116:                AR5K_REG_WRITE(AR5K_AR5212_QCU_CBRCFG(queue),
        !          1117:                    AR5K_REG_SM(tq->tqi_cbr_period,
        !          1118:                    AR5K_AR5212_QCU_CBRCFG_INTVAL) |
        !          1119:                    AR5K_REG_SM(tq->tqi_cbr_overflow_limit,
        !          1120:                    AR5K_AR5212_QCU_CBRCFG_ORN_THRES));
        !          1121:                AR5K_REG_ENABLE_BITS(AR5K_AR5212_QCU_MISC(queue),
        !          1122:                    AR5K_AR5212_QCU_MISC_FRSHED_CBR);
        !          1123:                if (tq->tqi_cbr_overflow_limit)
        !          1124:                        AR5K_REG_ENABLE_BITS(AR5K_AR5212_QCU_MISC(queue),
        !          1125:                            AR5K_AR5212_QCU_MISC_CBR_THRES_ENABLE);
        !          1126:        }
        !          1127:
        !          1128:        if (tq->tqi_ready_time) {
        !          1129:                AR5K_REG_WRITE(AR5K_AR5212_QCU_RDYTIMECFG(queue),
        !          1130:                    AR5K_REG_SM(tq->tqi_ready_time,
        !          1131:                    AR5K_AR5212_QCU_RDYTIMECFG_INTVAL) |
        !          1132:                    AR5K_AR5212_QCU_RDYTIMECFG_ENABLE);
        !          1133:        }
        !          1134:
        !          1135:        if (tq->tqi_burst_time) {
        !          1136:                AR5K_REG_WRITE(AR5K_AR5212_DCU_CHAN_TIME(queue),
        !          1137:                    AR5K_REG_SM(tq->tqi_burst_time,
        !          1138:                    AR5K_AR5212_DCU_CHAN_TIME_DUR) |
        !          1139:                    AR5K_AR5212_DCU_CHAN_TIME_ENABLE);
        !          1140:
        !          1141:                if (tq->tqi_flags & AR5K_TXQ_FLAG_RDYTIME_EXP_POLICY_ENABLE) {
        !          1142:                        AR5K_REG_ENABLE_BITS(AR5K_AR5212_QCU_MISC(queue),
        !          1143:                            AR5K_AR5212_QCU_MISC_TXE);
        !          1144:                }
        !          1145:        }
        !          1146:
        !          1147:        if (tq->tqi_flags & AR5K_TXQ_FLAG_BACKOFF_DISABLE) {
        !          1148:                AR5K_REG_WRITE(AR5K_AR5212_DCU_MISC(queue),
        !          1149:                    AR5K_AR5212_DCU_MISC_POST_FR_BKOFF_DIS);
        !          1150:        }
        !          1151:
        !          1152:        if (tq->tqi_flags & AR5K_TXQ_FLAG_FRAG_BURST_BACKOFF_ENABLE) {
        !          1153:                AR5K_REG_WRITE(AR5K_AR5212_DCU_MISC(queue),
        !          1154:                    AR5K_AR5212_DCU_MISC_BACKOFF_FRAG);
        !          1155:        }
        !          1156:
        !          1157:        /*
        !          1158:         * Set registers by queue type
        !          1159:         */
        !          1160:        switch (tq->tqi_type) {
        !          1161:        case HAL_TX_QUEUE_BEACON:
        !          1162:                AR5K_REG_ENABLE_BITS(AR5K_AR5212_QCU_MISC(queue),
        !          1163:                    AR5K_AR5212_QCU_MISC_FRSHED_DBA_GT |
        !          1164:                    AR5K_AR5212_QCU_MISC_CBREXP_BCN |
        !          1165:                    AR5K_AR5212_QCU_MISC_BCN_ENABLE);
        !          1166:
        !          1167:                AR5K_REG_ENABLE_BITS(AR5K_AR5212_DCU_MISC(queue),
        !          1168:                    (AR5K_AR5212_DCU_MISC_ARBLOCK_CTL_GLOBAL <<
        !          1169:                    AR5K_AR5212_DCU_MISC_ARBLOCK_CTL_GLOBAL) |
        !          1170:                    AR5K_AR5212_DCU_MISC_POST_FR_BKOFF_DIS |
        !          1171:                    AR5K_AR5212_DCU_MISC_BCN_ENABLE);
        !          1172:
        !          1173:                AR5K_REG_WRITE(AR5K_AR5212_QCU_RDYTIMECFG(queue),
        !          1174:                    ((AR5K_TUNE_BEACON_INTERVAL -
        !          1175:                    (AR5K_TUNE_SW_BEACON_RESP - AR5K_TUNE_DMA_BEACON_RESP) -
        !          1176:                    AR5K_TUNE_ADDITIONAL_SWBA_BACKOFF) * 1024) |
        !          1177:                    AR5K_AR5212_QCU_RDYTIMECFG_ENABLE);
        !          1178:                break;
        !          1179:
        !          1180:        case HAL_TX_QUEUE_CAB:
        !          1181:                AR5K_REG_ENABLE_BITS(AR5K_AR5212_QCU_MISC(queue),
        !          1182:                    AR5K_AR5212_QCU_MISC_FRSHED_DBA_GT |
        !          1183:                    AR5K_AR5212_QCU_MISC_CBREXP |
        !          1184:                    AR5K_AR5212_QCU_MISC_CBREXP_BCN);
        !          1185:
        !          1186:                AR5K_REG_ENABLE_BITS(AR5K_AR5212_DCU_MISC(queue),
        !          1187:                    (AR5K_AR5212_DCU_MISC_ARBLOCK_CTL_GLOBAL <<
        !          1188:                    AR5K_AR5212_DCU_MISC_ARBLOCK_CTL_GLOBAL));
        !          1189:                break;
        !          1190:
        !          1191:        case HAL_TX_QUEUE_PSPOLL:
        !          1192:                AR5K_REG_ENABLE_BITS(AR5K_AR5212_QCU_MISC(queue),
        !          1193:                    AR5K_AR5212_QCU_MISC_CBREXP);
        !          1194:                break;
        !          1195:
        !          1196:        case HAL_TX_QUEUE_DATA:
        !          1197:        default:
        !          1198:                break;
        !          1199:        }
        !          1200:
        !          1201:        /*
        !          1202:         * Enable tx queue in the secondary interrupt mask registers
        !          1203:         */
        !          1204:        AR5K_REG_WRITE(AR5K_AR5212_SIMR0,
        !          1205:            AR5K_REG_SM(hal->ah_txq_interrupts, AR5K_AR5212_SIMR0_QCU_TXOK) |
        !          1206:            AR5K_REG_SM(hal->ah_txq_interrupts, AR5K_AR5212_SIMR0_QCU_TXDESC));
        !          1207:        AR5K_REG_WRITE(AR5K_AR5212_SIMR1,
        !          1208:            AR5K_REG_SM(hal->ah_txq_interrupts, AR5K_AR5212_SIMR1_QCU_TXERR));
        !          1209:        AR5K_REG_WRITE(AR5K_AR5212_SIMR2,
        !          1210:            AR5K_REG_SM(hal->ah_txq_interrupts, AR5K_AR5212_SIMR2_QCU_TXURN));
        !          1211:
        !          1212:        return (AH_TRUE);
        !          1213: }
        !          1214:
        !          1215: u_int32_t
        !          1216: ar5k_ar5212_get_tx_buf(struct ath_hal *hal, u_int queue)
        !          1217: {
        !          1218:        AR5K_ASSERT_ENTRY(queue, hal->ah_capabilities.cap_queues.q_tx_num);
        !          1219:
        !          1220:        /*
        !          1221:         * Get the transmit queue descriptor pointer from the selected queue
        !          1222:         */
        !          1223:        return (AR5K_REG_READ(AR5K_AR5212_QCU_TXDP(queue)));
        !          1224: }
        !          1225:
        !          1226: HAL_BOOL
        !          1227: ar5k_ar5212_put_tx_buf(struct ath_hal *hal, u_int queue, u_int32_t phys_addr)
        !          1228: {
        !          1229:        AR5K_ASSERT_ENTRY(queue, hal->ah_capabilities.cap_queues.q_tx_num);
        !          1230:
        !          1231:        /*
        !          1232:         * Set the transmit queue descriptor pointer for the selected queue
        !          1233:         * (this won't work if the queue is still active)
        !          1234:         */
        !          1235:        if (AR5K_REG_READ_Q(AR5K_AR5212_QCU_TXE, queue))
        !          1236:                return (AH_FALSE);
        !          1237:
        !          1238:        AR5K_REG_WRITE(AR5K_AR5212_QCU_TXDP(queue), phys_addr);
        !          1239:
        !          1240:        return (AH_TRUE);
        !          1241: }
        !          1242:
        !          1243: u_int32_t
        !          1244: ar5k_ar5212_num_tx_pending(struct ath_hal *hal, u_int queue)
        !          1245: {
        !          1246:        AR5K_ASSERT_ENTRY(queue, hal->ah_capabilities.cap_queues.q_tx_num);
        !          1247:        return (AR5K_AR5212_QCU_STS(queue) & AR5K_AR5212_QCU_STS_FRMPENDCNT);
        !          1248: }
        !          1249:
        !          1250: HAL_BOOL
        !          1251: ar5k_ar5212_tx_start(struct ath_hal *hal, u_int queue)
        !          1252: {
        !          1253:        AR5K_ASSERT_ENTRY(queue, hal->ah_capabilities.cap_queues.q_tx_num);
        !          1254:
        !          1255:        /* Return if queue is disabled */
        !          1256:        if (AR5K_REG_READ_Q(AR5K_AR5212_QCU_TXD, queue))
        !          1257:                return (AH_FALSE);
        !          1258:
        !          1259:        /* Start queue */
        !          1260:        AR5K_REG_WRITE_Q(AR5K_AR5212_QCU_TXE, queue);
        !          1261:
        !          1262:        return (AH_TRUE);
        !          1263: }
        !          1264:
        !          1265: HAL_BOOL
        !          1266: ar5k_ar5212_stop_tx_dma(struct ath_hal *hal, u_int queue)
        !          1267: {
        !          1268:        int i = 100, pending;
        !          1269:
        !          1270:        AR5K_ASSERT_ENTRY(queue, hal->ah_capabilities.cap_queues.q_tx_num);
        !          1271:
        !          1272:        /*
        !          1273:         * Schedule TX disable and wait until queue is empty
        !          1274:         */
        !          1275:        AR5K_REG_WRITE_Q(AR5K_AR5212_QCU_TXD, queue);
        !          1276:
        !          1277:        do {
        !          1278:                pending = AR5K_REG_READ(AR5K_AR5212_QCU_STS(queue)) &
        !          1279:                     AR5K_AR5212_QCU_STS_FRMPENDCNT;
        !          1280:                delay(100);
        !          1281:        } while (--i && pending);
        !          1282:
        !          1283:        /* Clear register */
        !          1284:        AR5K_REG_WRITE(AR5K_AR5212_QCU_TXD, 0);
        !          1285:
        !          1286:        return (AH_TRUE);
        !          1287: }
        !          1288:
        !          1289: HAL_BOOL
        !          1290: ar5k_ar5212_setup_tx_desc(struct ath_hal *hal, struct ath_desc *desc,
        !          1291:     u_int packet_length, u_int header_length, HAL_PKT_TYPE type, u_int tx_power,
        !          1292:     u_int tx_rate0, u_int tx_tries0, u_int key_index, u_int antenna_mode,
        !          1293:     u_int flags, u_int rtscts_rate, u_int rtscts_duration)
        !          1294: {
        !          1295:        struct ar5k_ar5212_tx_desc *tx_desc;
        !          1296:
        !          1297:        tx_desc = (struct ar5k_ar5212_tx_desc*)&desc->ds_ctl0;
        !          1298:
        !          1299:        /*
        !          1300:         * Validate input
        !          1301:         */
        !          1302:        if (tx_tries0 == 0)
        !          1303:                return (AH_FALSE);
        !          1304:
        !          1305:        if ((tx_desc->tx_control_0 = (packet_length &
        !          1306:            AR5K_AR5212_DESC_TX_CTL0_FRAME_LEN)) != packet_length)
        !          1307:                return (AH_FALSE);
        !          1308:
        !          1309:        tx_desc->tx_control_0 |=
        !          1310:            AR5K_REG_SM(tx_power, AR5K_AR5212_DESC_TX_CTL0_XMIT_POWER) |
        !          1311:            AR5K_REG_SM(antenna_mode, AR5K_AR5212_DESC_TX_CTL0_ANT_MODE_XMIT);
        !          1312:        tx_desc->tx_control_1 =
        !          1313:            AR5K_REG_SM(type, AR5K_AR5212_DESC_TX_CTL1_FRAME_TYPE);
        !          1314:        tx_desc->tx_control_2 =
        !          1315:            AR5K_REG_SM(tx_tries0 + AR5K_TUNE_HWTXTRIES,
        !          1316:            AR5K_AR5212_DESC_TX_CTL2_XMIT_TRIES0);
        !          1317:        tx_desc->tx_control_3 =
        !          1318:            tx_rate0 & AR5K_AR5212_DESC_TX_CTL3_XMIT_RATE0;
        !          1319:
        !          1320: #define _TX_FLAGS(_c, _flag)                                           \
        !          1321:        if (flags & HAL_TXDESC_##_flag)                                 \
        !          1322:                tx_desc->tx_control_##_c |=                             \
        !          1323:                        AR5K_AR5212_DESC_TX_CTL##_c##_##_flag
        !          1324:
        !          1325:        _TX_FLAGS(0, CLRDMASK);
        !          1326:        _TX_FLAGS(0, VEOL);
        !          1327:        _TX_FLAGS(0, INTREQ);
        !          1328:        _TX_FLAGS(0, RTSENA);
        !          1329:        _TX_FLAGS(0, CTSENA);
        !          1330:        _TX_FLAGS(1, NOACK);
        !          1331:
        !          1332: #undef _TX_FLAGS
        !          1333:
        !          1334:        /*
        !          1335:         * WEP crap
        !          1336:         */
        !          1337:        if (key_index != HAL_TXKEYIX_INVALID) {
        !          1338:                tx_desc->tx_control_0 |=
        !          1339:                    AR5K_AR5212_DESC_TX_CTL0_ENCRYPT_KEY_VALID;
        !          1340:                tx_desc->tx_control_1 |=
        !          1341:                    AR5K_REG_SM(key_index,
        !          1342:                    AR5K_AR5212_DESC_TX_CTL1_ENCRYPT_KEY_INDEX);
        !          1343:        }
        !          1344:
        !          1345:        /*
        !          1346:         * RTS/CTS
        !          1347:         */
        !          1348:        if (flags & (HAL_TXDESC_RTSENA | HAL_TXDESC_CTSENA)) {
        !          1349:                if ((flags & HAL_TXDESC_RTSENA) &&
        !          1350:                    (flags & HAL_TXDESC_CTSENA))
        !          1351:                        return (AH_FALSE);
        !          1352:                tx_desc->tx_control_2 |=
        !          1353:                    rtscts_duration & AR5K_AR5212_DESC_TX_CTL2_RTS_DURATION;
        !          1354:                tx_desc->tx_control_3 |=
        !          1355:                    AR5K_REG_SM(rtscts_rate,
        !          1356:                    AR5K_AR5212_DESC_TX_CTL3_RTS_CTS_RATE);
        !          1357:        }
        !          1358:
        !          1359:        return (AH_TRUE);
        !          1360: }
        !          1361:
        !          1362: HAL_BOOL
        !          1363: ar5k_ar5212_fill_tx_desc(struct ath_hal *hal, struct ath_desc *desc,
        !          1364:     u_int segment_length, HAL_BOOL first_segment, HAL_BOOL last_segment)
        !          1365: {
        !          1366:        struct ar5k_ar5212_tx_desc *tx_desc;
        !          1367:        struct ar5k_ar5212_tx_status *tx_status;
        !          1368:
        !          1369:        tx_desc = (struct ar5k_ar5212_tx_desc*)&desc->ds_ctl0;
        !          1370:        tx_status = (struct ar5k_ar5212_tx_status*)&desc->ds_hw[2];
        !          1371:
        !          1372:        /* Clear status descriptor */
        !          1373:        bzero(tx_status, sizeof(struct ar5k_ar5212_tx_status));
        !          1374:
        !          1375:        /* Validate segment length and initialize the descriptor */
        !          1376:        if ((tx_desc->tx_control_1 = (segment_length &
        !          1377:            AR5K_AR5212_DESC_TX_CTL1_BUF_LEN)) != segment_length)
        !          1378:                return (AH_FALSE);
        !          1379:
        !          1380:        if (first_segment != AH_TRUE)
        !          1381:                tx_desc->tx_control_0 &= ~AR5K_AR5212_DESC_TX_CTL0_FRAME_LEN;
        !          1382:
        !          1383:        if (last_segment != AH_TRUE)
        !          1384:                tx_desc->tx_control_1 |= AR5K_AR5212_DESC_TX_CTL1_MORE;
        !          1385:
        !          1386:        return (AH_TRUE);
        !          1387: }
        !          1388:
        !          1389: HAL_BOOL
        !          1390: ar5k_ar5212_setup_xtx_desc(struct ath_hal *hal, struct ath_desc *desc,
        !          1391:     u_int tx_rate1, u_int tx_tries1, u_int tx_rate2, u_int tx_tries2,
        !          1392:     u_int tx_rate3, u_int tx_tries3)
        !          1393: {
        !          1394:        struct ar5k_ar5212_tx_desc *tx_desc;
        !          1395:
        !          1396:        tx_desc = (struct ar5k_ar5212_tx_desc*)&desc->ds_ctl0;
        !          1397:
        !          1398: #define _XTX_TRIES(_n)                                                 \
        !          1399:        if (tx_tries##_n) {                                             \
        !          1400:                tx_desc->tx_control_2 |=                                \
        !          1401:                    AR5K_REG_SM(tx_tries##_n,                           \
        !          1402:                    AR5K_AR5212_DESC_TX_CTL2_XMIT_TRIES##_n);           \
        !          1403:                tx_desc->tx_control_3 |=                                \
        !          1404:                    AR5K_REG_SM(tx_rate##_n,                            \
        !          1405:                    AR5K_AR5212_DESC_TX_CTL3_XMIT_RATE##_n);            \
        !          1406:        }
        !          1407:
        !          1408:        _XTX_TRIES(1);
        !          1409:        _XTX_TRIES(2);
        !          1410:        _XTX_TRIES(3);
        !          1411:
        !          1412: #undef _XTX_TRIES
        !          1413:
        !          1414:        return (AH_TRUE);
        !          1415: }
        !          1416:
        !          1417: HAL_STATUS
        !          1418: ar5k_ar5212_proc_tx_desc(struct ath_hal *hal, struct ath_desc *desc)
        !          1419: {
        !          1420:        struct ar5k_ar5212_tx_status *tx_status;
        !          1421:        struct ar5k_ar5212_tx_desc *tx_desc;
        !          1422:
        !          1423:        tx_desc = (struct ar5k_ar5212_tx_desc*)&desc->ds_ctl0;
        !          1424:        tx_status = (struct ar5k_ar5212_tx_status*)&desc->ds_hw[2];
        !          1425:
        !          1426:        /* No frame has been send or error */
        !          1427:        if ((tx_status->tx_status_1 & AR5K_AR5212_DESC_TX_STATUS1_DONE) == 0)
        !          1428:                return (HAL_EINPROGRESS);
        !          1429:
        !          1430:        /*
        !          1431:         * Get descriptor status
        !          1432:         */
        !          1433:        desc->ds_us.tx.ts_tstamp =
        !          1434:            AR5K_REG_MS(tx_status->tx_status_0,
        !          1435:            AR5K_AR5212_DESC_TX_STATUS0_SEND_TIMESTAMP);
        !          1436:        desc->ds_us.tx.ts_shortretry =
        !          1437:            AR5K_REG_MS(tx_status->tx_status_0,
        !          1438:            AR5K_AR5212_DESC_TX_STATUS0_RTS_FAIL_COUNT);
        !          1439:        desc->ds_us.tx.ts_longretry =
        !          1440:            AR5K_REG_MS(tx_status->tx_status_0,
        !          1441:            AR5K_AR5212_DESC_TX_STATUS0_DATA_FAIL_COUNT);
        !          1442:        desc->ds_us.tx.ts_seqnum =
        !          1443:            AR5K_REG_MS(tx_status->tx_status_1,
        !          1444:            AR5K_AR5212_DESC_TX_STATUS1_SEQ_NUM);
        !          1445:        desc->ds_us.tx.ts_rssi =
        !          1446:            AR5K_REG_MS(tx_status->tx_status_1,
        !          1447:            AR5K_AR5212_DESC_TX_STATUS1_ACK_SIG_STRENGTH);
        !          1448:        desc->ds_us.tx.ts_antenna = (tx_status->tx_status_1 &
        !          1449:            AR5K_AR5212_DESC_TX_STATUS1_XMIT_ANTENNA) ? 2 : 1;
        !          1450:        desc->ds_us.tx.ts_status = 0;
        !          1451:
        !          1452:        switch (AR5K_REG_MS(tx_status->tx_status_1,
        !          1453:            AR5K_AR5212_DESC_TX_STATUS1_FINAL_TS_INDEX)) {
        !          1454:        case 0:
        !          1455:                desc->ds_us.tx.ts_rate = tx_desc->tx_control_3 &
        !          1456:                    AR5K_AR5212_DESC_TX_CTL3_XMIT_RATE0;
        !          1457:                break;
        !          1458:        case 1:
        !          1459:                desc->ds_us.tx.ts_rate =
        !          1460:                    AR5K_REG_MS(tx_desc->tx_control_3,
        !          1461:                    AR5K_AR5212_DESC_TX_CTL3_XMIT_RATE1);
        !          1462:                desc->ds_us.tx.ts_longretry +=
        !          1463:                    AR5K_REG_MS(tx_desc->tx_control_2,
        !          1464:                    AR5K_AR5212_DESC_TX_CTL2_XMIT_TRIES1);
        !          1465:                break;
        !          1466:        case 2:
        !          1467:                desc->ds_us.tx.ts_rate =
        !          1468:                    AR5K_REG_MS(tx_desc->tx_control_3,
        !          1469:                    AR5K_AR5212_DESC_TX_CTL3_XMIT_RATE2);
        !          1470:                desc->ds_us.tx.ts_longretry +=
        !          1471:                    AR5K_REG_MS(tx_desc->tx_control_2,
        !          1472:                    AR5K_AR5212_DESC_TX_CTL2_XMIT_TRIES2);
        !          1473:                break;
        !          1474:        case 3:
        !          1475:                desc->ds_us.tx.ts_rate =
        !          1476:                    AR5K_REG_MS(tx_desc->tx_control_3,
        !          1477:                    AR5K_AR5212_DESC_TX_CTL3_XMIT_RATE3);
        !          1478:                desc->ds_us.tx.ts_longretry +=
        !          1479:                    AR5K_REG_MS(tx_desc->tx_control_2,
        !          1480:                    AR5K_AR5212_DESC_TX_CTL2_XMIT_TRIES3);
        !          1481:                break;
        !          1482:        }
        !          1483:
        !          1484:        if ((tx_status->tx_status_0 &
        !          1485:            AR5K_AR5212_DESC_TX_STATUS0_FRAME_XMIT_OK) == 0) {
        !          1486:                if (tx_status->tx_status_0 &
        !          1487:                    AR5K_AR5212_DESC_TX_STATUS0_EXCESSIVE_RETRIES)
        !          1488:                        desc->ds_us.tx.ts_status |= HAL_TXERR_XRETRY;
        !          1489:
        !          1490:                if (tx_status->tx_status_0 &
        !          1491:                    AR5K_AR5212_DESC_TX_STATUS0_FIFO_UNDERRUN)
        !          1492:                        desc->ds_us.tx.ts_status |= HAL_TXERR_FIFO;
        !          1493:
        !          1494:                if (tx_status->tx_status_0 &
        !          1495:                    AR5K_AR5212_DESC_TX_STATUS0_FILTERED)
        !          1496:                        desc->ds_us.tx.ts_status |= HAL_TXERR_FILT;
        !          1497:        }
        !          1498:
        !          1499:        return (HAL_OK);
        !          1500: }
        !          1501:
        !          1502: HAL_BOOL
        !          1503: ar5k_ar5212_has_veol(struct ath_hal *hal)
        !          1504: {
        !          1505:        return (AH_TRUE);
        !          1506: }
        !          1507:
        !          1508: /*
        !          1509:  * Receive functions
        !          1510:  */
        !          1511:
        !          1512: u_int32_t
        !          1513: ar5k_ar5212_get_rx_buf(struct ath_hal *hal)
        !          1514: {
        !          1515:        return (AR5K_REG_READ(AR5K_AR5212_RXDP));
        !          1516: }
        !          1517:
        !          1518: void
        !          1519: ar5k_ar5212_put_rx_buf(struct ath_hal *hal, u_int32_t phys_addr)
        !          1520: {
        !          1521:        AR5K_REG_WRITE(AR5K_AR5212_RXDP, phys_addr);
        !          1522: }
        !          1523:
        !          1524: void
        !          1525: ar5k_ar5212_start_rx(struct ath_hal *hal)
        !          1526: {
        !          1527:        AR5K_REG_WRITE(AR5K_AR5212_CR, AR5K_AR5212_CR_RXE);
        !          1528: }
        !          1529:
        !          1530: HAL_BOOL
        !          1531: ar5k_ar5212_stop_rx_dma(struct ath_hal *hal)
        !          1532: {
        !          1533:        int i;
        !          1534:
        !          1535:        AR5K_REG_WRITE(AR5K_AR5212_CR, AR5K_AR5212_CR_RXD);
        !          1536:
        !          1537:        /*
        !          1538:         * It may take some time to disable the DMA receive unit
        !          1539:         */
        !          1540:        for (i = 2000;
        !          1541:             i > 0 && (AR5K_REG_READ(AR5K_AR5212_CR) & AR5K_AR5212_CR_RXE) != 0;
        !          1542:             i--)
        !          1543:                AR5K_DELAY(10);
        !          1544:
        !          1545:        return (i > 0 ? AH_TRUE : AH_FALSE);
        !          1546: }
        !          1547:
        !          1548: void
        !          1549: ar5k_ar5212_start_rx_pcu(struct ath_hal *hal)
        !          1550: {
        !          1551:        AR5K_REG_DISABLE_BITS(AR5K_AR5212_DIAG_SW, AR5K_AR5212_DIAG_SW_DIS_RX);
        !          1552: }
        !          1553:
        !          1554: void
        !          1555: ar5k_ar5212_stop_pcu_recv(struct ath_hal *hal)
        !          1556: {
        !          1557:        AR5K_REG_ENABLE_BITS(AR5K_AR5212_DIAG_SW, AR5K_AR5212_DIAG_SW_DIS_RX);
        !          1558: }
        !          1559:
        !          1560: void
        !          1561: ar5k_ar5212_set_mcast_filter(struct ath_hal *hal, u_int32_t filter0,
        !          1562:     u_int32_t filter1)
        !          1563: {
        !          1564:        /* Set the multicat filter */
        !          1565:        AR5K_REG_WRITE(AR5K_AR5212_MCAST_FIL0, filter0);
        !          1566:        AR5K_REG_WRITE(AR5K_AR5212_MCAST_FIL1, filter1);
        !          1567: }
        !          1568:
        !          1569: HAL_BOOL
        !          1570: ar5k_ar5212_set_mcast_filterindex(struct ath_hal *hal, u_int32_t index)
        !          1571: {
        !          1572:        if (index >= 64) {
        !          1573:            return (AH_FALSE);
        !          1574:        } else if (index >= 32) {
        !          1575:            AR5K_REG_ENABLE_BITS(AR5K_AR5212_MCAST_FIL1,
        !          1576:                (1 << (index - 32)));
        !          1577:        } else {
        !          1578:            AR5K_REG_ENABLE_BITS(AR5K_AR5212_MCAST_FIL0,
        !          1579:                (1 << index));
        !          1580:        }
        !          1581:
        !          1582:        return (AH_TRUE);
        !          1583: }
        !          1584:
        !          1585: HAL_BOOL
        !          1586: ar5k_ar5212_clear_mcast_filter_idx(struct ath_hal *hal, u_int32_t index)
        !          1587: {
        !          1588:
        !          1589:        if (index >= 64) {
        !          1590:            return (AH_FALSE);
        !          1591:        } else if (index >= 32) {
        !          1592:            AR5K_REG_DISABLE_BITS(AR5K_AR5212_MCAST_FIL1,
        !          1593:                (1 << (index - 32)));
        !          1594:        } else {
        !          1595:            AR5K_REG_DISABLE_BITS(AR5K_AR5212_MCAST_FIL0,
        !          1596:                (1 << index));
        !          1597:        }
        !          1598:
        !          1599:        return (AH_TRUE);
        !          1600: }
        !          1601:
        !          1602: u_int32_t
        !          1603: ar5k_ar5212_get_rx_filter(struct ath_hal *hal)
        !          1604: {
        !          1605:        u_int32_t data, filter = 0;
        !          1606:
        !          1607:        filter = AR5K_REG_READ(AR5K_AR5212_RX_FILTER);
        !          1608:        data = AR5K_REG_READ(AR5K_AR5212_PHY_ERR_FIL);
        !          1609:
        !          1610:        if (data & AR5K_AR5212_PHY_ERR_FIL_RADAR)
        !          1611:                filter |= HAL_RX_FILTER_PHYRADAR;
        !          1612:        if (data & (AR5K_AR5212_PHY_ERR_FIL_OFDM |
        !          1613:            AR5K_AR5212_PHY_ERR_FIL_CCK))
        !          1614:                filter |= HAL_RX_FILTER_PHYERR;
        !          1615:
        !          1616:        return (filter);
        !          1617: }
        !          1618:
        !          1619: void
        !          1620: ar5k_ar5212_set_rx_filter(struct ath_hal *hal, u_int32_t filter)
        !          1621: {
        !          1622:        u_int32_t data = 0;
        !          1623:
        !          1624:        if (filter & HAL_RX_FILTER_PHYRADAR)
        !          1625:                data |= AR5K_AR5212_PHY_ERR_FIL_RADAR;
        !          1626:        if (filter & HAL_RX_FILTER_PHYERR)
        !          1627:                data |= AR5K_AR5212_PHY_ERR_FIL_OFDM |
        !          1628:                    AR5K_AR5212_PHY_ERR_FIL_CCK;
        !          1629:
        !          1630:        if (data) {
        !          1631:                AR5K_REG_ENABLE_BITS(AR5K_AR5212_RXCFG,
        !          1632:                    AR5K_AR5212_RXCFG_ZLFDMA);
        !          1633:        } else {
        !          1634:                AR5K_REG_DISABLE_BITS(AR5K_AR5212_RXCFG,
        !          1635:                    AR5K_AR5212_RXCFG_ZLFDMA);
        !          1636:        }
        !          1637:
        !          1638:        AR5K_REG_WRITE(AR5K_AR5212_RX_FILTER, filter & 0xff);
        !          1639:        AR5K_REG_WRITE(AR5K_AR5212_PHY_ERR_FIL, data);
        !          1640: }
        !          1641:
        !          1642: HAL_BOOL
        !          1643: ar5k_ar5212_setup_rx_desc(struct ath_hal *hal, struct ath_desc *desc,
        !          1644:     u_int32_t size, u_int flags)
        !          1645: {
        !          1646:        struct ar5k_ar5212_rx_desc *rx_desc;
        !          1647:
        !          1648:        rx_desc = (struct ar5k_ar5212_rx_desc*)&desc->ds_ctl0;
        !          1649:
        !          1650:        if ((rx_desc->rx_control_1 = (size &
        !          1651:            AR5K_AR5212_DESC_RX_CTL1_BUF_LEN)) != size)
        !          1652:                return (AH_FALSE);
        !          1653:
        !          1654:        if (flags & HAL_RXDESC_INTREQ)
        !          1655:                rx_desc->rx_control_1 |= AR5K_AR5212_DESC_RX_CTL1_INTREQ;
        !          1656:
        !          1657:        return (AH_TRUE);
        !          1658: }
        !          1659:
        !          1660: HAL_STATUS
        !          1661: ar5k_ar5212_proc_rx_desc(struct ath_hal *hal, struct ath_desc *desc,
        !          1662:     u_int32_t phys_addr, struct ath_desc *next)
        !          1663: {
        !          1664:        struct ar5k_ar5212_rx_status *rx_status;
        !          1665:        struct ar5k_ar5212_rx_error *rx_err;
        !          1666:
        !          1667:        rx_status = (struct ar5k_ar5212_rx_status*)&desc->ds_hw[0];
        !          1668:
        !          1669:        /* Overlay on error */
        !          1670:        rx_err = (struct ar5k_ar5212_rx_error*)&desc->ds_hw[0];
        !          1671:
        !          1672:        /* No frame received / not ready */
        !          1673:        if ((rx_status->rx_status_1 & AR5K_AR5212_DESC_RX_STATUS1_DONE) == 0)
        !          1674:                return (HAL_EINPROGRESS);
        !          1675:
        !          1676:        /*
        !          1677:         * Frame receive status
        !          1678:         */
        !          1679:        desc->ds_us.rx.rs_datalen = rx_status->rx_status_0 &
        !          1680:            AR5K_AR5212_DESC_RX_STATUS0_DATA_LEN;
        !          1681:        desc->ds_us.rx.rs_rssi =
        !          1682:            AR5K_REG_MS(rx_status->rx_status_0,
        !          1683:            AR5K_AR5212_DESC_RX_STATUS0_RECEIVE_SIGNAL);
        !          1684:        desc->ds_us.rx.rs_rate =
        !          1685:            AR5K_REG_MS(rx_status->rx_status_0,
        !          1686:            AR5K_AR5212_DESC_RX_STATUS0_RECEIVE_RATE);
        !          1687:        desc->ds_us.rx.rs_antenna = rx_status->rx_status_0 &
        !          1688:            AR5K_AR5212_DESC_RX_STATUS0_RECEIVE_ANTENNA;
        !          1689:        desc->ds_us.rx.rs_more = rx_status->rx_status_0 &
        !          1690:            AR5K_AR5212_DESC_RX_STATUS0_MORE;
        !          1691:        desc->ds_us.rx.rs_tstamp =
        !          1692:            AR5K_REG_MS(rx_status->rx_status_1,
        !          1693:            AR5K_AR5212_DESC_RX_STATUS1_RECEIVE_TIMESTAMP);
        !          1694:        desc->ds_us.rx.rs_status = 0;
        !          1695:
        !          1696:        /*
        !          1697:         * Key table status
        !          1698:         */
        !          1699:        if (rx_status->rx_status_1 &
        !          1700:            AR5K_AR5212_DESC_RX_STATUS1_KEY_INDEX_VALID) {
        !          1701:                desc->ds_us.rx.rs_keyix =
        !          1702:                    AR5K_REG_MS(rx_status->rx_status_1,
        !          1703:                    AR5K_AR5212_DESC_RX_STATUS1_KEY_INDEX);
        !          1704:        } else {
        !          1705:                desc->ds_us.rx.rs_keyix = HAL_RXKEYIX_INVALID;
        !          1706:        }
        !          1707:
        !          1708:        /*
        !          1709:         * Receive/descriptor errors
        !          1710:         */
        !          1711:        if ((rx_status->rx_status_1 &
        !          1712:            AR5K_AR5212_DESC_RX_STATUS1_FRAME_RECEIVE_OK) == 0) {
        !          1713:                if (rx_status->rx_status_1 &
        !          1714:                    AR5K_AR5212_DESC_RX_STATUS1_CRC_ERROR)
        !          1715:                        desc->ds_us.rx.rs_status |= HAL_RXERR_CRC;
        !          1716:
        !          1717:                if (rx_status->rx_status_1 &
        !          1718:                    AR5K_AR5212_DESC_RX_STATUS1_PHY_ERROR) {
        !          1719:                        desc->ds_us.rx.rs_status |= HAL_RXERR_PHY;
        !          1720:                        desc->ds_us.rx.rs_phyerr =
        !          1721:                            AR5K_REG_MS(rx_err->rx_error_1,
        !          1722:                            AR5K_AR5212_DESC_RX_ERROR1_PHY_ERROR_CODE);
        !          1723:                }
        !          1724:
        !          1725:                if (rx_status->rx_status_1 &
        !          1726:                    AR5K_AR5212_DESC_RX_STATUS1_DECRYPT_CRC_ERROR)
        !          1727:                        desc->ds_us.rx.rs_status |= HAL_RXERR_DECRYPT;
        !          1728:
        !          1729:                if (rx_status->rx_status_1 &
        !          1730:                    AR5K_AR5212_DESC_RX_STATUS1_MIC_ERROR)
        !          1731:                        desc->ds_us.rx.rs_status |= HAL_RXERR_MIC;
        !          1732:        }
        !          1733:
        !          1734:        return (HAL_OK);
        !          1735: }
        !          1736:
        !          1737: void
        !          1738: ar5k_ar5212_set_rx_signal(struct ath_hal *hal)
        !          1739: {
        !          1740:        /* Signal state monitoring is not yet supported */
        !          1741: }
        !          1742:
        !          1743: /*
        !          1744:  * Misc functions
        !          1745:  */
        !          1746:
        !          1747: void
        !          1748: ar5k_ar5212_dump_state(struct ath_hal *hal)
        !          1749: {
        !          1750: #ifdef AR5K_DEBUG
        !          1751: #define AR5K_PRINT_REGISTER(_x)                                                \
        !          1752:        printf("(%s: %08x) ", #_x, AR5K_REG_READ(AR5K_AR5212_##_x));
        !          1753:
        !          1754:        printf("MAC registers:\n");
        !          1755:        AR5K_PRINT_REGISTER(CR);
        !          1756:        AR5K_PRINT_REGISTER(CFG);
        !          1757:        AR5K_PRINT_REGISTER(IER);
        !          1758:        AR5K_PRINT_REGISTER(TXCFG);
        !          1759:        AR5K_PRINT_REGISTER(RXCFG);
        !          1760:        AR5K_PRINT_REGISTER(MIBC);
        !          1761:        AR5K_PRINT_REGISTER(TOPS);
        !          1762:        AR5K_PRINT_REGISTER(RXNOFRM);
        !          1763:        AR5K_PRINT_REGISTER(RPGTO);
        !          1764:        AR5K_PRINT_REGISTER(RFCNT);
        !          1765:        AR5K_PRINT_REGISTER(MISC);
        !          1766:        AR5K_PRINT_REGISTER(PISR);
        !          1767:        AR5K_PRINT_REGISTER(SISR0);
        !          1768:        AR5K_PRINT_REGISTER(SISR1);
        !          1769:        AR5K_PRINT_REGISTER(SISR3);
        !          1770:        AR5K_PRINT_REGISTER(SISR4);
        !          1771:        AR5K_PRINT_REGISTER(DCM_ADDR);
        !          1772:        AR5K_PRINT_REGISTER(DCM_DATA);
        !          1773:        AR5K_PRINT_REGISTER(DCCFG);
        !          1774:        AR5K_PRINT_REGISTER(CCFG);
        !          1775:        AR5K_PRINT_REGISTER(CCFG_CUP);
        !          1776:        AR5K_PRINT_REGISTER(CPC0);
        !          1777:        AR5K_PRINT_REGISTER(CPC1);
        !          1778:        AR5K_PRINT_REGISTER(CPC2);
        !          1779:        AR5K_PRINT_REGISTER(CPCORN);
        !          1780:        AR5K_PRINT_REGISTER(QCU_TXE);
        !          1781:        AR5K_PRINT_REGISTER(QCU_TXD);
        !          1782:        AR5K_PRINT_REGISTER(DCU_GBL_IFS_SIFS);
        !          1783:        AR5K_PRINT_REGISTER(DCU_GBL_IFS_SLOT);
        !          1784:        AR5K_PRINT_REGISTER(DCU_FP);
        !          1785:        AR5K_PRINT_REGISTER(DCU_TXP);
        !          1786:        AR5K_PRINT_REGISTER(DCU_TX_FILTER);
        !          1787:        AR5K_PRINT_REGISTER(RC);
        !          1788:        AR5K_PRINT_REGISTER(SCR);
        !          1789:        AR5K_PRINT_REGISTER(INTPEND);
        !          1790:        AR5K_PRINT_REGISTER(PCICFG);
        !          1791:        AR5K_PRINT_REGISTER(GPIOCR);
        !          1792:        AR5K_PRINT_REGISTER(GPIODO);
        !          1793:        AR5K_PRINT_REGISTER(SREV);
        !          1794:        AR5K_PRINT_REGISTER(EEPROM_BASE);
        !          1795:        AR5K_PRINT_REGISTER(EEPROM_DATA);
        !          1796:        AR5K_PRINT_REGISTER(EEPROM_CMD);
        !          1797:        AR5K_PRINT_REGISTER(EEPROM_CFG);
        !          1798:        AR5K_PRINT_REGISTER(PCU_MIN);
        !          1799:        AR5K_PRINT_REGISTER(STA_ID0);
        !          1800:        AR5K_PRINT_REGISTER(STA_ID1);
        !          1801:        AR5K_PRINT_REGISTER(BSS_ID0);
        !          1802:        AR5K_PRINT_REGISTER(SLOT_TIME);
        !          1803:        AR5K_PRINT_REGISTER(TIME_OUT);
        !          1804:        AR5K_PRINT_REGISTER(RSSI_THR);
        !          1805:        AR5K_PRINT_REGISTER(BEACON);
        !          1806:        AR5K_PRINT_REGISTER(CFP_PERIOD);
        !          1807:        AR5K_PRINT_REGISTER(TIMER0);
        !          1808:        AR5K_PRINT_REGISTER(TIMER2);
        !          1809:        AR5K_PRINT_REGISTER(TIMER3);
        !          1810:        AR5K_PRINT_REGISTER(CFP_DUR);
        !          1811:        AR5K_PRINT_REGISTER(MCAST_FIL0);
        !          1812:        AR5K_PRINT_REGISTER(MCAST_FIL1);
        !          1813:        AR5K_PRINT_REGISTER(DIAG_SW);
        !          1814:        AR5K_PRINT_REGISTER(TSF_U32);
        !          1815:        AR5K_PRINT_REGISTER(ADDAC_TEST);
        !          1816:        AR5K_PRINT_REGISTER(DEFAULT_ANTENNA);
        !          1817:        AR5K_PRINT_REGISTER(LAST_TSTP);
        !          1818:        AR5K_PRINT_REGISTER(NAV);
        !          1819:        AR5K_PRINT_REGISTER(RTS_OK);
        !          1820:        AR5K_PRINT_REGISTER(ACK_FAIL);
        !          1821:        AR5K_PRINT_REGISTER(FCS_FAIL);
        !          1822:        AR5K_PRINT_REGISTER(BEACON_CNT);
        !          1823:        AR5K_PRINT_REGISTER(TSF_PARM);
        !          1824:        AR5K_PRINT_REGISTER(RATE_DUR_0);
        !          1825:        AR5K_PRINT_REGISTER(KEYTABLE_0);
        !          1826:        printf("\n");
        !          1827:
        !          1828:        printf("PHY registers:\n");
        !          1829:        AR5K_PRINT_REGISTER(PHY_TURBO);
        !          1830:        AR5K_PRINT_REGISTER(PHY_AGC);
        !          1831:        AR5K_PRINT_REGISTER(PHY_TIMING_3);
        !          1832:        AR5K_PRINT_REGISTER(PHY_CHIP_ID);
        !          1833:        AR5K_PRINT_REGISTER(PHY_AGCCTL);
        !          1834:        AR5K_PRINT_REGISTER(PHY_NF);
        !          1835:        AR5K_PRINT_REGISTER(PHY_SCR);
        !          1836:        AR5K_PRINT_REGISTER(PHY_SLMT);
        !          1837:        AR5K_PRINT_REGISTER(PHY_SCAL);
        !          1838:        AR5K_PRINT_REGISTER(PHY_RX_DELAY);
        !          1839:        AR5K_PRINT_REGISTER(PHY_IQ);
        !          1840:        AR5K_PRINT_REGISTER(PHY_PAPD_PROBE);
        !          1841:        AR5K_PRINT_REGISTER(PHY_TXPOWER_RATE1);
        !          1842:        AR5K_PRINT_REGISTER(PHY_TXPOWER_RATE2);
        !          1843:        AR5K_PRINT_REGISTER(PHY_FC);
        !          1844:        AR5K_PRINT_REGISTER(PHY_RADAR);
        !          1845:        AR5K_PRINT_REGISTER(PHY_ANT_SWITCH_TABLE_0);
        !          1846:        AR5K_PRINT_REGISTER(PHY_ANT_SWITCH_TABLE_1);
        !          1847:        printf("\n");
        !          1848: #endif
        !          1849: }
        !          1850:
        !          1851: HAL_BOOL
        !          1852: ar5k_ar5212_get_diag_state(struct ath_hal *hal, int id, void **device,
        !          1853:     u_int *size)
        !          1854: {
        !          1855:        /*
        !          1856:         * We'll ignore this right now. This seems to be some kind of an obscure
        !          1857:         * debugging interface for the binary-only HAL.
        !          1858:         */
        !          1859:        return (AH_FALSE);
        !          1860: }
        !          1861:
        !          1862: void
        !          1863: ar5k_ar5212_get_lladdr(struct ath_hal *hal, u_int8_t *mac)
        !          1864: {
        !          1865:        bcopy(hal->ah_sta_id, mac, IEEE80211_ADDR_LEN);
        !          1866: }
        !          1867:
        !          1868: HAL_BOOL
        !          1869: ar5k_ar5212_set_lladdr(struct ath_hal *hal, const u_int8_t *mac)
        !          1870: {
        !          1871:        u_int32_t low_id, high_id;
        !          1872:
        !          1873:        /* Set new station ID */
        !          1874:        bcopy(mac, hal->ah_sta_id, IEEE80211_ADDR_LEN);
        !          1875:
        !          1876:        low_id = AR5K_LOW_ID(mac);
        !          1877:        high_id = 0x0000ffff & AR5K_HIGH_ID(mac);
        !          1878:
        !          1879:        AR5K_REG_WRITE(AR5K_AR5212_STA_ID0, low_id);
        !          1880:        AR5K_REG_WRITE(AR5K_AR5212_STA_ID1, high_id);
        !          1881:
        !          1882:        return (AH_TRUE);
        !          1883: }
        !          1884:
        !          1885: HAL_BOOL
        !          1886: ar5k_ar5212_set_regdomain(struct ath_hal *hal, u_int16_t regdomain,
        !          1887:     HAL_STATUS *status)
        !          1888: {
        !          1889:        ieee80211_regdomain_t ieee_regdomain;
        !          1890:
        !          1891:        ieee_regdomain = ar5k_regdomain_to_ieee(regdomain);
        !          1892:
        !          1893:        if (ar5k_eeprom_regulation_domain(hal, AH_TRUE,
        !          1894:                &ieee_regdomain) == AH_TRUE) {
        !          1895:                *status = HAL_OK;
        !          1896:                return (AH_TRUE);
        !          1897:        }
        !          1898:
        !          1899:        *status = EIO;
        !          1900:
        !          1901:        return (AH_FALSE);
        !          1902: }
        !          1903:
        !          1904: void
        !          1905: ar5k_ar5212_set_ledstate(struct ath_hal *hal, HAL_LED_STATE state)
        !          1906: {
        !          1907:        u_int32_t led;
        !          1908:
        !          1909:        AR5K_REG_DISABLE_BITS(AR5K_AR5212_PCICFG,
        !          1910:            AR5K_AR5212_PCICFG_LEDMODE |  AR5K_AR5212_PCICFG_LED);
        !          1911:
        !          1912:        /*
        !          1913:         * Some blinking values, define at your wish
        !          1914:         */
        !          1915:        switch (state) {
        !          1916:        case IEEE80211_S_SCAN:
        !          1917:        case IEEE80211_S_AUTH:
        !          1918:                led = AR5K_AR5212_PCICFG_LEDMODE_PROP |
        !          1919:                    AR5K_AR5212_PCICFG_LED_PEND;
        !          1920:                break;
        !          1921:
        !          1922:        case IEEE80211_S_INIT:
        !          1923:                led = AR5K_AR5212_PCICFG_LEDMODE_PROP |
        !          1924:                    AR5K_AR5212_PCICFG_LED_NONE;
        !          1925:                break;
        !          1926:
        !          1927:        case IEEE80211_S_ASSOC:
        !          1928:        case IEEE80211_S_RUN:
        !          1929:                led = AR5K_AR5212_PCICFG_LEDMODE_PROP |
        !          1930:                    AR5K_AR5212_PCICFG_LED_ASSOC;
        !          1931:                break;
        !          1932:
        !          1933:        default:
        !          1934:                led = AR5K_AR5212_PCICFG_LEDMODE_PROM |
        !          1935:                    AR5K_AR5212_PCICFG_LED_NONE;
        !          1936:                break;
        !          1937:        }
        !          1938:
        !          1939:        AR5K_REG_ENABLE_BITS(AR5K_AR5212_PCICFG, led);
        !          1940: }
        !          1941:
        !          1942: void
        !          1943: ar5k_ar5212_set_associd(struct ath_hal *hal, const u_int8_t *bssid,
        !          1944:     u_int16_t assoc_id, u_int16_t tim_offset)
        !          1945: {
        !          1946:        u_int32_t low_id, high_id;
        !          1947:
        !          1948:        /*
        !          1949:         * Set simple BSSID mask
        !          1950:         */
        !          1951:        AR5K_REG_WRITE(AR5K_AR5212_BSS_IDM0, 0xfffffff);
        !          1952:        AR5K_REG_WRITE(AR5K_AR5212_BSS_IDM1, 0xfffffff);
        !          1953:
        !          1954:        /*
        !          1955:         * Set BSSID which triggers the "SME Join" operation
        !          1956:         */
        !          1957:        low_id = AR5K_LOW_ID(bssid);
        !          1958:        high_id = AR5K_HIGH_ID(bssid);
        !          1959:        AR5K_REG_WRITE(AR5K_AR5212_BSS_ID0, low_id);
        !          1960:        AR5K_REG_WRITE(AR5K_AR5212_BSS_ID1, high_id |
        !          1961:            ((assoc_id & 0x3fff) << AR5K_AR5212_BSS_ID1_AID_S));
        !          1962:        bcopy(bssid, &hal->ah_bssid, IEEE80211_ADDR_LEN);
        !          1963:
        !          1964:        if (assoc_id == 0) {
        !          1965:                ar5k_ar5212_disable_pspoll(hal);
        !          1966:                return;
        !          1967:        }
        !          1968:
        !          1969:        AR5K_REG_WRITE(AR5K_AR5212_BEACON,
        !          1970:            (AR5K_REG_READ(AR5K_AR5212_BEACON) &
        !          1971:            ~AR5K_AR5212_BEACON_TIM) |
        !          1972:            (((tim_offset ? tim_offset + 4 : 0) <<
        !          1973:            AR5K_AR5212_BEACON_TIM_S) &
        !          1974:            AR5K_AR5212_BEACON_TIM));
        !          1975:
        !          1976:        ar5k_ar5212_enable_pspoll(hal, NULL, 0);
        !          1977: }
        !          1978:
        !          1979: HAL_BOOL
        !          1980: ar5k_ar5212_set_bssid_mask(struct ath_hal *hal, const u_int8_t* mask)
        !          1981: {
        !          1982:        u_int32_t low_id, high_id;
        !          1983:
        !          1984:        low_id = AR5K_LOW_ID(mask);
        !          1985:        high_id = 0x0000ffff & AR5K_HIGH_ID(mask);
        !          1986:
        !          1987:        AR5K_REG_WRITE(AR5K_AR5212_BSS_IDM0, low_id);
        !          1988:        AR5K_REG_WRITE(AR5K_AR5212_BSS_IDM1, high_id);
        !          1989:
        !          1990:        return (AH_TRUE);
        !          1991: }
        !          1992:
        !          1993: HAL_BOOL
        !          1994: ar5k_ar5212_set_gpio_output(struct ath_hal *hal, u_int32_t gpio)
        !          1995: {
        !          1996:        if (gpio > AR5K_AR5212_NUM_GPIO)
        !          1997:                return (AH_FALSE);
        !          1998:
        !          1999:        AR5K_REG_WRITE(AR5K_AR5212_GPIOCR,
        !          2000:            (AR5K_REG_READ(AR5K_AR5212_GPIOCR) &~ AR5K_AR5212_GPIOCR_ALL(gpio))
        !          2001:            | AR5K_AR5212_GPIOCR_ALL(gpio));
        !          2002:
        !          2003:        return (AH_TRUE);
        !          2004: }
        !          2005:
        !          2006: HAL_BOOL
        !          2007: ar5k_ar5212_set_gpio_input(struct ath_hal *hal, u_int32_t gpio)
        !          2008: {
        !          2009:        if (gpio > AR5K_AR5212_NUM_GPIO)
        !          2010:                return (AH_FALSE);
        !          2011:
        !          2012:        AR5K_REG_WRITE(AR5K_AR5212_GPIOCR,
        !          2013:            (AR5K_REG_READ(AR5K_AR5212_GPIOCR) &~ AR5K_AR5212_GPIOCR_ALL(gpio))
        !          2014:            | AR5K_AR5212_GPIOCR_NONE(gpio));
        !          2015:
        !          2016:        return (AH_TRUE);
        !          2017: }
        !          2018:
        !          2019: u_int32_t
        !          2020: ar5k_ar5212_get_gpio(struct ath_hal *hal, u_int32_t gpio)
        !          2021: {
        !          2022:        if (gpio > AR5K_AR5212_NUM_GPIO)
        !          2023:                return (0xffffffff);
        !          2024:
        !          2025:        /* GPIO input magic */
        !          2026:        return (((AR5K_REG_READ(AR5K_AR5212_GPIODI) &
        !          2027:            AR5K_AR5212_GPIODI_M) >> gpio) & 0x1);
        !          2028: }
        !          2029:
        !          2030: HAL_BOOL
        !          2031: ar5k_ar5212_set_gpio(struct ath_hal *hal, u_int32_t gpio, u_int32_t val)
        !          2032: {
        !          2033:        u_int32_t data;
        !          2034:
        !          2035:        if (gpio > AR5K_AR5212_NUM_GPIO)
        !          2036:                return (0xffffffff);
        !          2037:
        !          2038:        /* GPIO output magic */
        !          2039:        data =  AR5K_REG_READ(AR5K_AR5212_GPIODO);
        !          2040:
        !          2041:        data &= ~(1 << gpio);
        !          2042:        data |= (val&1) << gpio;
        !          2043:
        !          2044:        AR5K_REG_WRITE(AR5K_AR5212_GPIODO, data);
        !          2045:
        !          2046:        return (AH_TRUE);
        !          2047: }
        !          2048:
        !          2049: void
        !          2050: ar5k_ar5212_set_gpio_intr(struct ath_hal *hal, u_int gpio,
        !          2051:     u_int32_t interrupt_level)
        !          2052: {
        !          2053:        u_int32_t data;
        !          2054:
        !          2055:        if (gpio > AR5K_AR5212_NUM_GPIO)
        !          2056:                return;
        !          2057:
        !          2058:        /*
        !          2059:         * Set the GPIO interrupt
        !          2060:         */
        !          2061:        data = (AR5K_REG_READ(AR5K_AR5212_GPIOCR) &
        !          2062:            ~(AR5K_AR5212_GPIOCR_INT_SEL(gpio) | AR5K_AR5212_GPIOCR_INT_SELH |
        !          2063:            AR5K_AR5212_GPIOCR_INT_ENA | AR5K_AR5212_GPIOCR_ALL(gpio))) |
        !          2064:            (AR5K_AR5212_GPIOCR_INT_SEL(gpio) | AR5K_AR5212_GPIOCR_INT_ENA);
        !          2065:
        !          2066:        AR5K_REG_WRITE(AR5K_AR5212_GPIOCR,
        !          2067:            interrupt_level ? data : (data | AR5K_AR5212_GPIOCR_INT_SELH));
        !          2068:
        !          2069:        hal->ah_imr |= AR5K_AR5212_PIMR_GPIO;
        !          2070:
        !          2071:        /* Enable GPIO interrupts */
        !          2072:        AR5K_REG_ENABLE_BITS(AR5K_AR5212_PIMR, AR5K_AR5212_PIMR_GPIO);
        !          2073: }
        !          2074:
        !          2075: u_int32_t
        !          2076: ar5k_ar5212_get_tsf32(struct ath_hal *hal)
        !          2077: {
        !          2078:        return (AR5K_REG_READ(AR5K_AR5212_TSF_L32));
        !          2079: }
        !          2080:
        !          2081: u_int64_t
        !          2082: ar5k_ar5212_get_tsf64(struct ath_hal *hal)
        !          2083: {
        !          2084:        u_int64_t tsf = AR5K_REG_READ(AR5K_AR5212_TSF_U32);
        !          2085:
        !          2086:        return (AR5K_REG_READ(AR5K_AR5212_TSF_L32) | (tsf << 32));
        !          2087: }
        !          2088:
        !          2089: void
        !          2090: ar5k_ar5212_reset_tsf(struct ath_hal *hal)
        !          2091: {
        !          2092:        AR5K_REG_ENABLE_BITS(AR5K_AR5212_BEACON,
        !          2093:            AR5K_AR5212_BEACON_RESET_TSF);
        !          2094: }
        !          2095:
        !          2096: u_int16_t
        !          2097: ar5k_ar5212_get_regdomain(struct ath_hal *hal)
        !          2098: {
        !          2099:        return (ar5k_get_regdomain(hal));
        !          2100: }
        !          2101:
        !          2102: HAL_BOOL
        !          2103: ar5k_ar5212_detect_card_present(struct ath_hal *hal)
        !          2104: {
        !          2105:        u_int16_t magic;
        !          2106:
        !          2107:        /*
        !          2108:         * Checking the EEPROM's magic value could be an indication
        !          2109:         * if the card is still present. I didn't find another suitable
        !          2110:         * way to do this.
        !          2111:         */
        !          2112:        if (ar5k_ar5212_eeprom_read(hal, AR5K_EEPROM_MAGIC, &magic) != 0)
        !          2113:                return (AH_FALSE);
        !          2114:
        !          2115:        return (magic == AR5K_EEPROM_MAGIC_VALUE ? AH_TRUE : AH_FALSE);
        !          2116: }
        !          2117:
        !          2118: void
        !          2119: ar5k_ar5212_update_mib_counters(struct ath_hal *hal, HAL_MIB_STATS *statistics)
        !          2120: {
        !          2121:        /* Read-And-Clear */
        !          2122:        statistics->ackrcv_bad += AR5K_REG_READ(AR5K_AR5212_ACK_FAIL);
        !          2123:        statistics->rts_bad += AR5K_REG_READ(AR5K_AR5212_RTS_FAIL);
        !          2124:        statistics->rts_good += AR5K_REG_READ(AR5K_AR5212_RTS_OK);
        !          2125:        statistics->fcs_bad += AR5K_REG_READ(AR5K_AR5212_FCS_FAIL);
        !          2126:        statistics->beacons += AR5K_REG_READ(AR5K_AR5212_BEACON_CNT);
        !          2127:
        !          2128:        /* Reset profile count registers */
        !          2129:        AR5K_REG_WRITE(AR5K_AR5212_PROFCNT_TX, 0);
        !          2130:        AR5K_REG_WRITE(AR5K_AR5212_PROFCNT_RX, 0);
        !          2131:        AR5K_REG_WRITE(AR5K_AR5212_PROFCNT_RXCLR, 0);
        !          2132:        AR5K_REG_WRITE(AR5K_AR5212_PROFCNT_CYCLE, 0);
        !          2133: }
        !          2134:
        !          2135: HAL_RFGAIN
        !          2136: ar5k_ar5212_get_rf_gain(struct ath_hal *hal)
        !          2137: {
        !          2138:        u_int32_t data, type;
        !          2139:
        !          2140:        if ((hal->ah_rf_banks == NULL) || (!hal->ah_gain.g_active))
        !          2141:                return (HAL_RFGAIN_INACTIVE);
        !          2142:
        !          2143:        if (hal->ah_rf_gain != HAL_RFGAIN_READ_REQUESTED)
        !          2144:                goto done;
        !          2145:
        !          2146:        data = AR5K_REG_READ(AR5K_AR5212_PHY_PAPD_PROBE);
        !          2147:
        !          2148:        if (!(data & AR5K_AR5212_PHY_PAPD_PROBE_TX_NEXT)) {
        !          2149:                hal->ah_gain.g_current =
        !          2150:                    data >> AR5K_AR5212_PHY_PAPD_PROBE_GAINF_S;
        !          2151:                type = AR5K_REG_MS(data, AR5K_AR5212_PHY_PAPD_PROBE_TYPE);
        !          2152:
        !          2153:                if (type == AR5K_AR5212_PHY_PAPD_PROBE_TYPE_CCK)
        !          2154:                        hal->ah_gain.g_current += AR5K_GAIN_CCK_PROBE_CORR;
        !          2155:
        !          2156:                if (hal->ah_radio == AR5K_AR5112) {
        !          2157:                        ar5k_rfregs_gainf_corr(hal);
        !          2158:                        hal->ah_gain.g_current =
        !          2159:                            hal->ah_gain.g_current >= hal->ah_gain.g_f_corr ?
        !          2160:                            (hal->ah_gain.g_current - hal->ah_gain.g_f_corr) :
        !          2161:                            0;
        !          2162:                }
        !          2163:
        !          2164:                if (ar5k_rfregs_gain_readback(hal) &&
        !          2165:                    AR5K_GAIN_CHECK_ADJUST(&hal->ah_gain) &&
        !          2166:                    ar5k_rfregs_gain_adjust(hal))
        !          2167:                        hal->ah_rf_gain = HAL_RFGAIN_NEED_CHANGE;
        !          2168:        }
        !          2169:
        !          2170:  done:
        !          2171:        return (hal->ah_rf_gain);
        !          2172: }
        !          2173:
        !          2174: HAL_BOOL
        !          2175: ar5k_ar5212_set_slot_time(struct ath_hal *hal, u_int slot_time)
        !          2176: {
        !          2177:        if (slot_time < HAL_SLOT_TIME_9 || slot_time > HAL_SLOT_TIME_MAX)
        !          2178:                return (AH_FALSE);
        !          2179:
        !          2180:        AR5K_REG_WRITE(AR5K_AR5212_DCU_GBL_IFS_SLOT, slot_time);
        !          2181:
        !          2182:        return (AH_TRUE);
        !          2183: }
        !          2184:
        !          2185: u_int
        !          2186: ar5k_ar5212_get_slot_time(struct ath_hal *hal)
        !          2187: {
        !          2188:        return (AR5K_REG_READ(AR5K_AR5212_DCU_GBL_IFS_SLOT) & 0xffff);
        !          2189: }
        !          2190:
        !          2191: HAL_BOOL
        !          2192: ar5k_ar5212_set_ack_timeout(struct ath_hal *hal, u_int timeout)
        !          2193: {
        !          2194:        if (ar5k_clocktoh(AR5K_REG_MS(0xffffffff, AR5K_AR5212_TIME_OUT_ACK),
        !          2195:            hal->ah_turbo) <= timeout)
        !          2196:                return (AH_FALSE);
        !          2197:
        !          2198:        AR5K_REG_WRITE_BITS(AR5K_AR5212_TIME_OUT, AR5K_AR5212_TIME_OUT_ACK,
        !          2199:            ar5k_htoclock(timeout, hal->ah_turbo));
        !          2200:
        !          2201:        return (AH_TRUE);
        !          2202: }
        !          2203:
        !          2204: u_int
        !          2205: ar5k_ar5212_get_ack_timeout(struct ath_hal *hal)
        !          2206: {
        !          2207:        return (ar5k_clocktoh(AR5K_REG_MS(AR5K_REG_READ(AR5K_AR5212_TIME_OUT),
        !          2208:            AR5K_AR5212_TIME_OUT_ACK), hal->ah_turbo));
        !          2209: }
        !          2210:
        !          2211: HAL_BOOL
        !          2212: ar5k_ar5212_set_cts_timeout(struct ath_hal *hal, u_int timeout)
        !          2213: {
        !          2214:        if (ar5k_clocktoh(AR5K_REG_MS(0xffffffff, AR5K_AR5212_TIME_OUT_CTS),
        !          2215:            hal->ah_turbo) <= timeout)
        !          2216:                return (AH_FALSE);
        !          2217:
        !          2218:        AR5K_REG_WRITE_BITS(AR5K_AR5212_TIME_OUT, AR5K_AR5212_TIME_OUT_CTS,
        !          2219:            ar5k_htoclock(timeout, hal->ah_turbo));
        !          2220:
        !          2221:        return (AH_TRUE);
        !          2222: }
        !          2223:
        !          2224: u_int
        !          2225: ar5k_ar5212_get_cts_timeout(struct ath_hal *hal)
        !          2226: {
        !          2227:        return (ar5k_clocktoh(AR5K_REG_MS(AR5K_REG_READ(AR5K_AR5212_TIME_OUT),
        !          2228:            AR5K_AR5212_TIME_OUT_CTS), hal->ah_turbo));
        !          2229: }
        !          2230:
        !          2231: /*
        !          2232:  * Key table (WEP) functions
        !          2233:  */
        !          2234:
        !          2235: HAL_BOOL
        !          2236: ar5k_ar5212_is_cipher_supported(struct ath_hal *hal, HAL_CIPHER cipher)
        !          2237: {
        !          2238:        /*
        !          2239:         * The AR5212 only supports WEP
        !          2240:         */
        !          2241:        if (cipher == HAL_CIPHER_WEP)
        !          2242:                return (AH_TRUE);
        !          2243:
        !          2244:        return (AH_FALSE);
        !          2245: }
        !          2246:
        !          2247: u_int32_t
        !          2248: ar5k_ar5212_get_keycache_size(struct ath_hal *hal)
        !          2249: {
        !          2250:        return (AR5K_AR5212_KEYCACHE_SIZE);
        !          2251: }
        !          2252:
        !          2253: HAL_BOOL
        !          2254: ar5k_ar5212_reset_key(struct ath_hal *hal, u_int16_t entry)
        !          2255: {
        !          2256:        int i;
        !          2257:
        !          2258:        AR5K_ASSERT_ENTRY(entry, AR5K_AR5212_KEYTABLE_SIZE);
        !          2259:
        !          2260:        for (i = 0; i < AR5K_AR5212_KEYCACHE_SIZE; i++)
        !          2261:                AR5K_REG_WRITE(AR5K_AR5212_KEYTABLE_OFF(entry, i), 0);
        !          2262:
        !          2263:        /* Set NULL encryption */
        !          2264:        AR5K_REG_WRITE(AR5K_AR5212_KEYTABLE_TYPE(entry),
        !          2265:            AR5K_AR5212_KEYTABLE_TYPE_NULL);
        !          2266:
        !          2267:        return (AH_FALSE);
        !          2268: }
        !          2269:
        !          2270: HAL_BOOL
        !          2271: ar5k_ar5212_is_key_valid(struct ath_hal *hal, u_int16_t entry)
        !          2272: {
        !          2273:        AR5K_ASSERT_ENTRY(entry, AR5K_AR5212_KEYTABLE_SIZE);
        !          2274:
        !          2275:        /*
        !          2276:         * Check the validation flag at the end of the entry
        !          2277:         */
        !          2278:        if (AR5K_REG_READ(AR5K_AR5212_KEYTABLE_MAC1(entry)) &
        !          2279:            AR5K_AR5212_KEYTABLE_VALID)
        !          2280:                return (AH_TRUE);
        !          2281:
        !          2282:        return (AH_FALSE);
        !          2283: }
        !          2284:
        !          2285: HAL_BOOL
        !          2286: ar5k_ar5212_set_key(struct ath_hal *hal, u_int16_t entry,
        !          2287:     const HAL_KEYVAL *keyval, const u_int8_t *mac, int xor_notused)
        !          2288: {
        !          2289:        int i;
        !          2290:        u_int32_t key_v[AR5K_AR5212_KEYCACHE_SIZE - 2];
        !          2291:
        !          2292:        AR5K_ASSERT_ENTRY(entry, AR5K_AR5212_KEYTABLE_SIZE);
        !          2293:
        !          2294:        bzero(&key_v, sizeof(key_v));
        !          2295:
        !          2296:        switch (keyval->wk_len) {
        !          2297:        case AR5K_KEYVAL_LENGTH_40:
        !          2298:                bcopy(keyval->wk_key, &key_v[0], 4);
        !          2299:                bcopy(keyval->wk_key + 4, &key_v[1], 1);
        !          2300:                key_v[5] = AR5K_AR5212_KEYTABLE_TYPE_40;
        !          2301:                break;
        !          2302:
        !          2303:        case AR5K_KEYVAL_LENGTH_104:
        !          2304:                bcopy(keyval->wk_key, &key_v[0], 4);
        !          2305:                bcopy(keyval->wk_key + 4, &key_v[1], 2);
        !          2306:                bcopy(keyval->wk_key + 6, &key_v[2], 4);
        !          2307:                bcopy(keyval->wk_key + 10, &key_v[3], 2);
        !          2308:                bcopy(keyval->wk_key + 12, &key_v[4], 1);
        !          2309:                key_v[5] = AR5K_AR5212_KEYTABLE_TYPE_104;
        !          2310:                break;
        !          2311:
        !          2312:        case AR5K_KEYVAL_LENGTH_128:
        !          2313:                bcopy(keyval->wk_key, &key_v[0], 4);
        !          2314:                bcopy(keyval->wk_key + 4, &key_v[1], 2);
        !          2315:                bcopy(keyval->wk_key + 6, &key_v[2], 4);
        !          2316:                bcopy(keyval->wk_key + 10, &key_v[3], 2);
        !          2317:                bcopy(keyval->wk_key + 12, &key_v[4], 4);
        !          2318:                key_v[5] = AR5K_AR5212_KEYTABLE_TYPE_128;
        !          2319:                break;
        !          2320:
        !          2321:        default:
        !          2322:                /* Unsupported key length (not WEP40/104/128) */
        !          2323:                return (AH_FALSE);
        !          2324:        }
        !          2325:
        !          2326:        for (i = 0; i < AR5K_ELEMENTS(key_v); i++)
        !          2327:                AR5K_REG_WRITE(AR5K_AR5212_KEYTABLE_OFF(entry, i), key_v[i]);
        !          2328:
        !          2329:        return (ar5k_ar5212_set_key_lladdr(hal, entry, mac));
        !          2330: }
        !          2331:
        !          2332: HAL_BOOL
        !          2333: ar5k_ar5212_set_key_lladdr(struct ath_hal *hal, u_int16_t entry,
        !          2334:     const u_int8_t *mac)
        !          2335: {
        !          2336:        u_int32_t low_id, high_id;
        !          2337:        const u_int8_t *mac_v;
        !          2338:
        !          2339:        /*
        !          2340:         * Invalid entry (key table overflow)
        !          2341:         */
        !          2342:        AR5K_ASSERT_ENTRY(entry, AR5K_AR5212_KEYTABLE_SIZE);
        !          2343:
        !          2344:        /* MAC may be NULL if it's a broadcast key */
        !          2345:        mac_v = mac == NULL ? etherbroadcastaddr : mac;
        !          2346:
        !          2347:        low_id = AR5K_LOW_ID(mac_v);
        !          2348:        high_id = AR5K_HIGH_ID(mac_v) | AR5K_AR5212_KEYTABLE_VALID;
        !          2349:
        !          2350:        AR5K_REG_WRITE(AR5K_AR5212_KEYTABLE_MAC0(entry), low_id);
        !          2351:        AR5K_REG_WRITE(AR5K_AR5212_KEYTABLE_MAC1(entry), high_id);
        !          2352:
        !          2353:        return (AH_TRUE);
        !          2354: }
        !          2355:
        !          2356: /*
        !          2357:  * Power management functions
        !          2358:  */
        !          2359:
        !          2360: HAL_BOOL
        !          2361: ar5k_ar5212_set_power(struct ath_hal *hal, HAL_POWER_MODE mode,
        !          2362:     HAL_BOOL set_chip, u_int16_t sleep_duration)
        !          2363: {
        !          2364:        u_int32_t staid;
        !          2365:        int i;
        !          2366:
        !          2367:        staid = AR5K_REG_READ(AR5K_AR5212_STA_ID1);
        !          2368:
        !          2369:        switch (mode) {
        !          2370:        case HAL_PM_AUTO:
        !          2371:                staid &= ~AR5K_AR5212_STA_ID1_DEFAULT_ANTENNA;
        !          2372:                /* FALLTHROUGH */
        !          2373:        case HAL_PM_NETWORK_SLEEP:
        !          2374:                if (set_chip == AH_TRUE) {
        !          2375:                        AR5K_REG_WRITE(AR5K_AR5212_SCR,
        !          2376:                            AR5K_AR5212_SCR_SLE | sleep_duration);
        !          2377:                }
        !          2378:                staid |= AR5K_AR5212_STA_ID1_PWR_SV;
        !          2379:                break;
        !          2380:
        !          2381:        case HAL_PM_FULL_SLEEP:
        !          2382:                if (set_chip == AH_TRUE) {
        !          2383:                        AR5K_REG_WRITE(AR5K_AR5212_SCR,
        !          2384:                            AR5K_AR5212_SCR_SLE_SLP);
        !          2385:                }
        !          2386:                staid |= AR5K_AR5212_STA_ID1_PWR_SV;
        !          2387:                break;
        !          2388:
        !          2389:        case HAL_PM_AWAKE:
        !          2390:                if (set_chip == AH_FALSE)
        !          2391:                        goto commit;
        !          2392:
        !          2393:                AR5K_REG_WRITE(AR5K_AR5212_SCR, AR5K_AR5212_SCR_SLE_WAKE);
        !          2394:
        !          2395:                for (i = 5000; i > 0; i--) {
        !          2396:                        /* Check if the AR5212 did wake up */
        !          2397:                        if ((AR5K_REG_READ(AR5K_AR5212_PCICFG) &
        !          2398:                            AR5K_AR5212_PCICFG_SPWR_DN) == 0)
        !          2399:                                break;
        !          2400:
        !          2401:                        /* Wait a bit and retry */
        !          2402:                        AR5K_DELAY(200);
        !          2403:                        AR5K_REG_WRITE(AR5K_AR5212_SCR,
        !          2404:                            AR5K_AR5212_SCR_SLE_WAKE);
        !          2405:                }
        !          2406:
        !          2407:                /* Fail if the AR5212 didn't wake up */
        !          2408:                if (i <= 0)
        !          2409:                        return (AH_FALSE);
        !          2410:
        !          2411:                staid &= ~AR5K_AR5212_STA_ID1_PWR_SV;
        !          2412:                break;
        !          2413:
        !          2414:        default:
        !          2415:                return (AH_FALSE);
        !          2416:        }
        !          2417:
        !          2418:  commit:
        !          2419:        hal->ah_power_mode = mode;
        !          2420:
        !          2421:        AR5K_REG_WRITE(AR5K_AR5212_STA_ID1, staid);
        !          2422:
        !          2423:        return (AH_TRUE);
        !          2424: }
        !          2425:
        !          2426: HAL_POWER_MODE
        !          2427: ar5k_ar5212_get_power_mode(struct ath_hal *hal)
        !          2428: {
        !          2429:        return (hal->ah_power_mode);
        !          2430: }
        !          2431:
        !          2432: HAL_BOOL
        !          2433: ar5k_ar5212_query_pspoll_support(struct ath_hal *hal)
        !          2434: {
        !          2435:        /* nope */
        !          2436:        return (AH_FALSE);
        !          2437: }
        !          2438:
        !          2439: HAL_BOOL
        !          2440: ar5k_ar5212_init_pspoll(struct ath_hal *hal)
        !          2441: {
        !          2442:        /*
        !          2443:         * Not used on the AR5212
        !          2444:         */
        !          2445:        return (AH_FALSE);
        !          2446: }
        !          2447:
        !          2448: HAL_BOOL
        !          2449: ar5k_ar5212_enable_pspoll(struct ath_hal *hal, u_int8_t *bssid,
        !          2450:     u_int16_t assoc_id)
        !          2451: {
        !          2452:        return (AH_FALSE);
        !          2453: }
        !          2454:
        !          2455: HAL_BOOL
        !          2456: ar5k_ar5212_disable_pspoll(struct ath_hal *hal)
        !          2457: {
        !          2458:        return (AH_FALSE);
        !          2459: }
        !          2460:
        !          2461: /*
        !          2462:  * Beacon functions
        !          2463:  */
        !          2464:
        !          2465: void
        !          2466: ar5k_ar5212_init_beacon(struct ath_hal *hal, u_int32_t next_beacon,
        !          2467:     u_int32_t interval)
        !          2468: {
        !          2469:        u_int32_t timer1, timer2, timer3;
        !          2470:
        !          2471:        /*
        !          2472:         * Set the additional timers by mode
        !          2473:         */
        !          2474:        switch (hal->ah_op_mode) {
        !          2475:        case HAL_M_STA:
        !          2476:                timer1 = 0x0000ffff;
        !          2477:                timer2 = 0x0007ffff;
        !          2478:                break;
        !          2479:
        !          2480:        default:
        !          2481:                timer1 = (next_beacon - AR5K_TUNE_DMA_BEACON_RESP) <<
        !          2482:                    0x00000003;
        !          2483:                timer2 = (next_beacon - AR5K_TUNE_SW_BEACON_RESP) <<
        !          2484:                    0x00000003;
        !          2485:        }
        !          2486:
        !          2487:        timer3 = next_beacon +
        !          2488:            (hal->ah_atim_window ? hal->ah_atim_window : 1);
        !          2489:
        !          2490:        /*
        !          2491:         * Enable all timers and set the beacon register
        !          2492:         * (next beacon, DMA beacon, software beacon, ATIM window time)
        !          2493:         */
        !          2494:        AR5K_REG_WRITE(AR5K_AR5212_TIMER0, next_beacon);
        !          2495:        AR5K_REG_WRITE(AR5K_AR5212_TIMER1, timer1);
        !          2496:        AR5K_REG_WRITE(AR5K_AR5212_TIMER2, timer2);
        !          2497:        AR5K_REG_WRITE(AR5K_AR5212_TIMER3, timer3);
        !          2498:
        !          2499:        AR5K_REG_WRITE(AR5K_AR5212_BEACON, interval &
        !          2500:            (AR5K_AR5212_BEACON_PERIOD | AR5K_AR5212_BEACON_RESET_TSF |
        !          2501:            AR5K_AR5212_BEACON_ENABLE));
        !          2502: }
        !          2503:
        !          2504: void
        !          2505: ar5k_ar5212_set_beacon_timers(struct ath_hal *hal,
        !          2506:     const HAL_BEACON_STATE *state, u_int32_t tsf, u_int32_t dtim_count,
        !          2507:     u_int32_t cfp_count)
        !          2508: {
        !          2509:        u_int32_t cfp_period, next_cfp, dtim, interval, next_beacon;
        !          2510:
        !          2511:        /* Return on an invalid beacon state */
        !          2512:        if (state->bs_interval < 1)
        !          2513:                return;
        !          2514:
        !          2515:        interval = state->bs_intval;
        !          2516:        dtim = state->bs_dtimperiod;
        !          2517:
        !          2518:        /*
        !          2519:         * PCF support?
        !          2520:         */
        !          2521:        if (state->bs_cfp_period > 0) {
        !          2522:                /* Enable CFP mode and set the CFP and timer registers */
        !          2523:                cfp_period = state->bs_cfp_period * state->bs_dtim_period *
        !          2524:                    state->bs_interval;
        !          2525:                next_cfp = (cfp_count * state->bs_dtim_period + dtim_count) *
        !          2526:                    state->bs_interval;
        !          2527:
        !          2528:                AR5K_REG_ENABLE_BITS(AR5K_AR5212_STA_ID1,
        !          2529:                    AR5K_AR5212_STA_ID1_PCF);
        !          2530:                AR5K_REG_WRITE(AR5K_AR5212_CFP_PERIOD, cfp_period);
        !          2531:                AR5K_REG_WRITE(AR5K_AR5212_CFP_DUR, state->bs_cfp_max_duration);
        !          2532:                AR5K_REG_WRITE(AR5K_AR5212_TIMER2,
        !          2533:                    (tsf + (next_cfp == 0 ? cfp_period : next_cfp)) << 3);
        !          2534:        } else {
        !          2535:                /* Disable PCF mode */
        !          2536:                AR5K_REG_DISABLE_BITS(AR5K_AR5212_STA_ID1,
        !          2537:                    AR5K_AR5212_STA_ID1_PCF);
        !          2538:        }
        !          2539:
        !          2540:        /*
        !          2541:         * Enable the beacon timer register
        !          2542:         */
        !          2543:        AR5K_REG_WRITE(AR5K_AR5212_TIMER0, state->bs_next_beacon);
        !          2544:
        !          2545:        /*
        !          2546:         * Start the beacon timers
        !          2547:         */
        !          2548:        AR5K_REG_WRITE(AR5K_AR5212_BEACON,
        !          2549:            (AR5K_REG_READ(AR5K_AR5212_BEACON) &~
        !          2550:            (AR5K_AR5212_BEACON_PERIOD | AR5K_AR5212_BEACON_TIM)) |
        !          2551:            AR5K_REG_SM(state->bs_tim_offset ? state->bs_tim_offset + 4 : 0,
        !          2552:            AR5K_AR5212_BEACON_TIM) | AR5K_REG_SM(state->bs_interval,
        !          2553:            AR5K_AR5212_BEACON_PERIOD));
        !          2554:
        !          2555:        /*
        !          2556:         * Write new beacon miss threshold, if it appears to be valid
        !          2557:         */
        !          2558:        if ((AR5K_AR5212_RSSI_THR_BMISS >> AR5K_AR5212_RSSI_THR_BMISS_S) <
        !          2559:            state->bs_bmiss_threshold)
        !          2560:                return;
        !          2561:
        !          2562:        AR5K_REG_WRITE_BITS(AR5K_AR5212_RSSI_THR_M,
        !          2563:            AR5K_AR5212_RSSI_THR_BMISS, state->bs_bmiss_threshold);
        !          2564:
        !          2565:        /*
        !          2566:         * Set sleep registers
        !          2567:         */
        !          2568:        if ((state->bs_sleepduration > state->bs_interval) &&
        !          2569:            (roundup(state->bs_sleepduration, interval) ==
        !          2570:            state->bs_sleepduration))
        !          2571:                interval = state->bs_sleepduration;
        !          2572:
        !          2573:        if (state->bs_sleepduration > dtim &&
        !          2574:            (dtim == 0 || roundup(state->bs_sleepduration, dtim) ==
        !          2575:            state->bs_sleepduration))
        !          2576:                dtim = state->bs_sleepduration;
        !          2577:
        !          2578:        if (interval > dtim)
        !          2579:                return;
        !          2580:
        !          2581:        next_beacon = interval == dtim ?
        !          2582:            state->bs_nextdtim: state->bs_nexttbtt;
        !          2583:
        !          2584:        AR5K_REG_WRITE(AR5K_AR5212_SLEEP0,
        !          2585:            AR5K_REG_SM((state->bs_nextdtim - 3) << 3,
        !          2586:            AR5K_AR5212_SLEEP0_NEXT_DTIM) |
        !          2587:            AR5K_REG_SM(10, AR5K_AR5212_SLEEP0_CABTO) |
        !          2588:            AR5K_AR5212_SLEEP0_ENH_SLEEP_EN |
        !          2589:            AR5K_AR5212_SLEEP0_ASSUME_DTIM);
        !          2590:        AR5K_REG_WRITE(AR5K_AR5212_SLEEP1,
        !          2591:            AR5K_REG_SM((next_beacon - 3) << 3,
        !          2592:            AR5K_AR5212_SLEEP1_NEXT_TIM) |
        !          2593:            AR5K_REG_SM(10, AR5K_AR5212_SLEEP1_BEACON_TO));
        !          2594:        AR5K_REG_WRITE(AR5K_AR5212_SLEEP2,
        !          2595:            AR5K_REG_SM(interval, AR5K_AR5212_SLEEP2_TIM_PER) |
        !          2596:            AR5K_REG_SM(dtim, AR5K_AR5212_SLEEP2_DTIM_PER));
        !          2597: }
        !          2598:
        !          2599: void
        !          2600: ar5k_ar5212_reset_beacon(struct ath_hal *hal)
        !          2601: {
        !          2602:        /*
        !          2603:         * Disable beacon timer
        !          2604:         */
        !          2605:        AR5K_REG_WRITE(AR5K_AR5212_TIMER0, 0);
        !          2606:
        !          2607:        /*
        !          2608:         * Disable some beacon register values
        !          2609:         */
        !          2610:        AR5K_REG_DISABLE_BITS(AR5K_AR5212_STA_ID1,
        !          2611:            AR5K_AR5212_STA_ID1_DEFAULT_ANTENNA | AR5K_AR5212_STA_ID1_PCF);
        !          2612:        AR5K_REG_WRITE(AR5K_AR5212_BEACON, AR5K_AR5212_BEACON_PERIOD);
        !          2613: }
        !          2614:
        !          2615: HAL_BOOL
        !          2616: ar5k_ar5212_wait_for_beacon(struct ath_hal *hal, bus_addr_t phys_addr)
        !          2617: {
        !          2618:        HAL_BOOL ret;
        !          2619:
        !          2620:        /*
        !          2621:         * Wait for beaconn queue to be done
        !          2622:         */
        !          2623:        ret = ar5k_register_timeout(hal,
        !          2624:            AR5K_AR5212_QCU_STS(HAL_TX_QUEUE_ID_BEACON),
        !          2625:            AR5K_AR5212_QCU_STS_FRMPENDCNT, 0, AH_FALSE);
        !          2626:
        !          2627:        if (AR5K_REG_READ_Q(AR5K_AR5212_QCU_TXE, HAL_TX_QUEUE_ID_BEACON))
        !          2628:                return (AH_FALSE);
        !          2629:
        !          2630:        return (ret);
        !          2631: }
        !          2632:
        !          2633: /*
        !          2634:  * Interrupt handling
        !          2635:  */
        !          2636:
        !          2637: HAL_BOOL
        !          2638: ar5k_ar5212_is_intr_pending(struct ath_hal *hal)
        !          2639: {
        !          2640:        return (AR5K_REG_READ(AR5K_AR5212_INTPEND) == 0 ? AH_FALSE : AH_TRUE);
        !          2641: }
        !          2642:
        !          2643: HAL_BOOL
        !          2644: ar5k_ar5212_get_isr(struct ath_hal *hal, u_int32_t *interrupt_mask)
        !          2645: {
        !          2646:        u_int32_t data;
        !          2647:
        !          2648:        /*
        !          2649:         * Read interrupt status from the Read-And-Clear shadow register
        !          2650:         */
        !          2651:        data = AR5K_REG_READ(AR5K_AR5212_RAC_PISR);
        !          2652:
        !          2653:        /*
        !          2654:         * Get abstract interrupt mask (HAL-compatible)
        !          2655:         */
        !          2656:        *interrupt_mask = (data & HAL_INT_COMMON) & hal->ah_imr;
        !          2657:
        !          2658:        if (data == HAL_INT_NOCARD)
        !          2659:                return (AH_FALSE);
        !          2660:
        !          2661:        if (data & (AR5K_AR5212_PISR_RXOK | AR5K_AR5212_PISR_RXERR))
        !          2662:                *interrupt_mask |= HAL_INT_RX;
        !          2663:
        !          2664:        if (data & (AR5K_AR5212_PISR_TXOK | AR5K_AR5212_PISR_TXERR))
        !          2665:                *interrupt_mask |= HAL_INT_TX;
        !          2666:
        !          2667:        if (data & (AR5K_AR5212_PISR_HIUERR))
        !          2668:                *interrupt_mask |= HAL_INT_FATAL;
        !          2669:
        !          2670:        /*
        !          2671:         * Special interrupt handling (not caught by the driver)
        !          2672:         */
        !          2673:        if (((*interrupt_mask) & AR5K_AR5212_PISR_RXPHY) &&
        !          2674:            hal->ah_radar.r_enabled == AH_TRUE)
        !          2675:                ar5k_radar_alert(hal);
        !          2676:
        !          2677:        if (*interrupt_mask == 0)
        !          2678:                AR5K_PRINTF("0x%08x\n", data);
        !          2679:
        !          2680:        return (AH_TRUE);
        !          2681: }
        !          2682:
        !          2683: u_int32_t
        !          2684: ar5k_ar5212_get_intr(struct ath_hal *hal)
        !          2685: {
        !          2686:        /* Return the interrupt mask stored previously */
        !          2687:        return (hal->ah_imr);
        !          2688: }
        !          2689:
        !          2690: HAL_INT
        !          2691: ar5k_ar5212_set_intr(struct ath_hal *hal, HAL_INT new_mask)
        !          2692: {
        !          2693:        HAL_INT old_mask, int_mask;
        !          2694:
        !          2695:        /*
        !          2696:         * Disable card interrupts to prevent any race conditions
        !          2697:         * (they will be re-enabled afterwards).
        !          2698:         */
        !          2699:        AR5K_REG_WRITE(AR5K_AR5212_IER, AR5K_AR5212_IER_DISABLE);
        !          2700:
        !          2701:        old_mask = hal->ah_imr;
        !          2702:
        !          2703:        /*
        !          2704:         * Add additional, chipset-dependent interrupt mask flags
        !          2705:         * and write them to the IMR (interrupt mask register).
        !          2706:         */
        !          2707:        int_mask = new_mask & HAL_INT_COMMON;
        !          2708:
        !          2709:        if (new_mask & HAL_INT_RX)
        !          2710:                int_mask |=
        !          2711:                    AR5K_AR5212_PIMR_RXOK |
        !          2712:                    AR5K_AR5212_PIMR_RXERR |
        !          2713:                    AR5K_AR5212_PIMR_RXORN |
        !          2714:                    AR5K_AR5212_PIMR_RXDESC;
        !          2715:
        !          2716:        if (new_mask & HAL_INT_TX)
        !          2717:                int_mask |=
        !          2718:                    AR5K_AR5212_PIMR_TXOK |
        !          2719:                    AR5K_AR5212_PIMR_TXERR |
        !          2720:                    AR5K_AR5212_PIMR_TXDESC |
        !          2721:                    AR5K_AR5212_PIMR_TXURN;
        !          2722:
        !          2723:        if (new_mask & HAL_INT_FATAL) {
        !          2724:                int_mask |= AR5K_AR5212_PIMR_HIUERR;
        !          2725:                AR5K_REG_ENABLE_BITS(AR5K_AR5212_SIMR2,
        !          2726:                    AR5K_AR5212_SIMR2_MCABT |
        !          2727:                    AR5K_AR5212_SIMR2_SSERR |
        !          2728:                    AR5K_AR5212_SIMR2_DPERR);
        !          2729:        }
        !          2730:
        !          2731:        AR5K_REG_WRITE(AR5K_AR5212_PIMR, int_mask);
        !          2732:
        !          2733:        /* Store new interrupt mask */
        !          2734:        hal->ah_imr = new_mask;
        !          2735:
        !          2736:        /* ..re-enable interrupts */
        !          2737:        AR5K_REG_WRITE(AR5K_AR5212_IER, AR5K_AR5212_IER_ENABLE);
        !          2738:
        !          2739:        return (old_mask);
        !          2740: }
        !          2741:
        !          2742: /*
        !          2743:  * Misc internal functions
        !          2744:  */
        !          2745:
        !          2746: HAL_BOOL
        !          2747: ar5k_ar5212_get_capabilities(struct ath_hal *hal)
        !          2748: {
        !          2749:        u_int16_t ee_header;
        !          2750:
        !          2751:        /* Capabilities stored in the EEPROM */
        !          2752:        ee_header = hal->ah_capabilities.cap_eeprom.ee_header;
        !          2753:
        !          2754:        /*
        !          2755:         * XXX The AR5212 tranceiver supports frequencies from 4920 to 6100GHz
        !          2756:         * XXX and from 2312 to 2732GHz. There are problems with the current
        !          2757:         * XXX ieee80211 implementation because the IEEE channel mapping
        !          2758:         * XXX does not support negative channel numbers (2312MHz is channel
        !          2759:         * XXX -19). Of course, this doesn't matter because these channels
        !          2760:         * XXX are out of range but some regulation domains like MKK (Japan)
        !          2761:         * XXX will support frequencies somewhere around 4.8GHz.
        !          2762:         */
        !          2763:
        !          2764:        /*
        !          2765:         * Set radio capabilities
        !          2766:         */
        !          2767:
        !          2768:        if (AR5K_EEPROM_HDR_11A(ee_header)) {
        !          2769:                hal->ah_capabilities.cap_range.range_5ghz_min = 5005; /* 4920 */
        !          2770:                hal->ah_capabilities.cap_range.range_5ghz_max = 6100;
        !          2771:
        !          2772:                /* Set supported modes */
        !          2773:                hal->ah_capabilities.cap_mode =
        !          2774:                    HAL_MODE_11A | HAL_MODE_TURBO | HAL_MODE_XR;
        !          2775:        }
        !          2776:
        !          2777:        /* This chip will support 802.11b if the 2GHz radio is connected */
        !          2778:        if (AR5K_EEPROM_HDR_11B(ee_header) || AR5K_EEPROM_HDR_11G(ee_header)) {
        !          2779:                hal->ah_capabilities.cap_range.range_2ghz_min = 2412; /* 2312 */
        !          2780:                hal->ah_capabilities.cap_range.range_2ghz_max = 2732;
        !          2781:
        !          2782:                if (AR5K_EEPROM_HDR_11B(ee_header))
        !          2783:                        hal->ah_capabilities.cap_mode |= HAL_MODE_11B;
        !          2784: #if 0
        !          2785:                if (AR5K_EEPROM_HDR_11G(ee_header))
        !          2786:                        hal->ah_capabilities.cap_mode |= HAL_MODE_11G;
        !          2787: #endif
        !          2788:        }
        !          2789:
        !          2790:        /* GPIO */
        !          2791:        hal->ah_gpio_npins = AR5K_AR5212_NUM_GPIO;
        !          2792:
        !          2793:        /* Set number of supported TX queues */
        !          2794:        hal->ah_capabilities.cap_queues.q_tx_num = AR5K_AR5212_TX_NUM_QUEUES;
        !          2795:
        !          2796:        return (AH_TRUE);
        !          2797: }
        !          2798:
        !          2799: void
        !          2800: ar5k_ar5212_radar_alert(struct ath_hal *hal, HAL_BOOL enable)
        !          2801: {
        !          2802:        /*
        !          2803:         * Enable radar detection
        !          2804:         */
        !          2805:        AR5K_REG_WRITE(AR5K_AR5212_IER, AR5K_AR5212_IER_DISABLE);
        !          2806:
        !          2807:        if (enable == AH_TRUE) {
        !          2808:                AR5K_REG_WRITE(AR5K_AR5212_PHY_RADAR,
        !          2809:                    AR5K_AR5212_PHY_RADAR_ENABLE);
        !          2810:                AR5K_REG_ENABLE_BITS(AR5K_AR5212_PIMR,
        !          2811:                    AR5K_AR5212_PIMR_RXPHY);
        !          2812:        } else {
        !          2813:                AR5K_REG_WRITE(AR5K_AR5212_PHY_RADAR,
        !          2814:                    AR5K_AR5212_PHY_RADAR_DISABLE);
        !          2815:                AR5K_REG_DISABLE_BITS(AR5K_AR5212_PIMR,
        !          2816:                    AR5K_AR5212_PIMR_RXPHY);
        !          2817:        }
        !          2818:
        !          2819:        AR5K_REG_WRITE(AR5K_AR5212_IER, AR5K_AR5212_IER_ENABLE);
        !          2820: }
        !          2821:
        !          2822: /*
        !          2823:  * EEPROM access functions
        !          2824:  */
        !          2825:
        !          2826: HAL_BOOL
        !          2827: ar5k_ar5212_eeprom_is_busy(struct ath_hal *hal)
        !          2828: {
        !          2829:        return (AR5K_REG_READ(AR5K_AR5212_CFG) & AR5K_AR5212_CFG_EEBS ?
        !          2830:            AH_TRUE : AH_FALSE);
        !          2831: }
        !          2832:
        !          2833: int
        !          2834: ar5k_ar5212_eeprom_read(struct ath_hal *hal, u_int32_t offset, u_int16_t *data)
        !          2835: {
        !          2836:        u_int32_t status, i;
        !          2837:
        !          2838:        /*
        !          2839:         * Initialize EEPROM access
        !          2840:         */
        !          2841:        AR5K_REG_WRITE(AR5K_AR5212_EEPROM_BASE, (u_int8_t)offset);
        !          2842:        AR5K_REG_ENABLE_BITS(AR5K_AR5212_EEPROM_CMD,
        !          2843:            AR5K_AR5212_EEPROM_CMD_READ);
        !          2844:
        !          2845:        for (i = AR5K_TUNE_REGISTER_TIMEOUT; i > 0; i--) {
        !          2846:                status = AR5K_REG_READ(AR5K_AR5212_EEPROM_STATUS);
        !          2847:                if (status & AR5K_AR5212_EEPROM_STAT_RDDONE) {
        !          2848:                        if (status & AR5K_AR5212_EEPROM_STAT_RDERR)
        !          2849:                                return (EIO);
        !          2850:                        *data = (u_int16_t)
        !          2851:                            (AR5K_REG_READ(AR5K_AR5212_EEPROM_DATA) & 0xffff);
        !          2852:                        return (0);
        !          2853:                }
        !          2854:                AR5K_DELAY(15);
        !          2855:        }
        !          2856:
        !          2857:        return (ETIMEDOUT);
        !          2858: }
        !          2859:
        !          2860: int
        !          2861: ar5k_ar5212_eeprom_write(struct ath_hal *hal, u_int32_t offset, u_int16_t data)
        !          2862: {
        !          2863:        u_int32_t status, timeout;
        !          2864:
        !          2865:        /* Enable eeprom access */
        !          2866:        AR5K_REG_ENABLE_BITS(AR5K_AR5212_EEPROM_CMD,
        !          2867:            AR5K_AR5212_EEPROM_CMD_RESET);
        !          2868:        AR5K_REG_ENABLE_BITS(AR5K_AR5212_EEPROM_CMD,
        !          2869:            AR5K_AR5212_EEPROM_CMD_WRITE);
        !          2870:
        !          2871:        /*
        !          2872:         * Prime write pump
        !          2873:         */
        !          2874:        AR5K_REG_WRITE(AR5K_AR5212_EEPROM_BASE, (u_int8_t)offset - 1);
        !          2875:
        !          2876:        for (timeout = 10000; timeout > 0; timeout--) {
        !          2877:                AR5K_DELAY(1);
        !          2878:                status = AR5K_REG_READ(AR5K_AR5212_EEPROM_STATUS);
        !          2879:                if (status & AR5K_AR5212_EEPROM_STAT_WRDONE) {
        !          2880:                        if (status & AR5K_AR5212_EEPROM_STAT_WRERR)
        !          2881:                                return (EIO);
        !          2882:                        return (0);
        !          2883:                }
        !          2884:        }
        !          2885:
        !          2886:        return (ETIMEDOUT);
        !          2887: }
        !          2888:
        !          2889: /*
        !          2890:  * TX power setup
        !          2891:  */
        !          2892:
        !          2893: HAL_BOOL
        !          2894: ar5k_ar5212_txpower(struct ath_hal *hal, HAL_CHANNEL *channel, u_int txpower)
        !          2895: {
        !          2896:        HAL_BOOL tpc = hal->ah_txpower.txp_tpc;
        !          2897:        int i;
        !          2898:
        !          2899:        if (txpower > AR5K_TUNE_MAX_TXPOWER) {
        !          2900:                AR5K_PRINTF("invalid tx power: %u\n", txpower);
        !          2901:                return (AH_FALSE);
        !          2902:        }
        !          2903:
        !          2904:        /* Reset TX power values */
        !          2905:        bzero(&hal->ah_txpower, sizeof(hal->ah_txpower));
        !          2906:        hal->ah_txpower.txp_tpc = tpc;
        !          2907:
        !          2908:        /* Initialize TX power table */
        !          2909:        ar5k_txpower_table(hal, channel, txpower);
        !          2910:
        !          2911:        /*
        !          2912:         * Write TX power values
        !          2913:         */
        !          2914:        for (i = 0; i < (AR5K_EEPROM_POWER_TABLE_SIZE / 2); i++) {
        !          2915:                AR5K_REG_WRITE(AR5K_AR5212_PHY_PCDAC_TXPOWER(i),
        !          2916:                    ((((hal->ah_txpower.txp_pcdac[(i << 1) + 1] << 8) | 0xff) &
        !          2917:                    0xffff) << 16) | (((hal->ah_txpower.txp_pcdac[i << 1] << 8)
        !          2918:                    | 0xff) & 0xffff));
        !          2919:        }
        !          2920:
        !          2921:        AR5K_REG_WRITE(AR5K_AR5212_PHY_TXPOWER_RATE1,
        !          2922:            AR5K_TXPOWER_OFDM(3, 24) | AR5K_TXPOWER_OFDM(2, 16)
        !          2923:            | AR5K_TXPOWER_OFDM(1, 8) | AR5K_TXPOWER_OFDM(0, 0));
        !          2924:
        !          2925:        AR5K_REG_WRITE(AR5K_AR5212_PHY_TXPOWER_RATE2,
        !          2926:            AR5K_TXPOWER_OFDM(7, 24) | AR5K_TXPOWER_OFDM(6, 16)
        !          2927:            | AR5K_TXPOWER_OFDM(5, 8) | AR5K_TXPOWER_OFDM(4, 0));
        !          2928:
        !          2929:        AR5K_REG_WRITE(AR5K_AR5212_PHY_TXPOWER_RATE3,
        !          2930:            AR5K_TXPOWER_CCK(10, 24) | AR5K_TXPOWER_CCK(9, 16)
        !          2931:            | AR5K_TXPOWER_CCK(15, 8) | AR5K_TXPOWER_CCK(8, 0));
        !          2932:
        !          2933:        AR5K_REG_WRITE(AR5K_AR5212_PHY_TXPOWER_RATE4,
        !          2934:            AR5K_TXPOWER_CCK(14, 24) | AR5K_TXPOWER_CCK(13, 16)
        !          2935:            | AR5K_TXPOWER_CCK(12, 8) | AR5K_TXPOWER_CCK(11, 0));
        !          2936:
        !          2937:        if (hal->ah_txpower.txp_tpc == AH_TRUE) {
        !          2938:                AR5K_REG_WRITE(AR5K_AR5212_PHY_TXPOWER_RATE_MAX,
        !          2939:                    AR5K_AR5212_PHY_TXPOWER_RATE_MAX_TPC_ENABLE |
        !          2940:                    AR5K_TUNE_MAX_TXPOWER);
        !          2941:        } else {
        !          2942:                AR5K_REG_WRITE(AR5K_AR5212_PHY_TXPOWER_RATE_MAX,
        !          2943:                    AR5K_AR5212_PHY_TXPOWER_RATE_MAX |
        !          2944:                    AR5K_TUNE_MAX_TXPOWER);
        !          2945:        }
        !          2946:
        !          2947:        return (AH_TRUE);
        !          2948: }
        !          2949:
        !          2950: HAL_BOOL
        !          2951: ar5k_ar5212_set_txpower_limit(struct ath_hal *hal, u_int power)
        !          2952: {
        !          2953:        HAL_CHANNEL *channel = &hal->ah_current_channel;
        !          2954:
        !          2955:        AR5K_PRINTF("changing txpower to %d\n", power);
        !          2956:        return (ar5k_ar5212_txpower(hal, channel, power));
        !          2957: }

CVSweb