iwlwifi: introduce statistics lock

The statistics are currently only half-heartedly
locked against concurrent reading & modification
so introduce a lock to really protect them.

Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Signed-off-by: Wey-Yi Guy <wey-yi.w.guy@intel.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
This commit is contained in:
Johannes Berg 2012-03-05 11:24:26 -08:00 committed by John W. Linville
parent fa23cb04b3
commit 4ff70fcdf3
6 changed files with 40 additions and 9 deletions

View File

@ -655,7 +655,6 @@ void iwl_sensitivity_calibration(struct iwl_priv *priv)
struct iwl_sensitivity_data *data = NULL; struct iwl_sensitivity_data *data = NULL;
struct statistics_rx_non_phy *rx_info; struct statistics_rx_non_phy *rx_info;
struct statistics_rx_phy *ofdm, *cck; struct statistics_rx_phy *ofdm, *cck;
unsigned long flags;
struct statistics_general_data statis; struct statistics_general_data statis;
if (priv->disable_sens_cal) if (priv->disable_sens_cal)
@ -668,13 +667,13 @@ void iwl_sensitivity_calibration(struct iwl_priv *priv)
return; return;
} }
spin_lock_irqsave(&priv->shrd->lock, flags); spin_lock_bh(&priv->statistics.lock);
rx_info = &priv->statistics.rx_non_phy; rx_info = &priv->statistics.rx_non_phy;
ofdm = &priv->statistics.rx_ofdm; ofdm = &priv->statistics.rx_ofdm;
cck = &priv->statistics.rx_cck; cck = &priv->statistics.rx_cck;
if (rx_info->interference_data_flag != INTERFERENCE_DATA_AVAILABLE) { if (rx_info->interference_data_flag != INTERFERENCE_DATA_AVAILABLE) {
IWL_DEBUG_CALIB(priv, "<< invalid data.\n"); IWL_DEBUG_CALIB(priv, "<< invalid data.\n");
spin_unlock_irqrestore(&priv->shrd->lock, flags); spin_unlock_bh(&priv->statistics.lock);
return; return;
} }
@ -698,7 +697,7 @@ void iwl_sensitivity_calibration(struct iwl_priv *priv)
statis.beacon_energy_c = statis.beacon_energy_c =
le32_to_cpu(rx_info->beacon_energy_c); le32_to_cpu(rx_info->beacon_energy_c);
spin_unlock_irqrestore(&priv->shrd->lock, flags); spin_unlock_bh(&priv->statistics.lock);
IWL_DEBUG_CALIB(priv, "rx_enable_time = %u usecs\n", rx_enable_time); IWL_DEBUG_CALIB(priv, "rx_enable_time = %u usecs\n", rx_enable_time);
@ -958,7 +957,6 @@ void iwl_chain_noise_calibration(struct iwl_priv *priv)
u16 stat_chnum = INITIALIZATION_VALUE; u16 stat_chnum = INITIALIZATION_VALUE;
u8 rxon_band24; u8 rxon_band24;
u8 stat_band24; u8 stat_band24;
unsigned long flags;
struct statistics_rx_non_phy *rx_info; struct statistics_rx_non_phy *rx_info;
/* /*
@ -983,13 +981,13 @@ void iwl_chain_noise_calibration(struct iwl_priv *priv)
return; return;
} }
spin_lock_irqsave(&priv->shrd->lock, flags); spin_lock_bh(&priv->statistics.lock);
rx_info = &priv->statistics.rx_non_phy; rx_info = &priv->statistics.rx_non_phy;
if (rx_info->interference_data_flag != INTERFERENCE_DATA_AVAILABLE) { if (rx_info->interference_data_flag != INTERFERENCE_DATA_AVAILABLE) {
IWL_DEBUG_CALIB(priv, " << Interference data unavailable\n"); IWL_DEBUG_CALIB(priv, " << Interference data unavailable\n");
spin_unlock_irqrestore(&priv->shrd->lock, flags); spin_unlock_bh(&priv->statistics.lock);
return; return;
} }
@ -1004,7 +1002,7 @@ void iwl_chain_noise_calibration(struct iwl_priv *priv)
if ((rxon_chnum != stat_chnum) || (rxon_band24 != stat_band24)) { if ((rxon_chnum != stat_chnum) || (rxon_band24 != stat_band24)) {
IWL_DEBUG_CALIB(priv, "Stats not from chan=%d, band24=%d\n", IWL_DEBUG_CALIB(priv, "Stats not from chan=%d, band24=%d\n",
rxon_chnum, rxon_band24); rxon_chnum, rxon_band24);
spin_unlock_irqrestore(&priv->shrd->lock, flags); spin_unlock_bh(&priv->statistics.lock);
return; return;
} }
@ -1023,7 +1021,7 @@ void iwl_chain_noise_calibration(struct iwl_priv *priv)
chain_sig_b = le32_to_cpu(rx_info->beacon_rssi_b) & IN_BAND_FILTER; chain_sig_b = le32_to_cpu(rx_info->beacon_rssi_b) & IN_BAND_FILTER;
chain_sig_c = le32_to_cpu(rx_info->beacon_rssi_c) & IN_BAND_FILTER; chain_sig_c = le32_to_cpu(rx_info->beacon_rssi_c) & IN_BAND_FILTER;
spin_unlock_irqrestore(&priv->shrd->lock, flags); spin_unlock_bh(&priv->statistics.lock);
data->beacon_count++; data->beacon_count++;

View File

@ -88,6 +88,8 @@ int iwlagn_send_tx_power(struct iwl_priv *priv)
void iwlagn_temperature(struct iwl_priv *priv) void iwlagn_temperature(struct iwl_priv *priv)
{ {
lockdep_assert_held(&priv->statistics.lock);
/* store temperature from correct statistics (in Celsius) */ /* store temperature from correct statistics (in Celsius) */
priv->temperature = le32_to_cpu(priv->statistics.common.temperature); priv->temperature = le32_to_cpu(priv->statistics.common.temperature);
iwl_tt_handler(priv); iwl_tt_handler(priv);

