|
@@ -157,16 +157,19 @@ int ___ieee80211_stop_tx_ba_session(struct sta_info *sta, u16 tid,
|
|
bool tx)
|
|
bool tx)
|
|
{
|
|
{
|
|
struct ieee80211_local *local = sta->local;
|
|
struct ieee80211_local *local = sta->local;
|
|
- struct tid_ampdu_tx *tid_tx = sta->ampdu_mlme.tid_tx[tid];
|
|
|
|
|
|
+ struct tid_ampdu_tx *tid_tx;
|
|
int ret;
|
|
int ret;
|
|
|
|
|
|
lockdep_assert_held(&sta->ampdu_mlme.mtx);
|
|
lockdep_assert_held(&sta->ampdu_mlme.mtx);
|
|
|
|
|
|
- if (!tid_tx)
|
|
|
|
- return -ENOENT;
|
|
|
|
-
|
|
|
|
spin_lock_bh(&sta->lock);
|
|
spin_lock_bh(&sta->lock);
|
|
|
|
|
|
|
|
+ tid_tx = rcu_dereference_protected_tid_tx(sta, tid);
|
|
|
|
+ if (!tid_tx) {
|
|
|
|
+ spin_unlock_bh(&sta->lock);
|
|
|
|
+ return -ENOENT;
|
|
|
|
+ }
|
|
|
|
+
|
|
if (test_bit(HT_AGG_STATE_WANT_START, &tid_tx->state)) {
|
|
if (test_bit(HT_AGG_STATE_WANT_START, &tid_tx->state)) {
|
|
/* not even started yet! */
|
|
/* not even started yet! */
|
|
ieee80211_assign_tid_tx(sta, tid, NULL);
|
|
ieee80211_assign_tid_tx(sta, tid, NULL);
|
|
@@ -291,13 +294,13 @@ ieee80211_wake_queue_agg(struct ieee80211_local *local, int tid)
|
|
|
|
|
|
void ieee80211_tx_ba_session_handle_start(struct sta_info *sta, int tid)
|
|
void ieee80211_tx_ba_session_handle_start(struct sta_info *sta, int tid)
|
|
{
|
|
{
|
|
- struct tid_ampdu_tx *tid_tx = sta->ampdu_mlme.tid_tx[tid];
|
|
|
|
|
|
+ struct tid_ampdu_tx *tid_tx;
|
|
struct ieee80211_local *local = sta->local;
|
|
struct ieee80211_local *local = sta->local;
|
|
struct ieee80211_sub_if_data *sdata = sta->sdata;
|
|
struct ieee80211_sub_if_data *sdata = sta->sdata;
|
|
u16 start_seq_num;
|
|
u16 start_seq_num;
|
|
int ret;
|
|
int ret;
|
|
|
|
|
|
- lockdep_assert_held(&sta->ampdu_mlme.mtx);
|
|
|
|
|
|
+ tid_tx = rcu_dereference_protected_tid_tx(sta, tid);
|
|
|
|
|
|
/*
|
|
/*
|
|
* While we're asking the driver about the aggregation,
|
|
* While we're asking the driver about the aggregation,
|
|
@@ -404,7 +407,7 @@ int ieee80211_start_tx_ba_session(struct ieee80211_sta *pubsta, u16 tid,
|
|
goto err_unlock_sta;
|
|
goto err_unlock_sta;
|
|
}
|
|
}
|
|
|
|
|
|
- tid_tx = sta->ampdu_mlme.tid_tx[tid];
|
|
|
|
|
|
+ tid_tx = rcu_dereference_protected_tid_tx(sta, tid);
|
|
/* check if the TID is not in aggregation flow already */
|
|
/* check if the TID is not in aggregation flow already */
|
|
if (tid_tx || sta->ampdu_mlme.tid_start_tx[tid]) {
|
|
if (tid_tx || sta->ampdu_mlme.tid_start_tx[tid]) {
|
|
#ifdef CONFIG_MAC80211_HT_DEBUG
|
|
#ifdef CONFIG_MAC80211_HT_DEBUG
|
|
@@ -491,16 +494,19 @@ ieee80211_agg_splice_finish(struct ieee80211_local *local, u16 tid)
|
|
static void ieee80211_agg_tx_operational(struct ieee80211_local *local,
|
|
static void ieee80211_agg_tx_operational(struct ieee80211_local *local,
|
|
struct sta_info *sta, u16 tid)
|
|
struct sta_info *sta, u16 tid)
|
|
{
|
|
{
|
|
|
|
+ struct tid_ampdu_tx *tid_tx;
|
|
|
|
+
|
|
lockdep_assert_held(&sta->ampdu_mlme.mtx);
|
|
lockdep_assert_held(&sta->ampdu_mlme.mtx);
|
|
|
|
|
|
|
|
+ tid_tx = rcu_dereference_protected_tid_tx(sta, tid);
|
|
|
|
+
|
|
#ifdef CONFIG_MAC80211_HT_DEBUG
|
|
#ifdef CONFIG_MAC80211_HT_DEBUG
|
|
printk(KERN_DEBUG "Aggregation is on for tid %d\n", tid);
|
|
printk(KERN_DEBUG "Aggregation is on for tid %d\n", tid);
|
|
#endif
|
|
#endif
|
|
|
|
|
|
drv_ampdu_action(local, sta->sdata,
|
|
drv_ampdu_action(local, sta->sdata,
|
|
IEEE80211_AMPDU_TX_OPERATIONAL,
|
|
IEEE80211_AMPDU_TX_OPERATIONAL,
|
|
- &sta->sta, tid, NULL,
|
|
|
|
- sta->ampdu_mlme.tid_tx[tid]->buf_size);
|
|
|
|
|
|
+ &sta->sta, tid, NULL, tid_tx->buf_size);
|
|
|
|
|
|
/*
|
|
/*
|
|
* synchronize with TX path, while splicing the TX path
|
|
* synchronize with TX path, while splicing the TX path
|
|
@@ -508,13 +514,13 @@ static void ieee80211_agg_tx_operational(struct ieee80211_local *local,
|
|
*/
|
|
*/
|
|
spin_lock_bh(&sta->lock);
|
|
spin_lock_bh(&sta->lock);
|
|
|
|
|
|
- ieee80211_agg_splice_packets(local, sta->ampdu_mlme.tid_tx[tid], tid);
|
|
|
|
|
|
+ ieee80211_agg_splice_packets(local, tid_tx, tid);
|
|
/*
|
|
/*
|
|
* Now mark as operational. This will be visible
|
|
* Now mark as operational. This will be visible
|
|
* in the TX path, and lets it go lock-free in
|
|
* in the TX path, and lets it go lock-free in
|
|
* the common case.
|
|
* the common case.
|
|
*/
|
|
*/
|
|
- set_bit(HT_AGG_STATE_OPERATIONAL, &sta->ampdu_mlme.tid_tx[tid]->state);
|
|
|
|
|
|
+ set_bit(HT_AGG_STATE_OPERATIONAL, &tid_tx->state);
|
|
ieee80211_agg_splice_finish(local, tid);
|
|
ieee80211_agg_splice_finish(local, tid);
|
|
|
|
|
|
spin_unlock_bh(&sta->lock);
|
|
spin_unlock_bh(&sta->lock);
|
|
@@ -548,7 +554,7 @@ void ieee80211_start_tx_ba_cb(struct ieee80211_vif *vif, u8 *ra, u16 tid)
|
|
}
|
|
}
|
|
|
|
|
|
mutex_lock(&sta->ampdu_mlme.mtx);
|
|
mutex_lock(&sta->ampdu_mlme.mtx);
|
|
- tid_tx = sta->ampdu_mlme.tid_tx[tid];
|
|
|
|
|
|
+ tid_tx = rcu_dereference_protected_tid_tx(sta, tid);
|
|
|
|
|
|
if (WARN_ON(!tid_tx)) {
|
|
if (WARN_ON(!tid_tx)) {
|
|
#ifdef CONFIG_MAC80211_HT_DEBUG
|
|
#ifdef CONFIG_MAC80211_HT_DEBUG
|
|
@@ -626,7 +632,7 @@ int ieee80211_stop_tx_ba_session(struct ieee80211_sta *pubsta, u16 tid)
|
|
return -EINVAL;
|
|
return -EINVAL;
|
|
|
|
|
|
spin_lock_bh(&sta->lock);
|
|
spin_lock_bh(&sta->lock);
|
|
- tid_tx = sta->ampdu_mlme.tid_tx[tid];
|
|
|
|
|
|
+ tid_tx = rcu_dereference_protected_tid_tx(sta, tid);
|
|
|
|
|
|
if (!tid_tx) {
|
|
if (!tid_tx) {
|
|
ret = -ENOENT;
|
|
ret = -ENOENT;
|
|
@@ -682,7 +688,7 @@ void ieee80211_stop_tx_ba_cb(struct ieee80211_vif *vif, u8 *ra, u8 tid)
|
|
|
|
|
|
mutex_lock(&sta->ampdu_mlme.mtx);
|
|
mutex_lock(&sta->ampdu_mlme.mtx);
|
|
spin_lock_bh(&sta->lock);
|
|
spin_lock_bh(&sta->lock);
|
|
- tid_tx = sta->ampdu_mlme.tid_tx[tid];
|
|
|
|
|
|
+ tid_tx = rcu_dereference_protected_tid_tx(sta, tid);
|
|
|
|
|
|
if (!tid_tx || !test_bit(HT_AGG_STATE_STOPPING, &tid_tx->state)) {
|
|
if (!tid_tx || !test_bit(HT_AGG_STATE_STOPPING, &tid_tx->state)) {
|
|
#ifdef CONFIG_MAC80211_HT_DEBUG
|
|
#ifdef CONFIG_MAC80211_HT_DEBUG
|
|
@@ -763,7 +769,7 @@ void ieee80211_process_addba_resp(struct ieee80211_local *local,
|
|
|
|
|
|
mutex_lock(&sta->ampdu_mlme.mtx);
|
|
mutex_lock(&sta->ampdu_mlme.mtx);
|
|
|
|
|
|
- tid_tx = sta->ampdu_mlme.tid_tx[tid];
|
|
|
|
|
|
+ tid_tx = rcu_dereference_protected_tid_tx(sta, tid);
|
|
if (!tid_tx)
|
|
if (!tid_tx)
|
|
goto out;
|
|
goto out;
|
|
|
|
|