]> git.proxmox.com Git - mirror_ubuntu-hirsute-kernel.git/commitdiff
signal: Unify and correct copy_siginfo_from_user32
authorEric W. Biederman <ebiederm@xmission.com>
Mon, 31 Jul 2017 22:15:31 +0000 (17:15 -0500)
committerEric W. Biederman <ebiederm@xmission.com>
Mon, 15 Jan 2018 23:55:59 +0000 (17:55 -0600)
The function copy_siginfo_from_user32 is used for two things, in ptrace
since the dawn of siginfo for arbirarily modifying a signal that
user space sees, and in sigqueueinfo to send a signal with arbirary
siginfo data.

Create a single copy of copy_siginfo_from_user32 that all architectures
share, and teach it to handle all of the cases in the siginfo union.

In the generic version of copy_siginfo_from_user32 ensure that all
of the fields in siginfo are initialized so that the siginfo structure
can be safely copied to userspace if necessary.

When copying the embedded sigval union copy the si_int member.  That
ensures the 32bit values passes through the kernel unchanged.

Signed-off-by: "Eric W. Biederman" <ebiederm@xmission.com>
arch/arm64/kernel/signal32.c
arch/mips/kernel/signal32.c
arch/parisc/kernel/signal32.c
arch/parisc/kernel/signal32.h
arch/powerpc/kernel/signal_32.c
arch/s390/kernel/compat_signal.c
arch/sparc/kernel/signal32.c
arch/tile/kernel/compat_signal.c
arch/x86/kernel/signal_compat.c
include/linux/compat.h
kernel/signal.c

index 22711ee8e36c566eef59319a457e3d61e0f9d39d..4377907dbb70868f51857aef19ecf971e32a80cc 100644 (file)
@@ -195,16 +195,6 @@ int copy_siginfo_to_user32(compat_siginfo_t __user *to, const siginfo_t *from)
        return err;
 }
 
-int copy_siginfo_from_user32(siginfo_t *to, compat_siginfo_t __user *from)
-{
-       if (copy_from_user(to, from, __ARCH_SI_PREAMBLE_SIZE) ||
-           copy_from_user(to->_sifields._pad,
-                          from->_sifields._pad, SI_PAD_SIZE))
-               return -EFAULT;
-
-       return 0;
-}
-
 /*
  * VFP save/restore code.
  *
index cf5c7c05e5a38798ecb237fac321983809b8476b..500b5e4634ea2ae3a5bc4bd9f9ead824a2880446 100644 (file)
@@ -133,13 +133,3 @@ int copy_siginfo_to_user32(compat_siginfo_t __user *to, const siginfo_t *from)
        }
        return err;
 }
-
-int copy_siginfo_from_user32(siginfo_t *to, compat_siginfo_t __user *from)
-{
-       if (copy_from_user(to, from, 3*sizeof(int)) ||
-           copy_from_user(to->_sifields._pad,
-                          from->_sifields._pad, SI_PAD_SIZE32))
-               return -EFAULT;
-
-       return 0;
-}
index 41afa9cd1f55c9dc7d0d67434c218277cdbebbb8..558e32475c357734c3b1f0cf55179912757584d2 100644 (file)
@@ -261,50 +261,6 @@ setup_sigcontext32(struct compat_sigcontext __user *sc, struct compat_regfile __
        return err;
 }
 
-int
-copy_siginfo_from_user32 (siginfo_t *to, compat_siginfo_t __user *from)
-{
-       compat_uptr_t addr;
-       int err;
-
-       if (!access_ok(VERIFY_READ, from, sizeof(compat_siginfo_t)))
-               return -EFAULT;
-
-       err = __get_user(to->si_signo, &from->si_signo);
-       err |= __get_user(to->si_errno, &from->si_errno);
-       err |= __get_user(to->si_code, &from->si_code);
-
-       if (to->si_code < 0)
-               err |= __copy_from_user(&to->_sifields._pad, &from->_sifields._pad, SI_PAD_SIZE);
-       else {
-               switch (siginfo_layout(to->si_signo, to->si_code)) {
-                     case SIL_CHLD:
-                       err |= __get_user(to->si_utime, &from->si_utime);
-                       err |= __get_user(to->si_stime, &from->si_stime);
-                       err |= __get_user(to->si_status, &from->si_status);
-                     default:
-                     case SIL_KILL:
-                       err |= __get_user(to->si_pid, &from->si_pid);
-                       err |= __get_user(to->si_uid, &from->si_uid);
-                       break;
-                     case SIL_FAULT:
-                       err |= __get_user(addr, &from->si_addr);
-                       to->si_addr = compat_ptr(addr);
-                       break;
-                     case SIL_POLL:
-                       err |= __get_user(to->si_band, &from->si_band);
-                       err |= __get_user(to->si_fd, &from->si_fd);
-                       break;
-                     case SIL_RT:
-                       err |= __get_user(to->si_pid, &from->si_pid);
-                       err |= __get_user(to->si_uid, &from->si_uid);
-                       err |= __get_user(to->si_int, &from->si_int);
-                       break;
-               }
-       }
-       return err;
-}
-
 int
 copy_siginfo_to_user32 (compat_siginfo_t __user *to, const siginfo_t *from)
 {
index 719e7417732cc6f420111df3ebe7e2a4da606b69..d25858e4db630dab407e3a2c76eceaa45bb1b6f9 100644 (file)
@@ -35,7 +35,6 @@ struct compat_ucontext {
 /* ELF32 signal handling */
 
 int copy_siginfo_to_user32 (compat_siginfo_t __user *to, const siginfo_t *from);
