2005-04-17 02:20:36 +04:00
/*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file " COPYING " in the main directory of this archive
* for more details .
*
* Copyright ( C ) 1994 - 1999 , 2000 by Ralf Baechle and others .
2006-02-08 16:38:18 +03:00
* Copyright ( C ) 2005 , 2006 by Ralf Baechle ( ralf @ linux - mips . org )
2005-04-17 02:20:36 +04:00
* Copyright ( C ) 1999 , 2000 Silicon Graphics , Inc .
* Copyright ( C ) 2004 Thiemo Seufer
*/
# include <linux/errno.h>
# include <linux/module.h>
# include <linux/sched.h>
# include <linux/kernel.h>
# include <linux/mm.h>
# include <linux/stddef.h>
# include <linux/unistd.h>
# include <linux/ptrace.h>
# include <linux/slab.h>
# include <linux/mman.h>
# include <linux/personality.h>
# include <linux/sys.h>
# include <linux/user.h>
# include <linux/a.out.h>
# include <linux/init.h>
# include <linux/completion.h>
2006-02-07 19:48:03 +03:00
# include <linux/kallsyms.h>
2005-04-17 02:20:36 +04:00
2005-05-31 15:49:19 +04:00
# include <asm/abi.h>
2005-04-17 02:20:36 +04:00
# include <asm/bootinfo.h>
# include <asm/cpu.h>
2005-05-31 15:49:19 +04:00
# include <asm/dsp.h>
2005-04-17 02:20:36 +04:00
# include <asm/fpu.h>
# include <asm/pgtable.h>
# include <asm/system.h>
# include <asm/mipsregs.h>
# include <asm/processor.h>
# include <asm/uaccess.h>
# include <asm/io.h>
# include <asm/elf.h>
# include <asm/isadep.h>
# include <asm/inst.h>
2006-04-05 12:45:45 +04:00
# ifdef CONFIG_MIPS_MT_SMTC
# include <asm/mipsmtregs.h>
extern void smtc_idle_loop_hook ( void ) ;
# endif /* CONFIG_MIPS_MT_SMTC */
2005-04-17 02:20:36 +04:00
/*
* 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 )
*/
ATTRIB_NORET void cpu_idle ( void )
{
/* endless idle loop with no priority at all */
while ( 1 ) {
2006-04-05 12:45:45 +04:00
while ( ! need_resched ( ) ) {
# ifdef CONFIG_MIPS_MT_SMTC
smtc_idle_loop_hook ( ) ;
# endif /* CONFIG_MIPS_MT_SMTC */
2005-04-17 02:20:36 +04:00
if ( cpu_wait )
( * cpu_wait ) ( ) ;
2006-04-05 12:45:45 +04:00
}
2005-11-09 08:39:01 +03:00
preempt_enable_no_resched ( ) ;
2005-04-17 02:20:36 +04:00
schedule ( ) ;
2005-11-09 08:39:01 +03:00
preempt_disable ( ) ;
2005-04-17 02:20:36 +04:00
}
}
2005-05-31 15:49:19 +04:00
/*
* Native o32 and N64 ABI without DSP ASE
*/
struct mips_abi mips_abi = {
. do_signal = do_signal ,
# ifdef CONFIG_TRAD_SIGNALS
. setup_frame = setup_frame ,
# endif
. setup_rt_frame = setup_rt_frame
} ;
# ifdef CONFIG_MIPS32_O32
/*
* o32 compatibility on 64 - bit kernels , without DSP ASE
*/
struct mips_abi mips_abi_32 = {
. do_signal = do_signal32 ,
. setup_frame = setup_frame_32 ,
. setup_rt_frame = setup_rt_frame_32
} ;
# endif /* CONFIG_MIPS32_O32 */
# ifdef CONFIG_MIPS32_N32
/*
* N32 on 64 - bit kernels , without DSP ASE
*/
struct mips_abi mips_abi_n32 = {
. do_signal = do_signal ,
. setup_rt_frame = setup_rt_frame_n32
} ;
# endif /* CONFIG_MIPS32_N32 */
2005-04-17 02:20:36 +04:00
asmlinkage void ret_from_fork ( void ) ;
void start_thread ( struct pt_regs * regs , unsigned long pc , unsigned long sp )
{
unsigned long status ;
/* New thread loses kernel privileges. */
status = regs - > cp0_status & ~ ( ST0_CU0 | ST0_CU1 | KU_MASK ) ;
2005-09-04 02:56:16 +04:00
# ifdef CONFIG_64BIT
2005-04-17 02:20:36 +04:00
status & = ~ ST0_FR ;
status | = ( current - > thread . mflags & MF_32BIT_REGS ) ? 0 : ST0_FR ;
# endif
status | = KU_USER ;
regs - > cp0_status = status ;
clear_used_math ( ) ;
lose_fpu ( ) ;
2005-05-31 15:49:19 +04:00
if ( cpu_has_dsp )
__init_dsp ( ) ;
2005-04-17 02:20:36 +04:00
regs - > cp0_epc = pc ;
regs - > regs [ 29 ] = sp ;
current_thread_info ( ) - > addr_limit = USER_DS ;
}
void exit_thread ( void )
{
}
void flush_thread ( 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-01-12 12:06:08 +03:00
struct thread_info * ti = task_thread_info ( p ) ;
2005-04-17 02:20:36 +04:00
struct pt_regs * childregs ;
long childksp ;
2005-04-13 21:43:59 +04:00
p - > set_child_tid = p - > clear_child_tid = NULL ;
2005-04-17 02:20:36 +04:00
2006-01-12 12:06:08 +03:00
childksp = ( unsigned long ) task_stack_page ( p ) + THREAD_SIZE - 32 ;
2005-04-17 02:20:36 +04:00
preempt_disable ( ) ;
2005-05-31 15:49:19 +04:00
if ( is_fpu_owner ( ) )
2005-04-17 02:20:36 +04:00
save_fp ( p ) ;
2005-05-31 15:49:19 +04:00
if ( cpu_has_dsp )
save_dsp ( p ) ;
2005-04-17 02:20:36 +04:00
preempt_enable ( ) ;
/* set up new TSS. */
childregs = ( struct pt_regs * ) childksp - 1 ;
* childregs = * regs ;
childregs - > regs [ 7 ] = 0 ; /* Clear error flag */
# if defined(CONFIG_BINFMT_IRIX)
if ( current - > personality ! = PER_LINUX ) {
/* Under IRIX things are a little different. */
childregs - > regs [ 3 ] = 1 ;
regs - > regs [ 3 ] = 0 ;
}
# endif
childregs - > regs [ 2 ] = 0 ; /* Child gets zero as return value */
regs - > regs [ 2 ] = p - > pid ;
if ( childregs - > cp0_status & ST0_CU0 ) {
childregs - > regs [ 28 ] = ( unsigned long ) ti ;
childregs - > regs [ 29 ] = childksp ;
ti - > addr_limit = KERNEL_DS ;
} else {
childregs - > regs [ 29 ] = usp ;
ti - > addr_limit = USER_DS ;
}
p - > thread . reg29 = ( unsigned long ) childregs ;
p - > thread . reg31 = ( unsigned long ) ret_from_fork ;
/*
* New tasks lose permission to use the fpu . This accelerates context
* switching for most programs since they don ' t use the fpu .
*/
p - > thread . cp0_status = read_c0_status ( ) & ~ ( ST0_CU2 | ST0_CU1 ) ;
childregs - > cp0_status & = ~ ( ST0_CU2 | ST0_CU1 ) ;
clear_tsk_thread_flag ( p , TIF_USEDFPU ) ;
2006-04-05 12:45:47 +04:00
# ifdef CONFIG_MIPS_MT_FPAFF
/*
* FPU affinity support is cleaner if we track the
* user - visible CPU affinity from the very beginning .
* The generic cpus_allowed mask will already have
* been copied from the parent before copy_thread
* is invoked .
*/
p - > thread . user_cpus_allowed = p - > cpus_allowed ;
# endif /* CONFIG_MIPS_MT_FPAFF */
2005-04-13 21:43:59 +04:00
if ( clone_flags & CLONE_SETTLS )
ti - > tp_value = regs - > regs [ 7 ] ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
/* Fill in the fpu structure for a core dump.. */
int dump_fpu ( struct pt_regs * regs , elf_fpregset_t * r )
{
memcpy ( r , & current - > thread . fpu , sizeof ( current - > thread . fpu ) ) ;
return 1 ;
}
2005-12-17 01:40:47 +03:00
void elf_dump_regs ( elf_greg_t * gp , struct pt_regs * regs )
2005-04-17 02:20:36 +04:00
{
int i ;
for ( i = 0 ; i < EF_R0 ; i + + )
gp [ i ] = 0 ;
gp [ EF_R0 ] = 0 ;
for ( i = 1 ; i < = 31 ; i + + )
gp [ EF_R0 + i ] = regs - > regs [ i ] ;
gp [ EF_R26 ] = 0 ;
gp [ EF_R27 ] = 0 ;
gp [ EF_LO ] = regs - > lo ;
gp [ EF_HI ] = regs - > hi ;
gp [ EF_CP0_EPC ] = regs - > cp0_epc ;
gp [ EF_CP0_BADVADDR ] = regs - > cp0_badvaddr ;
gp [ EF_CP0_STATUS ] = regs - > cp0_status ;
gp [ EF_CP0_CAUSE ] = regs - > cp0_cause ;
# ifdef EF_UNUSED0
gp [ EF_UNUSED0 ] = 0 ;
# endif
}
2005-03-14 13:16:59 +03:00
int dump_task_regs ( struct task_struct * tsk , elf_gregset_t * regs )
{
2006-01-12 12:06:07 +03:00
elf_dump_regs ( * regs , task_pt_regs ( tsk ) ) ;
2005-03-14 13:16:59 +03:00
return 1 ;
}
2005-04-17 02:20:36 +04:00
int dump_task_fpu ( struct task_struct * t , elf_fpregset_t * fpr )
{
memcpy ( fpr , & t - > thread . fpu , sizeof ( current - > thread . fpu ) ) ;
return 1 ;
}
/*
* Create a kernel thread
*/
ATTRIB_NORET void kernel_thread_helper ( void * arg , int ( * fn ) ( void * ) )
{
do_exit ( fn ( arg ) ) ;
}
long kernel_thread ( int ( * fn ) ( void * ) , void * arg , unsigned long flags )
{
struct pt_regs regs ;
memset ( & regs , 0 , sizeof ( regs ) ) ;
regs . regs [ 4 ] = ( unsigned long ) arg ;
regs . regs [ 5 ] = ( unsigned long ) fn ;
regs . cp0_epc = ( unsigned long ) kernel_thread_helper ;
regs . cp0_status = read_c0_status ( ) ;
# if defined(CONFIG_CPU_R3000) || defined(CONFIG_CPU_TX39XX)
regs . cp0_status & = ~ ( ST0_KUP | ST0_IEC ) ;
regs . cp0_status | = ST0_IEP ;
# else
regs . cp0_status | = ST0_EXL ;
# endif
/* Ok, create the new process.. */
return do_fork ( flags | CLONE_VM | CLONE_UNTRACED , 0 , & regs , 0 , NULL , NULL ) ;
}
2005-02-21 13:55:16 +03:00
static struct mips_frame_info {
void * func ;
2006-02-07 19:48:03 +03:00
unsigned long func_size ;
int frame_size ;
2005-04-17 02:20:36 +04:00
int pc_offset ;
2006-02-07 19:48:03 +03:00
} * schedule_frame , mfinfo [ 64 ] ;
static int mfinfo_num ;
2005-02-21 13:55:16 +03:00
2006-08-03 11:29:15 +04:00
static inline int is_ra_save_ins ( union mips_instruction * ip )
{
/* sw / sd $ra, offset($sp) */
return ( ip - > i_format . opcode = = sw_op | | ip - > i_format . opcode = = sd_op ) & &
ip - > i_format . rs = = 29 & &
ip - > i_format . rt = = 31 ;
}
static inline int is_jal_jalr_jr_ins ( union mips_instruction * ip )
{
if ( ip - > j_format . opcode = = jal_op )
return 1 ;
if ( ip - > r_format . opcode ! = spec_op )
return 0 ;
return ip - > r_format . func = = jalr_op | | ip - > r_format . func = = jr_op ;
}
static inline int is_sp_move_ins ( union mips_instruction * ip )
{
/* addiu/daddiu sp,sp,-imm */
if ( ip - > i_format . rs ! = 29 | | ip - > i_format . rt ! = 29 )
return 0 ;
if ( ip - > i_format . opcode = = addiu_op | | ip - > i_format . opcode = = daddiu_op )
return 1 ;
return 0 ;
}
2006-07-29 18:27:20 +04:00
static int get_frame_info ( struct mips_frame_info * info )
2005-04-17 02:20:36 +04:00
{
2006-08-03 11:29:15 +04:00
union mips_instruction * ip = info - > func ;
int i , max_insns =
min ( 128UL , info - > func_size / sizeof ( union mips_instruction ) ) ;
2005-04-17 02:20:36 +04:00
info - > pc_offset = - 1 ;
2006-02-07 19:48:03 +03:00
info - > frame_size = 0 ;
2005-04-17 02:20:36 +04:00
2006-08-03 11:29:15 +04:00
for ( i = 0 ; i < max_insns ; i + + , ip + + ) {
if ( is_jal_jalr_jr_ins ( ip ) )
2006-02-07 19:48:03 +03:00
break ;
2006-08-03 11:29:20 +04:00
if ( ! info - > frame_size ) {
if ( is_sp_move_ins ( ip ) )
info - > frame_size = - ip - > i_format . simmediate ;
continue ;
2006-02-07 19:48:03 +03:00
}
2006-08-03 11:29:20 +04:00
if ( info - > pc_offset = = - 1 & & is_ra_save_ins ( ip ) ) {
2006-02-07 19:48:03 +03:00
info - > pc_offset =
ip - > i_format . simmediate / sizeof ( long ) ;
2006-08-03 11:29:20 +04:00
break ;
2005-04-17 02:20:36 +04:00
}
}
2006-07-29 18:27:20 +04:00
if ( info - > frame_size & & info - > pc_offset > = 0 ) /* nested */
return 0 ;
if ( info - > pc_offset < 0 ) /* leaf */
return 1 ;
/* prologue seems boggus... */
return - 1 ;
2005-04-17 02:20:36 +04:00
}
static int __init frame_info_init ( void )
{
2006-02-07 19:48:03 +03:00
int i ;
# ifdef CONFIG_KALLSYMS
char * modname ;
char namebuf [ KSYM_NAME_LEN + 1 ] ;
unsigned long start , size , ofs ;
extern char __sched_text_start [ ] , __sched_text_end [ ] ;
extern char __lock_text_start [ ] , __lock_text_end [ ] ;
start = ( unsigned long ) __sched_text_start ;
for ( i = 0 ; i < ARRAY_SIZE ( mfinfo ) ; i + + ) {
if ( start = = ( unsigned long ) schedule )
schedule_frame = & mfinfo [ i ] ;
if ( ! kallsyms_lookup ( start , & size , & ofs , & modname , namebuf ) )
break ;
mfinfo [ i ] . func = ( void * ) ( start + ofs ) ;
mfinfo [ i ] . func_size = size ;
start + = size - ofs ;
if ( start > = ( unsigned long ) __lock_text_end )
break ;
if ( start = = ( unsigned long ) __sched_text_end )
start = ( unsigned long ) __lock_text_start ;
}
# else
mfinfo [ 0 ] . func = schedule ;
schedule_frame = & mfinfo [ 0 ] ;
# endif
2006-08-03 11:29:18 +04:00
for ( i = 0 ; i < ARRAY_SIZE ( mfinfo ) & & mfinfo [ i ] . func ; i + + )
get_frame_info ( mfinfo + i ) ;
/*
* Without schedule ( ) frame info , result given by
* thread_saved_pc ( ) and get_wchan ( ) are not reliable .
*/
if ( schedule_frame - > pc_offset < 0 )
printk ( " Can't analyze schedule() prologue at %p \n " , schedule ) ;
2006-02-07 19:48:03 +03:00
mfinfo_num = i ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
arch_initcall ( frame_info_init ) ;
/*
* Return saved PC of a blocked thread .
*/
unsigned long thread_saved_pc ( struct task_struct * tsk )
{
struct thread_struct * t = & tsk - > thread ;
/* New born processes are a special case */
if ( t - > reg31 = = ( unsigned long ) ret_from_fork )
return t - > reg31 ;
2006-02-07 19:48:03 +03:00
if ( ! schedule_frame | | schedule_frame - > pc_offset < 0 )
2005-04-17 02:20:36 +04:00
return 0 ;
2006-02-07 19:48:03 +03:00
return ( ( unsigned long * ) t - > reg29 ) [ schedule_frame - > pc_offset ] ;
2005-04-17 02:20:36 +04:00
}
/* get_wchan - a maintenance nightmare^W^Wpain in the ass ... */
unsigned long get_wchan ( struct task_struct * p )
{
2005-02-21 13:55:16 +03:00
unsigned long stack_page ;
2006-02-07 19:48:03 +03:00
unsigned long pc ;
# ifdef CONFIG_KALLSYMS
unsigned long frame ;
# endif
2005-04-17 02:20:36 +04:00
if ( ! p | | p = = current | | p - > state = = TASK_RUNNING )
return 0 ;
2006-01-12 12:06:08 +03:00
stack_page = ( unsigned long ) task_stack_page ( p ) ;
2006-02-07 19:48:03 +03:00
if ( ! stack_page | | ! mfinfo_num )
2005-04-17 02:20:36 +04:00
return 0 ;
2005-02-21 13:55:16 +03:00
2005-04-17 02:20:36 +04:00
pc = thread_saved_pc ( p ) ;
2006-02-07 19:48:03 +03:00
# ifdef CONFIG_KALLSYMS
2005-04-17 02:20:36 +04:00
if ( ! in_sched_functions ( pc ) )
2005-02-21 13:55:16 +03:00
return pc ;
2005-04-17 02:20:36 +04:00
2006-02-07 19:48:03 +03:00
frame = p - > thread . reg29 + schedule_frame - > frame_size ;
2005-02-21 13:55:16 +03:00
do {
int i ;
2005-04-17 02:20:36 +04:00
2005-02-21 13:55:16 +03:00
if ( frame < stack_page | | frame > stack_page + THREAD_SIZE - 32 )
return 0 ;
2005-04-17 02:20:36 +04:00
2006-02-07 19:48:03 +03:00
for ( i = mfinfo_num - 1 ; i > = 0 ; i - - ) {
2005-02-21 13:55:16 +03:00
if ( pc > = ( unsigned long ) mfinfo [ i ] . func )
break ;
}
if ( i < 0 )
break ;
2005-04-17 02:20:36 +04:00
2006-07-29 18:27:20 +04:00
if ( mfinfo [ i ] . pc_offset < 0 )
break ;
2005-02-21 13:55:16 +03:00
pc = ( ( unsigned long * ) frame ) [ mfinfo [ i ] . pc_offset ] ;
2006-02-07 19:48:03 +03:00
if ( ! mfinfo [ i ] . frame_size )
break ;
frame + = mfinfo [ i ] . frame_size ;
2005-02-21 13:55:16 +03:00
} while ( in_sched_functions ( pc ) ) ;
2006-02-07 19:48:03 +03:00
# endif
2005-04-17 02:20:36 +04:00
return pc ;
}
2006-07-29 18:27:20 +04:00
# ifdef CONFIG_KALLSYMS
2006-08-03 11:29:21 +04:00
/* used by show_backtrace() */
unsigned long unwind_stack ( struct task_struct * task , unsigned long * sp ,
unsigned long pc , unsigned long ra )
2006-07-29 18:27:20 +04:00
{
unsigned long stack_page ;
struct mips_frame_info info ;
char * modname ;
char namebuf [ KSYM_NAME_LEN + 1 ] ;
unsigned long size , ofs ;
2006-08-03 11:29:21 +04:00
int leaf ;
2006-07-29 18:27:20 +04:00
stack_page = ( unsigned long ) task_stack_page ( task ) ;
if ( ! stack_page )
return 0 ;
if ( ! kallsyms_lookup ( pc , & size , & ofs , & modname , namebuf ) )
return 0 ;
if ( ofs = = 0 )
return 0 ;
info . func = ( void * ) ( pc - ofs ) ;
info . func_size = ofs ; /* analyze from start to ofs */
2006-08-03 11:29:21 +04:00
leaf = get_frame_info ( & info ) ;
if ( leaf < 0 )
2006-07-29 18:27:20 +04:00
return 0 ;
2006-08-03 11:29:21 +04:00
if ( * sp < stack_page | |
* sp + info . frame_size > stack_page + THREAD_SIZE - 32 )
2006-07-29 18:27:20 +04:00
return 0 ;
2006-08-03 11:29:21 +04:00
if ( leaf )
/*
* For some extreme cases , get_frame_info ( ) can
* consider wrongly a nested function as a leaf
* one . In that cases avoid to return always the
* same value .
*/
pc = pc ! = ra ? ra : 0 ;
else
pc = ( ( unsigned long * ) ( * sp ) ) [ info . pc_offset ] ;
* sp + = info . frame_size ;
return __kernel_text_address ( pc ) ? pc : 0 ;
2006-07-29 18:27:20 +04:00
}
# endif