2006-12-25 09:51:47 +09:00
/*
* arch / sh / kernel / process . c
2005-04-16 15:20:36 -07:00
*
2006-12-25 09:51:47 +09:00
* This file handles the architecture - dependent parts of process handling . .
2005-04-16 15:20:36 -07:00
*
* Copyright ( C ) 1995 Linus Torvalds
*
* SuperH version : Copyright ( C ) 1999 , 2000 Niibe Yutaka & Kaz Kojima
2006-10-12 12:16:13 +09:00
* Copyright ( C ) 2006 Lineo Solutions Inc . support SH4A UBC
2007-05-01 16:33:10 +09:00
* Copyright ( C ) 2002 - 2007 Paul Mundt
2005-04-16 15:20:36 -07:00
*/
# include <linux/module.h>
# include <linux/mm.h>
# include <linux/elfcore.h>
2006-02-01 03:06:08 -08:00
# include <linux/pm.h>
2005-04-16 15:20:36 -07:00
# include <linux/kallsyms.h>
2006-02-01 03:06:08 -08:00
# include <linux/kexec.h>
2007-05-09 10:55:38 +09:00
# include <linux/kdebug.h>
2007-05-09 17:33:24 +09:00
# include <linux/tick.h>
2007-05-14 12:52:56 +09:00
# include <linux/reboot.h>
2007-07-31 13:01:43 +09:00
# include <linux/fs.h>
2007-09-28 15:48:06 +09:00
# include <linux/preempt.h>
2005-04-16 15:20:36 -07:00
# include <asm/uaccess.h>
# include <asm/mmu_context.h>
2007-05-08 11:55:21 +09:00
# include <asm/pgalloc.h>
2007-05-08 14:50:59 +09:00
# include <asm/system.h>
2006-09-20 03:25:34 +09:00
# include <asm/ubc.h>
2008-03-26 19:02:47 +09:00
# include <asm/fpu.h>
2005-04-16 15:20:36 -07:00
2006-12-25 09:51:47 +09:00
static int hlt_counter ;
2005-04-16 15:20:36 -07:00
int ubc_usercnt = 0 ;
2006-02-01 03:06:08 -08:00
void ( * pm_idle ) ( void ) ;
void ( * pm_power_off ) ( void ) ;
EXPORT_SYMBOL ( pm_power_off ) ;
2007-05-14 19:39:48 +09:00
static int __init nohlt_setup ( char * __unused )
{
hlt_counter = 1 ;
return 1 ;
}
__setup ( " nohlt " , nohlt_setup ) ;
static int __init hlt_setup ( char * __unused )
{
hlt_counter = 0 ;
return 1 ;
}
__setup ( " hlt " , hlt_setup ) ;
2008-06-18 03:36:50 +03:00
static void default_idle ( void )
2006-02-01 03:06:08 -08:00
{
2007-05-14 19:39:48 +09:00
if ( ! hlt_counter ) {
clear_thread_flag ( TIF_POLLING_NRFLAG ) ;
smp_mb__after_clear_bit ( ) ;
set_bl_bit ( ) ;
while ( ! need_resched ( ) )
cpu_sleep ( ) ;
clear_bl_bit ( ) ;
set_thread_flag ( TIF_POLLING_NRFLAG ) ;
} else
while ( ! need_resched ( ) )
cpu_relax ( ) ;
2006-02-01 03:06:08 -08:00
}
[PATCH] sched: resched and cpu_idle rework
Make some changes to the NEED_RESCHED and POLLING_NRFLAG to reduce
confusion, and make their semantics rigid. Improves efficiency of
resched_task and some cpu_idle routines.
* In resched_task:
- TIF_NEED_RESCHED is only cleared with the task's runqueue lock held,
and as we hold it during resched_task, then there is no need for an
atomic test and set there. The only other time this should be set is
when the task's quantum expires, in the timer interrupt - this is
protected against because the rq lock is irq-safe.
- If TIF_NEED_RESCHED is set, then we don't need to do anything. It
won't get unset until the task get's schedule()d off.
- If we are running on the same CPU as the task we resched, then set
TIF_NEED_RESCHED and no further action is required.
- If we are running on another CPU, and TIF_POLLING_NRFLAG is *not* set
after TIF_NEED_RESCHED has been set, then we need to send an IPI.
Using these rules, we are able to remove the test and set operation in
resched_task, and make clear the previously vague semantics of
POLLING_NRFLAG.
* In idle routines:
- Enter cpu_idle with preempt disabled. When the need_resched() condition
becomes true, explicitly call schedule(). This makes things a bit clearer
(IMO), but haven't updated all architectures yet.
- Many do a test and clear of TIF_NEED_RESCHED for some reason. According
to the resched_task rules, this isn't needed (and actually breaks the
assumption that TIF_NEED_RESCHED is only cleared with the runqueue lock
held). So remove that. Generally one less locked memory op when switching
to the idle thread.
- Many idle routines clear TIF_POLLING_NRFLAG, and only set it in the inner
most polling idle loops. The above resched_task semantics allow it to be
set until before the last time need_resched() is checked before going into
a halt requiring interrupt wakeup.
Many idle routines simply never enter such a halt, and so POLLING_NRFLAG
can be always left set, completely eliminating resched IPIs when rescheduling
the idle task.
POLLING_NRFLAG width can be increased, to reduce the chance of resched IPIs.
Signed-off-by: Nick Piggin <npiggin@suse.de>
Cc: Ingo Molnar <mingo@elte.hu>
Cc: Con Kolivas <kernel@kolivas.org>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
2005-11-08 21:39:04 -08:00
void cpu_idle ( void )
2005-04-16 15:20:36 -07:00
{
2007-05-14 19:39:48 +09:00
set_thread_flag ( TIF_POLLING_NRFLAG ) ;
2005-04-16 15:20:36 -07:00
/* endless idle loop with no priority at all */
while ( 1 ) {
2006-02-01 03:06:08 -08:00
void ( * idle ) ( void ) = pm_idle ;
if ( ! idle )
idle = default_idle ;
2008-07-18 17:27:28 +02:00
tick_nohz_stop_sched_tick ( 1 ) ;
2006-02-01 03:06:08 -08:00
while ( ! need_resched ( ) )
idle ( ) ;
2007-05-09 17:33:24 +09:00
tick_nohz_restart_sched_tick ( ) ;
2005-04-16 15:20:36 -07:00
2005-11-08 21:39:01 -08:00
preempt_enable_no_resched ( ) ;
2005-04-16 15:20:36 -07:00
schedule ( ) ;
2005-11-08 21:39:01 -08:00
preempt_disable ( ) ;
2007-05-08 11:55:21 +09:00
check_pgt_cache ( ) ;
2005-04-16 15:20:36 -07:00
}
}
void machine_restart ( char * __unused )
{
/* SR.BL=1 and invoke address error to let CPU reset (manual reset) */
asm volatile ( " ldc %0, sr \n \t "
" mov.l @%1, %0 " : : " r " ( 0x10000000 ) , " r " ( 0x80000001 ) ) ;
}
void machine_halt ( void )
{
2006-02-01 03:06:08 -08:00
local_irq_disable ( ) ;
2005-04-16 15:20:36 -07:00
while ( 1 )
cpu_sleep ( ) ;
}
void machine_power_off ( void )
{
2006-02-01 03:06:08 -08:00
if ( pm_power_off )
pm_power_off ( ) ;
2005-04-16 15:20:36 -07:00
}
void show_regs ( struct pt_regs * regs )
{
printk ( " \n " ) ;
2007-10-18 23:40:41 -07:00
printk ( " Pid : %d, Comm: %20s \n " , task_pid_nr ( current ) , current - > comm ) ;
2006-10-12 17:07:45 +09:00
print_symbol ( " PC is at %s \n " , instruction_pointer ( regs ) ) ;
2005-04-16 15:20:36 -07:00
printk ( " PC : %08lx SP : %08lx SR : %08lx " ,
regs - > pc , regs - > regs [ 15 ] , regs - > sr ) ;
# ifdef CONFIG_MMU
printk ( " TEA : %08x " , ctrl_inl ( MMU_TEA ) ) ;
# else
printk ( " " ) ;
# endif
printk ( " %s \n " , print_tainted ( ) ) ;
printk ( " R0 : %08lx R1 : %08lx R2 : %08lx R3 : %08lx \n " ,
regs - > regs [ 0 ] , regs - > regs [ 1 ] ,
regs - > regs [ 2 ] , regs - > regs [ 3 ] ) ;
printk ( " R4 : %08lx R5 : %08lx R6 : %08lx R7 : %08lx \n " ,
regs - > regs [ 4 ] , regs - > regs [ 5 ] ,
regs - > regs [ 6 ] , regs - > regs [ 7 ] ) ;
printk ( " R8 : %08lx R9 : %08lx R10 : %08lx R11 : %08lx \n " ,
regs - > regs [ 8 ] , regs - > regs [ 9 ] ,
regs - > regs [ 10 ] , regs - > regs [ 11 ] ) ;
printk ( " R12 : %08lx R13 : %08lx R14 : %08lx \n " ,
regs - > regs [ 12 ] , regs - > regs [ 13 ] ,
regs - > regs [ 14 ] ) ;
printk ( " MACH: %08lx MACL: %08lx GBR : %08lx PR : %08lx \n " ,
regs - > mach , regs - > macl , regs - > gbr , regs - > pr ) ;
2006-10-12 17:07:45 +09:00
show_trace ( NULL , ( unsigned long * ) regs - > regs [ 15 ] , regs ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Create a kernel thread
*/
/*
* This is the mechanism for creating a new kernel thread .
*
*/
extern void kernel_thread_helper ( void ) ;
__asm__ ( " .align 5 \n "
" kernel_thread_helper: \n \t "
" jsr @r5 \n \t "
" nop \n \t "
" mov.l 1f, r1 \n \t "
" jsr @r1 \n \t "
" mov r0, r4 \n \t "
" .align 2 \n \t "
" 1:.long do_exit " ) ;
2006-12-25 09:51:47 +09:00
/* Don't use this in BL=1(cli). Or else, CPU resets! */
2005-04-16 15:20:36 -07:00
int kernel_thread ( int ( * fn ) ( void * ) , void * arg , unsigned long flags )
2006-12-25 09:51:47 +09:00
{
2005-04-16 15:20:36 -07:00
struct pt_regs regs ;
memset ( & regs , 0 , sizeof ( regs ) ) ;
2006-12-25 09:51:47 +09:00
regs . regs [ 4 ] = ( unsigned long ) arg ;
regs . regs [ 5 ] = ( unsigned long ) fn ;
2005-04-16 15:20:36 -07:00
2006-12-25 09:51:47 +09:00
regs . pc = ( unsigned long ) kernel_thread_helper ;
2005-04-16 15:20:36 -07:00
regs . sr = ( 1 < < 30 ) ;
/* Ok, create the new process.. */
2006-12-25 09:51:47 +09:00
return do_fork ( flags | CLONE_VM | CLONE_UNTRACED , 0 ,
& regs , 0 , NULL , NULL ) ;
2005-04-16 15:20:36 -07:00
}
/*
* Free current thread data structures etc . .
*/
void exit_thread ( void )
{
if ( current - > thread . ubc_pc ) {
current - > thread . ubc_pc = 0 ;
ubc_usercnt - = 1 ;
}
}
void flush_thread ( void )
{
# if defined(CONFIG_SH_FPU)
struct task_struct * tsk = current ;
/* Forget lazy FPU state */
2006-01-12 01:05:44 -08:00
clear_fpu ( tsk , task_pt_regs ( tsk ) ) ;
2005-04-16 15:20:36 -07:00
clear_used_math ( ) ;
# endif
}
void release_thread ( struct task_struct * dead_task )
{
/* do nothing */
}
/* Fill in the fpu structure for a core dump.. */
int dump_fpu ( struct pt_regs * regs , elf_fpregset_t * fpu )
{
int fpvalid = 0 ;
# if defined(CONFIG_SH_FPU)
struct task_struct * tsk = current ;
fpvalid = ! ! tsk_used_math ( tsk ) ;
if ( fpvalid ) {
unlazy_fpu ( tsk , regs ) ;
memcpy ( fpu , & tsk - > thread . fpu . hard , sizeof ( * fpu ) ) ;
}
# endif
return fpvalid ;
}
asmlinkage void ret_from_fork ( void ) ;
int copy_thread ( int nr , unsigned long clone_flags , unsigned long usp ,
unsigned long unused ,
struct task_struct * p , struct pt_regs * regs )
{
2006-09-27 17:07:07 +09:00
struct thread_info * ti = task_thread_info ( p ) ;
2005-04-16 15:20:36 -07:00
struct pt_regs * childregs ;
# if defined(CONFIG_SH_FPU)
struct task_struct * tsk = current ;
unlazy_fpu ( tsk , regs ) ;
p - > thread . fpu = tsk - > thread . fpu ;
copy_to_stopped_child_used_math ( p ) ;
# endif
2006-01-12 01:05:44 -08:00
childregs = task_pt_regs ( p ) ;
2005-04-16 15:20:36 -07:00
* childregs = * regs ;
if ( user_mode ( regs ) ) {
childregs - > regs [ 15 ] = usp ;
2006-09-27 17:07:07 +09:00
ti - > addr_limit = USER_DS ;
2005-04-16 15:20:36 -07:00
} else {
2007-02-28 18:35:42 +09:00
childregs - > regs [ 15 ] = ( unsigned long ) childregs ;
2006-09-27 17:07:07 +09:00
ti - > addr_limit = KERNEL_DS ;
2005-04-16 15:20:36 -07:00
}
2006-12-25 09:51:47 +09:00
2007-02-28 18:35:42 +09:00
if ( clone_flags & CLONE_SETTLS )
2005-04-16 15:20:36 -07:00
childregs - > gbr = childregs - > regs [ 0 ] ;
2006-12-25 09:51:47 +09:00
2005-04-16 15:20:36 -07:00
childregs - > regs [ 0 ] = 0 ; /* Set return value for child */
p - > thread . sp = ( unsigned long ) childregs ;
p - > thread . pc = ( unsigned long ) ret_from_fork ;
p - > thread . ubc_pc = 0 ;
return 0 ;
}
/* Tracing by user break controller. */
2006-12-25 09:51:47 +09:00
static void ubc_set_tracing ( int asid , unsigned long pc )
2005-04-16 15:20:36 -07:00
{
2006-10-12 12:16:13 +09:00
# if defined(CONFIG_CPU_SH4A)
unsigned long val ;
val = ( UBC_CBR_ID_INST | UBC_CBR_RW_READ | UBC_CBR_CE ) ;
val | = ( UBC_CBR_AIE | UBC_CBR_AIV_SET ( asid ) ) ;
ctrl_outl ( val , UBC_CBR0 ) ;
ctrl_outl ( pc , UBC_CAR0 ) ;
ctrl_outl ( 0x0 , UBC_CAMR0 ) ;
ctrl_outl ( 0x0 , UBC_CBCR ) ;
val = ( UBC_CRR_RES | UBC_CRR_PCB | UBC_CRR_BIE ) ;
ctrl_outl ( val , UBC_CRR0 ) ;
2006-12-25 09:51:47 +09:00
/* Read UBC register that we wrote last, for checking update */
2006-10-12 12:16:13 +09:00
val = ctrl_inl ( UBC_CRR0 ) ;
# else /* CONFIG_CPU_SH4A */
2005-04-16 15:20:36 -07:00
ctrl_outl ( pc , UBC_BARA ) ;
2006-09-27 17:25:07 +09:00
# ifdef CONFIG_MMU
2007-06-11 15:32:07 +09:00
ctrl_outb ( asid , UBC_BASRA ) ;
2006-09-27 17:25:07 +09:00
# endif
2005-04-16 15:20:36 -07:00
ctrl_outl ( 0 , UBC_BAMRA ) ;
2006-12-25 10:19:56 +09:00
if ( current_cpu_data . type = = CPU_SH7729 | |
2007-05-01 16:33:10 +09:00
current_cpu_data . type = = CPU_SH7710 | |
current_cpu_data . type = = CPU_SH7712 ) {
2005-04-16 15:20:36 -07:00
ctrl_outw ( BBR_INST | BBR_READ | BBR_CPU , UBC_BBRA ) ;
ctrl_outl ( BRCR_PCBA | BRCR_PCTE , UBC_BRCR ) ;
} else {
ctrl_outw ( BBR_INST | BBR_READ , UBC_BBRA ) ;
ctrl_outw ( BRCR_PCBA , UBC_BRCR ) ;
}
2006-10-12 12:16:13 +09:00
# endif /* CONFIG_CPU_SH4A */
2005-04-16 15:20:36 -07:00
}
/*
* switch_to ( x , y ) should switch tasks from x to y .
*
*/
2006-12-25 09:51:47 +09:00
struct task_struct * __switch_to ( struct task_struct * prev ,
struct task_struct * next )
2005-04-16 15:20:36 -07:00
{
# if defined(CONFIG_SH_FPU)
2006-01-12 01:05:44 -08:00
unlazy_fpu ( prev , task_pt_regs ( prev ) ) ;
2005-04-16 15:20:36 -07:00
# endif
2006-09-27 17:25:07 +09:00
# ifdef CONFIG_MMU
2005-04-16 15:20:36 -07:00
/*
* Restore the kernel mode register
2006-12-25 09:51:47 +09:00
* k7 ( r7_bank1 )
2005-04-16 15:20:36 -07:00
*/
asm volatile ( " ldc %0, r7_bank "
: /* no output */
2006-01-12 01:05:45 -08:00
: " r " ( task_thread_info ( next ) ) ) ;
2006-09-27 17:25:07 +09:00
# endif
2005-04-16 15:20:36 -07:00
/* If no tasks are using the UBC, we're done */
if ( ubc_usercnt = = 0 )
/* If no tasks are using the UBC, we're done */ ;
else if ( next - > thread . ubc_pc & & next - > mm ) {
2006-09-27 17:25:07 +09:00
int asid = 0 ;
# ifdef CONFIG_MMU
2006-12-25 09:51:47 +09:00
asid | = cpu_asid ( smp_processor_id ( ) , next - > mm ) ;
2006-09-27 17:25:07 +09:00
# endif
ubc_set_tracing ( asid , next - > thread . ubc_pc ) ;
2005-04-16 15:20:36 -07:00
} else {
2006-10-12 12:16:13 +09:00
# if defined(CONFIG_CPU_SH4A)
ctrl_outl ( UBC_CBR_INIT , UBC_CBR0 ) ;
ctrl_outl ( UBC_CRR_INIT , UBC_CRR0 ) ;
# else
2005-04-16 15:20:36 -07:00
ctrl_outw ( 0 , UBC_BBRA ) ;
ctrl_outw ( 0 , UBC_BBRB ) ;
2006-10-12 12:16:13 +09:00
# endif
2005-04-16 15:20:36 -07:00
}
return prev ;
}
asmlinkage int sys_fork ( unsigned long r4 , unsigned long r5 ,
unsigned long r6 , unsigned long r7 ,
2006-11-21 11:16:57 +09:00
struct pt_regs __regs )
2005-04-16 15:20:36 -07:00
{
# ifdef CONFIG_MMU
2007-05-14 17:26:34 +09:00
struct pt_regs * regs = RELOC_HIDE ( & __regs , 0 ) ;
2006-11-21 11:16:57 +09:00
return do_fork ( SIGCHLD , regs - > regs [ 15 ] , regs , 0 , NULL , NULL ) ;
2005-04-16 15:20:36 -07:00
# else
/* fork almost works, enough to trick you into looking elsewhere :-( */
return - EINVAL ;
# endif
}
asmlinkage int sys_clone ( unsigned long clone_flags , unsigned long newsp ,
unsigned long parent_tidptr ,
unsigned long child_tidptr ,
2006-11-21 11:16:57 +09:00
struct pt_regs __regs )
2005-04-16 15:20:36 -07:00
{
2006-11-21 11:16:57 +09:00
struct pt_regs * regs = RELOC_HIDE ( & __regs , 0 ) ;
2005-04-16 15:20:36 -07:00
if ( ! newsp )
2006-11-21 11:16:57 +09:00
newsp = regs - > regs [ 15 ] ;
return do_fork ( clone_flags , newsp , regs , 0 ,
2006-12-25 09:51:47 +09:00
( int __user * ) parent_tidptr ,
( int __user * ) child_tidptr ) ;
2005-04-16 15:20:36 -07:00
}
/*
* This is trivial , and on the face of it looks like it
* could equally well be done in user mode .
*
* Not so , for quite unobvious reasons - register pressure .
* In user mode vfork ( ) cannot have a stack frame , and if
* done by calling the " clone() " system call directly , you
* do not have enough call - clobbered registers to hold all
* the information you need .
*/
asmlinkage int sys_vfork ( unsigned long r4 , unsigned long r5 ,
unsigned long r6 , unsigned long r7 ,
2006-11-21 11:16:57 +09:00
struct pt_regs __regs )
2005-04-16 15:20:36 -07:00
{
2006-11-21 11:16:57 +09:00
struct pt_regs * regs = RELOC_HIDE ( & __regs , 0 ) ;
return do_fork ( CLONE_VFORK | CLONE_VM | SIGCHLD , regs - > regs [ 15 ] , regs ,
2005-04-16 15:20:36 -07:00
0 , NULL , NULL ) ;
}
/*
* sys_execve ( ) executes a new program .
*/
2007-05-14 12:52:56 +09:00
asmlinkage int sys_execve ( char __user * ufilename , char __user * __user * uargv ,
char __user * __user * uenvp , unsigned long r7 ,
2006-11-21 11:16:57 +09:00
struct pt_regs __regs )
2005-04-16 15:20:36 -07:00
{
2006-11-21 11:16:57 +09:00
struct pt_regs * regs = RELOC_HIDE ( & __regs , 0 ) ;
2005-04-16 15:20:36 -07:00
int error ;
char * filename ;
2007-05-14 12:52:56 +09:00
filename = getname ( ufilename ) ;
2005-04-16 15:20:36 -07:00
error = PTR_ERR ( filename ) ;
if ( IS_ERR ( filename ) )
goto out ;
2007-05-14 12:52:56 +09:00
error = do_execve ( filename , uargv , uenvp , regs ) ;
2005-04-16 15:20:36 -07:00
if ( error = = 0 ) {
task_lock ( current ) ;
current - > ptrace & = ~ PT_DTRACE ;
task_unlock ( current ) ;
}
putname ( filename ) ;
out :
return error ;
}
unsigned long get_wchan ( struct task_struct * p )
{
unsigned long pc ;
if ( ! p | | p = = current | | p - > state = = TASK_RUNNING )
return 0 ;
/*
* The same comment as on the Alpha applies here , too . . .
*/
pc = thread_saved_pc ( p ) ;
2007-07-26 17:46:07 +09:00
# ifdef CONFIG_FRAME_POINTER
2005-04-16 15:20:36 -07:00
if ( in_sched_functions ( pc ) ) {
2007-07-26 17:46:07 +09:00
unsigned long schedule_frame = ( unsigned long ) p - > thread . sp ;
2006-12-08 17:46:29 +09:00
return ( ( unsigned long * ) schedule_frame ) [ 21 ] ;
2005-04-16 15:20:36 -07:00
}
2007-07-26 17:46:07 +09:00
# endif
2006-12-08 17:46:29 +09:00
2005-04-16 15:20:36 -07:00
return pc ;
}
2006-11-21 11:16:57 +09:00
asmlinkage void break_point_trap ( void )
2005-04-16 15:20:36 -07:00
{
/* Clear tracing. */
2006-10-12 12:16:13 +09:00
# if defined(CONFIG_CPU_SH4A)
ctrl_outl ( UBC_CBR_INIT , UBC_CBR0 ) ;
ctrl_outl ( UBC_CRR_INIT , UBC_CRR0 ) ;
# else
2005-04-16 15:20:36 -07:00
ctrl_outw ( 0 , UBC_BBRA ) ;
ctrl_outw ( 0 , UBC_BBRB ) ;
2006-10-12 12:16:13 +09:00
# endif
2005-04-16 15:20:36 -07:00
current - > thread . ubc_pc = 0 ;
ubc_usercnt - = 1 ;
force_sig ( SIGTRAP , current ) ;
}