rt2x00: Fix BUG_ON() with antenna handling
With the new configuration handling, and more specifically splitting the configuration of the antenna from the normal configuration steps allowed a BUG_ON() to be triggered in the driver because the SW_DIVERSITY was send to the driver. This fixes that by catching the value early in rt2x00config.c and replacing it with a sensible value. This also fixes a problem where the antenna is not being initialized at all when the radio is enabled. Since it no longer is part of the mac80211 configuration the only place where rt2x00 configured it was the SW diversity handler. Obviously this is broken for all non-diversity hardware and breaks SW diversity due to a broken initialization. When the radio is enabled the antenna will be configured once as soon as the config() callback function is called. Signed-off-by: Ivo van Doorn <IvDoorn@gmail.com> Signed-off-by: John W. Linville <linville@tuxdriver.com>
This commit is contained in:
parent
8058409c4f
commit
6d64360ac5
|
@ -723,8 +723,7 @@ struct rt2x00_dev {
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This is the default TX/RX antenna setup as indicated
|
* This is the default TX/RX antenna setup as indicated
|
||||||
* by the device's EEPROM. When mac80211 sets its
|
* by the device's EEPROM.
|
||||||
* antenna value to 0 we should be using these values.
|
|
||||||
*/
|
*/
|
||||||
struct antenna_setup default_ant;
|
struct antenna_setup default_ant;
|
||||||
|
|
||||||
|
|
|
@ -109,15 +109,32 @@ void rt2x00lib_config_erp(struct rt2x00_dev *rt2x00dev,
|
||||||
}
|
}
|
||||||
|
|
||||||
void rt2x00lib_config_antenna(struct rt2x00_dev *rt2x00dev,
|
void rt2x00lib_config_antenna(struct rt2x00_dev *rt2x00dev,
|
||||||
enum antenna rx, enum antenna tx)
|
struct antenna_setup *ant)
|
||||||
{
|
{
|
||||||
struct antenna_setup ant;
|
/*
|
||||||
|
* Failsafe: Make sure we are not sending the
|
||||||
|
* ANTENNA_SW_DIVERSITY state to the driver.
|
||||||
|
* If that happes fallback to hardware default,
|
||||||
|
* or our own default.
|
||||||
|
*/
|
||||||
|
if (ant->rx == ANTENNA_SW_DIVERSITY) {
|
||||||
|
if (rt2x00dev->default_ant.rx == ANTENNA_SW_DIVERSITY)
|
||||||
|
ant->rx = ANTENNA_B;
|
||||||
|
else
|
||||||
|
ant->rx = rt2x00dev->default_ant.rx;
|
||||||
|
}
|
||||||
|
if (ant->tx == ANTENNA_SW_DIVERSITY) {
|
||||||
|
if (rt2x00dev->default_ant.tx == ANTENNA_SW_DIVERSITY)
|
||||||
|
ant->tx = ANTENNA_B;
|
||||||
|
else
|
||||||
|
ant->tx = rt2x00dev->default_ant.tx;
|
||||||
|
}
|
||||||
|
|
||||||
ant.rx = rx;
|
/*
|
||||||
ant.tx = tx;
|
* Only reconfigure when something has changed.
|
||||||
|
*/
|
||||||
if (rx == rt2x00dev->link.ant.active.rx &&
|
if (ant->rx == rt2x00dev->link.ant.active.rx &&
|
||||||
tx == rt2x00dev->link.ant.active.tx)
|
ant->tx == rt2x00dev->link.ant.active.tx)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -132,12 +149,12 @@ void rt2x00lib_config_antenna(struct rt2x00_dev *rt2x00dev,
|
||||||
* The latter is required since we need to recalibrate the
|
* The latter is required since we need to recalibrate the
|
||||||
* noise-sensitivity ratio for the new setup.
|
* noise-sensitivity ratio for the new setup.
|
||||||
*/
|
*/
|
||||||
rt2x00dev->ops->lib->config_ant(rt2x00dev, &ant);
|
rt2x00dev->ops->lib->config_ant(rt2x00dev, ant);
|
||||||
|
|
||||||
rt2x00lib_reset_link_tuner(rt2x00dev);
|
rt2x00lib_reset_link_tuner(rt2x00dev);
|
||||||
rt2x00_reset_link_ant_rssi(&rt2x00dev->link);
|
rt2x00_reset_link_ant_rssi(&rt2x00dev->link);
|
||||||
|
|
||||||
memcpy(&rt2x00dev->link.ant.active, &ant, sizeof(ant));
|
memcpy(&rt2x00dev->link.ant.active, ant, sizeof(*ant));
|
||||||
|
|
||||||
if (test_bit(DEVICE_STATE_ENABLED_RADIO, &rt2x00dev->flags))
|
if (test_bit(DEVICE_STATE_ENABLED_RADIO, &rt2x00dev->flags))
|
||||||
rt2x00lib_toggle_rx(rt2x00dev, STATE_RADIO_RX_ON_LINK);
|
rt2x00lib_toggle_rx(rt2x00dev, STATE_RADIO_RX_ON_LINK);
|
||||||
|
|
|
@ -176,13 +176,14 @@ void rt2x00lib_toggle_rx(struct rt2x00_dev *rt2x00dev, enum dev_state state)
|
||||||
|
|
||||||
static void rt2x00lib_evaluate_antenna_sample(struct rt2x00_dev *rt2x00dev)
|
static void rt2x00lib_evaluate_antenna_sample(struct rt2x00_dev *rt2x00dev)
|
||||||
{
|
{
|
||||||
enum antenna rx = rt2x00dev->link.ant.active.rx;
|
struct antenna_setup ant;
|
||||||
enum antenna tx = rt2x00dev->link.ant.active.tx;
|
|
||||||
int sample_a =
|
int sample_a =
|
||||||
rt2x00_get_link_ant_rssi_history(&rt2x00dev->link, ANTENNA_A);
|
rt2x00_get_link_ant_rssi_history(&rt2x00dev->link, ANTENNA_A);
|
||||||
int sample_b =
|
int sample_b =
|
||||||
rt2x00_get_link_ant_rssi_history(&rt2x00dev->link, ANTENNA_B);
|
rt2x00_get_link_ant_rssi_history(&rt2x00dev->link, ANTENNA_B);
|
||||||
|
|
||||||
|
memcpy(&ant, &rt2x00dev->link.ant.active, sizeof(ant));
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We are done sampling. Now we should evaluate the results.
|
* We are done sampling. Now we should evaluate the results.
|
||||||
*/
|
*/
|
||||||
|
@ -200,21 +201,22 @@ static void rt2x00lib_evaluate_antenna_sample(struct rt2x00_dev *rt2x00dev)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (rt2x00dev->link.ant.flags & ANTENNA_RX_DIVERSITY)
|
if (rt2x00dev->link.ant.flags & ANTENNA_RX_DIVERSITY)
|
||||||
rx = (sample_a > sample_b) ? ANTENNA_A : ANTENNA_B;
|
ant.rx = (sample_a > sample_b) ? ANTENNA_A : ANTENNA_B;
|
||||||
|
|
||||||
if (rt2x00dev->link.ant.flags & ANTENNA_TX_DIVERSITY)
|
if (rt2x00dev->link.ant.flags & ANTENNA_TX_DIVERSITY)
|
||||||
tx = (sample_a > sample_b) ? ANTENNA_A : ANTENNA_B;
|
ant.tx = (sample_a > sample_b) ? ANTENNA_A : ANTENNA_B;
|
||||||
|
|
||||||
rt2x00lib_config_antenna(rt2x00dev, rx, tx);
|
rt2x00lib_config_antenna(rt2x00dev, &ant);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void rt2x00lib_evaluate_antenna_eval(struct rt2x00_dev *rt2x00dev)
|
static void rt2x00lib_evaluate_antenna_eval(struct rt2x00_dev *rt2x00dev)
|
||||||
{
|
{
|
||||||
enum antenna rx = rt2x00dev->link.ant.active.rx;
|
struct antenna_setup ant;
|
||||||
enum antenna tx = rt2x00dev->link.ant.active.tx;
|
|
||||||
int rssi_curr = rt2x00_get_link_ant_rssi(&rt2x00dev->link);
|
int rssi_curr = rt2x00_get_link_ant_rssi(&rt2x00dev->link);
|
||||||
int rssi_old = rt2x00_update_ant_rssi(&rt2x00dev->link, rssi_curr);
|
int rssi_old = rt2x00_update_ant_rssi(&rt2x00dev->link, rssi_curr);
|
||||||
|
|
||||||
|
memcpy(&ant, &rt2x00dev->link.ant.active, sizeof(ant));
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Legacy driver indicates that we should swap antenna's
|
* Legacy driver indicates that we should swap antenna's
|
||||||
* when the difference in RSSI is greater that 5. This
|
* when the difference in RSSI is greater that 5. This
|
||||||
|
@ -230,12 +232,12 @@ static void rt2x00lib_evaluate_antenna_eval(struct rt2x00_dev *rt2x00dev)
|
||||||
rt2x00dev->link.ant.flags |= ANTENNA_MODE_SAMPLE;
|
rt2x00dev->link.ant.flags |= ANTENNA_MODE_SAMPLE;
|
||||||
|
|
||||||
if (rt2x00dev->link.ant.flags & ANTENNA_RX_DIVERSITY)
|
if (rt2x00dev->link.ant.flags & ANTENNA_RX_DIVERSITY)
|
||||||
rx = (rx == ANTENNA_A) ? ANTENNA_B : ANTENNA_A;
|
ant.rx = (ant.rx == ANTENNA_A) ? ANTENNA_B : ANTENNA_A;
|
||||||
|
|
||||||
if (rt2x00dev->link.ant.flags & ANTENNA_TX_DIVERSITY)
|
if (rt2x00dev->link.ant.flags & ANTENNA_TX_DIVERSITY)
|
||||||
tx = (tx == ANTENNA_A) ? ANTENNA_B : ANTENNA_A;
|
ant.tx = (ant.tx == ANTENNA_A) ? ANTENNA_B : ANTENNA_A;
|
||||||
|
|
||||||
rt2x00lib_config_antenna(rt2x00dev, rx, tx);
|
rt2x00lib_config_antenna(rt2x00dev, &ant);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void rt2x00lib_evaluate_antenna(struct rt2x00_dev *rt2x00dev)
|
static void rt2x00lib_evaluate_antenna(struct rt2x00_dev *rt2x00dev)
|
||||||
|
|
|
@ -93,7 +93,7 @@ void rt2x00lib_config_erp(struct rt2x00_dev *rt2x00dev,
|
||||||
struct rt2x00_intf *intf,
|
struct rt2x00_intf *intf,
|
||||||
struct ieee80211_bss_conf *conf);
|
struct ieee80211_bss_conf *conf);
|
||||||
void rt2x00lib_config_antenna(struct rt2x00_dev *rt2x00dev,
|
void rt2x00lib_config_antenna(struct rt2x00_dev *rt2x00dev,
|
||||||
enum antenna rx, enum antenna tx);
|
struct antenna_setup *ant);
|
||||||
void rt2x00lib_config(struct rt2x00_dev *rt2x00dev,
|
void rt2x00lib_config(struct rt2x00_dev *rt2x00dev,
|
||||||
struct ieee80211_conf *conf,
|
struct ieee80211_conf *conf,
|
||||||
const unsigned int changed_flags);
|
const unsigned int changed_flags);
|
||||||
|
|
|
@ -339,7 +339,6 @@ int rt2x00mac_config(struct ieee80211_hw *hw, u32 changed)
|
||||||
{
|
{
|
||||||
struct rt2x00_dev *rt2x00dev = hw->priv;
|
struct rt2x00_dev *rt2x00dev = hw->priv;
|
||||||
struct ieee80211_conf *conf = &hw->conf;
|
struct ieee80211_conf *conf = &hw->conf;
|
||||||
int radio_on;
|
|
||||||
int status;
|
int status;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -356,7 +355,6 @@ int rt2x00mac_config(struct ieee80211_hw *hw, u32 changed)
|
||||||
* some configuration parameters (e.g. channel and antenna values) can
|
* some configuration parameters (e.g. channel and antenna values) can
|
||||||
* only be set when the radio is enabled.
|
* only be set when the radio is enabled.
|
||||||
*/
|
*/
|
||||||
radio_on = test_bit(DEVICE_STATE_ENABLED_RADIO, &rt2x00dev->flags);
|
|
||||||
if (conf->radio_enabled) {
|
if (conf->radio_enabled) {
|
||||||
/* For programming the values, we have to turn RX off */
|
/* For programming the values, we have to turn RX off */
|
||||||
rt2x00lib_toggle_rx(rt2x00dev, STATE_RADIO_RX_OFF);
|
rt2x00lib_toggle_rx(rt2x00dev, STATE_RADIO_RX_OFF);
|
||||||
|
@ -372,6 +370,17 @@ int rt2x00mac_config(struct ieee80211_hw *hw, u32 changed)
|
||||||
*/
|
*/
|
||||||
rt2x00lib_config(rt2x00dev, conf, changed);
|
rt2x00lib_config(rt2x00dev, conf, changed);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The radio was enabled, configure the antenna to the
|
||||||
|
* default settings, the link tuner will later start
|
||||||
|
* continue configuring the antenna based on the software
|
||||||
|
* diversity. But for non-diversity configurations, we need
|
||||||
|
* to have configured the correct state now.
|
||||||
|
*/
|
||||||
|
if (changed & IEEE80211_CONF_CHANGE_RADIO_ENABLED)
|
||||||
|
rt2x00lib_config_antenna(rt2x00dev,
|
||||||
|
&rt2x00dev->default_ant);
|
||||||
|
|
||||||
/* Turn RX back on */
|
/* Turn RX back on */
|
||||||
rt2x00lib_toggle_rx(rt2x00dev, STATE_RADIO_RX_ON);
|
rt2x00lib_toggle_rx(rt2x00dev, STATE_RADIO_RX_ON);
|
||||||
} else {
|
} else {
|
||||||
|
|
Loading…
Reference in New Issue