View File

@ -266,6 +266,8 @@ static bool iwlagn_good_ack_health(struct iwl_priv *priv,
if (priv->agg_tids_count) if (priv->agg_tids_count)
return true; return true;
lockdep_assert_held(&priv->statistics.lock);
old = &priv->statistics.tx; old = &priv->statistics.tx;
actual_delta = le32_to_cpu(cur->actual_ack_cnt) - actual_delta = le32_to_cpu(cur->actual_ack_cnt) -
@ -509,6 +511,8 @@ static int iwlagn_rx_statistics(struct iwl_priv *priv,
IWL_DEBUG_RX(priv, "Statistics notification received (%d bytes).\n", IWL_DEBUG_RX(priv, "Statistics notification received (%d bytes).\n",
len); len);
spin_lock(&priv->statistics.lock);
if (len == sizeof(struct iwl_bt_notif_statistics)) { if (len == sizeof(struct iwl_bt_notif_statistics)) {
struct iwl_bt_notif_statistics *stats; struct iwl_bt_notif_statistics *stats;
stats = &pkt->u.stats_bt; stats = &pkt->u.stats_bt;
@ -542,6 +546,7 @@ static int iwlagn_rx_statistics(struct iwl_priv *priv,
WARN_ONCE(1, "len %d doesn't match BT (%zu) or normal (%zu)\n", WARN_ONCE(1, "len %d doesn't match BT (%zu) or normal (%zu)\n",
len, sizeof(struct iwl_bt_notif_statistics), len, sizeof(struct iwl_bt_notif_statistics),
sizeof(struct iwl_notif_statistics)); sizeof(struct iwl_notif_statistics));
spin_unlock(&priv->statistics.lock);
return 0; return 0;
} }
@ -585,6 +590,9 @@ static int iwlagn_rx_statistics(struct iwl_priv *priv,
} }
if (cfg(priv)->lib->temperature && change) if (cfg(priv)->lib->temperature && change)
cfg(priv)->lib->temperature(priv); cfg(priv)->lib->temperature(priv);
spin_unlock(&priv->statistics.lock);
return 0; return 0;
} }

View File

