blob: 81abcf6a737b244467db7763ac005421b5503b81 [file] [log] [blame]
Thomas Gleixner2522fe42019-05-28 09:57:20 -07001/* SPDX-License-Identifier: GPL-2.0-only */
Roland McGrathf1ba1282008-07-27 16:51:35 +10002/*
3 * Access to user system call parameters and results
4 *
5 * Copyright (C) 2008 Red Hat, Inc. All rights reserved.
6 *
Roland McGrathf1ba1282008-07-27 16:51:35 +10007 * See asm-generic/syscall.h for descriptions of what we must do here.
8 */
9
10#ifndef _ASM_SYSCALL_H
11#define _ASM_SYSCALL_H 1
12
Eric Parisce5d1122014-03-11 13:50:46 -040013#include <uapi/linux/audit.h>
Roland McGrathf1ba1282008-07-27 16:51:35 +100014#include <linux/sched.h>
Eric Paris75dddcb2014-04-22 12:07:30 -040015#include <linux/thread_info.h>
Roland McGrathf1ba1282008-07-27 16:51:35 +100016
Ian Munsie02424d82011-02-02 17:27:24 +000017/* ftrace syscalls requires exporting the sys_call_table */
Romeo Cane1028ccf2014-10-02 15:41:39 +010018extern const unsigned long sys_call_table[];
Firoz Khanfbf508d2018-12-17 16:10:35 +053019extern const unsigned long compat_sys_call_table[];
Ian Munsie02424d82011-02-02 17:27:24 +000020
Michael Ellermane9fbe682015-07-23 20:21:07 +100021static inline int syscall_get_nr(struct task_struct *task, struct pt_regs *regs)
Roland McGrathf1ba1282008-07-27 16:51:35 +100022{
Michael Ellermane9fbe682015-07-23 20:21:07 +100023 /*
24 * Note that we are returning an int here. That means 0xffffffff, ie.
25 * 32-bit negative 1, will be interpreted as -1 on a 64-bit kernel.
26 * This is important for seccomp so that compat tasks can set r0 = -1
27 * to reject the syscall.
28 */
29 return TRAP(regs) == 0xc00 ? regs->gpr[0] : -1;
Roland McGrathf1ba1282008-07-27 16:51:35 +100030}
31
32static inline void syscall_rollback(struct task_struct *task,
33 struct pt_regs *regs)
34{
35 regs->gpr[3] = regs->orig_gpr3;
36}
37
Roland McGrathf1ba1282008-07-27 16:51:35 +100038static inline long syscall_get_return_value(struct task_struct *task,
39 struct pt_regs *regs)
40{
41 return regs->gpr[3];
42}
43
44static inline void syscall_set_return_value(struct task_struct *task,
45 struct pt_regs *regs,
46 int error, long val)
47{
Michael Ellerman1b1a3702015-07-23 20:21:04 +100048 /*
49 * In the general case it's not obvious that we must deal with CCR
50 * here, as the syscall exit path will also do that for us. However
51 * there are some places, eg. the signal code, which check ccr to
52 * decide if the value in r3 is actually an error.
53 */
Roland McGrathf1ba1282008-07-27 16:51:35 +100054 if (error) {
Nathan Lynch409d2412010-03-12 13:16:02 +000055 regs->ccr |= 0x10000000L;
Michael Ellerman1b1a3702015-07-23 20:21:04 +100056 regs->gpr[3] = error;
Roland McGrathf1ba1282008-07-27 16:51:35 +100057 } else {
Nathan Lynch409d2412010-03-12 13:16:02 +000058 regs->ccr &= ~0x10000000L;
Roland McGrathf1ba1282008-07-27 16:51:35 +100059 regs->gpr[3] = val;
60 }
61}
62
63static inline void syscall_get_arguments(struct task_struct *task,
64 struct pt_regs *regs,
Roland McGrathf1ba1282008-07-27 16:51:35 +100065 unsigned long *args)
66{
Michael Ellerman1cb98392015-07-23 20:21:06 +100067 unsigned long val, mask = -1UL;
Steven Rostedt (Red Hat)b35f5492016-11-07 16:26:37 -050068 unsigned int n = 6;
Michael Ellermana7657842015-07-23 20:21:05 +100069
70#ifdef CONFIG_COMPAT
71 if (test_tsk_thread_flag(task, TIF_32BIT))
72 mask = 0xffffffff;
Roland McGrathf1ba1282008-07-27 16:51:35 +100073#endif
Michael Ellerman1cb98392015-07-23 20:21:06 +100074 while (n--) {
Steven Rostedt (Red Hat)b35f5492016-11-07 16:26:37 -050075 if (n == 0)
Michael Ellerman1cb98392015-07-23 20:21:06 +100076 val = regs->orig_gpr3;
77 else
Steven Rostedt (Red Hat)b35f5492016-11-07 16:26:37 -050078 val = regs->gpr[3 + n];
Michael Ellerman1cb98392015-07-23 20:21:06 +100079
80 args[n] = val & mask;
81 }
Roland McGrathf1ba1282008-07-27 16:51:35 +100082}
83
84static inline void syscall_set_arguments(struct task_struct *task,
85 struct pt_regs *regs,
Roland McGrathf1ba1282008-07-27 16:51:35 +100086 const unsigned long *args)
87{
Steven Rostedt (VMware)32d92582019-03-27 20:07:31 -040088 memcpy(&regs->gpr[3], args, 6 * sizeof(args[0]));
Michael Ellerman1cb98392015-07-23 20:21:06 +100089
90 /* Also copy the first argument into orig_gpr3 */
Steven Rostedt (VMware)32d92582019-03-27 20:07:31 -040091 regs->orig_gpr3 = args[0];
Roland McGrathf1ba1282008-07-27 16:51:35 +100092}
93
Dmitry V. Levin16add412019-03-18 02:30:18 +030094static inline int syscall_get_arch(struct task_struct *task)
Eric Parisce5d1122014-03-11 13:50:46 -040095{
Dmitry V. Levin16add412019-03-18 02:30:18 +030096 int arch;
97
98 if (IS_ENABLED(CONFIG_PPC64) && !test_tsk_thread_flag(task, TIF_32BIT))
99 arch = AUDIT_ARCH_PPC64;
100 else
101 arch = AUDIT_ARCH_PPC;
102
Richard Guy Briggs63f13442014-12-09 15:37:07 -0500103#ifdef __LITTLE_ENDIAN__
104 arch |= __AUDIT_ARCH_LE;
105#endif
106 return arch;
Eric Parisce5d1122014-03-11 13:50:46 -0400107}
Roland McGrathf1ba1282008-07-27 16:51:35 +1000108#endif /* _ASM_SYSCALL_H */