2014-01-26 23:42:01 -05:00
/*
* Generic entry point for the idle threads
*/
# include <linux/sched.h>
# include <linux/cpu.h>
# include <linux/cpuidle.h>
# include <linux/tick.h>
# include <linux/mm.h>
# include <linux/stackprotector.h>
# include <asm/tlb.h>
# include <trace/events/power.h>
2014-06-04 10:31:18 -07:00
# include "sched.h"
2014-01-26 23:42:01 -05:00
static int __read_mostly cpu_idle_force_poll ;
void cpu_idle_poll_ctrl ( bool enable )
{
if ( enable ) {
cpu_idle_force_poll + + ;
} else {
cpu_idle_force_poll - - ;
WARN_ON_ONCE ( cpu_idle_force_poll < 0 ) ;
}
}
# ifdef CONFIG_GENERIC_IDLE_POLL_SETUP
static int __init cpu_idle_poll_setup ( char * __unused )
{
cpu_idle_force_poll = 1 ;
return 1 ;
}
__setup ( " nohlt " , cpu_idle_poll_setup ) ;
static int __init cpu_idle_nopoll_setup ( char * __unused )
{
cpu_idle_force_poll = 0 ;
return 1 ;
}
__setup ( " hlt " , cpu_idle_nopoll_setup ) ;
# endif
static inline int cpu_idle_poll ( void )
{
rcu_idle_enter ( ) ;
trace_cpu_idle_rcuidle ( 0 , smp_processor_id ( ) ) ;
local_irq_enable ( ) ;
while ( ! tif_need_resched ( ) )
cpu_relax ( ) ;
trace_cpu_idle_rcuidle ( PWR_EVENT_EXIT , smp_processor_id ( ) ) ;
rcu_idle_exit ( ) ;
return 1 ;
}
/* Weak implementations for optional arch specific functions */
void __weak arch_cpu_idle_prepare ( void ) { }
void __weak arch_cpu_idle_enter ( void ) { }
void __weak arch_cpu_idle_exit ( void ) { }
void __weak arch_cpu_idle_dead ( void ) { }
void __weak arch_cpu_idle ( void )
{
cpu_idle_force_poll = 1 ;
local_irq_enable ( ) ;
}
2014-03-03 08:48:51 +01:00
/**
* cpuidle_idle_call - the main idle function
*
* NOTE : no locks or semaphores should be used here
2014-06-04 10:31:16 -07:00
*
* On archs that support TIF_POLLING_NRFLAG , is called with polling
* set , and it returns with polling set . If it ever stops polling , it
* must clear the polling bit .
2014-03-03 08:48:51 +01:00
*/
2014-04-21 01:26:58 +02:00
static void cpuidle_idle_call ( void )
2014-03-03 08:48:51 +01:00
{
struct cpuidle_device * dev = __this_cpu_read ( cpuidle_devices ) ;
struct cpuidle_driver * drv = cpuidle_get_cpu_driver ( dev ) ;
2014-04-11 13:55:48 +02:00
int next_state , entered_state ;
2014-06-24 10:01:01 +05:30
unsigned int broadcast ;
2014-03-03 08:48:51 +01:00
2014-03-03 08:48:54 +01:00
/*
* Check if the idle task must be rescheduled . If it is the
2014-04-11 13:47:16 +02:00
* case , exit the function after re - enabling the local irq .
2014-03-03 08:48:54 +01:00
*/
2014-04-11 13:47:16 +02:00
if ( need_resched ( ) ) {
2014-03-03 08:48:53 +01:00
local_irq_enable ( ) ;
2014-04-21 01:26:58 +02:00
return ;
2014-03-03 08:48:53 +01:00
}
2014-03-03 08:48:54 +01:00
/*
* During the idle period , stop measuring the disabled irqs
* critical sections latencies
*/
2014-03-03 08:48:52 +01:00
stop_critical_timings ( ) ;
2014-03-03 08:48:54 +01:00
/*
* Tell the RCU framework we are entering an idle section ,
* so no more rcu read side critical sections and one more
* step to the grace period
*/
2014-03-03 08:48:52 +01:00
rcu_idle_enter ( ) ;
2014-03-03 08:48:54 +01:00
/*
2014-05-01 00:13:47 +02:00
* Ask the cpuidle framework to choose a convenient idle state .
2014-05-22 10:37:06 +02:00
* Fall back to the default arch idle method on errors .
2014-03-03 08:48:54 +01:00
*/
2014-05-01 00:13:47 +02:00
next_state = cpuidle_select ( drv , dev ) ;
2014-05-22 10:37:06 +02:00
if ( next_state < 0 ) {
2014-04-11 13:55:48 +02:00
use_default :
2014-03-03 08:48:54 +01:00
/*
2014-04-11 13:55:48 +02:00
* We can ' t use the cpuidle framework , let ' s use the default
* idle routine .
2014-03-03 08:48:54 +01:00
*/
2014-04-11 13:55:48 +02:00
if ( current_clr_polling_and_test ( ) )
2014-03-03 08:48:53 +01:00
local_irq_enable ( ) ;
2014-04-11 13:55:48 +02:00
else
arch_cpu_idle ( ) ;
goto exit_idle ;
2014-03-03 08:48:53 +01:00
}
2014-04-11 13:55:48 +02:00
/*
* The idle task must be scheduled , it is pointless to
* go to idle , just update no idle residency and get
* out of this function
*/
if ( current_clr_polling_and_test ( ) ) {
dev - > last_residency = 0 ;
entered_state = next_state ;
local_irq_enable ( ) ;
goto exit_idle ;
2014-04-11 13:47:16 +02:00
}
2014-03-03 08:48:53 +01:00
2014-06-24 10:01:01 +05:30
broadcast = drv - > states [ next_state ] . flags & CPUIDLE_FLAG_TIMER_STOP ;
2014-04-11 13:55:48 +02:00
/*
* Tell the time framework to switch to a broadcast timer
* because our local timer will be shutdown . If a local timer
* is used from another cpu as a broadcast timer , this call may
* fail if it is not available
*/
if ( broadcast & &
clockevents_notify ( CLOCK_EVT_NOTIFY_BROADCAST_ENTER , & dev - > cpu ) )
goto use_default ;
/*
* Enter the idle state previously returned by the governor decision .
* This function will block until an interrupt occurs and will take
* care of re - enabling the local interrupts
*/
entered_state = cpuidle_enter ( drv , dev , next_state ) ;
if ( broadcast )
clockevents_notify ( CLOCK_EVT_NOTIFY_BROADCAST_EXIT , & dev - > cpu ) ;
/*
* Give the governor an opportunity to reflect on the outcome
*/
cpuidle_reflect ( dev , entered_state ) ;
exit_idle :
2014-03-03 08:48:53 +01:00
__current_set_polling ( ) ;
2014-03-03 08:48:51 +01:00
2014-03-03 08:48:54 +01:00
/*
2014-04-11 13:55:48 +02:00
* It is up to the idle functions to reenable local interrupts
2014-03-03 08:48:54 +01:00
*/
2014-03-03 08:48:52 +01:00
if ( WARN_ON_ONCE ( irqs_disabled ( ) ) )
local_irq_enable ( ) ;
rcu_idle_exit ( ) ;
start_critical_timings ( ) ;
2014-03-03 08:48:51 +01:00
}
2014-01-26 23:42:01 -05:00
/*
* Generic idle loop implementation
2014-06-04 10:31:16 -07:00
*
* Called with polling cleared .
2014-01-26 23:42:01 -05:00
*/
static void cpu_idle_loop ( void )
{
while ( 1 ) {
2014-06-04 10:31:16 -07:00
/*
* If the arch has a polling bit , we maintain an invariant :
*
* Our polling bit is clear if we ' re not scheduled ( i . e . if
* rq - > curr ! = rq - > idle ) . This means that , if rq - > idle has
* the polling bit set , then setting need_resched is
* guaranteed to cause the cpu to reschedule .
*/
__current_set_polling ( ) ;
2014-01-26 23:42:01 -05:00
tick_nohz_idle_enter ( ) ;
while ( ! need_resched ( ) ) {
check_pgt_cache ( ) ;
rmb ( ) ;
if ( cpu_is_offline ( smp_processor_id ( ) ) )
arch_cpu_idle_dead ( ) ;
local_irq_disable ( ) ;
arch_cpu_idle_enter ( ) ;
/*
* In poll mode we reenable interrupts and spin .
*
* Also if we detected in the wakeup from idle
* path that the tick broadcast device expired
* for us , we don ' t want to go deep idle as we
* know that the IPI is going to arrive right
* away
*/
2014-03-03 08:48:53 +01:00
if ( cpu_idle_force_poll | | tick_check_broadcast_expired ( ) )
2014-01-26 23:42:01 -05:00
cpu_idle_poll ( ) ;
2014-03-03 08:48:53 +01:00
else
cpuidle_idle_call ( ) ;
2014-01-26 23:42:01 -05:00
arch_cpu_idle_exit ( ) ;
}
2014-02-24 18:22:07 +01:00
/*
* Since we fell out of the loop above , we know
* TIF_NEED_RESCHED must be set , propagate it into
* PREEMPT_NEED_RESCHED .
*
* This is required because for polling idle loops we will
* not have had an IPI to fold the state for us .
*/
preempt_set_need_resched ( ) ;
2014-01-26 23:42:01 -05:00
tick_nohz_idle_exit ( ) ;
2014-06-04 10:31:16 -07:00
__current_clr_polling ( ) ;
/*
2014-06-04 10:31:18 -07:00
* We promise to call sched_ttwu_pending and reschedule
* if need_resched is set while polling is set . That
* means that clearing polling needs to be visible
* before doing these things .
2014-06-04 10:31:16 -07:00
*/
smp_mb__after_atomic ( ) ;
2014-06-04 10:31:18 -07:00
sched_ttwu_pending ( ) ;
2014-01-26 23:42:01 -05:00
schedule_preempt_disabled ( ) ;
}
}
void cpu_startup_entry ( enum cpuhp_state state )
{
/*
* This # ifdef needs to die , but it ' s too late in the cycle to
* make this generic ( arm and sh have never invoked the canary
* init for the non boot cpus ! ) . Will be fixed in 3.11
*/
# ifdef CONFIG_X86
/*
* If we ' re the non - boot CPU , nothing set the stack canary up
* for us . The boot CPU already has it initialized but no harm
* in doing it again . This is a good place for updating it , as
* we wont ever return from this function ( so the invalid
* canaries already on the stack wont ever trigger ) .
*/
boot_init_stack_canary ( ) ;
# endif
arch_cpu_idle_prepare ( ) ;
cpu_idle_loop ( ) ;
}