1 // SPDX-License-Identifier: GPL-2.0-only
3 * Based on arch/arm/kernel/signal.c
5 * Copyright (C) 1995-2009 Russell King
6 * Copyright (C) 2012 ARM Ltd.
7 * Modified by Will Deacon <will.deacon@arm.com>
10 #include <linux/compat.h>
11 #include <linux/signal.h>
12 #include <linux/syscalls.h>
13 #include <linux/ratelimit.h>
16 #include <asm/fpsimd.h>
17 #include <asm/signal32.h>
18 #include <asm/traps.h>
19 #include <linux/uaccess.h>
20 #include <asm/unistd_compat_32.h>
23 struct compat_vfp_sigframe
{
26 struct compat_user_vfp
{
27 compat_u64 fpregs
[32];
30 struct compat_user_vfp_exc
{
32 compat_ulong_t fpinst
;
33 compat_ulong_t fpinst2
;
35 } __attribute__((__aligned__(8)));
37 #define VFP_MAGIC 0x56465001
38 #define VFP_STORAGE_SIZE sizeof(struct compat_vfp_sigframe)
40 #define FSR_WRITE_SHIFT (11)
42 struct compat_aux_sigframe
{
43 struct compat_vfp_sigframe vfp
;
45 /* Something that isn't a valid magic number for any coprocessor. */
46 unsigned long end_magic
;
47 } __attribute__((__aligned__(8)));
49 static inline int put_sigset_t(compat_sigset_t __user
*uset
, sigset_t
*set
)
53 cset
.sig
[0] = set
->sig
[0] & 0xffffffffull
;
54 cset
.sig
[1] = set
->sig
[0] >> 32;
56 return copy_to_user(uset
, &cset
, sizeof(*uset
));
59 static inline int get_sigset_t(sigset_t
*set
,
60 const compat_sigset_t __user
*uset
)
64 if (copy_from_user(&s32
, uset
, sizeof(*uset
)))
67 set
->sig
[0] = s32
.sig
[0] | (((long)s32
.sig
[1]) << 32);
72 * VFP save/restore code.
74 * We have to be careful with endianness, since the fpsimd context-switch
75 * code operates on 128-bit (Q) register values whereas the compat ABI
76 * uses an array of 64-bit (D) registers. Consequently, we need to swap
77 * the two halves of each Q register when running on a big-endian CPU.
92 static int compat_preserve_vfp_context(struct compat_vfp_sigframe __user
*frame
)
94 struct user_fpsimd_state
const *fpsimd
=
95 ¤t
->thread
.uw
.fpsimd_state
;
96 compat_ulong_t magic
= VFP_MAGIC
;
97 compat_ulong_t size
= VFP_STORAGE_SIZE
;
98 compat_ulong_t fpscr
, fpexc
;
102 * Save the hardware registers to the fpsimd_state structure.
103 * Note that this also saves V16-31, which aren't visible
106 fpsimd_signal_preserve_current_state();
108 /* Place structure header on the stack */
109 __put_user_error(magic
, &frame
->magic
, err
);
110 __put_user_error(size
, &frame
->size
, err
);
113 * Now copy the FP registers. Since the registers are packed,
114 * we can copy the prefix we want (V0-V15) as it is.
116 for (i
= 0; i
< ARRAY_SIZE(frame
->ufp
.fpregs
); i
+= 2) {
117 union __fpsimd_vreg vreg
= {
118 .raw
= fpsimd
->vregs
[i
>> 1],
121 __put_user_error(vreg
.lo
, &frame
->ufp
.fpregs
[i
], err
);
122 __put_user_error(vreg
.hi
, &frame
->ufp
.fpregs
[i
+ 1], err
);
125 /* Create an AArch32 fpscr from the fpsr and the fpcr. */
126 fpscr
= (fpsimd
->fpsr
& VFP_FPSCR_STAT_MASK
) |
127 (fpsimd
->fpcr
& VFP_FPSCR_CTRL_MASK
);
128 __put_user_error(fpscr
, &frame
->ufp
.fpscr
, err
);
131 * The exception register aren't available so we fake up a
132 * basic FPEXC and zero everything else.
135 __put_user_error(fpexc
, &frame
->ufp_exc
.fpexc
, err
);
136 __put_user_error(0, &frame
->ufp_exc
.fpinst
, err
);
137 __put_user_error(0, &frame
->ufp_exc
.fpinst2
, err
);
139 return err
? -EFAULT
: 0;
142 static int compat_restore_vfp_context(struct compat_vfp_sigframe __user
*frame
)
144 struct user_fpsimd_state fpsimd
;
145 compat_ulong_t magic
= VFP_MAGIC
;
146 compat_ulong_t size
= VFP_STORAGE_SIZE
;
147 compat_ulong_t fpscr
;
150 __get_user_error(magic
, &frame
->magic
, err
);
151 __get_user_error(size
, &frame
->size
, err
);
155 if (magic
!= VFP_MAGIC
|| size
!= VFP_STORAGE_SIZE
)
158 /* Copy the FP registers into the start of the fpsimd_state. */
159 for (i
= 0; i
< ARRAY_SIZE(frame
->ufp
.fpregs
); i
+= 2) {
160 union __fpsimd_vreg vreg
;
162 __get_user_error(vreg
.lo
, &frame
->ufp
.fpregs
[i
], err
);
163 __get_user_error(vreg
.hi
, &frame
->ufp
.fpregs
[i
+ 1], err
);
164 fpsimd
.vregs
[i
>> 1] = vreg
.raw
;
167 /* Extract the fpsr and the fpcr from the fpscr */
168 __get_user_error(fpscr
, &frame
->ufp
.fpscr
, err
);
169 fpsimd
.fpsr
= fpscr
& VFP_FPSCR_STAT_MASK
;
170 fpsimd
.fpcr
= fpscr
& VFP_FPSCR_CTRL_MASK
;
173 * We don't need to touch the exception register, so
174 * reload the hardware state.
177 fpsimd_update_current_state(&fpsimd
);
179 return err
? -EFAULT
: 0;
182 static int compat_restore_sigframe(struct pt_regs
*regs
,
183 struct compat_sigframe __user
*sf
)
187 struct compat_aux_sigframe __user
*aux
;
190 err
= get_sigset_t(&set
, &sf
->uc
.uc_sigmask
);
192 set_current_blocked(&set
);
194 __get_user_error(regs
->regs
[0], &sf
->uc
.uc_mcontext
.arm_r0
, err
);
195 __get_user_error(regs
->regs
[1], &sf
->uc
.uc_mcontext
.arm_r1
, err
);
196 __get_user_error(regs
->regs
[2], &sf
->uc
.uc_mcontext
.arm_r2
, err
);
197 __get_user_error(regs
->regs
[3], &sf
->uc
.uc_mcontext
.arm_r3
, err
);
198 __get_user_error(regs
->regs
[4], &sf
->uc
.uc_mcontext
.arm_r4
, err
);
199 __get_user_error(regs
->regs
[5], &sf
->uc
.uc_mcontext
.arm_r5
, err
);
200 __get_user_error(regs
->regs
[6], &sf
->uc
.uc_mcontext
.arm_r6
, err
);
201 __get_user_error(regs
->regs
[7], &sf
->uc
.uc_mcontext
.arm_r7
, err
);
202 __get_user_error(regs
->regs
[8], &sf
->uc
.uc_mcontext
.arm_r8
, err
);
203 __get_user_error(regs
->regs
[9], &sf
->uc
.uc_mcontext
.arm_r9
, err
);
204 __get_user_error(regs
->regs
[10], &sf
->uc
.uc_mcontext
.arm_r10
, err
);
205 __get_user_error(regs
->regs
[11], &sf
->uc
.uc_mcontext
.arm_fp
, err
);
206 __get_user_error(regs
->regs
[12], &sf
->uc
.uc_mcontext
.arm_ip
, err
);
207 __get_user_error(regs
->compat_sp
, &sf
->uc
.uc_mcontext
.arm_sp
, err
);
208 __get_user_error(regs
->compat_lr
, &sf
->uc
.uc_mcontext
.arm_lr
, err
);
209 __get_user_error(regs
->pc
, &sf
->uc
.uc_mcontext
.arm_pc
, err
);
210 __get_user_error(psr
, &sf
->uc
.uc_mcontext
.arm_cpsr
, err
);
212 regs
->pstate
= compat_psr_to_pstate(psr
);
215 * Avoid compat_sys_sigreturn() restarting.
217 forget_syscall(regs
);
219 err
|= !valid_user_regs(®s
->user_regs
, current
);
221 aux
= (struct compat_aux_sigframe __user
*) sf
->uc
.uc_regspace
;
222 if (err
== 0 && system_supports_fpsimd())
223 err
|= compat_restore_vfp_context(&aux
->vfp
);
228 COMPAT_SYSCALL_DEFINE0(sigreturn
)
230 struct pt_regs
*regs
= current_pt_regs();
231 struct compat_sigframe __user
*frame
;
233 /* Always make any pending restarted system calls return -EINTR */
234 current
->restart_block
.fn
= do_no_restart_syscall
;
237 * Since we stacked the signal on a 64-bit boundary,
238 * then 'sp' should be word aligned here. If it's
239 * not, then the user is trying to mess with us.
241 if (regs
->compat_sp
& 7)
244 frame
= (struct compat_sigframe __user
*)regs
->compat_sp
;
246 if (!access_ok(frame
, sizeof (*frame
)))
249 if (compat_restore_sigframe(regs
, frame
))
252 return regs
->regs
[0];
255 arm64_notify_segfault(regs
->compat_sp
);
259 COMPAT_SYSCALL_DEFINE0(rt_sigreturn
)
261 struct pt_regs
*regs
= current_pt_regs();
262 struct compat_rt_sigframe __user
*frame
;
264 /* Always make any pending restarted system calls return -EINTR */
265 current
->restart_block
.fn
= do_no_restart_syscall
;
268 * Since we stacked the signal on a 64-bit boundary,
269 * then 'sp' should be word aligned here. If it's
270 * not, then the user is trying to mess with us.
272 if (regs
->compat_sp
& 7)
275 frame
= (struct compat_rt_sigframe __user
*)regs
->compat_sp
;
277 if (!access_ok(frame
, sizeof (*frame
)))
280 if (compat_restore_sigframe(regs
, &frame
->sig
))
283 if (compat_restore_altstack(&frame
->sig
.uc
.uc_stack
))
286 return regs
->regs
[0];
289 arm64_notify_segfault(regs
->compat_sp
);
293 static void __user
*compat_get_sigframe(struct ksignal
*ksig
,
294 struct pt_regs
*regs
,
297 compat_ulong_t sp
= sigsp(regs
->compat_sp
, ksig
);
301 * ATPCS B01 mandates 8-byte alignment
303 frame
= compat_ptr((compat_uptr_t
)((sp
- framesize
) & ~7));
306 * Check that we can actually write to the signal frame.
308 if (!access_ok(frame
, framesize
))
314 static void compat_setup_return(struct pt_regs
*regs
, struct k_sigaction
*ka
,
315 compat_ulong_t __user
*rc
, void __user
*frame
,
318 compat_ulong_t handler
= ptr_to_compat(ka
->sa
.sa_handler
);
319 compat_ulong_t retcode
;
320 compat_ulong_t spsr
= regs
->pstate
& ~(PSR_f
| PSR_AA32_E_BIT
);
323 /* Check if the handler is written for ARM or Thumb */
327 spsr
|= PSR_AA32_T_BIT
;
329 spsr
&= ~PSR_AA32_T_BIT
;
331 /* The IT state must be cleared for both ARM and Thumb-2 */
332 spsr
&= ~PSR_AA32_IT_MASK
;
334 /* Restore the original endianness */
335 spsr
|= PSR_AA32_ENDSTATE
;
337 if (ka
->sa
.sa_flags
& SA_RESTORER
) {
338 retcode
= ptr_to_compat(ka
->sa
.sa_restorer
);
340 /* Set up sigreturn pointer */
341 unsigned int idx
= thumb
<< 1;
343 if (ka
->sa
.sa_flags
& SA_SIGINFO
)
346 retcode
= (unsigned long)current
->mm
->context
.sigpage
+
350 regs
->regs
[0] = usig
;
351 regs
->compat_sp
= ptr_to_compat(frame
);
352 regs
->compat_lr
= retcode
;
357 static int compat_setup_sigframe(struct compat_sigframe __user
*sf
,
358 struct pt_regs
*regs
, sigset_t
*set
)
360 struct compat_aux_sigframe __user
*aux
;
361 unsigned long psr
= pstate_to_compat_psr(regs
->pstate
);
364 __put_user_error(regs
->regs
[0], &sf
->uc
.uc_mcontext
.arm_r0
, err
);
365 __put_user_error(regs
->regs
[1], &sf
->uc
.uc_mcontext
.arm_r1
, err
);
366 __put_user_error(regs
->regs
[2], &sf
->uc
.uc_mcontext
.arm_r2
, err
);
367 __put_user_error(regs
->regs
[3], &sf
->uc
.uc_mcontext
.arm_r3
, err
);
368 __put_user_error(regs
->regs
[4], &sf
->uc
.uc_mcontext
.arm_r4
, err
);
369 __put_user_error(regs
->regs
[5], &sf
->uc
.uc_mcontext
.arm_r5
, err
);
370 __put_user_error(regs
->regs
[6], &sf
->uc
.uc_mcontext
.arm_r6
, err
);
371 __put_user_error(regs
->regs
[7], &sf
->uc
.uc_mcontext
.arm_r7
, err
);
372 __put_user_error(regs
->regs
[8], &sf
->uc
.uc_mcontext
.arm_r8
, err
);
373 __put_user_error(regs
->regs
[9], &sf
->uc
.uc_mcontext
.arm_r9
, err
);
374 __put_user_error(regs
->regs
[10], &sf
->uc
.uc_mcontext
.arm_r10
, err
);
375 __put_user_error(regs
->regs
[11], &sf
->uc
.uc_mcontext
.arm_fp
, err
);
376 __put_user_error(regs
->regs
[12], &sf
->uc
.uc_mcontext
.arm_ip
, err
);
377 __put_user_error(regs
->compat_sp
, &sf
->uc
.uc_mcontext
.arm_sp
, err
);
378 __put_user_error(regs
->compat_lr
, &sf
->uc
.uc_mcontext
.arm_lr
, err
);
379 __put_user_error(regs
->pc
, &sf
->uc
.uc_mcontext
.arm_pc
, err
);
380 __put_user_error(psr
, &sf
->uc
.uc_mcontext
.arm_cpsr
, err
);
382 __put_user_error((compat_ulong_t
)0, &sf
->uc
.uc_mcontext
.trap_no
, err
);
383 /* set the compat FSR WnR */
384 __put_user_error(!!(current
->thread
.fault_code
& ESR_ELx_WNR
) <<
385 FSR_WRITE_SHIFT
, &sf
->uc
.uc_mcontext
.error_code
, err
);
386 __put_user_error(current
->thread
.fault_address
, &sf
->uc
.uc_mcontext
.fault_address
, err
);
387 __put_user_error(set
->sig
[0], &sf
->uc
.uc_mcontext
.oldmask
, err
);
389 err
|= put_sigset_t(&sf
->uc
.uc_sigmask
, set
);
391 aux
= (struct compat_aux_sigframe __user
*) sf
->uc
.uc_regspace
;
393 if (err
== 0 && system_supports_fpsimd())
394 err
|= compat_preserve_vfp_context(&aux
->vfp
);
395 __put_user_error(0, &aux
->end_magic
, err
);
401 * 32-bit signal handling routines called from signal.c
403 int compat_setup_rt_frame(int usig
, struct ksignal
*ksig
,
404 sigset_t
*set
, struct pt_regs
*regs
)
406 struct compat_rt_sigframe __user
*frame
;
409 frame
= compat_get_sigframe(ksig
, regs
, sizeof(*frame
));
414 err
|= copy_siginfo_to_user32(&frame
->info
, &ksig
->info
);
416 __put_user_error(0, &frame
->sig
.uc
.uc_flags
, err
);
417 __put_user_error(0, &frame
->sig
.uc
.uc_link
, err
);
419 err
|= __compat_save_altstack(&frame
->sig
.uc
.uc_stack
, regs
->compat_sp
);
421 err
|= compat_setup_sigframe(&frame
->sig
, regs
, set
);
424 compat_setup_return(regs
, &ksig
->ka
, frame
->sig
.retcode
, frame
, usig
);
425 regs
->regs
[1] = (compat_ulong_t
)(unsigned long)&frame
->info
;
426 regs
->regs
[2] = (compat_ulong_t
)(unsigned long)&frame
->sig
.uc
;
432 int compat_setup_frame(int usig
, struct ksignal
*ksig
, sigset_t
*set
,
433 struct pt_regs
*regs
)
435 struct compat_sigframe __user
*frame
;
438 frame
= compat_get_sigframe(ksig
, regs
, sizeof(*frame
));
443 __put_user_error(0x5ac3c35a, &frame
->uc
.uc_flags
, err
);
445 err
|= compat_setup_sigframe(frame
, regs
, set
);
447 compat_setup_return(regs
, &ksig
->ka
, frame
->retcode
, frame
, usig
);
452 void compat_setup_restart_syscall(struct pt_regs
*regs
)
454 regs
->regs
[7] = __NR_compat32_restart_syscall
;
458 * Compile-time assertions for siginfo_t offsets. Check NSIG* as well, as
459 * changes likely come with new fields that should be added below.
461 static_assert(NSIGILL
== 11);
462 static_assert(NSIGFPE
== 15);
463 static_assert(NSIGSEGV
== 10);
464 static_assert(NSIGBUS
== 5);
465 static_assert(NSIGTRAP
== 6);
466 static_assert(NSIGCHLD
== 6);
467 static_assert(NSIGSYS
== 2);
468 static_assert(sizeof(compat_siginfo_t
) == 128);
469 static_assert(__alignof__(compat_siginfo_t
) == 4);
470 static_assert(offsetof(compat_siginfo_t
, si_signo
) == 0x00);
471 static_assert(offsetof(compat_siginfo_t
, si_errno
) == 0x04);
472 static_assert(offsetof(compat_siginfo_t
, si_code
) == 0x08);
473 static_assert(offsetof(compat_siginfo_t
, si_pid
) == 0x0c);
474 static_assert(offsetof(compat_siginfo_t
, si_uid
) == 0x10);
475 static_assert(offsetof(compat_siginfo_t
, si_tid
) == 0x0c);
476 static_assert(offsetof(compat_siginfo_t
, si_overrun
) == 0x10);
477 static_assert(offsetof(compat_siginfo_t
, si_status
) == 0x14);
478 static_assert(offsetof(compat_siginfo_t
, si_utime
) == 0x18);
479 static_assert(offsetof(compat_siginfo_t
, si_stime
) == 0x1c);
480 static_assert(offsetof(compat_siginfo_t
, si_value
) == 0x14);
481 static_assert(offsetof(compat_siginfo_t
, si_int
) == 0x14);
482 static_assert(offsetof(compat_siginfo_t
, si_ptr
) == 0x14);
483 static_assert(offsetof(compat_siginfo_t
, si_addr
) == 0x0c);
484 static_assert(offsetof(compat_siginfo_t
, si_addr_lsb
) == 0x10);
485 static_assert(offsetof(compat_siginfo_t
, si_lower
) == 0x14);
486 static_assert(offsetof(compat_siginfo_t
, si_upper
) == 0x18);
487 static_assert(offsetof(compat_siginfo_t
, si_pkey
) == 0x14);
488 static_assert(offsetof(compat_siginfo_t
, si_perf_data
) == 0x10);
489 static_assert(offsetof(compat_siginfo_t
, si_perf_type
) == 0x14);
490 static_assert(offsetof(compat_siginfo_t
, si_perf_flags
) == 0x18);
491 static_assert(offsetof(compat_siginfo_t
, si_band
) == 0x0c);
492 static_assert(offsetof(compat_siginfo_t
, si_fd
) == 0x10);
493 static_assert(offsetof(compat_siginfo_t
, si_call_addr
) == 0x0c);
494 static_assert(offsetof(compat_siginfo_t
, si_syscall
) == 0x10);
495 static_assert(offsetof(compat_siginfo_t
, si_arch
) == 0x14);