156e62a73SSven Schnelle /* SPDX-License-Identifier: GPL-2.0 */
256e62a73SSven Schnelle #ifndef ARCH_S390_ENTRY_COMMON_H
356e62a73SSven Schnelle #define ARCH_S390_ENTRY_COMMON_H
456e62a73SSven Schnelle 
556e62a73SSven Schnelle #include <linux/sched.h>
656e62a73SSven Schnelle #include <linux/audit.h>
7*bae1cd36SSven Schnelle #include <linux/randomize_kstack.h>
856e62a73SSven Schnelle #include <linux/tracehook.h>
956e62a73SSven Schnelle #include <linux/processor.h>
1056e62a73SSven Schnelle #include <linux/uaccess.h>
11*bae1cd36SSven Schnelle #include <asm/timex.h>
1256e62a73SSven Schnelle #include <asm/fpu/api.h>
1356e62a73SSven Schnelle 
1456e62a73SSven Schnelle #define ARCH_EXIT_TO_USER_MODE_WORK (_TIF_GUARDED_STORAGE | _TIF_PER_TRAP)
1556e62a73SSven Schnelle 
1656e62a73SSven Schnelle void do_per_trap(struct pt_regs *regs);
1756e62a73SSven Schnelle void do_syscall(struct pt_regs *regs);
1856e62a73SSven Schnelle 
1956e62a73SSven Schnelle #ifdef CONFIG_DEBUG_ENTRY
2056e62a73SSven Schnelle static __always_inline void arch_check_user_regs(struct pt_regs *regs)
2156e62a73SSven Schnelle {
2256e62a73SSven Schnelle 	debug_user_asce(0);
2356e62a73SSven Schnelle }
2456e62a73SSven Schnelle 
2556e62a73SSven Schnelle #define arch_check_user_regs arch_check_user_regs
2656e62a73SSven Schnelle #endif /* CONFIG_DEBUG_ENTRY */
2756e62a73SSven Schnelle 
2856e62a73SSven Schnelle static __always_inline void arch_exit_to_user_mode_work(struct pt_regs *regs,
2956e62a73SSven Schnelle 							unsigned long ti_work)
3056e62a73SSven Schnelle {
3156e62a73SSven Schnelle 	if (ti_work & _TIF_PER_TRAP) {
3256e62a73SSven Schnelle 		clear_thread_flag(TIF_PER_TRAP);
3356e62a73SSven Schnelle 		do_per_trap(regs);
3456e62a73SSven Schnelle 	}
3556e62a73SSven Schnelle 
3656e62a73SSven Schnelle 	if (ti_work & _TIF_GUARDED_STORAGE)
3756e62a73SSven Schnelle 		gs_load_bc_cb(regs);
3856e62a73SSven Schnelle }
3956e62a73SSven Schnelle 
4056e62a73SSven Schnelle #define arch_exit_to_user_mode_work arch_exit_to_user_mode_work
4156e62a73SSven Schnelle 
4256e62a73SSven Schnelle static __always_inline void arch_exit_to_user_mode(void)
4356e62a73SSven Schnelle {
4456e62a73SSven Schnelle 	if (test_cpu_flag(CIF_FPU))
4556e62a73SSven Schnelle 		__load_fpu_regs();
4656e62a73SSven Schnelle 
4756e62a73SSven Schnelle 	if (IS_ENABLED(CONFIG_DEBUG_ENTRY))
4856e62a73SSven Schnelle 		debug_user_asce(1);
4956e62a73SSven Schnelle }
5056e62a73SSven Schnelle 
5156e62a73SSven Schnelle #define arch_exit_to_user_mode arch_exit_to_user_mode
5256e62a73SSven Schnelle 
53*bae1cd36SSven Schnelle static inline void arch_exit_to_user_mode_prepare(struct pt_regs *regs,
54*bae1cd36SSven Schnelle 						  unsigned long ti_work)
55*bae1cd36SSven Schnelle {
56*bae1cd36SSven Schnelle 	choose_random_kstack_offset(get_tod_clock_fast() & 0xff);
57*bae1cd36SSven Schnelle }
58*bae1cd36SSven Schnelle 
59*bae1cd36SSven Schnelle #define arch_exit_to_user_mode_prepare arch_exit_to_user_mode_prepare
60*bae1cd36SSven Schnelle 
6156e62a73SSven Schnelle static inline bool on_thread_stack(void)
6256e62a73SSven Schnelle {
6356e62a73SSven Schnelle 	return !(((unsigned long)(current->stack) ^ current_stack_pointer()) & ~(THREAD_SIZE - 1));
6456e62a73SSven Schnelle }
6556e62a73SSven Schnelle 
6656e62a73SSven Schnelle #endif
67