@ -1233,6 +1233,7 @@ static struct iwl_op_mode *iwl_op_mode_dvm_start(struct iwl_trans *trans)
*/ */
spin_lock_init(&trans(priv)->reg_lock); spin_lock_init(&trans(priv)->reg_lock);
spin_lock_init(&priv->shrd->lock); spin_lock_init(&priv->shrd->lock);
spin_lock_init(&priv->statistics.lock);
/*********************** /***********************
* 3. Read REV register * 3. Read REV register

View File

@ -920,6 +920,8 @@ static int iwl_statistics_flag(struct iwl_priv *priv, char *buf, int bufsz)
int p = 0; int p = 0;
u32 flag; u32 flag;
lockdep_assert_held(&priv->statistics.lock);
flag = le32_to_cpu(priv->statistics.flag); flag = le32_to_cpu(priv->statistics.flag);
p += scnprintf(buf + p, bufsz - p, "Statistics Flag(0x%X):\n", flag); p += scnprintf(buf + p, bufsz - p, "Statistics Flag(0x%X):\n", flag);
@ -967,6 +969,7 @@ static ssize_t iwl_dbgfs_ucode_rx_stats_read(struct file *file,
* the last statistics notification from uCode * the last statistics notification from uCode
* might not reflect the current uCode activity * might not reflect the current uCode activity
*/ */
spin_lock_bh(&priv->statistics.lock);
ofdm = &priv->statistics.rx_ofdm; ofdm = &priv->statistics.rx_ofdm;
cck = &priv->statistics.rx_cck; cck = &priv->statistics.rx_cck;
general = &priv->statistics.rx_non_phy; general = &priv->statistics.rx_non_phy;
@ -1363,6 +1366,8 @@ static ssize_t iwl_dbgfs_ucode_rx_stats_read(struct file *file,
accum_ht->unsupport_mcs, accum_ht->unsupport_mcs,
delta_ht->unsupport_mcs, max_ht->unsupport_mcs); delta_ht->unsupport_mcs, max_ht->unsupport_mcs);
spin_unlock_bh(&priv->statistics.lock);
ret = simple_read_from_buffer(user_buf, count, ppos, buf, pos); ret = simple_read_from_buffer(user_buf, count, ppos, buf, pos);
kfree(buf); kfree(buf);
return ret; return ret;
@ -1392,6 +1397,8 @@ static ssize_t iwl_dbgfs_ucode_tx_stats_read(struct file *file,
* the last statistics notification from uCode * the last statistics notification from uCode
* might not reflect the current uCode activity * might not reflect the current uCode activity
*/ */
spin_lock_bh(&priv->statistics.lock);
tx = &priv->statistics.tx; tx = &priv->statistics.tx;
accum_tx = &priv->accum_stats.tx; accum_tx = &priv->accum_stats.tx;
delta_tx = &priv->delta_stats.tx; delta_tx = &priv->delta_stats.tx;
@ -1554,6 +1561,9 @@ static ssize_t iwl_dbgfs_ucode_tx_stats_read(struct file *file,
fmt_hex, "antenna C:", fmt_hex, "antenna C:",
tx->tx_power.ant_c); tx->tx_power.ant_c);
} }
spin_unlock_bh(&priv->statistics.lock);
ret = simple_read_from_buffer(user_buf, count, ppos, buf, pos); ret = simple_read_from_buffer(user_buf, count, ppos, buf, pos);
kfree(buf); kfree(buf);
return ret; return ret;
@ -1586,6 +1596,9 @@ static ssize_t iwl_dbgfs_ucode_general_stats_read(struct file *file,
* the last statistics notification from uCode * the last statistics notification from uCode
* might not reflect the current uCode activity * might not reflect the current uCode activity
*/ */
spin_lock_bh(&priv->statistics.lock);
general = &priv->statistics.common; general = &priv->statistics.common;
dbg = &priv->statistics.common.dbg; dbg = &priv->statistics.common.dbg;
div = &priv->statistics.common.div; div = &priv->statistics.common.div;
@ -1670,6 +1683,9 @@ static ssize_t iwl_dbgfs_ucode_general_stats_read(struct file *file,
accum_general->num_of_sos_states, accum_general->num_of_sos_states,
delta_general->num_of_sos_states, delta_general->num_of_sos_states,
max_general->num_of_sos_states); max_general->num_of_sos_states);
spin_unlock_bh(&priv->statistics.lock);
ret = simple_read_from_buffer(user_buf, count, ppos, buf, pos); ret = simple_read_from_buffer(user_buf, count, ppos, buf, pos);
kfree(buf); kfree(buf);
return ret; return ret;
@ -1713,6 +1729,9 @@ static ssize_t iwl_dbgfs_ucode_bt_stats_read(struct file *file,
* the last statistics notification from uCode * the last statistics notification from uCode
* might not reflect the current uCode activity * might not reflect the current uCode activity
*/ */
spin_lock_bh(&priv->statistics.lock);
bt = &priv->statistics.bt_activity; bt = &priv->statistics.bt_activity;
accum_bt = &priv->accum_stats.bt_activity; accum_bt = &priv->accum_stats.bt_activity;
@ -1758,6 +1777,8 @@ static ssize_t iwl_dbgfs_ucode_bt_stats_read(struct file *file,
le32_to_cpu(priv->statistics.num_bt_kills), le32_to_cpu(priv->statistics.num_bt_kills),
priv->statistics.accum_num_bt_kills); priv->statistics.accum_num_bt_kills);
spin_unlock_bh(&priv->statistics.lock);
ret = simple_read_from_buffer(user_buf, count, ppos, buf, pos); ret = simple_read_from_buffer(user_buf, count, ppos, buf, pos);
kfree(buf); kfree(buf);
return ret; return ret;

View File

@ -851,6 +851,7 @@ struct iwl_priv {
struct statistics_bt_activity bt_activity; struct statistics_bt_activity bt_activity;
__le32 num_bt_kills, accum_num_bt_kills; __le32 num_bt_kills, accum_num_bt_kills;
#endif #endif
spinlock_t lock;
} statistics; } statistics;
#ifdef CONFIG_IWLWIFI_DEBUGFS #ifdef CONFIG_IWLWIFI_DEBUGFS
struct { struct {