2008-04-29 00:59:43 -07:00
/*
* ratelimit . c - Do something with rate limit .
*
* Isolated from kernel / printk . c by Dave Young < hidave . darkstar @ gmail . com >
*
2008-07-25 01:45:58 -07:00
* 2008 - 05 - 01 rewrite the function and use a ratelimit_state data struct as
* parameter . Now every user can use their own standalone ratelimit_state .
*
2008-04-29 00:59:43 -07:00
* This file is released under the GPLv2 .
*/
2009-09-22 16:18:09 +02:00
# include <linux/ratelimit.h>
2008-04-29 00:59:43 -07:00
# include <linux/jiffies.h>
# include <linux/module.h>
/*
* __ratelimit - rate limiting
2008-07-25 01:45:58 -07:00
* @ rs : ratelimit_state data
2008-04-29 00:59:43 -07:00
*
2008-07-25 01:45:58 -07:00
* This enforces a rate limit : not more than @ rs - > ratelimit_burst callbacks
* in every @ rs - > ratelimit_jiffies
2008-04-29 00:59:43 -07:00
*/
2009-10-23 14:58:11 +02:00
int ___ratelimit ( struct ratelimit_state * rs , const char * func )
2008-04-29 00:59:43 -07:00
{
2008-07-28 15:46:21 -07:00
unsigned long flags ;
2009-09-22 14:44:11 +02:00
int ret ;
2008-07-28 15:46:21 -07:00
2008-07-25 01:45:58 -07:00
if ( ! rs - > interval )
return 1 ;
2008-04-29 00:59:43 -07:00
2009-09-22 14:44:11 +02:00
/*
* If we contend on this state ' s lock then almost
* by definition we are too busy to print a message ,
* in addition to the one that will be printed by
* the entity that is holding the lock already :
*/
if ( ! spin_trylock_irqsave ( & rs - > lock , flags ) )
return 1 ;
2008-07-25 01:45:58 -07:00
if ( ! rs - > begin )
rs - > begin = jiffies ;
2008-04-29 00:59:43 -07:00
2008-07-25 01:45:58 -07:00
if ( time_is_before_jiffies ( rs - > begin + rs - > interval ) ) {
if ( rs - > missed )
printk ( KERN_WARNING " %s: %d callbacks suppressed \n " ,
2009-10-23 14:58:11 +02:00
func , rs - > missed ) ;
2009-09-22 14:44:11 +02:00
rs - > begin = 0 ;
2008-07-25 01:45:58 -07:00
rs - > printed = 0 ;
2009-09-22 14:44:11 +02:00
rs - > missed = 0 ;
2008-04-29 00:59:43 -07:00
}
2009-09-22 14:44:11 +02:00
if ( rs - > burst & & rs - > burst > rs - > printed ) {
rs - > printed + + ;
ret = 1 ;
} else {
rs - > missed + + ;
ret = 0 ;
}
spin_unlock_irqrestore ( & rs - > lock , flags ) ;
2008-07-25 01:45:58 -07:00
2009-09-22 14:44:11 +02:00
return ret ;
2008-04-29 00:59:43 -07:00
}
2009-10-23 14:58:11 +02:00
EXPORT_SYMBOL ( ___ratelimit ) ;