-int copy_siginfo_from_user32 (siginfo_t *to, compat_siginfo_t __user *from);
 
 /* In a deft move of uber-hackery, we decide to carry the top half of all
  * 64-bit registers in a non-portable, non-ABI, hidden structure.
index 9ffd73296f649490cc5c31c3b14b789244a5b337..ee62ff7b296cdf29a51d837b57e9ad378ba4a4d1 100644 (file)
@@ -933,15 +933,6 @@ int copy_siginfo_to_user32(struct compat_siginfo __user *d, const siginfo_t *s)
 
 #define copy_siginfo_to_user   copy_siginfo_to_user32
 
-int copy_siginfo_from_user32(siginfo_t *to, struct compat_siginfo __user *from)
-{
-       if (copy_from_user(to, from, 3*sizeof(int)) ||
-           copy_from_user(to->_sifields._pad,
-                          from->_sifields._pad, SI_PAD_SIZE32))
-               return -EFAULT;
-
-       return 0;
-}
 #endif /* CONFIG_PPC64 */
 
 /*
index ef246940b44c514f7c3c101b6b65f20cf20be7df..d77ce14ffa5c8403310ec471b181fece7ff577dc 100644 (file)
@@ -102,54 +102,6 @@ int copy_siginfo_to_user32(compat_siginfo_t __user *to, const siginfo_t *from)
        return err ? -EFAULT : 0;
 }
 
-int copy_siginfo_from_user32(siginfo_t *to, compat_siginfo_t __user *from)
-{
-       int err;
-       u32 tmp;
-
-       err = __get_user(to->si_signo, &from->si_signo);
-       err |= __get_user(to->si_errno, &from->si_errno);
-       err |= __get_user(to->si_code, &from->si_code);
-
-       if (to->si_code < 0)
-               err |= __copy_from_user(&to->_sifields._pad, &from->_sifields._pad, SI_PAD_SIZE);
-       else {
-               switch (siginfo_layout(to->si_signo, to->si_code)) {
-               case SIL_RT:
-                       err |= __get_user(to->si_int, &from->si_int);
-                       /* fallthrough */
-               case SIL_KILL:
-                       err |= __get_user(to->si_pid, &from->si_pid);
-                       err |= __get_user(to->si_uid, &from->si_uid);
-                       break;
-               case SIL_CHLD:
-                       err |= __get_user(to->si_pid, &from->si_pid);
-                       err |= __get_user(to->si_uid, &from->si_uid);
-                       err |= __get_user(to->si_utime, &from->si_utime);
-                       err |= __get_user(to->si_stime, &from->si_stime);
-                       err |= __get_user(to->si_status, &from->si_status);
-                       break;
-               case SIL_FAULT:
-                       err |= __get_user(tmp, &from->si_addr);
-                       to->si_addr = (void __force __user *)
-                               (u64) (tmp & PSW32_ADDR_INSN);
-                       break;
-               case SIL_POLL:
-                       err |= __get_user(to->si_band, &from->si_band);
-                       err |= __get_user(to->si_fd, &from->si_fd);
-                       break;
-               case SIL_TIMER:
-                       err |= __get_user(to->si_tid, &from->si_tid);
-                       err |= __get_user(to->si_overrun, &from->si_overrun);
-                       err |= __get_user(to->si_int, &from->si_int);
-                       break;
-               default:
-                       break;
-               }
-       }
-       return err ? -EFAULT : 0;
-}
-
 /* Store registers needed to create the signal frame */
 static void store_sigregs(void)
 {
index 54a6159b9cd8f1aaaacce11759b99f4778da3cdc..8022bb4c65a55fc365840d6e10484fef0be4da5a 100644 (file)
@@ -123,22 +123,6 @@ int copy_siginfo_to_user32(compat_siginfo_t __user *to, const siginfo_t *from)
        return err;
 }
 
