[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <4c8da7b3-6235-4d0a-aeb1-81c8bdfd051a@nvidia.com>
Date: Thu, 28 Aug 2025 10:07:03 -0700
From: Fenghua Yu <fenghuay@...dia.com>
To: James Morse <james.morse@....com>, linux-kernel@...r.kernel.org,
linux-arm-kernel@...ts.infradead.org, linux-acpi@...r.kernel.org,
devicetree@...r.kernel.org
Cc: shameerali.kolothum.thodi@...wei.com,
D Scott Phillips OS <scott@...amperecomputing.com>,
carl@...amperecomputing.com, lcherian@...vell.com,
bobo.shaobowang@...wei.com, tan.shaopeng@...itsu.com,
baolin.wang@...ux.alibaba.com, Jamie Iles <quic_jiles@...cinc.com>,
Xin Hao <xhao@...ux.alibaba.com>, peternewman@...gle.com,
dfustini@...libre.com, amitsinght@...vell.com,
David Hildenbrand <david@...hat.com>, Rex Nie <rex.nie@...uarmicro.com>,
Dave Martin <dave.martin@....com>, Koba Ko <kobak@...dia.com>,
Shanker Donthineni <sdonthineni@...dia.com>, baisheng.gao@...soc.com,
Jonathan Cameron <jonathan.cameron@...wei.com>, Rob Herring
<robh@...nel.org>, Rohit Mathew <rohit.mathew@....com>,
Rafael Wysocki <rafael@...nel.org>, Len Brown <lenb@...nel.org>,
Lorenzo Pieralisi <lpieralisi@...nel.org>, Hanjun Guo
<guohanjun@...wei.com>, Sudeep Holla <sudeep.holla@....com>,
Krzysztof Kozlowski <krzk+dt@...nel.org>, Conor Dooley
<conor+dt@...nel.org>, Catalin Marinas <catalin.marinas@....com>,
Will Deacon <will@...nel.org>,
Greg Kroah-Hartman <gregkh@...uxfoundation.org>,
Danilo Krummrich <dakr@...nel.org>
Subject: Re: [PATCH 16/33] arm_mpam: Add helpers for managing the locking
around the mon_sel registers
Hi, James,
On 8/22/25 08:29, James Morse wrote:
> The MSC MON_SEL register needs to be accessed from hardirq context by the
> PMU drivers, making an irqsave spinlock the obvious lock to protect these
> registers. On systems with SCMI mailboxes it must be able to sleep, meaning
> a mutex must be used.
>
> Clearly these two can't exist at the same time.
>
> Add helpers for the MON_SEL locking. The outer lock must be taken in a
> pre-emptible context before the inner lock can be taken. On systems with
> SCMI mailboxes where the MON_SEL accesses must sleep - the inner lock
> will fail to be 'taken' if the caller is unable to sleep. This will allow
> the PMU driver to fail without having to check the interface type of
> each MSC.
>
> Signed-off-by: James Morse <james.morse@....com>
> ---
> drivers/resctrl/mpam_internal.h | 57 ++++++++++++++++++++++++++++++++-
> 1 file changed, 56 insertions(+), 1 deletion(-)
>
> diff --git a/drivers/resctrl/mpam_internal.h b/drivers/resctrl/mpam_internal.h
> index a623f405ddd8..c6f087f9fa7d 100644
> --- a/drivers/resctrl/mpam_internal.h
> +++ b/drivers/resctrl/mpam_internal.h
> @@ -68,10 +68,19 @@ struct mpam_msc {
>
> /*
> * mon_sel_lock protects access to the MSC hardware registers that are
> - * affeted by MPAMCFG_MON_SEL.
> + * affected by MPAMCFG_MON_SEL, and the mbwu_state.
> + * Both the 'inner' and 'outer' must be taken.
> + * For real MMIO MSC, the outer lock is unnecessary - but keeps the
> + * code common with:
> + * Firmware backed MSC need to sleep when accessing the MSC, which
> + * means some code-paths will always fail. For these MSC the outer
> + * lock is providing the protection, and the inner lock fails to
> + * be taken if the task is unable to sleep.
> + *
> * If needed, take msc->probe_lock first.
> */
> struct mutex outer_mon_sel_lock;
> + bool outer_lock_held;
Is it better to define outer_lock_held at atomic_t?
> raw_spinlock_t inner_mon_sel_lock;
> unsigned long inner_mon_sel_flags;
>
> @@ -81,6 +90,52 @@ struct mpam_msc {
> struct mpam_garbage garbage;
> };
>
> +static inline bool __must_check mpam_mon_sel_inner_lock(struct mpam_msc *msc)
> +{
> + /*
> + * The outer lock may be taken by a CPU that then issues an IPI to run
> + * a helper that takes the inner lock. lockdep can't help us here.
> + */
> + WARN_ON_ONCE(!msc->outer_lock_held);
At this point, msc->outer_lock_held might not be true yet due to no
memory barrier on it on this CPU. If it's atomic_t and it's set as true
on another CPU by smp_store_release(), it's guaranteed to be visible as
true on this CPU. Without atomic setting, we may see a false warning
here and cause debug difficult.
> +
> + if (msc->iface == MPAM_IFACE_MMIO) {
> + raw_spin_lock_irqsave(&msc->inner_mon_sel_lock, msc->inner_mon_sel_flags);
> + return true;
> + }
> +
> + /* Accesses must fail if we are not pre-emptible */
> + return !!preemptible();
> +}
> +
> +static inline void mpam_mon_sel_inner_unlock(struct mpam_msc *msc)
> +{
> + WARN_ON_ONCE(!msc->outer_lock_held);
> +
> + if (msc->iface == MPAM_IFACE_MMIO)
> + raw_spin_unlock_irqrestore(&msc->inner_mon_sel_lock, msc->inner_mon_sel_flags);
> +}
> +
> +static inline void mpam_mon_sel_outer_lock(struct mpam_msc *msc)
> +{
> + mutex_lock(&msc->outer_mon_sel_lock);
> + msc->outer_lock_held = true;
> +}
> +
> +static inline void mpam_mon_sel_outer_unlock(struct mpam_msc *msc)
> +{
> + msc->outer_lock_held = false;
> + mutex_unlock(&msc->outer_mon_sel_lock);
> +}
> +
> +static inline void mpam_mon_sel_lock_held(struct mpam_msc *msc)
> +{
> + WARN_ON_ONCE(!msc->outer_lock_held);
> + if (msc->iface == MPAM_IFACE_MMIO)
> + lockdep_assert_held_once(&msc->inner_mon_sel_lock);
> + else
> + lockdep_assert_preemption_enabled();
> +}
> +
> struct mpam_class {
> /* mpam_components in this class */
> struct list_head components;
Thanks.
-Fenghua
Powered by blists - more mailing lists