1 /* SPDX-License-Identifier: GPL-2.0-only */
2 /*
3 * Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com)
4 */
5
6 #ifndef _ASM_ARC_SYSCALL_H
7 #define _ASM_ARC_SYSCALL_H 1
8
9 #include <uapi/linux/audit.h>
10 #include <linux/err.h>
11 #include <linux/sched.h>
12 #include <asm/unistd.h>
13 #include <asm/ptrace.h> /* in_syscall() */
14
15 static inline long
syscall_get_nr(struct task_struct * task,struct pt_regs * regs)16 syscall_get_nr(struct task_struct *task, struct pt_regs *regs)
17 {
18 if (user_mode(regs) && in_syscall(regs))
19 return regs->r8;
20 else
21 return -1;
22 }
23
24 static inline void
syscall_rollback(struct task_struct * task,struct pt_regs * regs)25 syscall_rollback(struct task_struct *task, struct pt_regs *regs)
26 {
27 regs->r0 = regs->orig_r0;
28 }
29
30 static inline long
syscall_get_error(struct task_struct * task,struct pt_regs * regs)31 syscall_get_error(struct task_struct *task, struct pt_regs *regs)
32 {
33 /* 0 if syscall succeeded, otherwise -Errorcode */
34 return IS_ERR_VALUE(regs->r0) ? regs->r0 : 0;
35 }
36
37 static inline long
syscall_get_return_value(struct task_struct * task,struct pt_regs * regs)38 syscall_get_return_value(struct task_struct *task, struct pt_regs *regs)
39 {
40 return regs->r0;
41 }
42
43 static inline void
syscall_set_return_value(struct task_struct * task,struct pt_regs * regs,int error,long val)44 syscall_set_return_value(struct task_struct *task, struct pt_regs *regs,
45 int error, long val)
46 {
47 regs->r0 = (long) error ?: val;
48 }
49
50 /*
51 * @i: argument index [0,5]
52 * @n: number of arguments; n+i must be [1,6].
53 */
54 static inline void
syscall_get_arguments(struct task_struct * task,struct pt_regs * regs,unsigned long * args)55 syscall_get_arguments(struct task_struct *task, struct pt_regs *regs,
56 unsigned long *args)
57 {
58 unsigned long *inside_ptregs = &(regs->r0);
59 unsigned int n = 6;
60 unsigned int i = 0;
61
62 while (n--) {
63 args[i++] = (*inside_ptregs);
64 inside_ptregs--;
65 }
66 }
67
68 static inline int
syscall_get_arch(struct task_struct * task)69 syscall_get_arch(struct task_struct *task)
70 {
71 return IS_ENABLED(CONFIG_ISA_ARCOMPACT)
72 ? (IS_ENABLED(CONFIG_CPU_BIG_ENDIAN)
73 ? AUDIT_ARCH_ARCOMPACTBE : AUDIT_ARCH_ARCOMPACT)
74 : (IS_ENABLED(CONFIG_CPU_BIG_ENDIAN)
75 ? AUDIT_ARCH_ARCV2BE : AUDIT_ARCH_ARCV2);
76 }
77
78 #endif
79