diff options
| -rw-r--r-- | drivers/cpuidle/cpuidle.c | 13 | ||||
| -rw-r--r-- | include/linux/cpu.h | 7 | ||||
| -rw-r--r-- | include/linux/cpuidle.h | 6 | ||||
| -rw-r--r-- | kernel/sched/idle.c | 14 | 
4 files changed, 23 insertions, 17 deletions
diff --git a/drivers/cpuidle/cpuidle.c b/drivers/cpuidle/cpuidle.c index bf9b030cd7e1..12077db1158e 100644 --- a/drivers/cpuidle/cpuidle.c +++ b/drivers/cpuidle/cpuidle.c @@ -99,20 +99,21 @@ static int find_deepest_state(struct cpuidle_driver *drv,  }  /** - * cpuidle_use_deepest_state - Set/clear governor override flag. - * @enable: New value of the flag. + * cpuidle_use_deepest_state - Set/unset governor override mode. + * @latency_limit_ns: Idle state exit latency limit (or no override if 0).   * - * Set/unset the current CPU to use the deepest idle state (override governors - * going forward if set). + * If @latency_limit_ns is nonzero, set the current CPU to use the deepest idle + * state with exit latency within @latency_limit_ns (override governors going + * forward), or do not override governors if it is zero.   */ -void cpuidle_use_deepest_state(bool enable) +void cpuidle_use_deepest_state(u64 latency_limit_ns)  {  	struct cpuidle_device *dev;  	preempt_disable();  	dev = cpuidle_get_device();  	if (dev) -		dev->use_deepest_state = enable; +		dev->forced_idle_latency_limit_ns = latency_limit_ns;  	preempt_enable();  } diff --git a/include/linux/cpu.h b/include/linux/cpu.h index d0633ebdaa9c..cc03a7848b63 100644 --- a/include/linux/cpu.h +++ b/include/linux/cpu.h @@ -179,7 +179,12 @@ void arch_cpu_idle_dead(void);  int cpu_report_state(int cpu);  int cpu_check_up_prepare(int cpu);  void cpu_set_state_online(int cpu); -void play_idle(unsigned long duration_us); +void play_idle_precise(u64 duration_ns, u64 latency_ns); + +static inline void play_idle(unsigned long duration_us) +{ +	play_idle_precise(duration_us * NSEC_PER_USEC, U64_MAX); +}  #ifdef CONFIG_HOTPLUG_CPU  bool cpu_wait_death(unsigned int cpu, int seconds); diff --git a/include/linux/cpuidle.h b/include/linux/cpuidle.h index afb6a573b46d..72b26ff1de4b 100644 --- a/include/linux/cpuidle.h +++ b/include/linux/cpuidle.h @@ -85,7 +85,6 @@ struct cpuidle_driver_kobj;  struct cpuidle_device {  	unsigned int		registered:1;  	unsigned int		enabled:1; -	unsigned int		use_deepest_state:1;  	unsigned int		poll_time_limit:1;  	unsigned int		cpu;  	ktime_t			next_hrtimer; @@ -93,6 +92,7 @@ struct cpuidle_device {  	int			last_state_idx;  	u64			last_residency_ns;  	u64			poll_limit_ns; +	u64			forced_idle_latency_limit_ns;  	struct cpuidle_state_usage	states_usage[CPUIDLE_STATE_MAX];  	struct cpuidle_state_kobj *kobjs[CPUIDLE_STATE_MAX];  	struct cpuidle_driver_kobj *kobj_driver; @@ -216,7 +216,7 @@ extern int cpuidle_find_deepest_state(struct cpuidle_driver *drv,  				      struct cpuidle_device *dev);  extern int cpuidle_enter_s2idle(struct cpuidle_driver *drv,  				struct cpuidle_device *dev); -extern void cpuidle_use_deepest_state(bool enable); +extern void cpuidle_use_deepest_state(u64 latency_limit_ns);  #else  static inline int cpuidle_find_deepest_state(struct cpuidle_driver *drv,  					     struct cpuidle_device *dev) @@ -224,7 +224,7 @@ static inline int cpuidle_find_deepest_state(struct cpuidle_driver *drv,  static inline int cpuidle_enter_s2idle(struct cpuidle_driver *drv,  				       struct cpuidle_device *dev)  {return -ENODEV; } -static inline void cpuidle_use_deepest_state(bool enable) +static inline void cpuidle_use_deepest_state(u64 latency_limit_ns)  {  }  #endif diff --git a/kernel/sched/idle.c b/kernel/sched/idle.c index 1aa260702b38..cd05ffa0abfe 100644 --- a/kernel/sched/idle.c +++ b/kernel/sched/idle.c @@ -165,7 +165,7 @@ static void cpuidle_idle_call(void)  	 * until a proper wakeup interrupt happens.  	 */ -	if (idle_should_enter_s2idle() || dev->use_deepest_state) { +	if (idle_should_enter_s2idle() || dev->forced_idle_latency_limit_ns) {  		if (idle_should_enter_s2idle()) {  			rcu_idle_enter(); @@ -311,7 +311,7 @@ static enum hrtimer_restart idle_inject_timer_fn(struct hrtimer *timer)  	return HRTIMER_NORESTART;  } -void play_idle(unsigned long duration_us) +void play_idle_precise(u64 duration_ns, u64 latency_ns)  {  	struct idle_timer it; @@ -323,29 +323,29 @@ void play_idle(unsigned long duration_us)  	WARN_ON_ONCE(current->nr_cpus_allowed != 1);  	WARN_ON_ONCE(!(current->flags & PF_KTHREAD));  	WARN_ON_ONCE(!(current->flags & PF_NO_SETAFFINITY)); -	WARN_ON_ONCE(!duration_us); +	WARN_ON_ONCE(!duration_ns);  	rcu_sleep_check();  	preempt_disable();  	current->flags |= PF_IDLE; -	cpuidle_use_deepest_state(true); +	cpuidle_use_deepest_state(latency_ns);  	it.done = 0;  	hrtimer_init_on_stack(&it.timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL);  	it.timer.function = idle_inject_timer_fn; -	hrtimer_start(&it.timer, ns_to_ktime(duration_us * NSEC_PER_USEC), +	hrtimer_start(&it.timer, ns_to_ktime(duration_ns),  		      HRTIMER_MODE_REL_PINNED);  	while (!READ_ONCE(it.done))  		do_idle(); -	cpuidle_use_deepest_state(false); +	cpuidle_use_deepest_state(0);  	current->flags &= ~PF_IDLE;  	preempt_fold_need_resched();  	preempt_enable();  } -EXPORT_SYMBOL_GPL(play_idle); +EXPORT_SYMBOL_GPL(play_idle_precise);  void cpu_startup_entry(enum cpuhp_state state)  {  |