On Wed, Jul 30, 2025 at 10:10:32AM +0200, Janosch Frank wrote: > From: Mark Rutland <mark.rutland@xxxxxxx> > > All architectures have an interruptible RCU extended quiescent state > (EQS) as part of their idle sequences, where interrupts can occur > without RCU watching. Entry code must account for this and wake RCU as > necessary; the common entry code deals with this in irqentry_enter() by > treating any interrupt from an idle thread as potentially having > occurred within an EQS and waking RCU for the duration of the interrupt > via rcu_irq_enter() .. rcu_irq_exit(). > > Some architectures may have other interruptible EQSs which require > similar treatment. For example, on s390 it is necessary to enable > interrupts around guest entry in the middle of a period where core KVM > code has entered an EQS. > > So that architectures can wake RCU in these cases, this patch adds a > new arch_in_rcu_eqs() hook to the common entry code which is checked in > addition to the existing is_idle_thread() check, with RCU woken if > either returns true. A default implementation is provided which always > returns false, which suffices for most architectures. > > As no architectures currently implement arch_in_rcu_eqs(), there should > be no functional change as a result of this patch alone. A subsequent > patch will add an s390 implementation to fix a latent bug with missing > RCU wakeups. > > [ajd@xxxxxxxxxxxxx: rebase, fix commit message] > > Signed-off-by: Mark Rutland <mark.rutland@xxxxxxx> > Cc: Andy Lutomirski <luto@xxxxxxxxxx> > Cc: Christian Borntraeger <borntraeger@xxxxxxxxxxxxx> > Cc: Heiko Carstens <hca@xxxxxxxxxxxxx> > Cc: Paolo Bonzini <pbonzini@xxxxxxxxxx> > Cc: Paul E. McKenney <paulmck@xxxxxxxxxx> > Cc: Peter Zijlstra <peterz@xxxxxxxxxxxxx> > Cc: Sven Schnelle <svens@xxxxxxxxxxxxx> > Cc: Thomas Gleixner <tglx@xxxxxxxxxxxxx> > Cc: Claudio Imbrenda <imbrenda@xxxxxxxxxxxxx> > Cc: Vasily Gorbik <gor@xxxxxxxxxxxxx> > Cc: Alexander Gordeev <agordeev@xxxxxxxxxxxxx> > Cc: Janosch Frank <frankja@xxxxxxxxxxxxx> > Reviewed-by: Christian Borntraeger <borntraeger@xxxxxxxxxxxxx> > Signed-off-by: Andrew Donnellan <ajd@xxxxxxxxxxxxx> > Acked-by: Peter Zijlstra (Intel) <peterz@xxxxxxxxxxxxx> > Reviewed-by: Janosch Frank <frankja@xxxxxxxxxxxxx> > Link: https://lore.kernel.org/r/20250708092742.104309-2-ajd@xxxxxxxxxxxxx > Signed-off-by: Janosch Frank <frankja@xxxxxxxxxxxxx> > Message-ID: <20250708092742.104309-2-ajd@xxxxxxxxxxxxx> Reviewed-by: Paul E. McKenney <paulmck@xxxxxxxxxx> > --- > include/linux/entry-common.h | 16 ++++++++++++++++ > kernel/entry/common.c | 3 ++- > 2 files changed, 18 insertions(+), 1 deletion(-) > > diff --git a/include/linux/entry-common.h b/include/linux/entry-common.h > index f94f3fdf15fc..3bf99cbad8a3 100644 > --- a/include/linux/entry-common.h > +++ b/include/linux/entry-common.h > @@ -86,6 +86,22 @@ static __always_inline void arch_enter_from_user_mode(struct pt_regs *regs); > static __always_inline void arch_enter_from_user_mode(struct pt_regs *regs) {} > #endif > > +/** > + * arch_in_rcu_eqs - Architecture specific check for RCU extended quiescent > + * states. > + * > + * Returns: true if the CPU is potentially in an RCU EQS, false otherwise. > + * > + * Architectures only need to define this if threads other than the idle thread > + * may have an interruptible EQS. This does not need to handle idle threads. It > + * is safe to over-estimate at the cost of redundant RCU management work. > + * > + * Invoked from irqentry_enter() > + */ > +#ifndef arch_in_rcu_eqs > +static __always_inline bool arch_in_rcu_eqs(void) { return false; } > +#endif > + > /** > * enter_from_user_mode - Establish state when coming from user mode > * > diff --git a/kernel/entry/common.c b/kernel/entry/common.c > index a8dd1f27417c..eb52d38e8099 100644 > --- a/kernel/entry/common.c > +++ b/kernel/entry/common.c > @@ -220,7 +220,8 @@ noinstr irqentry_state_t irqentry_enter(struct pt_regs *regs) > * TINY_RCU does not support EQS, so let the compiler eliminate > * this part when enabled. > */ > - if (!IS_ENABLED(CONFIG_TINY_RCU) && is_idle_task(current)) { > + if (!IS_ENABLED(CONFIG_TINY_RCU) && > + (is_idle_task(current) || arch_in_rcu_eqs())) { > /* > * If RCU is not watching then the same careful > * sequence vs. lockdep and tracing is required > -- > 2.50.1 >