mirror of
https://github.com/torvalds/linux.git
synced 2025-01-01 15:51:46 +00:00
d4173023e6
Pull siginfo cleanups from Eric Biederman: "Long ago when 2.4 was just a testing release copy_siginfo_to_user was made to copy individual fields to userspace, possibly for efficiency and to ensure initialized values were not copied to userspace. Unfortunately the design was complex, it's assumptions unstated, and humans are fallible and so while it worked much of the time that design failed to ensure unitialized memory is not copied to userspace. This set of changes is part of a new design to clean up siginfo and simplify things, and hopefully make the siginfo handling robust enough that a simple inspection of the code can be made to ensure we don't copy any unitializied fields to userspace. The design is to unify struct siginfo and struct compat_siginfo into a single definition that is shared between all architectures so that anyone adding to the set of information shared with struct siginfo can see the whole picture. Hopefully ensuring all future si_code assignments are arch independent. The design is to unify copy_siginfo_to_user32 and copy_siginfo_from_user32 so that those function are complete and cope with all of the different cases documented in signinfo_layout. I don't think there was a single implementation of either of those functions that was complete and correct before my changes unified them. The design is to introduce a series of helpers including force_siginfo_fault that take the values that are needed in struct siginfo and build the siginfo structure for their callers. Ensuring struct siginfo is built correctly. The remaining work for 4.17 (unless someone thinks it is post -rc1 material) is to push usage of those helpers down into the architectures so that architecture specific code will not need to deal with the fiddly work of intializing struct siginfo, and then when struct siginfo is guaranteed to be fully initialized change copy siginfo_to_user into a simple wrapper around copy_to_user. Further there is work in progress on the issues that have been documented requires arch specific knowledge to sort out. The changes below fix or at least document all of the issues that have been found with siginfo generation. Then proceed to unify struct siginfo the 32 bit helpers that copy siginfo to and from userspace, and generally clean up anything that is not arch specific with regards to siginfo generation. It is a lot but with the unification you can of siginfo you can already see the code reduction in the kernel" * 'siginfo-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/ebiederm/user-namespace: (45 commits) signal/memory-failure: Use force_sig_mceerr and send_sig_mceerr mm/memory_failure: Remove unused trapno from memory_failure signal/ptrace: Add force_sig_ptrace_errno_trap and use it where needed signal/powerpc: Remove unnecessary signal_code parameter of do_send_trap signal: Helpers for faults with specialized siginfo layouts signal: Add send_sig_fault and force_sig_fault signal: Replace memset(info,...) with clear_siginfo for clarity signal: Don't use structure initializers for struct siginfo signal/arm64: Better isolate the COMPAT_TASK portion of ptrace_hbptriggered ptrace: Use copy_siginfo in setsiginfo and getsiginfo signal: Unify and correct copy_siginfo_to_user32 signal: Remove the code to clear siginfo before calling copy_siginfo_from_user32 signal: Unify and correct copy_siginfo_from_user32 signal/blackfin: Remove pointless UID16_SIGINFO_COMPAT_NEEDED signal/blackfin: Move the blackfin specific si_codes to asm-generic/siginfo.h signal/tile: Move the tile specific si_codes to asm-generic/siginfo.h signal/frv: Move the frv specific si_codes to asm-generic/siginfo.h signal/ia64: Move the ia64 specific si_codes to asm-generic/siginfo.h signal/powerpc: Remove redefinition of NSIGTRAP on powerpc signal: Move addr_lsb into the _sigfault union for clarity ...
174 lines
4.4 KiB
C
174 lines
4.4 KiB
C
/*
|
|
* Traps/Non-MMU Exception handling for ARC
|
|
*
|
|
* Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com)
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2 as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* vineetg: May 2011
|
|
* -user-space unaligned access emulation
|
|
*
|
|
* Rahul Trivedi: Codito Technologies 2004
|
|
*/
|
|
|
|
#include <linux/sched/signal.h>
|
|
#include <linux/kdebug.h>
|
|
#include <linux/uaccess.h>
|
|
#include <linux/ptrace.h>
|
|
#include <linux/kprobes.h>
|
|
#include <linux/kgdb.h>
|
|
#include <asm/setup.h>
|
|
#include <asm/unaligned.h>
|
|
#include <asm/kprobes.h>
|
|
|
|
void __init trap_init(void)
|
|
{
|
|
return;
|
|
}
|
|
|
|
void die(const char *str, struct pt_regs *regs, unsigned long address)
|
|
{
|
|
show_kernel_fault_diag(str, regs, address);
|
|
|
|
/* DEAD END */
|
|
__asm__("flag 1");
|
|
}
|
|
|
|
/*
|
|
* Helper called for bulk of exceptions NOT needing specific handling
|
|
* -for user faults enqueues requested signal
|
|
* -for kernel, chk if due to copy_(to|from)_user, otherwise die()
|
|
*/
|
|
static noinline int
|
|
unhandled_exception(const char *str, struct pt_regs *regs, siginfo_t *info)
|
|
{
|
|
if (user_mode(regs)) {
|
|
struct task_struct *tsk = current;
|
|
|
|
tsk->thread.fault_address = (__force unsigned int)info->si_addr;
|
|
|
|
force_sig_info(info->si_signo, info, tsk);
|
|
|
|
} else {
|
|
/* If not due to copy_(to|from)_user, we are doomed */
|
|
if (fixup_exception(regs))
|
|
return 0;
|
|
|
|
die(str, regs, (unsigned long)info->si_addr);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
#define DO_ERROR_INFO(signr, str, name, sicode) \
|
|
int name(unsigned long address, struct pt_regs *regs) \
|
|
{ \
|
|
siginfo_t info; \
|
|
\
|
|
clear_siginfo(&info); \
|
|
info.si_signo = signr; \
|
|
info.si_errno = 0; \
|
|
info.si_code = sicode; \
|
|
info.si_addr = (void __user *)address; \
|
|
\
|
|
return unhandled_exception(str, regs, &info);\
|
|
}
|
|
|
|
/*
|
|
* Entry points for exceptions NOT needing specific handling
|
|
*/
|
|
DO_ERROR_INFO(SIGILL, "Priv Op/Disabled Extn", do_privilege_fault, ILL_PRVOPC)
|
|
DO_ERROR_INFO(SIGILL, "Invalid Extn Insn", do_extension_fault, ILL_ILLOPC)
|
|
DO_ERROR_INFO(SIGILL, "Illegal Insn (or Seq)", insterror_is_error, ILL_ILLOPC)
|
|
DO_ERROR_INFO(SIGBUS, "Invalid Mem Access", __weak do_memory_error, BUS_ADRERR)
|
|
DO_ERROR_INFO(SIGTRAP, "Breakpoint Set", trap_is_brkpt, TRAP_BRKPT)
|
|
DO_ERROR_INFO(SIGBUS, "Misaligned Access", do_misaligned_error, BUS_ADRALN)
|
|
DO_ERROR_INFO(SIGSEGV, "gcc generated __builtin_trap", do_trap5_error, 0)
|
|
|
|
/*
|
|
* Entry Point for Misaligned Data access Exception, for emulating in software
|
|
*/
|
|
int do_misaligned_access(unsigned long address, struct pt_regs *regs,
|
|
struct callee_regs *cregs)
|
|
{
|
|
/* If emulation not enabled, or failed, kill the task */
|
|
if (misaligned_fixup(address, regs, cregs) != 0)
|
|
return do_misaligned_error(address, regs);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Entry point for miscll errors such as Nested Exceptions
|
|
* -Duplicate TLB entry is handled seperately though
|
|
*/
|
|
void do_machine_check_fault(unsigned long address, struct pt_regs *regs)
|
|
{
|
|
die("Unhandled Machine Check Exception", regs, address);
|
|
}
|
|
|
|
|
|
/*
|
|
* Entry point for traps induced by ARCompact TRAP_S <n> insn
|
|
* This is same family as TRAP0/SWI insn (use the same vector).
|
|
* The only difference being SWI insn take no operand, while TRAP_S does
|
|
* which reflects in ECR Reg as 8 bit param.
|
|
* Thus TRAP_S <n> can be used for specific purpose
|
|
* -1 used for software breakpointing (gdb)
|
|
* -2 used by kprobes
|
|
* -5 __builtin_trap() generated by gcc (2018.03 onwards) for toggle such as
|
|
* -fno-isolate-erroneous-paths-dereference
|
|
*/
|
|
void do_non_swi_trap(unsigned long address, struct pt_regs *regs)
|
|
{
|
|
unsigned int param = regs->ecr_param;
|
|
|
|
switch (param) {
|
|
case 1:
|
|
trap_is_brkpt(address, regs);
|
|
break;
|
|
|
|
case 2:
|
|
trap_is_kprobe(address, regs);
|
|
break;
|
|
|
|
case 3:
|
|
case 4:
|
|
kgdb_trap(regs);
|
|
break;
|
|
|
|
case 5:
|
|
do_trap5_error(address, regs);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Entry point for Instruction Error Exception
|
|
* -For a corner case, ARC kprobes implementation resorts to using
|
|
* this exception, hence the check
|
|
*/
|
|
void do_insterror_or_kprobe(unsigned long address, struct pt_regs *regs)
|
|
{
|
|
int rc;
|
|
|
|
/* Check if this exception is caused by kprobes */
|
|
rc = notify_die(DIE_IERR, "kprobe_ierr", regs, address, 0, SIGILL);
|
|
if (rc == NOTIFY_STOP)
|
|
return;
|
|
|
|
insterror_is_error(address, regs);
|
|
}
|
|
|
|
/*
|
|
* abort() call generated by older gcc for __builtin_trap()
|
|
*/
|
|
void abort(void)
|
|
{
|
|
__asm__ __volatile__("trap_s 5\n");
|
|
}
|