2005-04-17 02:20:36 +04:00
/*
* Simple traffic shaper for Linux NET3 .
*
* ( c ) Copyright 1996 Alan Cox < alan @ redhat . com > , All Rights Reserved .
* http : //www.redhat.com
*
* 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 .
*
2006-09-13 21:24:59 +04:00
* Neither Alan Cox nor CymruNet Ltd . admit liability nor provide
* warranty for any of this software . This material is provided
* " AS-IS " and at no charge .
*
*
2005-04-17 02:20:36 +04:00
* Algorithm :
*
* Queue Frame :
* Compute time length of frame at regulated speed
* Add frame to queue at appropriate point
* Adjust time length computation for followup frames
* Any frame that falls outside of its boundaries is freed
*
* We work to the following constants
*
* SHAPER_QLEN Maximum queued frames
* SHAPER_LATENCY Bounding latency on a frame . Leaving this latency
2006-09-13 21:24:59 +04:00
* window drops the frame . This stops us queueing
2005-04-17 02:20:36 +04:00
* frames for a long time and confusing a remote
* host .
* SHAPER_MAXSLIP Maximum time a priority frame may jump forward .
* That bounds the penalty we will inflict on low
* priority traffic .
* SHAPER_BURST Time range we call " now " in order to reduce
* system load . The more we make this the burstier
* the behaviour , the better local performance you
* get through packet clustering on routers and the
* worse the remote end gets to judge rtts .
*
* This is designed to handle lower speed links ( < 200 K / second or so ) . We
* run off a 100 - 150 Hz base clock typically . This gives us a resolution at
* 200 Kbit / second of about 2 Kbit or 256 bytes . Above that our timer
* resolution may start to cause much more burstiness in the traffic . We
2006-09-13 21:24:59 +04:00
* could avoid a lot of that by calling kick_shaper ( ) at the end of the
* tied device transmissions . If you run above about 100 K second you
2005-04-17 02:20:36 +04:00
* may need to tune the supposed speed rate for the right values .
*
* BUGS :
* Downing the interface under the shaper before the shaper
* will render your machine defunct . Don ' t for now shape over
* PPP or SLIP therefore !
* This will be fixed in BETA4
*
* Update History :
*
* bh_atomic ( ) SMP races fixes and rewritten the locking code to
* be SMP safe and irq - mask friendly .
* NOTE : we can ' t use start_bh_atomic ( ) in kick_shaper ( )
* because it ' s going to be recalled from an irq handler ,
* and synchronize_bh ( ) is a nono if called from irq context .
* 1999 Andrea Arcangeli
*
* Device statistics ( tx_pakets , tx_bytes ,
* tx_drops : queue_over_time and collisions : max_queue_exceded )
* 1999 / 06 / 18 Jordi Murgo < savage @ apostols . org >
*
* Use skb - > cb for private data .
* 2000 / 03 Andi Kleen
*/
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/fcntl.h>
# include <linux/mm.h>
# include <linux/slab.h>
# include <linux/string.h>
# include <linux/errno.h>
# include <linux/netdevice.h>
# include <linux/etherdevice.h>
# include <linux/skbuff.h>
# include <linux/if_arp.h>
# include <linux/init.h>
# include <linux/if_shaper.h>
2006-01-10 05:37:15 +03:00
# include <linux/jiffies.h>
2005-04-17 02:20:36 +04:00
# include <net/dst.h>
# include <net/arp.h>
2007-09-17 22:56:21 +04:00
# include <net/net_namespace.h>
2005-04-17 02:20:36 +04:00
2006-09-13 21:24:59 +04:00
struct shaper_cb {
2005-04-17 02:20:36 +04:00
unsigned long shapeclock ; /* Time it should go out */
unsigned long shapestamp ; /* Stamp for shaper */
__u32 shapelatency ; /* Latency on frame */
__u32 shapelen ; /* Frame length in clocks */
__u16 shapepend ; /* Pending */
2006-09-13 21:24:59 +04:00
} ;
2005-04-17 02:20:36 +04:00
# define SHAPERCB(skb) ((struct shaper_cb *) ((skb)->cb))
static int sh_debug ; /* Debug flag */
# define SHAPER_BANNER "CymruNet Traffic Shaper BETA 0.04 for Linux 2.1\n"
static void shaper_kick ( struct shaper * sh ) ;
/*
* Compute clocks on a buffer
*/
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
static int shaper_clocks ( struct shaper * shaper , struct sk_buff * skb )
{
int t = skb - > len / shaper - > bytespertick ;
return t ;
}
/*
* Set the speed of a shaper . We compute this in bytes per tick since
* thats how the machine wants to run . Quoted input is in bits per second
2006-09-13 21:24:59 +04:00
* as is traditional ( note not BAUD ) . We assume 8 bit bytes .
2005-04-17 02:20:36 +04:00
*/
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
static void shaper_setspeed ( struct shaper * shaper , int bitspersec )
{
shaper - > bitspersec = bitspersec ;
shaper - > bytespertick = ( bitspersec / HZ ) / 8 ;
if ( ! shaper - > bytespertick )
shaper - > bytespertick + + ;
}
/*
* Throw a frame at a shaper .
*/
2006-09-13 21:24:59 +04:00
2005-06-03 03:36:00 +04:00
static int shaper_start_xmit ( struct sk_buff * skb , struct net_device * dev )
2005-04-17 02:20:36 +04:00
{
2005-06-03 03:36:00 +04:00
struct shaper * shaper = dev - > priv ;
2005-04-17 02:20:36 +04:00
struct sk_buff * ptr ;
2006-09-13 21:24:59 +04:00
2005-07-06 02:03:46 +04:00
spin_lock ( & shaper - > lock ) ;
2005-04-17 02:20:36 +04:00
ptr = shaper - > sendq . prev ;
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
/*
* Set up our packet details
*/
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
SHAPERCB ( skb ) - > shapelatency = 0 ;
SHAPERCB ( skb ) - > shapeclock = shaper - > recovery ;
if ( time_before ( SHAPERCB ( skb ) - > shapeclock , jiffies ) )
SHAPERCB ( skb ) - > shapeclock = jiffies ;
skb - > priority = 0 ; /* short term bug fix */
SHAPERCB ( skb ) - > shapestamp = jiffies ;
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
/*
* Time slots for this packet .
*/
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
SHAPERCB ( skb ) - > shapelen = shaper_clocks ( shaper , skb ) ;
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
{
struct sk_buff * tmp ;
/*
* Up our shape clock by the time pending on the queue
* ( Should keep this in the shaper as a variable . . )
*/
2006-09-13 21:24:59 +04:00
for ( tmp = skb_peek ( & shaper - > sendq ) ; tmp ! = NULL & &
2005-04-17 02:20:36 +04:00
tmp ! = ( struct sk_buff * ) & shaper - > sendq ; tmp = tmp - > next )
SHAPERCB ( skb ) - > shapeclock + = SHAPERCB ( tmp ) - > shapelen ;
/*
* Queue over time . Spill packet .
*/
2006-01-10 05:37:15 +03:00
if ( time_after ( SHAPERCB ( skb ) - > shapeclock , jiffies + SHAPER_LATENCY ) ) {
2005-04-17 02:20:36 +04:00
dev_kfree_skb ( skb ) ;
2007-10-04 04:41:50 +04:00
dev - > stats . tx_dropped + + ;
2005-04-17 02:20:36 +04:00
} else
skb_queue_tail ( & shaper - > sendq , skb ) ;
}
2005-08-10 06:25:21 +04:00
2005-04-17 02:20:36 +04:00
if ( sh_debug )
printk ( " Frame queued. \n " ) ;
if ( skb_queue_len ( & shaper - > sendq ) > SHAPER_QLEN )
{
ptr = skb_dequeue ( & shaper - > sendq ) ;
dev_kfree_skb ( ptr ) ;
2007-10-04 04:41:50 +04:00
dev - > stats . collisions + + ;
2005-04-17 02:20:36 +04:00
}
2005-06-03 03:36:00 +04:00
shaper_kick ( shaper ) ;
2005-07-06 02:03:46 +04:00
spin_unlock ( & shaper - > lock ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
/*
* Transmit from a shaper
*/
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
static void shaper_queue_xmit ( struct shaper * shaper , struct sk_buff * skb )
{
struct sk_buff * newskb = skb_clone ( skb , GFP_ATOMIC ) ;
if ( sh_debug )
printk ( " Kick frame on %p \n " , newskb ) ;
if ( newskb )
{
newskb - > dev = shaper - > dev ;
newskb - > priority = 2 ;
if ( sh_debug )
printk ( " Kick new frame to %s, %d \n " ,
shaper - > dev - > name , newskb - > priority ) ;
dev_queue_xmit ( newskb ) ;
2007-10-04 04:41:50 +04:00
shaper - > dev - > stats . tx_bytes + = skb - > len ;
shaper - > dev - > stats . tx_packets + + ;
2005-04-17 02:20:36 +04:00
if ( sh_debug )
printk ( " Kicked new frame out. \n " ) ;
dev_kfree_skb ( skb ) ;
}
}
/*
* Timer handler for shaping clock
*/
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
static void shaper_timer ( unsigned long data )
{
2005-06-03 03:36:00 +04:00
struct shaper * shaper = ( struct shaper * ) data ;
2005-07-06 02:03:46 +04:00
spin_lock ( & shaper - > lock ) ;
shaper_kick ( shaper ) ;
spin_unlock ( & shaper - > lock ) ;
2005-04-17 02:20:36 +04:00
}
/*
2006-09-13 21:24:59 +04:00
* Kick a shaper queue and try and do something sensible with the
* queue .
2005-04-17 02:20:36 +04:00
*/
static void shaper_kick ( struct shaper * shaper )
{
struct sk_buff * skb ;
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
/*
* Walk the list ( may be empty )
*/
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
while ( ( skb = skb_peek ( & shaper - > sendq ) ) ! = NULL )
{
/*
* Each packet due to go out by now ( within an error
2006-09-13 21:24:59 +04:00
* of SHAPER_BURST ) gets kicked onto the link
2005-04-17 02:20:36 +04:00
*/
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
if ( sh_debug )
printk ( " Clock = %ld, jiffies = %ld \n " , SHAPERCB ( skb ) - > shapeclock , jiffies ) ;
if ( time_before_eq ( SHAPERCB ( skb ) - > shapeclock , jiffies + SHAPER_BURST ) )
{
/*
* Pull the frame and get interrupts back on .
*/
2006-09-13 21:24:59 +04:00
2005-08-10 06:25:21 +04:00
skb_unlink ( skb , & shaper - > sendq ) ;
2006-09-13 21:24:59 +04:00
if ( shaper - > recovery <
2005-04-17 02:20:36 +04:00
SHAPERCB ( skb ) - > shapeclock + SHAPERCB ( skb ) - > shapelen )
shaper - > recovery = SHAPERCB ( skb ) - > shapeclock + SHAPERCB ( skb ) - > shapelen ;
/*
* Pass on to the physical target device via
* our low level packet thrower .
*/
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
SHAPERCB ( skb ) - > shapepend = 0 ;
shaper_queue_xmit ( shaper , skb ) ; /* Fire */
}
else
break ;
}
/*
* Next kick .
*/
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
if ( skb ! = NULL )
mod_timer ( & shaper - > timer , SHAPERCB ( skb ) - > shapeclock ) ;
}
/*
2006-09-13 21:24:59 +04:00
* Bring the interface up . We just disallow this until a
2005-04-17 02:20:36 +04:00
* bind .
*/
static int shaper_open ( struct net_device * dev )
{
struct shaper * shaper = dev - > priv ;
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
/*
* Can ' t open until attached .
* Also can ' t open until speed is set , or we ' ll get
* a division by zero .
*/
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
if ( shaper - > dev = = NULL )
return - ENODEV ;
if ( shaper - > bitspersec = = 0 )
return - EINVAL ;
return 0 ;
}
/*
* Closing a shaper flushes the queues .
*/
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
static int shaper_close ( struct net_device * dev )
{
struct shaper * shaper = dev - > priv ;
2005-07-06 02:03:46 +04:00
struct sk_buff * skb ;
while ( ( skb = skb_dequeue ( & shaper - > sendq ) ) ! = NULL )
dev_kfree_skb ( skb ) ;
spin_lock_bh ( & shaper - > lock ) ;
shaper_kick ( shaper ) ;
spin_unlock_bh ( & shaper - > lock ) ;
2005-04-17 02:20:36 +04:00
del_timer_sync ( & shaper - > timer ) ;
return 0 ;
}
/*
* Revectored calls . We alter the parameters and call the functions
* for our attached device . This enables us to bandwidth allocate after
* ARP and other resolutions and not before .
*/
2006-09-13 21:24:59 +04:00
static int shaper_header ( struct sk_buff * skb , struct net_device * dev ,
2007-10-09 12:40:57 +04:00
unsigned short type ,
const void * daddr , const void * saddr , unsigned len )
2005-04-17 02:20:36 +04:00
{
struct shaper * sh = dev - > priv ;
int v ;
if ( sh_debug )
printk ( " Shaper header \n " ) ;
2007-10-09 12:40:57 +04:00
skb - > dev = sh - > dev ;
v = dev_hard_header ( skb , sh - > dev , type , daddr , saddr , len ) ;
skb - > dev = dev ;
2005-04-17 02:20:36 +04:00
return v ;
}
static int shaper_rebuild_header ( struct sk_buff * skb )
{
struct shaper * sh = skb - > dev - > priv ;
struct net_device * dev = skb - > dev ;
int v ;
if ( sh_debug )
printk ( " Shaper rebuild header \n " ) ;
skb - > dev = sh - > dev ;
2007-10-09 12:40:57 +04:00
v = sh - > dev - > header_ops - > rebuild ( skb ) ;
2005-04-17 02:20:36 +04:00
skb - > dev = dev ;
return v ;
}
#if 0
static int shaper_cache ( struct neighbour * neigh , struct hh_cache * hh )
{
struct shaper * sh = neigh - > dev - > priv ;
struct net_device * tmp ;
int ret ;
if ( sh_debug )
printk ( " Shaper header cache bind \n " ) ;
tmp = neigh - > dev ;
neigh - > dev = sh - > dev ;
ret = sh - > hard_header_cache ( neigh , hh ) ;
neigh - > dev = tmp ;
return ret ;
}
static void shaper_cache_update ( struct hh_cache * hh , struct net_device * dev ,
unsigned char * haddr )
{
struct shaper * sh = dev - > priv ;
if ( sh_debug )
printk ( " Shaper cache update \n " ) ;
sh - > header_cache_update ( hh , sh - > dev , haddr ) ;
}
# endif
# ifdef CONFIG_INET
static int shaper_neigh_setup ( struct neighbour * n )
{
# ifdef CONFIG_INET
if ( n - > nud_state = = NUD_NONE ) {
n - > ops = & arp_broken_ops ;
n - > output = n - > ops - > output ;
}
2006-09-13 21:24:59 +04:00
# endif
2005-04-17 02:20:36 +04:00
return 0 ;
}
static int shaper_neigh_setup_dev ( struct net_device * dev , struct neigh_parms * p )
{
# ifdef CONFIG_INET
if ( p - > tbl - > family = = AF_INET ) {
p - > neigh_setup = shaper_neigh_setup ;
p - > ucast_probes = 0 ;
p - > mcast_probes = 0 ;
}
2006-09-13 21:24:59 +04:00
# endif
2005-04-17 02:20:36 +04:00
return 0 ;
}
# else /* !(CONFIG_INET) */
static int shaper_neigh_setup_dev ( struct net_device * dev , struct neigh_parms * p )
{
return 0 ;
}
# endif
2007-10-09 12:40:57 +04:00
static const struct header_ops shaper_ops = {
. create = shaper_header ,
. rebuild = shaper_rebuild_header ,
} ;
2005-04-17 02:20:36 +04:00
static int shaper_attach ( struct net_device * shdev , struct shaper * sh , struct net_device * dev )
{
sh - > dev = dev ;
sh - > get_stats = dev - > get_stats ;
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
shdev - > neigh_setup = shaper_neigh_setup_dev ;
shdev - > hard_header_len = dev - > hard_header_len ;
shdev - > type = dev - > type ;
shdev - > addr_len = dev - > addr_len ;
shdev - > mtu = dev - > mtu ;
sh - > bitspersec = 0 ;
return 0 ;
}
static int shaper_ioctl ( struct net_device * dev , struct ifreq * ifr , int cmd )
{
struct shaperconf * ss = ( struct shaperconf * ) & ifr - > ifr_ifru ;
struct shaper * sh = dev - > priv ;
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
if ( ss - > ss_cmd = = SHAPER_SET_DEV | | ss - > ss_cmd = = SHAPER_SET_SPEED )
{
if ( ! capable ( CAP_NET_ADMIN ) )
return - EPERM ;
}
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
switch ( ss - > ss_cmd )
{
case SHAPER_SET_DEV :
{
2007-09-17 22:56:21 +04:00
struct net_device * them = __dev_get_by_name ( & init_net , ss - > ss_name ) ;
2005-04-17 02:20:36 +04:00
if ( them = = NULL )
return - ENODEV ;
if ( sh - > dev )
return - EBUSY ;
return shaper_attach ( dev , dev - > priv , them ) ;
}
case SHAPER_GET_DEV :
if ( sh - > dev = = NULL )
return - ENODEV ;
strcpy ( ss - > ss_name , sh - > dev - > name ) ;
return 0 ;
case SHAPER_SET_SPEED :
shaper_setspeed ( sh , ss - > ss_speed ) ;
return 0 ;
case SHAPER_GET_SPEED :
ss - > ss_speed = sh - > bitspersec ;
return 0 ;
default :
return - EINVAL ;
}
}
static void shaper_init_priv ( struct net_device * dev )
{
struct shaper * sh = dev - > priv ;
skb_queue_head_init ( & sh - > sendq ) ;
init_timer ( & sh - > timer ) ;
sh - > timer . function = shaper_timer ;
sh - > timer . data = ( unsigned long ) sh ;
2005-07-06 02:03:46 +04:00
spin_lock_init ( & sh - > lock ) ;
2005-04-17 02:20:36 +04:00
}
/*
* Add a shaper device to the system
*/
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
static void __init shaper_setup ( struct net_device * dev )
{
/*
* Set up the shaper .
*/
shaper_init_priv ( dev ) ;
dev - > open = shaper_open ;
dev - > stop = shaper_close ;
dev - > hard_start_xmit = shaper_start_xmit ;
dev - > set_multicast_list = NULL ;
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
/*
* Intialise the packet queues
*/
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
/*
* Handlers for when we attach to a device .
*/
dev - > neigh_setup = shaper_neigh_setup_dev ;
dev - > do_ioctl = shaper_ioctl ;
dev - > hard_header_len = 0 ;
dev - > type = ARPHRD_ETHER ; /* initially */
dev - > set_mac_address = NULL ;
dev - > mtu = 1500 ;
dev - > addr_len = 0 ;
dev - > tx_queue_len = 10 ;
dev - > flags = 0 ;
}
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
static int shapers = 1 ;
# ifdef MODULE
module_param ( shapers , int , 0 ) ;
MODULE_PARM_DESC ( shapers , " Traffic shaper: maximum number of shapers " ) ;
# else /* MODULE */
static int __init set_num_shapers ( char * str )
{
shapers = simple_strtol ( str , NULL , 0 ) ;
return 1 ;
}
__setup ( " shapers= " , set_num_shapers ) ;
# endif /* MODULE */
static struct net_device * * devs ;
static unsigned int shapers_registered = 0 ;
static int __init shaper_init ( void )
{
int i ;
size_t alloc_size ;
struct net_device * dev ;
char name [ IFNAMSIZ ] ;
if ( shapers < 1 )
return - ENODEV ;
alloc_size = sizeof ( * dev ) * shapers ;
some kmalloc/memset ->kzalloc (tree wide)
Transform some calls to kmalloc/memset to a single kzalloc (or kcalloc).
Here is a short excerpt of the semantic patch performing
this transformation:
@@
type T2;
expression x;
identifier f,fld;
expression E;
expression E1,E2;
expression e1,e2,e3,y;
statement S;
@@
x =
- kmalloc
+ kzalloc
(E1,E2)
... when != \(x->fld=E;\|y=f(...,x,...);\|f(...,x,...);\|x=E;\|while(...) S\|for(e1;e2;e3) S\)
- memset((T2)x,0,E1);
@@
expression E1,E2,E3;
@@
- kzalloc(E1 * E2,E3)
+ kcalloc(E1,E2,E3)
[akpm@linux-foundation.org: get kcalloc args the right way around]
Signed-off-by: Yoann Padioleau <padator@wanadoo.fr>
Cc: Richard Henderson <rth@twiddle.net>
Cc: Ivan Kokshaysky <ink@jurassic.park.msu.ru>
Acked-by: Russell King <rmk@arm.linux.org.uk>
Cc: Bryan Wu <bryan.wu@analog.com>
Acked-by: Jiri Slaby <jirislaby@gmail.com>
Cc: Dave Airlie <airlied@linux.ie>
Acked-by: Roland Dreier <rolandd@cisco.com>
Cc: Jiri Kosina <jkosina@suse.cz>
Acked-by: Dmitry Torokhov <dtor@mail.ru>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Acked-by: Mauro Carvalho Chehab <mchehab@infradead.org>
Acked-by: Pierre Ossman <drzeus-list@drzeus.cx>
Cc: Jeff Garzik <jeff@garzik.org>
Cc: "David S. Miller" <davem@davemloft.net>
Acked-by: Greg KH <greg@kroah.com>
Cc: James Bottomley <James.Bottomley@steeleye.com>
Cc: "Antonino A. Daplas" <adaplas@pol.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-07-19 12:49:03 +04:00
devs = kzalloc ( alloc_size , GFP_KERNEL ) ;
2005-04-17 02:20:36 +04:00
if ( ! devs )
return - ENOMEM ;
for ( i = 0 ; i < shapers ; i + + ) {
snprintf ( name , IFNAMSIZ , " shaper%d " , i ) ;
dev = alloc_netdev ( sizeof ( struct shaper ) , name ,
shaper_setup ) ;
2006-09-13 21:24:59 +04:00
if ( ! dev )
2005-04-17 02:20:36 +04:00
break ;
if ( register_netdev ( dev ) ) {
free_netdev ( dev ) ;
break ;
}
devs [ i ] = dev ;
shapers_registered + + ;
}
if ( ! shapers_registered ) {
kfree ( devs ) ;
devs = NULL ;
}
return ( shapers_registered ? 0 : - ENODEV ) ;
}
static void __exit shaper_exit ( void )
{
int i ;
for ( i = 0 ; i < shapers_registered ; i + + ) {
if ( devs [ i ] ) {
unregister_netdev ( devs [ i ] ) ;
free_netdev ( devs [ i ] ) ;
}
}
kfree ( devs ) ;
devs = NULL ;
}
module_init ( shaper_init ) ;
module_exit ( shaper_exit ) ;
MODULE_LICENSE ( " GPL " ) ;