1234567891011121314151617181920212223242526272829303132333435363738394041424344454647 |
- /* vim: set expandtab ts=4 sw=4: */
- /*
- * You may redistribute this program and/or modify it under the terms of
- * the GNU General Public License as published by the Free Software Foundation,
- * either version 3 of the License, or (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program. If not, see <https://www.gnu.org/licenses/>.
- */
- struct Average
- {
- uint64_t shiftedAvg;
- int shiftCount;
- int weight;
- };
- static inline void Average_init(struct Average* average, uint64_t factor, uint64_t weight)
- {
- average->weight = Bits_log2x64(weight);
- average->shiftCount = Bits_log2x64(factor);
- average->shiftedAvg = 0;
- // must be a power of 2
- Assert_true((1 << average->shiftCount) == factor);
- Assert_true((1 << average->weight) == weight);
- }
- static inline void Average_accumulate(struct Average* average, uint64_t value)
- {
- if (!average->shiftedAvg) {
- average->shiftedAvg = value << average->shiftCount;
- return;
- }
- uint64_t sha = average->shiftedAvg;
- sha = ( (sha << average->weight) - sha + (value << avg->shiftCount) ) >> average->weight;
- average->shiftedAvg = sha;
- }
- static inline uint64_t Average_value(struct Average* average)
- {
- return avg->shiftedAvg >> avg->shiftCount;
- }
|