Received: by 2002:a05:7412:e794:b0:fa:551:50a7 with SMTP id o20csp617345rdd; Tue, 9 Jan 2024 14:25:54 -0800 (PST) X-Google-Smtp-Source: AGHT+IGKxmaG7tYUBjbB9skThJ3NGdKb5iNulslCBF8SY7kddZsIG1u/Vm7CrqjcgCF4kQwsD5Im X-Received: by 2002:a05:6a20:e68a:b0:199:d3e3:93df with SMTP id mz10-20020a056a20e68a00b00199d3e393dfmr19337pzb.14.1704839153808; Tue, 09 Jan 2024 14:25:53 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1704839153; cv=none; d=google.com; s=arc-20160816; b=UKBkjXVtIhr1J5L7qIhRp5FFrUicEoHncySOFtgNziaiJuK/0l2FjTSZ2J2xmdR+Eq +HnDErY2H+u0LVl0SritDoCU8L+lK9LxvBj7pSzxz7Edig7KZwiokDjkQz31xD1AmrUF RH25d0xrN92IHjgAxYHkBzWI/sg8zf8NmUih5/K1Yhh8G8An0wiQPAeC7xaR81vTNQyz U5+sqaWQ/+MBWGXsXhty1xQnKMU1/waFgKVVYcKgpzc82e7Q28zpPbXEWDrPNFsS7zgS RHjxbJR+T1eC2x8OjRvFNc5ndEFmjlqVY7kubOXMhjjtRlcyigCkI5NfPkC5CsYnEgU2 Fm9Q== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=content-transfer-encoding:mime-version:list-unsubscribe :list-subscribe:list-id:precedence:references:in-reply-to:message-id :date:subject:cc:to:from:dkim-signature; bh=x0ICXYKJsN1FiZyAGJk5wb/Ce75ZaCCBxgmi3e2tJao=; fh=9CmRWAX07YUuDz14FceEnhk8zrN4bQcb1x8/mZIL04w=; b=FhjXKwX/JStNj+vvHPwFQT7Ku3+LJaZaDCQq5fZoB9m7dp9jKmPgFysyeVKwTdxLAT 0HUmWoBljWD+Y9rODmGRQfcPZlyur0lIt34und7yKyEOD2/J5K/7Y59vI4iVUHYC2Kyh wr+4/qc+8uTp3MG/0qz/M/aEK0fvgJACYnGWeHUfafGW6CNT0TPZfqvHfajgnOPYmAQL kQr+Q2YJ8k7DI6z9TcQkCaSmzQXFn7lHunEFVOJ6y8IRytpXHc/wkXZw8SldwRhFWTU1 z/y19NYwfDAMrF42QRO8eu8dMvkJULmhspCWEGYzaxfz1qig1wFLAEH43QCJN6sR8b7O wYQA== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@kernel.org header.s=k20201202 header.b=nYxyiNPt; spf=pass (google.com: domain of linux-kernel+bounces-21472-linux.lists.archive=gmail.com@vger.kernel.org designates 147.75.48.161 as permitted sender) smtp.mailfrom="linux-kernel+bounces-21472-linux.lists.archive=gmail.com@vger.kernel.org"; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=kernel.org Return-Path: Received: from sy.mirrors.kernel.org (sy.mirrors.kernel.org. [147.75.48.161]) by mx.google.com with ESMTPS id jc30-20020a056a006c9e00b006dafc2e8b56si2251194pfb.361.2024.01.09.14.25.53 for (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Tue, 09 Jan 2024 14:25:53 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel+bounces-21472-linux.lists.archive=gmail.com@vger.kernel.org designates 147.75.48.161 as permitted sender) client-ip=147.75.48.161; Authentication-Results: mx.google.com; dkim=pass header.i=@kernel.org header.s=k20201202 header.b=nYxyiNPt; spf=pass (google.com: domain of linux-kernel+bounces-21472-linux.lists.archive=gmail.com@vger.kernel.org designates 147.75.48.161 as permitted sender) smtp.mailfrom="linux-kernel+bounces-21472-linux.lists.archive=gmail.com@vger.kernel.org"; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=kernel.org Received: from smtp.subspace.kernel.org (wormhole.subspace.kernel.org [52.25.139.140]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by sy.mirrors.kernel.org (Postfix) with ESMTPS id 0572EB22532 for ; Tue, 9 Jan 2024 22:24:43 +0000 (UTC) Received: from localhost.localdomain (localhost.localdomain [127.0.0.1]) by smtp.subspace.kernel.org (Postfix) with ESMTP id 1B0023EA6F; Tue, 9 Jan 2024 22:24:18 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; dkim=pass (2048-bit key) header.d=kernel.org header.i=@kernel.org header.b="nYxyiNPt" Received: from smtp.kernel.org (aws-us-west-2-korg-mail-1.web.codeaurora.org [10.30.226.201]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id 3904C3E499; Tue, 9 Jan 2024 22:24:16 +0000 (UTC) Received: by smtp.kernel.org (Postfix) with ESMTPSA id E16ADC43394; Tue, 9 Jan 2024 22:24:13 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=kernel.org; s=k20201202; t=1704839056; bh=7G6zMOBlrDBrz9IFrxxD1LvMGu57Oi45V5z6uSEnxiQ=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=nYxyiNPtWbIoYE9pwkV4RYD6SmcHZMyJUWHEKhbMfcNfLUFd4lOBhLi+dVwmFq3NB jB12yA9WYekWi0BaJKW3GgppBEpsWOli+NDzLWrO27ef35XypOoANF7alTuMNVcblw 4EkErYPnSAVjA/SSdvxGXXPtwbX6lQDqikqPDDe7Fin/REZIIFkvUf+NtvvknCMYvH R4ids+fOckanL17PCX1NEK00uJRcKdXhfrXSpcJNJmxS7b7+mvpxZZMcPd80olEagW VqHH3Yai1Tk57E88NcOuPxPHy1xiNug6ct64kAta8ELHW7xycD5f+jTa4aurhV/tqy tJxCT8UuRoKVA== From: Frederic Weisbecker To: LKML Cc: Frederic Weisbecker , Boqun Feng , Joel Fernandes , Josh Triplett , Lai Jiangshan , Mathieu Desnoyers , Neeraj Upadhyay , Steven Rostedt , Uladzislau Rezki , Zqiang , rcu , "Paul E . McKenney" Subject: [PATCH 2/2] rcu/nocb: Re-arrange call_rcu() NOCB specific code Date: Tue, 9 Jan 2024 23:24:01 +0100 Message-ID: <20240109222401.28961-3-frederic@kernel.org> X-Mailer: git-send-email 2.43.0 In-Reply-To: <20240109222401.28961-1-frederic@kernel.org> References: <20240109222401.28961-1-frederic@kernel.org> Precedence: bulk X-Mailing-List: linux-kernel@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Currently the call_rcu() function interleaves NOCB and !NOCB enqueue code in a complicated way such that: * The bypass enqueue code may or may not have enqueued and may or may not have locked the ->nocb_lock. Everything that follows is in a Schrödinger locking state for the unwary reviewer's eyes. * The was_alldone is always set but only used in NOCB related code. * The NOCB wake up is distantly related to the locking hopefully performed by the bypass enqueue code that did not enqueue on the bypass list. Unconfuse the whole and gather NOCB and !NOCB specific enqueue code to their own functions. Signed-off-by: Frederic Weisbecker --- kernel/rcu/tree.c | 44 +++++++++++++++++++----------------------- kernel/rcu/tree.h | 9 ++++----- kernel/rcu/tree_nocb.h | 18 ++++++++++++++--- 3 files changed, 39 insertions(+), 32 deletions(-) diff --git a/kernel/rcu/tree.c b/kernel/rcu/tree.c index 91b2eb772e86..de5796ce024f 100644 --- a/kernel/rcu/tree.c +++ b/kernel/rcu/tree.c @@ -2597,12 +2597,26 @@ static int __init rcu_spawn_core_kthreads(void) return 0; } +static void rcutree_enqueue(struct rcu_data *rdp, struct rcu_head *head, rcu_callback_t func) +{ + rcu_segcblist_enqueue(&rdp->cblist, head); + if (__is_kvfree_rcu_offset((unsigned long)func)) + trace_rcu_kvfree_callback(rcu_state.name, head, + (unsigned long)func, + rcu_segcblist_n_cbs(&rdp->cblist)); + else + trace_rcu_callback(rcu_state.name, head, + rcu_segcblist_n_cbs(&rdp->cblist)); + trace_rcu_segcb_stats(&rdp->cblist, TPS("SegCBQueued")); +} + /* * Handle any core-RCU processing required by a call_rcu() invocation. */ -static void __call_rcu_core(struct rcu_data *rdp, struct rcu_head *head, - unsigned long flags) +static void call_rcu_core(struct rcu_data *rdp, struct rcu_head *head, + rcu_callback_t func, unsigned long flags) { + rcutree_enqueue(rdp, head, func); /* * If called from an extended quiescent state, invoke the RCU * core in order to force a re-evaluation of RCU's idleness. @@ -2698,7 +2712,6 @@ __call_rcu_common(struct rcu_head *head, rcu_callback_t func, bool lazy_in) unsigned long flags; bool lazy; struct rcu_data *rdp; - bool was_alldone; /* Misaligned rcu_head! */ WARN_ON_ONCE((unsigned long)head & (sizeof(void *) - 1)); @@ -2735,28 +2748,11 @@ __call_rcu_common(struct rcu_head *head, rcu_callback_t func, bool lazy_in) } check_cb_ovld(rdp); - if (rcu_nocb_try_bypass(rdp, head, &was_alldone, flags, lazy)) { - local_irq_restore(flags); - return; // Enqueued onto ->nocb_bypass, so just leave. - } - // If no-CBs CPU gets here, rcu_nocb_try_bypass() acquired ->nocb_lock. - rcu_segcblist_enqueue(&rdp->cblist, head); - if (__is_kvfree_rcu_offset((unsigned long)func)) - trace_rcu_kvfree_callback(rcu_state.name, head, - (unsigned long)func, - rcu_segcblist_n_cbs(&rdp->cblist)); + + if (unlikely(rcu_rdp_is_offloaded(rdp))) + call_rcu_nocb(rdp, head, func, flags, lazy); else - trace_rcu_callback(rcu_state.name, head, - rcu_segcblist_n_cbs(&rdp->cblist)); - - trace_rcu_segcb_stats(&rdp->cblist, TPS("SegCBQueued")); - - /* Go handle any RCU core processing required. */ - if (unlikely(rcu_rdp_is_offloaded(rdp))) { - __call_rcu_nocb_wake(rdp, was_alldone, flags); /* unlocks */ - } else { - __call_rcu_core(rdp, head, flags); - } + call_rcu_core(rdp, head, func, flags); local_irq_restore(flags); } diff --git a/kernel/rcu/tree.h b/kernel/rcu/tree.h index e9821a8422db..bf478da89a8f 100644 --- a/kernel/rcu/tree.h +++ b/kernel/rcu/tree.h @@ -467,11 +467,10 @@ static void rcu_init_one_nocb(struct rcu_node *rnp); static bool wake_nocb_gp(struct rcu_data *rdp, bool force); static bool rcu_nocb_flush_bypass(struct rcu_data *rdp, struct rcu_head *rhp, unsigned long j, bool lazy); -static bool rcu_nocb_try_bypass(struct rcu_data *rdp, struct rcu_head *rhp, - bool *was_alldone, unsigned long flags, - bool lazy); -static void __call_rcu_nocb_wake(struct rcu_data *rdp, bool was_empty, - unsigned long flags); +static void call_rcu_nocb(struct rcu_data *rdp, struct rcu_head *head, + rcu_callback_t func, unsigned long flags, bool lazy); +static void __maybe_unused __call_rcu_nocb_wake(struct rcu_data *rdp, bool was_empty, + unsigned long flags); static int rcu_nocb_need_deferred_wakeup(struct rcu_data *rdp, int level); static bool do_nocb_deferred_wakeup(struct rcu_data *rdp); static void rcu_boot_init_nocb_percpu_data(struct rcu_data *rdp); diff --git a/kernel/rcu/tree_nocb.h b/kernel/rcu/tree_nocb.h index 06c8ff85850c..5fd47ea6d20e 100644 --- a/kernel/rcu/tree_nocb.h +++ b/kernel/rcu/tree_nocb.h @@ -622,6 +622,18 @@ static void __call_rcu_nocb_wake(struct rcu_data *rdp, bool was_alldone, } } +static void call_rcu_nocb(struct rcu_data *rdp, struct rcu_head *head, + rcu_callback_t func, unsigned long flags, bool lazy) +{ + bool was_alldone; + + if (!rcu_nocb_try_bypass(rdp, head, &was_alldone, flags, lazy)) { + /* Not enqueued on bypass but locked, do regular enqueue */ + rcutree_enqueue(rdp, head, func); + __call_rcu_nocb_wake(rdp, was_alldone, flags); /* unlocks */ + } +} + static int nocb_gp_toggle_rdp(struct rcu_data *rdp, bool *wake_state) { @@ -1764,10 +1776,10 @@ static bool rcu_nocb_flush_bypass(struct rcu_data *rdp, struct rcu_head *rhp, return true; } -static bool rcu_nocb_try_bypass(struct rcu_data *rdp, struct rcu_head *rhp, - bool *was_alldone, unsigned long flags, bool lazy) +static void call_rcu_nocb(struct rcu_data *rdp, struct rcu_head *head, + rcu_callback_t func, unsigned long flags, bool lazy) { - return false; + WARN_ON_ONCE(1); /* Should be dead code! */ } static void __call_rcu_nocb_wake(struct rcu_data *rdp, bool was_empty, -- 2.43.0