diff options
Diffstat (limited to 'net/mac80211/agg-tx.c')
| -rw-r--r-- | net/mac80211/agg-tx.c | 57 | 
1 files changed, 29 insertions, 28 deletions
| diff --git a/net/mac80211/agg-tx.c b/net/mac80211/agg-tx.c index cbd48762256c..5f8ab5be369f 100644 --- a/net/mac80211/agg-tx.c +++ b/net/mac80211/agg-tx.c @@ -226,7 +226,11 @@ ieee80211_agg_start_txq(struct sta_info *sta, int tid, bool enable)  		clear_bit(IEEE80211_TXQ_AMPDU, &txqi->flags);  	clear_bit(IEEE80211_TXQ_STOP, &txqi->flags); +	local_bh_disable(); +	rcu_read_lock();  	drv_wake_tx_queue(sta->sdata->local, txqi); +	rcu_read_unlock(); +	local_bh_enable();  }  /* @@ -326,6 +330,11 @@ int ___ieee80211_stop_tx_ba_session(struct sta_info *sta, u16 tid,  	spin_lock_bh(&sta->lock); +	/* free struct pending for start, if present */ +	tid_tx = sta->ampdu_mlme.tid_start_tx[tid]; +	kfree(tid_tx); +	sta->ampdu_mlme.tid_start_tx[tid] = NULL; +  	tid_tx = rcu_dereference_protected_tid_tx(sta, tid);  	if (!tid_tx) {  		spin_unlock_bh(&sta->lock); @@ -418,15 +427,12 @@ int ___ieee80211_stop_tx_ba_session(struct sta_info *sta, u16 tid,   * add Block Ack response will arrive from the recipient.   * If this timer expires sta_addba_resp_timer_expired will be executed.   */ -static void sta_addba_resp_timer_expired(unsigned long data) +static void sta_addba_resp_timer_expired(struct timer_list *t)  { -	/* not an elegant detour, but there is no choice as the timer passes -	 * only one argument, and both sta_info and TID are needed, so init -	 * flow in sta_info_create gives the TID as data, while the timer_to_id -	 * array gives the sta through container_of */ -	u16 tid = *(u8 *)data; -	struct sta_info *sta = container_of((void *)data, -		struct sta_info, timer_to_tid[tid]); +	struct tid_ampdu_tx *tid_tx_timer = +		from_timer(tid_tx_timer, t, addba_resp_timer); +	struct sta_info *sta = tid_tx_timer->sta; +	u8 tid = tid_tx_timer->tid;  	struct tid_ampdu_tx *tid_tx;  	/* check if the TID waits for addBA response */ @@ -436,7 +442,7 @@ static void sta_addba_resp_timer_expired(unsigned long data)  	    test_bit(HT_AGG_STATE_RESPONSE_RECEIVED, &tid_tx->state)) {  		rcu_read_unlock();  		ht_dbg(sta->sdata, -		       "timer expired on %pM tid %d but we are not (or no longer) expecting addBA response there\n", +		       "timer expired on %pM tid %d not expecting addBA response\n",  		       sta->sta.addr, tid);  		return;  	} @@ -521,21 +527,17 @@ void ieee80211_tx_ba_session_handle_start(struct sta_info *sta, int tid)   * After accepting the AddBA Response we activated a timer,   * resetting it after each frame that we send.   */ -static void sta_tx_agg_session_timer_expired(unsigned long data) +static void sta_tx_agg_session_timer_expired(struct timer_list *t)  { -	/* not an elegant detour, but there is no choice as the timer passes -	 * only one argument, and various sta_info are needed here, so init -	 * flow in sta_info_create gives the TID as data, while the timer_to_id -	 * array gives the sta through container_of */ -	u8 *ptid = (u8 *)data; -	u8 *timer_to_id = ptid - *ptid; -	struct sta_info *sta = container_of(timer_to_id, struct sta_info, -					 timer_to_tid[0]); +	struct tid_ampdu_tx *tid_tx_timer = +		from_timer(tid_tx_timer, t, session_timer); +	struct sta_info *sta = tid_tx_timer->sta; +	u8 tid = tid_tx_timer->tid;  	struct tid_ampdu_tx *tid_tx;  	unsigned long timeout;  	rcu_read_lock(); -	tid_tx = rcu_dereference(sta->ampdu_mlme.tid_tx[*ptid]); +	tid_tx = rcu_dereference(sta->ampdu_mlme.tid_tx[tid]);  	if (!tid_tx || test_bit(HT_AGG_STATE_STOPPING, &tid_tx->state)) {  		rcu_read_unlock();  		return; @@ -551,9 +553,9 @@ static void sta_tx_agg_session_timer_expired(unsigned long data)  	rcu_read_unlock();  	ht_dbg(sta->sdata, "tx session timer expired on %pM tid %d\n", -	       sta->sta.addr, (u16)*ptid); +	       sta->sta.addr, tid); -	ieee80211_stop_tx_ba_session(&sta->sta, *ptid); +	ieee80211_stop_tx_ba_session(&sta->sta, tid);  }  int ieee80211_start_tx_ba_session(struct ieee80211_sta *pubsta, u16 tid, @@ -639,7 +641,7 @@ int ieee80211_start_tx_ba_session(struct ieee80211_sta *pubsta, u16 tid,  	    time_before(jiffies, sta->ampdu_mlme.last_addba_req_time[tid] +  			HT_AGG_RETRIES_PERIOD)) {  		ht_dbg(sdata, -		       "BA request denied - waiting a grace period after %d failed requests on %pM tid %u\n", +		       "BA request denied - %d failed requests on %pM tid %u\n",  		       sta->ampdu_mlme.addba_req_num[tid], sta->sta.addr, tid);  		ret = -EBUSY;  		goto err_unlock_sta; @@ -666,16 +668,15 @@ int ieee80211_start_tx_ba_session(struct ieee80211_sta *pubsta, u16 tid,  	__set_bit(HT_AGG_STATE_WANT_START, &tid_tx->state);  	tid_tx->timeout = timeout; +	tid_tx->sta = sta; +	tid_tx->tid = tid;  	/* response timer */ -	setup_timer(&tid_tx->addba_resp_timer, -		    sta_addba_resp_timer_expired, -		    (unsigned long)&sta->timer_to_tid[tid]); +	timer_setup(&tid_tx->addba_resp_timer, sta_addba_resp_timer_expired, 0);  	/* tx timer */ -	setup_deferrable_timer(&tid_tx->session_timer, -			       sta_tx_agg_session_timer_expired, -			       (unsigned long)&sta->timer_to_tid[tid]); +	timer_setup(&tid_tx->session_timer, +		    sta_tx_agg_session_timer_expired, TIMER_DEFERRABLE);  	/* assign a dialog token */  	sta->ampdu_mlme.dialog_token_allocator++; | 
