forked from Minki/linux
c633f9e788
In the old times, the whole idle task was considered as an RCU quiescent state. But as RCU became more and more successful overtime, some RCU read side critical section have been added even in the code of some architectures idle tasks, for tracing for example. So nowadays, rcu_idle_enter() and rcu_idle_exit() must be called by the architecture to tell RCU about the part in the idle loop that doesn't make use of rcu read side critical sections, typically the part that puts the CPU in low power mode. This is necessary for RCU to find the quiescent states in idle in order to complete grace periods. Add this missing pair of calls in the Cris's idle loop. Reported-by: Paul E. McKenney <paulmck@linux.vnet.ibm.com> Signed-off-by: Frederic Weisbecker <fweisbec@gmail.com> Cc: Mikael Starvik <starvik@axis.com> Cc: Jesper Nilsson <jesper.nilsson@axis.com> Cc: Cris <linux-cris-kernel@axis.com> Cc: <stable@vger.kernel.org> # 3.3+ Signed-off-by: Paul E. McKenney <paulmck@linux.vnet.ibm.com> Reviewed-by: Josh Triplett <josh@joshtriplett.org>
134 lines
2.8 KiB
C
134 lines
2.8 KiB
C
/*
|
|
* linux/arch/cris/kernel/process.c
|
|
*
|
|
* Copyright (C) 1995 Linus Torvalds
|
|
* Copyright (C) 2000-2002 Axis Communications AB
|
|
*
|
|
* Authors: Bjorn Wesen (bjornw@axis.com)
|
|
*
|
|
*/
|
|
|
|
/*
|
|
* This file handles the architecture-dependent parts of process handling..
|
|
*/
|
|
|
|
#include <linux/atomic.h>
|
|
#include <asm/pgtable.h>
|
|
#include <asm/uaccess.h>
|
|
#include <asm/irq.h>
|
|
#include <linux/module.h>
|
|
#include <linux/spinlock.h>
|
|
#include <linux/init_task.h>
|
|
#include <linux/sched.h>
|
|
#include <linux/fs.h>
|
|
#include <linux/user.h>
|
|
#include <linux/elfcore.h>
|
|
#include <linux/mqueue.h>
|
|
#include <linux/reboot.h>
|
|
#include <linux/rcupdate.h>
|
|
|
|
//#define DEBUG
|
|
|
|
/*
|
|
* The hlt_counter, disable_hlt and enable_hlt is just here as a hook if
|
|
* there would ever be a halt sequence (for power save when idle) with
|
|
* some largish delay when halting or resuming *and* a driver that can't
|
|
* afford that delay. The hlt_counter would then be checked before
|
|
* executing the halt sequence, and the driver marks the unhaltable
|
|
* region by enable_hlt/disable_hlt.
|
|
*/
|
|
|
|
int cris_hlt_counter=0;
|
|
|
|
void disable_hlt(void)
|
|
{
|
|
cris_hlt_counter++;
|
|
}
|
|
|
|
EXPORT_SYMBOL(disable_hlt);
|
|
|
|
void enable_hlt(void)
|
|
{
|
|
cris_hlt_counter--;
|
|
}
|
|
|
|
EXPORT_SYMBOL(enable_hlt);
|
|
|
|
/*
|
|
* The following aren't currently used.
|
|
*/
|
|
void (*pm_idle)(void);
|
|
|
|
extern void default_idle(void);
|
|
|
|
void (*pm_power_off)(void);
|
|
EXPORT_SYMBOL(pm_power_off);
|
|
|
|
/*
|
|
* The idle thread. There's no useful work to be
|
|
* done, so just try to conserve power and have a
|
|
* low exit latency (ie sit in a loop waiting for
|
|
* somebody to say that they'd like to reschedule)
|
|
*/
|
|
|
|
void cpu_idle (void)
|
|
{
|
|
/* endless idle loop with no priority at all */
|
|
while (1) {
|
|
rcu_idle_enter();
|
|
while (!need_resched()) {
|
|
void (*idle)(void);
|
|
/*
|
|
* Mark this as an RCU critical section so that
|
|
* synchronize_kernel() in the unload path waits
|
|
* for our completion.
|
|
*/
|
|
idle = pm_idle;
|
|
if (!idle)
|
|
idle = default_idle;
|
|
idle();
|
|
}
|
|
rcu_idle_exit();
|
|
schedule_preempt_disabled();
|
|
}
|
|
}
|
|
|
|
void hard_reset_now (void);
|
|
|
|
void machine_restart(char *cmd)
|
|
{
|
|
hard_reset_now();
|
|
}
|
|
|
|
/*
|
|
* Similar to machine_power_off, but don't shut off power. Add code
|
|
* here to freeze the system for e.g. post-mortem debug purpose when
|
|
* possible. This halt has nothing to do with the idle halt.
|
|
*/
|
|
|
|
void machine_halt(void)
|
|
{
|
|
}
|
|
|
|
/* If or when software power-off is implemented, add code here. */
|
|
|
|
void machine_power_off(void)
|
|
{
|
|
}
|
|
|
|
/*
|
|
* When a process does an "exec", machine state like FPU and debug
|
|
* registers need to be reset. This is a hook function for that.
|
|
* Currently we don't have any such state to reset, so this is empty.
|
|
*/
|
|
|
|
void flush_thread(void)
|
|
{
|
|
}
|
|
|
|
/* Fill in the fpu structure for a core dump. */
|
|
int dump_fpu(struct pt_regs *regs, elf_fpregset_t *fpu)
|
|
{
|
|
return 0;
|
|
}
|