Skip to content

Commit a4fcfbe

Browse files
qiangzh3paulmckrcu
authored andcommitted
rcu-tasks: Handle queue-shrink/callback-enqueue race condition
The rcu_tasks_need_gpcb() determines whether or not: (1) There are callbacks needing another grace period, (2) There are callbacks ready to be invoked, and (3) It would be a good time to shrink back down to a single-CPU callback list. This third case is interesting because some other CPU might be adding new callbacks, which might suddenly make this a very bad time to be shrinking. This is currently handled by requiring call_rcu_tasks_generic() to enqueue callbacks under the protection of rcu_read_lock() and requiring rcu_tasks_need_gpcb() to wait for an RCU grace period to elapse before finalizing the transition. This works well in practice. Unfortunately, the current code assumes that a grace period whose end is detected by the poll_state_synchronize_rcu() in the second "if" condition actually ended before the earlier code counted the callbacks queued on CPUs other than CPU 0 (local variable "ncbsnz"). Given the current code, it is possible that a long-delayed call_rcu_tasks_generic() invocation will queue a callback on a non-zero CPU after these CPUs have had their callbacks counted and zero has been stored to ncbsnz. Such a callback would trigger the WARN_ON_ONCE() in the second "if" statement. To see this, consider the following sequence of events: o CPU 0 invokes rcu_tasks_one_gp(), and counts fewer than rcu_task_collapse_lim callbacks. It sees at least one callback queued on some other CPU, thus setting ncbsnz to a non-zero value. o CPU 1 invokes call_rcu_tasks_generic() and loads 42 from ->percpu_enqueue_lim. It therefore decides to enqueue its callback onto CPU 1's callback list, but is delayed. o CPU 0 sees the rcu_task_cb_adjust is non-zero and that the number of callbacks does not exceed rcu_task_collapse_lim. It therefore checks percpu_enqueue_lim, and sees that its value is greater than the value one. CPU 0 therefore starts the shift back to a single callback list. It sets ->percpu_enqueue_lim to 1, but CPU 1 has already read the old value of 42. It also gets a grace-period state value from get_state_synchronize_rcu(). o CPU 0 sees that ncbsnz is non-zero in its second "if" statement, so it declines to finalize the shrink operation. o CPU 0 again invokes rcu_tasks_one_gp(), and counts fewer than rcu_task_collapse_lim callbacks. It also sees that there are no callback queued on any other CPU, and thus sets ncbsnz to zero. o CPU 1 resumes execution and enqueues its callback onto its own list. This invalidates the value of ncbsnz. o CPU 0 sees the rcu_task_cb_adjust is non-zero and that the number of callbacks does not exceed rcu_task_collapse_lim. It therefore checks percpu_enqueue_lim, but sees that its value is already unity. It therefore does not get a new grace-period state value. o CPU 0 sees that rcu_task_cb_adjust is non-zero, ncbsnz is zero, and that poll_state_synchronize_rcu() says that the grace period has completed. it therefore finalizes the shrink operation, setting ->percpu_dequeue_lim to the value one. o CPU 0 does a debug check, scanning the other CPUs' callback lists. It sees that CPU 1's list has a callback, so it (rightly) triggers the WARN_ON_ONCE(). After all, the new value of ->percpu_dequeue_lim says to not bother looking at CPU 1's callback list, which means that this callback will never be invoked. This can result in hangs and maybe even OOMs. Based on long experience with rcutorture, this is an extremely low-probability race condition, but it really can happen, especially in preemptible kernels or within guest OSes. This commit therefore checks for completion of the grace period before counting callbacks. With this change, in the above failure scenario CPU 0 would know not to prematurely end the shrink operation because the grace period would not have completed before the count operation started. [ paulmck: Adjust grace-period end rather than adding RCU reader. ] [ paulmck: Avoid spurious WARN_ON_ONCE() with ->percpu_dequeue_lim check. ] Signed-off-by: Zqiang <[email protected]> Signed-off-by: Paul E. McKenney <[email protected]>
1 parent ea5c898 commit a4fcfbe

File tree

1 file changed

+8
-5
lines changed

1 file changed

+8
-5
lines changed

kernel/rcu/tasks.h

Lines changed: 8 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -384,6 +384,7 @@ static int rcu_tasks_need_gpcb(struct rcu_tasks *rtp)
384384
{
385385
int cpu;
386386
unsigned long flags;
387+
bool gpdone = poll_state_synchronize_rcu(rtp->percpu_dequeue_gpseq);
387388
long n;
388389
long ncbs = 0;
389390
long ncbsnz = 0;
@@ -425,21 +426,23 @@ static int rcu_tasks_need_gpcb(struct rcu_tasks *rtp)
425426
WRITE_ONCE(rtp->percpu_enqueue_shift, order_base_2(nr_cpu_ids));
426427
smp_store_release(&rtp->percpu_enqueue_lim, 1);
427428
rtp->percpu_dequeue_gpseq = get_state_synchronize_rcu();
429+
gpdone = false;
428430
pr_info("Starting switch %s to CPU-0 callback queuing.\n", rtp->name);
429431
}
430432
raw_spin_unlock_irqrestore(&rtp->cbs_gbl_lock, flags);
431433
}
432-
if (rcu_task_cb_adjust && !ncbsnz &&
433-
poll_state_synchronize_rcu(rtp->percpu_dequeue_gpseq)) {
434+
if (rcu_task_cb_adjust && !ncbsnz && gpdone) {
434435
raw_spin_lock_irqsave(&rtp->cbs_gbl_lock, flags);
435436
if (rtp->percpu_enqueue_lim < rtp->percpu_dequeue_lim) {
436437
WRITE_ONCE(rtp->percpu_dequeue_lim, 1);
437438
pr_info("Completing switch %s to CPU-0 callback queuing.\n", rtp->name);
438439
}
439-
for (cpu = rtp->percpu_dequeue_lim; cpu < nr_cpu_ids; cpu++) {
440-
struct rcu_tasks_percpu *rtpcp = per_cpu_ptr(rtp->rtpcpu, cpu);
440+
if (rtp->percpu_dequeue_lim == 1) {
441+
for (cpu = rtp->percpu_dequeue_lim; cpu < nr_cpu_ids; cpu++) {
442+
struct rcu_tasks_percpu *rtpcp = per_cpu_ptr(rtp->rtpcpu, cpu);
441443

442-
WARN_ON_ONCE(rcu_segcblist_n_cbs(&rtpcp->cblist));
444+
WARN_ON_ONCE(rcu_segcblist_n_cbs(&rtpcp->cblist));
445+
}
443446
}
444447
raw_spin_unlock_irqrestore(&rtp->cbs_gbl_lock, flags);
445448
}

0 commit comments

Comments
 (0)