Philipp Schrader | 29d54f2 | 2016-04-02 22:14:48 +0000 | [diff] [blame] | 1 | #ifndef FRC971_ZEROING_AVERAGER_H_ |
| 2 | #define FRC971_ZEROING_AVERAGER_H_ |
| 3 | |
| 4 | #include <algorithm> |
| 5 | #include <array> |
| 6 | #include <stdint.h> |
| 7 | |
James Kuszmaul | d3f9eb2 | 2020-01-12 15:02:07 -0800 | [diff] [blame] | 8 | #include "Eigen/Dense" |
| 9 | #include "glog/logging.h" |
| 10 | |
Philipp Schrader | 29d54f2 | 2016-04-02 22:14:48 +0000 | [diff] [blame] | 11 | namespace frc971 { |
| 12 | namespace zeroing { |
| 13 | |
| 14 | // Averages a set of given numbers. Numbers are given one at a time. Once full |
| 15 | // the average may be requested. |
James Kuszmaul | d3f9eb2 | 2020-01-12 15:02:07 -0800 | [diff] [blame] | 16 | // TODO(james): Look at deduplicating this with some of the work in the |
| 17 | // MoveDetector. |
| 18 | template <typename Scalar, size_t num_samples, int rows_per_sample = 1> |
Philipp Schrader | 29d54f2 | 2016-04-02 22:14:48 +0000 | [diff] [blame] | 19 | class Averager { |
| 20 | public: |
James Kuszmaul | d3f9eb2 | 2020-01-12 15:02:07 -0800 | [diff] [blame] | 21 | typedef Eigen::Matrix<Scalar, rows_per_sample, 1> Vector; |
James Kuszmaul | b1e2937 | 2020-02-11 16:55:36 -0800 | [diff] [blame] | 22 | Averager() { |
| 23 | for (Vector &datum : data_) { |
| 24 | datum.setZero(); |
| 25 | } |
| 26 | } |
Philipp Schrader | 29d54f2 | 2016-04-02 22:14:48 +0000 | [diff] [blame] | 27 | // Adds one data point to the set of data points to be averaged. |
James Kuszmaul | d3f9eb2 | 2020-01-12 15:02:07 -0800 | [diff] [blame] | 28 | // If more than "num_samples" samples are added, they will start overwriting |
Philipp Schrader | 29d54f2 | 2016-04-02 22:14:48 +0000 | [diff] [blame] | 29 | // the oldest ones. |
James Kuszmaul | d3f9eb2 | 2020-01-12 15:02:07 -0800 | [diff] [blame] | 30 | void AddData(Scalar data) { |
| 31 | CHECK_EQ(1, rows_per_sample); |
| 32 | AddData(Vector(data)); |
| 33 | } |
| 34 | void AddData(const Vector &data) { |
Philipp Schrader | 29d54f2 | 2016-04-02 22:14:48 +0000 | [diff] [blame] | 35 | data_[data_point_index_] = data; |
James Kuszmaul | d3f9eb2 | 2020-01-12 15:02:07 -0800 | [diff] [blame] | 36 | num_data_points_ = std::min(num_samples, num_data_points_ + 1); |
| 37 | data_point_index_ = (data_point_index_ + 1) % num_samples; |
Philipp Schrader | 29d54f2 | 2016-04-02 22:14:48 +0000 | [diff] [blame] | 38 | } |
| 39 | |
| 40 | // Returns the average of the data points. |
James Kuszmaul | d3f9eb2 | 2020-01-12 15:02:07 -0800 | [diff] [blame] | 41 | Vector GetAverage() const { |
Philipp Schrader | 29d54f2 | 2016-04-02 22:14:48 +0000 | [diff] [blame] | 42 | if (num_data_points_ == 0) { |
James Kuszmaul | d3f9eb2 | 2020-01-12 15:02:07 -0800 | [diff] [blame] | 43 | return Vector::Zero(); |
Philipp Schrader | 29d54f2 | 2016-04-02 22:14:48 +0000 | [diff] [blame] | 44 | } |
| 45 | |
James Kuszmaul | d3f9eb2 | 2020-01-12 15:02:07 -0800 | [diff] [blame] | 46 | Vector average; |
| 47 | average.setZero(); |
| 48 | for (const Vector &data : data_) { |
Philipp Schrader | 29d54f2 | 2016-04-02 22:14:48 +0000 | [diff] [blame] | 49 | average += data; |
| 50 | } |
| 51 | return average / num_data_points_; |
| 52 | } |
| 53 | |
James Kuszmaul | d3f9eb2 | 2020-01-12 15:02:07 -0800 | [diff] [blame] | 54 | // Return the difference between the min and max values in the data buffer. |
| 55 | Scalar GetRange() const { |
| 56 | if (num_data_points_ == 0) { |
| 57 | return 0.0; |
| 58 | } |
| 59 | Vector min_value; |
| 60 | min_value.setConstant(std::numeric_limits<Scalar>::max()); |
| 61 | Vector max_value; |
| 62 | max_value.setConstant(std::numeric_limits<Scalar>::lowest()); |
| 63 | // The array will always fill up starting at zero, so we can iterate from |
| 64 | // zero safely. |
| 65 | for (size_t ii = 0; ii < num_data_points_; ++ii) { |
| 66 | const Vector &value = data_[ii]; |
| 67 | min_value = min_value.cwiseMin(value); |
| 68 | max_value = max_value.cwiseMax(value); |
| 69 | } |
| 70 | return (max_value - min_value).maxCoeff(); |
| 71 | } |
Philipp Schrader | 29d54f2 | 2016-04-02 22:14:48 +0000 | [diff] [blame] | 72 | |
James Kuszmaul | d3f9eb2 | 2020-01-12 15:02:07 -0800 | [diff] [blame] | 73 | void Reset() { |
| 74 | num_data_points_ = 0; |
| 75 | data_point_index_ = 0; |
| 76 | } |
| 77 | |
| 78 | // Returns true when we've gathered num_samples data points. |
| 79 | bool full() const { return num_data_points_ >= num_samples; }; |
| 80 | |
| 81 | size_t size() const { return num_samples; } |
Philipp Schrader | 29d54f2 | 2016-04-02 22:14:48 +0000 | [diff] [blame] | 82 | |
| 83 | private: |
| 84 | // Data points to be averaged. |
James Kuszmaul | d3f9eb2 | 2020-01-12 15:02:07 -0800 | [diff] [blame] | 85 | std::array<Vector, num_samples> data_; |
Philipp Schrader | 29d54f2 | 2016-04-02 22:14:48 +0000 | [diff] [blame] | 86 | // Which data point in "data_" will be filled in next. |
| 87 | size_t data_point_index_ = 0; |
| 88 | // Number of data points added via AddData(). |
| 89 | size_t num_data_points_ = 0; |
| 90 | }; |
| 91 | |
| 92 | } // namespace zeroing |
| 93 | } // namespace frc971 |
| 94 | |
| 95 | #endif // FRC971_ZEROING_AVERAGER_H_ |