]> www.infradead.org Git - users/willy/linux.git/commitdiff
refperf: More closely synchronize reader start times
authorPaul E. McKenney <paulmck@kernel.org>
Tue, 26 May 2020 18:40:52 +0000 (11:40 -0700)
committerPaul E. McKenney <paulmck@kernel.org>
Mon, 29 Jun 2020 19:00:45 +0000 (12:00 -0700)
Currently, readers are awakened individually.  On most systems, this
results in significant wakeup delay from one reader to the next, which
can result in the first and last reader having sole access to the
synchronization primitive in question.  If that synchronization primitive
involves shared memory, those readers will rack up a huge number of
operations in a very short time, causing large perturbations in the
results.

This commit therefore has the readers busy-wait after being awakened,
and uses a new n_started variable to synchronize their start times.

Cc: Joel Fernandes (Google) <joel@joelfernandes.org>
Signed-off-by: Paul E. McKenney <paulmck@kernel.org>
kernel/rcu/refperf.c

index 2fd3ed1a0d0d544c909c9e24d83d3809055eb196..234bb0e84a8b9d71418da197dd56221a005b0132 100644 (file)
@@ -99,6 +99,7 @@ static atomic_t nreaders_exp;
 
 // Use to wait for all threads to start.
 static atomic_t n_init;
+static atomic_t n_started;
 
 // Track which experiment is currently running.
 static int exp_idx;
@@ -253,6 +254,9 @@ repeat:
        WARN_ON_ONCE(smp_processor_id() != me);
 
        WRITE_ONCE(rt->start_reader, 0);
+       if (!atomic_dec_return(&n_started))
+               while (atomic_read_acquire(&n_started))
+                       cpu_relax();
 
        VERBOSE_PERFOUT("ref_perf_reader %ld: experiment %d started", me, exp_idx);
 
@@ -367,6 +371,7 @@ static int main_func(void *arg)
 
                reset_readers();
                atomic_set(&nreaders_exp, nreaders);
+               atomic_set(&n_started, nreaders);
 
                exp_idx = exp;