Commit 7bcaad1f authored by Vasily Gorbik's avatar Vasily Gorbik

s390: avoid misusing CALL_ON_STACK for task stack setup

CALL_ON_STACK is intended to be used for temporary stack switching with
potential return to the caller.

When CALL_ON_STACK is misused to switch from nodat stack to task stack
back_chain information would later lead stack unwinder from task stack into
(per cpu) nodat stack which is reused for other purposes. This would
yield confusing unwinding result or errors.

To avoid that introduce CALL_ON_STACK_NORETURN to be used instead. It
makes sure that back_chain is zeroed and unwinder finishes gracefully
ending up at task pt_regs.
Reviewed-by: default avatarHeiko Carstens <heiko.carstens@de.ibm.com>
Signed-off-by: default avatarVasily Gorbik <gor@linux.ibm.com>
parent 75794257
...@@ -124,4 +124,15 @@ struct stack_frame { ...@@ -124,4 +124,15 @@ struct stack_frame {
r2; \ r2; \
}) })
#define CALL_ON_STACK_NORETURN(fn, stack) \
({ \
asm volatile( \
" la 15,0(%[_stack])\n" \
" xc %[_bc](8,15),%[_bc](15)\n" \
" brasl 14,%[_fn]\n" \
::[_bc] "i" (offsetof(struct stack_frame, back_chain)), \
[_stack] "a" (stack), [_fn] "X" (fn)); \
BUG(); \
})
#endif /* _ASM_S390_STACKTRACE_H */ #endif /* _ASM_S390_STACKTRACE_H */
...@@ -355,7 +355,6 @@ early_initcall(async_stack_realloc); ...@@ -355,7 +355,6 @@ early_initcall(async_stack_realloc);
void __init arch_call_rest_init(void) void __init arch_call_rest_init(void)
{ {
struct stack_frame *frame;
unsigned long stack; unsigned long stack;
stack = stack_alloc(); stack = stack_alloc();
...@@ -368,13 +367,7 @@ void __init arch_call_rest_init(void) ...@@ -368,13 +367,7 @@ void __init arch_call_rest_init(void)
set_task_stack_end_magic(current); set_task_stack_end_magic(current);
stack += STACK_INIT_OFFSET; stack += STACK_INIT_OFFSET;
S390_lowcore.kernel_stack = stack; S390_lowcore.kernel_stack = stack;
frame = (struct stack_frame *) stack; CALL_ON_STACK_NORETURN(rest_init, stack);
memset(frame, 0, sizeof(*frame));
/* Branch to rest_init on the new stack, never returns */
asm volatile(
" la 15,0(%[_frame])\n"
" jg rest_init\n"
: : [_frame] "a" (frame));
} }
static void __init setup_lowcore_dat_off(void) static void __init setup_lowcore_dat_off(void)
......
...@@ -876,7 +876,7 @@ static void __no_sanitize_address smp_start_secondary(void *cpuvoid) ...@@ -876,7 +876,7 @@ static void __no_sanitize_address smp_start_secondary(void *cpuvoid)
S390_lowcore.restart_source = -1UL; S390_lowcore.restart_source = -1UL;
__ctl_load(S390_lowcore.cregs_save_area, 0, 15); __ctl_load(S390_lowcore.cregs_save_area, 0, 15);
__load_psw_mask(PSW_KERNEL_BITS | PSW_MASK_DAT); __load_psw_mask(PSW_KERNEL_BITS | PSW_MASK_DAT);
CALL_ON_STACK(smp_init_secondary, S390_lowcore.kernel_stack, 0); CALL_ON_STACK_NORETURN(smp_init_secondary, S390_lowcore.kernel_stack);
} }
/* Upping and downing of CPUs */ /* Upping and downing of CPUs */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment