2005-04-17 02:20:36 +04:00
/*
* net / sched / sch_api . c Packet scheduler API .
*
* 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 >
*
* Fixes :
*
* Rani Assaf < rani @ magic . metawire . com > : 980802 : JIFFIES and CPU clock sources are repaired .
* Eduardo J . Blanco < ejbs @ netlabs . com . uy > : 990222 : kmod support
* Jamal Hadi Salim < hadi @ nortelnetworks . com > : 990601 : ingress support
*/
# include <linux/module.h>
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/string.h>
# include <linux/errno.h>
# include <linux/skbuff.h>
# include <linux/init.h>
# include <linux/proc_fs.h>
# include <linux/seq_file.h>
# include <linux/kmod.h>
# include <linux/list.h>
2007-03-16 11:19:15 +03:00
# include <linux/hrtimer.h>
2008-08-19 07:53:34 +04:00
# include <linux/lockdep.h>
2005-04-17 02:20:36 +04:00
2007-09-12 14:01:34 +04:00
# include <net/net_namespace.h>
2007-11-30 16:21:31 +03:00
# include <net/sock.h>
2007-03-26 10:06:12 +04:00
# include <net/netlink.h>
2005-04-17 02:20:36 +04:00
# include <net/pkt_sched.h>
static int qdisc_notify ( struct sk_buff * oskb , struct nlmsghdr * n , u32 clid ,
struct Qdisc * old , struct Qdisc * new ) ;
static int tclass_notify ( struct sk_buff * oskb , struct nlmsghdr * n ,
struct Qdisc * q , unsigned long cl , int event ) ;
/*
Short review .
- - - - - - - - - - - - -
This file consists of two interrelated parts :
1. queueing disciplines manager frontend .
2. traffic classes manager frontend .
Generally , queueing discipline ( " qdisc " ) is a black box ,
which is able to enqueue packets and to dequeue them ( when
device is ready to send something ) in order and at times
determined by algorithm hidden in it .
qdisc ' s are divided to two categories :
- " queues " , which have no internal structure visible from outside .
- " schedulers " , which split all the packets to " traffic classes " ,
using " packet classifiers " ( look at cls_api . c )
In turn , classes may have child qdiscs ( as rule , queues )
attached to them etc . etc . etc .
The goal of the routines in this file is to translate
information supplied by user in the form of handles
to more intelligible for kernel form , to make some sanity
checks and part of work , which is common to all qdiscs
and to provide rtnetlink notifications .
All real intelligent work is done inside qdisc modules .
Every discipline has two major routines : enqueue and dequeue .
- - - dequeue
dequeue usually returns a skb to send . It is allowed to return NULL ,
but it does not mean that queue is empty , it just means that
discipline does not want to send anything this time .
Queue is really empty if q - > q . qlen = = 0.
For complicated disciplines with multiple queues q - > q is not
real packet queue , but however q - > q . qlen must be valid .
- - - enqueue
enqueue returns 0 , if packet was enqueued successfully .
If packet ( this one or another one ) was dropped , it returns
not zero error code .
NET_XMIT_DROP - this packet dropped
Expected action : do not backoff , but wait until queue will clear .
NET_XMIT_CN - probably this packet enqueued , but another one dropped .
Expected action : backoff or ignore
NET_XMIT_POLICED - dropped by police .
Expected action : backoff or error to real - time apps .
Auxiliary routines :
- - - requeue
requeues once dequeued packet . It is used for non - standard or
2008-07-09 03:46:01 +04:00
just buggy devices , which can defer output even if netif_queue_stopped ( ) = 0.
2005-04-17 02:20:36 +04:00
- - - reset
returns qdisc to initial state : purge all buffers , clear all
timers , counters ( except for statistics ) etc .
- - - init
initializes newly created qdisc .
- - - destroy
destroys resources allocated by init and during lifetime of qdisc .
- - - change
changes qdisc parameters .
*/
/* Protects list of registered TC modules. It is pure SMP lock. */
static DEFINE_RWLOCK ( qdisc_mod_lock ) ;
/************************************************
* Queueing disciplines manipulation . *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/* The list of all installed queueing disciplines. */
static struct Qdisc_ops * qdisc_base ;
/* Register/uregister queueing discipline */
int register_qdisc ( struct Qdisc_ops * qops )
{
struct Qdisc_ops * q , * * qp ;
int rc = - EEXIST ;
write_lock ( & qdisc_mod_lock ) ;
for ( qp = & qdisc_base ; ( q = * qp ) ! = NULL ; qp = & q - > next )
if ( ! strcmp ( qops - > id , q - > id ) )
goto out ;
if ( qops - > enqueue = = NULL )
qops - > enqueue = noop_qdisc_ops . enqueue ;
if ( qops - > requeue = = NULL )
qops - > requeue = noop_qdisc_ops . requeue ;
if ( qops - > dequeue = = NULL )
qops - > dequeue = noop_qdisc_ops . dequeue ;
qops - > next = NULL ;
* qp = qops ;
rc = 0 ;
out :
write_unlock ( & qdisc_mod_lock ) ;
return rc ;
}
2008-01-23 09:10:23 +03:00
EXPORT_SYMBOL ( register_qdisc ) ;
2005-04-17 02:20:36 +04:00
int unregister_qdisc ( struct Qdisc_ops * qops )
{
struct Qdisc_ops * q , * * qp ;
int err = - ENOENT ;
write_lock ( & qdisc_mod_lock ) ;
for ( qp = & qdisc_base ; ( q = * qp ) ! = NULL ; qp = & q - > next )
if ( q = = qops )
break ;
if ( q ) {
* qp = q - > next ;
q - > next = NULL ;
err = 0 ;
}
write_unlock ( & qdisc_mod_lock ) ;
return err ;
}
2008-01-23 09:10:23 +03:00
EXPORT_SYMBOL ( unregister_qdisc ) ;
2005-04-17 02:20:36 +04:00
/* We know handle. Find qdisc among all qdisc's attached to device
( root qdisc , all its children , children of children etc . )
*/
2008-08-09 10:23:39 +04:00
struct Qdisc * qdisc_match_from_root ( struct Qdisc * root , u32 handle )
{
struct Qdisc * q ;
if ( ! ( root - > flags & TCQ_F_BUILTIN ) & &
root - > handle = = handle )
return root ;
list_for_each_entry ( q , & root - > list , list ) {
if ( q - > handle = = handle )
return q ;
}
return NULL ;
}
2008-08-22 14:24:05 +04:00
/*
* This lock is needed until some qdiscs stop calling qdisc_tree_decrease_qlen ( )
* without rtnl_lock ( ) ; currently hfsc_dequeue ( ) , netem_dequeue ( ) , tbf_dequeue ( )
*/
static DEFINE_SPINLOCK ( qdisc_list_lock ) ;
static void qdisc_list_add ( struct Qdisc * q )
{
if ( ( q - > parent ! = TC_H_ROOT ) & & ! ( q - > flags & TCQ_F_INGRESS ) ) {
spin_lock_bh ( & qdisc_list_lock ) ;
list_add_tail ( & q - > list , & qdisc_root_sleeping ( q ) - > list ) ;
spin_unlock_bh ( & qdisc_list_lock ) ;
}
}
void qdisc_list_del ( struct Qdisc * q )
{
if ( ( q - > parent ! = TC_H_ROOT ) & & ! ( q - > flags & TCQ_F_INGRESS ) ) {
spin_lock_bh ( & qdisc_list_lock ) ;
list_del ( & q - > list ) ;
spin_unlock_bh ( & qdisc_list_lock ) ;
}
}
EXPORT_SYMBOL ( qdisc_list_del ) ;
2008-07-17 11:50:32 +04:00
struct Qdisc * qdisc_lookup ( struct net_device * dev , u32 handle )
2005-04-17 02:20:36 +04:00
{
2008-07-19 09:50:15 +04:00
unsigned int i ;
2008-08-22 14:24:05 +04:00
struct Qdisc * q ;
spin_lock_bh ( & qdisc_list_lock ) ;
2008-07-19 09:50:15 +04:00
for ( i = 0 ; i < dev - > num_tx_queues ; i + + ) {
struct netdev_queue * txq = netdev_get_tx_queue ( dev , i ) ;
2008-08-22 14:24:05 +04:00
struct Qdisc * txq_root = txq - > qdisc_sleeping ;
2005-04-17 02:20:36 +04:00
2008-08-09 10:23:39 +04:00
q = qdisc_match_from_root ( txq_root , handle ) ;
if ( q )
2008-08-22 14:24:05 +04:00
goto unlock ;
2005-04-17 02:20:36 +04:00
}
2008-08-22 14:24:05 +04:00
q = qdisc_match_from_root ( dev - > rx_queue . qdisc_sleeping , handle ) ;
unlock :
spin_unlock_bh ( & qdisc_list_lock ) ;
return q ;
2005-04-17 02:20:36 +04:00
}
static struct Qdisc * qdisc_leaf ( struct Qdisc * p , u32 classid )
{
unsigned long cl ;
struct Qdisc * leaf ;
2007-11-14 12:44:41 +03:00
const struct Qdisc_class_ops * cops = p - > ops - > cl_ops ;
2005-04-17 02:20:36 +04:00
if ( cops = = NULL )
return NULL ;
cl = cops - > get ( p , classid ) ;
if ( cl = = 0 )
return NULL ;
leaf = cops - > leaf ( p , cl ) ;
cops - > put ( p , cl ) ;
return leaf ;
}
/* Find queueing discipline by name */
2008-01-23 09:11:17 +03:00
static struct Qdisc_ops * qdisc_lookup_ops ( struct nlattr * kind )
2005-04-17 02:20:36 +04:00
{
struct Qdisc_ops * q = NULL ;
if ( kind ) {
read_lock ( & qdisc_mod_lock ) ;
for ( q = qdisc_base ; q ; q = q - > next ) {
2008-01-23 09:11:17 +03:00
if ( nla_strcmp ( kind , q - > id ) = = 0 ) {
2005-04-17 02:20:36 +04:00
if ( ! try_module_get ( q - > owner ) )
q = NULL ;
break ;
}
}
read_unlock ( & qdisc_mod_lock ) ;
}
return q ;
}
static struct qdisc_rate_table * qdisc_rtab_list ;
2008-01-23 09:11:17 +03:00
struct qdisc_rate_table * qdisc_get_rtab ( struct tc_ratespec * r , struct nlattr * tab )
2005-04-17 02:20:36 +04:00
{
struct qdisc_rate_table * rtab ;
for ( rtab = qdisc_rtab_list ; rtab ; rtab = rtab - > next ) {
if ( memcmp ( & rtab - > rate , r , sizeof ( struct tc_ratespec ) ) = = 0 ) {
rtab - > refcnt + + ;
return rtab ;
}
}
2008-01-24 07:35:19 +03:00
if ( tab = = NULL | | r - > rate = = 0 | | r - > cell_log = = 0 | |
nla_len ( tab ) ! = TC_RTAB_SIZE )
2005-04-17 02:20:36 +04:00
return NULL ;
rtab = kmalloc ( sizeof ( * rtab ) , GFP_KERNEL ) ;
if ( rtab ) {
rtab - > rate = * r ;
rtab - > refcnt = 1 ;
2008-01-23 09:11:17 +03:00
memcpy ( rtab - > data , nla_data ( tab ) , 1024 ) ;
2005-04-17 02:20:36 +04:00
rtab - > next = qdisc_rtab_list ;
qdisc_rtab_list = rtab ;
}
return rtab ;
}
2008-01-23 09:10:23 +03:00
EXPORT_SYMBOL ( qdisc_get_rtab ) ;
2005-04-17 02:20:36 +04:00
void qdisc_put_rtab ( struct qdisc_rate_table * tab )
{
struct qdisc_rate_table * rtab , * * rtabp ;
if ( ! tab | | - - tab - > refcnt )
return ;
for ( rtabp = & qdisc_rtab_list ; ( rtab = * rtabp ) ! = NULL ; rtabp = & rtab - > next ) {
if ( rtab = = tab ) {
* rtabp = rtab - > next ;
kfree ( rtab ) ;
return ;
}
}
}
2008-01-23 09:10:23 +03:00
EXPORT_SYMBOL ( qdisc_put_rtab ) ;
2005-04-17 02:20:36 +04:00
2008-07-20 11:08:47 +04:00
static LIST_HEAD ( qdisc_stab_list ) ;
static DEFINE_SPINLOCK ( qdisc_stab_lock ) ;
static const struct nla_policy stab_policy [ TCA_STAB_MAX + 1 ] = {
[ TCA_STAB_BASE ] = { . len = sizeof ( struct tc_sizespec ) } ,
[ TCA_STAB_DATA ] = { . type = NLA_BINARY } ,
} ;
static struct qdisc_size_table * qdisc_get_stab ( struct nlattr * opt )
{
struct nlattr * tb [ TCA_STAB_MAX + 1 ] ;
struct qdisc_size_table * stab ;
struct tc_sizespec * s ;
unsigned int tsize = 0 ;
u16 * tab = NULL ;
int err ;
err = nla_parse_nested ( tb , TCA_STAB_MAX , opt , stab_policy ) ;
if ( err < 0 )
return ERR_PTR ( err ) ;
if ( ! tb [ TCA_STAB_BASE ] )
return ERR_PTR ( - EINVAL ) ;
s = nla_data ( tb [ TCA_STAB_BASE ] ) ;
if ( s - > tsize > 0 ) {
if ( ! tb [ TCA_STAB_DATA ] )
return ERR_PTR ( - EINVAL ) ;
tab = nla_data ( tb [ TCA_STAB_DATA ] ) ;
tsize = nla_len ( tb [ TCA_STAB_DATA ] ) / sizeof ( u16 ) ;
}
if ( ! s | | tsize ! = s - > tsize | | ( ! tab & & tsize > 0 ) )
return ERR_PTR ( - EINVAL ) ;
2008-08-19 09:33:05 +04:00
spin_lock ( & qdisc_stab_lock ) ;
2008-07-20 11:08:47 +04:00
list_for_each_entry ( stab , & qdisc_stab_list , list ) {
if ( memcmp ( & stab - > szopts , s , sizeof ( * s ) ) )
continue ;
if ( tsize > 0 & & memcmp ( stab - > data , tab , tsize * sizeof ( u16 ) ) )
continue ;
stab - > refcnt + + ;
2008-08-19 09:33:05 +04:00
spin_unlock ( & qdisc_stab_lock ) ;
2008-07-20 11:08:47 +04:00
return stab ;
}
2008-08-19 09:33:05 +04:00
spin_unlock ( & qdisc_stab_lock ) ;
2008-07-20 11:08:47 +04:00
stab = kmalloc ( sizeof ( * stab ) + tsize * sizeof ( u16 ) , GFP_KERNEL ) ;
if ( ! stab )
return ERR_PTR ( - ENOMEM ) ;
stab - > refcnt = 1 ;
stab - > szopts = * s ;
if ( tsize > 0 )
memcpy ( stab - > data , tab , tsize * sizeof ( u16 ) ) ;
2008-08-19 09:33:05 +04:00
spin_lock ( & qdisc_stab_lock ) ;
2008-07-20 11:08:47 +04:00
list_add_tail ( & stab - > list , & qdisc_stab_list ) ;
2008-08-19 09:33:05 +04:00
spin_unlock ( & qdisc_stab_lock ) ;
2008-07-20 11:08:47 +04:00
return stab ;
}
void qdisc_put_stab ( struct qdisc_size_table * tab )
{
if ( ! tab )
return ;
2008-08-19 09:33:05 +04:00
spin_lock ( & qdisc_stab_lock ) ;
2008-07-20 11:08:47 +04:00
if ( - - tab - > refcnt = = 0 ) {
list_del ( & tab - > list ) ;
kfree ( tab ) ;
}
2008-08-19 09:33:05 +04:00
spin_unlock ( & qdisc_stab_lock ) ;
2008-07-20 11:08:47 +04:00
}
EXPORT_SYMBOL ( qdisc_put_stab ) ;
static int qdisc_dump_stab ( struct sk_buff * skb , struct qdisc_size_table * stab )
{
struct nlattr * nest ;
nest = nla_nest_start ( skb , TCA_STAB ) ;
NLA_PUT ( skb , TCA_STAB_BASE , sizeof ( stab - > szopts ) , & stab - > szopts ) ;
nla_nest_end ( skb , nest ) ;
return skb - > len ;
nla_put_failure :
return - 1 ;
}
void qdisc_calculate_pkt_len ( struct sk_buff * skb , struct qdisc_size_table * stab )
{
int pkt_len , slot ;
pkt_len = skb - > len + stab - > szopts . overhead ;
if ( unlikely ( ! stab - > szopts . tsize ) )
goto out ;
slot = pkt_len + stab - > szopts . cell_align ;
if ( unlikely ( slot < 0 ) )
slot = 0 ;
slot > > = stab - > szopts . cell_log ;
if ( likely ( slot < stab - > szopts . tsize ) )
pkt_len = stab - > data [ slot ] ;
else
pkt_len = stab - > data [ stab - > szopts . tsize - 1 ] *
( slot / stab - > szopts . tsize ) +
stab - > data [ slot % stab - > szopts . tsize ] ;
pkt_len < < = stab - > szopts . size_log ;
out :
if ( unlikely ( pkt_len < 1 ) )
pkt_len = 1 ;
qdisc_skb_cb ( skb ) - > pkt_len = pkt_len ;
}
EXPORT_SYMBOL ( qdisc_calculate_pkt_len ) ;
2007-03-16 11:19:15 +03:00
static enum hrtimer_restart qdisc_watchdog ( struct hrtimer * timer )
{
struct qdisc_watchdog * wd = container_of ( timer , struct qdisc_watchdog ,
timer ) ;
wd - > qdisc - > flags & = ~ TCQ_F_THROTTLED ;
2007-03-22 22:17:42 +03:00
smp_wmb ( ) ;
2008-08-19 07:51:18 +04:00
__netif_schedule ( qdisc_root ( wd - > qdisc ) ) ;
2007-03-22 22:18:35 +03:00
2007-03-16 11:19:15 +03:00
return HRTIMER_NORESTART ;
}
void qdisc_watchdog_init ( struct qdisc_watchdog * wd , struct Qdisc * qdisc )
{
hrtimer_init ( & wd - > timer , CLOCK_MONOTONIC , HRTIMER_MODE_ABS ) ;
wd - > timer . function = qdisc_watchdog ;
wd - > qdisc = qdisc ;
}
EXPORT_SYMBOL ( qdisc_watchdog_init ) ;
void qdisc_watchdog_schedule ( struct qdisc_watchdog * wd , psched_time_t expires )
{
ktime_t time ;
2008-08-21 16:11:14 +04:00
if ( test_bit ( __QDISC_STATE_DEACTIVATED ,
& qdisc_root_sleeping ( wd - > qdisc ) - > state ) )
return ;
2007-03-16 11:19:15 +03:00
wd - > qdisc - > flags | = TCQ_F_THROTTLED ;
time = ktime_set ( 0 , 0 ) ;
time = ktime_add_ns ( time , PSCHED_US2NS ( expires ) ) ;
hrtimer_start ( & wd - > timer , time , HRTIMER_MODE_ABS ) ;
}
EXPORT_SYMBOL ( qdisc_watchdog_schedule ) ;
void qdisc_watchdog_cancel ( struct qdisc_watchdog * wd )
{
hrtimer_cancel ( & wd - > timer ) ;
wd - > qdisc - > flags & = ~ TCQ_F_THROTTLED ;
}
EXPORT_SYMBOL ( qdisc_watchdog_cancel ) ;
2005-04-17 02:20:36 +04:00
2008-07-23 01:20:11 +04:00
static struct hlist_head * qdisc_class_hash_alloc ( unsigned int n )
2008-07-06 10:21:31 +04:00
{
unsigned int size = n * sizeof ( struct hlist_head ) , i ;
struct hlist_head * h ;
if ( size < = PAGE_SIZE )
h = kmalloc ( size , GFP_KERNEL ) ;
else
h = ( struct hlist_head * )
__get_free_pages ( GFP_KERNEL , get_order ( size ) ) ;
if ( h ! = NULL ) {
for ( i = 0 ; i < n ; i + + )
INIT_HLIST_HEAD ( & h [ i ] ) ;
}
return h ;
}
static void qdisc_class_hash_free ( struct hlist_head * h , unsigned int n )
{
unsigned int size = n * sizeof ( struct hlist_head ) ;
if ( size < = PAGE_SIZE )
kfree ( h ) ;
else
free_pages ( ( unsigned long ) h , get_order ( size ) ) ;
}
void qdisc_class_hash_grow ( struct Qdisc * sch , struct Qdisc_class_hash * clhash )
{
struct Qdisc_class_common * cl ;
struct hlist_node * n , * next ;
struct hlist_head * nhash , * ohash ;
unsigned int nsize , nmask , osize ;
unsigned int i , h ;
/* Rehash when load factor exceeds 0.75 */
if ( clhash - > hashelems * 4 < = clhash - > hashsize * 3 )
return ;
nsize = clhash - > hashsize * 2 ;
nmask = nsize - 1 ;
nhash = qdisc_class_hash_alloc ( nsize ) ;
if ( nhash = = NULL )
return ;
ohash = clhash - > hash ;
osize = clhash - > hashsize ;
sch_tree_lock ( sch ) ;
for ( i = 0 ; i < osize ; i + + ) {
hlist_for_each_entry_safe ( cl , n , next , & ohash [ i ] , hnode ) {
h = qdisc_class_hash ( cl - > classid , nmask ) ;
hlist_add_head ( & cl - > hnode , & nhash [ h ] ) ;
}
}
clhash - > hash = nhash ;
clhash - > hashsize = nsize ;
clhash - > hashmask = nmask ;
sch_tree_unlock ( sch ) ;
qdisc_class_hash_free ( ohash , osize ) ;
}
EXPORT_SYMBOL ( qdisc_class_hash_grow ) ;
int qdisc_class_hash_init ( struct Qdisc_class_hash * clhash )
{
unsigned int size = 4 ;
clhash - > hash = qdisc_class_hash_alloc ( size ) ;
if ( clhash - > hash = = NULL )
return - ENOMEM ;
clhash - > hashsize = size ;
clhash - > hashmask = size - 1 ;
clhash - > hashelems = 0 ;
return 0 ;
}
EXPORT_SYMBOL ( qdisc_class_hash_init ) ;
void qdisc_class_hash_destroy ( struct Qdisc_class_hash * clhash )
{
qdisc_class_hash_free ( clhash - > hash , clhash - > hashsize ) ;
}
EXPORT_SYMBOL ( qdisc_class_hash_destroy ) ;
void qdisc_class_hash_insert ( struct Qdisc_class_hash * clhash ,
struct Qdisc_class_common * cl )
{
unsigned int h ;
INIT_HLIST_NODE ( & cl - > hnode ) ;
h = qdisc_class_hash ( cl - > classid , clhash - > hashmask ) ;
hlist_add_head ( & cl - > hnode , & clhash - > hash [ h ] ) ;
clhash - > hashelems + + ;
}
EXPORT_SYMBOL ( qdisc_class_hash_insert ) ;
void qdisc_class_hash_remove ( struct Qdisc_class_hash * clhash ,
struct Qdisc_class_common * cl )
{
hlist_del ( & cl - > hnode ) ;
clhash - > hashelems - - ;
}
EXPORT_SYMBOL ( qdisc_class_hash_remove ) ;
2005-04-17 02:20:36 +04:00
/* Allocate an unique handle from space managed by kernel */
static u32 qdisc_alloc_handle ( struct net_device * dev )
{
int i = 0x10000 ;
static u32 autohandle = TC_H_MAKE ( 0x80000000U , 0 ) ;
do {
autohandle + = TC_H_MAKE ( 0x10000U , 0 ) ;
if ( autohandle = = TC_H_MAKE ( TC_H_ROOT , 0 ) )
autohandle = TC_H_MAKE ( 0x80000000U , 0 ) ;
} while ( qdisc_lookup ( dev , autohandle ) & & - - i > 0 ) ;
return i > 0 ? autohandle : 0 ;
}
2008-07-17 15:54:10 +04:00
/* Attach toplevel qdisc to device queue. */
2005-04-17 02:20:36 +04:00
2008-07-17 15:54:10 +04:00
static struct Qdisc * dev_graft_qdisc ( struct netdev_queue * dev_queue ,
struct Qdisc * qdisc )
2005-04-17 02:20:36 +04:00
{
2008-07-30 13:37:46 +04:00
struct Qdisc * oqdisc = dev_queue - > qdisc_sleeping ;
2008-07-16 14:00:19 +04:00
spinlock_t * root_lock ;
2008-08-27 13:12:52 +04:00
root_lock = qdisc_lock ( oqdisc ) ;
2008-07-16 14:00:19 +04:00
spin_lock_bh ( root_lock ) ;
2008-07-30 13:37:46 +04:00
/* Prune old scheduler */
if ( oqdisc & & atomic_read ( & oqdisc - > refcnt ) < = 1 )
qdisc_reset ( oqdisc ) ;
2005-04-17 02:20:36 +04:00
2008-07-30 13:37:46 +04:00
/* ... and graft new one */
if ( qdisc = = NULL )
qdisc = & noop_qdisc ;
dev_queue - > qdisc_sleeping = qdisc ;
2008-08-27 13:22:07 +04:00
rcu_assign_pointer ( dev_queue - > qdisc , & noop_qdisc ) ;
2005-04-17 02:20:36 +04:00
2008-07-16 14:00:19 +04:00
spin_unlock_bh ( root_lock ) ;
2005-04-17 02:20:36 +04:00
return oqdisc ;
}
2006-11-30 04:35:48 +03:00
void qdisc_tree_decrease_qlen ( struct Qdisc * sch , unsigned int n )
{
2007-11-14 12:44:41 +03:00
const struct Qdisc_class_ops * cops ;
2006-11-30 04:35:48 +03:00
unsigned long cl ;
u32 parentid ;
if ( n = = 0 )
return ;
while ( ( parentid = sch - > parent ) ) {
2008-04-15 02:10:42 +04:00
if ( TC_H_MAJ ( parentid ) = = TC_H_MAJ ( TC_H_INGRESS ) )
return ;
2008-07-09 04:06:30 +04:00
sch = qdisc_lookup ( qdisc_dev ( sch ) , TC_H_MAJ ( parentid ) ) ;
2007-07-31 04:11:50 +04:00
if ( sch = = NULL ) {
WARN_ON ( parentid ! = TC_H_ROOT ) ;
return ;
}
2006-11-30 04:35:48 +03:00
cops = sch - > ops - > cl_ops ;
if ( cops - > qlen_notify ) {
cl = cops - > get ( sch , parentid ) ;
cops - > qlen_notify ( sch , cl ) ;
cops - > put ( sch , cl ) ;
}
sch - > q . qlen - = n ;
}
}
EXPORT_SYMBOL ( qdisc_tree_decrease_qlen ) ;
2005-04-17 02:20:36 +04:00
2008-07-17 15:54:10 +04:00
static void notify_and_destroy ( struct sk_buff * skb , struct nlmsghdr * n , u32 clid ,
struct Qdisc * old , struct Qdisc * new )
{
if ( new | | old )
qdisc_notify ( skb , n , clid , old , new ) ;
2005-04-17 02:20:36 +04:00
2008-08-19 08:03:15 +04:00
if ( old )
2008-07-17 15:54:10 +04:00
qdisc_destroy ( old ) ;
}
/* Graft qdisc "new" to class "classid" of qdisc "parent" or
* to device " dev " .
*
* When appropriate send a netlink notification using ' skb '
* and " n " .
*
* On success , destroy old qdisc .
2005-04-17 02:20:36 +04:00
*/
static int qdisc_graft ( struct net_device * dev , struct Qdisc * parent ,
2008-07-17 15:54:10 +04:00
struct sk_buff * skb , struct nlmsghdr * n , u32 classid ,
struct Qdisc * new , struct Qdisc * old )
2005-04-17 02:20:36 +04:00
{
2008-07-17 15:54:10 +04:00
struct Qdisc * q = old ;
2005-04-17 02:20:36 +04:00
int err = 0 ;
2007-02-09 17:25:16 +03:00
if ( parent = = NULL ) {
2008-07-17 15:54:10 +04:00
unsigned int i , num_q , ingress ;
ingress = 0 ;
num_q = dev - > num_tx_queues ;
2008-07-30 13:37:46 +04:00
if ( ( q & & q - > flags & TCQ_F_INGRESS ) | |
( new & & new - > flags & TCQ_F_INGRESS ) ) {
2008-07-17 15:54:10 +04:00
num_q = 1 ;
ingress = 1 ;
}
if ( dev - > flags & IFF_UP )
dev_deactivate ( dev ) ;
for ( i = 0 ; i < num_q ; i + + ) {
struct netdev_queue * dev_queue = & dev - > rx_queue ;
if ( ! ingress )
dev_queue = netdev_get_tx_queue ( dev , i ) ;
2008-07-30 13:37:46 +04:00
old = dev_graft_qdisc ( dev_queue , new ) ;
if ( new & & i > 0 )
atomic_inc ( & new - > refcnt ) ;
2008-07-17 15:54:10 +04:00
notify_and_destroy ( skb , n , classid , old , new ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-17 15:54:10 +04:00
if ( dev - > flags & IFF_UP )
dev_activate ( dev ) ;
2005-04-17 02:20:36 +04:00
} else {
2007-11-14 12:44:41 +03:00
const struct Qdisc_class_ops * cops = parent - > ops - > cl_ops ;
2005-04-17 02:20:36 +04:00
err = - EINVAL ;
if ( cops ) {
unsigned long cl = cops - > get ( parent , classid ) ;
if ( cl ) {
2008-07-17 15:54:10 +04:00
err = cops - > graft ( parent , cl , new , & old ) ;
2005-04-17 02:20:36 +04:00
cops - > put ( parent , cl ) ;
}
}
2008-07-17 15:54:10 +04:00
if ( ! err )
notify_and_destroy ( skb , n , classid , old , new ) ;
2005-04-17 02:20:36 +04:00
}
return err ;
}
2008-08-19 07:53:34 +04:00
/* lockdep annotation is needed for ingress; egress gets it only for name */
static struct lock_class_key qdisc_tx_lock ;
static struct lock_class_key qdisc_rx_lock ;
2005-04-17 02:20:36 +04:00
/*
Allocate and initialize new qdisc .
Parameters are passed via opt .
*/
static struct Qdisc *
2008-07-09 03:55:56 +04:00
qdisc_create ( struct net_device * dev , struct netdev_queue * dev_queue ,
u32 parent , u32 handle , struct nlattr * * tca , int * errp )
2005-04-17 02:20:36 +04:00
{
int err ;
2008-01-23 09:11:17 +03:00
struct nlattr * kind = tca [ TCA_KIND ] ;
2005-04-17 02:20:36 +04:00
struct Qdisc * sch ;
struct Qdisc_ops * ops ;
2008-07-20 11:08:47 +04:00
struct qdisc_size_table * stab ;
2005-04-17 02:20:36 +04:00
ops = qdisc_lookup_ops ( kind ) ;
2008-10-17 02:24:51 +04:00
# ifdef CONFIG_MODULES
2005-04-17 02:20:36 +04:00
if ( ops = = NULL & & kind ! = NULL ) {
char name [ IFNAMSIZ ] ;
2008-01-23 09:11:17 +03:00
if ( nla_strlcpy ( name , kind , IFNAMSIZ ) < IFNAMSIZ ) {
2005-04-17 02:20:36 +04:00
/* We dropped the RTNL semaphore in order to
* perform the module load . So , even if we
* succeeded in loading the module we have to
* tell the caller to replay the request . We
* indicate this using - EAGAIN .
* We replay the request because the device may
* go away in the mean time .
*/
rtnl_unlock ( ) ;
request_module ( " sch_%s " , name ) ;
rtnl_lock ( ) ;
ops = qdisc_lookup_ops ( kind ) ;
if ( ops ! = NULL ) {
/* We will try again qdisc_lookup_ops,
* so don ' t keep a reference .
*/
module_put ( ops - > owner ) ;
err = - EAGAIN ;
goto err_out ;
}
}
}
# endif
2006-08-04 03:36:51 +04:00
err = - ENOENT ;
2005-04-17 02:20:36 +04:00
if ( ops = = NULL )
goto err_out ;
2008-07-09 04:06:30 +04:00
sch = qdisc_alloc ( dev_queue , ops ) ;
2005-07-06 01:15:09 +04:00
if ( IS_ERR ( sch ) ) {
err = PTR_ERR ( sch ) ;
2005-04-17 02:20:36 +04:00
goto err_out2 ;
2005-07-06 01:15:09 +04:00
}
2005-04-17 02:20:36 +04:00
2007-07-31 04:11:50 +04:00
sch - > parent = parent ;
2005-07-06 01:15:09 +04:00
if ( handle = = TC_H_INGRESS ) {
2005-04-17 02:20:36 +04:00
sch - > flags | = TCQ_F_INGRESS ;
2005-07-06 01:15:09 +04:00
handle = TC_H_MAKE ( TC_H_INGRESS , 0 ) ;
2008-08-19 07:53:34 +04:00
lockdep_set_class ( qdisc_lock ( sch ) , & qdisc_rx_lock ) ;
2007-04-17 04:07:08 +04:00
} else {
if ( handle = = 0 ) {
handle = qdisc_alloc_handle ( dev ) ;
err = - ENOMEM ;
if ( handle = = 0 )
goto err_out3 ;
}
2008-08-19 07:53:34 +04:00
lockdep_set_class ( qdisc_lock ( sch ) , & qdisc_tx_lock ) ;
2005-04-17 02:20:36 +04:00
}
2005-07-06 01:15:09 +04:00
sch - > handle = handle ;
2005-04-17 02:20:36 +04:00
2008-01-23 09:11:17 +03:00
if ( ! ops - > init | | ( err = ops - > init ( sch , tca [ TCA_OPTIONS ] ) ) = = 0 ) {
2008-07-20 11:08:47 +04:00
if ( tca [ TCA_STAB ] ) {
stab = qdisc_get_stab ( tca [ TCA_STAB ] ) ;
if ( IS_ERR ( stab ) ) {
err = PTR_ERR ( stab ) ;
goto err_out3 ;
}
sch - > stab = stab ;
}
2008-01-23 09:11:17 +03:00
if ( tca [ TCA_RATE ] ) {
2008-08-27 13:25:17 +04:00
spinlock_t * root_lock ;
if ( ( sch - > parent ! = TC_H_ROOT ) & &
! ( sch - > flags & TCQ_F_INGRESS ) )
root_lock = qdisc_root_sleeping_lock ( sch ) ;
else
root_lock = qdisc_lock ( sch ) ;
2005-07-06 01:15:53 +04:00
err = gen_new_estimator ( & sch - > bstats , & sch - > rate_est ,
2008-08-27 13:25:17 +04:00
root_lock , tca [ TCA_RATE ] ) ;
2005-07-06 01:15:53 +04:00
if ( err ) {
/*
* Any broken qdiscs that would require
* a ops - > reset ( ) here ? The qdisc was never
* in action so it shouldn ' t be necessary .
*/
if ( ops - > destroy )
ops - > destroy ( sch ) ;
goto err_out3 ;
}
}
2008-08-22 14:24:05 +04:00
qdisc_list_add ( sch ) ;
2005-04-17 02:20:36 +04:00
return sch ;
}
err_out3 :
2008-07-20 11:08:47 +04:00
qdisc_put_stab ( sch - > stab ) ;
2005-04-17 02:20:36 +04:00
dev_put ( dev ) ;
2005-07-06 01:15:09 +04:00
kfree ( ( char * ) sch - sch - > padded ) ;
2005-04-17 02:20:36 +04:00
err_out2 :
module_put ( ops - > owner ) ;
err_out :
* errp = err ;
return NULL ;
}
2008-01-23 09:11:17 +03:00
static int qdisc_change ( struct Qdisc * sch , struct nlattr * * tca )
2005-04-17 02:20:36 +04:00
{
2008-07-20 11:08:47 +04:00
struct qdisc_size_table * stab = NULL ;
int err = 0 ;
2005-04-17 02:20:36 +04:00
2008-07-20 11:08:47 +04:00
if ( tca [ TCA_OPTIONS ] ) {
2005-04-17 02:20:36 +04:00
if ( sch - > ops - > change = = NULL )
return - EINVAL ;
2008-01-23 09:11:17 +03:00
err = sch - > ops - > change ( sch , tca [ TCA_OPTIONS ] ) ;
2005-04-17 02:20:36 +04:00
if ( err )
return err ;
}
2008-07-20 11:08:47 +04:00
if ( tca [ TCA_STAB ] ) {
stab = qdisc_get_stab ( tca [ TCA_STAB ] ) ;
if ( IS_ERR ( stab ) )
return PTR_ERR ( stab ) ;
}
qdisc_put_stab ( sch - > stab ) ;
sch - > stab = stab ;
2008-01-23 09:11:17 +03:00
if ( tca [ TCA_RATE ] )
2005-04-17 02:20:36 +04:00
gen_replace_estimator ( & sch - > bstats , & sch - > rate_est ,
2008-08-27 13:25:17 +04:00
qdisc_root_sleeping_lock ( sch ) ,
tca [ TCA_RATE ] ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
struct check_loop_arg
{
struct qdisc_walker w ;
struct Qdisc * p ;
int depth ;
} ;
static int check_loop_fn ( struct Qdisc * q , unsigned long cl , struct qdisc_walker * w ) ;
static int check_loop ( struct Qdisc * q , struct Qdisc * p , int depth )
{
struct check_loop_arg arg ;
if ( q - > ops - > cl_ops = = NULL )
return 0 ;
arg . w . stop = arg . w . skip = arg . w . count = 0 ;
arg . w . fn = check_loop_fn ;
arg . depth = depth ;
arg . p = p ;
q - > ops - > cl_ops - > walk ( q , & arg . w ) ;
return arg . w . stop ? - ELOOP : 0 ;
}
static int
check_loop_fn ( struct Qdisc * q , unsigned long cl , struct qdisc_walker * w )
{
struct Qdisc * leaf ;
2007-11-14 12:44:41 +03:00
const struct Qdisc_class_ops * cops = q - > ops - > cl_ops ;
2005-04-17 02:20:36 +04:00
struct check_loop_arg * arg = ( struct check_loop_arg * ) w ;
leaf = cops - > leaf ( q , cl ) ;
if ( leaf ) {
if ( leaf = = arg - > p | | arg - > depth > 7 )
return - ELOOP ;
return check_loop ( leaf , arg - > p , arg - > depth + 1 ) ;
}
return 0 ;
}
/*
* Delete / get qdisc .
*/
static int tc_get_qdisc ( struct sk_buff * skb , struct nlmsghdr * n , void * arg )
{
2008-03-25 20:26:21 +03:00
struct net * net = sock_net ( skb - > sk ) ;
2005-04-17 02:20:36 +04:00
struct tcmsg * tcm = NLMSG_DATA ( n ) ;
2008-01-23 09:11:17 +03:00
struct nlattr * tca [ TCA_MAX + 1 ] ;
2005-04-17 02:20:36 +04:00
struct net_device * dev ;
u32 clid = tcm - > tcm_parent ;
struct Qdisc * q = NULL ;
struct Qdisc * p = NULL ;
int err ;
2007-11-30 16:21:31 +03:00
if ( net ! = & init_net )
return - EINVAL ;
2007-09-17 22:56:21 +04:00
if ( ( dev = __dev_get_by_index ( & init_net , tcm - > tcm_ifindex ) ) = = NULL )
2005-04-17 02:20:36 +04:00
return - ENODEV ;
2008-01-23 09:11:17 +03:00
err = nlmsg_parse ( n , sizeof ( * tcm ) , tca , TCA_MAX , NULL ) ;
if ( err < 0 )
return err ;
2005-04-17 02:20:36 +04:00
if ( clid ) {
if ( clid ! = TC_H_ROOT ) {
if ( TC_H_MAJ ( clid ) ! = TC_H_MAJ ( TC_H_INGRESS ) ) {
if ( ( p = qdisc_lookup ( dev , TC_H_MAJ ( clid ) ) ) = = NULL )
return - ENOENT ;
q = qdisc_leaf ( p , clid ) ;
} else { /* ingress */
2008-08-09 10:23:39 +04:00
q = dev - > rx_queue . qdisc_sleeping ;
2007-02-09 17:25:16 +03:00
}
2005-04-17 02:20:36 +04:00
} else {
2008-07-17 11:34:19 +04:00
struct netdev_queue * dev_queue ;
dev_queue = netdev_get_tx_queue ( dev , 0 ) ;
2008-07-09 04:42:10 +04:00
q = dev_queue - > qdisc_sleeping ;
2005-04-17 02:20:36 +04:00
}
if ( ! q )
return - ENOENT ;
if ( tcm - > tcm_handle & & q - > handle ! = tcm - > tcm_handle )
return - EINVAL ;
} else {
if ( ( q = qdisc_lookup ( dev , tcm - > tcm_handle ) ) = = NULL )
return - ENOENT ;
}
2008-01-23 09:11:17 +03:00
if ( tca [ TCA_KIND ] & & nla_strcmp ( tca [ TCA_KIND ] , q - > ops - > id ) )
2005-04-17 02:20:36 +04:00
return - EINVAL ;
if ( n - > nlmsg_type = = RTM_DELQDISC ) {
if ( ! clid )
return - EINVAL ;
if ( q - > handle = = 0 )
return - ENOENT ;
2008-07-17 15:54:10 +04:00
if ( ( err = qdisc_graft ( dev , p , skb , n , clid , NULL , q ) ) ! = 0 )
2005-04-17 02:20:36 +04:00
return err ;
} else {
qdisc_notify ( skb , n , clid , NULL , q ) ;
}
return 0 ;
}
/*
Create / change qdisc .
*/
static int tc_modify_qdisc ( struct sk_buff * skb , struct nlmsghdr * n , void * arg )
{
2008-03-25 20:26:21 +03:00
struct net * net = sock_net ( skb - > sk ) ;
2005-04-17 02:20:36 +04:00
struct tcmsg * tcm ;
2008-01-23 09:11:17 +03:00
struct nlattr * tca [ TCA_MAX + 1 ] ;
2005-04-17 02:20:36 +04:00
struct net_device * dev ;
u32 clid ;
struct Qdisc * q , * p ;
int err ;
2007-11-30 16:21:31 +03:00
if ( net ! = & init_net )
return - EINVAL ;
2005-04-17 02:20:36 +04:00
replay :
/* Reinit, just in case something touches this. */
tcm = NLMSG_DATA ( n ) ;
clid = tcm - > tcm_parent ;
q = p = NULL ;
2007-09-17 22:56:21 +04:00
if ( ( dev = __dev_get_by_index ( & init_net , tcm - > tcm_ifindex ) ) = = NULL )
2005-04-17 02:20:36 +04:00
return - ENODEV ;
2008-01-23 09:11:17 +03:00
err = nlmsg_parse ( n , sizeof ( * tcm ) , tca , TCA_MAX , NULL ) ;
if ( err < 0 )
return err ;
2005-04-17 02:20:36 +04:00
if ( clid ) {
if ( clid ! = TC_H_ROOT ) {
if ( clid ! = TC_H_INGRESS ) {
if ( ( p = qdisc_lookup ( dev , TC_H_MAJ ( clid ) ) ) = = NULL )
return - ENOENT ;
q = qdisc_leaf ( p , clid ) ;
} else { /*ingress */
2008-08-09 10:23:39 +04:00
q = dev - > rx_queue . qdisc_sleeping ;
2005-04-17 02:20:36 +04:00
}
} else {
2008-07-17 11:34:19 +04:00
struct netdev_queue * dev_queue ;
dev_queue = netdev_get_tx_queue ( dev , 0 ) ;
2008-07-09 04:42:10 +04:00
q = dev_queue - > qdisc_sleeping ;
2005-04-17 02:20:36 +04:00
}
/* It may be default qdisc, ignore it */
if ( q & & q - > handle = = 0 )
q = NULL ;
if ( ! q | | ! tcm - > tcm_handle | | q - > handle ! = tcm - > tcm_handle ) {
if ( tcm - > tcm_handle ) {
if ( q & & ! ( n - > nlmsg_flags & NLM_F_REPLACE ) )
return - EEXIST ;
if ( TC_H_MIN ( tcm - > tcm_handle ) )
return - EINVAL ;
if ( ( q = qdisc_lookup ( dev , tcm - > tcm_handle ) ) = = NULL )
goto create_n_graft ;
if ( n - > nlmsg_flags & NLM_F_EXCL )
return - EEXIST ;
2008-01-23 09:11:17 +03:00
if ( tca [ TCA_KIND ] & & nla_strcmp ( tca [ TCA_KIND ] , q - > ops - > id ) )
2005-04-17 02:20:36 +04:00
return - EINVAL ;
if ( q = = p | |
( p & & check_loop ( q , p , 0 ) ) )
return - ELOOP ;
atomic_inc ( & q - > refcnt ) ;
goto graft ;
} else {
if ( q = = NULL )
goto create_n_graft ;
/* This magic test requires explanation.
*
* We know , that some child q is already
* attached to this parent and have choice :
* either to change it or to create / graft new one .
*
* 1. We are allowed to create / graft only
* if CREATE and REPLACE flags are set .
*
* 2. If EXCL is set , requestor wanted to say ,
* that qdisc tcm_handle is not expected
* to exist , so that we choose create / graft too .
*
* 3. The last case is when no flags are set .
* Alas , it is sort of hole in API , we
* cannot decide what to do unambiguously .
* For now we select create / graft , if
* user gave KIND , which does not match existing .
*/
if ( ( n - > nlmsg_flags & NLM_F_CREATE ) & &
( n - > nlmsg_flags & NLM_F_REPLACE ) & &
( ( n - > nlmsg_flags & NLM_F_EXCL ) | |
2008-01-23 09:11:17 +03:00
( tca [ TCA_KIND ] & &
nla_strcmp ( tca [ TCA_KIND ] , q - > ops - > id ) ) ) )
2005-04-17 02:20:36 +04:00
goto create_n_graft ;
}
}
} else {
if ( ! tcm - > tcm_handle )
return - EINVAL ;
q = qdisc_lookup ( dev , tcm - > tcm_handle ) ;
}
/* Change qdisc parameters */
if ( q = = NULL )
return - ENOENT ;
if ( n - > nlmsg_flags & NLM_F_EXCL )
return - EEXIST ;
2008-01-23 09:11:17 +03:00
if ( tca [ TCA_KIND ] & & nla_strcmp ( tca [ TCA_KIND ] , q - > ops - > id ) )
2005-04-17 02:20:36 +04:00
return - EINVAL ;
err = qdisc_change ( q , tca ) ;
if ( err = = 0 )
qdisc_notify ( skb , n , clid , NULL , q ) ;
return err ;
create_n_graft :
if ( ! ( n - > nlmsg_flags & NLM_F_CREATE ) )
return - ENOENT ;
if ( clid = = TC_H_INGRESS )
2008-07-09 03:55:56 +04:00
q = qdisc_create ( dev , & dev - > rx_queue ,
tcm - > tcm_parent , tcm - > tcm_parent ,
2007-07-31 04:11:50 +04:00
tca , & err ) ;
2007-02-09 17:25:16 +03:00
else
2008-07-17 11:34:19 +04:00
q = qdisc_create ( dev , netdev_get_tx_queue ( dev , 0 ) ,
2008-07-09 03:55:56 +04:00
tcm - > tcm_parent , tcm - > tcm_handle ,
2007-07-31 04:11:50 +04:00
tca , & err ) ;
2005-04-17 02:20:36 +04:00
if ( q = = NULL ) {
if ( err = = - EAGAIN )
goto replay ;
return err ;
}
graft :
2008-08-19 09:30:01 +04:00
err = qdisc_graft ( dev , p , skb , n , clid , q , NULL ) ;
if ( err ) {
if ( q )
qdisc_destroy ( q ) ;
return err ;
2005-04-17 02:20:36 +04:00
}
2008-08-19 09:30:01 +04:00
2005-04-17 02:20:36 +04:00
return 0 ;
}
static int tc_fill_qdisc ( struct sk_buff * skb , struct Qdisc * q , u32 clid ,
2005-06-19 09:55:31 +04:00
u32 pid , u32 seq , u16 flags , int event )
2005-04-17 02:20:36 +04:00
{
struct tcmsg * tcm ;
struct nlmsghdr * nlh ;
2007-04-20 07:29:13 +04:00
unsigned char * b = skb_tail_pointer ( skb ) ;
2005-04-17 02:20:36 +04:00
struct gnet_dump d ;
2005-06-19 09:55:31 +04:00
nlh = NLMSG_NEW ( skb , pid , seq , event , sizeof ( * tcm ) , flags ) ;
2005-04-17 02:20:36 +04:00
tcm = NLMSG_DATA ( nlh ) ;
tcm - > tcm_family = AF_UNSPEC ;
2005-06-28 23:55:30 +04:00
tcm - > tcm__pad1 = 0 ;
tcm - > tcm__pad2 = 0 ;
2008-07-09 04:06:30 +04:00
tcm - > tcm_ifindex = qdisc_dev ( q ) - > ifindex ;
2005-04-17 02:20:36 +04:00
tcm - > tcm_parent = clid ;
tcm - > tcm_handle = q - > handle ;
tcm - > tcm_info = atomic_read ( & q - > refcnt ) ;
2008-01-24 07:34:28 +03:00
NLA_PUT_STRING ( skb , TCA_KIND , q - > ops - > id ) ;
2005-04-17 02:20:36 +04:00
if ( q - > ops - > dump & & q - > ops - > dump ( q , skb ) < 0 )
2008-01-23 09:11:17 +03:00
goto nla_put_failure ;
2005-04-17 02:20:36 +04:00
q - > qstats . qlen = q - > q . qlen ;
2008-07-20 11:08:47 +04:00
if ( q - > stab & & qdisc_dump_stab ( skb , q - > stab ) < 0 )
goto nla_put_failure ;
2008-08-30 01:21:52 +04:00
if ( gnet_stats_start_copy_compat ( skb , TCA_STATS2 , TCA_STATS , TCA_XSTATS ,
qdisc_root_sleeping_lock ( q ) , & d ) < 0 )
2008-01-23 09:11:17 +03:00
goto nla_put_failure ;
2005-04-17 02:20:36 +04:00
if ( q - > ops - > dump_stats & & q - > ops - > dump_stats ( q , & d ) < 0 )
2008-01-23 09:11:17 +03:00
goto nla_put_failure ;
2005-04-17 02:20:36 +04:00
if ( gnet_stats_copy_basic ( & d , & q - > bstats ) < 0 | |
gnet_stats_copy_rate_est ( & d , & q - > rate_est ) < 0 | |
gnet_stats_copy_queue ( & d , & q - > qstats ) < 0 )
2008-01-23 09:11:17 +03:00
goto nla_put_failure ;
2007-02-09 17:25:16 +03:00
2005-04-17 02:20:36 +04:00
if ( gnet_stats_finish_copy ( & d ) < 0 )
2008-01-23 09:11:17 +03:00
goto nla_put_failure ;
2007-02-09 17:25:16 +03:00
2007-04-20 07:29:13 +04:00
nlh - > nlmsg_len = skb_tail_pointer ( skb ) - b ;
2005-04-17 02:20:36 +04:00
return skb - > len ;
nlmsg_failure :
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 ;
}
static int qdisc_notify ( struct sk_buff * oskb , struct nlmsghdr * n ,
u32 clid , struct Qdisc * old , struct Qdisc * new )
{
struct sk_buff * skb ;
u32 pid = oskb ? NETLINK_CB ( oskb ) . pid : 0 ;
skb = alloc_skb ( NLMSG_GOODSIZE , GFP_KERNEL ) ;
if ( ! skb )
return - ENOBUFS ;
if ( old & & old - > handle ) {
if ( tc_fill_qdisc ( skb , old , clid , pid , n - > nlmsg_seq , 0 , RTM_DELQDISC ) < 0 )
goto err_out ;
}
if ( new ) {
if ( tc_fill_qdisc ( skb , new , clid , pid , n - > nlmsg_seq , old ? NLM_F_REPLACE : 0 , RTM_NEWQDISC ) < 0 )
goto err_out ;
}
if ( skb - > len )
2007-11-20 09:26:51 +03:00
return rtnetlink_send ( skb , & init_net , pid , RTNLGRP_TC , n - > nlmsg_flags & NLM_F_ECHO ) ;
2005-04-17 02:20:36 +04:00
err_out :
kfree_skb ( skb ) ;
return - EINVAL ;
}
2008-07-19 09:50:15 +04:00
static bool tc_qdisc_dump_ignore ( struct Qdisc * q )
{
return ( q - > flags & TCQ_F_BUILTIN ) ? true : false ;
}
static int tc_dump_qdisc_root ( struct Qdisc * root , struct sk_buff * skb ,
struct netlink_callback * cb ,
int * q_idx_p , int s_q_idx )
{
int ret = 0 , q_idx = * q_idx_p ;
struct Qdisc * q ;
if ( ! root )
return 0 ;
q = root ;
if ( q_idx < s_q_idx ) {
q_idx + + ;
} else {
if ( ! tc_qdisc_dump_ignore ( q ) & &
tc_fill_qdisc ( skb , q , q - > parent , NETLINK_CB ( cb - > skb ) . pid ,
cb - > nlh - > nlmsg_seq , NLM_F_MULTI , RTM_NEWQDISC ) < = 0 )
goto done ;
q_idx + + ;
}
list_for_each_entry ( q , & root - > list , list ) {
if ( q_idx < s_q_idx ) {
q_idx + + ;
continue ;
}
if ( ! tc_qdisc_dump_ignore ( q ) & &
tc_fill_qdisc ( skb , q , q - > parent , NETLINK_CB ( cb - > skb ) . pid ,
cb - > nlh - > nlmsg_seq , NLM_F_MULTI , RTM_NEWQDISC ) < = 0 )
goto done ;
q_idx + + ;
}
out :
* q_idx_p = q_idx ;
return ret ;
done :
ret = - 1 ;
goto out ;
}
2005-04-17 02:20:36 +04:00
static int tc_dump_qdisc ( struct sk_buff * skb , struct netlink_callback * cb )
{
2008-03-25 20:26:21 +03:00
struct net * net = sock_net ( skb - > sk ) ;
2005-04-17 02:20:36 +04:00
int idx , q_idx ;
int s_idx , s_q_idx ;
struct net_device * dev ;
2007-11-30 16:21:31 +03:00
if ( net ! = & init_net )
return 0 ;
2005-04-17 02:20:36 +04:00
s_idx = cb - > args [ 0 ] ;
s_q_idx = q_idx = cb - > args [ 1 ] ;
read_lock ( & dev_base_lock ) ;
2007-05-04 02:13:45 +04:00
idx = 0 ;
2007-09-17 22:56:21 +04:00
for_each_netdev ( & init_net , dev ) {
2008-07-19 09:50:15 +04:00
struct netdev_queue * dev_queue ;
2005-04-17 02:20:36 +04:00
if ( idx < s_idx )
2007-05-04 02:13:45 +04:00
goto cont ;
2005-04-17 02:20:36 +04:00
if ( idx > s_idx )
s_q_idx = 0 ;
q_idx = 0 ;
2008-07-19 09:50:15 +04:00
dev_queue = netdev_get_tx_queue ( dev , 0 ) ;
2008-08-08 07:26:40 +04:00
if ( tc_dump_qdisc_root ( dev_queue - > qdisc_sleeping , skb , cb , & q_idx , s_q_idx ) < 0 )
2008-07-19 09:50:15 +04:00
goto done ;
dev_queue = & dev - > rx_queue ;
2008-08-08 07:26:40 +04:00
if ( tc_dump_qdisc_root ( dev_queue - > qdisc_sleeping , skb , cb , & q_idx , s_q_idx ) < 0 )
2008-07-19 09:50:15 +04:00
goto done ;
2007-05-04 02:13:45 +04:00
cont :
idx + + ;
2005-04-17 02:20:36 +04:00
}
done :
read_unlock ( & dev_base_lock ) ;
cb - > args [ 0 ] = idx ;
cb - > args [ 1 ] = q_idx ;
return skb - > len ;
}
/************************************************
* Traffic classes manipulation . *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int tc_ctl_tclass ( struct sk_buff * skb , struct nlmsghdr * n , void * arg )
{
2008-03-25 20:26:21 +03:00
struct net * net = sock_net ( skb - > sk ) ;
2008-07-09 04:42:10 +04:00
struct netdev_queue * dev_queue ;
2005-04-17 02:20:36 +04:00
struct tcmsg * tcm = NLMSG_DATA ( n ) ;
2008-01-23 09:11:17 +03:00
struct nlattr * tca [ TCA_MAX + 1 ] ;
2005-04-17 02:20:36 +04:00
struct net_device * dev ;
struct Qdisc * q = NULL ;
2007-11-14 12:44:41 +03:00
const struct Qdisc_class_ops * cops ;
2005-04-17 02:20:36 +04:00
unsigned long cl = 0 ;
unsigned long new_cl ;
u32 pid = tcm - > tcm_parent ;
u32 clid = tcm - > tcm_handle ;
u32 qid = TC_H_MAJ ( clid ) ;
int err ;
2007-11-30 16:21:31 +03:00
if ( net ! = & init_net )
return - EINVAL ;
2007-09-17 22:56:21 +04:00
if ( ( dev = __dev_get_by_index ( & init_net , tcm - > tcm_ifindex ) ) = = NULL )
2005-04-17 02:20:36 +04:00
return - ENODEV ;
2008-01-23 09:11:17 +03:00
err = nlmsg_parse ( n , sizeof ( * tcm ) , tca , TCA_MAX , NULL ) ;
if ( err < 0 )
return err ;
2005-04-17 02:20:36 +04:00
/*
parent = = TC_H_UNSPEC - unspecified parent .
parent = = TC_H_ROOT - class is root , which has no parent .
parent = = X : 0 - parent is root class .
parent = = X : Y - parent is a node in hierarchy .
parent = = 0 : Y - parent is X : Y , where X : 0 is qdisc .
handle = = 0 : 0 - generate handle from kernel pool .
handle = = 0 : Y - class is X : Y , where X : 0 is qdisc .
handle = = X : Y - clear .
handle = = X : 0 - root class .
*/
/* Step 1. Determine qdisc handle X:0 */
2008-07-17 11:34:19 +04:00
dev_queue = netdev_get_tx_queue ( dev , 0 ) ;
2005-04-17 02:20:36 +04:00
if ( pid ! = TC_H_ROOT ) {
u32 qid1 = TC_H_MAJ ( pid ) ;
if ( qid & & qid1 ) {
/* If both majors are known, they must be identical. */
if ( qid ! = qid1 )
return - EINVAL ;
} else if ( qid1 ) {
qid = qid1 ;
} else if ( qid = = 0 )
2008-07-09 04:42:10 +04:00
qid = dev_queue - > qdisc_sleeping - > handle ;
2005-04-17 02:20:36 +04:00
/* Now qid is genuine qdisc handle consistent
both with parent and child .
TC_H_MAJ ( pid ) still may be unspecified , complete it now .
*/
if ( pid )
pid = TC_H_MAKE ( qid , pid ) ;
} else {
if ( qid = = 0 )
2008-07-09 04:42:10 +04:00
qid = dev_queue - > qdisc_sleeping - > handle ;
2005-04-17 02:20:36 +04:00
}
/* OK. Locate qdisc */
2007-02-09 17:25:16 +03:00
if ( ( q = qdisc_lookup ( dev , qid ) ) = = NULL )
2005-04-17 02:20:36 +04:00
return - ENOENT ;
/* An check that it supports classes */
cops = q - > ops - > cl_ops ;
if ( cops = = NULL )
return - EINVAL ;
/* Now try to get class */
if ( clid = = 0 ) {
if ( pid = = TC_H_ROOT )
clid = qid ;
} else
clid = TC_H_MAKE ( qid , clid ) ;
if ( clid )
cl = cops - > get ( q , clid ) ;
if ( cl = = 0 ) {
err = - ENOENT ;
if ( n - > nlmsg_type ! = RTM_NEWTCLASS | | ! ( n - > nlmsg_flags & NLM_F_CREATE ) )
goto out ;
} else {
switch ( n - > nlmsg_type ) {
2007-02-09 17:25:16 +03:00
case RTM_NEWTCLASS :
2005-04-17 02:20:36 +04:00
err = - EEXIST ;
if ( n - > nlmsg_flags & NLM_F_EXCL )
goto out ;
break ;
case RTM_DELTCLASS :
err = cops - > delete ( q , cl ) ;
if ( err = = 0 )
tclass_notify ( skb , n , q , cl , RTM_DELTCLASS ) ;
goto out ;
case RTM_GETTCLASS :
err = tclass_notify ( skb , n , q , cl , RTM_NEWTCLASS ) ;
goto out ;
default :
err = - EINVAL ;
goto out ;
}
}
new_cl = cl ;
err = cops - > change ( q , clid , pid , tca , & new_cl ) ;
if ( err = = 0 )
tclass_notify ( skb , n , q , new_cl , RTM_NEWTCLASS ) ;
out :
if ( cl )
cops - > put ( q , cl ) ;
return err ;
}
static int tc_fill_tclass ( struct sk_buff * skb , struct Qdisc * q ,
unsigned long cl ,
2005-06-19 09:55:31 +04:00
u32 pid , u32 seq , u16 flags , int event )
2005-04-17 02:20:36 +04:00
{
struct tcmsg * tcm ;
struct nlmsghdr * nlh ;
2007-04-20 07:29:13 +04:00
unsigned char * b = skb_tail_pointer ( skb ) ;
2005-04-17 02:20:36 +04:00
struct gnet_dump d ;
2007-11-14 12:44:41 +03:00
const struct Qdisc_class_ops * cl_ops = q - > ops - > cl_ops ;
2005-04-17 02:20:36 +04:00
2005-06-19 09:55:31 +04:00
nlh = NLMSG_NEW ( skb , pid , seq , event , sizeof ( * tcm ) , flags ) ;
2005-04-17 02:20:36 +04:00
tcm = NLMSG_DATA ( nlh ) ;
tcm - > tcm_family = AF_UNSPEC ;
2008-07-09 04:06:30 +04:00
tcm - > tcm_ifindex = qdisc_dev ( q ) - > ifindex ;
2005-04-17 02:20:36 +04:00
tcm - > tcm_parent = q - > handle ;
tcm - > tcm_handle = q - > handle ;
tcm - > tcm_info = 0 ;
2008-01-24 07:34:28 +03:00
NLA_PUT_STRING ( skb , TCA_KIND , q - > ops - > id ) ;
2005-04-17 02:20:36 +04:00
if ( cl_ops - > dump & & cl_ops - > dump ( q , cl , skb , tcm ) < 0 )
2008-01-23 09:11:17 +03:00
goto nla_put_failure ;
2005-04-17 02:20:36 +04:00
2008-08-30 01:21:52 +04:00
if ( gnet_stats_start_copy_compat ( skb , TCA_STATS2 , TCA_STATS , TCA_XSTATS ,
qdisc_root_sleeping_lock ( q ) , & d ) < 0 )
2008-01-23 09:11:17 +03:00
goto nla_put_failure ;
2005-04-17 02:20:36 +04:00
if ( cl_ops - > dump_stats & & cl_ops - > dump_stats ( q , cl , & d ) < 0 )
2008-01-23 09:11:17 +03:00
goto nla_put_failure ;
2005-04-17 02:20:36 +04:00
if ( gnet_stats_finish_copy ( & d ) < 0 )
2008-01-23 09:11:17 +03:00
goto nla_put_failure ;
2005-04-17 02:20:36 +04:00
2007-04-20 07:29:13 +04:00
nlh - > nlmsg_len = skb_tail_pointer ( skb ) - b ;
2005-04-17 02:20:36 +04:00
return skb - > len ;
nlmsg_failure :
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 ;
}
static int tclass_notify ( struct sk_buff * oskb , struct nlmsghdr * n ,
struct Qdisc * q , unsigned long cl , int event )
{
struct sk_buff * skb ;
u32 pid = oskb ? NETLINK_CB ( oskb ) . pid : 0 ;
skb = alloc_skb ( NLMSG_GOODSIZE , GFP_KERNEL ) ;
if ( ! skb )
return - ENOBUFS ;
if ( tc_fill_tclass ( skb , q , cl , pid , n - > nlmsg_seq , 0 , event ) < 0 ) {
kfree_skb ( skb ) ;
return - EINVAL ;
}
2007-11-20 09:26:51 +03:00
return rtnetlink_send ( skb , & init_net , pid , RTNLGRP_TC , n - > nlmsg_flags & NLM_F_ECHO ) ;
2005-04-17 02:20:36 +04:00
}
struct qdisc_dump_args
{
struct qdisc_walker w ;
struct sk_buff * skb ;
struct netlink_callback * cb ;
} ;
static int qdisc_class_dump ( struct Qdisc * q , unsigned long cl , struct qdisc_walker * arg )
{
struct qdisc_dump_args * a = ( struct qdisc_dump_args * ) arg ;
return tc_fill_tclass ( a - > skb , q , cl , NETLINK_CB ( a - > cb - > skb ) . pid ,
a - > cb - > nlh - > nlmsg_seq , NLM_F_MULTI , RTM_NEWTCLASS ) ;
}
2008-07-19 09:50:15 +04:00
static int tc_dump_tclass_qdisc ( struct Qdisc * q , struct sk_buff * skb ,
struct tcmsg * tcm , struct netlink_callback * cb ,
int * t_p , int s_t )
{
struct qdisc_dump_args arg ;
if ( tc_qdisc_dump_ignore ( q ) | |
* t_p < s_t | | ! q - > ops - > cl_ops | |
( tcm - > tcm_parent & &
TC_H_MAJ ( tcm - > tcm_parent ) ! = q - > handle ) ) {
( * t_p ) + + ;
return 0 ;
}
if ( * t_p > s_t )
memset ( & cb - > args [ 1 ] , 0 , sizeof ( cb - > args ) - sizeof ( cb - > args [ 0 ] ) ) ;
arg . w . fn = qdisc_class_dump ;
arg . skb = skb ;
arg . cb = cb ;
arg . w . stop = 0 ;
arg . w . skip = cb - > args [ 1 ] ;
arg . w . count = 0 ;
q - > ops - > cl_ops - > walk ( q , & arg . w ) ;
cb - > args [ 1 ] = arg . w . count ;
if ( arg . w . stop )
return - 1 ;
( * t_p ) + + ;
return 0 ;
}
static int tc_dump_tclass_root ( struct Qdisc * root , struct sk_buff * skb ,
struct tcmsg * tcm , struct netlink_callback * cb ,
int * t_p , int s_t )
{
struct Qdisc * q ;
if ( ! root )
return 0 ;
if ( tc_dump_tclass_qdisc ( root , skb , tcm , cb , t_p , s_t ) < 0 )
return - 1 ;
list_for_each_entry ( q , & root - > list , list ) {
if ( tc_dump_tclass_qdisc ( q , skb , tcm , cb , t_p , s_t ) < 0 )
return - 1 ;
}
return 0 ;
}
2005-04-17 02:20:36 +04:00
static int tc_dump_tclass ( struct sk_buff * skb , struct netlink_callback * cb )
{
2008-07-19 09:50:15 +04:00
struct tcmsg * tcm = ( struct tcmsg * ) NLMSG_DATA ( cb - > nlh ) ;
2008-03-25 20:26:21 +03:00
struct net * net = sock_net ( skb - > sk ) ;
2008-07-19 09:50:15 +04:00
struct netdev_queue * dev_queue ;
2005-04-17 02:20:36 +04:00
struct net_device * dev ;
2008-07-19 09:50:15 +04:00
int t , s_t ;
2005-04-17 02:20:36 +04:00
2007-11-30 16:21:31 +03:00
if ( net ! = & init_net )
return 0 ;
2005-04-17 02:20:36 +04:00
if ( cb - > nlh - > nlmsg_len < NLMSG_LENGTH ( sizeof ( * tcm ) ) )
return 0 ;
2007-09-17 22:56:21 +04:00
if ( ( dev = dev_get_by_index ( & init_net , tcm - > tcm_ifindex ) ) = = NULL )
2005-04-17 02:20:36 +04:00
return 0 ;
s_t = cb - > args [ 0 ] ;
t = 0 ;
2008-07-19 09:50:15 +04:00
dev_queue = netdev_get_tx_queue ( dev , 0 ) ;
2008-08-09 10:23:39 +04:00
if ( tc_dump_tclass_root ( dev_queue - > qdisc_sleeping , skb , tcm , cb , & t , s_t ) < 0 )
2008-07-19 09:50:15 +04:00
goto done ;
dev_queue = & dev - > rx_queue ;
2008-08-09 10:23:39 +04:00
if ( tc_dump_tclass_root ( dev_queue - > qdisc_sleeping , skb , tcm , cb , & t , s_t ) < 0 )
2008-07-19 09:50:15 +04:00
goto done ;
2005-04-17 02:20:36 +04:00
2008-07-19 09:50:15 +04:00
done :
2005-04-17 02:20:36 +04:00
cb - > args [ 0 ] = t ;
dev_put ( dev ) ;
return skb - > len ;
}
/* Main classifier routine: scans classifier chain attached
to this qdisc , ( optionally ) tests for protocol and asks
specific classifiers .
*/
2007-07-15 11:02:31 +04:00
int tc_classify_compat ( struct sk_buff * skb , struct tcf_proto * tp ,
struct tcf_result * res )
{
__be16 protocol = skb - > protocol ;
int err = 0 ;
for ( ; tp ; tp = tp - > next ) {
if ( ( tp - > protocol = = protocol | |
tp - > protocol = = htons ( ETH_P_ALL ) ) & &
( err = tp - > classify ( skb , tp , res ) ) > = 0 ) {
# ifdef CONFIG_NET_CLS_ACT
if ( err ! = TC_ACT_RECLASSIFY & & skb - > tc_verd )
skb - > tc_verd = SET_TC_VERD ( skb - > tc_verd , 0 ) ;
# endif
return err ;
}
}
return - 1 ;
}
EXPORT_SYMBOL ( tc_classify_compat ) ;
2005-04-17 02:20:36 +04:00
int tc_classify ( struct sk_buff * skb , struct tcf_proto * tp ,
2007-07-15 11:02:31 +04:00
struct tcf_result * res )
2005-04-17 02:20:36 +04:00
{
int err = 0 ;
2007-07-15 11:02:31 +04:00
__be16 protocol ;
2005-04-17 02:20:36 +04:00
# ifdef CONFIG_NET_CLS_ACT
struct tcf_proto * otp = tp ;
reclassify :
# endif
protocol = skb - > protocol ;
2007-07-15 11:02:31 +04:00
err = tc_classify_compat ( skb , tp , res ) ;
2005-04-17 02:20:36 +04:00
# ifdef CONFIG_NET_CLS_ACT
2007-07-15 11:02:31 +04:00
if ( err = = TC_ACT_RECLASSIFY ) {
u32 verd = G_TC_VERD ( skb - > tc_verd ) ;
tp = otp ;
if ( verd + + > = MAX_REC_LOOP ) {
printk ( " rule prio %u protocol %02x reclassify loop, "
" packet dropped \n " ,
tp - > prio & 0xffff , ntohs ( tp - > protocol ) ) ;
return TC_ACT_SHOT ;
2005-04-17 02:20:36 +04:00
}
2007-07-15 11:02:31 +04:00
skb - > tc_verd = SET_TC_VERD ( skb - > tc_verd , verd ) ;
goto reclassify ;
2005-04-17 02:20:36 +04:00
}
2007-07-15 11:02:31 +04:00
# endif
return err ;
2005-04-17 02:20:36 +04:00
}
2007-07-15 11:02:31 +04:00
EXPORT_SYMBOL ( tc_classify ) ;
2005-04-17 02:20:36 +04:00
2007-03-23 21:29:43 +03:00
void tcf_destroy ( struct tcf_proto * tp )
{
tp - > ops - > destroy ( tp ) ;
module_put ( tp - > ops - > owner ) ;
kfree ( tp ) ;
}
2008-07-02 06:52:38 +04:00
void tcf_destroy_chain ( struct tcf_proto * * fl )
2007-03-23 21:29:43 +03:00
{
struct tcf_proto * tp ;
2008-07-02 06:52:38 +04:00
while ( ( tp = * fl ) ! = NULL ) {
* fl = tp - > next ;
2007-03-23 21:29:43 +03:00
tcf_destroy ( tp ) ;
}
}
EXPORT_SYMBOL ( tcf_destroy_chain ) ;
2005-04-17 02:20:36 +04:00
# ifdef CONFIG_PROC_FS
static int psched_show ( struct seq_file * seq , void * v )
{
2007-10-11 03:32:41 +04:00
struct timespec ts ;
hrtimer_get_res ( CLOCK_MONOTONIC , & ts ) ;
2005-04-17 02:20:36 +04:00
seq_printf ( seq , " %08x %08x %08x %08x \n " ,
2007-03-16 11:18:42 +03:00
( u32 ) NSEC_PER_USEC , ( u32 ) PSCHED_US2NS ( 1 ) ,
2007-03-16 22:34:52 +03:00
1000000 ,
2007-10-11 03:32:41 +04:00
( u32 ) NSEC_PER_SEC / ( u32 ) ktime_to_ns ( timespec_to_ktime ( ts ) ) ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
static int psched_open ( struct inode * inode , struct file * file )
{
return single_open ( file , psched_show , PDE ( inode ) - > data ) ;
}
2007-02-12 11:55:36 +03:00
static const struct file_operations psched_fops = {
2005-04-17 02:20:36 +04:00
. owner = THIS_MODULE ,
. open = psched_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
2007-02-09 17:25:16 +03:00
} ;
2005-04-17 02:20:36 +04:00
# endif
static int __init pktsched_init ( void )
{
register_qdisc ( & pfifo_qdisc_ops ) ;
register_qdisc ( & bfifo_qdisc_ops ) ;
2007-09-12 14:01:34 +04:00
proc_net_fops_create ( & init_net , " psched " , 0 , & psched_fops ) ;
2005-04-17 02:20:36 +04:00
2007-03-22 21:55:50 +03:00
rtnl_register ( PF_UNSPEC , RTM_NEWQDISC , tc_modify_qdisc , NULL ) ;
rtnl_register ( PF_UNSPEC , RTM_DELQDISC , tc_get_qdisc , NULL ) ;
rtnl_register ( PF_UNSPEC , RTM_GETQDISC , tc_get_qdisc , tc_dump_qdisc ) ;
rtnl_register ( PF_UNSPEC , RTM_NEWTCLASS , tc_ctl_tclass , NULL ) ;
rtnl_register ( PF_UNSPEC , RTM_DELTCLASS , tc_ctl_tclass , NULL ) ;
rtnl_register ( PF_UNSPEC , RTM_GETTCLASS , tc_ctl_tclass , tc_dump_tclass ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
subsys_initcall ( pktsched_init ) ;