|
@@ -186,7 +186,40 @@ static u32 ar9003_hw_compute_pll_control(struct ath_hw *ah,
|
|
|
static void ar9003_hw_set_channel_regs(struct ath_hw *ah,
|
|
|
struct ath9k_channel *chan)
|
|
|
{
|
|
|
- /* TODO */
|
|
|
+ u32 phymode;
|
|
|
+ u32 enableDacFifo = 0;
|
|
|
+
|
|
|
+ enableDacFifo =
|
|
|
+ (REG_READ(ah, AR_PHY_GEN_CTRL) & AR_PHY_GC_ENABLE_DAC_FIFO);
|
|
|
+
|
|
|
+ /* Enable 11n HT, 20 MHz */
|
|
|
+ phymode = AR_PHY_GC_HT_EN | AR_PHY_GC_SINGLE_HT_LTF1 | AR_PHY_GC_WALSH |
|
|
|
+ AR_PHY_GC_SHORT_GI_40 | enableDacFifo;
|
|
|
+
|
|
|
+ /* Configure baseband for dynamic 20/40 operation */
|
|
|
+ if (IS_CHAN_HT40(chan)) {
|
|
|
+ phymode |= AR_PHY_GC_DYN2040_EN;
|
|
|
+ /* Configure control (primary) channel at +-10MHz */
|
|
|
+ if ((chan->chanmode == CHANNEL_A_HT40PLUS) ||
|
|
|
+ (chan->chanmode == CHANNEL_G_HT40PLUS))
|
|
|
+ phymode |= AR_PHY_GC_DYN2040_PRI_CH;
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ /* make sure we preserve INI settings */
|
|
|
+ phymode |= REG_READ(ah, AR_PHY_GEN_CTRL);
|
|
|
+ /* turn off Green Field detection for STA for now */
|
|
|
+ phymode &= ~AR_PHY_GC_GF_DETECT_EN;
|
|
|
+
|
|
|
+ REG_WRITE(ah, AR_PHY_GEN_CTRL, phymode);
|
|
|
+
|
|
|
+ /* Configure MAC for 20/40 operation */
|
|
|
+ ath9k_hw_set11nmac2040(ah);
|
|
|
+
|
|
|
+ /* global transmit timeout (25 TUs default)*/
|
|
|
+ REG_WRITE(ah, AR_GTXTO, 25 << AR_GTXTO_TIMEOUT_LIMIT_S);
|
|
|
+ /* carrier sense timeout */
|
|
|
+ REG_WRITE(ah, AR_CST, 0xF << AR_CST_TIMEOUT_LIMIT_S);
|
|
|
}
|
|
|
|
|
|
static void ar9003_hw_init_bb(struct ath_hw *ah,
|
|
@@ -195,11 +228,158 @@ static void ar9003_hw_init_bb(struct ath_hw *ah,
|
|
|
/* TODO */
|
|
|
}
|
|
|
|
|
|
+void ar9003_hw_set_chain_masks(struct ath_hw *ah, u8 rx, u8 tx)
|
|
|
+{
|
|
|
+ switch (rx) {
|
|
|
+ case 0x5:
|
|
|
+ REG_SET_BIT(ah, AR_PHY_ANALOG_SWAP,
|
|
|
+ AR_PHY_SWAP_ALT_CHAIN);
|
|
|
+ case 0x3:
|
|
|
+ case 0x1:
|
|
|
+ case 0x2:
|
|
|
+ case 0x7:
|
|
|
+ REG_WRITE(ah, AR_PHY_RX_CHAINMASK, rx);
|
|
|
+ REG_WRITE(ah, AR_PHY_CAL_CHAINMASK, rx);
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+ REG_WRITE(ah, AR_SELFGEN_MASK, tx);
|
|
|
+ if (tx == 0x5) {
|
|
|
+ REG_SET_BIT(ah, AR_PHY_ANALOG_SWAP,
|
|
|
+ AR_PHY_SWAP_ALT_CHAIN);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+/*
|
|
|
+ * Override INI values with chip specific configuration.
|
|
|
+ */
|
|
|
+static void ar9003_hw_override_ini(struct ath_hw *ah)
|
|
|
+{
|
|
|
+ u32 val;
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Set the RX_ABORT and RX_DIS and clear it only after
|
|
|
+ * RXE is set for MAC. This prevents frames with
|
|
|
+ * corrupted descriptor status.
|
|
|
+ */
|
|
|
+ REG_SET_BIT(ah, AR_DIAG_SW, (AR_DIAG_RX_DIS | AR_DIAG_RX_ABORT));
|
|
|
+
|
|
|
+ /*
|
|
|
+ * For AR9280 and above, there is a new feature that allows
|
|
|
+ * Multicast search based on both MAC Address and Key ID. By default,
|
|
|
+ * this feature is enabled. But since the driver is not using this
|
|
|
+ * feature, we switch it off; otherwise multicast search based on
|
|
|
+ * MAC addr only will fail.
|
|
|
+ */
|
|
|
+ val = REG_READ(ah, AR_PCU_MISC_MODE2) & (~AR_ADHOC_MCAST_KEYID_ENABLE);
|
|
|
+ REG_WRITE(ah, AR_PCU_MISC_MODE2,
|
|
|
+ val | AR_AGG_WEP_ENABLE_FIX | AR_AGG_WEP_ENABLE);
|
|
|
+}
|
|
|
+
|
|
|
+static void ar9003_hw_prog_ini(struct ath_hw *ah,
|
|
|
+ struct ar5416IniArray *iniArr,
|
|
|
+ int column)
|
|
|
+{
|
|
|
+ unsigned int i, regWrites = 0;
|
|
|
+
|
|
|
+ /* New INI format: Array may be undefined (pre, core, post arrays) */
|
|
|
+ if (!iniArr->ia_array)
|
|
|
+ return;
|
|
|
+
|
|
|
+ /*
|
|
|
+ * New INI format: Pre, core, and post arrays for a given subsystem
|
|
|
+ * may be modal (> 2 columns) or non-modal (2 columns). Determine if
|
|
|
+ * the array is non-modal and force the column to 1.
|
|
|
+ */
|
|
|
+ if (column >= iniArr->ia_columns)
|
|
|
+ column = 1;
|
|
|
+
|
|
|
+ for (i = 0; i < iniArr->ia_rows; i++) {
|
|
|
+ u32 reg = INI_RA(iniArr, i, 0);
|
|
|
+ u32 val = INI_RA(iniArr, i, column);
|
|
|
+
|
|
|
+ REG_WRITE(ah, reg, val);
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Determine if this is a shift register value, and insert the
|
|
|
+ * configured delay if so.
|
|
|
+ */
|
|
|
+ if (reg >= 0x16000 && reg < 0x17000
|
|
|
+ && ah->config.analog_shiftreg)
|
|
|
+ udelay(100);
|
|
|
+
|
|
|
+ DO_DELAY(regWrites);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
static int ar9003_hw_process_ini(struct ath_hw *ah,
|
|
|
struct ath9k_channel *chan)
|
|
|
{
|
|
|
- /* TODO */
|
|
|
- return -1;
|
|
|
+ struct ath_regulatory *regulatory = ath9k_hw_regulatory(ah);
|
|
|
+ unsigned int regWrites = 0, i;
|
|
|
+ struct ieee80211_channel *channel = chan->chan;
|
|
|
+ u32 modesIndex, freqIndex;
|
|
|
+
|
|
|
+ switch (chan->chanmode) {
|
|
|
+ case CHANNEL_A:
|
|
|
+ case CHANNEL_A_HT20:
|
|
|
+ modesIndex = 1;
|
|
|
+ freqIndex = 1;
|
|
|
+ break;
|
|
|
+ case CHANNEL_A_HT40PLUS:
|
|
|
+ case CHANNEL_A_HT40MINUS:
|
|
|
+ modesIndex = 2;
|
|
|
+ freqIndex = 1;
|
|
|
+ break;
|
|
|
+ case CHANNEL_G:
|
|
|
+ case CHANNEL_G_HT20:
|
|
|
+ case CHANNEL_B:
|
|
|
+ modesIndex = 4;
|
|
|
+ freqIndex = 2;
|
|
|
+ break;
|
|
|
+ case CHANNEL_G_HT40PLUS:
|
|
|
+ case CHANNEL_G_HT40MINUS:
|
|
|
+ modesIndex = 3;
|
|
|
+ freqIndex = 2;
|
|
|
+ break;
|
|
|
+
|
|
|
+ default:
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ for (i = 0; i < ATH_INI_NUM_SPLIT; i++) {
|
|
|
+ ar9003_hw_prog_ini(ah, &ah->iniSOC[i], modesIndex);
|
|
|
+ ar9003_hw_prog_ini(ah, &ah->iniMac[i], modesIndex);
|
|
|
+ ar9003_hw_prog_ini(ah, &ah->iniBB[i], modesIndex);
|
|
|
+ ar9003_hw_prog_ini(ah, &ah->iniRadio[i], modesIndex);
|
|
|
+ }
|
|
|
+
|
|
|
+ REG_WRITE_ARRAY(&ah->iniModesRxGain, 1, regWrites);
|
|
|
+ REG_WRITE_ARRAY(&ah->iniModesTxGain, modesIndex, regWrites);
|
|
|
+
|
|
|
+ /*
|
|
|
+ * For 5GHz channels requiring Fast Clock, apply
|
|
|
+ * different modal values.
|
|
|
+ */
|
|
|
+ if (IS_CHAN_A_5MHZ_SPACED(chan))
|
|
|
+ REG_WRITE_ARRAY(&ah->iniModesAdditional,
|
|
|
+ modesIndex, regWrites);
|
|
|
+
|
|
|
+ ar9003_hw_override_ini(ah);
|
|
|
+ ar9003_hw_set_channel_regs(ah, chan);
|
|
|
+ ar9003_hw_set_chain_masks(ah, ah->rxchainmask, ah->txchainmask);
|
|
|
+
|
|
|
+ /* Set TX power */
|
|
|
+ ah->eep_ops->set_txpower(ah, chan,
|
|
|
+ ath9k_regd_get_ctl(regulatory, chan),
|
|
|
+ channel->max_antenna_gain * 2,
|
|
|
+ channel->max_power * 2,
|
|
|
+ min((u32) MAX_RATE_POWER,
|
|
|
+ (u32) regulatory->power_limit));
|
|
|
+
|
|
|
+ return 0;
|
|
|
}
|
|
|
|
|
|
static void ar9003_hw_set_rfmode(struct ath_hw *ah,
|