-
Notifications
You must be signed in to change notification settings - Fork 190
/
Copy pathRollingAverage.h
213 lines (174 loc) · 6 KB
/
RollingAverage.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
/*
* RollingAverage.h
*
Draws on Arduino Smoothing example,
Created 22 April 2007
By David A. Mellis <[email protected]>
modified 9 Apr 2012
by Tom Igoe
http://www.arduino.cc/en/Tutorial/Smoothing
* This file is part of Mozzi.
*
* Copyright 2013-2024 Tim Barrass and the Mozzi Team
*
* Mozzi is licensed under the GNU Lesser General Public Licence (LGPL) Version 2.1 or later.
*
*/
#ifndef ROLLINGAVERAGE_H
#define ROLLINGAVERAGE_H
#include "mozzi_utils.h" // for trailingZeros()
/** @ingroup sensortools
Calculates a running average over a
specified number of the most recent readings.
Like Smooth(), this is good for smoothing analog inputs in updateControl().
@tparam WINDOW_LENGTH the number of readings to include in the rolling average.
It must be a power of two (unless you're averaging floats). The higher the
number, the more the readings will be smoothed, but the slower the output
will respond to the input.
*/
template <class T, int WINDOW_LENGTH>
class
RollingAverage {
public:
/** Constructor.
@tparam T the type of numbers to average, eg. int, unsigned int, float etc. It will be relatively slow with
floating point numbers, as it will use a divide operation for the averaging.
Nevertheless, there might be a time when it's useful.
@tparam WINDOW_LENGTH the number of readings to keep track of. It must be a power of two (unless
you're averaging floats). The higher the number, the more the readings will be
smoothed, but the slower the output will respond to the input.
@note Watch out for overflows!
*/
RollingAverage():index(0),total(0), WINDOW_LENGTH_AS_RSHIFT(trailingZerosConst(WINDOW_LENGTH))
{
// initialize all the readings to 0:
for (int thisReading = 0; thisReading < WINDOW_LENGTH; thisReading++)
readings[thisReading] = 0;
}
/** Give the average of the last WINDOW_LENGTH.
@param input a control signal such as an analog input which needs smoothing.
@return the smoothed result.
@note unsigned int timing 5.7us
*/
T next(T input)
{
return add(input)>>WINDOW_LENGTH_AS_RSHIFT;
}
protected:
inline
T add(T input){
// out with the old
total -= readings[index];
// in with the new
total += input;
readings[index] = input;
// advance and wrap index
++index &= WINDOW_LENGTH -1;
return total;
}
private:
T readings[WINDOW_LENGTH]; // the readings from the analog input
unsigned int index; // the index of the current reading
long total; // the running total
const uint8_t WINDOW_LENGTH_AS_RSHIFT;
};
// no need to show the specialisations
/** @cond */
/** unsigned int partial specialisation of RollingAverage template.
This is needed because unsigned types need to remind (unsigned) for rshift.
*/
template <int WINDOW_LENGTH>
class RollingAverage <unsigned int, WINDOW_LENGTH>
{
public:
/** Constructor.
@tparam WINDOW_LENGTH A power of two, the number of readings to keep track of.
The higher the number, the more the readings will be smoothed, but the slower the output will
respond to the input.
@note The internal total of all the values being averaged is held in a long (4 uint8_t) integer, to avoid overflowing.
However, watch out for overflows if you are averaging a long number types!
*/
RollingAverage():index(0),total(0), WINDOW_LENGTH_AS_RSHIFT(trailingZerosConst(WINDOW_LENGTH))
{
// initialize all the readings to 0:
for (int thisReading = 0; thisReading < WINDOW_LENGTH; thisReading++)
readings[thisReading] = 0;
}
/** Give the average of the last WINDOW_LENGTH.
@param a control signal such as an analog input which needs smoothing.
@return the smoothed result.
@note timing for int 6us
*/
unsigned int next(unsigned int input)
{
// calculate the average:
// this unsigned cast is the only difference between the int and unsigned int specialisations
// it tells the shift not to sign extend in from the left
return (unsigned)add(input)>>WINDOW_LENGTH_AS_RSHIFT;
}
protected:
inline
unsigned int add(unsigned int input){
// out with the old
total -= readings[index];
// in with the new
total += input;
readings[index] = input;
// advance and wrap index
++index &= WINDOW_LENGTH -1;
return total;
}
private:
unsigned int readings[WINDOW_LENGTH]; // the readings from the analog input
unsigned int index; // the index of the current reading
long total; // the running total
const uint8_t WINDOW_LENGTH_AS_RSHIFT;
};
/** float partial specialisation of RollingAverage template*/
template <int WINDOW_LENGTH>
class RollingAverage <float, WINDOW_LENGTH>
{
public:
/** Constructor.
@tparam WINDOW_LENGTH A power of two, the number of readings to keep track of.
The higher the number, the more the readings will be smoothed, but the slower the output will
respond to the input.
@note The internal total of all the values being averaged is held in a long (4 uint8_t) integer, to avoid overflowing.
However, watch out for overflows if you are averaging a long number types!
*/
RollingAverage():index(0),total(0.0)
{
// initialize all the readings to 0:
for (int thisReading = 0; thisReading < WINDOW_LENGTH; thisReading++)
readings[thisReading] = 0.0;
}
/** Give the average of the last WINDOW_LENGTH.
@param a control signal such as an analog input which needs smoothing.
@return the smoothed result.
@note timing for float 37us
*/
float next(float input)
{
// out with the old
total -= readings[index];
// in with the new
total += input;
readings[index] = input;
// advance and wrap index
++index &= WINDOW_LENGTH -1;
// calculate the average:
return total/WINDOW_LENGTH;
}
private:
float readings[WINDOW_LENGTH]; // the readings from the analog input
unsigned int index; // the index of the current reading
float total; // the running total
};
// no need to show the specialisations
/** @endcond
*/
/**
@example 03.Sensors/Knob_LDR_x2_WavePacket/Knob_LDR_x2_WavePacket.ino
This example demonstrates the RollingAverage class.
*/
#endif // #ifndef ROLLINGAVERAGE_H