2005-04-16 15:20:36 -07:00
/* net/sched/sch_dsmark.c - Differentiated Services field marker */
/* Written 1998-2000 by Werner Almesberger, EPFL ICA */
# include <linux/module.h>
# include <linux/init.h>
# include <linux/types.h>
# include <linux/string.h>
# include <linux/errno.h>
# include <linux/skbuff.h>
# include <linux/rtnetlink.h>
# include <net/pkt_sched.h>
# include <net/dsfield.h>
# include <net/inet_ecn.h>
# include <asm/byteorder.h>
2005-05-31 15:17:28 -07:00
#if 0 /* control */
2005-04-16 15:20:36 -07:00
# define DPRINTK(format,args...) printk(KERN_DEBUG format,##args)
# else
# define DPRINTK(format,args...)
# endif
#if 0 /* data */
# define D2PRINTK(format,args...) printk(KERN_DEBUG format,##args)
# else
# define D2PRINTK(format,args...)
# endif
2005-06-18 22:53:29 -07:00
# define PRIV(sch) ((struct dsmark_qdisc_data *) qdisc_priv(sch))
2005-04-16 15:20:36 -07:00
/*
* classid class marking
* - - - - - - - - - - - - - - - - - - -
* n / a 0 n / a
* x : 0 1 use entry [ 0 ]
* . . . . . . . . .
* x : y y > 0 y + 1 use entry [ y ]
* . . . . . . . . .
* x : indices - 1 indices use entry [ indices - 1 ]
* . . . . . . . . .
* x : y y + 1 use entry [ y & ( indices - 1 ) ]
* . . . . . . . . .
* 0xffff 0x10000 use entry [ indices - 1 ]
*/
# define NO_DEFAULT_INDEX (1 << 16)
struct dsmark_qdisc_data {
struct Qdisc * q ;
struct tcf_proto * filter_list ;
2005-06-18 22:53:29 -07:00
u8 * mask ; /* "owns" the array */
u8 * value ;
u16 indices ;
u32 default_index ; /* index range is 0...0xffff */
2005-04-16 15:20:36 -07:00
int set_tc_index ;
} ;
2005-06-18 22:52:54 -07:00
static inline int dsmark_valid_indices ( u16 indices )
{
while ( indices ! = 1 ) {
if ( indices & 1 )
return 0 ;
indices > > = 1 ;
}
2007-02-09 23:25:16 +09:00
2005-06-18 22:52:54 -07:00
return 1 ;
}
static inline int dsmark_valid_index ( struct dsmark_qdisc_data * p , u16 index )
{
return ( index < = p - > indices & & index > 0 ) ;
}
2005-04-16 15:20:36 -07:00
/* ------------------------- Class/flow operations ------------------------- */
2005-06-18 22:53:29 -07:00
static int dsmark_graft ( struct Qdisc * sch , unsigned long arg ,
struct Qdisc * new , struct Qdisc * * old )
2005-04-16 15:20:36 -07:00
{
struct dsmark_qdisc_data * p = PRIV ( sch ) ;
2005-06-18 22:53:29 -07:00
DPRINTK ( " dsmark_graft(sch %p,[qdisc %p],new %p,old %p) \n " ,
sch , p , new , old ) ;
2005-05-31 15:16:52 -07:00
if ( new = = NULL ) {
2006-11-29 17:35:18 -08:00
new = qdisc_create_dflt ( sch - > dev , & pfifo_qdisc_ops ,
sch - > handle ) ;
2005-05-31 15:16:52 -07:00
if ( new = = NULL )
new = & noop_qdisc ;
}
2005-04-16 15:20:36 -07:00
sch_tree_lock ( sch ) ;
2005-06-18 22:53:29 -07:00
* old = xchg ( & p - > q , new ) ;
2006-11-29 17:36:20 -08:00
qdisc_tree_decrease_qlen ( * old , ( * old ) - > q . qlen ) ;
2005-06-18 22:53:29 -07:00
qdisc_reset ( * old ) ;
sch_tree_unlock ( sch ) ;
2007-02-09 23:25:16 +09:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
static struct Qdisc * dsmark_leaf ( struct Qdisc * sch , unsigned long arg )
{
2005-06-18 22:53:29 -07:00
return PRIV ( sch ) - > q ;
2005-04-16 15:20:36 -07:00
}
2005-06-18 22:53:29 -07:00
static unsigned long dsmark_get ( struct Qdisc * sch , u32 classid )
2005-04-16 15:20:36 -07:00
{
2005-06-18 22:53:29 -07:00
DPRINTK ( " dsmark_get(sch %p,[qdisc %p],classid %x) \n " ,
sch , PRIV ( sch ) , classid ) ;
2005-04-16 15:20:36 -07:00
2005-06-18 22:53:29 -07:00
return TC_H_MIN ( classid ) + 1 ;
2005-04-16 15:20:36 -07:00
}
static unsigned long dsmark_bind_filter ( struct Qdisc * sch ,
2005-06-18 22:53:29 -07:00
unsigned long parent , u32 classid )
2005-04-16 15:20:36 -07:00
{
2005-06-18 22:53:29 -07:00
return dsmark_get ( sch , classid ) ;
2005-04-16 15:20:36 -07:00
}
static void dsmark_put ( struct Qdisc * sch , unsigned long cl )
{
}
static int dsmark_change ( struct Qdisc * sch , u32 classid , u32 parent ,
2005-06-18 22:52:54 -07:00
struct rtattr * * tca , unsigned long * arg )
2005-04-16 15:20:36 -07:00
{
struct dsmark_qdisc_data * p = PRIV ( sch ) ;
struct rtattr * opt = tca [ TCA_OPTIONS - 1 ] ;
struct rtattr * tb [ TCA_DSMARK_MAX ] ;
2005-06-18 22:52:54 -07:00
int err = - EINVAL ;
u8 mask = 0 ;
2005-04-16 15:20:36 -07:00
DPRINTK ( " dsmark_change(sch %p,[qdisc %p],classid %x,parent %x), "
2005-06-18 22:52:54 -07:00
" arg 0x%lx \n " , sch , p , classid , parent , * arg ) ;
if ( ! dsmark_valid_index ( p , * arg ) ) {
err = - ENOENT ;
goto rtattr_failure ;
2005-04-16 15:20:36 -07:00
}
2005-06-18 22:52:54 -07:00
if ( ! opt | | rtattr_parse_nested ( tb , TCA_DSMARK_MAX , opt ) )
goto rtattr_failure ;
if ( tb [ TCA_DSMARK_MASK - 1 ] )
mask = RTA_GET_U8 ( tb [ TCA_DSMARK_MASK - 1 ] ) ;
if ( tb [ TCA_DSMARK_VALUE - 1 ] )
p - > value [ * arg - 1 ] = RTA_GET_U8 ( tb [ TCA_DSMARK_VALUE - 1 ] ) ;
2007-02-09 23:25:16 +09:00
2005-06-18 22:52:54 -07:00
if ( tb [ TCA_DSMARK_MASK - 1 ] )
p - > mask [ * arg - 1 ] = mask ;
err = 0 ;
rtattr_failure :
return err ;
}
2005-04-16 15:20:36 -07:00
2005-06-18 22:53:29 -07:00
static int dsmark_delete ( struct Qdisc * sch , unsigned long arg )
2005-04-16 15:20:36 -07:00
{
struct dsmark_qdisc_data * p = PRIV ( sch ) ;
2005-06-18 22:53:29 -07:00
if ( ! dsmark_valid_index ( p , arg ) )
2005-04-16 15:20:36 -07:00
return - EINVAL ;
2007-02-09 23:25:16 +09:00
2005-04-16 15:20:36 -07:00
p - > mask [ arg - 1 ] = 0xff ;
p - > value [ arg - 1 ] = 0 ;
2005-06-18 22:53:29 -07:00
2005-04-16 15:20:36 -07:00
return 0 ;
}
static void dsmark_walk ( struct Qdisc * sch , struct qdisc_walker * walker )
{
struct dsmark_qdisc_data * p = PRIV ( sch ) ;
int i ;
2005-06-18 22:53:29 -07:00
DPRINTK ( " dsmark_walk(sch %p,[qdisc %p],walker %p) \n " , sch , p , walker ) ;
2005-04-16 15:20:36 -07:00
if ( walker - > stop )
return ;
2005-06-18 22:53:29 -07:00
2005-04-16 15:20:36 -07:00
for ( i = 0 ; i < p - > indices ; i + + ) {
if ( p - > mask [ i ] = = 0xff & & ! p - > value [ i ] )
2005-05-31 15:15:58 -07:00
goto ignore ;
2005-04-16 15:20:36 -07:00
if ( walker - > count > = walker - > skip ) {
if ( walker - > fn ( sch , i + 1 , walker ) < 0 ) {
walker - > stop = 1 ;
break ;
}
}
2007-02-09 23:25:16 +09:00
ignore :
2005-05-31 15:15:58 -07:00
walker - > count + + ;
2007-02-09 23:25:16 +09:00
}
2005-04-16 15:20:36 -07:00
}
static struct tcf_proto * * dsmark_find_tcf ( struct Qdisc * sch , unsigned long cl )
{
2005-06-18 22:53:29 -07:00
return & PRIV ( sch ) - > filter_list ;
2005-04-16 15:20:36 -07:00
}
/* --------------------------- Qdisc operations ---------------------------- */
static int dsmark_enqueue ( struct sk_buff * skb , struct Qdisc * sch )
{
struct dsmark_qdisc_data * p = PRIV ( sch ) ;
2005-06-18 22:53:29 -07:00
int err ;
D2PRINTK ( " dsmark_enqueue(skb %p,sch %p,[qdisc %p]) \n " , skb , sch , p ) ;
2005-04-16 15:20:36 -07:00
if ( p - > set_tc_index ) {
/* FIXME: Safe with non-linear skbs? --RR */
switch ( skb - > protocol ) {
case __constant_htons ( ETH_P_IP ) :
2007-04-20 22:47:35 -07:00
skb - > tc_index = ipv4_get_dsfield ( ip_hdr ( skb ) )
2005-04-16 15:20:36 -07:00
& ~ INET_ECN_MASK ;
break ;
case __constant_htons ( ETH_P_IPV6 ) :
2007-04-25 17:54:47 -07:00
skb - > tc_index = ipv6_get_dsfield ( ipv6_hdr ( skb ) )
2005-04-16 15:20:36 -07:00
& ~ INET_ECN_MASK ;
break ;
default :
skb - > tc_index = 0 ;
break ;
2007-04-20 17:09:22 -07:00
}
2005-04-16 15:20:36 -07:00
}
2005-06-18 22:53:29 -07:00
if ( TC_H_MAJ ( skb - > priority ) = = sch - > handle )
2005-04-16 15:20:36 -07:00
skb - > tc_index = TC_H_MIN ( skb - > priority ) ;
2005-06-18 22:53:29 -07:00
else {
struct tcf_result res ;
int result = tc_classify ( skb , p - > filter_list , & res ) ;
D2PRINTK ( " result %d class 0x%04x \n " , result , res . classid ) ;
2005-04-16 15:20:36 -07:00
switch ( result ) {
2007-07-15 00:02:10 -07:00
# ifdef CONFIG_NET_CLS_ACT
case TC_ACT_QUEUED :
case TC_ACT_STOLEN :
kfree_skb ( skb ) ;
return NET_XMIT_SUCCESS ;
case TC_ACT_SHOT :
kfree_skb ( skb ) ;
sch - > qstats . drops + + ;
return NET_XMIT_BYPASS ;
2005-04-16 15:20:36 -07:00
# endif
2007-07-15 00:03:05 -07:00
case TC_ACT_OK :
2007-07-15 00:02:10 -07:00
skb - > tc_index = TC_H_MIN ( res . classid ) ;
break ;
default :
if ( p - > default_index ! = NO_DEFAULT_INDEX )
skb - > tc_index = p - > default_index ;
break ;
2007-04-20 17:09:22 -07:00
}
2005-04-16 15:20:36 -07:00
}
2005-06-18 22:53:29 -07:00
err = p - > q - > enqueue ( skb , p - > q ) ;
if ( err ! = NET_XMIT_SUCCESS ) {
2005-04-16 15:20:36 -07:00
sch - > qstats . drops + + ;
2005-06-18 22:53:29 -07:00
return err ;
2005-04-16 15:20:36 -07:00
}
2005-06-18 22:53:29 -07:00
2005-04-16 15:20:36 -07:00
sch - > bstats . bytes + = skb - > len ;
sch - > bstats . packets + + ;
sch - > q . qlen + + ;
2005-06-18 22:53:29 -07:00
return NET_XMIT_SUCCESS ;
}
2005-04-16 15:20:36 -07:00
static struct sk_buff * dsmark_dequeue ( struct Qdisc * sch )
{
struct dsmark_qdisc_data * p = PRIV ( sch ) ;
struct sk_buff * skb ;
2005-06-18 22:53:29 -07:00
u32 index ;
D2PRINTK ( " dsmark_dequeue(sch %p,[qdisc %p]) \n " , sch , p ) ;
2005-04-16 15:20:36 -07:00
skb = p - > q - > ops - > dequeue ( p - > q ) ;
2005-06-18 22:53:29 -07:00
if ( skb = = NULL )
2005-04-16 15:20:36 -07:00
return NULL ;
2005-06-18 22:53:29 -07:00
2005-04-16 15:20:36 -07:00
sch - > q . qlen - - ;
2005-06-18 22:53:29 -07:00
index = skb - > tc_index & ( p - > indices - 1 ) ;
D2PRINTK ( " index %d->%d \n " , skb - > tc_index , index ) ;
2005-04-16 15:20:36 -07:00
switch ( skb - > protocol ) {
case __constant_htons ( ETH_P_IP ) :
2007-04-20 22:47:35 -07:00
ipv4_change_dsfield ( ip_hdr ( skb ) , p - > mask [ index ] ,
2005-06-18 22:53:29 -07:00
p - > value [ index ] ) ;
2005-04-16 15:20:36 -07:00
break ;
case __constant_htons ( ETH_P_IPV6 ) :
2007-04-25 17:54:47 -07:00
ipv6_change_dsfield ( ipv6_hdr ( skb ) , p - > mask [ index ] ,
2005-06-18 22:53:29 -07:00
p - > value [ index ] ) ;
2005-04-16 15:20:36 -07:00
break ;
default :
/*
* Only complain if a change was actually attempted .
* This way , we can send non - IP traffic through dsmark
* and don ' t need yet another qdisc as a bypass .
*/
if ( p - > mask [ index ] ! = 0xff | | p - > value [ index ] )
printk ( KERN_WARNING " dsmark_dequeue: "
" unsupported protocol %d \n " ,
2006-11-20 18:07:51 -08:00
ntohs ( skb - > protocol ) ) ;
2005-04-16 15:20:36 -07:00
break ;
2007-04-20 17:09:22 -07:00
}
2005-06-18 22:53:29 -07:00
2005-04-16 15:20:36 -07:00
return skb ;
}
static int dsmark_requeue ( struct sk_buff * skb , struct Qdisc * sch )
{
struct dsmark_qdisc_data * p = PRIV ( sch ) ;
2005-06-18 22:53:29 -07:00
int err ;
2005-04-16 15:20:36 -07:00
2005-06-18 22:53:29 -07:00
D2PRINTK ( " dsmark_requeue(skb %p,sch %p,[qdisc %p]) \n " , skb , sch , p ) ;
err = p - > q - > ops - > requeue ( skb , p - > q ) ;
if ( err ! = NET_XMIT_SUCCESS ) {
sch - > qstats . drops + + ;
return err ;
2005-04-16 15:20:36 -07:00
}
2005-06-18 22:53:29 -07:00
sch - > q . qlen + + ;
sch - > qstats . requeues + + ;
return NET_XMIT_SUCCESS ;
}
2005-04-16 15:20:36 -07:00
static unsigned int dsmark_drop ( struct Qdisc * sch )
{
struct dsmark_qdisc_data * p = PRIV ( sch ) ;
unsigned int len ;
2007-02-09 23:25:16 +09:00
2005-06-18 22:53:29 -07:00
DPRINTK ( " dsmark_reset(sch %p,[qdisc %p]) \n " , sch , p ) ;
if ( p - > q - > ops - > drop = = NULL )
2005-04-16 15:20:36 -07:00
return 0 ;
2005-06-18 22:53:29 -07:00
len = p - > q - > ops - > drop ( p - > q ) ;
if ( len )
sch - > q . qlen - - ;
2005-04-16 15:20:36 -07:00
return len ;
}
2005-06-18 22:52:54 -07:00
static int dsmark_init ( struct Qdisc * sch , struct rtattr * opt )
2005-04-16 15:20:36 -07:00
{
struct dsmark_qdisc_data * p = PRIV ( sch ) ;
struct rtattr * tb [ TCA_DSMARK_MAX ] ;
2005-06-18 22:52:54 -07:00
int err = - EINVAL ;
u32 default_index = NO_DEFAULT_INDEX ;
u16 indices ;
u8 * mask ;
DPRINTK ( " dsmark_init(sch %p,[qdisc %p],opt %p) \n " , sch , p , opt ) ;
if ( ! opt | | rtattr_parse_nested ( tb , TCA_DSMARK_MAX , opt ) < 0 )
goto errout ;
indices = RTA_GET_U16 ( tb [ TCA_DSMARK_INDICES - 1 ] ) ;
if ( ! indices | | ! dsmark_valid_indices ( indices ) )
goto errout ;
if ( tb [ TCA_DSMARK_DEFAULT_INDEX - 1 ] )
default_index = RTA_GET_U16 ( tb [ TCA_DSMARK_DEFAULT_INDEX - 1 ] ) ;
mask = kmalloc ( indices * 2 , GFP_KERNEL ) ;
if ( mask = = NULL ) {
err = - ENOMEM ;
goto errout ;
2005-04-16 15:20:36 -07:00
}
2005-06-18 22:52:54 -07:00
p - > mask = mask ;
memset ( p - > mask , 0xff , indices ) ;
p - > value = p - > mask + indices ;
memset ( p - > value , 0 , indices ) ;
p - > indices = indices ;
p - > default_index = default_index ;
p - > set_tc_index = RTA_GET_FLAG ( tb [ TCA_DSMARK_SET_TC_INDEX - 1 ] ) ;
2006-11-29 17:35:18 -08:00
p - > q = qdisc_create_dflt ( sch - > dev , & pfifo_qdisc_ops , sch - > handle ) ;
2005-06-18 22:52:54 -07:00
if ( p - > q = = NULL )
2005-04-16 15:20:36 -07:00
p - > q = & noop_qdisc ;
2005-06-18 22:52:54 -07:00
DPRINTK ( " dsmark_init: qdisc %p \n " , p - > q ) ;
err = 0 ;
errout :
rtattr_failure :
return err ;
2005-04-16 15:20:36 -07:00
}
static void dsmark_reset ( struct Qdisc * sch )
{
struct dsmark_qdisc_data * p = PRIV ( sch ) ;
2005-06-18 22:53:29 -07:00
DPRINTK ( " dsmark_reset(sch %p,[qdisc %p]) \n " , sch , p ) ;
2005-04-16 15:20:36 -07:00
qdisc_reset ( p - > q ) ;
sch - > q . qlen = 0 ;
}
static void dsmark_destroy ( struct Qdisc * sch )
{
struct dsmark_qdisc_data * p = PRIV ( sch ) ;
2005-06-18 22:53:29 -07:00
DPRINTK ( " dsmark_destroy(sch %p,[qdisc %p]) \n " , sch , p ) ;
2007-03-23 11:29:43 -07:00
tcf_destroy_chain ( p - > filter_list ) ;
2005-04-16 15:20:36 -07:00
qdisc_destroy ( p - > q ) ;
kfree ( p - > mask ) ;
}
static int dsmark_dump_class ( struct Qdisc * sch , unsigned long cl ,
2005-06-18 22:53:12 -07:00
struct sk_buff * skb , struct tcmsg * tcm )
2005-04-16 15:20:36 -07:00
{
struct dsmark_qdisc_data * p = PRIV ( sch ) ;
2005-06-18 22:53:12 -07:00
struct rtattr * opts = NULL ;
2005-04-16 15:20:36 -07:00
2005-06-18 22:53:12 -07:00
DPRINTK ( " dsmark_dump_class(sch %p,[qdisc %p],class %ld \n " , sch , p , cl ) ;
if ( ! dsmark_valid_index ( p , cl ) )
2005-04-16 15:20:36 -07:00
return - EINVAL ;
2005-06-18 22:53:12 -07:00
tcm - > tcm_handle = TC_H_MAKE ( TC_H_MAJ ( sch - > handle ) , cl - 1 ) ;
2006-03-20 19:01:06 -08:00
tcm - > tcm_info = p - > q - > handle ;
2005-06-18 22:53:12 -07:00
opts = RTA_NEST ( skb , TCA_OPTIONS ) ;
RTA_PUT_U8 ( skb , TCA_DSMARK_MASK , p - > mask [ cl - 1 ] ) ;
RTA_PUT_U8 ( skb , TCA_DSMARK_VALUE , p - > value [ cl - 1 ] ) ;
return RTA_NEST_END ( skb , opts ) ;
2005-04-16 15:20:36 -07:00
rtattr_failure :
2005-06-18 22:53:12 -07:00
return RTA_NEST_CANCEL ( skb , opts ) ;
2005-04-16 15:20:36 -07:00
}
static int dsmark_dump ( struct Qdisc * sch , struct sk_buff * skb )
{
struct dsmark_qdisc_data * p = PRIV ( sch ) ;
2005-06-18 22:53:12 -07:00
struct rtattr * opts = NULL ;
2005-04-16 15:20:36 -07:00
2005-06-18 22:53:12 -07:00
opts = RTA_NEST ( skb , TCA_OPTIONS ) ;
RTA_PUT_U16 ( skb , TCA_DSMARK_INDICES , p - > indices ) ;
if ( p - > default_index ! = NO_DEFAULT_INDEX )
RTA_PUT_U16 ( skb , TCA_DSMARK_DEFAULT_INDEX , p - > default_index ) ;
2005-04-16 15:20:36 -07:00
if ( p - > set_tc_index )
2005-06-18 22:53:12 -07:00
RTA_PUT_FLAG ( skb , TCA_DSMARK_SET_TC_INDEX ) ;
return RTA_NEST_END ( skb , opts ) ;
2005-04-16 15:20:36 -07:00
rtattr_failure :
2005-06-18 22:53:12 -07:00
return RTA_NEST_CANCEL ( skb , opts ) ;
2005-04-16 15:20:36 -07:00
}
static struct Qdisc_class_ops dsmark_class_ops = {
. graft = dsmark_graft ,
. leaf = dsmark_leaf ,
. get = dsmark_get ,
. put = dsmark_put ,
. change = dsmark_change ,
. delete = dsmark_delete ,
. walk = dsmark_walk ,
. tcf_chain = dsmark_find_tcf ,
. bind_tcf = dsmark_bind_filter ,
. unbind_tcf = dsmark_put ,
. dump = dsmark_dump_class ,
} ;
static struct Qdisc_ops dsmark_qdisc_ops = {
. next = NULL ,
. cl_ops = & dsmark_class_ops ,
. id = " dsmark " ,
. priv_size = sizeof ( struct dsmark_qdisc_data ) ,
. enqueue = dsmark_enqueue ,
. dequeue = dsmark_dequeue ,
. requeue = dsmark_requeue ,
. drop = dsmark_drop ,
. init = dsmark_init ,
. reset = dsmark_reset ,
. destroy = dsmark_destroy ,
. change = NULL ,
. dump = dsmark_dump ,
. owner = THIS_MODULE ,
} ;
static int __init dsmark_module_init ( void )
{
return register_qdisc ( & dsmark_qdisc_ops ) ;
}
2005-06-18 22:53:29 -07:00
2007-02-09 23:25:16 +09:00
static void __exit dsmark_module_exit ( void )
2005-04-16 15:20:36 -07:00
{
unregister_qdisc ( & dsmark_qdisc_ops ) ;
}
2005-06-18 22:53:29 -07:00
2005-04-16 15:20:36 -07:00
module_init ( dsmark_module_init )
module_exit ( dsmark_module_exit )
2005-06-18 22:53:29 -07:00
2005-04-16 15:20:36 -07:00
MODULE_LICENSE ( " GPL " ) ;