From: Paul E. McKenney Date: Sat, 22 Jun 2013 00:10:40 +0000 (-0700) Subject: nohz_full: Force RCU's grace-period kthreads onto timekeeping CPU X-Git-Tag: next-20130717~25^2 X-Git-Url: https://git.karo-electronics.de/?a=commitdiff_plain;h=0f791c956aa26d6d8142f7b3e7a9fbd8bdeffcd0;p=karo-tx-linux.git nohz_full: Force RCU's grace-period kthreads onto timekeeping CPU Because RCU's quiescent-state-forcing mechanism is used to drive the full-system-idle state machine, and because this mechanism is executed by RCU's grace-period kthreads, this commit forces these kthreads to run on the timekeeping CPU (tick_do_timer_cpu). To do otherwise would mean that the RCU grace-period kthreads would force the system into non-idle state every time they drove the state machine, which would be just a bit on the futile side. Signed-off-by: Paul E. McKenney Cc: Frederic Weisbecker Cc: Steven Rostedt --- diff --git a/kernel/rcutree.c b/kernel/rcutree.c index 7f8ac153c203..aa3f525e2a45 100644 --- a/kernel/rcutree.c +++ b/kernel/rcutree.c @@ -1286,6 +1286,7 @@ static int rcu_gp_init(struct rcu_state *rsp) struct rcu_data *rdp; struct rcu_node *rnp = rcu_get_root(rsp); + rcu_bind_gp_kthread(); raw_spin_lock_irq(&rnp->lock); rsp->gp_flags = 0; /* Clear all flags: New grace period. */ diff --git a/kernel/rcutree.h b/kernel/rcutree.h index 7326a3c58da7..1602c21bb012 100644 --- a/kernel/rcutree.h +++ b/kernel/rcutree.h @@ -558,6 +558,7 @@ static void rcu_sysidle_exit(struct rcu_dynticks *rdtp, int irq); static void rcu_sysidle_check_cpu(struct rcu_data *rdp, bool *isidle, unsigned long *maxj); static bool is_sysidle_rcu_state(struct rcu_state *rsp); +static void rcu_bind_gp_kthread(void); static void rcu_sysidle_report(struct rcu_state *rsp, int isidle, unsigned long maxj); static void rcu_sysidle_init_percpu_data(struct rcu_dynticks *rdtp); diff --git a/kernel/rcutree_plugin.h b/kernel/rcutree_plugin.h index b47ffb004d1d..a4d44c3e3320 100644 --- a/kernel/rcutree_plugin.h +++ b/kernel/rcutree_plugin.h @@ -2544,7 +2544,7 @@ static void rcu_sysidle_check_cpu(struct rcu_data *rdp, bool *isidle, if (!*isidle || rdp->rsp != rcu_sysidle_state || cpu_is_offline(rdp->cpu) || rdp->cpu == tick_do_timer_cpu) return; - /* WARN_ON_ONCE(smp_processor_id() != tick_do_timer_cpu); */ + WARN_ON_ONCE(smp_processor_id() != tick_do_timer_cpu); /* Pick up current idle and NMI-nesting counter and check. */ cur = atomic_read(&rdtp->dynticks_idle); @@ -2569,6 +2569,20 @@ static bool is_sysidle_rcu_state(struct rcu_state *rsp) return rsp == rcu_sysidle_state; } +/* + * Bind the grace-period kthread for the sysidle flavor of RCU to the + * timekeeping CPU. + */ +static void rcu_bind_gp_kthread(void) +{ + int cpu = ACCESS_ONCE(tick_do_timer_cpu); + + if (cpu < 0 || cpu >= nr_cpu_ids) + return; + if (raw_smp_processor_id() != cpu) + set_cpus_allowed_ptr(current, cpumask_of(cpu)); +} + /* * Return a delay in jiffies based on the number of CPUs, rcu_node * leaf fanout, and jiffies tick rate. The idea is to allow larger @@ -2758,6 +2772,10 @@ static bool is_sysidle_rcu_state(struct rcu_state *rsp) return false; } +static void rcu_bind_gp_kthread(void) +{ +} + static void rcu_sysidle_report(struct rcu_state *rsp, int isidle, unsigned long maxj) {