Re: [PATCH] rcu: Benefit from expedited grace period in __wait_rcu_gp

From: Raslan, KarimAllah
Date: Fri Oct 19 2018 - 15:46:11 EST


On Fri, 2018-10-19 at 05:31 -0700, Paul E. McKenney wrote:
> On Fri, Oct 19, 2018 at 02:49:05AM +0200, KarimAllah Ahmed wrote:
> >
> > When expedited grace-period is set, both synchronize_sched
> > synchronize_rcu_bh can be optimized to have a significantly lower latency.
> >
> > Improve wait_rcu_gp handling to also account for expedited grace-period.
> > The downside is that wait_rcu_gp will not wait anymore for all RCU variants
> > concurrently when an expedited grace-period is set, however, given the
> > improved latency it does not really matter.
> >
> > Cc: Paul E. McKenney <paulmck@xxxxxxxxxxxxxxxxxx>
> > Cc: Josh Triplett <josh@xxxxxxxxxxxxxxxx>
> > Cc: Steven Rostedt <rostedt@xxxxxxxxxxx>
> > Cc: Mathieu Desnoyers <mathieu.desnoyers@xxxxxxxxxxxx>
> > Cc: Lai Jiangshan <jiangshanlai@xxxxxxxxx>
> > Cc: linux-kernel@xxxxxxxxxxxxxxx
> > Signed-off-by: KarimAllah Ahmed <karahmed@xxxxxxxxx>
>
> Cute!
>
> Unfortunately, there are a few problems with this patch:
>
> 1. I will be eliminating synchronize_rcu_mult() due to the fact that
> the upcoming RCU flavor consolidation eliminates its sole caller.
> See 5fc9d4e000b1 ("rcu: Eliminate synchronize_rcu_mult()")
> in my -rcu tree. This would of course also eliminate the effects
> of this patch.

Your patch covers our use-case already, but I still think that the semanticsÂ
forÂwait_rcu_gp is not clear to me.

The problem for us was that sched_cpu_deactivate would call
synchronize_rcu_mult which does not check for "expedited" at all. So even
though we are already using rcu_expedited sysctl variable, synchronize_rcu_multÂ
was just ignoring it.

That being said, I indeed overlooked rcu_normal and that it takes precedenceÂ
over expedited and I did not notice at all the deadlock you mentioned below!

That can however be easily fixed by also checking for !rcu_gp_is_normal.

>
> 2. The real-time guys' users are not going to be at all happy
> with the IPIs resulting from the _expedited() API members.
> Yes, they can boot with rcupdate.rcu_normal=1, but they don't
> always need that big a hammer, and use of this kernel parameter
> can slow down boot, hibernation, suspend, network configuration,
> and much else besides. We therefore don't want them to have to
> use rcupdate.rcu_normal=1 unless absolutely necessary.

I might be missing something here. Why would they need to "explicitly" useÂ
rcu_normal? If rcu_expedited is set, would not the expected behavior is to callÂ
into the expedited version?

My patch should only activate *expedited* if and only if it is set.

I think I might be misunderstanding the expected behaviorÂ
fromÂsynchronize_rcu_mult. My understanding is that something like:

synchronize_rcu_mult(call_rcu_sched) and synchronize_rcu() should have anÂ
identical behavior, right?

At least in this commit:

commit d7d34d5e46140 ("sched: Rely on synchronize_rcu_mult() de-duplication")

.. the change clearly gives the impression that they can be usedÂ
interchangeably. The problem is that this is not true when you look at theÂ
implementation. One of them (i.e. synchronize_rcu) will respect the
expedite_rcu flag set by sysfs while the other (i.e. synchronize_rcu_mult)Â
simply ignores it.

So my patch is about making sure that both of the variants actually respectÂ
it.


> 3. If the real-time guys' users were to have booted with
> rcupdate.rcu_normal=1, then synchronize_sched_expedited()
> would invoke _synchronize_rcu_expedited, which would invoke
> wait_rcu_gp(), which would invoke _wait_rcu_gp() which would
> invoke __wait_rcu_gp(), which, given your patch, would in turn
> invoke synchronize_sched_expedited(). This situation could
> well prevent their systems from meeting their response-time
> requirements.
>
> So I cannot accept this patch nor for that matter any similar patch.
>
> But what were you really trying to get done here? If you were thinking
> of adding another synchronize_rcu_mult(), the flavor consolidation will
> make that unnecessary in most cases. If you are trying to speed up
> CPU-hotplug operations, I suggest using the rcu_expedited sysctl variable
> when taking a CPU offline. If something else, please let me know what
> it is so that we can work out how the problem might best be solved.
>
> Thanx, Paul
>
> >
> > ---
> > kernel/rcu/update.c | 34 ++++++++++++++++++++++++++++------
> > 1 file changed, 28 insertions(+), 6 deletions(-)
> >
> > diff --git a/kernel/rcu/update.c b/kernel/rcu/update.c
> > index 68fa19a..44b8817 100644
> > --- a/kernel/rcu/update.c
> > +++ b/kernel/rcu/update.c
> > @@ -392,13 +392,27 @@ void __wait_rcu_gp(bool checktiny, int n, call_rcu_func_t *crcu_array,
> > might_sleep();
> > continue;
> > }
> > - init_rcu_head_on_stack(&rs_array[i].head);
> > - init_completion(&rs_array[i].completion);
> > +
> > for (j = 0; j < i; j++)
> > if (crcu_array[j] == crcu_array[i])
> > break;
> > - if (j == i)
> > - (crcu_array[i])(&rs_array[i].head, wakeme_after_rcu);
> > + if (j != i)
> > + continue;
> > +
> > + if ((crcu_array[i] == call_rcu_sched ||
> > + crcu_array[i] == call_rcu_bh)
> > + && rcu_gp_is_expedited()) {
> > + if (crcu_array[i] == call_rcu_sched)
> > + synchronize_sched_expedited();
> > + else
> > + synchronize_rcu_bh_expedited();
> > +
> > + continue;
> > + }
> > +
> > + init_rcu_head_on_stack(&rs_array[i].head);
> > + init_completion(&rs_array[i].completion);
> > + (crcu_array[i])(&rs_array[i].head, wakeme_after_rcu);
> > }
> >
> > /* Wait for all callbacks to be invoked. */
> > @@ -407,11 +421,19 @@ void __wait_rcu_gp(bool checktiny, int n, call_rcu_func_t *crcu_array,
> > (crcu_array[i] == call_rcu ||
> > crcu_array[i] == call_rcu_bh))
> > continue;
> > +
> > + if ((crcu_array[i] == call_rcu_sched ||
> > + crcu_array[i] == call_rcu_bh)
> > + && rcu_gp_is_expedited())
> > + continue;
> > +
> > for (j = 0; j < i; j++)
> > if (crcu_array[j] == crcu_array[i])
> > break;
> > - if (j == i)
> > - wait_for_completion(&rs_array[i].completion);
> > + if (j != i)
> > + continue;
> > +
> > + wait_for_completion(&rs_array[i].completion);
> > destroy_rcu_head_on_stack(&rs_array[i].head);
> > }
> > }
> > --
> > 2.7.4
> >
>
Amazon Development Center Germany GmbH
Berlin - Dresden - Aachen
main office: Krausenstr. 38, 10117 Berlin
Geschaeftsfuehrer: Dr. Ralf Herbrich, Christian Schlaeger
Ust-ID: DE289237879
Eingetragen am Amtsgericht Charlottenburg HRB 149173 B