191 строка
5.5 KiB
C
191 строка
5.5 KiB
C
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
#include <linux/compat.h>
|
|
#include <linux/context_tracking.h>
|
|
#include <linux/randomize_kstack.h>
|
|
|
|
#include <asm/interrupt.h>
|
|
#include <asm/kup.h>
|
|
#include <asm/syscall.h>
|
|
#include <asm/time.h>
|
|
#include <asm/tm.h>
|
|
#include <asm/unistd.h>
|
|
|
|
|
|
typedef long (*syscall_fn)(long, long, long, long, long, long);
|
|
|
|
/* Has to run notrace because it is entered not completely "reconciled" */
|
|
notrace long system_call_exception(long r3, long r4, long r5,
|
|
long r6, long r7, long r8,
|
|
unsigned long r0, struct pt_regs *regs)
|
|
{
|
|
long ret;
|
|
syscall_fn f;
|
|
|
|
kuap_lock();
|
|
|
|
add_random_kstack_offset();
|
|
regs->orig_gpr3 = r3;
|
|
|
|
if (IS_ENABLED(CONFIG_PPC_IRQ_SOFT_MASK_DEBUG))
|
|
BUG_ON(irq_soft_mask_return() != IRQS_ALL_DISABLED);
|
|
|
|
trace_hardirqs_off(); /* finish reconciling */
|
|
|
|
CT_WARN_ON(ct_state() == CONTEXT_KERNEL);
|
|
user_exit_irqoff();
|
|
|
|
BUG_ON(regs_is_unrecoverable(regs));
|
|
BUG_ON(!(regs->msr & MSR_PR));
|
|
BUG_ON(arch_irq_disabled_regs(regs));
|
|
|
|
#ifdef CONFIG_PPC_PKEY
|
|
if (mmu_has_feature(MMU_FTR_PKEY)) {
|
|
unsigned long amr, iamr;
|
|
bool flush_needed = false;
|
|
/*
|
|
* When entering from userspace we mostly have the AMR/IAMR
|
|
* different from kernel default values. Hence don't compare.
|
|
*/
|
|
amr = mfspr(SPRN_AMR);
|
|
iamr = mfspr(SPRN_IAMR);
|
|
regs->amr = amr;
|
|
regs->iamr = iamr;
|
|
if (mmu_has_feature(MMU_FTR_BOOK3S_KUAP)) {
|
|
mtspr(SPRN_AMR, AMR_KUAP_BLOCKED);
|
|
flush_needed = true;
|
|
}
|
|
if (mmu_has_feature(MMU_FTR_BOOK3S_KUEP)) {
|
|
mtspr(SPRN_IAMR, AMR_KUEP_BLOCKED);
|
|
flush_needed = true;
|
|
}
|
|
if (flush_needed)
|
|
isync();
|
|
} else
|
|
#endif
|
|
kuap_assert_locked();
|
|
|
|
booke_restore_dbcr0();
|
|
|
|
account_cpu_user_entry();
|
|
|
|
account_stolen_time();
|
|
|
|
/*
|
|
* This is not required for the syscall exit path, but makes the
|
|
* stack frame look nicer. If this was initialised in the first stack
|
|
* frame, or if the unwinder was taught the first stack frame always
|
|
* returns to user with IRQS_ENABLED, this store could be avoided!
|
|
*/
|
|
irq_soft_mask_regs_set_state(regs, IRQS_ENABLED);
|
|
|
|
/*
|
|
* If system call is called with TM active, set _TIF_RESTOREALL to
|
|
* prevent RFSCV being used to return to userspace, because POWER9
|
|
* TM implementation has problems with this instruction returning to
|
|
* transactional state. Final register values are not relevant because
|
|
* the transaction will be aborted upon return anyway. Or in the case
|
|
* of unsupported_scv SIGILL fault, the return state does not much
|
|
* matter because it's an edge case.
|
|
*/
|
|
if (IS_ENABLED(CONFIG_PPC_TRANSACTIONAL_MEM) &&
|
|
unlikely(MSR_TM_TRANSACTIONAL(regs->msr)))
|
|
set_bits(_TIF_RESTOREALL, ¤t_thread_info()->flags);
|
|
|
|
/*
|
|
* If the system call was made with a transaction active, doom it and
|
|
* return without performing the system call. Unless it was an
|
|
* unsupported scv vector, in which case it's treated like an illegal
|
|
* instruction.
|
|
*/
|
|
#ifdef CONFIG_PPC_TRANSACTIONAL_MEM
|
|
if (unlikely(MSR_TM_TRANSACTIONAL(regs->msr)) &&
|
|
!trap_is_unsupported_scv(regs)) {
|
|
/* Enable TM in the kernel, and disable EE (for scv) */
|
|
hard_irq_disable();
|
|
mtmsr(mfmsr() | MSR_TM);
|
|
|
|
/* tabort, this dooms the transaction, nothing else */
|
|
asm volatile(".long 0x7c00071d | ((%0) << 16)"
|
|
:: "r"(TM_CAUSE_SYSCALL|TM_CAUSE_PERSISTENT));
|
|
|
|
/*
|
|
* Userspace will never see the return value. Execution will
|
|
* resume after the tbegin. of the aborted transaction with the
|
|
* checkpointed register state. A context switch could occur
|
|
* or signal delivered to the process before resuming the
|
|
* doomed transaction context, but that should all be handled
|
|
* as expected.
|
|
*/
|
|
return -ENOSYS;
|
|
}
|
|
#endif // CONFIG_PPC_TRANSACTIONAL_MEM
|
|
|
|
local_irq_enable();
|
|
|
|
if (unlikely(read_thread_flags() & _TIF_SYSCALL_DOTRACE)) {
|
|
if (unlikely(trap_is_unsupported_scv(regs))) {
|
|
/* Unsupported scv vector */
|
|
_exception(SIGILL, regs, ILL_ILLOPC, regs->nip);
|
|
return regs->gpr[3];
|
|
}
|
|
/*
|
|
* We use the return value of do_syscall_trace_enter() as the
|
|
* syscall number. If the syscall was rejected for any reason
|
|
* do_syscall_trace_enter() returns an invalid syscall number
|
|
* and the test against NR_syscalls will fail and the return
|
|
* value to be used is in regs->gpr[3].
|
|
*/
|
|
r0 = do_syscall_trace_enter(regs);
|
|
if (unlikely(r0 >= NR_syscalls))
|
|
return regs->gpr[3];
|
|
r3 = regs->gpr[3];
|
|
r4 = regs->gpr[4];
|
|
r5 = regs->gpr[5];
|
|
r6 = regs->gpr[6];
|
|
r7 = regs->gpr[7];
|
|
r8 = regs->gpr[8];
|
|
|
|
} else if (unlikely(r0 >= NR_syscalls)) {
|
|
if (unlikely(trap_is_unsupported_scv(regs))) {
|
|
/* Unsupported scv vector */
|
|
_exception(SIGILL, regs, ILL_ILLOPC, regs->nip);
|
|
return regs->gpr[3];
|
|
}
|
|
return -ENOSYS;
|
|
}
|
|
|
|
/* May be faster to do array_index_nospec? */
|
|
barrier_nospec();
|
|
|
|
if (unlikely(is_compat_task())) {
|
|
f = (void *)compat_sys_call_table[r0];
|
|
|
|
r3 &= 0x00000000ffffffffULL;
|
|
r4 &= 0x00000000ffffffffULL;
|
|
r5 &= 0x00000000ffffffffULL;
|
|
r6 &= 0x00000000ffffffffULL;
|
|
r7 &= 0x00000000ffffffffULL;
|
|
r8 &= 0x00000000ffffffffULL;
|
|
|
|
} else {
|
|
f = (void *)sys_call_table[r0];
|
|
}
|
|
|
|
ret = f(r3, r4, r5, r6, r7, r8);
|
|
|
|
/*
|
|
* Ultimately, this value will get limited by KSTACK_OFFSET_MAX(),
|
|
* so the maximum stack offset is 1k bytes (10 bits).
|
|
*
|
|
* The actual entropy will be further reduced by the compiler when
|
|
* applying stack alignment constraints: the powerpc architecture
|
|
* may have two kinds of stack alignment (16-bytes and 8-bytes).
|
|
*
|
|
* So the resulting 6 or 7 bits of entropy is seen in SP[9:4] or SP[9:3].
|
|
*/
|
|
choose_random_kstack_offset(mftb());
|
|
|
|
return ret;
|
|
}
|