#ifndef __LINUX_PCOUNTER_H #define __LINUX_PCOUNTER_H /* * Using a dynamic percpu 'int' variable has a cost : * 1) Extra dereference * Current per_cpu_ptr() implementation uses an array per 'percpu variable'. * 2) memory cost of NR_CPUS*(32+sizeof(void *)) instead of num_possible_cpus()*4 * * This pcounter implementation is an abstraction to be able to use * either a static or a dynamic per cpu variable. * One dynamic per cpu variable gets a fast & cheap implementation, we can * change pcounter implementation too. */ struct pcounter { #ifdef CONFIG_SMP void (*add)(struct pcounter *self, int inc); int (*getval)(const struct pcounter *self, int cpu); int *per_cpu_values; #else int val; #endif }; #ifdef CONFIG_SMP #include <linux/percpu.h> #define DEFINE_PCOUNTER(NAME) \ static DEFINE_PER_CPU(int, NAME##_pcounter_values); \ static void NAME##_pcounter_add(struct pcounter *self, int val) \ { \ __get_cpu_var(NAME##_pcounter_values) += val; \ } \ static int NAME##_pcounter_getval(const struct pcounter *self, int cpu) \ { \ return per_cpu(NAME##_pcounter_values, cpu); \ } \ #define PCOUNTER_MEMBER_INITIALIZER(NAME, MEMBER) \ MEMBER = { \ .add = NAME##_pcounter_add, \ .getval = NAME##_pcounter_getval, \ } static inline void pcounter_add(struct pcounter *self, int inc) { self->add(self, inc); } extern int pcounter_getval(const struct pcounter *self); extern int pcounter_alloc(struct pcounter *self); extern void pcounter_free(struct pcounter *self); #else /* CONFIG_SMP */ static inline void pcounter_add(struct pcounter *self, int inc) { self->val += inc; } static inline int pcounter_getval(const struct pcounter *self) { return self->val; } #define DEFINE_PCOUNTER(NAME) #define PCOUNTER_MEMBER_INITIALIZER(NAME, MEMBER) #define pcounter_alloc(self) 0 #define pcounter_free(self) #endif /* CONFIG_SMP */ #endif /* __LINUX_PCOUNTER_H */