| #ifndef _LINUX_AVERAGE_H |
| #define _LINUX_AVERAGE_H |
| |
| /* Exponentially weighted moving average (EWMA) */ |
| |
| #define DECLARE_EWMA(name, _factor, _weight) \ |
| struct ewma_##name { \ |
| unsigned long internal; \ |
| }; \ |
| static inline void ewma_##name##_init(struct ewma_##name *e) \ |
| { \ |
| BUILD_BUG_ON(!__builtin_constant_p(_factor)); \ |
| BUILD_BUG_ON(!__builtin_constant_p(_weight)); \ |
| BUILD_BUG_ON_NOT_POWER_OF_2(_factor); \ |
| BUILD_BUG_ON_NOT_POWER_OF_2(_weight); \ |
| e->internal = 0; \ |
| } \ |
| static inline unsigned long \ |
| ewma_##name##_read(struct ewma_##name *e) \ |
| { \ |
| BUILD_BUG_ON(!__builtin_constant_p(_factor)); \ |
| BUILD_BUG_ON(!__builtin_constant_p(_weight)); \ |
| BUILD_BUG_ON_NOT_POWER_OF_2(_factor); \ |
| BUILD_BUG_ON_NOT_POWER_OF_2(_weight); \ |
| return e->internal >> ilog2(_factor); \ |
| } \ |
| static inline void ewma_##name##_add(struct ewma_##name *e, \ |
| unsigned long val) \ |
| { \ |
| unsigned long internal = ACCESS_ONCE(e->internal); \ |
| unsigned long weight = ilog2(_weight); \ |
| unsigned long factor = ilog2(_factor); \ |
| \ |
| BUILD_BUG_ON(!__builtin_constant_p(_factor)); \ |
| BUILD_BUG_ON(!__builtin_constant_p(_weight)); \ |
| BUILD_BUG_ON_NOT_POWER_OF_2(_factor); \ |
| BUILD_BUG_ON_NOT_POWER_OF_2(_weight); \ |
| \ |
| ACCESS_ONCE(e->internal) = internal ? \ |
| (((internal << weight) - internal) + \ |
| (val << factor)) >> weight : \ |
| (val << factor); \ |
| } |
| |
| #endif /* _LINUX_AVERAGE_H */ |