-/* CAUTION: This is just a very minimalist implementation for the
- *          sake of compat_sys_rt_sigqueueinfo()
- */
-int copy_siginfo_from_user32(siginfo_t *to, compat_siginfo_t __user *from)
-{
-       if (!access_ok(VERIFY_WRITE, from, sizeof(compat_siginfo_t)))
-               return -EFAULT;
-
-       if (copy_from_user(to, from, 3*sizeof(int)) ||
-           copy_from_user(to->_sifields._pad, from->_sifields._pad,
-                          SI_PAD_SIZE))
-               return -EFAULT;
-
-       return 0;
-}
-
 /* Checks if the fp is valid.  We always build signal frames which are
  * 16-byte aligned, therefore we can always enforce that the restore
  * frame has that property as well.
index 971d87a1d8cf8a9241c026e7d83f33bb8581c9eb..4e7f40a10eb351961a233e82b9eeccd53fe95268 100644 (file)
@@ -105,24 +105,6 @@ int copy_siginfo_to_user32(struct compat_siginfo __user *to, const siginfo_t *fr
        return err;
 }
 
-int copy_siginfo_from_user32(siginfo_t *to, struct compat_siginfo __user *from)
-{
-       int err;
-
-       if (!access_ok(VERIFY_READ, from, sizeof(struct compat_siginfo)))
-               return -EFAULT;
-
-       err = __get_user(to->si_signo, &from->si_signo);
-       err |= __get_user(to->si_errno, &from->si_errno);
-       err |= __get_user(to->si_code, &from->si_code);
-
-       err |= __get_user(to->si_pid, &from->si_pid);
-       err |= __get_user(to->si_uid, &from->si_uid);
-       err |= __get_user(to->si_int, &from->si_int);
-
-       return err;
-}
-
 /* The assembly shim for this function arranges to ignore the return value. */
 long compat_sys_rt_sigreturn(void)
 {
index feb3ac135d0c4bed86f21f908db7b3a6acbf91fd..59148de2d83f5cfa1825d51c9b7d60484e1bd429 100644 (file)
@@ -207,24 +207,3 @@ int copy_siginfo_to_user32(compat_siginfo_t __user *to, const siginfo_t *from)
        return __copy_siginfo_to_user32(to, from, in_x32_syscall());
 }
 
-int copy_siginfo_from_user32(siginfo_t *to, compat_siginfo_t __user *from)
-{
-       int err = 0;
-       u32 ptr32;
-
-       if (!access_ok(VERIFY_READ, from, sizeof(compat_siginfo_t)))
-               return -EFAULT;
-
-       get_user_try {
-               get_user_ex(to->si_signo, &from->si_signo);
-               get_user_ex(to->si_errno, &from->si_errno);
-               get_user_ex(to->si_code, &from->si_code);
-
-               get_user_ex(to->si_pid, &from->si_pid);
-               get_user_ex(to->si_uid, &from->si_uid);
-               get_user_ex(ptr32, &from->si_ptr);
-               to->si_ptr = compat_ptr(ptr32);
-       } get_user_catch(err);
-
-       return err;
-}
index e698ec1908d970280bd9eeb65a37640ab49d5d18..8a9643857c4a13be60726a18bfe51ccfb1557c46 100644 (file)
@@ -510,7 +510,7 @@ long compat_get_bitmap(unsigned long *mask, const compat_ulong_t __user *umask,
                       unsigned long bitmap_size);
 long compat_put_bitmap(compat_ulong_t __user *umask, unsigned long *mask,
                       unsigned long bitmap_size);
