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
|
2023-07-24 12:51:25 +02:00
|
|
|
// VERSION: 0.3.4
|
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
|
2021-12-19 13:52:01 +01:00
|
|
|
|
2013-09-29 19:28:34 +02:00
|
|
|
|
|
|
|
#include "histogram.h"
|
|
|
|
|
2021-11-04 12:32:04 +01:00
|
|
|
|
|
|
|
Histogram::Histogram(const uint16_t length, float *bounds)
|
2013-09-29 19:28:34 +02:00
|
|
|
{
|
2017-07-16 08:04:20 +02:00
|
|
|
_bounds = bounds;
|
2021-11-04 12:32:04 +01:00
|
|
|
_length = length + 1;
|
|
|
|
_data = (int32_t *) malloc((_length) * sizeof(int32_t));
|
2023-02-22 10:34:45 +01:00
|
|
|
if (_data != NULL)
|
|
|
|
{
|
|
|
|
clear();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_length = 0;
|
|
|
|
}
|
2021-11-04 12:32:04 +01:00
|
|
|
_count = 0;
|
2023-07-24 12:51:25 +02:00
|
|
|
_status = HISTO_OK;
|
|
|
|
_maxBucket = 2147483647;
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
2021-11-04 12:32:04 +01:00
|
|
|
|
2013-09-29 19:28:34 +02:00
|
|
|
Histogram::~Histogram()
|
|
|
|
{
|
2023-02-22 10:34:45 +01:00
|
|
|
if (_data != NULL)
|
|
|
|
{
|
|
|
|
free(_data);
|
|
|
|
}
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
2021-11-04 12:32:04 +01:00
|
|
|
|
2022-11-09 10:42:12 +01:00
|
|
|
// resets all counters to value (default 0)
|
2023-07-24 12:51:25 +02:00
|
|
|
uint8_t Histogram::clear(int32_t value)
|
2013-09-29 21:55:02 +02:00
|
|
|
{
|
2023-02-22 10:34:45 +01:00
|
|
|
for (uint16_t i = 0; i < _length; i++)
|
|
|
|
{
|
|
|
|
_data[i] = value;
|
|
|
|
}
|
2021-11-04 12:32:04 +01:00
|
|
|
_count = 0;
|
2023-07-24 12:51:25 +02:00
|
|
|
_status = HISTO_OK;
|
|
|
|
if (value == _maxBucket) _status = HISTO_FULL;
|
|
|
|
return _status;
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
2021-11-04 12:32:04 +01:00
|
|
|
|
2023-07-24 12:51:25 +02:00
|
|
|
uint8_t Histogram::setBucket(const uint16_t index, int32_t value)
|
2023-02-22 10:34:45 +01:00
|
|
|
{
|
|
|
|
_data[index] = value;
|
2023-07-24 12:51:25 +02:00
|
|
|
_status = HISTO_OK;
|
|
|
|
if (value == _maxBucket) _status = HISTO_FULL;
|
|
|
|
return _status;
|
2023-02-22 10:34:45 +01:00
|
|
|
}
|
2022-11-09 10:42:12 +01:00
|
|
|
|
|
|
|
|
|
|
|
// adds a new value to the histogram - increasing
|
2023-07-24 12:51:25 +02:00
|
|
|
uint8_t Histogram::add(const float value)
|
2013-09-29 19:28:34 +02:00
|
|
|
{
|
2023-07-24 12:51:25 +02:00
|
|
|
if (_length == 0)
|
2017-07-16 08:04:20 +02:00
|
|
|
{
|
2023-07-24 12:51:25 +02:00
|
|
|
_status = HISTO_ERR_LENGTH;
|
|
|
|
return _status;
|
2017-07-16 08:04:20 +02:00
|
|
|
}
|
2023-07-24 12:51:25 +02:00
|
|
|
uint16_t index = find(value);
|
|
|
|
if (_data[index] == _maxBucket)
|
|
|
|
{
|
|
|
|
_status = HISTO_ERR_FULL;
|
|
|
|
return _status;
|
|
|
|
}
|
|
|
|
_data[index]++;
|
|
|
|
_count++;
|
|
|
|
_status = HISTO_OK;
|
|
|
|
if (_data[index] == _maxBucket) _status = HISTO_FULL;
|
|
|
|
return _status;
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
2021-11-04 12:32:04 +01:00
|
|
|
|
2022-11-09 10:42:12 +01:00
|
|
|
// adds a new value to the histogram - decreasing
|
2023-07-24 12:51:25 +02:00
|
|
|
uint8_t Histogram::sub(const float value)
|
2013-09-29 19:28:34 +02:00
|
|
|
{
|
2023-07-24 12:51:25 +02:00
|
|
|
if (_length == 0)
|
2017-07-16 08:04:20 +02:00
|
|
|
{
|
2023-07-24 12:51:25 +02:00
|
|
|
_status = HISTO_ERR_LENGTH;
|
|
|
|
return _status;
|
2017-07-16 08:04:20 +02:00
|
|
|
}
|
2023-07-24 12:51:25 +02:00
|
|
|
uint16_t index = find(value);
|
|
|
|
if (_data[index] == -_maxBucket)
|
|
|
|
{
|
|
|
|
_status = HISTO_ERR_FULL;
|
|
|
|
return _status;
|
|
|
|
}
|
|
|
|
_data[index]--;
|
|
|
|
_count++;
|
|
|
|
_status = HISTO_OK;
|
|
|
|
if (_data[index] == _maxBucket) _status = HISTO_FULL;
|
|
|
|
return _status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint8_t Histogram::status()
|
|
|
|
{
|
|
|
|
return _status;
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
2021-11-04 12:32:04 +01:00
|
|
|
|
2022-11-09 10:42:12 +01:00
|
|
|
// number of buckets
|
2023-02-22 10:34:45 +01:00
|
|
|
uint16_t Histogram::size()
|
|
|
|
{
|
|
|
|
return _length;
|
|
|
|
}
|
2022-11-09 10:42:12 +01:00
|
|
|
|
|
|
|
|
|
|
|
// number of values added to all buckets
|
|
|
|
uint32_t Histogram::count()
|
2023-02-22 10:34:45 +01:00
|
|
|
{
|
2022-11-09 10:42:12 +01:00
|
|
|
return _count;
|
2023-02-22 10:34:45 +01:00
|
|
|
}
|
2022-11-09 10:42:12 +01:00
|
|
|
|
|
|
|
|
|
|
|
// returns the count of a bucket
|
2021-11-04 12:32:04 +01:00
|
|
|
int32_t Histogram::bucket(const uint16_t index)
|
2013-09-29 19:28:34 +02:00
|
|
|
{
|
2021-11-04 12:32:04 +01:00
|
|
|
if (index > _length) return 0;
|
|
|
|
return _data[index];
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
2021-11-04 12:32:04 +01:00
|
|
|
|
2022-11-09 10:42:12 +01:00
|
|
|
// returns the relative frequency of a bucket
|
2021-11-04 12:32:04 +01:00
|
|
|
float Histogram::frequency(const uint16_t index)
|
2013-09-29 19:28:34 +02:00
|
|
|
{
|
2021-11-04 12:32:04 +01:00
|
|
|
if ((_count == 0) || (_length == 0)) return NAN;
|
2017-07-27 13:13:41 +02:00
|
|
|
|
2023-02-22 10:34:45 +01:00
|
|
|
if (index > _length) return 0; // differs from PMF()
|
2021-11-04 12:32:04 +01:00
|
|
|
return (1.0 * _data[index]) / _count;
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
2021-11-04 12:32:04 +01:00
|
|
|
|
2022-11-09 10:42:12 +01:00
|
|
|
// EXPERIMENTAL
|
|
|
|
// returns the probability of the bucket of a value
|
2021-11-04 12:32:04 +01:00
|
|
|
float Histogram::PMF(const float value)
|
2013-09-29 19:32:20 +02:00
|
|
|
{
|
2021-11-04 12:32:04 +01:00
|
|
|
if ((_count == 0) || (_length == 0)) return NAN;
|
2017-07-27 13:13:41 +02:00
|
|
|
|
2021-11-04 12:32:04 +01:00
|
|
|
uint16_t index = find(value);
|
|
|
|
return (1.0 * _data[index]) / _count;
|
2013-09-29 19:32:20 +02:00
|
|
|
}
|
|
|
|
|
2021-11-04 12:32:04 +01:00
|
|
|
|
2022-11-09 10:42:12 +01:00
|
|
|
// EXPERIMENTAL
|
|
|
|
// returns the cumulative probability of
|
|
|
|
// values <= value
|
2021-11-04 12:32:04 +01:00
|
|
|
float Histogram::CDF(const float value)
|
2013-09-29 19:28:34 +02:00
|
|
|
{
|
2021-11-04 12:32:04 +01:00
|
|
|
if ((_count == 0) || (_length == 0)) return NAN;
|
2017-07-27 13:13:41 +02:00
|
|
|
|
2022-11-09 10:42:12 +01:00
|
|
|
// TODO: could be done in one loop?
|
2021-11-04 12:32:04 +01:00
|
|
|
uint16_t index = find(value);
|
|
|
|
int32_t sum = 0;
|
|
|
|
for (uint16_t i = 0; i <= index; i++)
|
2017-07-16 08:04:20 +02:00
|
|
|
{
|
|
|
|
sum += _data[i];
|
|
|
|
}
|
2021-11-04 12:32:04 +01:00
|
|
|
return (1.0 * sum) / _count;
|
2013-09-29 19:32:20 +02:00
|
|
|
}
|
|
|
|
|
2021-11-04 12:32:04 +01:00
|
|
|
|
2022-11-09 10:42:12 +01:00
|
|
|
// EXPERIMENTAL
|
|
|
|
// returns the value of the original array for
|
2023-02-22 10:34:45 +01:00
|
|
|
// which the CDF is at least probability.
|
|
|
|
// must start at 0.
|
|
|
|
float Histogram::VAL(const float probability)
|
2013-09-29 19:32:20 +02:00
|
|
|
{
|
2021-11-04 12:32:04 +01:00
|
|
|
if ((_count == 0) || (_length == 0)) return NAN;
|
|
|
|
|
2023-02-22 10:34:45 +01:00
|
|
|
float prob = probability;
|
|
|
|
if (prob < 0.0) prob = 0.0;
|
|
|
|
else if (prob > 1.0) prob = 1.0;
|
2017-07-16 08:04:20 +02:00
|
|
|
|
2023-02-22 10:34:45 +01:00
|
|
|
prob *= _count;
|
2017-07-16 08:04:20 +02:00
|
|
|
int32_t sum = 0;
|
2021-11-04 12:32:04 +01:00
|
|
|
for (uint16_t i = 0; i < _length; i++)
|
2017-07-16 08:04:20 +02:00
|
|
|
{
|
|
|
|
sum += _data[i];
|
2023-02-22 10:34:45 +01:00
|
|
|
if ((sum >= prob) && (i < (_length - 1)))
|
2021-11-04 12:32:04 +01:00
|
|
|
{
|
|
|
|
return _bounds[i];
|
|
|
|
}
|
2017-07-16 08:04:20 +02:00
|
|
|
}
|
|
|
|
return INFINITY;
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
2021-11-04 12:32:04 +01:00
|
|
|
|
2023-07-24 12:51:25 +02:00
|
|
|
int32_t Histogram::sum()
|
|
|
|
{
|
|
|
|
int32_t _sum = 0;
|
|
|
|
for (uint16_t i = 0; i < _length; i++)
|
|
|
|
{
|
|
|
|
_sum += _data[i];
|
|
|
|
}
|
|
|
|
return _sum;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-02-22 10:34:45 +01:00
|
|
|
// returns the bucket number for value
|
2021-11-04 12:32:04 +01:00
|
|
|
// - binary search, more memory ; faster for #buckets > 20 ?
|
|
|
|
// uint16_t Histogram::find(const float value)
|
|
|
|
// {
|
|
|
|
// if (_length <= 0) return -1;
|
|
|
|
|
|
|
|
// uint16_t low = 0, high = _length;
|
|
|
|
// uint16_t mid;
|
|
|
|
// while (high - low > 1)
|
|
|
|
// {
|
|
|
|
// mid = (low + high)/2;
|
|
|
|
// if (_bounds[mid] > value)
|
|
|
|
// {
|
|
|
|
// high = mid;
|
|
|
|
// }
|
|
|
|
// else
|
|
|
|
// {
|
|
|
|
// low = mid;
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// if (_bounds[mid] > value) return mid;
|
|
|
|
// return _length - 1;
|
|
|
|
// }
|
|
|
|
|
|
|
|
|
2022-11-09 10:42:12 +01:00
|
|
|
// returns the bucket number for value
|
2023-02-22 10:34:45 +01:00
|
|
|
// must start at 0.
|
2021-11-04 12:32:04 +01:00
|
|
|
uint16_t Histogram::find(const float value)
|
|
|
|
{
|
|
|
|
if (_length <= 0) return -1;
|
|
|
|
|
|
|
|
for (uint16_t i = 0; i < (_length - 1); i++)
|
|
|
|
{
|
|
|
|
if (_bounds[i] >= value)
|
|
|
|
{
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return _length - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-11-09 10:42:12 +01:00
|
|
|
// returns the (first) index of the bucket with minimum value.
|
2021-11-04 12:32:04 +01:00
|
|
|
uint16_t Histogram::findMin()
|
|
|
|
{
|
|
|
|
if (_length <= 0) return -1;
|
|
|
|
uint16_t index = 0;
|
2023-02-22 10:34:45 +01:00
|
|
|
uint16_t n = _length;
|
|
|
|
while (n > 1)
|
2021-11-04 12:32:04 +01:00
|
|
|
{
|
2023-02-22 10:34:45 +01:00
|
|
|
n--;
|
|
|
|
if (_data[n] < _data[index]) index = n;
|
2021-11-04 12:32:04 +01:00
|
|
|
}
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-11-09 10:42:12 +01:00
|
|
|
// returns the (first) index of the bucket with maximum value.
|
2021-11-04 12:32:04 +01:00
|
|
|
uint16_t Histogram::findMax()
|
|
|
|
{
|
|
|
|
if (_length <= 0) return -1;
|
|
|
|
uint16_t index = 0;
|
2023-02-22 10:34:45 +01:00
|
|
|
uint16_t n = _length;
|
|
|
|
while (n > 1)
|
2021-11-04 12:32:04 +01:00
|
|
|
{
|
2023-02-22 10:34:45 +01:00
|
|
|
n--;
|
|
|
|
if (_data[n] > _data[index]) index = n;
|
2021-11-04 12:32:04 +01:00
|
|
|
}
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-02-22 10:34:45 +01:00
|
|
|
// returns the number of buckets with an exact level.
|
2021-11-04 12:32:04 +01:00
|
|
|
uint16_t Histogram::countLevel(const int32_t level)
|
|
|
|
{
|
|
|
|
if (_length <= 0) return -1;
|
|
|
|
|
|
|
|
uint16_t buckets = 0;
|
2023-02-22 10:34:45 +01:00
|
|
|
uint16_t n = _length;
|
|
|
|
while (n > 0)
|
2021-11-04 12:32:04 +01:00
|
|
|
{
|
2023-02-22 10:34:45 +01:00
|
|
|
n--;
|
|
|
|
if (_data[n] == level) buckets++;
|
2021-11-04 12:32:04 +01:00
|
|
|
}
|
|
|
|
return buckets;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-11-09 10:42:12 +01:00
|
|
|
// returns the number of buckets above a certain level.
|
2021-11-04 12:32:04 +01:00
|
|
|
uint16_t Histogram::countAbove(const int32_t level)
|
2013-09-29 19:28:34 +02:00
|
|
|
{
|
2021-11-04 12:32:04 +01:00
|
|
|
if (_length <= 0) return -1;
|
2017-07-27 13:13:41 +02:00
|
|
|
|
2021-11-04 12:32:04 +01:00
|
|
|
uint16_t buckets = 0;
|
2023-02-22 10:34:45 +01:00
|
|
|
uint16_t n = _length;
|
|
|
|
while (n > 0)
|
2017-07-16 08:04:20 +02:00
|
|
|
{
|
2023-02-22 10:34:45 +01:00
|
|
|
n--;
|
|
|
|
if (_data[n] > level) buckets++;
|
2017-07-16 08:04:20 +02:00
|
|
|
}
|
2021-11-04 12:32:04 +01:00
|
|
|
return buckets;
|
2013-09-29 19:28:34 +02:00
|
|
|
}
|
|
|
|
|
2021-11-04 12:32:04 +01:00
|
|
|
|
2022-11-09 10:42:12 +01:00
|
|
|
// returns the number of buckets below a certain level.
|
2021-11-04 12:32:04 +01:00
|
|
|
uint16_t Histogram::countBelow(const int32_t level)
|
|
|
|
{
|
|
|
|
if (_length <= 0) return -1;
|
|
|
|
|
|
|
|
uint16_t buckets = 0;
|
2023-02-22 10:34:45 +01:00
|
|
|
uint16_t n = _length;
|
|
|
|
while (n > 0)
|
2021-11-04 12:32:04 +01:00
|
|
|
{
|
2023-02-22 10:34:45 +01:00
|
|
|
n--;
|
|
|
|
if (_data[n] < level) buckets++;
|
2021-11-04 12:32:04 +01:00
|
|
|
}
|
|
|
|
return buckets;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-07-24 12:51:25 +02:00
|
|
|
|
|
|
|
// experimental use with care
|
|
|
|
int32_t Histogram::getMaxBucket()
|
|
|
|
{
|
|
|
|
return _maxBucket;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Histogram::setMaxBucket(int32_t value)
|
|
|
|
{
|
|
|
|
_maxBucket = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-04 12:32:04 +01:00
|
|
|
//////////////////////////////////////////////////////////////
|
|
|
|
//
|
2023-07-24 12:51:25 +02:00
|
|
|
// DERIVED CLASS - HISTOGRAM16
|
2021-11-04 12:32:04 +01:00
|
|
|
//
|
|
|
|
Histogram16::Histogram16(const uint16_t length, float *bounds) : Histogram(length, bounds)
|
|
|
|
{
|
2023-07-24 12:51:25 +02:00
|
|
|
_bounds = bounds;
|
|
|
|
_length = length + 1;
|
|
|
|
_data = (int16_t *) malloc((_length) * sizeof(int16_t));
|
|
|
|
if (_data != NULL)
|
|
|
|
{
|
|
|
|
clear();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_length = 0;
|
|
|
|
}
|
|
|
|
_count = 0;
|
|
|
|
_status = HISTO_OK;
|
|
|
|
_maxBucket = 32767;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Histogram16::~Histogram16()
|
|
|
|
{
|
|
|
|
if (_data) free(_data);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint8_t Histogram16::setBucket(const uint16_t index, int16_t value)
|
|
|
|
{
|
|
|
|
_data[index] = value;
|
|
|
|
_status = HISTO_OK;
|
|
|
|
if (value == _maxBucket) _status = HISTO_FULL;
|
|
|
|
return _status;
|
2023-02-22 10:34:45 +01:00
|
|
|
}
|
2021-11-04 12:32:04 +01:00
|
|
|
|
|
|
|
|
2023-07-24 12:51:25 +02:00
|
|
|
//////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// DERIVED CLASS - HISTOGRAM8
|
|
|
|
//
|
2021-11-04 12:32:04 +01:00
|
|
|
Histogram8::Histogram8(const uint16_t length, float *bounds) : Histogram(length, bounds)
|
|
|
|
{
|
2023-07-24 12:51:25 +02:00
|
|
|
_bounds = bounds;
|
|
|
|
_length = length + 1;
|
|
|
|
_data = (int8_t *) malloc((_length) * sizeof(int8_t));
|
|
|
|
if (_data != NULL)
|
|
|
|
{
|
|
|
|
clear();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_length = 0;
|
|
|
|
}
|
|
|
|
_count = 0;
|
|
|
|
_status = HISTO_OK;
|
|
|
|
_maxBucket = 127;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Histogram8::~Histogram8()
|
|
|
|
{
|
|
|
|
if (_data) free(_data);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint8_t Histogram8::setBucket(const uint16_t index, int8_t value)
|
|
|
|
{
|
|
|
|
_data[index] = value;
|
|
|
|
_status = HISTO_OK;
|
|
|
|
if (value == _maxBucket) _status = HISTO_FULL;
|
|
|
|
return _status;
|
2023-02-22 10:34:45 +01:00
|
|
|
}
|
2021-11-04 12:32:04 +01:00
|
|
|
|
2022-11-09 10:42:12 +01:00
|
|
|
|
2023-02-22 10:34:45 +01:00
|
|
|
// -- END OF FILE --
|
2021-11-04 12:32:04 +01:00
|
|
|
|