11547db7dSXiu Jianfeng // SPDX-License-Identifier: GPL-2.0-or-later 21547db7dSXiu Jianfeng 31547db7dSXiu Jianfeng #include <linux/compat.h> 41547db7dSXiu Jianfeng #include <linux/context_tracking.h> 5*f4a0318fSXiu Jianfeng #include <linux/randomize_kstack.h> 61547db7dSXiu Jianfeng 71547db7dSXiu Jianfeng #include <asm/interrupt.h> 81547db7dSXiu Jianfeng #include <asm/kup.h> 91547db7dSXiu Jianfeng #include <asm/syscall.h> 101547db7dSXiu Jianfeng #include <asm/time.h> 111547db7dSXiu Jianfeng #include <asm/tm.h> 121547db7dSXiu Jianfeng #include <asm/unistd.h> 131547db7dSXiu Jianfeng 141547db7dSXiu Jianfeng 151547db7dSXiu Jianfeng typedef long (*syscall_fn)(long, long, long, long, long, long); 161547db7dSXiu Jianfeng 171547db7dSXiu Jianfeng /* Has to run notrace because it is entered not completely "reconciled" */ 181547db7dSXiu Jianfeng notrace long system_call_exception(long r3, long r4, long r5, 191547db7dSXiu Jianfeng long r6, long r7, long r8, 201547db7dSXiu Jianfeng unsigned long r0, struct pt_regs *regs) 211547db7dSXiu Jianfeng { 22*f4a0318fSXiu Jianfeng long ret; 231547db7dSXiu Jianfeng syscall_fn f; 241547db7dSXiu Jianfeng 251547db7dSXiu Jianfeng kuap_lock(); 261547db7dSXiu Jianfeng 27*f4a0318fSXiu Jianfeng add_random_kstack_offset(); 281547db7dSXiu Jianfeng regs->orig_gpr3 = r3; 291547db7dSXiu Jianfeng 301547db7dSXiu Jianfeng if (IS_ENABLED(CONFIG_PPC_IRQ_SOFT_MASK_DEBUG)) 311547db7dSXiu Jianfeng BUG_ON(irq_soft_mask_return() != IRQS_ALL_DISABLED); 321547db7dSXiu Jianfeng 331547db7dSXiu Jianfeng trace_hardirqs_off(); /* finish reconciling */ 341547db7dSXiu Jianfeng 351547db7dSXiu Jianfeng CT_WARN_ON(ct_state() == CONTEXT_KERNEL); 361547db7dSXiu Jianfeng user_exit_irqoff(); 371547db7dSXiu Jianfeng 381547db7dSXiu Jianfeng BUG_ON(regs_is_unrecoverable(regs)); 391547db7dSXiu Jianfeng BUG_ON(!(regs->msr & MSR_PR)); 401547db7dSXiu Jianfeng BUG_ON(arch_irq_disabled_regs(regs)); 411547db7dSXiu Jianfeng 421547db7dSXiu Jianfeng #ifdef CONFIG_PPC_PKEY 431547db7dSXiu Jianfeng if (mmu_has_feature(MMU_FTR_PKEY)) { 441547db7dSXiu Jianfeng unsigned long amr, iamr; 451547db7dSXiu Jianfeng bool flush_needed = false; 461547db7dSXiu Jianfeng /* 471547db7dSXiu Jianfeng * When entering from userspace we mostly have the AMR/IAMR 481547db7dSXiu Jianfeng * different from kernel default values. Hence don't compare. 491547db7dSXiu Jianfeng */ 501547db7dSXiu Jianfeng amr = mfspr(SPRN_AMR); 511547db7dSXiu Jianfeng iamr = mfspr(SPRN_IAMR); 521547db7dSXiu Jianfeng regs->amr = amr; 531547db7dSXiu Jianfeng regs->iamr = iamr; 541547db7dSXiu Jianfeng if (mmu_has_feature(MMU_FTR_BOOK3S_KUAP)) { 551547db7dSXiu Jianfeng mtspr(SPRN_AMR, AMR_KUAP_BLOCKED); 561547db7dSXiu Jianfeng flush_needed = true; 571547db7dSXiu Jianfeng } 581547db7dSXiu Jianfeng if (mmu_has_feature(MMU_FTR_BOOK3S_KUEP)) { 591547db7dSXiu Jianfeng mtspr(SPRN_IAMR, AMR_KUEP_BLOCKED); 601547db7dSXiu Jianfeng flush_needed = true; 611547db7dSXiu Jianfeng } 621547db7dSXiu Jianfeng if (flush_needed) 631547db7dSXiu Jianfeng isync(); 641547db7dSXiu Jianfeng } else 651547db7dSXiu Jianfeng #endif 661547db7dSXiu Jianfeng kuap_assert_locked(); 671547db7dSXiu Jianfeng 681547db7dSXiu Jianfeng booke_restore_dbcr0(); 691547db7dSXiu Jianfeng 701547db7dSXiu Jianfeng account_cpu_user_entry(); 711547db7dSXiu Jianfeng 721547db7dSXiu Jianfeng account_stolen_time(); 731547db7dSXiu Jianfeng 741547db7dSXiu Jianfeng /* 751547db7dSXiu Jianfeng * This is not required for the syscall exit path, but makes the 761547db7dSXiu Jianfeng * stack frame look nicer. If this was initialised in the first stack 771547db7dSXiu Jianfeng * frame, or if the unwinder was taught the first stack frame always 781547db7dSXiu Jianfeng * returns to user with IRQS_ENABLED, this store could be avoided! 791547db7dSXiu Jianfeng */ 801547db7dSXiu Jianfeng irq_soft_mask_regs_set_state(regs, IRQS_ENABLED); 811547db7dSXiu Jianfeng 821547db7dSXiu Jianfeng /* 831547db7dSXiu Jianfeng * If system call is called with TM active, set _TIF_RESTOREALL to 841547db7dSXiu Jianfeng * prevent RFSCV being used to return to userspace, because POWER9 851547db7dSXiu Jianfeng * TM implementation has problems with this instruction returning to 861547db7dSXiu Jianfeng * transactional state. Final register values are not relevant because 871547db7dSXiu Jianfeng * the transaction will be aborted upon return anyway. Or in the case 881547db7dSXiu Jianfeng * of unsupported_scv SIGILL fault, the return state does not much 891547db7dSXiu Jianfeng * matter because it's an edge case. 901547db7dSXiu Jianfeng */ 911547db7dSXiu Jianfeng if (IS_ENABLED(CONFIG_PPC_TRANSACTIONAL_MEM) && 921547db7dSXiu Jianfeng unlikely(MSR_TM_TRANSACTIONAL(regs->msr))) 931547db7dSXiu Jianfeng set_bits(_TIF_RESTOREALL, ¤t_thread_info()->flags); 941547db7dSXiu Jianfeng 951547db7dSXiu Jianfeng /* 961547db7dSXiu Jianfeng * If the system call was made with a transaction active, doom it and 971547db7dSXiu Jianfeng * return without performing the system call. Unless it was an 981547db7dSXiu Jianfeng * unsupported scv vector, in which case it's treated like an illegal 991547db7dSXiu Jianfeng * instruction. 1001547db7dSXiu Jianfeng */ 1011547db7dSXiu Jianfeng #ifdef CONFIG_PPC_TRANSACTIONAL_MEM 1021547db7dSXiu Jianfeng if (unlikely(MSR_TM_TRANSACTIONAL(regs->msr)) && 1031547db7dSXiu Jianfeng !trap_is_unsupported_scv(regs)) { 1041547db7dSXiu Jianfeng /* Enable TM in the kernel, and disable EE (for scv) */ 1051547db7dSXiu Jianfeng hard_irq_disable(); 1061547db7dSXiu Jianfeng mtmsr(mfmsr() | MSR_TM); 1071547db7dSXiu Jianfeng 1081547db7dSXiu Jianfeng /* tabort, this dooms the transaction, nothing else */ 1091547db7dSXiu Jianfeng asm volatile(".long 0x7c00071d | ((%0) << 16)" 1101547db7dSXiu Jianfeng :: "r"(TM_CAUSE_SYSCALL|TM_CAUSE_PERSISTENT)); 1111547db7dSXiu Jianfeng 1121547db7dSXiu Jianfeng /* 1131547db7dSXiu Jianfeng * Userspace will never see the return value. Execution will 1141547db7dSXiu Jianfeng * resume after the tbegin. of the aborted transaction with the 1151547db7dSXiu Jianfeng * checkpointed register state. A context switch could occur 1161547db7dSXiu Jianfeng * or signal delivered to the process before resuming the 1171547db7dSXiu Jianfeng * doomed transaction context, but that should all be handled 1181547db7dSXiu Jianfeng * as expected. 1191547db7dSXiu Jianfeng */ 1201547db7dSXiu Jianfeng return -ENOSYS; 1211547db7dSXiu Jianfeng } 1221547db7dSXiu Jianfeng #endif // CONFIG_PPC_TRANSACTIONAL_MEM 1231547db7dSXiu Jianfeng 1241547db7dSXiu Jianfeng local_irq_enable(); 1251547db7dSXiu Jianfeng 1261547db7dSXiu Jianfeng if (unlikely(read_thread_flags() & _TIF_SYSCALL_DOTRACE)) { 1271547db7dSXiu Jianfeng if (unlikely(trap_is_unsupported_scv(regs))) { 1281547db7dSXiu Jianfeng /* Unsupported scv vector */ 1291547db7dSXiu Jianfeng _exception(SIGILL, regs, ILL_ILLOPC, regs->nip); 1301547db7dSXiu Jianfeng return regs->gpr[3]; 1311547db7dSXiu Jianfeng } 1321547db7dSXiu Jianfeng /* 1331547db7dSXiu Jianfeng * We use the return value of do_syscall_trace_enter() as the 1341547db7dSXiu Jianfeng * syscall number. If the syscall was rejected for any reason 1351547db7dSXiu Jianfeng * do_syscall_trace_enter() returns an invalid syscall number 1361547db7dSXiu Jianfeng * and the test against NR_syscalls will fail and the return 1371547db7dSXiu Jianfeng * value to be used is in regs->gpr[3]. 1381547db7dSXiu Jianfeng */ 1391547db7dSXiu Jianfeng r0 = do_syscall_trace_enter(regs); 1401547db7dSXiu Jianfeng if (unlikely(r0 >= NR_syscalls)) 1411547db7dSXiu Jianfeng return regs->gpr[3]; 1421547db7dSXiu Jianfeng r3 = regs->gpr[3]; 1431547db7dSXiu Jianfeng r4 = regs->gpr[4]; 1441547db7dSXiu Jianfeng r5 = regs->gpr[5]; 1451547db7dSXiu Jianfeng r6 = regs->gpr[6]; 1461547db7dSXiu Jianfeng r7 = regs->gpr[7]; 1471547db7dSXiu Jianfeng r8 = regs->gpr[8]; 1481547db7dSXiu Jianfeng 1491547db7dSXiu Jianfeng } else if (unlikely(r0 >= NR_syscalls)) { 1501547db7dSXiu Jianfeng if (unlikely(trap_is_unsupported_scv(regs))) { 1511547db7dSXiu Jianfeng /* Unsupported scv vector */ 1521547db7dSXiu Jianfeng _exception(SIGILL, regs, ILL_ILLOPC, regs->nip); 1531547db7dSXiu Jianfeng return regs->gpr[3]; 1541547db7dSXiu Jianfeng } 1551547db7dSXiu Jianfeng return -ENOSYS; 1561547db7dSXiu Jianfeng } 1571547db7dSXiu Jianfeng 1581547db7dSXiu Jianfeng /* May be faster to do array_index_nospec? */ 1591547db7dSXiu Jianfeng barrier_nospec(); 1601547db7dSXiu Jianfeng 1611547db7dSXiu Jianfeng if (unlikely(is_compat_task())) { 1621547db7dSXiu Jianfeng f = (void *)compat_sys_call_table[r0]; 1631547db7dSXiu Jianfeng 1641547db7dSXiu Jianfeng r3 &= 0x00000000ffffffffULL; 1651547db7dSXiu Jianfeng r4 &= 0x00000000ffffffffULL; 1661547db7dSXiu Jianfeng r5 &= 0x00000000ffffffffULL; 1671547db7dSXiu Jianfeng r6 &= 0x00000000ffffffffULL; 1681547db7dSXiu Jianfeng r7 &= 0x00000000ffffffffULL; 1691547db7dSXiu Jianfeng r8 &= 0x00000000ffffffffULL; 1701547db7dSXiu Jianfeng 1711547db7dSXiu Jianfeng } else { 1721547db7dSXiu Jianfeng f = (void *)sys_call_table[r0]; 1731547db7dSXiu Jianfeng } 1741547db7dSXiu Jianfeng 175*f4a0318fSXiu Jianfeng ret = f(r3, r4, r5, r6, r7, r8); 176*f4a0318fSXiu Jianfeng 177*f4a0318fSXiu Jianfeng /* 178*f4a0318fSXiu Jianfeng * Ultimately, this value will get limited by KSTACK_OFFSET_MAX(), 179*f4a0318fSXiu Jianfeng * so the maximum stack offset is 1k bytes (10 bits). 180*f4a0318fSXiu Jianfeng * 181*f4a0318fSXiu Jianfeng * The actual entropy will be further reduced by the compiler when 182*f4a0318fSXiu Jianfeng * applying stack alignment constraints: the powerpc architecture 183*f4a0318fSXiu Jianfeng * may have two kinds of stack alignment (16-bytes and 8-bytes). 184*f4a0318fSXiu Jianfeng * 185*f4a0318fSXiu Jianfeng * So the resulting 6 or 7 bits of entropy is seen in SP[9:4] or SP[9:3]. 186*f4a0318fSXiu Jianfeng */ 187*f4a0318fSXiu Jianfeng choose_random_kstack_offset(mftb()); 188*f4a0318fSXiu Jianfeng 189*f4a0318fSXiu Jianfeng return ret; 1901547db7dSXiu Jianfeng } 191