-int copy_siginfo_from_user32(siginfo_t *to, struct compat_siginfo __user *from);
+int copy_siginfo_from_user32(siginfo_t *to, const struct compat_siginfo __user *from);
 int copy_siginfo_to_user32(struct compat_siginfo __user *to, const siginfo_t *from);
 int get_compat_sigevent(struct sigevent *event,
                const struct compat_sigevent __user *u_event);
index 4c3f4448c5f19287cde90999abccb5b0f7be3e64..5211b1b57163c9917f39a3e22847f5c2dc0e7a34 100644 (file)
@@ -2814,6 +2814,87 @@ int copy_siginfo_to_user(siginfo_t __user *to, const siginfo_t *from)
        return err;
 }
 
+#ifdef CONFIG_COMPAT
+int copy_siginfo_from_user32(struct siginfo *to,
+                            const struct compat_siginfo __user *ufrom)
+{
+       struct compat_siginfo from;
+
+       if (copy_from_user(&from, ufrom, sizeof(struct compat_siginfo)))
+               return -EFAULT;
+
+       clear_siginfo(to);
+       to->si_signo = from.si_signo;
+       to->si_errno = from.si_errno;
+       to->si_code  = from.si_code;
+       switch(siginfo_layout(from.si_signo, from.si_code)) {
+       case SIL_KILL:
+               to->si_pid = from.si_pid;
+               to->si_uid = from.si_uid;
+               break;
+       case SIL_TIMER:
+               to->si_tid     = from.si_tid;
+               to->si_overrun = from.si_overrun;
+               to->si_int     = from.si_int;
+               break;
+       case SIL_POLL:
+               to->si_band = from.si_band;
+               to->si_fd   = from.si_fd;
+               break;
+       case SIL_FAULT:
+               to->si_addr = compat_ptr(from.si_addr);
+#ifdef __ARCH_SI_TRAPNO
+               to->si_trapno = from.si_trapno;
+#endif
+#ifdef BUS_MCEERR_AR
+               if ((from.si_signo == SIGBUS) && (from.si_code == BUS_MCEERR_AR))
+                       to->si_addr_lsb = from.si_addr_lsb;
+#endif
+#ifdef BUS_MCEER_AO
+               if ((from.si_signo == SIGBUS) && (from.si_code == BUS_MCEERR_AO))
+                       to->si_addr_lsb = from.si_addr_lsb;
+#endif
+#ifdef SEGV_BNDERR
+               if ((from.si_signo == SIGSEGV) && (from.si_code == SEGV_BNDERR)) {
+                       to->si_lower = compat_ptr(from.si_lower);
+                       to->si_upper = compat_ptr(from.si_upper);
+               }
+#endif
+#ifdef SEGV_PKUERR
+               if ((from.si_signo == SIGSEGV) && (from.si_code == SEGV_PKUERR))
+                       to->si_pkey = from.si_pkey;
+#endif
+               break;
+       case SIL_CHLD:
+               to->si_pid    = from.si_pid;
+               to->si_uid    = from.si_uid;
+               to->si_status = from.si_status;
+#ifdef CONFIG_X86_X32_ABI
+               if (in_x32_syscall()) {
+                       to->si_utime = from._sifields._sigchld_x32._utime;
+                       to->si_stime = from._sifields._sigchld_x32._stime;
+               } else
+#endif
+               {
+                       to->si_utime = from.si_utime;
+                       to->si_stime = from.si_stime;
+               }
+               break;
+       case SIL_RT:
+               to->si_pid = from.si_pid;
+               to->si_uid = from.si_uid;
+               to->si_int = from.si_int;
+               break;
+       case SIL_SYS:
+               to->si_call_addr = compat_ptr(from.si_call_addr);
+               to->si_syscall   = from.si_syscall;
+               to->si_arch      = from.si_arch;
+               break;
+       }
+       return 0;
+}
+#endif /* CONFIG_COMPAT */
+
 /**
  *  do_sigtimedwait - wait for queued signals specified in @which
  *  @which: queued signals to wait for