mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	rcu: Make RCU_FAST_NO_HZ take advantage of numbered callbacks
Because RCU callbacks are now associated with the number of the grace period that they must wait for, CPUs can now take advance callbacks corresponding to grace periods that ended while a given CPU was in dyntick-idle mode. This eliminates the need to try forcing the RCU state machine while entering idle, thus reducing the CPU intensiveness of RCU_FAST_NO_HZ, which should increase its energy efficiency. Signed-off-by: Paul E. McKenney <paul.mckenney@linaro.org> Signed-off-by: Paul E. McKenney <paulmck@linux.vnet.ibm.com>
This commit is contained in:
		
							parent
							
								
									b11cc5760a
								
							
						
					
					
						commit
						c0f4dfd4f9
					
				| @ -2490,6 +2490,17 @@ bytes respectively. Such letter suffixes can also be entirely omitted. | |||||||
| 			leaf rcu_node structure.  Useful for very large | 			leaf rcu_node structure.  Useful for very large | ||||||
| 			systems. | 			systems. | ||||||
| 
 | 
 | ||||||
|  | 	rcutree.jiffies_till_first_fqs= [KNL,BOOT] | ||||||
|  | 			Set delay from grace-period initialization to | ||||||
|  | 			first attempt to force quiescent states. | ||||||
|  | 			Units are jiffies, minimum value is zero, | ||||||
|  | 			and maximum value is HZ. | ||||||
|  | 
 | ||||||
|  | 	rcutree.jiffies_till_next_fqs= [KNL,BOOT] | ||||||
|  | 			Set delay between subsequent attempts to force | ||||||
|  | 			quiescent states.  Units are jiffies, minimum | ||||||
|  | 			value is one, and maximum value is HZ. | ||||||
|  | 
 | ||||||
| 	rcutree.qhimark=	[KNL,BOOT] | 	rcutree.qhimark=	[KNL,BOOT] | ||||||
| 			Set threshold of queued | 			Set threshold of queued | ||||||
| 			RCU callbacks over which batch limiting is disabled. | 			RCU callbacks over which batch limiting is disabled. | ||||||
| @ -2504,16 +2515,15 @@ bytes respectively. Such letter suffixes can also be entirely omitted. | |||||||
| 	rcutree.rcu_cpu_stall_timeout= [KNL,BOOT] | 	rcutree.rcu_cpu_stall_timeout= [KNL,BOOT] | ||||||
| 			Set timeout for RCU CPU stall warning messages. | 			Set timeout for RCU CPU stall warning messages. | ||||||
| 
 | 
 | ||||||
| 	rcutree.jiffies_till_first_fqs= [KNL,BOOT] | 	rcutree.rcu_idle_gp_delay=	[KNL,BOOT] | ||||||
| 			Set delay from grace-period initialization to | 			Set wakeup interval for idle CPUs that have | ||||||
| 			first attempt to force quiescent states. | 			RCU callbacks (RCU_FAST_NO_HZ=y). | ||||||
| 			Units are jiffies, minimum value is zero, |  | ||||||
| 			and maximum value is HZ. |  | ||||||
| 
 | 
 | ||||||
| 	rcutree.jiffies_till_next_fqs= [KNL,BOOT] | 	rcutree.rcu_idle_lazy_gp_delay=	[KNL,BOOT] | ||||||
| 			Set delay between subsequent attempts to force | 			Set wakeup interval for idle CPUs that have | ||||||
| 			quiescent states.  Units are jiffies, minimum | 			only "lazy" RCU callbacks (RCU_FAST_NO_HZ=y). | ||||||
| 			value is one, and maximum value is HZ. | 			Lazy RCU callbacks are those which RCU can | ||||||
|  | 			prove do nothing more than free memory. | ||||||
| 
 | 
 | ||||||
