2013-09-29 19:28:34 +02:00
|
|
|
//
|
2017-07-16 08:04:20 +02:00
|
|
|
// FILE: Histogram.cpp
|
2013-09-29 21:55:02 +02:00
|
|
|
// AUTHOR: Rob Tillaart
|
2017-07-27 13:13:41 +02:00
|
|
|
// VERSION: 0.1.6
|
2013-09-29 19:28:34 +02:00
|
|
|
// PURPOSE: Histogram library for Arduino
|
2013-09-29 21:55:02 +02:00
|
|
|
// DATE: 2012-11-10
|
|
|
|
//
|
|
|
|
// Released to the public domain
|
2013-09-29 19:28:34 +02:00
|
|
|
//
|
|
|
|
// HISTORY:
|
|
|
|
// 0.1.0 - 2012-11-10 initial version
|
|
|
|
// 0.1.1 - 2012-11-10 added PMF() and CDF()
|
2013-09-29 19:32:20 +02:00
|
|
|
// 0.1.2 - 2012-12-23 changed float to double; some comments
|
2017-07-16 08:04:20 +02:00
|
|
|
// 0.1.3 - 2013-09-29 testing a lot & refactoring
|
|
|
|
// 0.1.4 - 2015-03-06 stricter interface
|
|
|
|
// 0.1.5 - 2017-07-16 refactor, support for > 256 buckets; prevent alloc errors
|
2017-07-27 13:13:41 +02:00
|
|
|
// 0.1.6 - 2017-07-27 revert double to float (issue #33)
|
2013-09-29 19:28:34 +02:00
|
|
|
//
|
|
|
|
// Released to the public domain
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "histogram.h"
|
|
|
|
|
2017-07-27 13:13:41 +02:00
|
|
|
Histogram::Histogram(const int16_t len, float *bounds)
|
2013-09-29 19:28:34 +02:00
|
|
|
{
|
2017-07-16 08:04:20 +02:00
|
|
|
_bounds = bounds;
|
|
|
|
_len = len + 1;
|
|
|
|
_data = (int32_t *) malloc((_len) * sizeof(int32_t));
|
|
|
|
if (_data) clear();
|
|
|
|
else _len = 0;
|
|
|
|
_cnt = 0;
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Histogram::~Histogram()
|
|
|
|
{
|
2017-07-16 08:04:20 +02:00
|
|
|
if (_data) free(_data);
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// resets all counters
|
|
|
|
void Histogram::clear()
|
2013-09-29 21:55:02 +02:00
|
|
|
{
|
2017-07-16 08:04:20 +02:00
|
|
|
for (int16_t i = 0; i < _len; i++) _data[i] = 0;
|
|
|
|
_cnt = 0;
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// adds a new value to the histogram - increasing
|
2017-07-27 13:13:41 +02:00
|
|
|
void Histogram::add(const float f)
|
2013-09-29 19:28:34 +02:00
|
|
|
{
|
2017-07-16 08:04:20 +02:00
|
|
|
if (_len > 0)
|
|
|
|
{
|
2013-09-29 21:55:02 +02:00
|
|
|
_data[find(f)]++;
|
|
|
|
_cnt++;
|
2017-07-16 08:04:20 +02:00
|
|
|
}
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// adds a new value to the histogram - decreasing
|
2017-07-27 13:13:41 +02:00
|
|
|
void Histogram::sub(const float f)
|
2013-09-29 19:28:34 +02:00
|
|
|
{
|
2017-07-16 08:04:20 +02:00
|
|
|
if (_len > 0)
|
|
|
|
{
|
2013-09-29 21:55:02 +02:00
|
|
|
_data[find(f)]--;
|
2017-07-16 08:04:20 +02:00
|
|
|
_cnt++;
|
|
|
|
}
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
2013-09-29 19:32:20 +02:00
|
|
|
// returns the count of a bucket
|
2017-07-16 08:04:20 +02:00
|
|
|
int32_t Histogram::bucket(const int16_t idx)
|
2013-09-29 19:28:34 +02:00
|
|
|
{
|
2017-07-16 08:04:20 +02:00
|
|
|
if (idx > _len) return 0;
|
|
|
|
return _data[idx];
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// returns the relative frequency of a bucket
|
2017-07-27 13:13:41 +02:00
|
|
|
float Histogram::frequency(const int16_t idx)
|
2013-09-29 19:28:34 +02:00
|
|
|
{
|
2017-07-16 08:04:20 +02:00
|
|
|
if (_cnt == 0 || _len == 0) return NAN;
|
2017-07-27 13:13:41 +02:00
|
|
|
|
2017-07-16 08:04:20 +02:00
|
|
|
if (idx > _len) return 0; // diff with PMF
|
|
|
|
return (1.0 * _data[idx]) / _cnt;
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
2013-09-29 21:55:02 +02:00
|
|
|
// EXPERIMENTAL
|
2013-09-29 19:32:20 +02:00
|
|
|
// returns the probability of the bucket of a value
|
2017-07-27 13:13:41 +02:00
|
|
|
float Histogram::PMF(const float val)
|
2013-09-29 19:32:20 +02:00
|
|
|
{
|
2017-07-16 08:04:20 +02:00
|
|
|
if (_cnt == 0 || _len == 0) return NAN;
|
2017-07-27 13:13:41 +02:00
|
|
|
|
2017-07-16 08:04:20 +02:00
|
|
|
int16_t idx = find(val);
|
|
|
|
return (1.0 * _data[idx]) / _cnt;
|
2013-09-29 19:32:20 +02:00
|
|
|
}
|
|
|
|
|
2013-09-29 21:55:02 +02:00
|
|
|
// EXPERIMENTAL
|
|
|
|
// returns the cummulative probability of
|
2013-09-29 19:32:20 +02:00
|
|
|
// values <= value
|
2017-07-27 13:13:41 +02:00
|
|
|
float Histogram::CDF(const float val)
|
2013-09-29 19:28:34 +02:00
|
|
|
{
|
2017-07-16 08:04:20 +02:00
|
|
|
if (_cnt == 0 || _len == 0) return NAN;
|
2017-07-27 13:13:41 +02:00
|
|
|
|
2017-07-16 08:04:20 +02:00
|
|
|
int16_t idx = find(val);
|
|
|
|
int32_t sum = 0;
|
|
|
|
for (int16_t i = 0; i <= idx; i++)
|
|
|
|
{
|
|
|
|
sum += _data[i];
|
|
|
|
}
|
|
|
|
return (1.0 * sum) / _cnt;
|
2013-09-29 19:32:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// EXPERIMENTAL
|
2013-09-29 21:55:02 +02:00
|
|
|
// returns the value of the original array for
|
2013-09-29 19:32:20 +02:00
|
|
|
// which the CDF is at least prob.
|
2017-07-27 13:13:41 +02:00
|
|
|
float Histogram::VAL(const float prob)
|
2013-09-29 19:32:20 +02:00
|
|
|
{
|
2017-07-16 08:04:20 +02:00
|
|
|
if (_cnt == 0 || _len == 0) return NAN;
|
2017-07-27 13:13:41 +02:00
|
|
|
float p = prob;
|
2017-07-16 08:04:20 +02:00
|
|
|
if (p < 0.0) p = 0.0;
|
|
|
|
if (p > 1.0) p = 1.0;
|
|
|
|
|
2017-07-27 13:13:41 +02:00
|
|
|
float probability = p * _cnt;
|
2017-07-16 08:04:20 +02:00
|
|
|
int32_t sum = 0;
|
|
|
|
for (int16_t i = 0; i < _len; i++)
|
|
|
|
{
|
|
|
|
sum += _data[i];
|
|
|
|
if (sum >= probability && (i <(_len-1)) ) return _bounds[i];
|
|
|
|
}
|
|
|
|
return INFINITY;
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
2013-09-29 21:55:02 +02:00
|
|
|
// returns the bucket number for value val
|
2017-07-27 13:13:41 +02:00
|
|
|
int16_t Histogram::find(const float val)
|
2013-09-29 19:28:34 +02:00
|
|
|
{
|
2017-07-16 08:04:20 +02:00
|
|
|
if (_len <= 0) return -1;
|
2017-07-27 13:13:41 +02:00
|
|
|
|
2017-07-16 08:04:20 +02:00
|
|
|
for (int16_t i = 0; i < (_len-1); i++)
|
|
|
|
{
|
|
|
|
if (_bounds[i] >= val) return i;
|
|
|
|
}
|
|
|
|
return _len-1; // len?
|
|
|
|
// int16_t i = 0;
|
|
|
|
// while ((i < (_len-1)) && (_bounds[i] < val)) i++;
|
|
|
|
// return i;
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// END OF FILE
|