2008-04-29 11:59:43 +04:00
/*
* ratelimit . c - Do something with rate limit .
*
* Isolated from kernel / printk . c by Dave Young < hidave . darkstar @ gmail . com >
*
2008-07-25 12:45:58 +04: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 11:59:43 +04:00
* This file is released under the GPLv2 .
*/
# include <linux/kernel.h>
# include <linux/jiffies.h>
# include <linux/module.h>
/*
* __ratelimit - rate limiting
2008-07-25 12:45:58 +04:00
* @ rs : ratelimit_state data
2008-04-29 11:59:43 +04:00
*
2008-07-25 12:45:58 +04:00
* This enforces a rate limit : not more than @ rs - > ratelimit_burst callbacks
* in every @ rs - > ratelimit_jiffies
2008-04-29 11:59:43 +04:00
*/
2008-07-25 12:45:58 +04:00
int __ratelimit ( struct ratelimit_state * rs )
2008-04-29 11:59:43 +04:00
{
2008-07-29 02:46:21 +04:00
unsigned long flags ;
2009-09-22 16:44:11 +04:00
int ret ;
2008-07-29 02:46:21 +04:00
2008-07-25 12:45:58 +04:00
if ( ! rs - > interval )
return 1 ;
2008-04-29 11:59:43 +04:00
2009-09-22 16:44:11 +04: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 12:45:58 +04:00
if ( ! rs - > begin )
rs - > begin = jiffies ;
2008-04-29 11:59:43 +04:00
2008-07-25 12:45:58 +04:00
if ( time_is_before_jiffies ( rs - > begin + rs - > interval ) ) {
if ( rs - > missed )
printk ( KERN_WARNING " %s: %d callbacks suppressed \n " ,
__func__ , rs - > missed ) ;
2009-09-22 16:44:11 +04:00
rs - > begin = 0 ;
2008-07-25 12:45:58 +04:00
rs - > printed = 0 ;
2009-09-22 16:44:11 +04:00
rs - > missed = 0 ;
2008-04-29 11:59:43 +04:00
}
2009-09-22 16:44:11 +04: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 12:45:58 +04:00
2009-09-22 16:44:11 +04:00
return ret ;
2008-04-29 11:59:43 +04:00
}
EXPORT_SYMBOL ( __ratelimit ) ;