2010-11-16 10:58:37 +09:00
/*
* lib / average . c
*
* This source code is licensed under the GNU General Public License ,
* Version 2. See the file COPYING for more details .
*/
2011-11-16 21:29:17 -05:00
# include <linux/export.h>
2010-11-16 10:58:37 +09:00
# include <linux/average.h>
2011-11-16 21:29:17 -05:00
# include <linux/kernel.h>
2010-11-16 10:58:37 +09:00
# include <linux/bug.h>
2010-12-02 19:50:37 +09:00
# include <linux/log2.h>
2010-11-16 10:58:37 +09:00
/**
* DOC : Exponentially Weighted Moving Average ( EWMA )
*
* These are generic functions for calculating Exponentially Weighted Moving
* Averages ( EWMA ) . We keep a structure with the EWMA parameters and a scaled
* up internal representation of the average value to prevent rounding errors .
* The factor for scaling up and the exponential weight ( or decay rate ) have to
* be specified thru the init fuction . The structure should not be accessed
* directly but only thru the helper functions .
*/
/**
* ewma_init ( ) - Initialize EWMA parameters
* @ avg : Average structure
* @ factor : Factor to use for the scaled up internal value . The maximum value
2010-12-02 19:50:37 +09:00
* of averages can be ULONG_MAX / ( factor * weight ) . For performance reasons
* factor has to be a power of 2.
2010-11-16 10:58:37 +09:00
* @ weight : Exponential weight , or decay rate . This defines how fast the
2010-12-02 19:50:37 +09:00
* influence of older values decreases . For performance reasons weight has
* to be a power of 2.
2010-11-16 10:58:37 +09:00
*
* Initialize the EWMA parameters for a given struct ewma @ avg .
*/
void ewma_init ( struct ewma * avg , unsigned long factor , unsigned long weight )
{
2010-12-02 19:50:37 +09:00
WARN_ON ( ! is_power_of_2 ( weight ) | | ! is_power_of_2 ( factor ) ) ;
avg - > weight = ilog2 ( weight ) ;
avg - > factor = ilog2 ( factor ) ;
2010-11-16 10:58:37 +09:00
avg - > internal = 0 ;
}
EXPORT_SYMBOL ( ewma_init ) ;
/**
* ewma_add ( ) - Exponentially weighted moving average ( EWMA )
* @ avg : Average structure
* @ val : Current value
*
* Add a sample to the average .
*/
struct ewma * ewma_add ( struct ewma * avg , unsigned long val )
{
2014-01-16 22:23:29 -08:00
unsigned long internal = ACCESS_ONCE ( avg - > internal ) ;
ACCESS_ONCE ( avg - > internal ) = internal ?
( ( ( internal < < avg - > weight ) - internal ) +
2010-12-02 19:50:37 +09:00
( val < < avg - > factor ) ) > > avg - > weight :
( val < < avg - > factor ) ;
2010-11-16 10:58:37 +09:00
return avg ;
}
EXPORT_SYMBOL ( ewma_add ) ;