Viewing file: average.h (1.48 KB) -rw-r--r-- Select action/file-type: (+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
#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 */
|