diff options
Diffstat (limited to 'net/mac80211/agg-rx.c')
| -rw-r--r-- | net/mac80211/agg-rx.c | 167 | 
1 files changed, 95 insertions, 72 deletions
diff --git a/net/mac80211/agg-rx.c b/net/mac80211/agg-rx.c index 720b7a84af5..31bf2586fb8 100644 --- a/net/mac80211/agg-rx.c +++ b/net/mac80211/agg-rx.c @@ -38,6 +38,7 @@  #include <linux/ieee80211.h>  #include <linux/slab.h> +#include <linux/export.h>  #include <net/mac80211.h>  #include "ieee80211_i.h"  #include "driver-ops.h" @@ -48,6 +49,8 @@ static void ieee80211_free_tid_rx(struct rcu_head *h)  		container_of(h, struct tid_ampdu_rx, rcu_head);  	int i; +	del_timer_sync(&tid_rx->reorder_timer); +  	for (i = 0; i < tid_rx->buf_size; i++)  		dev_kfree_skb(tid_rx->reorder_buf[i]);  	kfree(tid_rx->reorder_buf); @@ -63,30 +66,32 @@ void ___ieee80211_stop_rx_ba_session(struct sta_info *sta, u16 tid,  	lockdep_assert_held(&sta->ampdu_mlme.mtx); -	tid_rx = sta->ampdu_mlme.tid_rx[tid]; +	tid_rx = rcu_dereference_protected(sta->ampdu_mlme.tid_rx[tid], +					lockdep_is_held(&sta->ampdu_mlme.mtx));  	if (!tid_rx)  		return; -	rcu_assign_pointer(sta->ampdu_mlme.tid_rx[tid], NULL); +	RCU_INIT_POINTER(sta->ampdu_mlme.tid_rx[tid], NULL); -#ifdef CONFIG_MAC80211_HT_DEBUG -	printk(KERN_DEBUG "Rx BA session stop requested for %pM tid %u\n", -	       sta->sta.addr, tid); -#endif /* CONFIG_MAC80211_HT_DEBUG */ +	ht_dbg(sta->sdata, +	       "Rx BA session stop requested for %pM tid %u %s reason: %d\n", +	       sta->sta.addr, tid, +	       initiator == WLAN_BACK_RECIPIENT ? "recipient" : "inititator", +	       (int)reason);  	if (drv_ampdu_action(local, sta->sdata, IEEE80211_AMPDU_RX_STOP, -			     &sta->sta, tid, NULL)) -		printk(KERN_DEBUG "HW problem - can not stop rx " -				"aggregation for tid %d\n", tid); +			     &sta->sta, tid, NULL, 0)) +		sdata_info(sta->sdata, +			   "HW problem - can not stop rx aggregation for %pM tid %d\n", +			   sta->sta.addr, tid);  	/* check if this is a self generated aggregation halt */  	if (initiator == WLAN_BACK_RECIPIENT && tx)  		ieee80211_send_delba(sta->sdata, sta->sta.addr, -				     tid, 0, reason); +				     tid, WLAN_BACK_RECIPIENT, reason);  	del_timer_sync(&tid_rx->session_timer); -	del_timer_sync(&tid_rx->reorder_timer);  	call_rcu(&tid_rx->rcu_head, ieee80211_free_tid_rx);  } @@ -99,6 +104,29 @@ void __ieee80211_stop_rx_ba_session(struct sta_info *sta, u16 tid,  	mutex_unlock(&sta->ampdu_mlme.mtx);  } +void ieee80211_stop_rx_ba_session(struct ieee80211_vif *vif, u16 ba_rx_bitmap, +				  const u8 *addr) +{ +	struct ieee80211_sub_if_data *sdata = vif_to_sdata(vif); +	struct sta_info *sta; +	int i; + +	rcu_read_lock(); +	sta = sta_info_get_bss(sdata, addr); +	if (!sta) { +		rcu_read_unlock(); +		return; +	} + +	for (i = 0; i < IEEE80211_NUM_TIDS; i++) +		if (ba_rx_bitmap & BIT(i)) +			set_bit(i, sta->ampdu_mlme.tid_rx_stop_requested); + +	ieee80211_queue_work(&sta->local->hw, &sta->ampdu_mlme.work); +	rcu_read_unlock(); +} +EXPORT_SYMBOL(ieee80211_stop_rx_ba_session); +  /*   * After accepting the AddBA Request we activated a timer,   * resetting it after each frame that arrives from the originator. @@ -113,10 +141,27 @@ static void sta_rx_agg_session_timer_expired(unsigned long 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_rx *tid_rx; +	unsigned long timeout; + +	rcu_read_lock(); +	tid_rx = rcu_dereference(sta->ampdu_mlme.tid_rx[*ptid]); +	if (!tid_rx) { +		rcu_read_unlock(); +		return; +	} + +	timeout = tid_rx->last_rx + TU_TO_JIFFIES(tid_rx->timeout); +	if (time_is_after_jiffies(timeout)) { +		mod_timer(&tid_rx->session_timer, timeout); +		rcu_read_unlock(); +		return; +	} +	rcu_read_unlock(); + +	ht_dbg(sta->sdata, "RX session timer expired on %pM tid %d\n", +	       sta->sta.addr, (u16)*ptid); -#ifdef CONFIG_MAC80211_HT_DEBUG -	printk(KERN_DEBUG "rx session timer expired on tid %d\n", (u16)*ptid); -#endif  	set_bit(*ptid, sta->ampdu_mlme.tid_rx_timer_expired);  	ieee80211_queue_work(&sta->local->hw, &sta->ampdu_mlme.work);  } @@ -129,9 +174,7 @@ static void sta_rx_agg_reorder_timer_expired(unsigned long data)  			timer_to_tid[0]);  	rcu_read_lock(); -	spin_lock(&sta->lock);  	ieee80211_release_reorder_timeout(sta, *ptid); -	spin_unlock(&sta->lock);  	rcu_read_unlock();  } @@ -145,12 +188,8 @@ static void ieee80211_send_addba_resp(struct ieee80211_sub_if_data *sdata, u8 *d  	u16 capab;  	skb = dev_alloc_skb(sizeof(*mgmt) + local->hw.extra_tx_headroom); - -	if (!skb) { -		printk(KERN_DEBUG "%s: failed to allocate buffer " -		       "for addba resp frame\n", sdata->name); +	if (!skb)  		return; -	}  	skb_reserve(skb, local->hw.extra_tx_headroom);  	mgmt = (struct ieee80211_mgmt *) skb_put(skb, 24); @@ -158,10 +197,13 @@ static void ieee80211_send_addba_resp(struct ieee80211_sub_if_data *sdata, u8 *d  	memcpy(mgmt->da, da, ETH_ALEN);  	memcpy(mgmt->sa, sdata->vif.addr, ETH_ALEN);  	if (sdata->vif.type == NL80211_IFTYPE_AP || -	    sdata->vif.type == NL80211_IFTYPE_AP_VLAN) +	    sdata->vif.type == NL80211_IFTYPE_AP_VLAN || +	    sdata->vif.type == NL80211_IFTYPE_MESH_POINT)  		memcpy(mgmt->bssid, sdata->vif.addr, ETH_ALEN);  	else if (sdata->vif.type == NL80211_IFTYPE_STATION)  		memcpy(mgmt->bssid, sdata->u.mgd.bssid, ETH_ALEN); +	else if (sdata->vif.type == NL80211_IFTYPE_ADHOC) +		memcpy(mgmt->bssid, sdata->u.ibss.bssid, ETH_ALEN);  	mgmt->frame_control = cpu_to_le16(IEEE80211_FTYPE_MGMT |  					  IEEE80211_STYPE_ACTION); @@ -187,8 +229,6 @@ void ieee80211_process_addba_request(struct ieee80211_local *local,  				     struct ieee80211_mgmt *mgmt,  				     size_t len)  { -	struct ieee80211_hw *hw = &local->hw; -	struct ieee80211_conf *conf = &hw->conf;  	struct tid_ampdu_rx *tid_agg_rx;  	u16 capab, tid, timeout, ba_policy, buf_size, start_seq_num, status;  	u8 dialog_token; @@ -207,11 +247,10 @@ void ieee80211_process_addba_request(struct ieee80211_local *local,  	status = WLAN_STATUS_REQUEST_DECLINED; -	if (test_sta_flags(sta, WLAN_STA_BLOCK_BA)) { -#ifdef CONFIG_MAC80211_HT_DEBUG -		printk(KERN_DEBUG "Suspend in progress. " -		       "Denying ADDBA request\n"); -#endif +	if (test_sta_flag(sta, WLAN_STA_BLOCK_BA)) { +		ht_dbg(sta->sdata, +		       "Suspend in progress - Denying ADDBA request (%pM tid %d)\n", +		       sta->sta.addr, tid);  		goto end_no_lock;  	} @@ -223,55 +262,44 @@ void ieee80211_process_addba_request(struct ieee80211_local *local,  	     (!(sta->sta.ht_cap.cap & IEEE80211_HT_CAP_DELAY_BA))) ||  	    (buf_size > IEEE80211_MAX_AMPDU_BUF)) {  		status = WLAN_STATUS_INVALID_QOS_PARAM; -#ifdef CONFIG_MAC80211_HT_DEBUG -		if (net_ratelimit()) -			printk(KERN_DEBUG "AddBA Req with bad params from " -				"%pM on tid %u. policy %d, buffer size %d\n", -				mgmt->sa, tid, ba_policy, -				buf_size); -#endif /* CONFIG_MAC80211_HT_DEBUG */ +		ht_dbg_ratelimited(sta->sdata, +				   "AddBA Req with bad params from %pM on tid %u. policy %d, buffer size %d\n", +				   mgmt->sa, tid, ba_policy, buf_size);  		goto end_no_lock;  	}  	/* determine default buffer size */ -	if (buf_size == 0) { -		struct ieee80211_supported_band *sband; - -		sband = local->hw.wiphy->bands[conf->channel->band]; -		buf_size = IEEE80211_MIN_AMPDU_BUF; -		buf_size = buf_size << sband->ht_cap.ampdu_factor; -	} +	if (buf_size == 0) +		buf_size = IEEE80211_MAX_AMPDU_BUF; +	/* make sure the size doesn't exceed the maximum supported by the hw */ +	if (buf_size > local->hw.max_rx_aggregation_subframes) +		buf_size = local->hw.max_rx_aggregation_subframes;  	/* examine state machine */  	mutex_lock(&sta->ampdu_mlme.mtx);  	if (sta->ampdu_mlme.tid_rx[tid]) { -#ifdef CONFIG_MAC80211_HT_DEBUG -		if (net_ratelimit()) -			printk(KERN_DEBUG "unexpected AddBA Req from " -				"%pM on tid %u\n", -				mgmt->sa, tid); -#endif /* CONFIG_MAC80211_HT_DEBUG */ -		goto end; +		ht_dbg_ratelimited(sta->sdata, +				   "unexpected AddBA Req from %pM on tid %u\n", +				   mgmt->sa, tid); + +		/* delete existing Rx BA session on the same tid */ +		___ieee80211_stop_rx_ba_session(sta, tid, WLAN_BACK_RECIPIENT, +						WLAN_STATUS_UNSPECIFIED_QOS, +						false);  	}  	/* prepare A-MPDU MLME for Rx aggregation */ -	tid_agg_rx = kmalloc(sizeof(struct tid_ampdu_rx), GFP_ATOMIC); -	if (!tid_agg_rx) { -#ifdef CONFIG_MAC80211_HT_DEBUG -		if (net_ratelimit()) -			printk(KERN_ERR "allocate rx mlme to tid %d failed\n", -					tid); -#endif +	tid_agg_rx = kmalloc(sizeof(struct tid_ampdu_rx), GFP_KERNEL); +	if (!tid_agg_rx)  		goto end; -	}  	spin_lock_init(&tid_agg_rx->reorder_lock);  	/* rx timer */  	tid_agg_rx->session_timer.function = sta_rx_agg_session_timer_expired;  	tid_agg_rx->session_timer.data = (unsigned long)&sta->timer_to_tid[tid]; -	init_timer(&tid_agg_rx->session_timer); +	init_timer_deferrable(&tid_agg_rx->session_timer);  	/* rx reorder timer */  	tid_agg_rx->reorder_timer.function = sta_rx_agg_reorder_timer_expired; @@ -280,15 +308,10 @@ void ieee80211_process_addba_request(struct ieee80211_local *local,  	/* prepare reordering buffer */  	tid_agg_rx->reorder_buf = -		kcalloc(buf_size, sizeof(struct sk_buff *), GFP_ATOMIC); +		kcalloc(buf_size, sizeof(struct sk_buff *), GFP_KERNEL);  	tid_agg_rx->reorder_time = -		kcalloc(buf_size, sizeof(unsigned long), GFP_ATOMIC); +		kcalloc(buf_size, sizeof(unsigned long), GFP_KERNEL);  	if (!tid_agg_rx->reorder_buf || !tid_agg_rx->reorder_time) { -#ifdef CONFIG_MAC80211_HT_DEBUG -		if (net_ratelimit()) -			printk(KERN_ERR "can not allocate reordering buffer " -			       "to tid %d\n", tid); -#endif  		kfree(tid_agg_rx->reorder_buf);  		kfree(tid_agg_rx->reorder_time);  		kfree(tid_agg_rx); @@ -296,11 +319,9 @@ void ieee80211_process_addba_request(struct ieee80211_local *local,  	}  	ret = drv_ampdu_action(local, sta->sdata, IEEE80211_AMPDU_RX_START, -			       &sta->sta, tid, &start_seq_num); -#ifdef CONFIG_MAC80211_HT_DEBUG -	printk(KERN_DEBUG "Rx A-MPDU request on tid %d result %d\n", tid, ret); -#endif /* CONFIG_MAC80211_HT_DEBUG */ - +			       &sta->sta, tid, &start_seq_num, 0); +	ht_dbg(sta->sdata, "Rx A-MPDU request on %pM tid %d result %d\n", +	       sta->sta.addr, tid, ret);  	if (ret) {  		kfree(tid_agg_rx->reorder_buf);  		kfree(tid_agg_rx->reorder_time); @@ -320,8 +341,10 @@ void ieee80211_process_addba_request(struct ieee80211_local *local,  	/* activate it for RX */  	rcu_assign_pointer(sta->ampdu_mlme.tid_rx[tid], tid_agg_rx); -	if (timeout) +	if (timeout) {  		mod_timer(&tid_agg_rx->session_timer, TU_TO_EXP_TIME(timeout)); +		tid_agg_rx->last_rx = jiffies; +	}  end:  	mutex_unlock(&sta->ampdu_mlme.mtx);  | 
