232 lines
6.2 KiB
C
232 lines
6.2 KiB
C
/*
|
|
* QEMU timed average computation
|
|
*
|
|
* Copyright (C) Nodalink, EURL. 2014
|
|
* Copyright (C) Igalia, S.L. 2015
|
|
*
|
|
* Authors:
|
|
* Benoît Canet <benoit.canet@nodalink.com>
|
|
* Alberto Garcia <berto@igalia.com>
|
|
*
|
|
* This program is free sofware: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Sofware Foundation, either version 2 of the License, or
|
|
* (at your option) version 3 or 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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <string.h>
|
|
|
|
#include "qemu/timed-average.h"
|
|
|
|
/* This module computes an average of a set of values within a time
|
|
* window.
|
|
*
|
|
* Algorithm:
|
|
*
|
|
* - Create two windows with a certain expiration period, and
|
|
* offsetted by period / 2.
|
|
* - Each time you want to account a new value, do it in both windows.
|
|
* - The minimum / maximum / average values are always returned from
|
|
* the oldest window.
|
|
*
|
|
* Example:
|
|
*
|
|
* t=0 |t=0.5 |t=1 |t=1.5 |t=2
|
|
* wnd0: [0,0.5)|wnd0: [0.5,1.5) | |wnd0: [1.5,2.5) |
|
|
* wnd1: [0,1) | |wnd1: [1,2) | |
|
|
*
|
|
* Values are returned from:
|
|
*
|
|
* wnd0---------|wnd1------------|wnd0---------|wnd1-------------|
|
|
*/
|
|
|
|
/* Update the expiration of a time window
|
|
*
|
|
* @w: the window used
|
|
* @now: the current time in nanoseconds
|
|
* @period: the expiration period in nanoseconds
|
|
*/
|
|
static void update_expiration(TimedAverageWindow *w, int64_t now,
|
|
int64_t period)
|
|
{
|
|
/* time elapsed since the last theoretical expiration */
|
|
int64_t elapsed = (now - w->expiration) % period;
|
|
/* time remaininging until the next expiration */
|
|
int64_t remaining = period - elapsed;
|
|
/* compute expiration */
|
|
w->expiration = now + remaining;
|
|
}
|
|
|
|
/* Reset a window
|
|
*
|
|
* @w: the window to reset
|
|
*/
|
|
static void window_reset(TimedAverageWindow *w)
|
|
{
|
|
w->min = UINT64_MAX;
|
|
w->max = 0;
|
|
w->sum = 0;
|
|
w->count = 0;
|
|
}
|
|
|
|
/* Get the current window (that is, the one with the earliest
|
|
* expiration time).
|
|
*
|
|
* @ta: the TimedAverage structure
|
|
* @ret: a pointer to the current window
|
|
*/
|
|
static TimedAverageWindow *current_window(TimedAverage *ta)
|
|
{
|
|
return &ta->windows[ta->current];
|
|
}
|
|
|
|
/* Initialize a TimedAverage structure
|
|
*
|
|
* @ta: the TimedAverage structure
|
|
* @clock_type: the type of clock to use
|
|
* @period: the time window period in nanoseconds
|
|
*/
|
|
void timed_average_init(TimedAverage *ta, QEMUClockType clock_type,
|
|
uint64_t period)
|
|
{
|
|
int64_t now = qemu_clock_get_ns(clock_type);
|
|
|
|
/* Returned values are from the oldest window, so they belong to
|
|
* the interval [ta->period/2,ta->period). By adjusting the
|
|
* requested period by 4/3, we guarantee that they're in the
|
|
* interval [2/3 period,4/3 period), closer to the requested
|
|
* period on average */
|
|
ta->period = (uint64_t) period * 4 / 3;
|
|
ta->clock_type = clock_type;
|
|
ta->current = 0;
|
|
|
|
window_reset(&ta->windows[0]);
|
|
window_reset(&ta->windows[1]);
|
|
|
|
/* Both windows are offsetted by half a period */
|
|
ta->windows[0].expiration = now + ta->period / 2;
|
|
ta->windows[1].expiration = now + ta->period;
|
|
}
|
|
|
|
/* Check if the time windows have expired, updating their counters and
|
|
* expiration time if that's the case.
|
|
*
|
|
* @ta: the TimedAverage structure
|
|
* @elapsed: if non-NULL, the elapsed time (in ns) within the current
|
|
* window will be stored here
|
|
*/
|
|
static void check_expirations(TimedAverage *ta, uint64_t *elapsed)
|
|
{
|
|
int64_t now = qemu_clock_get_ns(ta->clock_type);
|
|
int i;
|
|
|
|
assert(ta->period != 0);
|
|
|
|
/* Check if the windows have expired */
|
|
for (i = 0; i < 2; i++) {
|
|
TimedAverageWindow *w = &ta->windows[i];
|
|
if (w->expiration <= now) {
|
|
window_reset(w);
|
|
update_expiration(w, now, ta->period);
|
|
}
|
|
}
|
|
|
|
/* Make ta->current point to the oldest window */
|
|
if (ta->windows[0].expiration < ta->windows[1].expiration) {
|
|
ta->current = 0;
|
|
} else {
|
|
ta->current = 1;
|
|
}
|
|
|
|
/* Calculate the elapsed time within the current window */
|
|
if (elapsed) {
|
|
int64_t remaining = ta->windows[ta->current].expiration - now;
|
|
*elapsed = ta->period - remaining;
|
|
}
|
|
}
|
|
|
|
/* Account a value
|
|
*
|
|
* @ta: the TimedAverage structure
|
|
* @value: the value to account
|
|
*/
|
|
void timed_average_account(TimedAverage *ta, uint64_t value)
|
|
{
|
|
int i;
|
|
check_expirations(ta, NULL);
|
|
|
|
/* Do the accounting in both windows at the same time */
|
|
for (i = 0; i < 2; i++) {
|
|
TimedAverageWindow *w = &ta->windows[i];
|
|
|
|
w->sum += value;
|
|
w->count++;
|
|
|
|
if (value < w->min) {
|
|
w->min = value;
|
|
}
|
|
|
|
if (value > w->max) {
|
|
w->max = value;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Get the minimum value
|
|
*
|
|
* @ta: the TimedAverage structure
|
|
* @ret: the minimum value
|
|
*/
|
|
uint64_t timed_average_min(TimedAverage *ta)
|
|
{
|
|
TimedAverageWindow *w;
|
|
check_expirations(ta, NULL);
|
|
w = current_window(ta);
|
|
return w->min < UINT64_MAX ? w->min : 0;
|
|
}
|
|
|
|
/* Get the average value
|
|
*
|
|
* @ta: the TimedAverage structure
|
|
* @ret: the average value
|
|
*/
|
|
uint64_t timed_average_avg(TimedAverage *ta)
|
|
{
|
|
TimedAverageWindow *w;
|
|
check_expirations(ta, NULL);
|
|
w = current_window(ta);
|
|
return w->count > 0 ? w->sum / w->count : 0;
|
|
}
|
|
|
|
/* Get the maximum value
|
|
*
|
|
* @ta: the TimedAverage structure
|
|
* @ret: the maximum value
|
|
*/
|
|
uint64_t timed_average_max(TimedAverage *ta)
|
|
{
|
|
check_expirations(ta, NULL);
|
|
return current_window(ta)->max;
|
|
}
|
|
|
|
/* Get the sum of all accounted values
|
|
* @ta: the TimedAverage structure
|
|
* @elapsed: if non-NULL, the elapsed time (in ns) will be stored here
|
|
* @ret: the sum of all accounted values
|
|
*/
|
|
uint64_t timed_average_sum(TimedAverage *ta, uint64_t *elapsed)
|
|
{
|
|
TimedAverageWindow *w;
|
|
check_expirations(ta, elapsed);
|
|
w = current_window(ta);
|
|
return w->sum;
|
|
}
|