Commit 193df183 authored by Lars Ericsson's avatar Lars Ericsson Committed by John W. Linville

rt2x00: Fix quality houskeeping for software diversity

Antanna quality statistics is not handled correctly, which leads to
software diversity being shutdown completly.

The main problem is that during antenna diversity statistics can
be reset resulting in loosing the signal strength just before
evaluation.

rssi history is not updated correctly leading to incorrect comparison
material for basing antenna switching on.
Signed-off-by: default avatarLars Ericsson <Lars_Ericsson@telia.com>
Signed-off-by: default avatarIvo van Doorn <IvDoorn@gmail.com>
Signed-off-by: default avatarJohn W. Linville <linville@tuxdriver.com>
parent 93354cbb
...@@ -245,14 +245,11 @@ struct link_ant { ...@@ -245,14 +245,11 @@ struct link_ant {
struct antenna_setup active; struct antenna_setup active;
/* /*
* RSSI information for the different antennas. * RSSI history information for the antenna.
* These statistics are used to determine when * Used to determine when to switch antenna
* to switch antenna when using software diversity. * when using software diversity.
*
* rssi[0] -> Antenna A RSSI
* rssi[1] -> Antenna B RSSI
*/ */
int rssi_history[2]; int rssi_history;
/* /*
* Current RSSI average of the currently active antenna. * Current RSSI average of the currently active antenna.
......
...@@ -103,39 +103,21 @@ static int rt2x00link_antenna_get_link_rssi(struct rt2x00_dev *rt2x00dev) ...@@ -103,39 +103,21 @@ static int rt2x00link_antenna_get_link_rssi(struct rt2x00_dev *rt2x00dev)
return DEFAULT_RSSI; return DEFAULT_RSSI;
} }
static int rt2x00link_antenna_get_rssi_history(struct rt2x00_dev *rt2x00dev, static int rt2x00link_antenna_get_rssi_history(struct rt2x00_dev *rt2x00dev)
enum antenna antenna)
{ {
struct link_ant *ant = &rt2x00dev->link.ant; struct link_ant *ant = &rt2x00dev->link.ant;
if (ant->rssi_history[antenna - ANTENNA_A]) if (ant->rssi_history)
return ant->rssi_history[antenna - ANTENNA_A]; return ant->rssi_history;
return DEFAULT_RSSI; return DEFAULT_RSSI;
} }
/* Small wrapper for rt2x00link_antenna_get_rssi_history() */
#define rt2x00link_antenna_get_rssi_rx_history(__dev) \
rt2x00link_antenna_get_rssi_history((__dev), \
(__dev)->link.ant.active.rx)
#define rt2x00link_antenna_get_rssi_tx_history(__dev) \
rt2x00link_antenna_get_rssi_history((__dev), \
(__dev)->link.ant.active.tx)
static void rt2x00link_antenna_update_rssi_history(struct rt2x00_dev *rt2x00dev, static void rt2x00link_antenna_update_rssi_history(struct rt2x00_dev *rt2x00dev,
enum antenna antenna,
int rssi) int rssi)
{ {
struct link_ant *ant = &rt2x00dev->link.ant; struct link_ant *ant = &rt2x00dev->link.ant;
ant->rssi_history[ant->active.rx - ANTENNA_A] = rssi; ant->rssi_history = rssi;
} }
/* Small wrapper for rt2x00link_antenna_get_rssi_history() */
#define rt2x00link_antenna_update_rssi_rx_history(__dev, __rssi) \
rt2x00link_antenna_update_rssi_history((__dev), \
(__dev)->link.ant.active.rx, \
(__rssi))
#define rt2x00link_antenna_update_rssi_tx_history(__dev, __rssi) \
rt2x00link_antenna_update_rssi_history((__dev), \
(__dev)->link.ant.active.tx, \
(__rssi))
static void rt2x00link_antenna_reset(struct rt2x00_dev *rt2x00dev) static void rt2x00link_antenna_reset(struct rt2x00_dev *rt2x00dev)
{ {
...@@ -146,8 +128,10 @@ static void rt2x00lib_antenna_diversity_sample(struct rt2x00_dev *rt2x00dev) ...@@ -146,8 +128,10 @@ static void rt2x00lib_antenna_diversity_sample(struct rt2x00_dev *rt2x00dev)
{ {
struct link_ant *ant = &rt2x00dev->link.ant; struct link_ant *ant = &rt2x00dev->link.ant;
struct antenna_setup new_ant; struct antenna_setup new_ant;
int sample_a = rt2x00link_antenna_get_rssi_history(rt2x00dev, ANTENNA_A); int other_antenna;
int sample_b = rt2x00link_antenna_get_rssi_history(rt2x00dev, ANTENNA_B);
int sample_current = rt2x00link_antenna_get_link_rssi(rt2x00dev);
int sample_other = rt2x00link_antenna_get_rssi_history(rt2x00dev);
memcpy(&new_ant, &ant->active, sizeof(new_ant)); memcpy(&new_ant, &ant->active, sizeof(new_ant));
...@@ -161,17 +145,22 @@ static void rt2x00lib_antenna_diversity_sample(struct rt2x00_dev *rt2x00dev) ...@@ -161,17 +145,22 @@ static void rt2x00lib_antenna_diversity_sample(struct rt2x00_dev *rt2x00dev)
* from both antennas. It now is time to determine * from both antennas. It now is time to determine
* which antenna demonstrated the best performance. * which antenna demonstrated the best performance.
* When we are already on the antenna with the best * When we are already on the antenna with the best
* performance, then there really is nothing for us * performance, just create a good starting point
* left to do. * for the history and we are done.
*/ */
if (sample_a == sample_b) if (sample_current >= sample_other) {
rt2x00link_antenna_update_rssi_history(rt2x00dev,
sample_current);
return; return;
}
other_antenna = (ant->active.rx == ANTENNA_A) ? ANTENNA_B : ANTENNA_A;
if (ant->flags & ANTENNA_RX_DIVERSITY) if (ant->flags & ANTENNA_RX_DIVERSITY)
new_ant.rx = (sample_a > sample_b) ? ANTENNA_A : ANTENNA_B; new_ant.rx = other_antenna;
if (ant->flags & ANTENNA_TX_DIVERSITY) if (ant->flags & ANTENNA_TX_DIVERSITY)
new_ant.tx = (sample_a > sample_b) ? ANTENNA_A : ANTENNA_B; new_ant.tx = other_antenna;
rt2x00lib_config_antenna(rt2x00dev, new_ant); rt2x00lib_config_antenna(rt2x00dev, new_ant);
} }
...@@ -190,8 +179,8 @@ static void rt2x00lib_antenna_diversity_eval(struct rt2x00_dev *rt2x00dev) ...@@ -190,8 +179,8 @@ static void rt2x00lib_antenna_diversity_eval(struct rt2x00_dev *rt2x00dev)
* after that update the history with the current value. * after that update the history with the current value.
*/ */
rssi_curr = rt2x00link_antenna_get_link_rssi(rt2x00dev); rssi_curr = rt2x00link_antenna_get_link_rssi(rt2x00dev);
rssi_old = rt2x00link_antenna_get_rssi_rx_history(rt2x00dev); rssi_old = rt2x00link_antenna_get_rssi_history(rt2x00dev);
rt2x00link_antenna_update_rssi_rx_history(rt2x00dev, rssi_curr); rt2x00link_antenna_update_rssi_history(rt2x00dev, rssi_curr);
/* /*
* Legacy driver indicates that we should swap antenna's * Legacy driver indicates that we should swap antenna's
...@@ -216,7 +205,7 @@ static void rt2x00lib_antenna_diversity_eval(struct rt2x00_dev *rt2x00dev) ...@@ -216,7 +205,7 @@ static void rt2x00lib_antenna_diversity_eval(struct rt2x00_dev *rt2x00dev)
rt2x00lib_config_antenna(rt2x00dev, new_ant); rt2x00lib_config_antenna(rt2x00dev, new_ant);
} }
static void rt2x00lib_antenna_diversity(struct rt2x00_dev *rt2x00dev) static bool rt2x00lib_antenna_diversity(struct rt2x00_dev *rt2x00dev)
{ {
struct link_ant *ant = &rt2x00dev->link.ant; struct link_ant *ant = &rt2x00dev->link.ant;
unsigned int flags = ant->flags; unsigned int flags = ant->flags;
...@@ -238,7 +227,7 @@ static void rt2x00lib_antenna_diversity(struct rt2x00_dev *rt2x00dev) ...@@ -238,7 +227,7 @@ static void rt2x00lib_antenna_diversity(struct rt2x00_dev *rt2x00dev)
if (!(ant->flags & ANTENNA_RX_DIVERSITY) && if (!(ant->flags & ANTENNA_RX_DIVERSITY) &&
!(ant->flags & ANTENNA_TX_DIVERSITY)) { !(ant->flags & ANTENNA_TX_DIVERSITY)) {
ant->flags = 0; ant->flags = 0;
return; return true;
} }
/* Update flags */ /* Update flags */
...@@ -250,10 +239,15 @@ static void rt2x00lib_antenna_diversity(struct rt2x00_dev *rt2x00dev) ...@@ -250,10 +239,15 @@ static void rt2x00lib_antenna_diversity(struct rt2x00_dev *rt2x00dev)
* the data. The latter should only be performed once * the data. The latter should only be performed once
* every 2 seconds. * every 2 seconds.
*/ */
if (ant->flags & ANTENNA_MODE_SAMPLE) if (ant->flags & ANTENNA_MODE_SAMPLE) {
rt2x00lib_antenna_diversity_sample(rt2x00dev); rt2x00lib_antenna_diversity_sample(rt2x00dev);
else if (rt2x00dev->link.count & 1) return true;
} else if (rt2x00dev->link.count & 1) {
rt2x00lib_antenna_diversity_eval(rt2x00dev); rt2x00lib_antenna_diversity_eval(rt2x00dev);
return true;
}
return false;
} }
void rt2x00link_update_stats(struct rt2x00_dev *rt2x00dev, void rt2x00link_update_stats(struct rt2x00_dev *rt2x00dev,
...@@ -451,15 +445,12 @@ static void rt2x00link_tuner(struct work_struct *work) ...@@ -451,15 +445,12 @@ static void rt2x00link_tuner(struct work_struct *work)
rt2x00leds_led_quality(rt2x00dev, link->avg_rssi); rt2x00leds_led_quality(rt2x00dev, link->avg_rssi);
/* /*
* Evaluate antenna setup, make this the last step since this could * Evaluate antenna setup, make this the last step when
* possibly reset some statistics. * rt2x00lib_antenna_diversity made changes the quality
*/ * statistics will be reset.
rt2x00lib_antenna_diversity(rt2x00dev);
/*
* Reset the quality counters which recounted during each period.
*/ */
rt2x00link_reset_qual(rt2x00dev); if (rt2x00lib_antenna_diversity(rt2x00dev))
rt2x00link_reset_qual(rt2x00dev);
/* /*
* Increase tuner counter, and reschedule the next link tuner run. * Increase tuner counter, and reschedule the next link tuner run.
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment