xref: /openbmc/linux/arch/powerpc/kernel/syscall.c (revision f4a0318f278d98d9492916722e85f258c2221f88)
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, &current_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