2007-05-05 22:45:53 +04: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"
/* maximum number of hardware queues we support. */
2008-04-30 20:51:21 +04:00
# define QD_MAX_QUEUES (IEEE80211_MAX_AMPDU_QUEUES + IEEE80211_MAX_QUEUES)
/* current number of hardware queues we support. */
# define QD_NUM(hw) ((hw)->queues + (hw)->ampdu_queues)
2008-01-28 15:07:18 +03:00
2008-04-30 20:51:21 +04:00
/*
* Default mapping in classifier to work with default
* queue setup .
*/
2008-01-28 15:07:18 +03:00
const int ieee802_1d_to_ac [ 8 ] = { 2 , 3 , 3 , 2 , 1 , 1 , 0 , 0 } ;
2007-05-05 22:45:53 +04:00
struct ieee80211_sched_data
{
2008-04-30 20:51:21 +04:00
unsigned long qdisc_pool [ BITS_TO_LONGS ( QD_MAX_QUEUES ) ] ;
2007-05-05 22:45:53 +04:00
struct tcf_proto * filter_list ;
2008-04-30 20:51:21 +04:00
struct Qdisc * queues [ QD_MAX_QUEUES ] ;
struct sk_buff_head requeued [ QD_MAX_QUEUES ] ;
2007-05-05 22:45:53 +04:00
} ;
2008-01-09 20:12:48 +03:00
static const char llc_ip_hdr [ 8 ] = { 0xAA , 0xAA , 0x3 , 0 , 0 , 0 , 0x08 , 0 } ;
2007-05-05 22:45:53 +04:00
/* given a data frame determine the 802.1p/1d tag to use */
static inline unsigned classify_1d ( struct sk_buff * skb , struct Qdisc * qd )
{
struct iphdr * ip ;
int dscp ;
int offset ;
struct ieee80211_sched_data * q = qdisc_priv ( qd ) ;
struct tcf_result res = { - 1 , 0 } ;
/* if there is a user set filter list, call out to that */
if ( q - > filter_list ) {
tc_classify ( skb , q - > filter_list , & res ) ;
if ( res . class ! = - 1 )
return res . class ;
}
/* skb->priority values from 256->263 are magic values to
* directly indicate a specific 802.1 d priority .
* This is used to allow 802.1 d priority to be passed directly in
* from VLAN tags , etc . */
if ( skb - > priority > = 256 & & skb - > priority < = 263 )
return skb - > priority - 256 ;
/* check there is a valid IP header present */
2008-01-09 20:12:48 +03:00
offset = ieee80211_get_hdrlen_from_skb ( skb ) ;
if ( skb - > len < offset + sizeof ( llc_ip_hdr ) + sizeof ( * ip ) | |
memcmp ( skb - > data + offset , llc_ip_hdr , sizeof ( llc_ip_hdr ) ) )
2007-05-05 22:45:53 +04:00
return 0 ;
2008-01-09 20:12:48 +03:00
ip = ( struct iphdr * ) ( skb - > data + offset + sizeof ( llc_ip_hdr ) ) ;
2007-05-05 22:45:53 +04:00
dscp = ip - > tos & 0xfc ;
if ( dscp & 0x1c )
return 0 ;
return dscp > > 5 ;
}
static inline int wme_downgrade_ac ( struct sk_buff * skb )
{
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 ;
}
}
/* positive return value indicates which queue to use
* negative return value indicates to drop the frame */
2008-04-30 20:51:21 +04:00
static int classify80211 ( struct sk_buff * skb , struct Qdisc * qd )
2007-05-05 22:45:53 +04:00
{
struct ieee80211_local * local = wdev_priv ( qd - > dev - > ieee80211_ptr ) ;
struct ieee80211_hdr * hdr = ( struct ieee80211_hdr * ) skb - > data ;
2008-06-12 01:21:59 +04:00
if ( ! ieee80211_is_data ( hdr - > frame_control ) ) {
2007-05-05 22:45:53 +04:00
/* management frames go on AC_VO queue, but are sent
* without QoS control fields */
2008-04-30 20:51:21 +04:00
return 0 ;
2007-05-05 22:45:53 +04:00
}
2007-09-28 16:02:09 +04:00
if ( 0 /* injected */ ) {
/* use AC from radiotap */
2007-05-05 22:45:53 +04:00
}
2008-06-12 01:21:59 +04:00
if ( ! ieee80211_is_data_qos ( hdr - > frame_control ) ) {
2007-05-05 22:45:53 +04:00
skb - > priority = 0 ; /* required for correct WPA/11i MIC */
return ieee802_1d_to_ac [ skb - > priority ] ;
}
/* use the data classifier to determine what 802.1d tag the
2007-08-29 01:01:55 +04:00
* data frame has */
2007-05-05 22:45:53 +04:00
skb - > priority = classify_1d ( skb , qd ) ;
2007-08-29 01:01:55 +04:00
/* in case we are a client verify acm is not set for this ac */
2007-05-05 22:45:53 +04:00
while ( unlikely ( local - > wmm_acm & BIT ( skb - > priority ) ) ) {
if ( wme_downgrade_ac ( skb ) ) {
2007-08-29 01:01:55 +04:00
/* No AC with lower priority has acm=0, drop packet. */
2007-05-05 22:45:53 +04:00
return - 1 ;
}
}
/* look up which queue to use for frames with this 1d tag */
return ieee802_1d_to_ac [ skb - > priority ] ;
}
static int wme_qdiscop_enqueue ( struct sk_buff * skb , struct Qdisc * qd )
{
struct ieee80211_local * local = wdev_priv ( qd - > dev - > ieee80211_ptr ) ;
2008-04-30 20:51:21 +04:00
struct ieee80211_hw * hw = & local - > hw ;
2007-05-05 22:45:53 +04:00
struct ieee80211_sched_data * q = qdisc_priv ( qd ) ;
2008-05-15 14:55:29 +04:00
struct ieee80211_tx_info * info = IEEE80211_SKB_CB ( skb ) ;
2007-05-05 22:45:53 +04:00
struct ieee80211_hdr * hdr = ( struct ieee80211_hdr * ) skb - > data ;
struct Qdisc * qdisc ;
2008-01-28 15:07:18 +03:00
struct sta_info * sta ;
2008-05-03 02:44:09 +04:00
int err , queue ;
2008-01-28 15:07:18 +03:00
u8 tid ;
2007-05-05 22:45:53 +04:00
2008-05-15 14:55:29 +04:00
if ( info - > flags & IEEE80211_TX_CTL_REQUEUE ) {
2008-05-17 02:57:14 +04:00
queue = skb_get_queue_mapping ( skb ) ;
2008-02-25 18:27:46 +03:00
rcu_read_lock ( ) ;
2008-01-28 15:07:18 +03:00
sta = sta_info_get ( local , hdr - > addr1 ) ;
2008-07-02 22:05:34 +04:00
tid = skb - > priority & IEEE80211_QOS_CTL_TAG1D_MASK ;
2008-01-28 15:07:18 +03:00
if ( sta ) {
int ampdu_queue = sta - > tid_to_tx_q [ tid ] ;
2008-04-30 20:51:21 +04:00
if ( ( ampdu_queue < QD_NUM ( hw ) ) & &
2008-01-30 13:58:45 +03:00
test_bit ( ampdu_queue , q - > qdisc_pool ) ) {
2008-01-28 15:07:18 +03:00
queue = ampdu_queue ;
2008-05-15 14:55:29 +04:00
info - > flags | = IEEE80211_TX_CTL_AMPDU ;
2008-01-28 15:07:18 +03:00
} else {
2008-05-15 14:55:29 +04:00
info - > flags & = ~ IEEE80211_TX_CTL_AMPDU ;
2008-01-28 15:07:18 +03:00
}
}
2008-02-25 18:27:46 +03:00
rcu_read_unlock ( ) ;
2008-01-28 15:07:18 +03:00
skb_queue_tail ( & q - > requeued [ queue ] , skb ) ;
2007-05-05 22:45:53 +04:00
qd - > q . qlen + + ;
return 0 ;
}
queue = classify80211 ( skb , qd ) ;
2008-04-30 20:51:21 +04:00
if ( unlikely ( queue > = local - > hw . queues ) )
queue = local - > hw . queues - 1 ;
2007-05-05 22:45:53 +04:00
/* now we know the 1d priority, fill in the QoS header if there is one
*/
2008-06-12 01:21:59 +04:00
if ( ieee80211_is_data_qos ( hdr - > frame_control ) ) {
u8 * p = ieee80211_get_qos_ctl ( hdr ) ;
2008-01-28 15:07:18 +03:00
u8 ack_policy = 0 ;
2008-07-02 22:05:34 +04:00
tid = skb - > priority & IEEE80211_QOS_CTL_TAG1D_MASK ;
2007-05-05 22:45:53 +04:00
if ( local - > wifi_wme_noack_test )
2008-01-28 15:07:18 +03:00
ack_policy | = QOS_CONTROL_ACK_POLICY_NOACK < <
2007-05-05 22:45:53 +04:00
QOS_CONTROL_ACK_POLICY_SHIFT ;
/* qos header is 2 bytes, second reserved */
2008-06-12 01:21:59 +04:00
* p + + = ack_policy | tid ;
2007-05-05 22:45:53 +04:00
* p = 0 ;
2008-01-28 15:07:18 +03:00
2008-02-25 18:27:46 +03:00
rcu_read_lock ( ) ;
2008-01-28 15:07:18 +03:00
sta = sta_info_get ( local , hdr - > addr1 ) ;
if ( sta ) {
int ampdu_queue = sta - > tid_to_tx_q [ tid ] ;
2008-04-30 20:51:21 +04:00
if ( ( ampdu_queue < QD_NUM ( hw ) ) & &
test_bit ( ampdu_queue , q - > qdisc_pool ) ) {
2008-01-28 15:07:18 +03:00
queue = ampdu_queue ;
2008-05-15 14:55:29 +04:00
info - > flags | = IEEE80211_TX_CTL_AMPDU ;
2008-01-28 15:07:18 +03:00
} else {
2008-05-15 14:55:29 +04:00
info - > flags & = ~ IEEE80211_TX_CTL_AMPDU ;
2008-01-28 15:07:18 +03:00
}
}
2008-02-25 18:27:46 +03:00
rcu_read_unlock ( ) ;
2007-05-05 22:45:53 +04:00
}
2008-05-03 02:44:09 +04:00
if ( unlikely ( queue < 0 ) ) {
kfree_skb ( skb ) ;
err = NET_XMIT_DROP ;
} else {
2008-05-17 02:57:14 +04:00
skb_set_queue_mapping ( skb , queue ) ;
2008-05-03 02:44:09 +04:00
qdisc = q - > queues [ queue ] ;
err = qdisc - > enqueue ( skb , qdisc ) ;
if ( err = = NET_XMIT_SUCCESS ) {
qd - > q . qlen + + ;
qd - > bstats . bytes + = skb - > len ;
qd - > bstats . packets + + ;
return NET_XMIT_SUCCESS ;
}
2007-05-05 22:45:53 +04:00
}
qd - > qstats . drops + + ;
return err ;
}
/* TODO: clean up the cases where master_hard_start_xmit
* returns non 0 - it shouldn ' t ever do that . Once done we
* can remove this function */
static int wme_qdiscop_requeue ( struct sk_buff * skb , struct Qdisc * qd )
{
struct ieee80211_sched_data * q = qdisc_priv ( qd ) ;
struct Qdisc * qdisc ;
int err ;
/* we recorded which queue to use earlier! */
2008-05-17 02:57:14 +04:00
qdisc = q - > queues [ skb_get_queue_mapping ( skb ) ] ;
2007-05-05 22:45:53 +04:00
if ( ( err = qdisc - > ops - > requeue ( skb , qdisc ) ) = = 0 ) {
qd - > q . qlen + + ;
return 0 ;
}
qd - > qstats . drops + + ;
return err ;
}
static struct sk_buff * wme_qdiscop_dequeue ( struct Qdisc * qd )
{
struct ieee80211_sched_data * q = qdisc_priv ( qd ) ;
struct net_device * dev = qd - > dev ;
struct ieee80211_local * local = wdev_priv ( dev - > ieee80211_ptr ) ;
struct ieee80211_hw * hw = & local - > hw ;
struct sk_buff * skb ;
struct Qdisc * qdisc ;
int queue ;
/* check all the h/w queues in numeric/priority order */
2008-04-30 20:51:21 +04:00
for ( queue = 0 ; queue < QD_NUM ( hw ) ; queue + + ) {
2007-05-05 22:45:53 +04:00
/* see if there is room in this hardware queue */
2008-05-17 02:57:14 +04:00
if ( __netif_subqueue_stopped ( local - > mdev , queue ) | |
! test_bit ( queue , q - > qdisc_pool ) )
2007-05-05 22:45:53 +04:00
continue ;
/* there is space - try and get a frame */
skb = skb_dequeue ( & q - > requeued [ queue ] ) ;
if ( skb ) {
qd - > q . qlen - - ;
return skb ;
}
qdisc = q - > queues [ queue ] ;
skb = qdisc - > dequeue ( qdisc ) ;
if ( skb ) {
qd - > q . qlen - - ;
return skb ;
}
}
/* returning a NULL here when all the h/w queues are full means we
* never need to call netif_stop_queue in the driver */
return NULL ;
}
static void wme_qdiscop_reset ( struct Qdisc * qd )
{
struct ieee80211_sched_data * q = qdisc_priv ( qd ) ;
struct ieee80211_local * local = wdev_priv ( qd - > dev - > ieee80211_ptr ) ;
struct ieee80211_hw * hw = & local - > hw ;
int queue ;
/* QUESTION: should we have some hardware flush functionality here? */
2008-04-30 20:51:21 +04:00
for ( queue = 0 ; queue < QD_NUM ( hw ) ; queue + + ) {
2007-05-05 22:45:53 +04:00
skb_queue_purge ( & q - > requeued [ queue ] ) ;
qdisc_reset ( q - > queues [ queue ] ) ;
}
qd - > q . qlen = 0 ;
}
static void wme_qdiscop_destroy ( struct Qdisc * qd )
{
struct ieee80211_sched_data * q = qdisc_priv ( qd ) ;
struct ieee80211_local * local = wdev_priv ( qd - > dev - > ieee80211_ptr ) ;
struct ieee80211_hw * hw = & local - > hw ;
int queue ;
2008-07-02 06:52:38 +04:00
tcf_destroy_chain ( & q - > filter_list ) ;
2007-05-05 22:45:53 +04:00
2008-04-30 20:51:21 +04:00
for ( queue = 0 ; queue < QD_NUM ( hw ) ; queue + + ) {
2007-05-05 22:45:53 +04:00
skb_queue_purge ( & q - > requeued [ queue ] ) ;
qdisc_destroy ( q - > queues [ queue ] ) ;
q - > queues [ queue ] = & noop_qdisc ;
}
}
/* called whenever parameters are updated on existing qdisc */
2008-01-23 09:11:17 +03:00
static int wme_qdiscop_tune ( struct Qdisc * qd , struct nlattr * opt )
2007-05-05 22:45:53 +04:00
{
return 0 ;
}
/* called during initial creation of qdisc on device */
2008-01-23 09:11:17 +03:00
static int wme_qdiscop_init ( struct Qdisc * qd , struct nlattr * opt )
2007-05-05 22:45:53 +04:00
{
struct ieee80211_sched_data * q = qdisc_priv ( qd ) ;
struct net_device * dev = qd - > dev ;
struct ieee80211_local * local ;
2008-04-30 20:51:21 +04:00
struct ieee80211_hw * hw ;
2007-05-05 22:45:53 +04:00
int err = 0 , i ;
/* check that device is a mac80211 device */
if ( ! dev - > ieee80211_ptr | |
dev - > ieee80211_ptr - > wiphy - > privid ! = mac80211_wiphy_privid )
return - EINVAL ;
2008-04-30 20:51:21 +04:00
local = wdev_priv ( dev - > ieee80211_ptr ) ;
hw = & local - > hw ;
/* only allow on master dev */
if ( dev ! = local - > mdev )
2007-05-05 22:45:53 +04:00
return - EINVAL ;
2008-04-30 20:51:21 +04:00
/* ensure that we are root qdisc */
if ( qd - > parent ! = TC_H_ROOT )
2007-05-05 22:45:53 +04:00
return - EINVAL ;
if ( qd - > flags & TCQ_F_INGRESS )
return - EINVAL ;
/* if options were passed in, set them */
2008-04-30 20:51:21 +04:00
if ( opt )
2007-05-05 22:45:53 +04:00
err = wme_qdiscop_tune ( qd , opt ) ;
/* create child queues */
2008-04-30 20:51:21 +04:00
for ( i = 0 ; i < QD_NUM ( hw ) ; i + + ) {
2007-05-05 22:45:53 +04:00
skb_queue_head_init ( & q - > requeued [ i ] ) ;
q - > queues [ i ] = qdisc_create_dflt ( qd - > dev , & pfifo_qdisc_ops ,
qd - > handle ) ;
2007-09-10 15:55:08 +04:00
if ( ! q - > queues [ i ] ) {
2007-05-05 22:45:53 +04:00
q - > queues [ i ] = & noop_qdisc ;
2008-05-12 17:02:24 +04:00
printk ( KERN_ERR " %s child qdisc %i creation failed \n " ,
dev - > name , i ) ;
2007-05-05 22:45:53 +04:00
}
}
2008-04-30 20:51:21 +04:00
/* non-aggregation queues: reserve/mark as used */
for ( i = 0 ; i < local - > hw . queues ; i + + )
2008-01-30 13:58:45 +03:00
set_bit ( i , q - > qdisc_pool ) ;
2008-01-28 15:07:18 +03:00
2007-05-05 22:45:53 +04:00
return err ;
}
static int wme_qdiscop_dump ( struct Qdisc * qd , struct sk_buff * skb )
{
return - 1 ;
}
static int wme_classop_graft ( struct Qdisc * qd , unsigned long arg ,
struct Qdisc * new , struct Qdisc * * old )
{
struct ieee80211_sched_data * q = qdisc_priv ( qd ) ;
struct ieee80211_local * local = wdev_priv ( qd - > dev - > ieee80211_ptr ) ;
struct ieee80211_hw * hw = & local - > hw ;
unsigned long queue = arg - 1 ;
2008-04-30 20:51:21 +04:00
if ( queue > = QD_NUM ( hw ) )
2007-05-05 22:45:53 +04:00
return - EINVAL ;
if ( ! new )
new = & noop_qdisc ;
sch_tree_lock ( qd ) ;
* old = q - > queues [ queue ] ;
q - > queues [ queue ] = new ;
qdisc_reset ( * old ) ;
sch_tree_unlock ( qd ) ;
return 0 ;
}
static struct Qdisc *
wme_classop_leaf ( struct Qdisc * qd , unsigned long arg )
{
struct ieee80211_sched_data * q = qdisc_priv ( qd ) ;
struct ieee80211_local * local = wdev_priv ( qd - > dev - > ieee80211_ptr ) ;
struct ieee80211_hw * hw = & local - > hw ;
unsigned long queue = arg - 1 ;
2008-04-30 20:51:21 +04:00
if ( queue > = QD_NUM ( hw ) )
2007-05-05 22:45:53 +04:00
return NULL ;
return q - > queues [ queue ] ;
}
static unsigned long wme_classop_get ( struct Qdisc * qd , u32 classid )
{
struct ieee80211_local * local = wdev_priv ( qd - > dev - > ieee80211_ptr ) ;
struct ieee80211_hw * hw = & local - > hw ;
unsigned long queue = TC_H_MIN ( classid ) ;
2008-04-30 20:51:21 +04:00
if ( queue - 1 > = QD_NUM ( hw ) )
2007-05-05 22:45:53 +04:00
return 0 ;
return queue ;
}
static unsigned long wme_classop_bind ( struct Qdisc * qd , unsigned long parent ,
u32 classid )
{
return wme_classop_get ( qd , classid ) ;
}
static void wme_classop_put ( struct Qdisc * q , unsigned long cl )
{
}
static int wme_classop_change ( struct Qdisc * qd , u32 handle , u32 parent ,
2008-01-23 09:11:17 +03:00
struct nlattr * * tca , unsigned long * arg )
2007-05-05 22:45:53 +04:00
{
unsigned long cl = * arg ;
struct ieee80211_local * local = wdev_priv ( qd - > dev - > ieee80211_ptr ) ;
struct ieee80211_hw * hw = & local - > hw ;
2008-04-30 20:51:21 +04:00
if ( cl - 1 > QD_NUM ( hw ) )
2007-05-05 22:45:53 +04:00
return - ENOENT ;
/* TODO: put code to program hardware queue parameters here,
* to allow programming from tc command line */
return 0 ;
}
/* we don't support deleting hardware queues
* when we add WMM - SA support - TSPECs may be deleted here */
static int wme_classop_delete ( struct Qdisc * qd , unsigned long cl )
{
struct ieee80211_local * local = wdev_priv ( qd - > dev - > ieee80211_ptr ) ;
struct ieee80211_hw * hw = & local - > hw ;
2008-04-30 20:51:21 +04:00
if ( cl - 1 > QD_NUM ( hw ) )
2007-05-05 22:45:53 +04:00
return - ENOENT ;
return 0 ;
}
static int wme_classop_dump_class ( struct Qdisc * qd , unsigned long cl ,
struct sk_buff * skb , struct tcmsg * tcm )
{
struct ieee80211_sched_data * q = qdisc_priv ( qd ) ;
struct ieee80211_local * local = wdev_priv ( qd - > dev - > ieee80211_ptr ) ;
struct ieee80211_hw * hw = & local - > hw ;
2008-04-30 20:51:21 +04:00
if ( cl - 1 > QD_NUM ( hw ) )
2007-05-05 22:45:53 +04:00
return - ENOENT ;
tcm - > tcm_handle = TC_H_MIN ( cl ) ;
tcm - > tcm_parent = qd - > handle ;
tcm - > tcm_info = q - > queues [ cl - 1 ] - > handle ; /* do we need this? */
return 0 ;
}
static void wme_classop_walk ( struct Qdisc * qd , struct qdisc_walker * arg )
{
struct ieee80211_local * local = wdev_priv ( qd - > dev - > ieee80211_ptr ) ;
struct ieee80211_hw * hw = & local - > hw ;
int queue ;
if ( arg - > stop )
return ;
2008-04-30 20:51:21 +04:00
for ( queue = 0 ; queue < QD_NUM ( hw ) ; queue + + ) {
2007-05-05 22:45:53 +04:00
if ( arg - > count < arg - > skip ) {
arg - > count + + ;
continue ;
}
/* we should return classids for our internal queues here
* as well as the external ones */
if ( arg - > fn ( qd , queue + 1 , arg ) < 0 ) {
arg - > stop = 1 ;
break ;
}
arg - > count + + ;
}
}
static struct tcf_proto * * wme_classop_find_tcf ( struct Qdisc * qd ,
unsigned long cl )
{
struct ieee80211_sched_data * q = qdisc_priv ( qd ) ;
if ( cl )
return NULL ;
return & q - > filter_list ;
}
/* this qdisc is classful (i.e. has classes, some of which may have leaf qdiscs attached)
* - these are the operations on the classes */
2007-11-14 12:44:41 +03:00
static const struct Qdisc_class_ops class_ops =
2007-05-05 22:45:53 +04:00
{
. graft = wme_classop_graft ,
. leaf = wme_classop_leaf ,
. get = wme_classop_get ,
. put = wme_classop_put ,
. change = wme_classop_change ,
. delete = wme_classop_delete ,
. walk = wme_classop_walk ,
. tcf_chain = wme_classop_find_tcf ,
. bind_tcf = wme_classop_bind ,
. unbind_tcf = wme_classop_put ,
. dump = wme_classop_dump_class ,
} ;
/* queueing discipline operations */
2007-11-14 12:44:41 +03:00
static struct Qdisc_ops wme_qdisc_ops __read_mostly =
2007-05-05 22:45:53 +04:00
{
. next = NULL ,
. cl_ops = & class_ops ,
. id = " ieee80211 " ,
. priv_size = sizeof ( struct ieee80211_sched_data ) ,
. enqueue = wme_qdiscop_enqueue ,
. dequeue = wme_qdiscop_dequeue ,
. requeue = wme_qdiscop_requeue ,
. drop = NULL , /* drop not needed since we are always the root qdisc */
. init = wme_qdiscop_init ,
. reset = wme_qdiscop_reset ,
. destroy = wme_qdiscop_destroy ,
. change = wme_qdiscop_tune ,
. dump = wme_qdiscop_dump ,
} ;
void ieee80211_install_qdisc ( struct net_device * dev )
{
struct Qdisc * qdisc ;
qdisc = qdisc_create_dflt ( dev , & wme_qdisc_ops , TC_H_ROOT ) ;
if ( ! qdisc ) {
printk ( KERN_ERR " %s: qdisc installation failed \n " , dev - > name ) ;
return ;
}
/* same handle as would be allocated by qdisc_alloc_handle() */
qdisc - > handle = 0x80010000 ;
qdisc_lock_tree ( dev ) ;
list_add_tail ( & qdisc - > list , & dev - > qdisc_list ) ;
dev - > qdisc_sleeping = qdisc ;
qdisc_unlock_tree ( dev ) ;
}
int ieee80211_qdisc_installed ( struct net_device * dev )
{
return dev - > qdisc_sleeping - > ops = = & wme_qdisc_ops ;
}
int ieee80211_wme_register ( void )
{
return register_qdisc ( & wme_qdisc_ops ) ;
}
void ieee80211_wme_unregister ( void )
{
unregister_qdisc ( & wme_qdisc_ops ) ;
}
2008-01-28 15:07:18 +03:00
int ieee80211_ht_agg_queue_add ( struct ieee80211_local * local ,
struct sta_info * sta , u16 tid )
{
int i ;
struct ieee80211_sched_data * q =
qdisc_priv ( local - > mdev - > qdisc_sleeping ) ;
DECLARE_MAC_BUF ( mac ) ;
/* prepare the filter and save it for the SW queue
2008-04-30 20:51:21 +04:00
* matching the received HW queue */
if ( ! local - > hw . ampdu_queues )
return - EPERM ;
2008-01-28 15:07:18 +03:00
/* try to get a Qdisc from the pool */
2008-04-30 20:51:21 +04:00
for ( i = local - > hw . queues ; i < QD_NUM ( & local - > hw ) ; i + + )
2008-01-30 13:58:45 +03:00
if ( ! test_and_set_bit ( i , q - > qdisc_pool ) ) {
2008-01-28 15:07:18 +03:00
ieee80211_stop_queue ( local_to_hw ( local ) , i ) ;
sta - > tid_to_tx_q [ tid ] = i ;
/* IF there are already pending packets
* on this tid first we need to drain them
* on the previous queue
* since HT is strict in order */
# ifdef CONFIG_MAC80211_HT_DEBUG
if ( net_ratelimit ( ) )
printk ( KERN_DEBUG " allocated aggregation queue "
2008-06-12 21:08:19 +04:00
" %d tid %d addr %s pool=0x%lX \n " ,
2008-01-28 15:07:18 +03:00
i , tid , print_mac ( mac , sta - > addr ) ,
2008-01-30 13:58:45 +03:00
q - > qdisc_pool [ 0 ] ) ;
2008-01-28 15:07:18 +03:00
# endif /* CONFIG_MAC80211_HT_DEBUG */
return 0 ;
}
return - EAGAIN ;
}
/**
* the caller needs to hold local - > mdev - > queue_lock
*/
void ieee80211_ht_agg_queue_remove ( struct ieee80211_local * local ,
struct sta_info * sta , u16 tid ,
u8 requeue )
{
2008-04-30 20:51:21 +04:00
struct ieee80211_hw * hw = & local - > hw ;
2008-01-28 15:07:18 +03:00
struct ieee80211_sched_data * q =
qdisc_priv ( local - > mdev - > qdisc_sleeping ) ;
int agg_queue = sta - > tid_to_tx_q [ tid ] ;
/* return the qdisc to the pool */
2008-01-30 13:58:45 +03:00
clear_bit ( agg_queue , q - > qdisc_pool ) ;
2008-04-30 20:51:21 +04:00
sta - > tid_to_tx_q [ tid ] = QD_NUM ( hw ) ;
2008-01-28 15:07:18 +03:00
if ( requeue )
ieee80211_requeue ( local , agg_queue ) ;
else
q - > queues [ agg_queue ] - > ops - > reset ( q - > queues [ agg_queue ] ) ;
}
void ieee80211_requeue ( struct ieee80211_local * local , int queue )
{
struct Qdisc * root_qd = local - > mdev - > qdisc_sleeping ;
struct ieee80211_sched_data * q = qdisc_priv ( root_qd ) ;
struct Qdisc * qdisc = q - > queues [ queue ] ;
struct sk_buff * skb = NULL ;
2008-04-23 14:45:12 +04:00
u32 len ;
2008-01-28 15:07:18 +03:00
if ( ! qdisc | | ! qdisc - > dequeue )
return ;
for ( len = qdisc - > q . qlen ; len > 0 ; len - - ) {
skb = qdisc - > dequeue ( qdisc ) ;
root_qd - > q . qlen - - ;
/* packet will be classified again and */
/* skb->packet_data->queue will be overridden if needed */
if ( skb )
wme_qdiscop_enqueue ( skb , root_qd ) ;
}
}