2007-05-05 11:45:53 -07:00
/*
* Copyright 2004 , Instant802 Networks , Inc .
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*/
# include <linux/netdevice.h>
# include <linux/skbuff.h>
# include <linux/module.h>
# include <linux/if_arp.h>
# include <linux/types.h>
# include <net/ip.h>
# include <net/pkt_sched.h>
# include <net/mac80211.h>
# include "ieee80211_i.h"
# include "wme.h"
2008-07-15 03:34:57 -07:00
/* Default mapping in classifier to work with default
2008-04-30 18:51:21 +02:00
* queue setup .
*/
2010-11-16 11:49:58 -08:00
const int ieee802_1d_to_ac [ 8 ] = {
IEEE80211_AC_BE ,
IEEE80211_AC_BK ,
IEEE80211_AC_BK ,
IEEE80211_AC_BE ,
IEEE80211_AC_VI ,
IEEE80211_AC_VI ,
IEEE80211_AC_VO ,
IEEE80211_AC_VO
} ;
2007-05-05 11:45:53 -07:00
2008-07-15 03:34:57 -07:00
static int wme_downgrade_ac ( struct sk_buff * skb )
2007-05-05 11:45:53 -07:00
{
switch ( skb - > priority ) {
case 6 :
case 7 :
skb - > priority = 5 ; /* VO -> VI */
return 0 ;
case 4 :
case 5 :
skb - > priority = 3 ; /* VI -> BE */
return 0 ;
case 0 :
case 3 :
skb - > priority = 2 ; /* BE -> BK */
return 0 ;
default :
return - 1 ;
}
}
2012-06-20 15:39:13 +03:00
static u16 ieee80211_downgrade_queue ( struct ieee80211_sub_if_data * sdata ,
2012-03-27 14:18:38 +02:00
struct sk_buff * skb )
{
/* in case we are a client verify acm is not set for this ac */
2012-06-20 15:39:13 +03:00
while ( unlikely ( sdata - > wmm_acm & BIT ( skb - > priority ) ) ) {
2012-03-27 14:18:38 +02:00
if ( wme_downgrade_ac ( skb ) ) {
/*
* This should not really happen . The AP has marked all
* lower ACs to require admission control which is not
* a reasonable configuration . Allow the frame to be
* transmitted using AC_BK as a workaround .
*/
break ;
}
}
/* look up which queue to use for frames with this 1d tag */
return ieee802_1d_to_ac [ skb - > priority ] ;
}
2011-11-24 17:15:23 -08:00
/* Indicate which queue to use for this fully formed 802.11 frame */
2012-06-20 15:39:13 +03:00
u16 ieee80211_select_queue_80211 ( struct ieee80211_sub_if_data * sdata ,
2011-11-24 17:15:23 -08:00
struct sk_buff * skb ,
struct ieee80211_hdr * hdr )
{
2012-06-20 15:39:13 +03:00
struct ieee80211_local * local = sdata - > local ;
2011-11-24 17:15:23 -08:00
u8 * p ;
2012-03-28 11:04:29 +02:00
if ( local - > hw . queues < IEEE80211_NUM_ACS )
2011-11-24 17:15:23 -08:00
return 0 ;
if ( ! ieee80211_is_data ( hdr - > frame_control ) ) {
skb - > priority = 7 ;
return ieee802_1d_to_ac [ skb - > priority ] ;
}
if ( ! ieee80211_is_data_qos ( hdr - > frame_control ) ) {
skb - > priority = 0 ;
return ieee802_1d_to_ac [ skb - > priority ] ;
}
p = ieee80211_get_qos_ctl ( hdr ) ;
skb - > priority = * p & IEEE80211_QOS_CTL_TAG1D_MASK ;
2012-06-20 15:39:13 +03:00
return ieee80211_downgrade_queue ( sdata , skb ) ;
2011-11-24 17:15:23 -08:00
}
2007-05-05 11:45:53 -07:00
2010-01-05 18:00:58 +01:00
/* Indicate which queue to use. */
u16 ieee80211_select_queue ( struct ieee80211_sub_if_data * sdata ,
struct sk_buff * skb )
2007-05-05 11:45:53 -07:00
{
2010-01-05 18:00:58 +01:00
struct ieee80211_local * local = sdata - > local ;
struct sta_info * sta = NULL ;
const u8 * ra = NULL ;
bool qos = false ;
2007-05-05 11:45:53 -07:00
2012-03-28 11:04:29 +02:00
if ( local - > hw . queues < IEEE80211_NUM_ACS | | skb - > len < 6 ) {
2010-01-05 18:00:58 +01:00
skb - > priority = 0 ; /* required for correct WPA/11i MIC */
2012-03-28 11:04:26 +02:00
return 0 ;
2010-01-05 18:00:58 +01:00
}
rcu_read_lock ( ) ;
switch ( sdata - > vif . type ) {
case NL80211_IFTYPE_AP_VLAN :
sta = rcu_dereference ( sdata - > u . vlan . sta ) ;
2010-12-22 10:15:30 +01:00
if ( sta ) {
2011-09-29 16:04:36 +02:00
qos = test_sta_flag ( sta , WLAN_STA_WME ) ;
2010-01-05 18:00:58 +01:00
break ;
2010-12-22 10:15:30 +01:00
}
2010-01-05 18:00:58 +01:00
case NL80211_IFTYPE_AP :
ra = skb - > data ;
break ;
case NL80211_IFTYPE_WDS :
ra = sdata - > u . wds . remote_addr ;
break ;
# ifdef CONFIG_MAC80211_MESH
case NL80211_IFTYPE_MESH_POINT :
2011-11-03 21:11:13 -07:00
qos = true ;
2010-01-05 18:00:58 +01:00
break ;
# endif
case NL80211_IFTYPE_STATION :
ra = sdata - > u . mgd . bssid ;
break ;
case NL80211_IFTYPE_ADHOC :
ra = skb - > data ;
break ;
default :
break ;
2007-05-05 11:45:53 -07:00
}
2010-01-05 18:00:58 +01:00
if ( ! sta & & ra & & ! is_multicast_ether_addr ( ra ) ) {
sta = sta_info_get ( sdata , ra ) ;
if ( sta )
2011-09-29 16:04:36 +02:00
qos = test_sta_flag ( sta , WLAN_STA_WME ) ;
2007-05-05 11:45:53 -07:00
}
2010-01-05 18:00:58 +01:00
rcu_read_unlock ( ) ;
if ( ! qos ) {
2007-05-05 11:45:53 -07:00
skb - > priority = 0 ; /* required for correct WPA/11i MIC */
2010-12-22 10:15:30 +01:00
return IEEE80211_AC_BE ;
2007-05-05 11:45:53 -07:00
}
/* use the data classifier to determine what 802.1d tag the
2007-08-28 17:01:55 -04:00
* data frame has */
2009-05-21 21:47:03 +08:00
skb - > priority = cfg80211_classify8021d ( skb ) ;
2007-05-05 11:45:53 -07:00
2012-06-20 15:39:13 +03:00
return ieee80211_downgrade_queue ( sdata , skb ) ;
2010-01-05 18:00:58 +01:00
}
2012-11-21 18:40:31 -08:00
/**
* ieee80211_set_qos_hdr - Fill in the QoS header if there is one .
*
* @ sdata : local subif
* @ skb : packet to be updated
*/
2011-09-07 17:49:53 -07:00
void ieee80211_set_qos_hdr ( struct ieee80211_sub_if_data * sdata ,
struct sk_buff * skb )
2007-05-05 11:45:53 -07:00
{
2010-01-05 18:00:58 +01:00
struct ieee80211_hdr * hdr = ( void * ) skb - > data ;
2011-11-18 14:20:44 +01:00
struct ieee80211_tx_info * info = IEEE80211_SKB_CB ( skb ) ;
2012-11-21 18:40:31 -08:00
u8 * p ;
u8 ack_policy , tid ;
2010-01-05 18:00:58 +01:00
2012-11-21 18:40:31 -08:00
if ( ! ieee80211_is_data_qos ( hdr - > frame_control ) )
return ;
2010-01-05 18:00:58 +01:00
2012-11-21 18:40:31 -08:00
p = ieee80211_get_qos_ctl ( hdr ) ;
tid = skb - > priority & IEEE80211_QOS_CTL_TAG1D_MASK ;
2011-11-03 09:59:39 +01:00
2012-11-21 18:40:31 -08:00
/* preserve EOSP bit */
ack_policy = * p & IEEE80211_QOS_CTL_EOSP ;
2011-11-18 14:20:44 +01:00
2012-11-21 18:40:31 -08:00
if ( is_multicast_ether_addr ( hdr - > addr1 ) | |
sdata - > noack_map & BIT ( tid ) ) {
ack_policy | = IEEE80211_QOS_CTL_ACK_POLICY_NOACK ;
info - > flags | = IEEE80211_TX_CTL_NO_ACK ;
2007-05-05 11:45:53 -07:00
}
2012-11-21 18:40:31 -08:00
/* qos header is 2 bytes */
* p + + = ack_policy | tid ;
2013-01-30 18:14:08 +01:00
if ( ieee80211_vif_is_mesh ( & sdata - > vif ) ) {
/* preserve RSPI and Mesh PS Level bit */
* p & = ( ( IEEE80211_QOS_CTL_RSPI |
IEEE80211_QOS_CTL_MESH_PS_LEVEL ) > > 8 ) ;
/* Nulls don't have a mesh header (frame body) */
if ( ! ieee80211_is_qos_nullfunc ( hdr - > frame_control ) )
* p | = ( IEEE80211_QOS_CTL_MESH_CONTROL_PRESENT > > 8 ) ;
} else {
* p = 0 ;
}
2007-05-05 11:45:53 -07:00
}