2005-04-17 02:20:36 +04:00
# ifndef __NET_SCHED_GENERIC_H
# define __NET_SCHED_GENERIC_H
# include <linux/netdevice.h>
# include <linux/types.h>
# include <linux/rcupdate.h>
# include <linux/module.h>
# include <linux/pkt_sched.h>
# include <linux/pkt_cls.h>
# include <net/gen_stats.h>
2007-03-22 21:55:50 +03:00
# include <net/rtnetlink.h>
2005-04-17 02:20:36 +04:00
struct Qdisc_ops ;
struct qdisc_walker ;
struct tcf_walker ;
struct module ;
struct qdisc_rate_table
{
struct tc_ratespec rate ;
u32 data [ 256 ] ;
struct qdisc_rate_table * next ;
int refcnt ;
} ;
struct Qdisc
{
int ( * enqueue ) ( struct sk_buff * skb , struct Qdisc * dev ) ;
struct sk_buff * ( * dequeue ) ( struct Qdisc * dev ) ;
unsigned flags ;
# define TCQ_F_BUILTIN 1
# define TCQ_F_THROTTLED 2
# define TCQ_F_INGRESS 4
int padded ;
struct Qdisc_ops * ops ;
u32 handle ;
u32 parent ;
atomic_t refcnt ;
struct sk_buff_head q ;
struct net_device * dev ;
struct list_head list ;
struct gnet_stats_basic bstats ;
struct gnet_stats_queue qstats ;
struct gnet_stats_rate_est rate_est ;
spinlock_t * stats_lock ;
struct rcu_head q_rcu ;
int ( * reshape_fail ) ( struct sk_buff * skb ,
struct Qdisc * q ) ;
/* This field is deprecated, but it is still used by CBQ
* and it will live until better solution will be invented .
*/
struct Qdisc * __parent ;
} ;
struct Qdisc_class_ops
{
/* Child qdisc manipulation */
int ( * graft ) ( struct Qdisc * , unsigned long cl ,
struct Qdisc * , struct Qdisc * * ) ;
struct Qdisc * ( * leaf ) ( struct Qdisc * , unsigned long cl ) ;
2006-11-30 04:35:48 +03:00
void ( * qlen_notify ) ( struct Qdisc * , unsigned long ) ;
2005-04-17 02:20:36 +04:00
/* Class manipulation routines */
unsigned long ( * get ) ( struct Qdisc * , u32 classid ) ;
void ( * put ) ( struct Qdisc * , unsigned long ) ;
int ( * change ) ( struct Qdisc * , u32 , u32 ,
2008-01-23 09:11:17 +03:00
struct nlattr * * , unsigned long * ) ;
2005-04-17 02:20:36 +04:00
int ( * delete ) ( struct Qdisc * , unsigned long ) ;
void ( * walk ) ( struct Qdisc * , struct qdisc_walker * arg ) ;
/* Filter manipulation */
struct tcf_proto * * ( * tcf_chain ) ( struct Qdisc * , unsigned long ) ;
unsigned long ( * bind_tcf ) ( struct Qdisc * , unsigned long ,
u32 classid ) ;
void ( * unbind_tcf ) ( struct Qdisc * , unsigned long ) ;
/* rtnetlink specific */
int ( * dump ) ( struct Qdisc * , unsigned long ,
struct sk_buff * skb , struct tcmsg * ) ;
int ( * dump_stats ) ( struct Qdisc * , unsigned long ,
struct gnet_dump * ) ;
} ;
struct Qdisc_ops
{
struct Qdisc_ops * next ;
2007-11-14 12:44:41 +03:00
const struct Qdisc_class_ops * cl_ops ;
2005-04-17 02:20:36 +04:00
char id [ IFNAMSIZ ] ;
int priv_size ;
int ( * enqueue ) ( struct sk_buff * , struct Qdisc * ) ;
struct sk_buff * ( * dequeue ) ( struct Qdisc * ) ;
int ( * requeue ) ( struct sk_buff * , struct Qdisc * ) ;
unsigned int ( * drop ) ( struct Qdisc * ) ;
2008-01-23 09:11:17 +03:00
int ( * init ) ( struct Qdisc * , struct nlattr * arg ) ;
2005-04-17 02:20:36 +04:00
void ( * reset ) ( struct Qdisc * ) ;
void ( * destroy ) ( struct Qdisc * ) ;
2008-01-23 09:11:17 +03:00
int ( * change ) ( struct Qdisc * , struct nlattr * arg ) ;
2005-04-17 02:20:36 +04:00
int ( * dump ) ( struct Qdisc * , struct sk_buff * ) ;
int ( * dump_stats ) ( struct Qdisc * , struct gnet_dump * ) ;
struct module * owner ;
} ;
struct tcf_result
{
unsigned long class ;
u32 classid ;
} ;
struct tcf_proto_ops
{
struct tcf_proto_ops * next ;
char kind [ IFNAMSIZ ] ;
int ( * classify ) ( struct sk_buff * , struct tcf_proto * ,
struct tcf_result * ) ;
int ( * init ) ( struct tcf_proto * ) ;
void ( * destroy ) ( struct tcf_proto * ) ;
unsigned long ( * get ) ( struct tcf_proto * , u32 handle ) ;
void ( * put ) ( struct tcf_proto * , unsigned long ) ;
int ( * change ) ( struct tcf_proto * , unsigned long ,
2008-01-23 09:11:33 +03:00
u32 handle , struct nlattr * * ,
2005-04-17 02:20:36 +04:00
unsigned long * ) ;
int ( * delete ) ( struct tcf_proto * , unsigned long ) ;
void ( * walk ) ( struct tcf_proto * , struct tcf_walker * arg ) ;
/* rtnetlink specific */
int ( * dump ) ( struct tcf_proto * , unsigned long ,
struct sk_buff * skb , struct tcmsg * ) ;
struct module * owner ;
} ;
struct tcf_proto
{
/* Fast access part */
struct tcf_proto * next ;
void * root ;
int ( * classify ) ( struct sk_buff * , struct tcf_proto * ,
struct tcf_result * ) ;
2006-11-21 05:07:51 +03:00
__be16 protocol ;
2005-04-17 02:20:36 +04:00
/* All the rest */
u32 prio ;
u32 classid ;
struct Qdisc * q ;
void * data ;
struct tcf_proto_ops * ops ;
} ;
extern void qdisc_lock_tree ( struct net_device * dev ) ;
extern void qdisc_unlock_tree ( struct net_device * dev ) ;
# define sch_tree_lock(q) qdisc_lock_tree((q)->dev)
# define sch_tree_unlock(q) qdisc_unlock_tree((q)->dev)
# define tcf_tree_lock(tp) qdisc_lock_tree((tp)->q->dev)
# define tcf_tree_unlock(tp) qdisc_unlock_tree((tp)->q->dev)
2005-07-06 01:14:30 +04:00
extern struct Qdisc noop_qdisc ;
extern struct Qdisc_ops noop_qdisc_ops ;
extern void dev_init_scheduler ( struct net_device * dev ) ;
extern void dev_shutdown ( struct net_device * dev ) ;
extern void dev_activate ( struct net_device * dev ) ;
extern void dev_deactivate ( struct net_device * dev ) ;
extern void qdisc_reset ( struct Qdisc * qdisc ) ;
extern void qdisc_destroy ( struct Qdisc * qdisc ) ;
2006-11-30 04:35:48 +03:00
extern void qdisc_tree_decrease_qlen ( struct Qdisc * qdisc , unsigned int n ) ;
2005-07-06 01:15:09 +04:00
extern struct Qdisc * qdisc_alloc ( struct net_device * dev , struct Qdisc_ops * ops ) ;
2005-07-06 01:14:30 +04:00
extern struct Qdisc * qdisc_create_dflt ( struct net_device * dev ,
2006-11-30 04:35:18 +03:00
struct Qdisc_ops * ops , u32 parentid ) ;
2007-03-23 21:29:43 +03:00
extern void tcf_destroy ( struct tcf_proto * tp ) ;
extern void tcf_destroy_chain ( struct tcf_proto * fl ) ;
2005-04-17 02:20:36 +04:00
2005-06-19 09:57:26 +04:00
static inline int __qdisc_enqueue_tail ( struct sk_buff * skb , struct Qdisc * sch ,
struct sk_buff_head * list )
{
__skb_queue_tail ( list , skb ) ;
sch - > qstats . backlog + = skb - > len ;
sch - > bstats . bytes + = skb - > len ;
sch - > bstats . packets + + ;
return NET_XMIT_SUCCESS ;
}
static inline int qdisc_enqueue_tail ( struct sk_buff * skb , struct Qdisc * sch )
{
return __qdisc_enqueue_tail ( skb , sch , & sch - > q ) ;
}
static inline struct sk_buff * __qdisc_dequeue_head ( struct Qdisc * sch ,
struct sk_buff_head * list )
{
struct sk_buff * skb = __skb_dequeue ( list ) ;
if ( likely ( skb ! = NULL ) )
sch - > qstats . backlog - = skb - > len ;
return skb ;
}
static inline struct sk_buff * qdisc_dequeue_head ( struct Qdisc * sch )
{
return __qdisc_dequeue_head ( sch , & sch - > q ) ;
}
static inline struct sk_buff * __qdisc_dequeue_tail ( struct Qdisc * sch ,
struct sk_buff_head * list )
{
struct sk_buff * skb = __skb_dequeue_tail ( list ) ;
if ( likely ( skb ! = NULL ) )
sch - > qstats . backlog - = skb - > len ;
return skb ;
}
static inline struct sk_buff * qdisc_dequeue_tail ( struct Qdisc * sch )
{
return __qdisc_dequeue_tail ( sch , & sch - > q ) ;
}
static inline int __qdisc_requeue ( struct sk_buff * skb , struct Qdisc * sch ,
struct sk_buff_head * list )
{
__skb_queue_head ( list , skb ) ;
sch - > qstats . backlog + = skb - > len ;
sch - > qstats . requeues + + ;
return NET_XMIT_SUCCESS ;
}
static inline int qdisc_requeue ( struct sk_buff * skb , struct Qdisc * sch )
{
return __qdisc_requeue ( skb , sch , & sch - > q ) ;
}
static inline void __qdisc_reset_queue ( struct Qdisc * sch ,
struct sk_buff_head * list )
{
/*
* We do not know the backlog in bytes of this list , it
* is up to the caller to correct it
*/
skb_queue_purge ( list ) ;
}
static inline void qdisc_reset_queue ( struct Qdisc * sch )
{
__qdisc_reset_queue ( sch , & sch - > q ) ;
sch - > qstats . backlog = 0 ;
}
static inline unsigned int __qdisc_queue_drop ( struct Qdisc * sch ,
struct sk_buff_head * list )
{
struct sk_buff * skb = __qdisc_dequeue_tail ( sch , list ) ;
if ( likely ( skb ! = NULL ) ) {
unsigned int len = skb - > len ;
kfree_skb ( skb ) ;
return len ;
}
return 0 ;
}
static inline unsigned int qdisc_queue_drop ( struct Qdisc * sch )
{
return __qdisc_queue_drop ( sch , & sch - > q ) ;
}
static inline int qdisc_drop ( struct sk_buff * skb , struct Qdisc * sch )
{
kfree_skb ( skb ) ;
sch - > qstats . drops + + ;
return NET_XMIT_DROP ;
}
static inline int qdisc_reshape_fail ( struct sk_buff * skb , struct Qdisc * sch )
{
sch - > qstats . drops + + ;
2007-07-15 11:03:05 +04:00
# ifdef CONFIG_NET_CLS_ACT
2005-06-19 09:57:26 +04:00
if ( sch - > reshape_fail = = NULL | | sch - > reshape_fail ( skb , sch ) )
goto drop ;
return NET_XMIT_SUCCESS ;
drop :
# endif
kfree_skb ( skb ) ;
return NET_XMIT_DROP ;
}
2007-09-12 18:35:24 +04:00
/* Length to Time (L2T) lookup in a qdisc_rate_table, to determine how
long it will take to send a packet given its size .
*/
static inline u32 qdisc_l2t ( struct qdisc_rate_table * rtab , unsigned int pktlen )
{
2007-09-12 18:36:28 +04:00
int slot = pktlen + rtab - > rate . cell_align + rtab - > rate . overhead ;
if ( slot < 0 )
slot = 0 ;
2007-09-12 18:35:24 +04:00
slot > > = rtab - > rate . cell_log ;
if ( slot > 255 )
return ( rtab - > data [ 255 ] * ( slot > > 8 ) + rtab - > data [ slot & 0xFF ] ) ;
return rtab - > data [ slot ] ;
}
2007-10-26 13:47:23 +04:00
# ifdef CONFIG_NET_CLS_ACT
static inline struct sk_buff * skb_act_clone ( struct sk_buff * skb , gfp_t gfp_mask )
{
struct sk_buff * n = skb_clone ( skb , gfp_mask ) ;
if ( n ) {
n - > tc_verd = SET_TC_VERD ( n - > tc_verd , 0 ) ;
n - > tc_verd = CLR_TC_OK2MUNGE ( n - > tc_verd ) ;
n - > tc_verd = CLR_TC_MUNGED ( n - > tc_verd ) ;
}
return n ;
}
# endif
2005-04-17 02:20:36 +04:00
# endif