2006-06-23 02:05:40 -07:00
/*
* Fast batching percpu counters .
*/
# include <linux/percpu_counter.h>
2007-07-15 23:39:51 -07:00
# include <linux/notifier.h>
# include <linux/mutex.h>
# include <linux/init.h>
# include <linux/cpu.h>
2006-06-23 02:05:40 -07:00
# include <linux/module.h>
2010-10-26 14:23:05 -07:00
# include <linux/debugobjects.h>
2006-06-23 02:05:40 -07:00
2011-10-31 17:12:34 -07:00
# ifdef CONFIG_HOTPLUG_CPU
2007-07-15 23:39:51 -07:00
static LIST_HEAD ( percpu_counters ) ;
2012-07-31 09:28:31 +04:00
static DEFINE_SPINLOCK ( percpu_counters_lock ) ;
2011-10-31 17:12:34 -07:00
# endif
2007-07-15 23:39:51 -07:00
2010-10-26 14:23:05 -07:00
# ifdef CONFIG_DEBUG_OBJECTS_PERCPU_COUNTER
static struct debug_obj_descr percpu_counter_debug_descr ;
static int percpu_counter_fixup_free ( void * addr , enum debug_obj_state state )
{
struct percpu_counter * fbc = addr ;
switch ( state ) {
case ODEBUG_STATE_ACTIVE :
percpu_counter_destroy ( fbc ) ;
debug_object_free ( fbc , & percpu_counter_debug_descr ) ;
return 1 ;
default :
return 0 ;
}
}
static struct debug_obj_descr percpu_counter_debug_descr = {
. name = " percpu_counter " ,
. fixup_free = percpu_counter_fixup_free ,
} ;
static inline void debug_percpu_counter_activate ( struct percpu_counter * fbc )
{
debug_object_init ( fbc , & percpu_counter_debug_descr ) ;
debug_object_activate ( fbc , & percpu_counter_debug_descr ) ;
}
static inline void debug_percpu_counter_deactivate ( struct percpu_counter * fbc )
{
debug_object_deactivate ( fbc , & percpu_counter_debug_descr ) ;
debug_object_free ( fbc , & percpu_counter_debug_descr ) ;
}
# else /* CONFIG_DEBUG_OBJECTS_PERCPU_COUNTER */
static inline void debug_percpu_counter_activate ( struct percpu_counter * fbc )
{ }
static inline void debug_percpu_counter_deactivate ( struct percpu_counter * fbc )
{ }
# endif /* CONFIG_DEBUG_OBJECTS_PERCPU_COUNTER */
2007-10-16 23:25:44 -07:00
void percpu_counter_set ( struct percpu_counter * fbc , s64 amount )
{
int cpu ;
2013-10-24 09:06:45 +01:00
unsigned long flags ;
2007-10-16 23:25:44 -07:00
2013-10-24 09:06:45 +01:00
raw_spin_lock_irqsave ( & fbc - > lock , flags ) ;
2007-10-16 23:25:44 -07:00
for_each_possible_cpu ( cpu ) {
s32 * pcount = per_cpu_ptr ( fbc - > counters , cpu ) ;
* pcount = 0 ;
}
fbc - > count = amount ;
2013-10-24 09:06:45 +01:00
raw_spin_unlock_irqrestore ( & fbc - > lock , flags ) ;
2007-10-16 23:25:44 -07:00
}
EXPORT_SYMBOL ( percpu_counter_set ) ;
2007-10-16 23:25:43 -07:00
void __percpu_counter_add ( struct percpu_counter * fbc , s64 amount , s32 batch )
2006-06-23 02:05:40 -07:00
{
2007-10-16 23:25:43 -07:00
s64 count ;
2006-06-23 02:05:40 -07:00
2010-10-26 14:23:09 -07:00
preempt_disable ( ) ;
percpucounter: Optimize __percpu_counter_add a bit through the use of this_cpu() options.
The this_cpu_* options can be used to optimize __percpu_counter_add a bit. Avoids
some address arithmetic and saves 12 bytes.
Before:
00000000000001d3 <__percpu_counter_add>:
1d3: 55 push %rbp
1d4: 48 89 e5 mov %rsp,%rbp
1d7: 41 55 push %r13
1d9: 41 54 push %r12
1db: 53 push %rbx
1dc: 48 89 fb mov %rdi,%rbx
1df: 48 83 ec 08 sub $0x8,%rsp
1e3: 4c 8b 67 30 mov 0x30(%rdi),%r12
1e7: 65 4c 03 24 25 00 00 add %gs:0x0,%r12
1ee: 00 00
1f0: 4d 63 2c 24 movslq (%r12),%r13
1f4: 48 63 c2 movslq %edx,%rax
1f7: 49 01 f5 add %rsi,%r13
1fa: 49 39 c5 cmp %rax,%r13
1fd: 7d 0a jge 209 <__percpu_counter_add+0x36>
1ff: f7 da neg %edx
201: 48 63 d2 movslq %edx,%rdx
204: 49 39 d5 cmp %rdx,%r13
207: 7f 1e jg 227 <__percpu_counter_add+0x54>
209: 48 89 df mov %rbx,%rdi
20c: e8 00 00 00 00 callq 211 <__percpu_counter_add+0x3e>
211: 4c 01 6b 18 add %r13,0x18(%rbx)
215: 48 89 df mov %rbx,%rdi
218: 41 c7 04 24 00 00 00 movl $0x0,(%r12)
21f: 00
220: e8 00 00 00 00 callq 225 <__percpu_counter_add+0x52>
225: eb 04 jmp 22b <__percpu_counter_add+0x58>
227: 45 89 2c 24 mov %r13d,(%r12)
22b: 5b pop %rbx
22c: 5b pop %rbx
22d: 41 5c pop %r12
22f: 41 5d pop %r13
231: c9 leaveq
232: c3 retq
After:
00000000000001d3 <__percpu_counter_add>:
1d3: 55 push %rbp
1d4: 48 63 ca movslq %edx,%rcx
1d7: 48 89 e5 mov %rsp,%rbp
1da: 41 54 push %r12
1dc: 53 push %rbx
1dd: 48 89 fb mov %rdi,%rbx
1e0: 48 8b 47 30 mov 0x30(%rdi),%rax
1e4: 65 44 8b 20 mov %gs:(%rax),%r12d
1e8: 4d 63 e4 movslq %r12d,%r12
1eb: 49 01 f4 add %rsi,%r12
1ee: 49 39 cc cmp %rcx,%r12
1f1: 7d 0a jge 1fd <__percpu_counter_add+0x2a>
1f3: f7 da neg %edx
1f5: 48 63 d2 movslq %edx,%rdx
1f8: 49 39 d4 cmp %rdx,%r12
1fb: 7f 21 jg 21e <__percpu_counter_add+0x4b>
1fd: 48 89 df mov %rbx,%rdi
200: e8 00 00 00 00 callq 205 <__percpu_counter_add+0x32>
205: 4c 01 63 18 add %r12,0x18(%rbx)
209: 48 8b 43 30 mov 0x30(%rbx),%rax
20d: 48 89 df mov %rbx,%rdi
210: 65 c7 00 00 00 00 00 movl $0x0,%gs:(%rax)
217: e8 00 00 00 00 callq 21c <__percpu_counter_add+0x49>
21c: eb 04 jmp 222 <__percpu_counter_add+0x4f>
21e: 65 44 89 20 mov %r12d,%gs:(%rax)
222: 5b pop %rbx
223: 41 5c pop %r12
225: c9 leaveq
226: c3 retq
Reviewed-by: Pekka Enberg <penberg@kernel.org>
Reviewed-by: Tejun Heo <tj@kernel.org>
Reviewed-by: Mathieu Desnoyers <mathieu.desnoyers@efficios.com>
Acked-by: H. Peter Anvin <hpa@zytor.com>
Signed-off-by: Christoph Lameter <cl@linux.com>
Signed-off-by: Tejun Heo <tj@kernel.org>
2010-12-06 11:16:19 -06:00
count = __this_cpu_read ( * fbc - > counters ) + amount ;
2007-10-16 23:25:43 -07:00
if ( count > = batch | | count < = - batch ) {
2013-10-24 09:06:45 +01:00
unsigned long flags ;
raw_spin_lock_irqsave ( & fbc - > lock , flags ) ;
2006-06-23 02:05:40 -07:00
fbc - > count + = count ;
2014-01-16 15:26:48 -08:00
__this_cpu_sub ( * fbc - > counters , count - amount ) ;
2013-10-24 09:06:45 +01:00
raw_spin_unlock_irqrestore ( & fbc - > lock , flags ) ;
2006-06-23 02:05:40 -07:00
} else {
2014-01-14 17:56:42 -08:00
this_cpu_add ( * fbc - > counters , amount ) ;
2006-06-23 02:05:40 -07:00
}
2010-10-26 14:23:09 -07:00
preempt_enable ( ) ;
2006-06-23 02:05:40 -07:00
}
2007-10-16 23:25:43 -07:00
EXPORT_SYMBOL ( __percpu_counter_add ) ;
2006-06-23 02:05:40 -07:00
/*
* Add up all the per - cpu counts , return the result . This is a more accurate
* but much slower version of percpu_counter_read_positive ( )
*/
2008-12-09 13:14:14 -08:00
s64 __percpu_counter_sum ( struct percpu_counter * fbc )
2006-06-23 02:05:40 -07:00
{
2006-06-23 02:05:41 -07:00
s64 ret ;
2006-06-23 02:05:40 -07:00
int cpu ;
2013-10-24 09:06:45 +01:00
unsigned long flags ;
2006-06-23 02:05:40 -07:00
2013-10-24 09:06:45 +01:00
raw_spin_lock_irqsave ( & fbc - > lock , flags ) ;
2006-06-23 02:05:40 -07:00
ret = fbc - > count ;
2007-07-15 23:39:51 -07:00
for_each_online_cpu ( cpu ) {
2006-06-23 02:05:41 -07:00
s32 * pcount = per_cpu_ptr ( fbc - > counters , cpu ) ;
2006-06-23 02:05:40 -07:00
ret + = * pcount ;
}
2013-10-24 09:06:45 +01:00
raw_spin_unlock_irqrestore ( & fbc - > lock , flags ) ;
2007-10-16 23:25:45 -07:00
return ret ;
2006-06-23 02:05:40 -07:00
}
2007-10-16 23:25:45 -07:00
EXPORT_SYMBOL ( __percpu_counter_sum ) ;
2007-07-15 23:39:51 -07:00
2008-12-26 15:08:55 +01:00
int __percpu_counter_init ( struct percpu_counter * fbc , s64 amount ,
struct lock_class_key * key )
2007-07-15 23:39:51 -07:00
{
2009-07-25 16:21:48 +02:00
raw_spin_lock_init ( & fbc - > lock ) ;
2008-12-26 15:08:55 +01:00
lockdep_set_class ( & fbc - > lock , key ) ;
2007-07-15 23:39:51 -07:00
fbc - > count = amount ;
fbc - > counters = alloc_percpu ( s32 ) ;
2007-10-16 23:25:45 -07:00
if ( ! fbc - > counters )
return - ENOMEM ;
2010-10-26 14:23:05 -07:00
debug_percpu_counter_activate ( fbc ) ;
2007-07-15 23:39:51 -07:00
# ifdef CONFIG_HOTPLUG_CPU
2010-10-26 14:21:20 -07:00
INIT_LIST_HEAD ( & fbc - > list ) ;
2012-07-31 09:28:31 +04:00
spin_lock ( & percpu_counters_lock ) ;
2007-07-15 23:39:51 -07:00
list_add ( & fbc - > list , & percpu_counters ) ;
2012-07-31 09:28:31 +04:00
spin_unlock ( & percpu_counters_lock ) ;
2007-07-15 23:39:51 -07:00
# endif
2007-10-16 23:25:45 -07:00
return 0 ;
2007-07-15 23:39:51 -07:00
}
2008-12-26 15:08:55 +01:00
EXPORT_SYMBOL ( __percpu_counter_init ) ;
2007-07-15 23:39:51 -07:00
void percpu_counter_destroy ( struct percpu_counter * fbc )
{
2007-10-16 23:25:45 -07:00
if ( ! fbc - > counters )
return ;
2010-10-26 14:23:05 -07:00
debug_percpu_counter_deactivate ( fbc ) ;
2007-07-15 23:39:51 -07:00
# ifdef CONFIG_HOTPLUG_CPU
2012-07-31 09:28:31 +04:00
spin_lock ( & percpu_counters_lock ) ;
2007-07-15 23:39:51 -07:00
list_del ( & fbc - > list ) ;
2012-07-31 09:28:31 +04:00
spin_unlock ( & percpu_counters_lock ) ;
2007-07-15 23:39:51 -07:00
# endif
2008-12-09 13:14:11 -08:00
free_percpu ( fbc - > counters ) ;
fbc - > counters = NULL ;
2007-07-15 23:39:51 -07:00
}
EXPORT_SYMBOL ( percpu_counter_destroy ) ;
2009-01-06 14:41:04 -08:00
int percpu_counter_batch __read_mostly = 32 ;
EXPORT_SYMBOL ( percpu_counter_batch ) ;
static void compute_batch_value ( void )
{
int nr = num_online_cpus ( ) ;
percpu_counter_batch = max ( 32 , nr * 2 ) ;
}
2013-06-19 14:53:51 -04:00
static int percpu_counter_hotcpu_callback ( struct notifier_block * nb ,
2007-07-15 23:39:51 -07:00
unsigned long action , void * hcpu )
{
2009-01-06 14:41:04 -08:00
# ifdef CONFIG_HOTPLUG_CPU
2007-07-15 23:39:51 -07:00
unsigned int cpu ;
struct percpu_counter * fbc ;
2009-01-06 14:41:04 -08:00
compute_batch_value ( ) ;
2007-07-15 23:39:51 -07:00
if ( action ! = CPU_DEAD )
return NOTIFY_OK ;
cpu = ( unsigned long ) hcpu ;
2012-07-31 09:28:31 +04:00
spin_lock ( & percpu_counters_lock ) ;
2007-07-15 23:39:51 -07:00
list_for_each_entry ( fbc , & percpu_counters , list ) {
s32 * pcount ;
2007-10-18 23:40:47 -07:00
unsigned long flags ;
2007-07-15 23:39:51 -07:00
2009-07-25 16:21:48 +02:00
raw_spin_lock_irqsave ( & fbc - > lock , flags ) ;
2007-07-15 23:39:51 -07:00
pcount = per_cpu_ptr ( fbc - > counters , cpu ) ;
fbc - > count + = * pcount ;
* pcount = 0 ;
2009-07-25 16:21:48 +02:00
raw_spin_unlock_irqrestore ( & fbc - > lock , flags ) ;
2007-07-15 23:39:51 -07:00
}
2012-07-31 09:28:31 +04:00
spin_unlock ( & percpu_counters_lock ) ;
2009-01-06 14:41:04 -08:00
# endif
2007-07-15 23:39:51 -07:00
return NOTIFY_OK ;
}
2010-08-09 17:19:04 -07:00
/*
* Compare counter against given value .
* Return 1 if greater , 0 if equal and - 1 if less
*/
int percpu_counter_compare ( struct percpu_counter * fbc , s64 rhs )
{
s64 count ;
count = percpu_counter_read ( fbc ) ;
/* Check to see if rough count will be sufficient for comparison */
if ( abs ( count - rhs ) > ( percpu_counter_batch * num_online_cpus ( ) ) ) {
if ( count > rhs )
return 1 ;
else
return - 1 ;
}
/* Need to use precise count */
count = percpu_counter_sum ( fbc ) ;
if ( count > rhs )
return 1 ;
else if ( count < rhs )
return - 1 ;
else
return 0 ;
}
EXPORT_SYMBOL ( percpu_counter_compare ) ;
2007-07-15 23:39:51 -07:00
static int __init percpu_counter_startup ( void )
{
2009-01-06 14:41:04 -08:00
compute_batch_value ( ) ;
2007-07-15 23:39:51 -07:00
hotcpu_notifier ( percpu_counter_hotcpu_callback , 0 ) ;
return 0 ;
}
module_init ( percpu_counter_startup ) ;