| xj | b04a402 | 2021-11-25 15:01:52 +0800 | [diff] [blame] | 1 | /* | 
 | 2 |  * poll_state.c - Polling idle state | 
 | 3 |  * | 
 | 4 |  * This file is released under the GPLv2. | 
 | 5 |  */ | 
 | 6 |  | 
 | 7 | #include <linux/cpuidle.h> | 
 | 8 | #include <linux/sched.h> | 
 | 9 | #include <linux/sched/clock.h> | 
 | 10 | #include <linux/sched/idle.h> | 
 | 11 |  | 
 | 12 | #define POLL_IDLE_TIME_LIMIT	(TICK_NSEC / 16) | 
 | 13 | #define POLL_IDLE_RELAX_COUNT	200 | 
 | 14 |  | 
 | 15 | static int __cpuidle poll_idle(struct cpuidle_device *dev, | 
 | 16 | 			       struct cpuidle_driver *drv, int index) | 
 | 17 | { | 
 | 18 | 	u64 time_start = local_clock(); | 
 | 19 |  | 
 | 20 | 	dev->poll_time_limit = false; | 
 | 21 |  | 
 | 22 | 	local_irq_enable(); | 
 | 23 | 	if (!current_set_polling_and_test()) { | 
 | 24 | 		unsigned int loop_count = 0; | 
 | 25 |  | 
 | 26 | 		while (!need_resched()) { | 
 | 27 | 			cpu_relax(); | 
 | 28 | 			if (loop_count++ < POLL_IDLE_RELAX_COUNT) | 
 | 29 | 				continue; | 
 | 30 |  | 
 | 31 | 			loop_count = 0; | 
 | 32 | 			if (local_clock() - time_start > POLL_IDLE_TIME_LIMIT) { | 
 | 33 | 				dev->poll_time_limit = true; | 
 | 34 | 				break; | 
 | 35 | 			} | 
 | 36 | 		} | 
 | 37 | 	} | 
 | 38 | 	current_clr_polling(); | 
 | 39 |  | 
 | 40 | 	return index; | 
 | 41 | } | 
 | 42 |  | 
 | 43 | void cpuidle_poll_state_init(struct cpuidle_driver *drv) | 
 | 44 | { | 
 | 45 | 	struct cpuidle_state *state = &drv->states[0]; | 
 | 46 |  | 
 | 47 | 	snprintf(state->name, CPUIDLE_NAME_LEN, "POLL"); | 
 | 48 | 	snprintf(state->desc, CPUIDLE_DESC_LEN, "CPUIDLE CORE POLL IDLE"); | 
 | 49 | 	state->exit_latency = 0; | 
 | 50 | 	state->target_residency = 0; | 
 | 51 | 	state->power_usage = -1; | 
 | 52 | 	state->enter = poll_idle; | 
 | 53 | 	state->disabled = false; | 
 | 54 | 	state->flags = CPUIDLE_FLAG_POLLING; | 
 | 55 | } | 
 | 56 | EXPORT_SYMBOL_GPL(cpuidle_poll_state_init); |