rcu: Mark accesses to rcu_state.n_force_qs
authorPaul E. McKenney <paulmck@kernel.org>
Tue, 20 Jul 2021 13:16:27 +0000 (06:16 -0700)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Wed, 22 Dec 2021 08:30:59 +0000 (09:30 +0100)
commit 2431774f04d1050292054c763070021bade7b151 upstream.

This commit marks accesses to the rcu_state.n_force_qs.  These data
races are hard to make happen, but syzkaller was equal to the task.

Reported-by: syzbot+e08a83a1940ec3846cd5@syzkaller.appspotmail.com
Acked-by: Marco Elver <elver@google.com>
Signed-off-by: Paul E. McKenney <paulmck@kernel.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
kernel/rcu/tree.c

index 8c81c05..b74e7ac 100644 (file)
@@ -1888,7 +1888,7 @@ static void rcu_gp_fqs(bool first_time)
        struct rcu_node *rnp = rcu_get_root();
 
        WRITE_ONCE(rcu_state.gp_activity, jiffies);
-       rcu_state.n_force_qs++;
+       WRITE_ONCE(rcu_state.n_force_qs, rcu_state.n_force_qs + 1);
        if (first_time) {
                /* Collect dyntick-idle snapshots. */
                force_qs_rnp(dyntick_save_progress_counter);
@@ -2530,7 +2530,7 @@ static void rcu_do_batch(struct rcu_data *rdp)
        /* Reset ->qlen_last_fqs_check trigger if enough CBs have drained. */
        if (count == 0 && rdp->qlen_last_fqs_check != 0) {
                rdp->qlen_last_fqs_check = 0;
-               rdp->n_force_qs_snap = rcu_state.n_force_qs;
+               rdp->n_force_qs_snap = READ_ONCE(rcu_state.n_force_qs);
        } else if (count < rdp->qlen_last_fqs_check - qhimark)
                rdp->qlen_last_fqs_check = count;
 
@@ -2876,10 +2876,10 @@ static void __call_rcu_core(struct rcu_data *rdp, struct rcu_head *head,
                } else {
                        /* Give the grace period a kick. */
                        rdp->blimit = DEFAULT_MAX_RCU_BLIMIT;
-                       if (rcu_state.n_force_qs == rdp->n_force_qs_snap &&
+                       if (READ_ONCE(rcu_state.n_force_qs) == rdp->n_force_qs_snap &&
                            rcu_segcblist_first_pend_cb(&rdp->cblist) != head)
                                rcu_force_quiescent_state();
-                       rdp->n_force_qs_snap = rcu_state.n_force_qs;
+                       rdp->n_force_qs_snap = READ_ONCE(rcu_state.n_force_qs);
                        rdp->qlen_last_fqs_check = rcu_segcblist_n_cbs(&rdp->cblist);
                }
        }
@@ -3986,7 +3986,7 @@ int rcutree_prepare_cpu(unsigned int cpu)
        /* Set up local state, ensuring consistent view of global state. */
        raw_spin_lock_irqsave_rcu_node(rnp, flags);
        rdp->qlen_last_fqs_check = 0;
-       rdp->n_force_qs_snap = rcu_state.n_force_qs;
+       rdp->n_force_qs_snap = READ_ONCE(rcu_state.n_force_qs);
        rdp->blimit = blimit;
        if (rcu_segcblist_empty(&rdp->cblist) && /* No early-boot CBs? */
            !rcu_segcblist_is_offloaded(&rdp->cblist))