Commit d00a3810 authored by Will Deacon's avatar Will Deacon Committed by Catalin Marinas

arm64: context-switch user tls register tpidr_el0 for compat tasks

Since commit a4780ade ("ARM: 7735/2: Preserve the user r/w register
TPIDRURW on context switch and fork"), arch/arm/ has context switched
the user-writable TLS register, so do the same for compat tasks running
under the arm64 kernel.
Reported-by: default avatarAndré Hentschel <nerv@dawncrow.de>
Tested-by: default avatarAndré Hentschel <nerv@dawncrow.de>
Signed-off-by: default avatarWill Deacon <will.deacon@arm.com>
Signed-off-by: default avatarCatalin Marinas <catalin.marinas@arm.com>
parent 05981277
...@@ -78,13 +78,30 @@ struct cpu_context { ...@@ -78,13 +78,30 @@ struct cpu_context {
struct thread_struct { struct thread_struct {
struct cpu_context cpu_context; /* cpu context */ struct cpu_context cpu_context; /* cpu context */
unsigned long tp_value; unsigned long tp_value; /* TLS register */
#ifdef CONFIG_COMPAT
unsigned long tp2_value;
#endif
struct fpsimd_state fpsimd_state; struct fpsimd_state fpsimd_state;
unsigned long fault_address; /* fault info */ unsigned long fault_address; /* fault info */
unsigned long fault_code; /* ESR_EL1 value */ unsigned long fault_code; /* ESR_EL1 value */
struct debug_info debug; /* debugging */ struct debug_info debug; /* debugging */
}; };
#ifdef CONFIG_COMPAT
#define task_user_tls(t) \
({ \
unsigned long *__tls; \
if (is_compat_thread(task_thread_info(t))) \
__tls = &(t)->thread.tp2_value; \
else \
__tls = &(t)->thread.tp_value; \
__tls; \
})
#else
#define task_user_tls(t) (&(t)->thread.tp_value)
#endif
#define INIT_THREAD { } #define INIT_THREAD { }
static inline void start_thread_common(struct pt_regs *regs, unsigned long pc) static inline void start_thread_common(struct pt_regs *regs, unsigned long pc)
......
...@@ -244,35 +244,35 @@ int copy_thread(unsigned long clone_flags, unsigned long stack_start, ...@@ -244,35 +244,35 @@ int copy_thread(unsigned long clone_flags, unsigned long stack_start,
unsigned long stk_sz, struct task_struct *p) unsigned long stk_sz, struct task_struct *p)
{ {
struct pt_regs *childregs = task_pt_regs(p); struct pt_regs *childregs = task_pt_regs(p);
unsigned long tls = p->thread.tp_value;
memset(&p->thread.cpu_context, 0, sizeof(struct cpu_context)); memset(&p->thread.cpu_context, 0, sizeof(struct cpu_context));
if (likely(!(p->flags & PF_KTHREAD))) { if (likely(!(p->flags & PF_KTHREAD))) {
*childregs = *current_pt_regs(); *childregs = *current_pt_regs();
childregs->regs[0] = 0; childregs->regs[0] = 0;
if (is_compat_thread(task_thread_info(p))) {
if (stack_start)
childregs->compat_sp = stack_start;
} else {
/* /*
* Read the current TLS pointer from tpidr_el0 as it may be * Read the current TLS pointer from tpidr_el0 as it may be
* out-of-sync with the saved value. * out-of-sync with the saved value.
*/ */
asm("mrs %0, tpidr_el0" : "=r" (tls)); asm("mrs %0, tpidr_el0" : "=r" (*task_user_tls(p)));
if (stack_start) { if (stack_start) {
if (is_compat_thread(task_thread_info(p)))
childregs->compat_sp = stack_start;
/* 16-byte aligned stack mandatory on AArch64 */ /* 16-byte aligned stack mandatory on AArch64 */
if (stack_start & 15) else if (stack_start & 15)
return -EINVAL; return -EINVAL;
else
childregs->sp = stack_start; childregs->sp = stack_start;
} }
}
/* /*
* If a TLS pointer was passed to clone (4th argument), use it * If a TLS pointer was passed to clone (4th argument), use it
* for the new thread. * for the new thread.
*/ */
if (clone_flags & CLONE_SETTLS) if (clone_flags & CLONE_SETTLS)
tls = childregs->regs[3]; p->thread.tp_value = childregs->regs[3];
} else { } else {
memset(childregs, 0, sizeof(struct pt_regs)); memset(childregs, 0, sizeof(struct pt_regs));
childregs->pstate = PSR_MODE_EL1h; childregs->pstate = PSR_MODE_EL1h;
...@@ -281,7 +281,6 @@ int copy_thread(unsigned long clone_flags, unsigned long stack_start, ...@@ -281,7 +281,6 @@ int copy_thread(unsigned long clone_flags, unsigned long stack_start,
} }
p->thread.cpu_context.pc = (unsigned long)ret_from_fork; p->thread.cpu_context.pc = (unsigned long)ret_from_fork;
p->thread.cpu_context.sp = (unsigned long)childregs; p->thread.cpu_context.sp = (unsigned long)childregs;
p->thread.tp_value = tls;
ptrace_hw_copy_thread(p); ptrace_hw_copy_thread(p);
...@@ -292,18 +291,12 @@ static void tls_thread_switch(struct task_struct *next) ...@@ -292,18 +291,12 @@ static void tls_thread_switch(struct task_struct *next)
{ {
unsigned long tpidr, tpidrro; unsigned long tpidr, tpidrro;
if (!is_compat_task()) {
asm("mrs %0, tpidr_el0" : "=r" (tpidr)); asm("mrs %0, tpidr_el0" : "=r" (tpidr));
current->thread.tp_value = tpidr; *task_user_tls(current) = tpidr;
}
if (is_compat_thread(task_thread_info(next))) { tpidr = *task_user_tls(next);
tpidr = 0; tpidrro = is_compat_thread(task_thread_info(next)) ?
tpidrro = next->thread.tp_value; next->thread.tp_value : 0;
} else {
tpidr = next->thread.tp_value;
tpidrro = 0;
}
asm( asm(
" msr tpidr_el0, %0\n" " msr tpidr_el0, %0\n"
......
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