2005-04-17 06:20:36 +08:00
|
|
|
/*
|
2006-02-01 19:06:38 +08:00
|
|
|
* arch/s390/kernel/entry64.S
|
2005-04-17 06:20:36 +08:00
|
|
|
* S390 low-level entry points.
|
|
|
|
*
|
2006-02-01 19:06:38 +08:00
|
|
|
* Copyright (C) IBM Corp. 1999,2006
|
2005-04-17 06:20:36 +08:00
|
|
|
* Author(s): Martin Schwidefsky (schwidefsky@de.ibm.com),
|
2006-09-28 22:56:37 +08:00
|
|
|
* Hartmut Penner (hp@de.ibm.com),
|
|
|
|
* Denis Joseph Barrow (djbarrow@de.ibm.com,barrow_dj@yahoo.com),
|
2005-06-26 05:55:30 +08:00
|
|
|
* Heiko Carstens <heiko.carstens@de.ibm.com>
|
2005-04-17 06:20:36 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/sys.h>
|
|
|
|
#include <linux/linkage.h>
|
2008-02-05 23:50:40 +08:00
|
|
|
#include <linux/init.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
#include <asm/cache.h>
|
|
|
|
#include <asm/lowcore.h>
|
|
|
|
#include <asm/errno.h>
|
|
|
|
#include <asm/ptrace.h>
|
|
|
|
#include <asm/thread_info.h>
|
2005-09-10 02:57:26 +08:00
|
|
|
#include <asm/asm-offsets.h>
|
2005-04-17 06:20:36 +08:00
|
|
|
#include <asm/unistd.h>
|
|
|
|
#include <asm/page.h>
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Stack layout for the system_call stack entry.
|
|
|
|
* The first few entries are identical to the user_regs_struct.
|
|
|
|
*/
|
2006-09-28 22:56:37 +08:00
|
|
|
SP_PTREGS = STACK_FRAME_OVERHEAD
|
|
|
|
SP_ARGS = STACK_FRAME_OVERHEAD + __PT_ARGS
|
|
|
|
SP_PSW = STACK_FRAME_OVERHEAD + __PT_PSW
|
|
|
|
SP_R0 = STACK_FRAME_OVERHEAD + __PT_GPRS
|
|
|
|
SP_R1 = STACK_FRAME_OVERHEAD + __PT_GPRS + 8
|
|
|
|
SP_R2 = STACK_FRAME_OVERHEAD + __PT_GPRS + 16
|
|
|
|
SP_R3 = STACK_FRAME_OVERHEAD + __PT_GPRS + 24
|
|
|
|
SP_R4 = STACK_FRAME_OVERHEAD + __PT_GPRS + 32
|
|
|
|
SP_R5 = STACK_FRAME_OVERHEAD + __PT_GPRS + 40
|
|
|
|
SP_R6 = STACK_FRAME_OVERHEAD + __PT_GPRS + 48
|
|
|
|
SP_R7 = STACK_FRAME_OVERHEAD + __PT_GPRS + 56
|
|
|
|
SP_R8 = STACK_FRAME_OVERHEAD + __PT_GPRS + 64
|
|
|
|
SP_R9 = STACK_FRAME_OVERHEAD + __PT_GPRS + 72
|
|
|
|
SP_R10 = STACK_FRAME_OVERHEAD + __PT_GPRS + 80
|
|
|
|
SP_R11 = STACK_FRAME_OVERHEAD + __PT_GPRS + 88
|
|
|
|
SP_R12 = STACK_FRAME_OVERHEAD + __PT_GPRS + 96
|
|
|
|
SP_R13 = STACK_FRAME_OVERHEAD + __PT_GPRS + 104
|
|
|
|
SP_R14 = STACK_FRAME_OVERHEAD + __PT_GPRS + 112
|
|
|
|
SP_R15 = STACK_FRAME_OVERHEAD + __PT_GPRS + 120
|
|
|
|
SP_ORIG_R2 = STACK_FRAME_OVERHEAD + __PT_ORIG_GPR2
|
|
|
|
SP_ILC = STACK_FRAME_OVERHEAD + __PT_ILC
|
2008-11-27 18:05:55 +08:00
|
|
|
SP_SVCNR = STACK_FRAME_OVERHEAD + __PT_SVCNR
|
2006-09-28 22:56:37 +08:00
|
|
|
SP_SIZE = STACK_FRAME_OVERHEAD + __PT_SIZE
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
STACK_SHIFT = PAGE_SHIFT + THREAD_ORDER
|
|
|
|
STACK_SIZE = 1 << STACK_SHIFT
|
|
|
|
|
2008-10-11 03:33:20 +08:00
|
|
|
_TIF_WORK_SVC = (_TIF_SIGPENDING | _TIF_NOTIFY_RESUME | _TIF_NEED_RESCHED | \
|
2006-02-01 19:06:38 +08:00
|
|
|
_TIF_MCCK_PENDING | _TIF_RESTART_SVC | _TIF_SINGLE_STEP )
|
2008-10-11 03:33:20 +08:00
|
|
|
_TIF_WORK_INT = (_TIF_SIGPENDING | _TIF_NOTIFY_RESUME | _TIF_NEED_RESCHED | \
|
2006-02-01 19:06:38 +08:00
|
|
|
_TIF_MCCK_PENDING)
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
#define BASED(name) name-system_call(%r13)
|
|
|
|
|
2006-07-03 15:24:46 +08:00
|
|
|
#ifdef CONFIG_TRACE_IRQFLAGS
|
|
|
|
.macro TRACE_IRQS_ON
|
2008-11-15 01:18:05 +08:00
|
|
|
basr %r2,%r0
|
|
|
|
brasl %r14,trace_hardirqs_on_caller
|
2006-07-03 15:24:46 +08:00
|
|
|
.endm
|
|
|
|
|
|
|
|
.macro TRACE_IRQS_OFF
|
2008-11-15 01:18:05 +08:00
|
|
|
basr %r2,%r0
|
|
|
|
brasl %r14,trace_hardirqs_off_caller
|
2006-07-03 15:24:46 +08:00
|
|
|
.endm
|
2007-10-12 04:11:12 +08:00
|
|
|
|
2007-11-20 18:13:32 +08:00
|
|
|
.macro TRACE_IRQS_CHECK
|
2008-11-15 01:18:05 +08:00
|
|
|
basr %r2,%r0
|
2007-11-20 18:13:32 +08:00
|
|
|
tm SP_PSW(%r15),0x03 # irqs enabled?
|
|
|
|
jz 0f
|
2008-11-15 01:18:05 +08:00
|
|
|
brasl %r14,trace_hardirqs_on_caller
|
2007-11-20 18:13:32 +08:00
|
|
|
j 1f
|
2008-11-15 01:18:05 +08:00
|
|
|
0: brasl %r14,trace_hardirqs_off_caller
|
2007-11-20 18:13:32 +08:00
|
|
|
1:
|
2007-10-12 04:11:12 +08:00
|
|
|
.endm
|
2006-07-03 15:24:46 +08:00
|
|
|
#else
|
|
|
|
#define TRACE_IRQS_ON
|
|
|
|
#define TRACE_IRQS_OFF
|
2007-11-20 18:13:32 +08:00
|
|
|
#define TRACE_IRQS_CHECK
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_LOCKDEP
|
|
|
|
.macro LOCKDEP_SYS_EXIT
|
|
|
|
tm SP_PSW+1(%r15),0x01 # returning to user ?
|
|
|
|
jz 0f
|
|
|
|
brasl %r14,lockdep_sys_exit
|
|
|
|
0:
|
|
|
|
.endm
|
|
|
|
#else
|
2007-10-12 04:11:12 +08:00
|
|
|
#define LOCKDEP_SYS_EXIT
|
2006-07-03 15:24:46 +08:00
|
|
|
#endif
|
|
|
|
|
2006-09-28 22:56:37 +08:00
|
|
|
.macro UPDATE_VTIME lc_from,lc_to,lc_sum
|
2005-04-17 06:20:36 +08:00
|
|
|
lg %r10,\lc_from
|
|
|
|
slg %r10,\lc_to
|
|
|
|
alg %r10,\lc_sum
|
|
|
|
stg %r10,\lc_sum
|
|
|
|
.endm
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Register usage in interrupt handlers:
|
|
|
|
* R9 - pointer to current task structure
|
|
|
|
* R13 - pointer to literal pool
|
|
|
|
* R14 - return register for function calls
|
|
|
|
* R15 - kernel stack pointer
|
|
|
|
*/
|
|
|
|
|
2006-09-28 22:56:37 +08:00
|
|
|
.macro SAVE_ALL_BASE savearea
|
2005-04-17 06:20:36 +08:00
|
|
|
stmg %r12,%r15,\savearea
|
|
|
|
larl %r13,system_call
|
|
|
|
.endm
|
|
|
|
|
2007-07-10 17:24:18 +08:00
|
|
|
.macro SAVE_ALL_SVC psworg,savearea
|
|
|
|
la %r12,\psworg
|
|
|
|
lg %r15,__LC_KERNEL_STACK # problem state -> load ksp
|
|
|
|
.endm
|
|
|
|
|
2006-06-29 20:58:05 +08:00
|
|
|
.macro SAVE_ALL_SYNC psworg,savearea
|
2005-04-17 06:20:36 +08:00
|
|
|
la %r12,\psworg
|
|
|
|
tm \psworg+1,0x01 # test problem state bit
|
|
|
|
jz 2f # skip stack setup save
|
|
|
|
lg %r15,__LC_KERNEL_STACK # problem state -> load ksp
|
2006-06-29 20:58:05 +08:00
|
|
|
#ifdef CONFIG_CHECK_STACK
|
|
|
|
j 3f
|
|
|
|
2: tml %r15,STACK_SIZE - CONFIG_STACK_GUARD
|
|
|
|
jz stack_overflow
|
|
|
|
3:
|
|
|
|
#endif
|
|
|
|
2:
|
|
|
|
.endm
|
|
|
|
|
|
|
|
.macro SAVE_ALL_ASYNC psworg,savearea
|
|
|
|
la %r12,\psworg
|
2005-04-17 06:20:36 +08:00
|
|
|
tm \psworg+1,0x01 # test problem state bit
|
|
|
|
jnz 1f # from user -> load kernel stack
|
|
|
|
clc \psworg+8(8),BASED(.Lcritical_end)
|
|
|
|
jhe 0f
|
|
|
|
clc \psworg+8(8),BASED(.Lcritical_start)
|
|
|
|
jl 0f
|
|
|
|
brasl %r14,cleanup_critical
|
2005-09-17 10:27:42 +08:00
|
|
|
tm 1(%r12),0x01 # retest problem state after cleanup
|
2005-04-17 06:20:36 +08:00
|
|
|
jnz 1f
|
|
|
|
0: lg %r14,__LC_ASYNC_STACK # are we already on the async. stack ?
|
|
|
|
slgr %r14,%r15
|
|
|
|
srag %r14,%r14,STACK_SHIFT
|
|
|
|
jz 2f
|
|
|
|
1: lg %r15,__LC_ASYNC_STACK # load async stack
|
|
|
|
#ifdef CONFIG_CHECK_STACK
|
|
|
|
j 3f
|
|
|
|
2: tml %r15,STACK_SIZE - CONFIG_STACK_GUARD
|
|
|
|
jz stack_overflow
|
|
|
|
3:
|
|
|
|
#endif
|
2005-06-26 05:55:30 +08:00
|
|
|
2:
|
|
|
|
.endm
|
|
|
|
|
|
|
|
.macro CREATE_STACK_FRAME psworg,savearea
|
2006-09-28 22:56:37 +08:00
|
|
|
aghi %r15,-SP_SIZE # make room for registers & psw
|
|
|
|
mvc SP_PSW(16,%r15),0(%r12) # move user PSW to stack
|
2005-04-17 06:20:36 +08:00
|
|
|
stg %r2,SP_ORIG_R2(%r15) # store original content of gpr 2
|
2008-11-27 18:05:55 +08:00
|
|
|
icm %r12,3,__LC_SVC_ILC
|
2005-04-17 06:20:36 +08:00
|
|
|
stmg %r0,%r11,SP_R0(%r15) # store gprs %r0-%r11 to kernel stack
|
2008-11-27 18:05:55 +08:00
|
|
|
st %r12,SP_SVCNR(%r15)
|
2005-04-17 06:20:36 +08:00
|
|
|
mvc SP_R12(32,%r15),\savearea # move %r12-%r15 to stack
|
|
|
|
la %r12,0
|
|
|
|
stg %r12,__SF_BACKCHAIN(%r15)
|
2006-09-28 22:56:37 +08:00
|
|
|
.endm
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2005-09-04 06:57:56 +08:00
|
|
|
.macro RESTORE_ALL psworg,sync
|
|
|
|
mvc \psworg(16),SP_PSW(%r15) # move user PSW to lowcore
|
2005-04-17 06:20:36 +08:00
|
|
|
.if !\sync
|
2005-09-04 06:57:56 +08:00
|
|
|
ni \psworg+1,0xfd # clear wait state bit
|
2005-04-17 06:20:36 +08:00
|
|
|
.endif
|
|
|
|
lmg %r0,%r15,SP_R0(%r15) # load gprs 0-15 of user
|
2008-12-25 20:39:25 +08:00
|
|
|
stpt __LC_EXIT_TIMER
|
2005-09-04 06:57:56 +08:00
|
|
|
lpswe \psworg # back to caller
|
2005-04-17 06:20:36 +08:00
|
|
|
.endm
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Scheduler resume function, called by switch_to
|
|
|
|
* gpr2 = (task_struct *) prev
|
|
|
|
* gpr3 = (task_struct *) next
|
|
|
|
* Returns:
|
|
|
|
* gpr2 = prev
|
|
|
|
*/
|
2006-09-28 22:56:37 +08:00
|
|
|
.globl __switch_to
|
2005-04-17 06:20:36 +08:00
|
|
|
__switch_to:
|
|
|
|
tm __THREAD_per+4(%r3),0xe8 # is the new process using per ?
|
|
|
|
jz __switch_to_noper # if not we're fine
|
2006-09-28 22:56:37 +08:00
|
|
|
stctg %c9,%c11,__SF_EMPTY(%r15)# We are using per stuff
|
|
|
|
clc __THREAD_per(24,%r3),__SF_EMPTY(%r15)
|
|
|
|
je __switch_to_noper # we got away without bashing TLB's
|
|
|
|
lctlg %c9,%c11,__THREAD_per(%r3) # Nope we didn't
|
2005-04-17 06:20:36 +08:00
|
|
|
__switch_to_noper:
|
2006-09-28 22:56:37 +08:00
|
|
|
lg %r4,__THREAD_info(%r2) # get thread_info of prev
|
2005-06-26 05:55:30 +08:00
|
|
|
tm __TI_flags+7(%r4),_TIF_MCCK_PENDING # machine check pending?
|
|
|
|
jz __switch_to_no_mcck
|
|
|
|
ni __TI_flags+7(%r4),255-_TIF_MCCK_PENDING # clear flag in prev
|
|
|
|
lg %r4,__THREAD_info(%r3) # get thread_info of next
|
|
|
|
oi __TI_flags+7(%r4),_TIF_MCCK_PENDING # set it in next
|
|
|
|
__switch_to_no_mcck:
|
2006-09-28 22:56:37 +08:00
|
|
|
stmg %r6,%r15,__SF_GPRS(%r15)# store __switch_to registers of prev task
|
2005-04-17 06:20:36 +08:00
|
|
|
stg %r15,__THREAD_ksp(%r2) # store kernel stack to prev->tss.ksp
|
|
|
|
lg %r15,__THREAD_ksp(%r3) # load kernel stack from next->tss.ksp
|
2006-09-28 22:56:37 +08:00
|
|
|
lmg %r6,%r15,__SF_GPRS(%r15)# load __switch_to registers of next task
|
2005-04-17 06:20:36 +08:00
|
|
|
stg %r3,__LC_CURRENT # __LC_CURRENT = current task struct
|
|
|
|
lctl %c4,%c4,__TASK_pid(%r3) # load pid to control reg. 4
|
2006-09-28 22:56:37 +08:00
|
|
|
lg %r3,__THREAD_info(%r3) # load thread_info from task struct
|
2005-04-17 06:20:36 +08:00
|
|
|
stg %r3,__LC_THREAD_INFO
|
|
|
|
aghi %r3,STACK_SIZE
|
|
|
|
stg %r3,__LC_KERNEL_STACK # __LC_KERNEL_STACK = new kernel stack
|
|
|
|
br %r14
|
|
|
|
|
|
|
|
__critical_start:
|
|
|
|
/*
|
|
|
|
* SVC interrupt handler routine. System calls are synchronous events and
|
|
|
|
* are executed with interrupts enabled.
|
|
|
|
*/
|
|
|
|
|
2006-09-28 22:56:37 +08:00
|
|
|
.globl system_call
|
2005-04-17 06:20:36 +08:00
|
|
|
system_call:
|
2008-12-25 20:39:25 +08:00
|
|
|
stpt __LC_SYNC_ENTER_TIMER
|
2005-04-17 06:20:36 +08:00
|
|
|
sysc_saveall:
|
|
|
|
SAVE_ALL_BASE __LC_SAVE_AREA
|
2007-07-10 17:24:18 +08:00
|
|
|
SAVE_ALL_SVC __LC_SVC_OLD_PSW,__LC_SAVE_AREA
|
2006-09-28 22:56:37 +08:00
|
|
|
CREATE_STACK_FRAME __LC_SVC_OLD_PSW,__LC_SAVE_AREA
|
|
|
|
llgh %r7,__LC_SVC_INT_CODE # get svc number from lowcore
|
2005-04-17 06:20:36 +08:00
|
|
|
sysc_vtime:
|
|
|
|
UPDATE_VTIME __LC_EXIT_TIMER,__LC_SYNC_ENTER_TIMER,__LC_USER_TIMER
|
|
|
|
sysc_stime:
|
|
|
|
UPDATE_VTIME __LC_LAST_UPDATE_TIMER,__LC_EXIT_TIMER,__LC_SYSTEM_TIMER
|
|
|
|
sysc_update:
|
|
|
|
mvc __LC_LAST_UPDATE_TIMER(8),__LC_SYNC_ENTER_TIMER
|
|
|
|
sysc_do_svc:
|
|
|
|
lg %r9,__LC_THREAD_INFO # load pointer to thread_info struct
|
2008-11-27 18:05:55 +08:00
|
|
|
ltgr %r7,%r7 # test for svc 0
|
2005-04-17 06:20:36 +08:00
|
|
|
jnz sysc_nr_ok
|
|
|
|
# svc 0: system call number in %r1
|
|
|
|
cl %r1,BASED(.Lnr_syscalls)
|
|
|
|
jnl sysc_nr_ok
|
2006-09-28 22:56:37 +08:00
|
|
|
lgfr %r7,%r1 # clear high word in r1
|
2005-04-17 06:20:36 +08:00
|
|
|
sysc_nr_ok:
|
|
|
|
mvc SP_ARGS(8,%r15),SP_R7(%r15)
|
|
|
|
sysc_do_restart:
|
2008-11-27 18:05:55 +08:00
|
|
|
sth %r7,SP_SVCNR(%r15)
|
|
|
|
sllg %r7,%r7,2 # svc number * 4
|
2006-09-28 22:56:37 +08:00
|
|
|
larl %r10,sys_call_table
|
2006-01-06 16:19:28 +08:00
|
|
|
#ifdef CONFIG_COMPAT
|
2005-09-04 06:58:04 +08:00
|
|
|
tm __TI_flags+5(%r9),(_TIF_31BIT>>16) # running in 31 bit mode ?
|
|
|
|
jno sysc_noemu
|
2006-09-28 22:56:37 +08:00
|
|
|
larl %r10,sys_call_table_emu # use 31 bit emulation system calls
|
2005-04-17 06:20:36 +08:00
|
|
|
sysc_noemu:
|
|
|
|
#endif
|
|
|
|
tm __TI_flags+7(%r9),(_TIF_SYSCALL_TRACE|_TIF_SYSCALL_AUDIT)
|
2006-09-28 22:56:37 +08:00
|
|
|
lgf %r8,0(%r7,%r10) # load address of system call routine
|
|
|
|
jnz sysc_tracesys
|
|
|
|
basr %r14,%r8 # call sys_xxxx
|
|
|
|
stg %r2,SP_R2(%r15) # store return value (change R2 on stack)
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
sysc_return:
|
|
|
|
tm __TI_flags+7(%r9),_TIF_WORK_SVC
|
2006-09-28 22:56:37 +08:00
|
|
|
jnz sysc_work # there is work to do (signals etc.)
|
2007-11-20 18:13:32 +08:00
|
|
|
sysc_restore:
|
|
|
|
#ifdef CONFIG_TRACE_IRQFLAGS
|
|
|
|
larl %r1,sysc_restore_trace_psw
|
|
|
|
lpswe 0(%r1)
|
|
|
|
sysc_restore_trace:
|
|
|
|
TRACE_IRQS_CHECK
|
2007-10-12 04:11:12 +08:00
|
|
|
LOCKDEP_SYS_EXIT
|
2007-11-20 18:13:32 +08:00
|
|
|
#endif
|
2005-04-17 06:20:36 +08:00
|
|
|
sysc_leave:
|
2006-09-28 22:56:37 +08:00
|
|
|
RESTORE_ALL __LC_RETURN_PSW,1
|
2007-11-20 18:13:32 +08:00
|
|
|
sysc_done:
|
|
|
|
|
|
|
|
#ifdef CONFIG_TRACE_IRQFLAGS
|
|
|
|
.align 8
|
|
|
|
.globl sysc_restore_trace_psw
|
|
|
|
sysc_restore_trace_psw:
|
|
|
|
.quad 0, sysc_restore_trace
|
|
|
|
#endif
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
#
|
|
|
|
# recheck if there is more work to do
|
|
|
|
#
|
|
|
|
sysc_work_loop:
|
|
|
|
tm __TI_flags+7(%r9),_TIF_WORK_SVC
|
2007-11-20 18:13:32 +08:00
|
|
|
jz sysc_restore # there is no work to do
|
2005-04-17 06:20:36 +08:00
|
|
|
#
|
|
|
|
# One of the work bits is on. Find out which one.
|
|
|
|
#
|
|
|
|
sysc_work:
|
2008-05-07 15:22:52 +08:00
|
|
|
tm SP_PSW+1(%r15),0x01 # returning to user ?
|
|
|
|
jno sysc_restore
|
2005-06-26 05:55:30 +08:00
|
|
|
tm __TI_flags+7(%r9),_TIF_MCCK_PENDING
|
|
|
|
jo sysc_mcck_pending
|
2005-04-17 06:20:36 +08:00
|
|
|
tm __TI_flags+7(%r9),_TIF_NEED_RESCHED
|
|
|
|
jo sysc_reschedule
|
2008-04-30 15:53:08 +08:00
|
|
|
tm __TI_flags+7(%r9),_TIF_SIGPENDING
|
2006-02-01 19:06:38 +08:00
|
|
|
jnz sysc_sigpending
|
2008-10-11 03:33:20 +08:00
|
|
|
tm __TI_flags+7(%r9),_TIF_NOTIFY_RESUME
|
|
|
|
jnz sysc_notify_resume
|
2005-04-17 06:20:36 +08:00
|
|
|
tm __TI_flags+7(%r9),_TIF_RESTART_SVC
|
|
|
|
jo sysc_restart
|
|
|
|
tm __TI_flags+7(%r9),_TIF_SINGLE_STEP
|
|
|
|
jo sysc_singlestep
|
2007-11-20 18:13:32 +08:00
|
|
|
j sysc_restore
|
|
|
|
sysc_work_done:
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
#
|
|
|
|
# _TIF_NEED_RESCHED is set, call schedule
|
2006-09-28 22:56:37 +08:00
|
|
|
#
|
|
|
|
sysc_reschedule:
|
|
|
|
larl %r14,sysc_work_loop
|
|
|
|
jg schedule # return point is sysc_return
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2005-06-26 05:55:30 +08:00
|
|
|
#
|
|
|
|
# _TIF_MCCK_PENDING is set, call handler
|
|
|
|
#
|
|
|
|
sysc_mcck_pending:
|
|
|
|
larl %r14,sysc_work_loop
|
2006-09-28 22:56:37 +08:00
|
|
|
jg s390_handle_mcck # TIF bit will be cleared by handler
|
2005-06-26 05:55:30 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
#
|
2008-04-30 15:53:08 +08:00
|
|
|
# _TIF_SIGPENDING is set, call do_signal
|
2005-04-17 06:20:36 +08:00
|
|
|
#
|
2006-09-28 22:56:37 +08:00
|
|
|
sysc_sigpending:
|
2005-04-17 06:20:36 +08:00
|
|
|
ni __TI_flags+7(%r9),255-_TIF_SINGLE_STEP # clear TIF_SINGLE_STEP
|
2006-09-28 22:56:37 +08:00
|
|
|
la %r2,SP_PTREGS(%r15) # load pt_regs
|
|
|
|
brasl %r14,do_signal # call do_signal
|
2005-04-17 06:20:36 +08:00
|
|
|
tm __TI_flags+7(%r9),_TIF_RESTART_SVC
|
|
|
|
jo sysc_restart
|
|
|
|
tm __TI_flags+7(%r9),_TIF_SINGLE_STEP
|
|
|
|
jo sysc_singlestep
|
2005-11-07 16:59:02 +08:00
|
|
|
j sysc_work_loop
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2008-10-11 03:33:20 +08:00
|
|
|
#
|
|
|
|
# _TIF_NOTIFY_RESUME is set, call do_notify_resume
|
|
|
|
#
|
|
|
|
sysc_notify_resume:
|
|
|
|
la %r2,SP_PTREGS(%r15) # load pt_regs
|
|
|
|
larl %r14,sysc_work_loop
|
|
|
|
jg do_notify_resume # call do_notify_resume
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
#
|
|
|
|
# _TIF_RESTART_SVC is set, set up registers and restart svc
|
|
|
|
#
|
|
|
|
sysc_restart:
|
|
|
|
ni __TI_flags+7(%r9),255-_TIF_RESTART_SVC # clear TIF_RESTART_SVC
|
2006-09-28 22:56:37 +08:00
|
|
|
lg %r7,SP_R2(%r15) # load new svc number
|
2005-04-17 06:20:36 +08:00
|
|
|
mvc SP_R2(8,%r15),SP_ORIG_R2(%r15) # restore first argument
|
2006-09-28 22:56:37 +08:00
|
|
|
lmg %r2,%r6,SP_R2(%r15) # load svc arguments
|
|
|
|
j sysc_do_restart # restart svc
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
#
|
|
|
|
# _TIF_SINGLE_STEP is set, call do_single_step
|
|
|
|
#
|
|
|
|
sysc_singlestep:
|
2008-11-27 18:05:55 +08:00
|
|
|
ni __TI_flags+7(%r9),255-_TIF_SINGLE_STEP # clear TIF_SINGLE_STEP
|
|
|
|
xc SP_SVCNR(2,%r15),SP_SVCNR(%r15) # clear svc number
|
2005-04-17 06:20:36 +08:00
|
|
|
la %r2,SP_PTREGS(%r15) # address of register-save area
|
|
|
|
larl %r14,sysc_return # load adr. of system return
|
|
|
|
jg do_single_step # branch to do_sigtrap
|
|
|
|
|
|
|
|
#
|
2008-10-11 03:33:20 +08:00
|
|
|
# call tracehook_report_syscall_entry/tracehook_report_syscall_exit before
|
|
|
|
# and after the system call
|
2005-04-17 06:20:36 +08:00
|
|
|
#
|
|
|
|
sysc_tracesys:
|
2006-09-28 22:56:37 +08:00
|
|
|
la %r2,SP_PTREGS(%r15) # load pt_regs
|
2005-04-17 06:20:36 +08:00
|
|
|
la %r3,0
|
|
|
|
srl %r7,2
|
2006-09-28 22:56:37 +08:00
|
|
|
stg %r7,SP_R2(%r15)
|
2008-10-11 03:33:20 +08:00
|
|
|
brasl %r14,do_syscall_trace_enter
|
2005-04-17 06:20:36 +08:00
|
|
|
lghi %r0,NR_syscalls
|
2008-10-11 03:33:20 +08:00
|
|
|
clgr %r0,%r2
|
2005-04-17 06:20:36 +08:00
|
|
|
jnh sysc_tracenogo
|
2008-11-27 18:05:55 +08:00
|
|
|
sllg %r7,%r2,2 # svc number *4
|
2005-04-17 06:20:36 +08:00
|
|
|
lgf %r8,0(%r7,%r10)
|
|
|
|
sysc_tracego:
|
2006-09-28 22:56:37 +08:00
|
|
|
lmg %r3,%r6,SP_R3(%r15)
|
|
|
|
lg %r2,SP_ORIG_R2(%r15)
|
|
|
|
basr %r14,%r8 # call sys_xxx
|
|
|
|
stg %r2,SP_R2(%r15) # store return value
|
2005-04-17 06:20:36 +08:00
|
|
|
sysc_tracenogo:
|
|
|
|
tm __TI_flags+7(%r9),(_TIF_SYSCALL_TRACE|_TIF_SYSCALL_AUDIT)
|
2006-09-28 22:56:37 +08:00
|
|
|
jz sysc_return
|
|
|
|
la %r2,SP_PTREGS(%r15) # load pt_regs
|
|
|
|
larl %r14,sysc_return # return point is sysc_return
|
2008-10-11 03:33:20 +08:00
|
|
|
jg do_syscall_trace_exit
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
#
|
|
|
|
# a new process exits the kernel with ret_from_fork
|
|
|
|
#
|
2006-09-28 22:56:37 +08:00
|
|
|
.globl ret_from_fork
|
2005-04-17 06:20:36 +08:00
|
|
|
ret_from_fork:
|
|
|
|
lg %r13,__LC_SVC_NEW_PSW+8
|
|
|
|
lg %r9,__LC_THREAD_INFO # load pointer to thread_info struct
|
|
|
|
tm SP_PSW+1(%r15),0x01 # forking a kernel thread ?
|
|
|
|
jo 0f
|
|
|
|
stg %r15,SP_R15(%r15) # store stack pointer for new kthread
|
2006-09-28 22:56:37 +08:00
|
|
|
0: brasl %r14,schedule_tail
|
2006-07-03 15:24:46 +08:00
|
|
|
TRACE_IRQS_ON
|
2006-09-28 22:56:37 +08:00
|
|
|
stosm 24(%r15),0x03 # reenable interrupts
|
[S390] audit: get s390 ret_from_fork in sync with other architectures
On s390 we have ret_from_fork jump not to the "do all work we
normally do on return from syscall" as on x86, ppc, etc., but to the
"do all such work except audit". Historical reasons - the codepath
triggered when we have AUDIT process flag set is separated from the
normall one and they converge at sysc_return, which is the common
part of post-syscall work. And does not include calling audit_syscall_exit() -
that's done in the end of sysc_tracesys path, just before that path jumps
to sysc_return.
IOW, the child returning from fork()/clone()/vfork() doesn't
call audit_syscall_exit() at all, so no matter what we do with its
audit context, we are not going to see the audit entry.
The fix is simple: have ret_from_fork go to the point just past
the call of sys_.... in the 'we have AUDIT flag set' path. There we
have (64bit variant; for 31bit the situation is the same):
sysc_tracenogo:
tm __TI_flags+7(%r9),(_TIF_SYSCALL_TRACE|_TIF_SYSCALL_AUDIT)
jz sysc_return
la %r2,SP_PTREGS(%r15) # load pt_regs
larl %r14,sysc_return # return point is sysc_return
jg do_syscall_trace_exit
which is precisely what we need - check the flag, bugger off to sysc_return
if not set, otherwise call do_syscall_trace_exit() and bugger off to
sysc_return. r9 has just been properly set by ret_from_fork itself,
so we are fine.
Tested on s390x, seems to work fine. WARNING: it's been about
16 years since my last contact with 3X0 assembler[1], so additional
review would be very welcome. I don't think I've managed to screw it
up, but...
[1] that *was* in another country and besides, the box is dead...
Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
Signed-off-by: Heiko Carstens <heiko.carstens@de.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
2008-12-25 20:37:58 +08:00
|
|
|
j sysc_tracenogo
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
#
|
2007-04-27 22:01:40 +08:00
|
|
|
# kernel_execve function needs to deal with pt_regs that is not
|
|
|
|
# at the usual place
|
2005-04-17 06:20:36 +08:00
|
|
|
#
|
2007-04-27 22:01:40 +08:00
|
|
|
.globl kernel_execve
|
|
|
|
kernel_execve:
|
|
|
|
stmg %r12,%r15,96(%r15)
|
|
|
|
lgr %r14,%r15
|
|
|
|
aghi %r15,-SP_SIZE
|
|
|
|
stg %r14,__SF_BACKCHAIN(%r15)
|
|
|
|
la %r12,SP_PTREGS(%r15)
|
|
|
|
xc 0(__PT_SIZE,%r12),0(%r12)
|
|
|
|
lgr %r5,%r12
|
|
|
|
brasl %r14,do_execve
|
|
|
|
ltgfr %r2,%r2
|
|
|
|
je 0f
|
|
|
|
aghi %r15,SP_SIZE
|
|
|
|
lmg %r12,%r15,96(%r15)
|
|
|
|
br %r14
|
|
|
|
# execve succeeded.
|
|
|
|
0: stnsm __SF_EMPTY(%r15),0xfc # disable interrupts
|
|
|
|
lg %r15,__LC_KERNEL_STACK # load ksp
|
|
|
|
aghi %r15,-SP_SIZE # make room for registers & psw
|
|
|
|
lg %r13,__LC_SVC_NEW_PSW+8
|
|
|
|
lg %r9,__LC_THREAD_INFO
|
|
|
|
mvc SP_PTREGS(__PT_SIZE,%r15),0(%r12) # copy pt_regs
|
|
|
|
xc __SF_BACKCHAIN(8,%r15),__SF_BACKCHAIN(%r15)
|
|
|
|
stosm __SF_EMPTY(%r15),0x03 # reenable interrupts
|
|
|
|
brasl %r14,execve_tail
|
|
|
|
j sysc_return
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Program check handler routine
|
|
|
|
*/
|
|
|
|
|
2006-09-28 22:56:37 +08:00
|
|
|
.globl pgm_check_handler
|
2005-04-17 06:20:36 +08:00
|
|
|
pgm_check_handler:
|
|
|
|
/*
|
|
|
|
* First we need to check for a special case:
|
|
|
|
* Single stepping an instruction that disables the PER event mask will
|
|
|
|
* cause a PER event AFTER the mask has been set. Example: SVC or LPSW.
|
|
|
|
* For a single stepped SVC the program check handler gets control after
|
|
|
|
* the SVC new PSW has been loaded. But we want to execute the SVC first and
|
|
|
|
* then handle the PER event. Therefore we update the SVC old PSW to point
|
|
|
|
* to the pgm_check_handler and branch to the SVC handler after we checked
|
|
|
|
* if we have to load the kernel stack register.
|
|
|
|
* For every other possible cause for PER event without the PER mask set
|
|
|
|
* we just ignore the PER event (FIXME: is there anything we have to do
|
|
|
|
* for LPSW?).
|
|
|
|
*/
|
2008-12-25 20:39:25 +08:00
|
|
|
stpt __LC_SYNC_ENTER_TIMER
|
2005-04-17 06:20:36 +08:00
|
|
|
SAVE_ALL_BASE __LC_SAVE_AREA
|
2006-09-28 22:56:37 +08:00
|
|
|
tm __LC_PGM_INT_CODE+1,0x80 # check whether we got a per exception
|
|
|
|
jnz pgm_per # got per exception -> special case
|
2006-06-29 20:58:05 +08:00
|
|
|
SAVE_ALL_SYNC __LC_PGM_OLD_PSW,__LC_SAVE_AREA
|
2005-06-26 05:55:30 +08:00
|
|
|
CREATE_STACK_FRAME __LC_PGM_OLD_PSW,__LC_SAVE_AREA
|
2005-04-17 06:20:36 +08:00
|
|
|
tm SP_PSW+1(%r15),0x01 # interrupting from user ?
|
|
|
|
jz pgm_no_vtime
|
|
|
|
UPDATE_VTIME __LC_EXIT_TIMER,__LC_SYNC_ENTER_TIMER,__LC_USER_TIMER
|
|
|
|
UPDATE_VTIME __LC_LAST_UPDATE_TIMER,__LC_EXIT_TIMER,__LC_SYSTEM_TIMER
|
|
|
|
mvc __LC_LAST_UPDATE_TIMER(8),__LC_SYNC_ENTER_TIMER
|
|
|
|
pgm_no_vtime:
|
|
|
|
lg %r9,__LC_THREAD_INFO # load pointer to thread_info struct
|
[S390] kernel: show last breaking-event-address on oops
Newer s390 models have a breaking-event-address-recording register.
Each time an instruction causes a break in the sequential instruction
execution, the address is saved in that hardware register. On a program
interrupt the address is copied to the lowcore address 272-279, which
makes it software accessible.
This patch changes the program check handler and the stack overflow
checker to copy the value into the pt_regs argument.
The oops output is enhanced to show the last known breaking address.
It might give additional information if the stack trace is corrupted.
The feature is only available on 64 bit.
The new oops output looks like:
[---------snip----------]
Modules linked in: vmcp sunrpc qeth_l2 dm_mod qeth ccwgroup
CPU: 2 Not tainted 2.6.24zlive-host #8
Process modprobe (pid: 4788, task: 00000000bf3d8718, ksp: 00000000b2b0b8e0)
Krnl PSW : 0704200180000000 000003e000020028 (vmcp_init+0x28/0xe4 [vmcp])
R:0 T:1 IO:1 EX:1 Key:0 M:1 W:0 P:0 AS:0 CC:2 PM:0 EA:3
Krnl GPRS: 0000000004000002 000003e000020000 0000000000000000 0000000000000001
000000000015734c ffffffffffffffff 000003e0000b3b00 0000000000000000
000003e00007ca30 00000000b5bb5d40 00000000b5bb5800 000003e0000b3b00
000003e0000a2000 00000000003ecf50 00000000b2b0bd50 00000000b2b0bcb0
Krnl Code: 000003e000020018: c0c000040ff4 larl %r12,3e0000a2000
000003e00002001e: e3e0f0000024 stg %r14,0(%r15)
000003e000020024: a7f40001 brc 15,3e000020026
>000003e000020028: e310c0100004 lg %r1,16(%r12)
000003e00002002e: c020000413dc larl %r2,3e0000a27e6
000003e000020034: c0a00004aee6 larl %r10,3e0000b5e00
000003e00002003a: a7490001 lghi %r4,1
000003e00002003e: a75900f0 lghi %r5,240
Call Trace:
([<000000000014b300>] blocking_notifier_call_chain+0x2c/0x40)
[<000000000015735c>] sys_init_module+0x19d8/0x1b08
[<0000000000110afc>] sysc_noemu+0x10/0x16
[<000002000011cda2>] 0x2000011cda2
Last Breaking-Event-Address:
[<000003e000020024>] vmcp_init+0x24/0xe4 [vmcp]
[---------snip----------]
Signed-off-by: Christian Borntraeger <borntraeger@de.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
Signed-off-by: Heiko Carstens <heiko.carstens@de.ibm.com>
2008-04-17 13:46:30 +08:00
|
|
|
mvc SP_ARGS(8,%r15),__LC_LAST_BREAK
|
2007-11-20 18:13:32 +08:00
|
|
|
TRACE_IRQS_OFF
|
2006-09-28 22:56:37 +08:00
|
|
|
lgf %r3,__LC_PGM_ILC # load program interruption code
|
2005-04-17 06:20:36 +08:00
|
|
|
lghi %r8,0x7f
|
|
|
|
ngr %r8,%r3
|
|
|
|
pgm_do_call:
|
2006-09-28 22:56:37 +08:00
|
|
|
sll %r8,3
|
|
|
|
larl %r1,pgm_check_table
|
|
|
|
lg %r1,0(%r8,%r1) # load address of handler routine
|
|
|
|
la %r2,SP_PTREGS(%r15) # address of register-save area
|
2005-04-17 06:20:36 +08:00
|
|
|
larl %r14,sysc_return
|
2006-09-28 22:56:37 +08:00
|
|
|
br %r1 # branch to interrupt-handler
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
#
|
|
|
|
# handle per exception
|
|
|
|
#
|
|
|
|
pgm_per:
|
2006-09-28 22:56:37 +08:00
|
|
|
tm __LC_PGM_OLD_PSW,0x40 # test if per event recording is on
|
|
|
|
jnz pgm_per_std # ok, normal per event from user space
|
2005-04-17 06:20:36 +08:00
|
|
|
# ok its one of the special cases, now we need to find out which one
|
2006-09-28 22:56:37 +08:00
|
|
|
clc __LC_PGM_OLD_PSW(16),__LC_SVC_NEW_PSW
|
|
|
|
je pgm_svcper
|
2005-04-17 06:20:36 +08:00
|
|
|
# no interesting special case, ignore PER event
|
|
|
|
lmg %r12,%r15,__LC_SAVE_AREA
|
2006-09-28 22:56:37 +08:00
|
|
|
lpswe __LC_PGM_OLD_PSW
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
#
|
|
|
|
# Normal per exception
|
|
|
|
#
|
|
|
|
pgm_per_std:
|
2006-06-29 20:58:05 +08:00
|
|
|
SAVE_ALL_SYNC __LC_PGM_OLD_PSW,__LC_SAVE_AREA
|
2005-06-26 05:55:30 +08:00
|
|
|
CREATE_STACK_FRAME __LC_PGM_OLD_PSW,__LC_SAVE_AREA
|
2005-04-17 06:20:36 +08:00
|
|
|
tm SP_PSW+1(%r15),0x01 # interrupting from user ?
|
|
|
|
jz pgm_no_vtime2
|
|
|
|
UPDATE_VTIME __LC_EXIT_TIMER,__LC_SYNC_ENTER_TIMER,__LC_USER_TIMER
|
|
|
|
UPDATE_VTIME __LC_LAST_UPDATE_TIMER,__LC_EXIT_TIMER,__LC_SYSTEM_TIMER
|
|
|
|
mvc __LC_LAST_UPDATE_TIMER(8),__LC_SYNC_ENTER_TIMER
|
|
|
|
pgm_no_vtime2:
|
|
|
|
lg %r9,__LC_THREAD_INFO # load pointer to thread_info struct
|
2007-11-20 18:13:32 +08:00
|
|
|
TRACE_IRQS_OFF
|
2005-04-17 06:20:36 +08:00
|
|
|
lg %r1,__TI_task(%r9)
|
2006-09-20 21:58:39 +08:00
|
|
|
tm SP_PSW+1(%r15),0x01 # kernel per event ?
|
|
|
|
jz kernel_per
|
2005-04-17 06:20:36 +08:00
|
|
|
mvc __THREAD_per+__PER_atmid(2,%r1),__LC_PER_ATMID
|
|
|
|
mvc __THREAD_per+__PER_address(8,%r1),__LC_PER_ADDRESS
|
|
|
|
mvc __THREAD_per+__PER_access_id(1,%r1),__LC_PER_ACCESS_ID
|
|
|
|
oi __TI_flags+7(%r9),_TIF_SINGLE_STEP # set TIF_SINGLE_STEP
|
2006-09-28 22:56:37 +08:00
|
|
|
lgf %r3,__LC_PGM_ILC # load program interruption code
|
2005-04-17 06:20:36 +08:00
|
|
|
lghi %r8,0x7f
|
2006-09-28 22:56:37 +08:00
|
|
|
ngr %r8,%r3 # clear per-event-bit and ilc
|
2005-04-17 06:20:36 +08:00
|
|
|
je sysc_return
|
|
|
|
j pgm_do_call
|
|
|
|
|
|
|
|
#
|
|
|
|
# it was a single stepped SVC that is causing all the trouble
|
|
|
|
#
|
|
|
|
pgm_svcper:
|
2006-06-29 20:58:05 +08:00
|
|
|
SAVE_ALL_SYNC __LC_SVC_OLD_PSW,__LC_SAVE_AREA
|
2005-06-26 05:55:30 +08:00
|
|
|
CREATE_STACK_FRAME __LC_SVC_OLD_PSW,__LC_SAVE_AREA
|
2005-04-17 06:20:36 +08:00
|
|
|
UPDATE_VTIME __LC_EXIT_TIMER,__LC_SYNC_ENTER_TIMER,__LC_USER_TIMER
|
|
|
|
UPDATE_VTIME __LC_LAST_UPDATE_TIMER,__LC_EXIT_TIMER,__LC_SYSTEM_TIMER
|
|
|
|
mvc __LC_LAST_UPDATE_TIMER(8),__LC_SYNC_ENTER_TIMER
|
2006-09-28 22:56:37 +08:00
|
|
|
llgh %r7,__LC_SVC_INT_CODE # get svc number from lowcore
|
2005-04-17 06:20:36 +08:00
|
|
|
lg %r9,__LC_THREAD_INFO # load pointer to thread_info struct
|
|
|
|
lg %r1,__TI_task(%r9)
|
|
|
|
mvc __THREAD_per+__PER_atmid(2,%r1),__LC_PER_ATMID
|
|
|
|
mvc __THREAD_per+__PER_address(8,%r1),__LC_PER_ADDRESS
|
|
|
|
mvc __THREAD_per+__PER_access_id(1,%r1),__LC_PER_ACCESS_ID
|
|
|
|
oi __TI_flags+7(%r9),_TIF_SINGLE_STEP # set TIF_SINGLE_STEP
|
2006-07-03 15:24:46 +08:00
|
|
|
TRACE_IRQS_ON
|
2005-04-17 06:20:36 +08:00
|
|
|
stosm __SF_EMPTY(%r15),0x03 # reenable interrupts
|
|
|
|
j sysc_do_svc
|
|
|
|
|
2006-09-20 21:58:39 +08:00
|
|
|
#
|
|
|
|
# per was called from kernel, must be kprobes
|
|
|
|
#
|
|
|
|
kernel_per:
|
2008-11-27 18:05:55 +08:00
|
|
|
xc SP_SVCNR(2,%r15),SP_SVCNR(%r15) # clear svc number
|
2006-09-20 21:58:39 +08:00
|
|
|
la %r2,SP_PTREGS(%r15) # address of register-save area
|
2007-11-20 18:13:32 +08:00
|
|
|
larl %r14,sysc_restore # load adr. of system ret, no work
|
2006-09-20 21:58:39 +08:00
|
|
|
jg do_single_step # branch to do_single_step
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* IO interrupt handler routine
|
|
|
|
*/
|
2006-09-28 22:56:37 +08:00
|
|
|
.globl io_int_handler
|
2005-04-17 06:20:36 +08:00
|
|
|
io_int_handler:
|
2008-12-25 20:39:25 +08:00
|
|
|
stpt __LC_ASYNC_ENTER_TIMER
|
2005-04-17 06:20:36 +08:00
|
|
|
stck __LC_INT_CLOCK
|
|
|
|
SAVE_ALL_BASE __LC_SAVE_AREA+32
|
2006-06-29 20:58:05 +08:00
|
|
|
SAVE_ALL_ASYNC __LC_IO_OLD_PSW,__LC_SAVE_AREA+32
|
2005-06-26 05:55:30 +08:00
|
|
|
CREATE_STACK_FRAME __LC_IO_OLD_PSW,__LC_SAVE_AREA+32
|
2005-04-17 06:20:36 +08:00
|
|
|
tm SP_PSW+1(%r15),0x01 # interrupting from user ?
|
|
|
|
jz io_no_vtime
|
|
|
|
UPDATE_VTIME __LC_EXIT_TIMER,__LC_ASYNC_ENTER_TIMER,__LC_USER_TIMER
|
|
|
|
UPDATE_VTIME __LC_LAST_UPDATE_TIMER,__LC_EXIT_TIMER,__LC_SYSTEM_TIMER
|
|
|
|
mvc __LC_LAST_UPDATE_TIMER(8),__LC_ASYNC_ENTER_TIMER
|
|
|
|
io_no_vtime:
|
|
|
|
lg %r9,__LC_THREAD_INFO # load pointer to thread_info struct
|
2006-07-03 15:24:46 +08:00
|
|
|
TRACE_IRQS_OFF
|
2006-09-28 22:56:37 +08:00
|
|
|
la %r2,SP_PTREGS(%r15) # address of register-save area
|
|
|
|
brasl %r14,do_IRQ # call standard irq handler
|
2005-04-17 06:20:36 +08:00
|
|
|
io_return:
|
|
|
|
tm __TI_flags+7(%r9),_TIF_WORK_INT
|
2006-09-28 22:56:37 +08:00
|
|
|
jnz io_work # there is work to do (signals etc.)
|
2007-11-20 18:13:32 +08:00
|
|
|
io_restore:
|
|
|
|
#ifdef CONFIG_TRACE_IRQFLAGS
|
|
|
|
larl %r1,io_restore_trace_psw
|
|
|
|
lpswe 0(%r1)
|
|
|
|
io_restore_trace:
|
|
|
|
TRACE_IRQS_CHECK
|
2007-10-12 04:11:12 +08:00
|
|
|
LOCKDEP_SYS_EXIT
|
2007-11-20 18:13:32 +08:00
|
|
|
#endif
|
2005-04-17 06:20:36 +08:00
|
|
|
io_leave:
|
2006-09-28 22:56:37 +08:00
|
|
|
RESTORE_ALL __LC_RETURN_PSW,0
|
2005-09-04 06:57:56 +08:00
|
|
|
io_done:
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2007-11-20 18:13:32 +08:00
|
|
|
#ifdef CONFIG_TRACE_IRQFLAGS
|
|
|
|
.align 8
|
|
|
|
.globl io_restore_trace_psw
|
|
|
|
io_restore_trace_psw:
|
|
|
|
.quad 0, io_restore_trace
|
|
|
|
#endif
|
|
|
|
|
2008-05-07 15:22:52 +08:00
|
|
|
#
|
2008-05-07 15:22:53 +08:00
|
|
|
# There is work todo, we need to check if we return to userspace, then
|
|
|
|
# check, if we are in SIE, if yes leave it
|
2008-05-07 15:22:52 +08:00
|
|
|
#
|
|
|
|
io_work:
|
|
|
|
tm SP_PSW+1(%r15),0x01 # returning to user ?
|
|
|
|
#ifndef CONFIG_PREEMPT
|
2008-05-07 15:22:53 +08:00
|
|
|
#if defined(CONFIG_KVM) || defined(CONFIG_KVM_MODULE)
|
|
|
|
jnz io_work_user # yes -> no need to check for SIE
|
|
|
|
la %r1, BASED(sie_opcode) # we return to kernel here
|
|
|
|
lg %r2, SP_PSW+8(%r15)
|
|
|
|
clc 0(2,%r1), 0(%r2) # is current instruction = SIE?
|
|
|
|
jne io_restore # no-> return to kernel
|
|
|
|
lg %r1, SP_PSW+8(%r15) # yes-> add 4 bytes to leave SIE
|
|
|
|
aghi %r1, 4
|
|
|
|
stg %r1, SP_PSW+8(%r15)
|
|
|
|
j io_restore # return to kernel
|
|
|
|
#else
|
2008-05-07 15:22:52 +08:00
|
|
|
jno io_restore # no-> skip resched & signal
|
2008-05-07 15:22:53 +08:00
|
|
|
#endif
|
2008-05-07 15:22:52 +08:00
|
|
|
#else
|
|
|
|
jnz io_work_user # yes -> do resched & signal
|
2008-05-07 15:22:53 +08:00
|
|
|
#if defined(CONFIG_KVM) || defined(CONFIG_KVM_MODULE)
|
|
|
|
la %r1, BASED(sie_opcode)
|
|
|
|
lg %r2, SP_PSW+8(%r15)
|
|
|
|
clc 0(2,%r1), 0(%r2) # is current instruction = SIE?
|
|
|
|
jne 0f # no -> leave PSW alone
|
|
|
|
lg %r1, SP_PSW+8(%r15) # yes-> add 4 bytes to leave SIE
|
|
|
|
aghi %r1, 4
|
|
|
|
stg %r1, SP_PSW+8(%r15)
|
|
|
|
0:
|
|
|
|
#endif
|
2008-05-07 15:22:52 +08:00
|
|
|
# check for preemptive scheduling
|
2006-09-28 22:56:37 +08:00
|
|
|
icm %r0,15,__TI_precount(%r9)
|
2008-05-07 15:22:52 +08:00
|
|
|
jnz io_restore # preemption is disabled
|
2005-04-17 06:20:36 +08:00
|
|
|
# switch to kernel stack
|
|
|
|
lg %r1,SP_R15(%r15)
|
|
|
|
aghi %r1,-SP_SIZE
|
|
|
|
mvc SP_PTREGS(__PT_SIZE,%r1),SP_PTREGS(%r15)
|
2006-09-28 22:56:37 +08:00
|
|
|
xc __SF_BACKCHAIN(8,%r1),__SF_BACKCHAIN(%r1) # clear back chain
|
2005-04-17 06:20:36 +08:00
|
|
|
lgr %r15,%r1
|
|
|
|
io_resume_loop:
|
|
|
|
tm __TI_flags+7(%r9),_TIF_NEED_RESCHED
|
2007-11-20 18:13:32 +08:00
|
|
|
jno io_restore
|
2007-11-20 18:13:35 +08:00
|
|
|
larl %r14,io_resume_loop
|
|
|
|
jg preempt_schedule_irq
|
2005-04-17 06:20:36 +08:00
|
|
|
#endif
|
|
|
|
|
2008-05-07 15:22:52 +08:00
|
|
|
io_work_user:
|
2005-04-17 06:20:36 +08:00
|
|
|
lg %r1,__LC_KERNEL_STACK
|
|
|
|
aghi %r1,-SP_SIZE
|
|
|
|
mvc SP_PTREGS(__PT_SIZE,%r1),SP_PTREGS(%r15)
|
2006-09-28 22:56:37 +08:00
|
|
|
xc __SF_BACKCHAIN(8,%r1),__SF_BACKCHAIN(%r1) # clear back chain
|
2005-04-17 06:20:36 +08:00
|
|
|
lgr %r15,%r1
|
|
|
|
#
|
|
|
|
# One of the work bits is on. Find out which one.
|
2006-02-01 19:06:38 +08:00
|
|
|
# Checked are: _TIF_SIGPENDING, _TIF_RESTORE_SIGPENDING, _TIF_NEED_RESCHED
|
|
|
|
# and _TIF_MCCK_PENDING
|
2005-04-17 06:20:36 +08:00
|
|
|
#
|
|
|
|
io_work_loop:
|
2005-06-26 05:55:30 +08:00
|
|
|
tm __TI_flags+7(%r9),_TIF_MCCK_PENDING
|
|
|
|
jo io_mcck_pending
|
2005-04-17 06:20:36 +08:00
|
|
|
tm __TI_flags+7(%r9),_TIF_NEED_RESCHED
|
|
|
|
jo io_reschedule
|
2008-04-30 15:53:08 +08:00
|
|
|
tm __TI_flags+7(%r9),_TIF_SIGPENDING
|
2006-02-01 19:06:38 +08:00
|
|
|
jnz io_sigpending
|
2008-10-11 03:33:20 +08:00
|
|
|
tm __TI_flags+7(%r9),_TIF_NOTIFY_RESUME
|
|
|
|
jnz io_notify_resume
|
2007-11-20 18:13:32 +08:00
|
|
|
j io_restore
|
|
|
|
io_work_done:
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2008-05-07 15:22:53 +08:00
|
|
|
#if defined(CONFIG_KVM) || defined(CONFIG_KVM_MODULE)
|
|
|
|
sie_opcode:
|
|
|
|
.long 0xb2140000
|
|
|
|
#endif
|
|
|
|
|
2005-06-26 05:55:30 +08:00
|
|
|
#
|
|
|
|
# _TIF_MCCK_PENDING is set, call handler
|
|
|
|
#
|
|
|
|
io_mcck_pending:
|
2007-07-27 18:29:18 +08:00
|
|
|
brasl %r14,s390_handle_mcck # TIF bit will be cleared by handler
|
|
|
|
j io_work_loop
|
2005-06-26 05:55:30 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
#
|
|
|
|
# _TIF_NEED_RESCHED is set, call schedule
|
2006-09-28 22:56:37 +08:00
|
|
|
#
|
|
|
|
io_reschedule:
|
2007-11-20 18:13:32 +08:00
|
|
|
TRACE_IRQS_ON
|
2006-09-28 22:56:37 +08:00
|
|
|
stosm __SF_EMPTY(%r15),0x03 # reenable interrupts
|
|
|
|
brasl %r14,schedule # call scheduler
|
|
|
|
stnsm __SF_EMPTY(%r15),0xfc # disable I/O and ext. interrupts
|
2007-11-20 18:13:32 +08:00
|
|
|
TRACE_IRQS_OFF
|
2005-04-17 06:20:36 +08:00
|
|
|
tm __TI_flags+7(%r9),_TIF_WORK_INT
|
2007-11-20 18:13:32 +08:00
|
|
|
jz io_restore # there is no work to do
|
2005-04-17 06:20:36 +08:00
|
|
|
j io_work_loop
|
|
|
|
|
|
|
|
#
|
2008-04-30 15:53:08 +08:00
|
|
|
# _TIF_SIGPENDING or is set, call do_signal
|
2005-04-17 06:20:36 +08:00
|
|
|
#
|
2006-09-28 22:56:37 +08:00
|
|
|
io_sigpending:
|
2007-11-20 18:13:32 +08:00
|
|
|
TRACE_IRQS_ON
|
2006-09-28 22:56:37 +08:00
|
|
|
stosm __SF_EMPTY(%r15),0x03 # reenable interrupts
|
|
|
|
la %r2,SP_PTREGS(%r15) # load pt_regs
|
2005-04-17 06:20:36 +08:00
|
|
|
brasl %r14,do_signal # call do_signal
|
2006-09-28 22:56:37 +08:00
|
|
|
stnsm __SF_EMPTY(%r15),0xfc # disable I/O and ext. interrupts
|
2007-11-20 18:13:32 +08:00
|
|
|
TRACE_IRQS_OFF
|
2005-11-07 16:59:02 +08:00
|
|
|
j io_work_loop
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2008-10-11 03:33:20 +08:00
|
|
|
#
|
|
|
|
# _TIF_NOTIFY_RESUME or is set, call do_notify_resume
|
|
|
|
#
|
|
|
|
io_notify_resume:
|
|
|
|
TRACE_IRQS_ON
|
|
|
|
stosm __SF_EMPTY(%r15),0x03 # reenable interrupts
|
|
|
|
la %r2,SP_PTREGS(%r15) # load pt_regs
|
|
|
|
brasl %r14,do_notify_resume # call do_notify_resume
|
|
|
|
stnsm __SF_EMPTY(%r15),0xfc # disable I/O and ext. interrupts
|
|
|
|
TRACE_IRQS_OFF
|
|
|
|
j io_work_loop
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* External interrupt handler routine
|
|
|
|
*/
|
2006-09-28 22:56:37 +08:00
|
|
|
.globl ext_int_handler
|
2005-04-17 06:20:36 +08:00
|
|
|
ext_int_handler:
|
2008-12-25 20:39:25 +08:00
|
|
|
stpt __LC_ASYNC_ENTER_TIMER
|
2005-04-17 06:20:36 +08:00
|
|
|
stck __LC_INT_CLOCK
|
|
|
|
SAVE_ALL_BASE __LC_SAVE_AREA+32
|
2006-06-29 20:58:05 +08:00
|
|
|
SAVE_ALL_ASYNC __LC_EXT_OLD_PSW,__LC_SAVE_AREA+32
|
2005-06-26 05:55:30 +08:00
|
|
|
CREATE_STACK_FRAME __LC_EXT_OLD_PSW,__LC_SAVE_AREA+32
|
2005-04-17 06:20:36 +08:00
|
|
|
tm SP_PSW+1(%r15),0x01 # interrupting from user ?
|
|
|
|
jz ext_no_vtime
|
|
|
|
UPDATE_VTIME __LC_EXIT_TIMER,__LC_ASYNC_ENTER_TIMER,__LC_USER_TIMER
|
|
|
|
UPDATE_VTIME __LC_LAST_UPDATE_TIMER,__LC_EXIT_TIMER,__LC_SYSTEM_TIMER
|
|
|
|
mvc __LC_LAST_UPDATE_TIMER(8),__LC_ASYNC_ENTER_TIMER
|
|
|
|
ext_no_vtime:
|
|
|
|
lg %r9,__LC_THREAD_INFO # load pointer to thread_info struct
|
2006-07-03 15:24:46 +08:00
|
|
|
TRACE_IRQS_OFF
|
2006-09-28 22:56:37 +08:00
|
|
|
la %r2,SP_PTREGS(%r15) # address of register-save area
|
|
|
|
llgh %r3,__LC_EXT_INT_CODE # get interruption code
|
|
|
|
brasl %r14,do_extint
|
2005-04-17 06:20:36 +08:00
|
|
|
j io_return
|
|
|
|
|
2005-09-04 06:57:56 +08:00
|
|
|
__critical_end:
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* Machine check handler routines
|
|
|
|
*/
|
2006-09-28 22:56:37 +08:00
|
|
|
.globl mcck_int_handler
|
2005-04-17 06:20:36 +08:00
|
|
|
mcck_int_handler:
|
2005-06-26 05:55:30 +08:00
|
|
|
la %r1,4095 # revalidate r1
|
|
|
|
spt __LC_CPU_TIMER_SAVE_AREA-4095(%r1) # revalidate cpu timer
|
2006-09-28 22:56:37 +08:00
|
|
|
lmg %r0,%r15,__LC_GPREGS_SAVE_AREA-4095(%r1)# revalidate gprs
|
2005-04-17 06:20:36 +08:00
|
|
|
SAVE_ALL_BASE __LC_SAVE_AREA+64
|
2005-06-26 05:55:30 +08:00
|
|
|
la %r12,__LC_MCK_OLD_PSW
|
2006-09-28 22:56:37 +08:00
|
|
|
tm __LC_MCCK_CODE,0x80 # system damage?
|
2005-06-26 05:55:30 +08:00
|
|
|
jo mcck_int_main # yes -> rest of mcck code invalid
|
2006-06-29 20:58:05 +08:00
|
|
|
la %r14,4095
|
|
|
|
mvc __LC_SAVE_AREA+104(8),__LC_ASYNC_ENTER_TIMER
|
|
|
|
mvc __LC_ASYNC_ENTER_TIMER(8),__LC_CPU_TIMER_SAVE_AREA-4095(%r14)
|
|
|
|
tm __LC_MCCK_CODE+5,0x02 # stored cpu timer value valid?
|
|
|
|
jo 1f
|
|
|
|
la %r14,__LC_SYNC_ENTER_TIMER
|
|
|
|
clc 0(8,%r14),__LC_ASYNC_ENTER_TIMER
|
|
|
|
jl 0f
|
|
|
|
la %r14,__LC_ASYNC_ENTER_TIMER
|
|
|
|
0: clc 0(8,%r14),__LC_EXIT_TIMER
|
|
|
|
jl 0f
|
|
|
|
la %r14,__LC_EXIT_TIMER
|
|
|
|
0: clc 0(8,%r14),__LC_LAST_UPDATE_TIMER
|
|
|
|
jl 0f
|
|
|
|
la %r14,__LC_LAST_UPDATE_TIMER
|
|
|
|
0: spt 0(%r14)
|
|
|
|
mvc __LC_ASYNC_ENTER_TIMER(8),0(%r14)
|
2008-12-25 20:39:25 +08:00
|
|
|
1: tm __LC_MCCK_CODE+2,0x09 # mwp + ia of old psw valid?
|
2005-06-26 05:55:30 +08:00
|
|
|
jno mcck_int_main # no -> skip cleanup critical
|
2006-09-28 22:56:37 +08:00
|
|
|
tm __LC_MCK_OLD_PSW+1,0x01 # test problem state bit
|
2005-06-26 05:55:30 +08:00
|
|
|
jnz mcck_int_main # from user -> load kernel stack
|
|
|
|
clc __LC_MCK_OLD_PSW+8(8),BASED(.Lcritical_end)
|
|
|
|
jhe mcck_int_main
|
2006-09-28 22:56:37 +08:00
|
|
|
clc __LC_MCK_OLD_PSW+8(8),BASED(.Lcritical_start)
|
2005-06-26 05:55:30 +08:00
|
|
|
jl mcck_int_main
|
2006-09-28 22:56:37 +08:00
|
|
|
brasl %r14,cleanup_critical
|
2005-06-26 05:55:30 +08:00
|
|
|
mcck_int_main:
|
2006-09-28 22:56:37 +08:00
|
|
|
lg %r14,__LC_PANIC_STACK # are we already on the panic stack?
|
2005-06-26 05:55:30 +08:00
|
|
|
slgr %r14,%r15
|
|
|
|
srag %r14,%r14,PAGE_SHIFT
|
|
|
|
jz 0f
|
2006-09-28 22:56:37 +08:00
|
|
|
lg %r15,__LC_PANIC_STACK # load panic stack
|
2005-06-26 05:55:30 +08:00
|
|
|
0: CREATE_STACK_FRAME __LC_MCK_OLD_PSW,__LC_SAVE_AREA+64
|
2005-09-04 06:57:56 +08:00
|
|
|
tm __LC_MCCK_CODE+2,0x08 # mwp of old psw valid?
|
|
|
|
jno mcck_no_vtime # no -> no timer update
|
2006-06-29 20:58:05 +08:00
|
|
|
tm SP_PSW+1(%r15),0x01 # interrupting from user ?
|
2005-09-04 06:57:56 +08:00
|
|
|
jz mcck_no_vtime
|
|
|
|
UPDATE_VTIME __LC_EXIT_TIMER,__LC_ASYNC_ENTER_TIMER,__LC_USER_TIMER
|
|
|
|
UPDATE_VTIME __LC_LAST_UPDATE_TIMER,__LC_EXIT_TIMER,__LC_SYSTEM_TIMER
|
|
|
|
mvc __LC_LAST_UPDATE_TIMER(8),__LC_ASYNC_ENTER_TIMER
|
|
|
|
mcck_no_vtime:
|
2005-06-26 05:55:30 +08:00
|
|
|
lg %r9,__LC_THREAD_INFO # load pointer to thread_info struct
|
|
|
|
la %r2,SP_PTREGS(%r15) # load pt_regs
|
|
|
|
brasl %r14,s390_do_machine_check
|
2006-09-28 22:56:37 +08:00
|
|
|
tm SP_PSW+1(%r15),0x01 # returning to user ?
|
2005-06-26 05:55:30 +08:00
|
|
|
jno mcck_return
|
|
|
|
lg %r1,__LC_KERNEL_STACK # switch to kernel stack
|
|
|
|
aghi %r1,-SP_SIZE
|
|
|
|
mvc SP_PTREGS(__PT_SIZE,%r1),SP_PTREGS(%r15)
|
|
|
|
xc __SF_BACKCHAIN(8,%r1),__SF_BACKCHAIN(%r1) # clear back chain
|
|
|
|
lgr %r15,%r1
|
|
|
|
stosm __SF_EMPTY(%r15),0x04 # turn dat on
|
|
|
|
tm __TI_flags+7(%r9),_TIF_MCCK_PENDING
|
|
|
|
jno mcck_return
|
2006-07-03 15:24:46 +08:00
|
|
|
TRACE_IRQS_OFF
|
2005-06-26 05:55:30 +08:00
|
|
|
brasl %r14,s390_handle_mcck
|
2006-07-03 15:24:46 +08:00
|
|
|
TRACE_IRQS_ON
|
2005-04-17 06:20:36 +08:00
|
|
|
mcck_return:
|
2006-06-29 20:58:05 +08:00
|
|
|
mvc __LC_RETURN_MCCK_PSW(16),SP_PSW(%r15) # move return PSW
|
|
|
|
ni __LC_RETURN_MCCK_PSW+1,0xfd # clear wait state bit
|
|
|
|
lmg %r0,%r15,SP_R0(%r15) # load gprs 0-15
|
|
|
|
mvc __LC_ASYNC_ENTER_TIMER(8),__LC_SAVE_AREA+104
|
|
|
|
tm __LC_RETURN_MCCK_PSW+1,0x01 # returning to user ?
|
|
|
|
jno 0f
|
|
|
|
stpt __LC_EXIT_TIMER
|
2008-12-25 20:39:25 +08:00
|
|
|
0: lpswe __LC_RETURN_MCCK_PSW # back to caller
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Restart interruption handler, kick starter for additional CPUs
|
|
|
|
*/
|
2007-06-19 19:10:03 +08:00
|
|
|
#ifdef CONFIG_SMP
|
2008-02-05 23:50:40 +08:00
|
|
|
__CPUINIT
|
2006-09-28 22:56:37 +08:00
|
|
|
.globl restart_int_handler
|
2005-04-17 06:20:36 +08:00
|
|
|
restart_int_handler:
|
2006-09-28 22:56:37 +08:00
|
|
|
lg %r15,__LC_SAVE_AREA+120 # load ksp
|
|
|
|
lghi %r10,__LC_CREGS_SAVE_AREA
|
|
|
|
lctlg %c0,%c15,0(%r10) # get new ctl regs
|
|
|
|
lghi %r10,__LC_AREGS_SAVE_AREA
|
|
|
|
lam %a0,%a15,0(%r10)
|
|
|
|
lmg %r6,%r15,__SF_GPRS(%r15) # load registers from clone
|
|
|
|
stosm __SF_EMPTY(%r15),0x04 # now we can turn dat on
|
|
|
|
jg start_secondary
|
2007-06-19 19:10:03 +08:00
|
|
|
.previous
|
2005-04-17 06:20:36 +08:00
|
|
|
#else
|
|
|
|
/*
|
|
|
|
* If we do not run with SMP enabled, let the new CPU crash ...
|
|
|
|
*/
|
2006-09-28 22:56:37 +08:00
|
|
|
.globl restart_int_handler
|
2005-04-17 06:20:36 +08:00
|
|
|
restart_int_handler:
|
2006-09-28 22:56:37 +08:00
|
|
|
basr %r1,0
|
2005-04-17 06:20:36 +08:00
|
|
|
restart_base:
|
2006-09-28 22:56:37 +08:00
|
|
|
lpswe restart_crash-restart_base(%r1)
|
|
|
|
.align 8
|
2005-04-17 06:20:36 +08:00
|
|
|
restart_crash:
|
2006-09-28 22:56:37 +08:00
|
|
|
.long 0x000a0000,0x00000000,0x00000000,0x00000000
|
2005-04-17 06:20:36 +08:00
|
|
|
restart_go:
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_CHECK_STACK
|
|
|
|
/*
|
|
|
|
* The synchronous or the asynchronous stack overflowed. We are dead.
|
|
|
|
* No need to properly save the registers, we are going to panic anyway.
|
|
|
|
* Setup a pt_regs so that show_trace can provide a good call trace.
|
|
|
|
*/
|
|
|
|
stack_overflow:
|
|
|
|
lg %r15,__LC_PANIC_STACK # change to panic stack
|
2006-09-20 21:59:22 +08:00
|
|
|
aghi %r15,-SP_SIZE
|
2005-04-17 06:20:36 +08:00
|
|
|
mvc SP_PSW(16,%r15),0(%r12) # move user PSW to stack
|
|
|
|
stmg %r0,%r11,SP_R0(%r15) # store gprs %r0-%r11 to kernel stack
|
|
|
|
la %r1,__LC_SAVE_AREA
|
|
|
|
chi %r12,__LC_SVC_OLD_PSW
|
|
|
|
je 0f
|
|
|
|
chi %r12,__LC_PGM_OLD_PSW
|
|
|
|
je 0f
|
2006-09-20 21:59:22 +08:00
|
|
|
la %r1,__LC_SAVE_AREA+32
|
2006-09-28 22:56:37 +08:00
|
|
|
0: mvc SP_R12(32,%r15),0(%r1) # move %r12-%r15 to stack
|
[S390] kernel: show last breaking-event-address on oops
Newer s390 models have a breaking-event-address-recording register.
Each time an instruction causes a break in the sequential instruction
execution, the address is saved in that hardware register. On a program
interrupt the address is copied to the lowcore address 272-279, which
makes it software accessible.
This patch changes the program check handler and the stack overflow
checker to copy the value into the pt_regs argument.
The oops output is enhanced to show the last known breaking address.
It might give additional information if the stack trace is corrupted.
The feature is only available on 64 bit.
The new oops output looks like:
[---------snip----------]
Modules linked in: vmcp sunrpc qeth_l2 dm_mod qeth ccwgroup
CPU: 2 Not tainted 2.6.24zlive-host #8
Process modprobe (pid: 4788, task: 00000000bf3d8718, ksp: 00000000b2b0b8e0)
Krnl PSW : 0704200180000000 000003e000020028 (vmcp_init+0x28/0xe4 [vmcp])
R:0 T:1 IO:1 EX:1 Key:0 M:1 W:0 P:0 AS:0 CC:2 PM:0 EA:3
Krnl GPRS: 0000000004000002 000003e000020000 0000000000000000 0000000000000001
000000000015734c ffffffffffffffff 000003e0000b3b00 0000000000000000
000003e00007ca30 00000000b5bb5d40 00000000b5bb5800 000003e0000b3b00
000003e0000a2000 00000000003ecf50 00000000b2b0bd50 00000000b2b0bcb0
Krnl Code: 000003e000020018: c0c000040ff4 larl %r12,3e0000a2000
000003e00002001e: e3e0f0000024 stg %r14,0(%r15)
000003e000020024: a7f40001 brc 15,3e000020026
>000003e000020028: e310c0100004 lg %r1,16(%r12)
000003e00002002e: c020000413dc larl %r2,3e0000a27e6
000003e000020034: c0a00004aee6 larl %r10,3e0000b5e00
000003e00002003a: a7490001 lghi %r4,1
000003e00002003e: a75900f0 lghi %r5,240
Call Trace:
([<000000000014b300>] blocking_notifier_call_chain+0x2c/0x40)
[<000000000015735c>] sys_init_module+0x19d8/0x1b08
[<0000000000110afc>] sysc_noemu+0x10/0x16
[<000002000011cda2>] 0x2000011cda2
Last Breaking-Event-Address:
[<000003e000020024>] vmcp_init+0x24/0xe4 [vmcp]
[---------snip----------]
Signed-off-by: Christian Borntraeger <borntraeger@de.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
Signed-off-by: Heiko Carstens <heiko.carstens@de.ibm.com>
2008-04-17 13:46:30 +08:00
|
|
|
mvc SP_ARGS(8,%r15),__LC_LAST_BREAK
|
2006-09-28 22:56:37 +08:00
|
|
|
xc __SF_BACKCHAIN(8,%r15),__SF_BACKCHAIN(%r15) # clear back chain
|
|
|
|
la %r2,SP_PTREGS(%r15) # load pt_regs
|
2005-04-17 06:20:36 +08:00
|
|
|
jg kernel_stack_overflow
|
|
|
|
#endif
|
|
|
|
|
|
|
|
cleanup_table_system_call:
|
|
|
|
.quad system_call, sysc_do_svc
|
|
|
|
cleanup_table_sysc_return:
|
|
|
|
.quad sysc_return, sysc_leave
|
|
|
|
cleanup_table_sysc_leave:
|
2007-11-20 18:13:32 +08:00
|
|
|
.quad sysc_leave, sysc_done
|
2005-04-17 06:20:36 +08:00
|
|
|
cleanup_table_sysc_work_loop:
|
2007-11-20 18:13:32 +08:00
|
|
|
.quad sysc_work_loop, sysc_work_done
|
2006-06-29 20:58:05 +08:00
|
|
|
cleanup_table_io_return:
|
|
|
|
.quad io_return, io_leave
|
2005-09-04 06:57:56 +08:00
|
|
|
cleanup_table_io_leave:
|
|
|
|
.quad io_leave, io_done
|
|
|
|
cleanup_table_io_work_loop:
|
2007-11-20 18:13:32 +08:00
|
|
|
.quad io_work_loop, io_work_done
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
cleanup_critical:
|
|
|
|
clc 8(8,%r12),BASED(cleanup_table_system_call)
|
|
|
|
jl 0f
|
|
|
|
clc 8(8,%r12),BASED(cleanup_table_system_call+8)
|
|
|
|
jl cleanup_system_call
|
|
|
|
0:
|
|
|
|
clc 8(8,%r12),BASED(cleanup_table_sysc_return)
|
|
|
|
jl 0f
|
|
|
|
clc 8(8,%r12),BASED(cleanup_table_sysc_return+8)
|
|
|
|
jl cleanup_sysc_return
|
|
|
|
0:
|
|
|
|
clc 8(8,%r12),BASED(cleanup_table_sysc_leave)
|
|
|
|
jl 0f
|
|
|
|
clc 8(8,%r12),BASED(cleanup_table_sysc_leave+8)
|
|
|
|
jl cleanup_sysc_leave
|
|
|
|
0:
|
|
|
|
clc 8(8,%r12),BASED(cleanup_table_sysc_work_loop)
|
|
|
|
jl 0f
|
|
|
|
clc 8(8,%r12),BASED(cleanup_table_sysc_work_loop+8)
|
2005-06-26 05:55:30 +08:00
|
|
|
jl cleanup_sysc_return
|
2006-06-29 20:58:05 +08:00
|
|
|
0:
|
|
|
|
clc 8(8,%r12),BASED(cleanup_table_io_return)
|
|
|
|
jl 0f
|
|
|
|
clc 8(8,%r12),BASED(cleanup_table_io_return+8)
|
|
|
|
jl cleanup_io_return
|
2005-09-04 06:57:56 +08:00
|
|
|
0:
|
|
|
|
clc 8(8,%r12),BASED(cleanup_table_io_leave)
|
|
|
|
jl 0f
|
|
|
|
clc 8(8,%r12),BASED(cleanup_table_io_leave+8)
|
|
|
|
jl cleanup_io_leave
|
|
|
|
0:
|
|
|
|
clc 8(8,%r12),BASED(cleanup_table_io_work_loop)
|
|
|
|
jl 0f
|
|
|
|
clc 8(8,%r12),BASED(cleanup_table_io_work_loop+8)
|
|
|
|
jl cleanup_io_return
|
2005-04-17 06:20:36 +08:00
|
|
|
0:
|
|
|
|
br %r14
|
|
|
|
|
|
|
|
cleanup_system_call:
|
|
|
|
mvc __LC_RETURN_PSW(16),0(%r12)
|
2005-09-04 06:57:56 +08:00
|
|
|
cghi %r12,__LC_MCK_OLD_PSW
|
|
|
|
je 0f
|
|
|
|
la %r12,__LC_SAVE_AREA+32
|
|
|
|
j 1f
|
|
|
|
0: la %r12,__LC_SAVE_AREA+64
|
|
|
|
1:
|
2005-04-17 06:20:36 +08:00
|
|
|
clc __LC_RETURN_PSW+8(8),BASED(cleanup_system_call_insn+8)
|
|
|
|
jh 0f
|
|
|
|
mvc __LC_SYNC_ENTER_TIMER(8),__LC_ASYNC_ENTER_TIMER
|
|
|
|
0: clc __LC_RETURN_PSW+8(8),BASED(cleanup_system_call_insn+16)
|
|
|
|
jhe cleanup_vtime
|
|
|
|
clc __LC_RETURN_PSW+8(8),BASED(cleanup_system_call_insn)
|
|
|
|
jh 0f
|
2005-09-04 06:57:56 +08:00
|
|
|
mvc __LC_SAVE_AREA(32),0(%r12)
|
|
|
|
0: stg %r13,8(%r12)
|
|
|
|
stg %r12,__LC_SAVE_AREA+96 # argh
|
2006-06-29 20:58:05 +08:00
|
|
|
SAVE_ALL_SYNC __LC_SVC_OLD_PSW,__LC_SAVE_AREA
|
2005-06-26 05:55:30 +08:00
|
|
|
CREATE_STACK_FRAME __LC_SVC_OLD_PSW,__LC_SAVE_AREA
|
2005-09-04 06:57:56 +08:00
|
|
|
lg %r12,__LC_SAVE_AREA+96 # argh
|
|
|
|
stg %r15,24(%r12)
|
2005-04-17 06:20:36 +08:00
|
|
|
llgh %r7,__LC_SVC_INT_CODE
|
|
|
|
cleanup_vtime:
|
|
|
|
clc __LC_RETURN_PSW+8(8),BASED(cleanup_system_call_insn+24)
|
|
|
|
jhe cleanup_stime
|
|
|
|
UPDATE_VTIME __LC_EXIT_TIMER,__LC_SYNC_ENTER_TIMER,__LC_USER_TIMER
|
|
|
|
cleanup_stime:
|
|
|
|
clc __LC_RETURN_PSW+8(8),BASED(cleanup_system_call_insn+32)
|
|
|
|
jh cleanup_update
|
|
|
|
UPDATE_VTIME __LC_LAST_UPDATE_TIMER,__LC_EXIT_TIMER,__LC_SYSTEM_TIMER
|
|
|
|
cleanup_update:
|
|
|
|
mvc __LC_LAST_UPDATE_TIMER(8),__LC_SYNC_ENTER_TIMER
|
|
|
|
mvc __LC_RETURN_PSW+8(8),BASED(cleanup_table_system_call+8)
|
|
|
|
la %r12,__LC_RETURN_PSW
|
|
|
|
br %r14
|
|
|
|
cleanup_system_call_insn:
|
|
|
|
.quad sysc_saveall
|
2006-09-28 22:56:37 +08:00
|
|
|
.quad system_call
|
|
|
|
.quad sysc_vtime
|
|
|
|
.quad sysc_stime
|
|
|
|
.quad sysc_update
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
cleanup_sysc_return:
|
|
|
|
mvc __LC_RETURN_PSW(8),0(%r12)
|
|
|
|
mvc __LC_RETURN_PSW+8(8),BASED(cleanup_table_sysc_return)
|
|
|
|
la %r12,__LC_RETURN_PSW
|
|
|
|
br %r14
|
|
|
|
|
|
|
|
cleanup_sysc_leave:
|
|
|
|
clc 8(8,%r12),BASED(cleanup_sysc_leave_insn)
|
2005-09-04 06:57:56 +08:00
|
|
|
je 2f
|
2005-04-17 06:20:36 +08:00
|
|
|
mvc __LC_EXIT_TIMER(8),__LC_ASYNC_ENTER_TIMER
|
|
|
|
clc 8(8,%r12),BASED(cleanup_sysc_leave_insn+8)
|
2005-09-04 06:57:56 +08:00
|
|
|
je 2f
|
2005-04-17 06:20:36 +08:00
|
|
|
mvc __LC_RETURN_PSW(16),SP_PSW(%r15)
|
2005-09-04 06:57:56 +08:00
|
|
|
cghi %r12,__LC_MCK_OLD_PSW
|
|
|
|
jne 0f
|
|
|
|
mvc __LC_SAVE_AREA+64(32),SP_R12(%r15)
|
|
|
|
j 1f
|
|
|
|
0: mvc __LC_SAVE_AREA+32(32),SP_R12(%r15)
|
|
|
|
1: lmg %r0,%r11,SP_R0(%r15)
|
2005-04-17 06:20:36 +08:00
|
|
|
lg %r15,SP_R15(%r15)
|
2005-09-04 06:57:56 +08:00
|
|
|
2: la %r12,__LC_RETURN_PSW
|
2005-04-17 06:20:36 +08:00
|
|
|
br %r14
|
|
|
|
cleanup_sysc_leave_insn:
|
2007-11-20 18:13:32 +08:00
|
|
|
.quad sysc_done - 4
|
|
|
|
.quad sysc_done - 8
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2005-09-04 06:57:56 +08:00
|
|
|
cleanup_io_return:
|
|
|
|
mvc __LC_RETURN_PSW(8),0(%r12)
|
|
|
|
mvc __LC_RETURN_PSW+8(8),BASED(cleanup_table_io_work_loop)
|
|
|
|
la %r12,__LC_RETURN_PSW
|
|
|
|
br %r14
|
|
|
|
|
|
|
|
cleanup_io_leave:
|
|
|
|
clc 8(8,%r12),BASED(cleanup_io_leave_insn)
|
|
|
|
je 2f
|
|
|
|
mvc __LC_EXIT_TIMER(8),__LC_ASYNC_ENTER_TIMER
|
|
|
|
clc 8(8,%r12),BASED(cleanup_io_leave_insn+8)
|
|
|
|
je 2f
|
|
|
|
mvc __LC_RETURN_PSW(16),SP_PSW(%r15)
|
|
|
|
cghi %r12,__LC_MCK_OLD_PSW
|
|
|
|
jne 0f
|
|
|
|
mvc __LC_SAVE_AREA+64(32),SP_R12(%r15)
|
|
|
|
j 1f
|
|
|
|
0: mvc __LC_SAVE_AREA+32(32),SP_R12(%r15)
|
|
|
|
1: lmg %r0,%r11,SP_R0(%r15)
|
|
|
|
lg %r15,SP_R15(%r15)
|
|
|
|
2: la %r12,__LC_RETURN_PSW
|
|
|
|
br %r14
|
|
|
|
cleanup_io_leave_insn:
|
2007-11-20 18:13:32 +08:00
|
|
|
.quad io_done - 4
|
|
|
|
.quad io_done - 8
|
2005-09-04 06:57:56 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* Integer constants
|
|
|
|
*/
|
2006-09-28 22:56:37 +08:00
|
|
|
.align 4
|
2005-04-17 06:20:36 +08:00
|
|
|
.Lconst:
|
2006-09-28 22:56:37 +08:00
|
|
|
.Lnr_syscalls: .long NR_syscalls
|
|
|
|
.L0x0130: .short 0x130
|
|
|
|
.L0x0140: .short 0x140
|
|
|
|
.L0x0150: .short 0x150
|
|
|
|
.L0x0160: .short 0x160
|
|
|
|
.L0x0170: .short 0x170
|
2005-04-17 06:20:36 +08:00
|
|
|
.Lcritical_start:
|
2006-09-28 22:56:37 +08:00
|
|
|
.quad __critical_start
|
2005-04-17 06:20:36 +08:00
|
|
|
.Lcritical_end:
|
2006-09-28 22:56:37 +08:00
|
|
|
.quad __critical_end
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-09-28 22:56:37 +08:00
|
|
|
.section .rodata, "a"
|
2005-04-17 06:20:36 +08:00
|
|
|
#define SYSCALL(esa,esame,emu) .long esame
|
|
|
|
sys_call_table:
|
|
|
|
#include "syscalls.S"
|
|
|
|
#undef SYSCALL
|
|
|
|
|
2006-01-06 16:19:28 +08:00
|
|
|
#ifdef CONFIG_COMPAT
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
#define SYSCALL(esa,esame,emu) .long emu
|
|
|
|
sys_call_table_emu:
|
|
|
|
#include "syscalls.S"
|
|
|
|
#undef SYSCALL
|
|
|
|
#endif
|