aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--kernel/rcutree.c67
-rw-r--r--kernel/rcutree.h8
2 files changed, 34 insertions, 41 deletions
diff --git a/kernel/rcutree.c b/kernel/rcutree.c
index 69f4efec3449..26249abf24dc 100644
--- a/kernel/rcutree.c
+++ b/kernel/rcutree.c
@@ -178,9 +178,29 @@ static struct rcu_node *rcu_get_root(struct rcu_state *rsp)
178 return &rsp->node[0]; 178 return &rsp->node[0];
179} 179}
180 180
181/*
182 * Record the specified "completed" value, which is later used to validate
183 * dynticks counter manipulations and CPU-offline checks. Specify
184 * "rsp->completed - 1" to unconditionally invalidate any future dynticks
185 * manipulations and CPU-offline checks. Such invalidation is useful at
186 * the beginning of a grace period.
187 */
188static void dyntick_record_completed(struct rcu_state *rsp, long comp)
189{
190 rsp->dynticks_completed = comp;
191}
192
181#ifdef CONFIG_SMP 193#ifdef CONFIG_SMP
182 194
183/* 195/*
196 * Recall the previously recorded value of the completion for dynticks.
197 */
198static long dyntick_recall_completed(struct rcu_state *rsp)
199{
200 return rsp->dynticks_completed;
201}
202
203/*
184 * If the specified CPU is offline, tell the caller that it is in 204 * If the specified CPU is offline, tell the caller that it is in
185 * a quiescent state. Otherwise, whack it with a reschedule IPI. 205 * a quiescent state. Otherwise, whack it with a reschedule IPI.
186 * Grace periods can end up waiting on an offline CPU when that 206 * Grace periods can end up waiting on an offline CPU when that
@@ -337,28 +357,9 @@ void rcu_irq_exit(void)
337 set_need_resched(); 357 set_need_resched();
338} 358}
339 359
340/*
341 * Record the specified "completed" value, which is later used to validate
342 * dynticks counter manipulations. Specify "rsp->completed - 1" to
343 * unconditionally invalidate any future dynticks manipulations (which is
344 * useful at the beginning of a grace period).
345 */
346static void dyntick_record_completed(struct rcu_state *rsp, long comp)
347{
348 rsp->dynticks_completed = comp;
349}
350
351#ifdef CONFIG_SMP 360#ifdef CONFIG_SMP
352 361
353/* 362/*
354 * Recall the previously recorded value of the completion for dynticks.
355 */
356static long dyntick_recall_completed(struct rcu_state *rsp)
357{
358 return rsp->dynticks_completed;
359}
360
361/*
362 * Snapshot the specified CPU's dynticks counter so that we can later 363 * Snapshot the specified CPU's dynticks counter so that we can later
363 * credit them with an implicit quiescent state. Return 1 if this CPU 364 * credit them with an implicit quiescent state. Return 1 if this CPU
364 * is in dynticks idle mode, which is an extended quiescent state. 365 * is in dynticks idle mode, which is an extended quiescent state.
@@ -421,24 +422,8 @@ static int rcu_implicit_dynticks_qs(struct rcu_data *rdp)
421 422
422#else /* #ifdef CONFIG_NO_HZ */ 423#else /* #ifdef CONFIG_NO_HZ */
423 424
424static void dyntick_record_completed(struct rcu_state *rsp, long comp)
425{
426}
427
428#ifdef CONFIG_SMP 425#ifdef CONFIG_SMP
429 426
430/*
431 * If there are no dynticks, then the only way that a CPU can passively
432 * be in a quiescent state is to be offline. Unlike dynticks idle, which
433 * is a point in time during the prior (already finished) grace period,
434 * an offline CPU is always in a quiescent state, and thus can be
435 * unconditionally applied. So just return the current value of completed.
436 */
437static long dyntick_recall_completed(struct rcu_state *rsp)
438{
439 return rsp->completed;
440}
441
442static int dyntick_save_progress_counter(struct rcu_data *rdp) 427static int dyntick_save_progress_counter(struct rcu_data *rdp)
443{ 428{
444 return 0; 429 return 0;
@@ -1146,6 +1131,7 @@ static void force_quiescent_state(struct rcu_state *rsp, int relaxed)
1146 long lastcomp; 1131 long lastcomp;
1147 struct rcu_node *rnp = rcu_get_root(rsp); 1132 struct rcu_node *rnp = rcu_get_root(rsp);
1148 u8 signaled; 1133 u8 signaled;
1134 u8 forcenow;
1149 1135
1150 if (!rcu_gp_in_progress(rsp)) 1136 if (!rcu_gp_in_progress(rsp))
1151 return; /* No grace period in progress, nothing to force. */ 1137 return; /* No grace period in progress, nothing to force. */
@@ -1182,16 +1168,23 @@ static void force_quiescent_state(struct rcu_state *rsp, int relaxed)
1182 if (rcu_process_dyntick(rsp, lastcomp, 1168 if (rcu_process_dyntick(rsp, lastcomp,
1183 dyntick_save_progress_counter)) 1169 dyntick_save_progress_counter))
1184 goto unlock_ret; 1170 goto unlock_ret;
1171 /* fall into next case. */
1172
1173 case RCU_SAVE_COMPLETED:
1185 1174
1186 /* Update state, record completion counter. */ 1175 /* Update state, record completion counter. */
1176 forcenow = 0;
1187 spin_lock(&rnp->lock); 1177 spin_lock(&rnp->lock);
1188 if (lastcomp == rsp->completed && 1178 if (lastcomp == rsp->completed &&
1189 rsp->signaled == RCU_SAVE_DYNTICK) { 1179 rsp->signaled == signaled) {
1190 rsp->signaled = RCU_FORCE_QS; 1180 rsp->signaled = RCU_FORCE_QS;
1191 dyntick_record_completed(rsp, lastcomp); 1181 dyntick_record_completed(rsp, lastcomp);
1182 forcenow = signaled == RCU_SAVE_COMPLETED;
1192 } 1183 }
1193 spin_unlock(&rnp->lock); 1184 spin_unlock(&rnp->lock);
1194 break; 1185 if (!forcenow)
1186 break;
1187 /* fall into next case. */
1195 1188
1196 case RCU_FORCE_QS: 1189 case RCU_FORCE_QS:
1197 1190
diff --git a/kernel/rcutree.h b/kernel/rcutree.h
index 1899023b0962..8a4c1650ad8d 100644
--- a/kernel/rcutree.h
+++ b/kernel/rcutree.h
@@ -204,11 +204,12 @@ struct rcu_data {
204#define RCU_GP_IDLE 0 /* No grace period in progress. */ 204#define RCU_GP_IDLE 0 /* No grace period in progress. */
205#define RCU_GP_INIT 1 /* Grace period being initialized. */ 205#define RCU_GP_INIT 1 /* Grace period being initialized. */
206#define RCU_SAVE_DYNTICK 2 /* Need to scan dyntick state. */ 206#define RCU_SAVE_DYNTICK 2 /* Need to scan dyntick state. */
207#define RCU_FORCE_QS 3 /* Need to force quiescent state. */ 207#define RCU_SAVE_COMPLETED 3 /* Need to save rsp->completed. */
208#define RCU_FORCE_QS 4 /* Need to force quiescent state. */
208#ifdef CONFIG_NO_HZ 209#ifdef CONFIG_NO_HZ
209#define RCU_SIGNAL_INIT RCU_SAVE_DYNTICK 210#define RCU_SIGNAL_INIT RCU_SAVE_DYNTICK
210#else /* #ifdef CONFIG_NO_HZ */ 211#else /* #ifdef CONFIG_NO_HZ */
211#define RCU_SIGNAL_INIT RCU_FORCE_QS 212#define RCU_SIGNAL_INIT RCU_SAVE_COMPLETED
212#endif /* #else #ifdef CONFIG_NO_HZ */ 213#endif /* #else #ifdef CONFIG_NO_HZ */
213 214
214#define RCU_JIFFIES_TILL_FORCE_QS 3 /* for rsp->jiffies_force_qs */ 215#define RCU_JIFFIES_TILL_FORCE_QS 3 /* for rsp->jiffies_force_qs */
@@ -274,9 +275,8 @@ struct rcu_state {
274 unsigned long jiffies_stall; /* Time at which to check */ 275 unsigned long jiffies_stall; /* Time at which to check */
275 /* for CPU stalls. */ 276 /* for CPU stalls. */
276#endif /* #ifdef CONFIG_RCU_CPU_STALL_DETECTOR */ 277#endif /* #ifdef CONFIG_RCU_CPU_STALL_DETECTOR */
277#ifdef CONFIG_NO_HZ
278 long dynticks_completed; /* Value of completed @ snap. */ 278 long dynticks_completed; /* Value of completed @ snap. */
279#endif /* #ifdef CONFIG_NO_HZ */ 279 /* Protected by fqslock. */
280}; 280};
281 281
282#ifdef RCU_TREE_NONCORE 282#ifdef RCU_TREE_NONCORE