2005-04-17 02:20:36 +04:00
/*
* net / sched / sch_sfq . c Stochastic Fairness Queueing discipline .
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version
* 2 of the License , or ( at your option ) any later version .
*
* Authors : Alexey Kuznetsov , < kuznet @ ms2 . inr . ac . ru >
*/
# include <linux/module.h>
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/jiffies.h>
# include <linux/string.h>
# include <linux/in.h>
# include <linux/errno.h>
# include <linux/init.h>
# include <linux/ipv6.h>
# include <linux/skbuff.h>
2007-10-01 04:51:33 +04:00
# include <linux/jhash.h>
2007-07-03 09:49:07 +04:00
# include <net/ip.h>
# include <net/netlink.h>
2005-04-17 02:20:36 +04:00
# include <net/pkt_sched.h>
/* Stochastic Fairness Queuing algorithm.
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
Source :
Paul E . McKenney " Stochastic Fairness Queuing " ,
IEEE INFOCOMM ' 90 Proceedings , San Francisco , 1990.
Paul E . McKenney " Stochastic Fairness Queuing " ,
" Interworking: Research and Experience " , v .2 , 1991 , p .113 - 131.
See also :
M . Shreedhar and George Varghese " Efficient Fair
Queuing using Deficit Round Robin " , Proc. SIGCOMM 95.
2007-02-09 17:25:16 +03:00
This is not the thing that is usually called ( W ) FQ nowadays .
2005-04-17 02:20:36 +04:00
It does not use any timestamp mechanism , but instead
processes queues in round - robin order .
ADVANTAGE :
- It is very cheap . Both CPU and memory requirements are minimal .
DRAWBACKS :
2007-02-09 17:25:16 +03:00
- " Stochastic " - > It is not 100 % fair .
2005-04-17 02:20:36 +04:00
When hash collisions occur , several flows are considered as one .
- " Round-robin " - > It introduces larger delays than virtual clock
based schemes , and should not be used for isolating interactive
traffic from non - interactive . It means , that this scheduler
should be used as leaf of CBQ or P3 , which put interactive traffic
to higher priority band .
We still need true WFQ for top level CSZ , but using WFQ
for the best effort traffic is absolutely pointless :
SFQ is superior for this purpose .
IMPLEMENTATION :
This implementation limits maximal queue length to 128 ;
maximal mtu to 2 ^ 15 - 1 ; number of hash buckets to 1024.
The only goal of this restrictions was that all data
fit into one 4 K page : - ) . Struct sfq_sched_data is
organized in anti - cache manner : all the data for a bucket
are scattered over different locations . This is not good ,
but it allowed me to put it into 4 K .
It is easy to increase these values , but not in flight . */
# define SFQ_DEPTH 128
# define SFQ_HASH_DIVISOR 1024
/* This type should contain at least SFQ_DEPTH*2 values */
typedef unsigned char sfq_index ;
struct sfq_head
{
sfq_index next ;
sfq_index prev ;
} ;
struct sfq_sched_data
{
/* Parameters */
int perturb_period ;
unsigned quantum ; /* Allotment per round: MUST BE >= MTU */
int limit ;
/* Variables */
2008-02-01 05:36:52 +03:00
struct tcf_proto * filter_list ;
2005-04-17 02:20:36 +04:00
struct timer_list perturb_timer ;
2007-10-01 04:51:33 +04:00
u32 perturbation ;
2005-04-17 02:20:36 +04:00
sfq_index tail ; /* Index of current slot in round */
sfq_index max_depth ; /* Maximal depth */
sfq_index ht [ SFQ_HASH_DIVISOR ] ; /* Hash table */
sfq_index next [ SFQ_DEPTH ] ; /* Active slots link */
short allot [ SFQ_DEPTH ] ; /* Current allotment per slot */
unsigned short hash [ SFQ_DEPTH ] ; /* Hash value indexed by slots */
struct sk_buff_head qs [ SFQ_DEPTH ] ; /* Slot queue */
struct sfq_head dep [ SFQ_DEPTH * 2 ] ; /* Linked list of slots, indexed by depth */
} ;
static __inline__ unsigned sfq_fold_hash ( struct sfq_sched_data * q , u32 h , u32 h1 )
{
2007-10-01 04:51:33 +04:00
return jhash_2words ( h , h1 , q - > perturbation ) & ( SFQ_HASH_DIVISOR - 1 ) ;
2005-04-17 02:20:36 +04:00
}
static unsigned sfq_hash ( struct sfq_sched_data * q , struct sk_buff * skb )
{
u32 h , h2 ;
switch ( skb - > protocol ) {
2008-09-21 09:20:49 +04:00
case htons ( ETH_P_IP ) :
2005-04-17 02:20:36 +04:00
{
2007-04-21 09:47:35 +04:00
const struct iphdr * iph = ip_hdr ( skb ) ;
2005-04-17 02:20:36 +04:00
h = iph - > daddr ;
2008-01-21 04:20:56 +03:00
h2 = iph - > saddr ^ iph - > protocol ;
2005-04-17 02:20:36 +04:00
if ( ! ( iph - > frag_off & htons ( IP_MF | IP_OFFSET ) ) & &
( iph - > protocol = = IPPROTO_TCP | |
iph - > protocol = = IPPROTO_UDP | |
2007-02-08 02:07:43 +03:00
iph - > protocol = = IPPROTO_UDPLITE | |
2006-01-18 00:01:06 +03:00
iph - > protocol = = IPPROTO_SCTP | |
iph - > protocol = = IPPROTO_DCCP | |
2005-04-17 02:20:36 +04:00
iph - > protocol = = IPPROTO_ESP ) )
h2 ^ = * ( ( ( u32 * ) iph ) + iph - > ihl ) ;
break ;
}
2008-09-21 09:20:49 +04:00
case htons ( ETH_P_IPV6 ) :
2005-04-17 02:20:36 +04:00
{
2007-04-26 04:54:47 +04:00
struct ipv6hdr * iph = ipv6_hdr ( skb ) ;
2005-04-17 02:20:36 +04:00
h = iph - > daddr . s6_addr32 [ 3 ] ;
2008-01-21 04:20:56 +03:00
h2 = iph - > saddr . s6_addr32 [ 3 ] ^ iph - > nexthdr ;
2005-04-17 02:20:36 +04:00
if ( iph - > nexthdr = = IPPROTO_TCP | |
iph - > nexthdr = = IPPROTO_UDP | |
2007-02-08 02:07:43 +03:00
iph - > nexthdr = = IPPROTO_UDPLITE | |
2006-01-18 00:01:06 +03:00
iph - > nexthdr = = IPPROTO_SCTP | |
iph - > nexthdr = = IPPROTO_DCCP | |
2005-04-17 02:20:36 +04:00
iph - > nexthdr = = IPPROTO_ESP )
h2 ^ = * ( u32 * ) & iph [ 1 ] ;
break ;
}
default :
2008-01-21 04:20:56 +03:00
h = ( unsigned long ) skb - > dst ^ skb - > protocol ;
h2 = ( unsigned long ) skb - > sk ;
2005-04-17 02:20:36 +04:00
}
2008-01-21 04:20:56 +03:00
2005-04-17 02:20:36 +04:00
return sfq_fold_hash ( q , h , h2 ) ;
}
2008-02-01 05:36:52 +03:00
static unsigned int sfq_classify ( struct sk_buff * skb , struct Qdisc * sch ,
int * qerr )
{
struct sfq_sched_data * q = qdisc_priv ( sch ) ;
struct tcf_result res ;
int result ;
if ( TC_H_MAJ ( skb - > priority ) = = sch - > handle & &
TC_H_MIN ( skb - > priority ) > 0 & &
TC_H_MIN ( skb - > priority ) < = SFQ_HASH_DIVISOR )
return TC_H_MIN ( skb - > priority ) ;
if ( ! q - > filter_list )
return sfq_hash ( q , skb ) + 1 ;
2008-08-05 09:39:11 +04:00
* qerr = NET_XMIT_SUCCESS | __NET_XMIT_BYPASS ;
2008-02-01 05:36:52 +03:00
result = tc_classify ( skb , q - > filter_list , & res ) ;
if ( result > = 0 ) {
# ifdef CONFIG_NET_CLS_ACT
switch ( result ) {
case TC_ACT_STOLEN :
case TC_ACT_QUEUED :
2008-08-05 09:31:03 +04:00
* qerr = NET_XMIT_SUCCESS | __NET_XMIT_STOLEN ;
2008-02-01 05:36:52 +03:00
case TC_ACT_SHOT :
return 0 ;
}
# endif
if ( TC_H_MIN ( res . classid ) < = SFQ_HASH_DIVISOR )
return TC_H_MIN ( res . classid ) ;
}
return 0 ;
}
2005-04-17 02:20:36 +04:00
static inline void sfq_link ( struct sfq_sched_data * q , sfq_index x )
{
sfq_index p , n ;
int d = q - > qs [ x ] . qlen + SFQ_DEPTH ;
p = d ;
n = q - > dep [ d ] . next ;
q - > dep [ x ] . next = n ;
q - > dep [ x ] . prev = p ;
q - > dep [ p ] . next = q - > dep [ n ] . prev = x ;
}
static inline void sfq_dec ( struct sfq_sched_data * q , sfq_index x )
{
sfq_index p , n ;
n = q - > dep [ x ] . next ;
p = q - > dep [ x ] . prev ;
q - > dep [ p ] . next = n ;
q - > dep [ n ] . prev = p ;
if ( n = = p & & q - > max_depth = = q - > qs [ x ] . qlen + 1 )
q - > max_depth - - ;
sfq_link ( q , x ) ;
}
static inline void sfq_inc ( struct sfq_sched_data * q , sfq_index x )
{
sfq_index p , n ;
int d ;
n = q - > dep [ x ] . next ;
p = q - > dep [ x ] . prev ;
q - > dep [ p ] . next = n ;
q - > dep [ n ] . prev = p ;
d = q - > qs [ x ] . qlen ;
if ( q - > max_depth < d )
q - > max_depth = d ;
sfq_link ( q , x ) ;
}
static unsigned int sfq_drop ( struct Qdisc * sch )
{
struct sfq_sched_data * q = qdisc_priv ( sch ) ;
sfq_index d = q - > max_depth ;
struct sk_buff * skb ;
unsigned int len ;
/* Queue is full! Find the longest slot and
drop a packet from it */
if ( d > 1 ) {
2008-01-21 04:20:56 +03:00
sfq_index x = q - > dep [ d + SFQ_DEPTH ] . next ;
2005-04-17 02:20:36 +04:00
skb = q - > qs [ x ] . prev ;
2008-07-20 11:08:27 +04:00
len = qdisc_pkt_len ( skb ) ;
2005-04-17 02:20:36 +04:00
__skb_unlink ( skb , & q - > qs [ x ] ) ;
kfree_skb ( skb ) ;
sfq_dec ( q , x ) ;
sch - > q . qlen - - ;
sch - > qstats . drops + + ;
2006-03-21 06:01:38 +03:00
sch - > qstats . backlog - = len ;
2005-04-17 02:20:36 +04:00
return len ;
}
if ( d = = 1 ) {
/* It is difficult to believe, but ALL THE SLOTS HAVE LENGTH 1. */
d = q - > next [ q - > tail ] ;
q - > next [ q - > tail ] = q - > next [ d ] ;
q - > allot [ q - > next [ d ] ] + = q - > quantum ;
skb = q - > qs [ d ] . prev ;
2008-07-20 11:08:27 +04:00
len = qdisc_pkt_len ( skb ) ;
2005-04-17 02:20:36 +04:00
__skb_unlink ( skb , & q - > qs [ d ] ) ;
kfree_skb ( skb ) ;
sfq_dec ( q , d ) ;
sch - > q . qlen - - ;
q - > ht [ q - > hash [ d ] ] = SFQ_DEPTH ;
sch - > qstats . drops + + ;
2006-03-21 06:01:38 +03:00
sch - > qstats . backlog - = len ;
2005-04-17 02:20:36 +04:00
return len ;
}
return 0 ;
}
static int
2008-01-21 04:20:56 +03:00
sfq_enqueue ( struct sk_buff * skb , struct Qdisc * sch )
2005-04-17 02:20:36 +04:00
{
struct sfq_sched_data * q = qdisc_priv ( sch ) ;
2008-02-01 05:36:52 +03:00
unsigned int hash ;
2005-04-17 02:20:36 +04:00
sfq_index x ;
2008-12-22 07:14:48 +03:00
int uninitialized_var ( ret ) ;
2008-02-01 05:36:52 +03:00
hash = sfq_classify ( skb , sch , & ret ) ;
if ( hash = = 0 ) {
2008-08-05 09:39:11 +04:00
if ( ret & __NET_XMIT_BYPASS )
2008-02-01 05:36:52 +03:00
sch - > qstats . drops + + ;
kfree_skb ( skb ) ;
return ret ;
}
hash - - ;
2005-04-17 02:20:36 +04:00
x = q - > ht [ hash ] ;
if ( x = = SFQ_DEPTH ) {
q - > ht [ hash ] = x = q - > dep [ SFQ_DEPTH ] . next ;
q - > hash [ x ] = hash ;
}
2008-01-21 04:20:56 +03:00
2007-10-01 04:51:33 +04:00
/* If selected queue has length q->limit, this means that
* all another queues are empty and that we do simple tail drop ,
* i . e . drop _this_ packet .
*/
if ( q - > qs [ x ] . qlen > = q - > limit )
return qdisc_drop ( skb , sch ) ;
2008-07-20 11:08:27 +04:00
sch - > qstats . backlog + = qdisc_pkt_len ( skb ) ;
2005-04-17 02:20:36 +04:00
__skb_queue_tail ( & q - > qs [ x ] , skb ) ;
sfq_inc ( q , x ) ;
if ( q - > qs [ x ] . qlen = = 1 ) { /* The flow is new */
if ( q - > tail = = SFQ_DEPTH ) { /* It is the first flow */
q - > tail = x ;
q - > next [ x ] = x ;
q - > allot [ x ] = q - > quantum ;
} else {
q - > next [ x ] = q - > next [ q - > tail ] ;
q - > next [ q - > tail ] = x ;
q - > tail = x ;
}
}
2007-09-19 21:42:03 +04:00
if ( + + sch - > q . qlen < = q - > limit ) {
2008-07-20 11:08:27 +04:00
sch - > bstats . bytes + = qdisc_pkt_len ( skb ) ;
2005-04-17 02:20:36 +04:00
sch - > bstats . packets + + ;
return 0 ;
}
sfq_drop ( sch ) ;
return NET_XMIT_CN ;
}
2008-10-31 10:44:18 +03:00
static struct sk_buff *
sfq_peek ( struct Qdisc * sch )
{
struct sfq_sched_data * q = qdisc_priv ( sch ) ;
sfq_index a ;
2005-04-17 02:20:36 +04:00
2008-10-31 10:44:18 +03:00
/* No active slots */
if ( q - > tail = = SFQ_DEPTH )
return NULL ;
2005-04-17 02:20:36 +04:00
2008-10-31 10:44:18 +03:00
a = q - > next [ q - > tail ] ;
return skb_peek ( & q - > qs [ a ] ) ;
}
2005-04-17 02:20:36 +04:00
static struct sk_buff *
2008-01-21 04:20:56 +03:00
sfq_dequeue ( struct Qdisc * sch )
2005-04-17 02:20:36 +04:00
{
struct sfq_sched_data * q = qdisc_priv ( sch ) ;
struct sk_buff * skb ;
sfq_index a , old_a ;
/* No active slots */
if ( q - > tail = = SFQ_DEPTH )
return NULL ;
a = old_a = q - > next [ q - > tail ] ;
/* Grab packet */
skb = __skb_dequeue ( & q - > qs [ a ] ) ;
sfq_dec ( q , a ) ;
sch - > q . qlen - - ;
2008-07-20 11:08:27 +04:00
sch - > qstats . backlog - = qdisc_pkt_len ( skb ) ;
2005-04-17 02:20:36 +04:00
/* Is the slot empty? */
if ( q - > qs [ a ] . qlen = = 0 ) {
q - > ht [ q - > hash [ a ] ] = SFQ_DEPTH ;
a = q - > next [ a ] ;
if ( a = = old_a ) {
q - > tail = SFQ_DEPTH ;
return skb ;
}
q - > next [ q - > tail ] = a ;
q - > allot [ a ] + = q - > quantum ;
2008-07-20 11:08:27 +04:00
} else if ( ( q - > allot [ a ] - = qdisc_pkt_len ( skb ) ) < = 0 ) {
2005-04-17 02:20:36 +04:00
q - > tail = a ;
a = q - > next [ a ] ;
q - > allot [ a ] + = q - > quantum ;
}
return skb ;
}
static void
2008-01-21 04:20:56 +03:00
sfq_reset ( struct Qdisc * sch )
2005-04-17 02:20:36 +04:00
{
struct sk_buff * skb ;
while ( ( skb = sfq_dequeue ( sch ) ) ! = NULL )
kfree_skb ( skb ) ;
}
static void sfq_perturbation ( unsigned long arg )
{
2008-01-21 04:20:56 +03:00
struct Qdisc * sch = ( struct Qdisc * ) arg ;
2005-04-17 02:20:36 +04:00
struct sfq_sched_data * q = qdisc_priv ( sch ) ;
2008-01-21 04:19:43 +03:00
q - > perturbation = net_random ( ) ;
2005-04-17 02:20:36 +04:00
2007-10-01 04:51:33 +04:00
if ( q - > perturb_period )
mod_timer ( & q - > perturb_timer , jiffies + q - > perturb_period ) ;
2005-04-17 02:20:36 +04:00
}
2008-01-23 09:11:17 +03:00
static int sfq_change ( struct Qdisc * sch , struct nlattr * opt )
2005-04-17 02:20:36 +04:00
{
struct sfq_sched_data * q = qdisc_priv ( sch ) ;
2008-01-23 09:11:17 +03:00
struct tc_sfq_qopt * ctl = nla_data ( opt ) ;
2006-11-30 04:36:20 +03:00
unsigned int qlen ;
2005-04-17 02:20:36 +04:00
2008-01-23 09:11:17 +03:00
if ( opt - > nla_len < nla_attr_size ( sizeof ( * ctl ) ) )
2005-04-17 02:20:36 +04:00
return - EINVAL ;
sch_tree_lock ( sch ) ;
2008-07-09 04:06:30 +04:00
q - > quantum = ctl - > quantum ? : psched_mtu ( qdisc_dev ( sch ) ) ;
2008-01-21 04:20:56 +03:00
q - > perturb_period = ctl - > perturb_period * HZ ;
2005-04-17 02:20:36 +04:00
if ( ctl - > limit )
2007-10-01 04:51:33 +04:00
q - > limit = min_t ( u32 , ctl - > limit , SFQ_DEPTH - 1 ) ;
2005-04-17 02:20:36 +04:00
2006-11-30 04:36:20 +03:00
qlen = sch - > q . qlen ;
2007-09-19 21:42:03 +04:00
while ( sch - > q . qlen > q - > limit )
2005-04-17 02:20:36 +04:00
sfq_drop ( sch ) ;
2006-11-30 04:36:20 +03:00
qdisc_tree_decrease_qlen ( sch , qlen - sch - > q . qlen ) ;
2005-04-17 02:20:36 +04:00
del_timer ( & q - > perturb_timer ) ;
if ( q - > perturb_period ) {
2007-10-01 04:51:33 +04:00
mod_timer ( & q - > perturb_timer , jiffies + q - > perturb_period ) ;
2008-01-21 04:19:43 +03:00
q - > perturbation = net_random ( ) ;
2005-04-17 02:20:36 +04:00
}
sch_tree_unlock ( sch ) ;
return 0 ;
}
2008-01-23 09:11:17 +03:00
static int sfq_init ( struct Qdisc * sch , struct nlattr * opt )
2005-04-17 02:20:36 +04:00
{
struct sfq_sched_data * q = qdisc_priv ( sch ) ;
int i ;
2008-01-21 04:18:45 +03:00
q - > perturb_timer . function = sfq_perturbation ;
2009-01-08 05:09:08 +03:00
q - > perturb_timer . data = ( unsigned long ) sch ;
2008-01-21 04:18:45 +03:00
init_timer_deferrable ( & q - > perturb_timer ) ;
2005-04-17 02:20:36 +04:00
2008-01-21 04:20:56 +03:00
for ( i = 0 ; i < SFQ_HASH_DIVISOR ; i + + )
2005-04-17 02:20:36 +04:00
q - > ht [ i ] = SFQ_DEPTH ;
2008-01-21 04:20:56 +03:00
for ( i = 0 ; i < SFQ_DEPTH ; i + + ) {
2005-04-17 02:20:36 +04:00
skb_queue_head_init ( & q - > qs [ i ] ) ;
2008-01-21 04:20:56 +03:00
q - > dep [ i + SFQ_DEPTH ] . next = i + SFQ_DEPTH ;
q - > dep [ i + SFQ_DEPTH ] . prev = i + SFQ_DEPTH ;
2005-04-17 02:20:36 +04:00
}
2008-01-21 04:20:56 +03:00
2007-10-01 04:51:33 +04:00
q - > limit = SFQ_DEPTH - 1 ;
2005-04-17 02:20:36 +04:00
q - > max_depth = 0 ;
q - > tail = SFQ_DEPTH ;
if ( opt = = NULL ) {
2008-07-09 04:06:30 +04:00
q - > quantum = psched_mtu ( qdisc_dev ( sch ) ) ;
2005-04-17 02:20:36 +04:00
q - > perturb_period = 0 ;
2008-01-21 04:19:43 +03:00
q - > perturbation = net_random ( ) ;
2005-04-17 02:20:36 +04:00
} else {
int err = sfq_change ( sch , opt ) ;
if ( err )
return err ;
}
2008-01-21 04:20:56 +03:00
for ( i = 0 ; i < SFQ_DEPTH ; i + + )
2005-04-17 02:20:36 +04:00
sfq_link ( q , i ) ;
return 0 ;
}
static void sfq_destroy ( struct Qdisc * sch )
{
struct sfq_sched_data * q = qdisc_priv ( sch ) ;
2008-02-01 05:36:52 +03:00
2008-07-02 06:52:38 +04:00
tcf_destroy_chain ( & q - > filter_list ) ;
2008-04-29 14:29:03 +04:00
q - > perturb_period = 0 ;
del_timer_sync ( & q - > perturb_timer ) ;
2005-04-17 02:20:36 +04:00
}
static int sfq_dump ( struct Qdisc * sch , struct sk_buff * skb )
{
struct sfq_sched_data * q = qdisc_priv ( sch ) ;
2007-04-20 07:29:13 +04:00
unsigned char * b = skb_tail_pointer ( skb ) ;
2005-04-17 02:20:36 +04:00
struct tc_sfq_qopt opt ;
opt . quantum = q - > quantum ;
2008-01-21 04:20:56 +03:00
opt . perturb_period = q - > perturb_period / HZ ;
2005-04-17 02:20:36 +04:00
opt . limit = q - > limit ;
opt . divisor = SFQ_HASH_DIVISOR ;
2008-07-26 13:28:09 +04:00
opt . flows = q - > limit ;
2005-04-17 02:20:36 +04:00
2008-01-23 09:11:17 +03:00
NLA_PUT ( skb , TCA_OPTIONS , sizeof ( opt ) , & opt ) ;
2005-04-17 02:20:36 +04:00
return skb - > len ;
2008-01-23 09:11:17 +03:00
nla_put_failure :
2007-03-26 10:06:12 +04:00
nlmsg_trim ( skb , b ) ;
2005-04-17 02:20:36 +04:00
return - 1 ;
}
2008-02-01 05:36:52 +03:00
static int sfq_change_class ( struct Qdisc * sch , u32 classid , u32 parentid ,
struct nlattr * * tca , unsigned long * arg )
{
return - EOPNOTSUPP ;
}
static unsigned long sfq_get ( struct Qdisc * sch , u32 classid )
{
return 0 ;
}
static struct tcf_proto * * sfq_find_tcf ( struct Qdisc * sch , unsigned long cl )
{
struct sfq_sched_data * q = qdisc_priv ( sch ) ;
if ( cl )
return NULL ;
return & q - > filter_list ;
}
2008-02-01 05:37:16 +03:00
static int sfq_dump_class ( struct Qdisc * sch , unsigned long cl ,
struct sk_buff * skb , struct tcmsg * tcm )
{
tcm - > tcm_handle | = TC_H_MIN ( cl ) ;
return 0 ;
}
static int sfq_dump_class_stats ( struct Qdisc * sch , unsigned long cl ,
struct gnet_dump * d )
{
struct sfq_sched_data * q = qdisc_priv ( sch ) ;
sfq_index idx = q - > ht [ cl - 1 ] ;
struct gnet_stats_queue qs = { . qlen = q - > qs [ idx ] . qlen } ;
struct tc_sfq_xstats xstats = { . allot = q - > allot [ idx ] } ;
if ( gnet_stats_copy_queue ( d , & qs ) < 0 )
return - 1 ;
return gnet_stats_copy_app ( d , & xstats , sizeof ( xstats ) ) ;
}
2008-02-01 05:36:52 +03:00
static void sfq_walk ( struct Qdisc * sch , struct qdisc_walker * arg )
{
2008-02-01 05:37:16 +03:00
struct sfq_sched_data * q = qdisc_priv ( sch ) ;
unsigned int i ;
if ( arg - > stop )
return ;
for ( i = 0 ; i < SFQ_HASH_DIVISOR ; i + + ) {
if ( q - > ht [ i ] = = SFQ_DEPTH | |
arg - > count < arg - > skip ) {
arg - > count + + ;
continue ;
}
if ( arg - > fn ( sch , i + 1 , arg ) < 0 ) {
arg - > stop = 1 ;
break ;
}
arg - > count + + ;
}
2008-02-01 05:36:52 +03:00
}
static const struct Qdisc_class_ops sfq_class_ops = {
. get = sfq_get ,
. change = sfq_change_class ,
. tcf_chain = sfq_find_tcf ,
2008-02-01 05:37:16 +03:00
. dump = sfq_dump_class ,
. dump_stats = sfq_dump_class_stats ,
2008-02-01 05:36:52 +03:00
. walk = sfq_walk ,
} ;
2007-11-14 12:44:41 +03:00
static struct Qdisc_ops sfq_qdisc_ops __read_mostly = {
2008-02-01 05:36:52 +03:00
. cl_ops = & sfq_class_ops ,
2005-04-17 02:20:36 +04:00
. id = " sfq " ,
. priv_size = sizeof ( struct sfq_sched_data ) ,
. enqueue = sfq_enqueue ,
. dequeue = sfq_dequeue ,
2008-10-31 10:44:18 +03:00
. peek = sfq_peek ,
2005-04-17 02:20:36 +04:00
. drop = sfq_drop ,
. init = sfq_init ,
. reset = sfq_reset ,
. destroy = sfq_destroy ,
. change = NULL ,
. dump = sfq_dump ,
. owner = THIS_MODULE ,
} ;
static int __init sfq_module_init ( void )
{
return register_qdisc ( & sfq_qdisc_ops ) ;
}
2007-02-09 17:25:16 +03:00
static void __exit sfq_module_exit ( void )
2005-04-17 02:20:36 +04:00
{
unregister_qdisc ( & sfq_qdisc_ops ) ;
}
module_init ( sfq_module_init )
module_exit ( sfq_module_exit )
MODULE_LICENSE ( " GPL " ) ;