| 	rcutorture.fqs_duration= [KNL,BOOT] | 	rcutorture.fqs_duration= [KNL,BOOT] | ||||||
| 			Set duration of force_quiescent_state bursts. | 			Set duration of force_quiescent_state bursts. | ||||||
|  | |||||||
| @ -80,6 +80,7 @@ extern void do_trace_rcu_torture_read(char *rcutorturename, | |||||||
| #define UINT_CMP_LT(a, b)	(UINT_MAX / 2 < (a) - (b)) | #define UINT_CMP_LT(a, b)	(UINT_MAX / 2 < (a) - (b)) | ||||||
| #define ULONG_CMP_GE(a, b)	(ULONG_MAX / 2 >= (a) - (b)) | #define ULONG_CMP_GE(a, b)	(ULONG_MAX / 2 >= (a) - (b)) | ||||||
| #define ULONG_CMP_LT(a, b)	(ULONG_MAX / 2 < (a) - (b)) | #define ULONG_CMP_LT(a, b)	(ULONG_MAX / 2 < (a) - (b)) | ||||||
|  | #define ulong2long(a)		(*(long *)(&(a))) | ||||||
| 
 | 
 | ||||||
| /* Exported common interfaces */ | /* Exported common interfaces */ | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										15
									
								
								init/Kconfig
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								init/Kconfig
									
									
									
									
									
								
							| @ -582,13 +582,16 @@ config RCU_FAST_NO_HZ | |||||||
| 	depends on NO_HZ && SMP | 	depends on NO_HZ && SMP | ||||||
| 	default n | 	default n | ||||||
| 	help | 	help | ||||||
| 	  This option causes RCU to attempt to accelerate grace periods in | 	  This option permits CPUs to enter dynticks-idle state even if | ||||||
| 	  order to allow CPUs to enter dynticks-idle state more quickly. | 	  they have RCU callbacks queued, and prevents RCU from waking | ||||||
| 	  On the other hand, this option increases the overhead of the | 	  these CPUs up more than roughly once every four jiffies (by | ||||||
| 	  dynticks-idle checking, thus degrading scheduling latency. | 	  default, you can adjust this using the rcutree.rcu_idle_gp_delay | ||||||
|  | 	  parameter), thus improving energy efficiency.  On the other | ||||||
|  | 	  hand, this option increases the duration of RCU grace periods, | ||||||
|  | 	  for example, slowing down synchronize_rcu(). | ||||||
| 
 | 
 | ||||||
| 	  Say Y if energy efficiency is critically important, and you don't | 	  Say Y if energy efficiency is critically important, and you | ||||||
| 	  	care about real-time response. | 	  	don't care about increased grace-period durations. | ||||||
| 
 | 
 | ||||||
| 	  Say N if you are unsure. | 	  Say N if you are unsure. | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -2640,19 +2640,27 @@ static int rcu_pending(int cpu) | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Check to see if any future RCU-related work will need to be done |  * Return true if the specified CPU has any callback.  If all_lazy is | ||||||
|  * by the current CPU, even if none need be done immediately, returning |  * non-NULL, store an indication of whether all callbacks are lazy. | ||||||
|  * 1 if so. |  * (If there are no callbacks, all of them are deemed to be lazy.) | ||||||
|  */ |  */ | ||||||
| static int rcu_cpu_has_callbacks(int cpu) | static int rcu_cpu_has_callbacks(int cpu, bool *all_lazy) | ||||||
| { | { | ||||||
|  | 	bool al = true; | ||||||
|  | 	bool hc = false; | ||||||
|  | 	struct rcu_data *rdp; | ||||||
| 	struct rcu_state *rsp; | 	struct rcu_state *rsp; | ||||||
| 
 | 
 | ||||||
| 	/* RCU callbacks either ready or pending? */ | 	for_each_rcu_flavor(rsp) { | ||||||
| 	for_each_rcu_flavor(rsp) | 		rdp = per_cpu_ptr(rsp->rda, cpu); | ||||||
| 		if (per_cpu_ptr(rsp->rda, cpu)->nxtlist) | 		if (rdp->qlen != rdp->qlen_lazy) | ||||||
| 			return 1; | 			al = false; | ||||||
| 	return 0; | 		if (rdp->nxtlist) | ||||||
|  | 			hc = true; | ||||||
|  | 	} | ||||||
|  | 	if (all_lazy) | ||||||
|  | 		*all_lazy = al; | ||||||
|  | 	return hc; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
| @ -2871,7 +2879,6 @@ rcu_init_percpu_data(int cpu, struct rcu_state *rsp, int preemptible) | |||||||
| 	rdp->dynticks->dynticks_nesting = DYNTICK_TASK_EXIT_IDLE; | 	rdp->dynticks->dynticks_nesting = DYNTICK_TASK_EXIT_IDLE; | ||||||
| 	atomic_set(&rdp->dynticks->dynticks, | 	atomic_set(&rdp->dynticks->dynticks, | ||||||
| 		   (atomic_read(&rdp->dynticks->dynticks) & ~0x1) + 1); | 		   (atomic_read(&rdp->dynticks->dynticks) & ~0x1) + 1); | ||||||
| 	rcu_prepare_for_idle_init(cpu); |  | ||||||
| 	raw_spin_unlock(&rnp->lock);		/* irqs remain disabled. */ | 	raw_spin_unlock(&rnp->lock);		/* irqs remain disabled. */ | ||||||
| 
 | 
 | ||||||
| 	/* Add CPU to rcu_node bitmasks. */ | 	/* Add CPU to rcu_node bitmasks. */ | ||||||
| @ -2945,7 +2952,6 @@ static int __cpuinit rcu_cpu_notify(struct notifier_block *self, | |||||||
| 		 */ | 		 */ | ||||||
| 		for_each_rcu_flavor(rsp) | 		for_each_rcu_flavor(rsp) | ||||||
| 			rcu_cleanup_dying_cpu(rsp); | 			rcu_cleanup_dying_cpu(rsp); | ||||||
| 		rcu_cleanup_after_idle(cpu); |  | ||||||
| 		break; | 		break; | ||||||
| 	case CPU_DEAD: | 	case CPU_DEAD: | ||||||
| 	case CPU_DEAD_FROZEN: | 	case CPU_DEAD_FROZEN: | ||||||
|  | |||||||
| @ -88,18 +88,13 @@ struct rcu_dynticks { | |||||||
| 	int dynticks_nmi_nesting;   /* Track NMI nesting level. */ | 	int dynticks_nmi_nesting;   /* Track NMI nesting level. */ | ||||||
| 	atomic_t dynticks;	    /* Even value for idle, else odd. */ | 	atomic_t dynticks;	    /* Even value for idle, else odd. */ | ||||||
| #ifdef CONFIG_RCU_FAST_NO_HZ | #ifdef CONFIG_RCU_FAST_NO_HZ | ||||||
| 	int dyntick_drain;	    /* Prepare-for-idle state variable. */ | 	bool all_lazy;		    /* Are all CPU's CBs lazy? */ | ||||||
| 	unsigned long dyntick_holdoff; |  | ||||||
| 				    /* No retries for the jiffy of failure. */ |  | ||||||
| 	struct timer_list idle_gp_timer; |  | ||||||
| 				    /* Wake up CPU sleeping with callbacks. */ |  | ||||||
| 	unsigned long idle_gp_timer_expires; |  | ||||||
| 				    /* When to wake up CPU (for repost). */ |  | ||||||
| 	bool idle_first_pass;	    /* First pass of attempt to go idle? */ |  | ||||||
| 	unsigned long nonlazy_posted; | 	unsigned long nonlazy_posted; | ||||||
| 				    /* # times non-lazy CBs posted to CPU. */ | 				    /* # times non-lazy CBs posted to CPU. */ | ||||||
| 	unsigned long nonlazy_posted_snap; | 	unsigned long nonlazy_posted_snap; | ||||||
| 				    /* idle-period nonlazy_posted snapshot. */ | 				    /* idle-period nonlazy_posted snapshot. */ | ||||||
|  | 	unsigned long last_accelerate; | ||||||
|  | 				    /* Last jiffy CBs were accelerated. */ | ||||||
| 	int tick_nohz_enabled_snap; /* Previously seen value from sysfs. */ | 	int tick_nohz_enabled_snap; /* Previously seen value from sysfs. */ | ||||||
| #endif /* #ifdef CONFIG_RCU_FAST_NO_HZ */ | #endif /* #ifdef CONFIG_RCU_FAST_NO_HZ */ | ||||||
| }; | }; | ||||||
| @ -521,7 +516,6 @@ static int __cpuinit rcu_spawn_one_boost_kthread(struct rcu_state *rsp, | |||||||
| 						 struct rcu_node *rnp); | 						 struct rcu_node *rnp); | ||||||
| #endif /* #ifdef CONFIG_RCU_BOOST */ | #endif /* #ifdef CONFIG_RCU_BOOST */ | ||||||
| static void __cpuinit rcu_prepare_kthreads(int cpu); | static void __cpuinit rcu_prepare_kthreads(int cpu); | ||||||
| static void rcu_prepare_for_idle_init(int cpu); |  | ||||||
| static void rcu_cleanup_after_idle(int cpu); | static void rcu_cleanup_after_idle(int cpu); | ||||||
| static void rcu_prepare_for_idle(int cpu); | static void rcu_prepare_for_idle(int cpu); | ||||||
| static void rcu_idle_count_callbacks_posted(void); | static void rcu_idle_count_callbacks_posted(void); | ||||||
|  | |||||||
| @ -1543,14 +1543,7 @@ static void __cpuinit rcu_prepare_kthreads(int cpu) | |||||||
| int rcu_needs_cpu(int cpu, unsigned long *delta_jiffies) | int rcu_needs_cpu(int cpu, unsigned long *delta_jiffies) | ||||||
| { | { | ||||||
| 	*delta_jiffies = ULONG_MAX; | 	*delta_jiffies = ULONG_MAX; | ||||||
| 	return rcu_cpu_has_callbacks(cpu); | 	return rcu_cpu_has_callbacks(cpu, NULL); | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /*
 |  | ||||||
|  * Because we do not have RCU_FAST_NO_HZ, don't bother initializing for it. |  | ||||||
|  */ |  | ||||||
| static void rcu_prepare_for_idle_init(int cpu) |  | ||||||
| { |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
| @ -1587,16 +1580,6 @@ static void rcu_idle_count_callbacks_posted(void) | |||||||
|  * |  * | ||||||
|  * The following three proprocessor symbols control this state machine: |  * The following three proprocessor symbols control this state machine: | ||||||
|  * |  * | ||||||
|  * RCU_IDLE_FLUSHES gives the maximum number of times that we will attempt |  | ||||||
|  *	to satisfy RCU.  Beyond this point, it is better to incur a periodic |  | ||||||
|  *	scheduling-clock interrupt than to loop through the state machine |  | ||||||
|  *	at full power. |  | ||||||
|  * RCU_IDLE_OPT_FLUSHES gives the number of RCU_IDLE_FLUSHES that are |  | ||||||
|  *	optional if RCU does not need anything immediately from this |  | ||||||
|  *	CPU, even if this CPU still has RCU callbacks queued.  The first |  | ||||||
|  *	times through the state machine are mandatory: we need to give |  | ||||||
|  *	the state machine a chance to communicate a quiescent state |  | ||||||
|  *	to the RCU core. |  | ||||||
|  * RCU_IDLE_GP_DELAY gives the number of jiffies that a CPU is permitted |  * RCU_IDLE_GP_DELAY gives the number of jiffies that a CPU is permitted | ||||||
|  *	to sleep in dyntick-idle mode with RCU callbacks pending.  This |  *	to sleep in dyntick-idle mode with RCU callbacks pending.  This | ||||||
|  *	is sized to be roughly one RCU grace period.  Those energy-efficiency |  *	is sized to be roughly one RCU grace period.  Those energy-efficiency | ||||||
| @ -1612,15 +1595,9 @@ static void rcu_idle_count_callbacks_posted(void) | |||||||
|  * adjustment, they can be converted into kernel config parameters, though |  * adjustment, they can be converted into kernel config parameters, though | ||||||
|  * making the state machine smarter might be a better option. |  * making the state machine smarter might be a better option. | ||||||
|  */ |  */ | ||||||
| #define RCU_IDLE_FLUSHES 5		/* Number of dyntick-idle tries. */ |  | ||||||
| #define RCU_IDLE_OPT_FLUSHES 3		/* Optional dyntick-idle tries. */ |  | ||||||
| #define RCU_IDLE_GP_DELAY 4		/* Roughly one grace period. */ | #define RCU_IDLE_GP_DELAY 4		/* Roughly one grace period. */ | ||||||
| #define RCU_IDLE_LAZY_GP_DELAY (6 * HZ)	/* Roughly six seconds. */ | #define RCU_IDLE_LAZY_GP_DELAY (6 * HZ)	/* Roughly six seconds. */ | ||||||
| 
 | 
 | ||||||
| static int rcu_idle_flushes = RCU_IDLE_FLUSHES; |  | ||||||
| module_param(rcu_idle_flushes, int, 0644); |  | ||||||
| static int rcu_idle_opt_flushes = RCU_IDLE_OPT_FLUSHES; |  | ||||||
| module_param(rcu_idle_opt_flushes, int, 0644); |  | ||||||
| static int rcu_idle_gp_delay = RCU_IDLE_GP_DELAY; | static int rcu_idle_gp_delay = RCU_IDLE_GP_DELAY; | ||||||
| module_param(rcu_idle_gp_delay, int, 0644); | module_param(rcu_idle_gp_delay, int, 0644); | ||||||
| static int rcu_idle_lazy_gp_delay = RCU_IDLE_LAZY_GP_DELAY; | static int rcu_idle_lazy_gp_delay = RCU_IDLE_LAZY_GP_DELAY; | ||||||
| @ -1629,178 +1606,97 @@ module_param(rcu_idle_lazy_gp_delay, int, 0644); | |||||||
| extern int tick_nohz_enabled; | extern int tick_nohz_enabled; | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Does the specified flavor of RCU have non-lazy callbacks pending on |  * Try to advance callbacks for all flavors of RCU on the current CPU. | ||||||
|  * the specified CPU?  Both RCU flavor and CPU are specified by the |  * Afterwards, if there are any callbacks ready for immediate invocation, | ||||||
|  * rcu_data structure. |  * return true. | ||||||
|  */ |  */ | ||||||
| static bool __rcu_cpu_has_nonlazy_callbacks(struct rcu_data *rdp) | static bool rcu_try_advance_all_cbs(void) | ||||||
| { | { | ||||||
| 	return rdp->qlen != rdp->qlen_lazy; | 	bool cbs_ready = false; | ||||||
| } | 	struct rcu_data *rdp; | ||||||
|  | 	struct rcu_node *rnp; | ||||||
|  | 	struct rcu_state *rsp; | ||||||
| 
 | 
 | ||||||
| #ifdef CONFIG_TREE_PREEMPT_RCU | 	for_each_rcu_flavor(rsp) { | ||||||
|  | 		rdp = this_cpu_ptr(rsp->rda); | ||||||
|  | 		rnp = rdp->mynode; | ||||||
| 
 | 
 | ||||||
| /*
 | 		/*
 | ||||||
|  * Are there non-lazy RCU-preempt callbacks?  (There cannot be if there | 		 * Don't bother checking unless a grace period has | ||||||
|  * is no RCU-preempt in the kernel.) | 		 * completed since we last checked and there are | ||||||
|  | 		 * callbacks not yet ready to invoke. | ||||||
| 		 */ | 		 */ | ||||||
| static bool rcu_preempt_cpu_has_nonlazy_callbacks(int cpu) | 		if (rdp->completed != rnp->completed && | ||||||
| { | 		    rdp->nxttail[RCU_DONE_TAIL] != rdp->nxttail[RCU_NEXT_TAIL]) | ||||||
| 	struct rcu_data *rdp = &per_cpu(rcu_preempt_data, cpu); | 			rcu_process_gp_end(rsp, rdp); | ||||||
| 
 | 
 | ||||||
| 	return __rcu_cpu_has_nonlazy_callbacks(rdp); | 		if (cpu_has_callbacks_ready_to_invoke(rdp)) | ||||||
| } | 			cbs_ready = true; | ||||||
| 
 | 	} | ||||||
| #else /* #ifdef CONFIG_TREE_PREEMPT_RCU */ | 	return cbs_ready; | ||||||
| 
 |  | ||||||
| static bool rcu_preempt_cpu_has_nonlazy_callbacks(int cpu) |  | ||||||
| { |  | ||||||
| 	return 0; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #endif /* else #ifdef CONFIG_TREE_PREEMPT_RCU */ |  | ||||||
| 
 |  | ||||||
| /*
 |  | ||||||
|  * Does any flavor of RCU have non-lazy callbacks on the specified CPU? |  | ||||||
|  */ |  | ||||||
| static bool rcu_cpu_has_nonlazy_callbacks(int cpu) |  | ||||||
| { |  | ||||||
| 	return __rcu_cpu_has_nonlazy_callbacks(&per_cpu(rcu_sched_data, cpu)) || |  | ||||||
| 	       __rcu_cpu_has_nonlazy_callbacks(&per_cpu(rcu_bh_data, cpu)) || |  | ||||||
| 	       rcu_preempt_cpu_has_nonlazy_callbacks(cpu); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Allow the CPU to enter dyntick-idle mode if either: (1) There are no |  * Allow the CPU to enter dyntick-idle mode unless it has callbacks ready | ||||||
|  * callbacks on this CPU, (2) this CPU has not yet attempted to enter |  * to invoke.  If the CPU has callbacks, try to advance them.  Tell the | ||||||
|  * dyntick-idle mode, or (3) this CPU is in the process of attempting to |  * caller to set the timeout based on whether or not there are non-lazy | ||||||
|  * enter dyntick-idle mode.  Otherwise, if we have recently tried and failed |  * callbacks. | ||||||
|  * to enter dyntick-idle mode, we refuse to try to enter it.  After all, |  | ||||||
|  * it is better to incur scheduling-clock interrupts than to spin |  | ||||||
|  * continuously for the same time duration! |  | ||||||
|  * |  * | ||||||
|  * The delta_jiffies argument is used to store the time when RCU is |  * The caller must have disabled interrupts. | ||||||
|  * going to need the CPU again if it still has callbacks.  The reason |  | ||||||
|  * for this is that rcu_prepare_for_idle() might need to post a timer, |  | ||||||
|  * but if so, it will do so after tick_nohz_stop_sched_tick() has set |  | ||||||
|  * the wakeup time for this CPU.  This means that RCU's timer can be |  | ||||||
|  * delayed until the wakeup time, which defeats the purpose of posting |  | ||||||
|  * a timer. |  | ||||||
|  */ |  */ | ||||||
| int rcu_needs_cpu(int cpu, unsigned long *delta_jiffies) | int rcu_needs_cpu(int cpu, unsigned long *dj) | ||||||
| { | { | ||||||
| 	struct rcu_dynticks *rdtp = &per_cpu(rcu_dynticks, cpu); | 	struct rcu_dynticks *rdtp = &per_cpu(rcu_dynticks, cpu); | ||||||
| 
 | 
 | ||||||
| 	/* Flag a new idle sojourn to the idle-entry state machine. */ | 	/* Snapshot to detect later posting of non-lazy callback. */ | ||||||
| 	rdtp->idle_first_pass = 1; | 	rdtp->nonlazy_posted_snap = rdtp->nonlazy_posted; | ||||||
|  | 
 | ||||||
| 	/* If no callbacks, RCU doesn't need the CPU. */ | 	/* If no callbacks, RCU doesn't need the CPU. */ | ||||||
| 	if (!rcu_cpu_has_callbacks(cpu)) { | 	if (!rcu_cpu_has_callbacks(cpu, &rdtp->all_lazy)) { | ||||||
| 		*delta_jiffies = ULONG_MAX; | 		*dj = ULONG_MAX; | ||||||
| 		return 0; | 		return 0; | ||||||
| 	} | 	} | ||||||
| 	if (rdtp->dyntick_holdoff == jiffies) { | 
 | ||||||
| 		/* RCU recently tried and failed, so don't try again. */ | 	/* Attempt to advance callbacks. */ | ||||||
| 		*delta_jiffies = 1; | 	if (rcu_try_advance_all_cbs()) { | ||||||
|  | 		/* Some ready to invoke, so initiate later invocation. */ | ||||||
|  | 		invoke_rcu_core(); | ||||||
| 		return 1; | 		return 1; | ||||||
| 	} | 	} | ||||||
| 	/* Set up for the possibility that RCU will post a timer. */ | 	rdtp->last_accelerate = jiffies; | ||||||
| 	if (rcu_cpu_has_nonlazy_callbacks(cpu)) { | 
 | ||||||
| 		*delta_jiffies = round_up(rcu_idle_gp_delay + jiffies, | 	/* Request timer delay depending on laziness, and round. */ | ||||||
|  | 	if (rdtp->all_lazy) { | ||||||
|  | 		*dj = round_up(rcu_idle_gp_delay + jiffies, | ||||||
| 			       rcu_idle_gp_delay) - jiffies; | 			       rcu_idle_gp_delay) - jiffies; | ||||||
| 	} else { | 	} else { | ||||||
| 		*delta_jiffies = jiffies + rcu_idle_lazy_gp_delay; | 		*dj = round_jiffies(rcu_idle_lazy_gp_delay + jiffies) - jiffies; | ||||||
| 		*delta_jiffies = round_jiffies(*delta_jiffies) - jiffies; |  | ||||||
| 	} | 	} | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Handler for smp_call_function_single().  The only point of this |  * Prepare a CPU for idle from an RCU perspective.  The first major task | ||||||
|  * handler is to wake the CPU up, so the handler does only tracing. |  * is to sense whether nohz mode has been enabled or disabled via sysfs. | ||||||
|  */ |  * The second major task is to check to see if a non-lazy callback has | ||||||
| void rcu_idle_demigrate(void *unused) |  * arrived at a CPU that previously had only lazy callbacks.  The third | ||||||
| { |  * major task is to accelerate (that is, assign grace-period numbers to) | ||||||
| 	trace_rcu_prep_idle("Demigrate"); |  * any recently arrived callbacks. | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /*
 |  | ||||||
|  * Timer handler used to force CPU to start pushing its remaining RCU |  | ||||||
|  * callbacks in the case where it entered dyntick-idle mode with callbacks |  | ||||||
|  * pending.  The hander doesn't really need to do anything because the |  | ||||||
|  * real work is done upon re-entry to idle, or by the next scheduling-clock |  | ||||||
|  * interrupt should idle not be re-entered. |  | ||||||
|  * |  | ||||||
|  * One special case: the timer gets migrated without awakening the CPU |  | ||||||
|  * on which the timer was scheduled on.  In this case, we must wake up |  | ||||||
|  * that CPU.  We do so with smp_call_function_single(). |  | ||||||
|  */ |  | ||||||
| static void rcu_idle_gp_timer_func(unsigned long cpu_in) |  | ||||||
| { |  | ||||||
| 	int cpu = (int)cpu_in; |  | ||||||
| 
 |  | ||||||
| 	trace_rcu_prep_idle("Timer"); |  | ||||||
| 	if (cpu != smp_processor_id()) |  | ||||||
| 		smp_call_function_single(cpu, rcu_idle_demigrate, NULL, 0); |  | ||||||
| 	else |  | ||||||
| 		WARN_ON_ONCE(1); /* Getting here can hang the system... */ |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /*
 |  | ||||||
|  * Initialize the timer used to pull CPUs out of dyntick-idle mode. |  | ||||||
|  */ |  | ||||||
| static void rcu_prepare_for_idle_init(int cpu) |  | ||||||
| { |  | ||||||
| 	struct rcu_dynticks *rdtp = &per_cpu(rcu_dynticks, cpu); |  | ||||||
| 
 |  | ||||||
| 	rdtp->dyntick_holdoff = jiffies - 1; |  | ||||||
| 	setup_timer(&rdtp->idle_gp_timer, rcu_idle_gp_timer_func, cpu); |  | ||||||
| 	rdtp->idle_gp_timer_expires = jiffies - 1; |  | ||||||
| 	rdtp->idle_first_pass = 1; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /*
 |  | ||||||
|  * Clean up for exit from idle.  Because we are exiting from idle, there |  | ||||||
|  * is no longer any point to ->idle_gp_timer, so cancel it.  This will |  | ||||||
|  * do nothing if this timer is not active, so just cancel it unconditionally. |  | ||||||
|  */ |  | ||||||
| static void rcu_cleanup_after_idle(int cpu) |  | ||||||
| { |  | ||||||
| 	struct rcu_dynticks *rdtp = &per_cpu(rcu_dynticks, cpu); |  | ||||||
| 
 |  | ||||||
| 	del_timer(&rdtp->idle_gp_timer); |  | ||||||
| 	trace_rcu_prep_idle("Cleanup after idle"); |  | ||||||
| 	rdtp->tick_nohz_enabled_snap = ACCESS_ONCE(tick_nohz_enabled); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /*
 |  | ||||||
|  * Check to see if any RCU-related work can be done by the current CPU, |  | ||||||
|  * and if so, schedule a softirq to get it done.  This function is part |  | ||||||
|  * of the RCU implementation; it is -not- an exported member of the RCU API. |  | ||||||
|  * |  | ||||||
|  * The idea is for the current CPU to clear out all work required by the |  | ||||||
|  * RCU core for the current grace period, so that this CPU can be permitted |  | ||||||
|  * to enter dyntick-idle mode.  In some cases, it will need to be awakened |  | ||||||
|  * at the end of the grace period by whatever CPU ends the grace period. |  | ||||||
|  * This allows CPUs to go dyntick-idle more quickly, and to reduce the |  | ||||||
|  * number of wakeups by a modest integer factor. |  | ||||||
|  * |  | ||||||
|  * Because it is not legal to invoke rcu_process_callbacks() with irqs |  | ||||||
|  * disabled, we do one pass of force_quiescent_state(), then do a |  | ||||||
|  * invoke_rcu_core() to cause rcu_process_callbacks() to be invoked |  | ||||||
|  * later.  The ->dyntick_drain field controls the sequencing. |  | ||||||
|  * |  * | ||||||
|  * The caller must have disabled interrupts. |  * The caller must have disabled interrupts. | ||||||
|  */ |  */ | ||||||
| static void rcu_prepare_for_idle(int cpu) | static void rcu_prepare_for_idle(int cpu) | ||||||
| { | { | ||||||
| 	struct timer_list *tp; | 	struct rcu_data *rdp; | ||||||
| 	struct rcu_dynticks *rdtp = &per_cpu(rcu_dynticks, cpu); | 	struct rcu_dynticks *rdtp = &per_cpu(rcu_dynticks, cpu); | ||||||
|  | 	struct rcu_node *rnp; | ||||||
|  | 	struct rcu_state *rsp; | ||||||
| 	int tne; | 	int tne; | ||||||
| 
 | 
 | ||||||
| 	/* Handle nohz enablement switches conservatively. */ | 	/* Handle nohz enablement switches conservatively. */ | ||||||
| 	tne = ACCESS_ONCE(tick_nohz_enabled); | 	tne = ACCESS_ONCE(tick_nohz_enabled); | ||||||
| 	if (tne != rdtp->tick_nohz_enabled_snap) { | 	if (tne != rdtp->tick_nohz_enabled_snap) { | ||||||
| 		if (rcu_cpu_has_callbacks(cpu)) | 		if (rcu_cpu_has_callbacks(cpu, NULL)) | ||||||
| 			invoke_rcu_core(); /* force nohz to see update. */ | 			invoke_rcu_core(); /* force nohz to see update. */ | ||||||
| 		rdtp->tick_nohz_enabled_snap = tne; | 		rdtp->tick_nohz_enabled_snap = tne; | ||||||
| 		return; | 		return; | ||||||
| @ -1808,125 +1704,56 @@ static void rcu_prepare_for_idle(int cpu) | |||||||
| 	if (!tne) | 	if (!tne) | ||||||
| 		return; | 		return; | ||||||
| 
 | 
 | ||||||
| 	/* Adaptive-tick mode, where usermode execution is idle to RCU. */ | 	/* If this is a no-CBs CPU, no callbacks, just return. */ | ||||||
| 	if (!is_idle_task(current)) { | 	if (is_nocb_cpu(cpu)) | ||||||
| 		rdtp->dyntick_holdoff = jiffies - 1; |  | ||||||
| 		if (rcu_cpu_has_nonlazy_callbacks(cpu)) { |  | ||||||
| 			trace_rcu_prep_idle("User dyntick with callbacks"); |  | ||||||
| 			rdtp->idle_gp_timer_expires = |  | ||||||
| 				round_up(jiffies + rcu_idle_gp_delay, |  | ||||||
| 					 rcu_idle_gp_delay); |  | ||||||
| 		} else if (rcu_cpu_has_callbacks(cpu)) { |  | ||||||
| 			rdtp->idle_gp_timer_expires = |  | ||||||
| 				round_jiffies(jiffies + rcu_idle_lazy_gp_delay); |  | ||||||
| 			trace_rcu_prep_idle("User dyntick with lazy callbacks"); |  | ||||||
| 		} else { |  | ||||||
| 		return; | 		return; | ||||||
| 		} |  | ||||||
| 		tp = &rdtp->idle_gp_timer; |  | ||||||
| 		mod_timer_pinned(tp, rdtp->idle_gp_timer_expires); |  | ||||||
| 		return; |  | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * If this is an idle re-entry, for example, due to use of | 	 * If a non-lazy callback arrived at a CPU having only lazy | ||||||
| 	 * RCU_NONIDLE() or the new idle-loop tracing API within the idle | 	 * callbacks, invoke RCU core for the side-effect of recalculating | ||||||
| 	 * loop, then don't take any state-machine actions, unless the | 	 * idle duration on re-entry to idle. | ||||||
| 	 * momentary exit from idle queued additional non-lazy callbacks. |  | ||||||
| 	 * Instead, repost the ->idle_gp_timer if this CPU has callbacks |  | ||||||
| 	 * pending. |  | ||||||
| 	 */ | 	 */ | ||||||
| 	if (!rdtp->idle_first_pass && | 	if (rdtp->all_lazy && | ||||||
| 	    (rdtp->nonlazy_posted == rdtp->nonlazy_posted_snap)) { | 	    rdtp->nonlazy_posted != rdtp->nonlazy_posted_snap) { | ||||||
| 		if (rcu_cpu_has_callbacks(cpu)) { | 		invoke_rcu_core(); | ||||||
| 			tp = &rdtp->idle_gp_timer; | 		return; | ||||||
| 			mod_timer_pinned(tp, rdtp->idle_gp_timer_expires); | 	} | ||||||
| 		} | 
 | ||||||
| 		return; | 	/*
 | ||||||
| 	} | 	 * If we have not yet accelerated this jiffy, accelerate all | ||||||
| 	rdtp->idle_first_pass = 0; | 	 * callbacks on this CPU. | ||||||
| 	rdtp->nonlazy_posted_snap = rdtp->nonlazy_posted - 1; | 	 */ | ||||||
| 
 | 	if (rdtp->last_accelerate == jiffies) | ||||||
| 	/*
 | 		return; | ||||||
| 	 * If there are no callbacks on this CPU, enter dyntick-idle mode. | 	rdtp->last_accelerate = jiffies; | ||||||
| 	 * Also reset state to avoid prejudicing later attempts. | 	for_each_rcu_flavor(rsp) { | ||||||
| 	 */ | 		rdp = per_cpu_ptr(rsp->rda, cpu); | ||||||
| 	if (!rcu_cpu_has_callbacks(cpu)) { | 		if (!*rdp->nxttail[RCU_DONE_TAIL]) | ||||||
| 		rdtp->dyntick_holdoff = jiffies - 1; | 			continue; | ||||||
| 		rdtp->dyntick_drain = 0; | 		rnp = rdp->mynode; | ||||||
| 		trace_rcu_prep_idle("No callbacks"); | 		raw_spin_lock(&rnp->lock); /* irqs already disabled. */ | ||||||
| 		return; | 		rcu_accelerate_cbs(rsp, rnp, rdp); | ||||||
| 	} | 		raw_spin_unlock(&rnp->lock); /* irqs remain disabled. */ | ||||||
| 
 | 	} | ||||||
| 	/*
 | } | ||||||
| 	 * If in holdoff mode, just return.  We will presumably have | 
 | ||||||
| 	 * refrained from disabling the scheduling-clock tick. | /*
 | ||||||
| 	 */ |  * Clean up for exit from idle.  Attempt to advance callbacks based on | ||||||
| 	if (rdtp->dyntick_holdoff == jiffies) { |  * any grace periods that elapsed while the CPU was idle, and if any | ||||||
| 		trace_rcu_prep_idle("In holdoff"); |  * callbacks are now ready to invoke, initiate invocation. | ||||||
| 		return; |  */ | ||||||
| 	} | static void rcu_cleanup_after_idle(int cpu) | ||||||
| 
 | { | ||||||
| 	/* Check and update the ->dyntick_drain sequencing. */ | 	struct rcu_data *rdp; | ||||||
| 	if (rdtp->dyntick_drain <= 0) { | 	struct rcu_state *rsp; | ||||||
| 		/* First time through, initialize the counter. */ | 
 | ||||||
| 		rdtp->dyntick_drain = rcu_idle_flushes; | 	if (is_nocb_cpu(cpu)) | ||||||
| 	} else if (rdtp->dyntick_drain <= rcu_idle_opt_flushes && | 		return; | ||||||
| 		   !rcu_pending(cpu) && | 	rcu_try_advance_all_cbs(); | ||||||
| 		   !local_softirq_pending()) { | 	for_each_rcu_flavor(rsp) { | ||||||
| 		/* Can we go dyntick-idle despite still having callbacks? */ | 		rdp = per_cpu_ptr(rsp->rda, cpu); | ||||||
| 		rdtp->dyntick_drain = 0; | 		if (cpu_has_callbacks_ready_to_invoke(rdp)) | ||||||
| 		rdtp->dyntick_holdoff = jiffies; |  | ||||||
| 		if (rcu_cpu_has_nonlazy_callbacks(cpu)) { |  | ||||||
| 			trace_rcu_prep_idle("Dyntick with callbacks"); |  | ||||||
| 			rdtp->idle_gp_timer_expires = |  | ||||||
| 				round_up(jiffies + rcu_idle_gp_delay, |  | ||||||
| 					 rcu_idle_gp_delay); |  | ||||||
| 		} else { |  | ||||||
| 			rdtp->idle_gp_timer_expires = |  | ||||||
| 				round_jiffies(jiffies + rcu_idle_lazy_gp_delay); |  | ||||||
| 			trace_rcu_prep_idle("Dyntick with lazy callbacks"); |  | ||||||
| 		} |  | ||||||
| 		tp = &rdtp->idle_gp_timer; |  | ||||||
| 		mod_timer_pinned(tp, rdtp->idle_gp_timer_expires); |  | ||||||
| 		rdtp->nonlazy_posted_snap = rdtp->nonlazy_posted; |  | ||||||
| 		return; /* Nothing more to do immediately. */ |  | ||||||
| 	} else if (--(rdtp->dyntick_drain) <= 0) { |  | ||||||
| 		/* We have hit the limit, so time to give up. */ |  | ||||||
| 		rdtp->dyntick_holdoff = jiffies; |  | ||||||
| 		trace_rcu_prep_idle("Begin holdoff"); |  | ||||||
| 		invoke_rcu_core();  /* Force the CPU out of dyntick-idle. */ |  | ||||||
| 		return; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	/*
 |  | ||||||
| 	 * Do one step of pushing the remaining RCU callbacks through |  | ||||||
| 	 * the RCU core state machine. |  | ||||||
| 	 */ |  | ||||||
| #ifdef CONFIG_TREE_PREEMPT_RCU |  | ||||||
| 	if (per_cpu(rcu_preempt_data, cpu).nxtlist) { |  | ||||||
| 		rcu_preempt_qs(cpu); |  | ||||||
| 		force_quiescent_state(&rcu_preempt_state); |  | ||||||
| 	} |  | ||||||
| #endif /* #ifdef CONFIG_TREE_PREEMPT_RCU */ |  | ||||||
| 	if (per_cpu(rcu_sched_data, cpu).nxtlist) { |  | ||||||
| 		rcu_sched_qs(cpu); |  | ||||||
| 		force_quiescent_state(&rcu_sched_state); |  | ||||||
| 	} |  | ||||||
| 	if (per_cpu(rcu_bh_data, cpu).nxtlist) { |  | ||||||
| 		rcu_bh_qs(cpu); |  | ||||||
| 		force_quiescent_state(&rcu_bh_state); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	/*
 |  | ||||||
| 	 * If RCU callbacks are still pending, RCU still needs this CPU. |  | ||||||
| 	 * So try forcing the callbacks through the grace period. |  | ||||||
| 	 */ |  | ||||||
| 	if (rcu_cpu_has_callbacks(cpu)) { |  | ||||||
| 		trace_rcu_prep_idle("More callbacks"); |  | ||||||
| 			invoke_rcu_core(); | 			invoke_rcu_core(); | ||||||
| 	} else { |  | ||||||
| 		trace_rcu_prep_idle("Callbacks drained"); |  | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -2034,16 +1861,13 @@ early_initcall(rcu_register_oom_notifier); | |||||||
| static void print_cpu_stall_fast_no_hz(char *cp, int cpu) | static void print_cpu_stall_fast_no_hz(char *cp, int cpu) | ||||||
| { | { | ||||||
| 	struct rcu_dynticks *rdtp = &per_cpu(rcu_dynticks, cpu); | 	struct rcu_dynticks *rdtp = &per_cpu(rcu_dynticks, cpu); | ||||||
| 	struct timer_list *tltp = &rdtp->idle_gp_timer; | 	unsigned long nlpd = rdtp->nonlazy_posted - rdtp->nonlazy_posted_snap; | ||||||
| 	char c; |  | ||||||
| 
 | 
 | ||||||
| 	c = rdtp->dyntick_holdoff == jiffies ? 'H' : '.'; | 	sprintf(cp, "last_accelerate: %04lx/%04lx, nonlazy_posted: %ld, %c%c", | ||||||
| 	if (timer_pending(tltp)) | 		rdtp->last_accelerate & 0xffff, jiffies & 0xffff, | ||||||
| 		sprintf(cp, "drain=%d %c timer=%lu", | 		ulong2long(nlpd), | ||||||
| 			rdtp->dyntick_drain, c, tltp->expires - jiffies); | 		rdtp->all_lazy ? 'L' : '.', | ||||||
| 	else | 		rdtp->tick_nohz_enabled_snap ? '.' : 'D'); | ||||||
| 		sprintf(cp, "drain=%d %c timer not pending", |  | ||||||
| 			rdtp->dyntick_drain, c); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #else /* #ifdef CONFIG_RCU_FAST_NO_HZ */ | #else /* #ifdef CONFIG_RCU_FAST_NO_HZ */ | ||||||
|  | |||||||
| @ -46,8 +46,6 @@ | |||||||
| #define RCU_TREE_NONCORE | #define RCU_TREE_NONCORE | ||||||
| #include "rcutree.h" | #include "rcutree.h" | ||||||
| 
 | 
 | ||||||
| #define ulong2long(a) (*(long *)(&(a))) |  | ||||||
| 
 |  | ||||||
| static int r_open(struct inode *inode, struct file *file, | static int r_open(struct inode *inode, struct file *file, | ||||||
| 					const struct seq_operations *op) | 					const struct seq_operations *op) | ||||||
| { | { | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Paul E. McKenney
						Paul E. McKenney