Received: by 2002:a25:ad19:0:0:0:0:0 with SMTP id y25csp7665430ybi; Thu, 1 Aug 2019 11:44:15 -0700 (PDT) X-Google-Smtp-Source: APXvYqwXwdVm2v4iI3Ug4B13P/ecljNdhXysyr4PSbrd/jeNofqO1SBQmttk57foXZqgdhJv1sX1 X-Received: by 2002:a63:d941:: with SMTP id e1mr86318375pgj.75.1564685055121; Thu, 01 Aug 2019 11:44:15 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1564685055; cv=none; d=google.com; s=arc-20160816; b=e2bGU9p8bTBpeSD+MDW00kzw2PBj2kwtPnRy7JuqHXtVNGuqH9jr+9UaYvfhwWAmwF h0dhW/I2jE+WoJxogZBsQOHNYapL+MisU9wwDU6ycJ/zerX3WFNa45nyn7uFAqLfJV1A c+Uy7aTVU56QSdtlu1zElZvZV4IYZ6SrqCdZvTliC4uTUEL1sshYmAi8OarHbYyCUvH+ kkk1fxH0ct7CB3aij8D/hUjlMhsvt1iw6dQtSqNXZHUrVPC2wKogM589KNIzaVvCxcih rHXvh8WQrUlOSkpfa3+7q1hEpiHIUWnPrR3EgUkbvg133E3D1SWRYKUbnyQ2ksjJjfbJ EtqA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :references:in-reply-to:message-id:date:subject:cc:to:from :dkim-signature; bh=R5am/kvR4YdBRkExxjAP3dW6MnAlz9RnkWC4XFHMY/c=; b=DzhjWYO/HRwsN25T5yL1ELV1rtIrsab/vw/BzDP4dkqqLnnZNCnbL8BU0ouY6uzU39 3ZUc1TG9Hq1lvxGRxfgoi6r8CMt1Xyinl6RHZImA8hL31JbJXUcCkQxG5y+8rHApPfox 6WRsNZ9XCCDXq3u3KYuj2DJWDy6RUUgnVkUbiRNQEs4Q+rSuK32eP7dFLfenehGUF3tk 1Q3olockq4iFvKz+dMcFGwkCc6g/y0LyzikP1Hw0egn/0Y3NWGzogqOz1yq9/PcqvdM8 jaaPJq5AhxF+S02vhgKixt1PMbStnVq46Ot3UPyWsV7J2bpzsHhQi3o0gCxcWib8Ay+t A2Hg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@joelfernandes.org header.s=google header.b=VbNY1mev; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id v21si2598919pgb.209.2019.08.01.11.43.59; Thu, 01 Aug 2019 11:44:15 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@joelfernandes.org header.s=google header.b=VbNY1mev; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S2388291AbfHASO4 (ORCPT + 99 others); Thu, 1 Aug 2019 14:14:56 -0400 Received: from mail-pg1-f194.google.com ([209.85.215.194]:37811 "EHLO mail-pg1-f194.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S2388211AbfHASOf (ORCPT ); Thu, 1 Aug 2019 14:14:35 -0400 Received: by mail-pg1-f194.google.com with SMTP id d1so1830123pgp.4 for ; Thu, 01 Aug 2019 11:14:34 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=joelfernandes.org; s=google; h=from:to:cc:subject:date:message-id:in-reply-to:references :mime-version:content-transfer-encoding; bh=R5am/kvR4YdBRkExxjAP3dW6MnAlz9RnkWC4XFHMY/c=; b=VbNY1mevOcCg/oXTPML6dmX4Dzkfx46dtH9/yxVjJXmTgfbDKImxA8iP9mT3lTsE9L R2C7YsDFEqRskD+hY6yNvHsMcOffGuVY+gPgvoSGtxdJsFFAfIBWi2dU3rpM2OgGmZnz rRUzlwDDiABz67GwlKu6CMUwZhA20tpxHl2Ks= X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:from:to:cc:subject:date:message-id:in-reply-to :references:mime-version:content-transfer-encoding; bh=R5am/kvR4YdBRkExxjAP3dW6MnAlz9RnkWC4XFHMY/c=; b=J5keGDLwD5gEnO/Nd+AqbLC3ODH8VywIGpRa09P33hZqrRxzfFIwdXic2b7YA09x9p 93d1arFxi/6VXsC2A+qf/hoz4wDBvS8VuftHiHXBjBfyfQiodnKAyP//wwvZVNcux/wz Mj6+I7kjai8nl2sl7Si/8VWQPqWsE9fc/UhZniKawriyjYWvTw+makmDRqKJt19K/mWZ yDv9zc0bW40K+WkoCFSGgReuBZ7mtgx209go50wtVRKjfizr48GkTWFfgyU0+ieRyzby chpKhnjrSM07dK4pC9mhY68noQz48cNx4Y7hGnSUCeZNGdXwLW5bRa+nmfsJyKnndO6r kByA== X-Gm-Message-State: APjAAAWKm8MFs53L8bG+68iFL5tzz8zYGGQWhMIrRcghH10dgW0zU3PX opxjYEI1+WGh1NdDv3+I4HcAw9U5 X-Received: by 2002:a17:90a:35e6:: with SMTP id r93mr135658pjb.20.1564683273779; Thu, 01 Aug 2019 11:14:33 -0700 (PDT) Received: from joelaf.cam.corp.google.com ([2620:15c:6:12:9c46:e0da:efbf:69cc]) by smtp.gmail.com with ESMTPSA id g8sm82089165pgk.1.2019.08.01.11.14.32 (version=TLS1_3 cipher=AEAD-AES256-GCM-SHA384 bits=256/256); Thu, 01 Aug 2019 11:14:33 -0700 (PDT) From: "Joel Fernandes (Google)" To: linux-kernel@vger.kernel.org Cc: "Joel Fernandes (Google)" , "Paul E. McKenney" , rcu@vger.kernel.org Subject: [PATCH 8/9] Revert "Revert "rcu: Add support for consolidated-RCU reader checking"" Date: Thu, 1 Aug 2019 14:14:10 -0400 Message-Id: <20190801181411.96429-9-joel@joelfernandes.org> X-Mailer: git-send-email 2.22.0.770.g0f2c4a37fd-goog In-Reply-To: <20190801181411.96429-1-joel@joelfernandes.org> References: <20190801181411.96429-1-joel@joelfernandes.org> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org This reverts commit 24be1727c524b5874d5dc7828cd392cf86c3341e. --- include/linux/rculist.h | 32 ++++++++++++++++---- include/linux/rcupdate.h | 7 +++++ kernel/rcu/Kconfig.debug | 11 +++++++ kernel/rcu/update.c | 65 ++++++++++++++++++++++++++-------------- 4 files changed, 88 insertions(+), 27 deletions(-) diff --git a/include/linux/rculist.h b/include/linux/rculist.h index 932296144131..4158b7212936 100644 --- a/include/linux/rculist.h +++ b/include/linux/rculist.h @@ -40,6 +40,24 @@ static inline void INIT_LIST_HEAD_RCU(struct list_head *list) */ #define list_next_rcu(list) (*((struct list_head __rcu **)(&(list)->next))) +/* + * Check during list traversal that we are within an RCU reader + */ + +#define check_arg_count_one(dummy) + +#ifdef CONFIG_PROVE_RCU_LIST +#define __list_check_rcu(dummy, cond, extra...) \ + ({ \ + check_arg_count_one(extra); \ + RCU_LOCKDEP_WARN(!cond && !rcu_read_lock_any_held(), \ + "RCU-list traversed in non-reader section!"); \ + }) +#else +#define __list_check_rcu(dummy, cond, extra...) \ + ({ check_arg_count_one(extra); }) +#endif + /* * Insert a new entry between two known consecutive entries. * @@ -343,14 +361,16 @@ static inline void list_splice_tail_init_rcu(struct list_head *list, * @pos: the type * to use as a loop cursor. * @head: the head for your list. * @member: the name of the list_head within the struct. + * @cond: optional lockdep expression if called from non-RCU protection. * * This list-traversal primitive may safely run concurrently with * the _rcu list-mutation primitives such as list_add_rcu() * as long as the traversal is guarded by rcu_read_lock(). */ -#define list_for_each_entry_rcu(pos, head, member) \ - for (pos = list_entry_rcu((head)->next, typeof(*pos), member); \ - &pos->member != (head); \ +#define list_for_each_entry_rcu(pos, head, member, cond...) \ + for (__list_check_rcu(dummy, ## cond, 0), \ + pos = list_entry_rcu((head)->next, typeof(*pos), member); \ + &pos->member != (head); \ pos = list_entry_rcu(pos->member.next, typeof(*pos), member)) /** @@ -616,13 +636,15 @@ static inline void hlist_add_behind_rcu(struct hlist_node *n, * @pos: the type * to use as a loop cursor. * @head: the head for your list. * @member: the name of the hlist_node within the struct. + * @cond: optional lockdep expression if called from non-RCU protection. * * This list-traversal primitive may safely run concurrently with * the _rcu list-mutation primitives such as hlist_add_head_rcu() * as long as the traversal is guarded by rcu_read_lock(). */ -#define hlist_for_each_entry_rcu(pos, head, member) \ - for (pos = hlist_entry_safe(rcu_dereference_raw(hlist_first_rcu(head)),\ +#define hlist_for_each_entry_rcu(pos, head, member, cond...) \ + for (__list_check_rcu(dummy, ## cond, 0), \ + pos = hlist_entry_safe(rcu_dereference_raw(hlist_first_rcu(head)),\ typeof(*(pos)), member); \ pos; \ pos = hlist_entry_safe(rcu_dereference_raw(hlist_next_rcu(\ diff --git a/include/linux/rcupdate.h b/include/linux/rcupdate.h index bfcafbc1e301..80d6056f5855 100644 --- a/include/linux/rcupdate.h +++ b/include/linux/rcupdate.h @@ -221,6 +221,7 @@ int debug_lockdep_rcu_enabled(void); int rcu_read_lock_held(void); int rcu_read_lock_bh_held(void); int rcu_read_lock_sched_held(void); +int rcu_read_lock_any_held(void); #else /* #ifdef CONFIG_DEBUG_LOCK_ALLOC */ @@ -241,6 +242,12 @@ static inline int rcu_read_lock_sched_held(void) { return !preemptible(); } + +static inline int rcu_read_lock_any_held(void) +{ + return !preemptible(); +} + #endif /* #else #ifdef CONFIG_DEBUG_LOCK_ALLOC */ #ifdef CONFIG_PROVE_RCU diff --git a/kernel/rcu/Kconfig.debug b/kernel/rcu/Kconfig.debug index 5ec3ea4028e2..4aa02eee8f6c 100644 --- a/kernel/rcu/Kconfig.debug +++ b/kernel/rcu/Kconfig.debug @@ -8,6 +8,17 @@ menu "RCU Debugging" config PROVE_RCU def_bool PROVE_LOCKING +config PROVE_RCU_LIST + bool "RCU list lockdep debugging" + depends on PROVE_RCU && RCU_EXPERT + default n + help + Enable RCU lockdep checking for list usages. By default it is + turned off since there are several list RCU users that still + need to be converted to pass a lockdep expression. To prevent + false-positive splats, we keep it default disabled but once all + users are converted, we can remove this config option. + config TORTURE_TEST tristate default n diff --git a/kernel/rcu/update.c b/kernel/rcu/update.c index 64e9cc8609e7..c768f1fd6804 100644 --- a/kernel/rcu/update.c +++ b/kernel/rcu/update.c @@ -91,19 +91,30 @@ module_param(rcu_normal_after_boot, int, 0); * Similarly, we avoid claiming an SRCU read lock held if the current * CPU is offline. */ +static bool rcu_read_lock_held_common(bool *ret) +{ + if (!debug_lockdep_rcu_enabled()) { + *ret = 1; + return true; + } + if (!rcu_is_watching()) { + *ret = 0; + return true; + } + if (!rcu_lockdep_current_cpu_online()) { + *ret = 0; + return true; + } + return false; +} + int rcu_read_lock_sched_held(void) { - int lockdep_opinion = 0; + bool ret; - if (!debug_lockdep_rcu_enabled()) - return 1; - if (!rcu_is_watching()) - return 0; - if (!rcu_lockdep_current_cpu_online()) - return 0; - if (debug_locks) - lockdep_opinion = lock_is_held(&rcu_sched_lock_map); - return lockdep_opinion || !preemptible(); + if (rcu_read_lock_held_common(&ret)) + return ret; + return lock_is_held(&rcu_sched_lock_map) || !preemptible(); } EXPORT_SYMBOL(rcu_read_lock_sched_held); #endif @@ -260,12 +271,10 @@ NOKPROBE_SYMBOL(debug_lockdep_rcu_enabled); */ int rcu_read_lock_held(void) { - if (!debug_lockdep_rcu_enabled()) - return 1; - if (!rcu_is_watching()) - return 0; - if (!rcu_lockdep_current_cpu_online()) - return 0; + bool ret; + + if (rcu_read_lock_held_common(&ret)) + return ret; return lock_is_held(&rcu_lock_map); } EXPORT_SYMBOL_GPL(rcu_read_lock_held); @@ -287,16 +296,28 @@ EXPORT_SYMBOL_GPL(rcu_read_lock_held); */ int rcu_read_lock_bh_held(void) { - if (!debug_lockdep_rcu_enabled()) - return 1; - if (!rcu_is_watching()) - return 0; - if (!rcu_lockdep_current_cpu_online()) - return 0; + bool ret; + + if (rcu_read_lock_held_common(&ret)) + return ret; return in_softirq() || irqs_disabled(); } EXPORT_SYMBOL_GPL(rcu_read_lock_bh_held); +int rcu_read_lock_any_held(void) +{ + bool ret; + + if (rcu_read_lock_held_common(&ret)) + return ret; + if (lock_is_held(&rcu_lock_map) || + lock_is_held(&rcu_bh_lock_map) || + lock_is_held(&rcu_sched_lock_map)) + return 1; + return !preemptible(); +} +EXPORT_SYMBOL_GPL(rcu_read_lock_any_held); + #endif /* #ifdef CONFIG_DEBUG_LOCK_ALLOC */ /** -- 2.22.0.770.g0f2c4a37fd-goog