mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-01-12 07:26:15 +01:00
tests/performance: ensure to always warm up for 2 seconds
Despite all the efforts, there still seems to be a lot of noise in the performance measurement. Especially, the first iterations seem to run faster. Maybe that is because the kernel didn't yet determine that the process is CPU bound and is less likely to schedule it out Or maybe it's because burning the cycles heats up the CPU and it gets throttled after a while. It's unclear why, and it's even unclear whether this really happens. But from my observations, it seems to do. Hence, more warm up. - the first time we enter the test, ensure that we keep the CPU busy for at 2 seconds. This additional warm up (WARM_UP_ALWAYS_SEC) is global, and not per test. - for each test, ignore the first 5% of the runs. It seems those tend to run faster, thus skewing the results. - if the user specifies a "--factor", the warm up operations are the same and independent from external factors (such as time measurements). Note that this matters the most, when you want to run the executable twice in a row and compare the results.
This commit is contained in:
parent
29a69d5a1b
commit
282d536fd2
@ -22,6 +22,7 @@
|
||||
#include "../testcommon.h"
|
||||
|
||||
#define WARM_UP_N_RUNS 50
|
||||
#define WARM_UP_ALWAYS_SEC 2.0
|
||||
#define ESTIMATE_ROUND_TIME_N_RUNS 5
|
||||
#define DEFAULT_TEST_TIME 15 /* seconds */
|
||||
/* The time we want each round to take, in seconds, this should
|
||||
@ -34,6 +35,7 @@ static gboolean verbose = FALSE;
|
||||
static gboolean quiet = FALSE;
|
||||
static int test_length = DEFAULT_TEST_TIME;
|
||||
static double test_factor = 0;
|
||||
static GTimer *global_timer = NULL;
|
||||
|
||||
static GOptionEntry cmd_entries[] = {
|
||||
{"verbose", 'v', 0, G_OPTION_ARG_NONE, &verbose,
|
||||
@ -88,11 +90,43 @@ run_test (PerformanceTest *test)
|
||||
g_timer_start (timer);
|
||||
|
||||
/* Warm up the test by doing a few runs */
|
||||
for (i = 0; i < WARM_UP_N_RUNS; i++)
|
||||
for (i = 0; TRUE; i++)
|
||||
{
|
||||
test->init (test, data, 1.0);
|
||||
test->run (test, data);
|
||||
test->finish (test, data);
|
||||
|
||||
if (test_factor > 0)
|
||||
{
|
||||
/* The caller specified a constant factor. That makes mostly
|
||||
* sense, to ensure that the test run is independent from
|
||||
* external factors. In this case, don't make warm up dependent
|
||||
* on WARM_UP_ALWAYS_SEC. */
|
||||
}
|
||||
else if (global_timer)
|
||||
{
|
||||
if (g_timer_elapsed (global_timer, NULL) < WARM_UP_ALWAYS_SEC)
|
||||
{
|
||||
/* We always warm up for a certain time where we keep the
|
||||
* CPU busy.
|
||||
*
|
||||
* Note that when we run multiple tests, then this is only
|
||||
* performed once for the first test. */
|
||||
continue;
|
||||
}
|
||||
g_clear_pointer (&global_timer, g_timer_destroy);
|
||||
}
|
||||
|
||||
if (i >= WARM_UP_N_RUNS)
|
||||
break;
|
||||
|
||||
if (test_factor == 0 && g_timer_elapsed (timer, NULL) > test_length / 10)
|
||||
{
|
||||
/* The warm up should not take longer than 10 % of the entire
|
||||
* test run. Note that the warm up time for WARM_UP_ALWAYS_SEC
|
||||
* already passed. */
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
g_timer_stop (timer);
|
||||
@ -142,7 +176,7 @@ run_test (PerformanceTest *test)
|
||||
|
||||
/* Run the test */
|
||||
avg_elapsed = 0.0;
|
||||
min_elapsed = 0.0;
|
||||
min_elapsed = 1e100;
|
||||
max_elapsed = 0.0;
|
||||
for (i = 0; i < num_rounds; i++)
|
||||
{
|
||||
@ -151,16 +185,18 @@ run_test (PerformanceTest *test)
|
||||
test->run (test, data);
|
||||
g_timer_stop (timer);
|
||||
test->finish (test, data);
|
||||
|
||||
if (i < num_rounds / 20)
|
||||
{
|
||||
/* The first 5% are additional warm up. Ignore. */
|
||||
continue;
|
||||
}
|
||||
|
||||
elapsed = g_timer_elapsed (timer, NULL);
|
||||
|
||||
if (i == 0)
|
||||
max_elapsed = min_elapsed = avg_elapsed = elapsed;
|
||||
else
|
||||
{
|
||||
min_elapsed = MIN (min_elapsed, elapsed);
|
||||
max_elapsed = MAX (max_elapsed, elapsed);
|
||||
avg_elapsed += elapsed;
|
||||
}
|
||||
min_elapsed = MIN (min_elapsed, elapsed);
|
||||
max_elapsed = MAX (max_elapsed, elapsed);
|
||||
avg_elapsed += elapsed;
|
||||
}
|
||||
|
||||
if (num_rounds > 1)
|
||||
@ -1583,6 +1619,8 @@ main (int argc,
|
||||
return 1;
|
||||
}
|
||||
|
||||
global_timer = g_timer_new ();
|
||||
|
||||
if (argc > 1)
|
||||
{
|
||||
for (i = 1; i < argc; i++)
|
||||
@ -1600,5 +1638,6 @@ main (int argc,
|
||||
}
|
||||
|
||||
g_option_context_free (context);
|
||||
g_clear_pointer (&global_timer, g_timer_destroy);
|
||||
return 0;
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user