Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/arch/mips Handle gas/gcc generating a break/trap 6 for i...
details: https://anonhg.NetBSD.org/src/rev/dc521c51dcc5
branches: trunk
changeset: 1019644:dc521c51dcc5
user: simonb <simonb%NetBSD.org@localhost>
date: Wed Mar 17 11:05:37 2021 +0000
description:
Handle gas/gcc generating a break/trap 6 for integer overflow and
break/trap 7 for integer divide by zero and setting the SIGFPE
si_code of FPE_INTOVF or FPE_INTDIV respectively. The break/trap
6/7 seems to have existed since the early days of MIPS but not
well documented anywhere.
Fixes ATF lib/libc/gen/t_siginfo::sigfpe_int .
diffstat:
sys/arch/mips/include/cpuregs.h | 4 +-
sys/arch/mips/mips/trap.c | 71 ++++++++++++++++++++++++++++++++++++----
2 files changed, 66 insertions(+), 9 deletions(-)
diffs (147 lines):
diff -r ce5d76925948 -r dc521c51dcc5 sys/arch/mips/include/cpuregs.h
--- a/sys/arch/mips/include/cpuregs.h Wed Mar 17 08:13:29 2021 +0000
+++ b/sys/arch/mips/include/cpuregs.h Wed Mar 17 11:05:37 2021 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: cpuregs.h,v 1.109 2020/08/22 03:41:33 simonb Exp $ */
+/* $NetBSD: cpuregs.h,v 1.110 2021/03/17 11:05:37 simonb Exp $ */
/*
* Copyright (c) 2009 Miodrag Vallat.
@@ -679,6 +679,8 @@
#define MIPS_BREAK_INSTR 0x0000000d
#define MIPS_BREAK_VAL_MASK 0x03ff0000
#define MIPS_BREAK_VAL_SHIFT 16
+#define MIPS_BREAK_INTOVERFLOW 6 /* used by gas to indicate int overflow */
+#define MIPS_BREAK_INTDIVZERO 7 /* used by gas/gcc to indicate int div by zero */
#define MIPS_BREAK_KDB_VAL 512
#define MIPS_BREAK_SSTEP_VAL 513
#define MIPS_BREAK_BRKPT_VAL 514
diff -r ce5d76925948 -r dc521c51dcc5 sys/arch/mips/mips/trap.c
--- a/sys/arch/mips/mips/trap.c Wed Mar 17 08:13:29 2021 +0000
+++ b/sys/arch/mips/mips/trap.c Wed Mar 17 11:05:37 2021 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: trap.c,v 1.258 2021/03/13 17:14:11 skrll Exp $ */
+/* $NetBSD: trap.c,v 1.259 2021/03/17 11:05:37 simonb Exp $ */
/*
* Copyright (c) 1988 University of Utah.
@@ -39,7 +39,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: trap.c,v 1.258 2021/03/13 17:14:11 skrll Exp $");
+__KERNEL_RCSID(0, "$NetBSD: trap.c,v 1.259 2021/03/17 11:05:37 simonb Exp $");
#include "opt_cputype.h" /* which mips CPU levels do we support? */
#include "opt_ddb.h"
@@ -159,7 +159,6 @@
trap(uint32_t status, uint32_t cause, vaddr_t vaddr, vaddr_t pc,
struct trapframe *tf)
{
- int type;
struct lwp * const l = curlwp;
struct proc * const p = curproc;
struct trapframe * const utf = l->l_md.md_utf;
@@ -168,6 +167,9 @@
ksiginfo_t ksi;
extern void fswintrberr(void);
void *onfault;
+ InstFmt insn;
+ uint32_t instr;
+ int type;
int rv = 0;
KSI_INIT_TRAP(&ksi);
@@ -547,21 +549,46 @@
goto dopanic;
#endif
case T_BREAK+T_USER: {
- uint32_t instr;
-
/* compute address of break instruction */
vaddr_t va = pc + (cause & MIPS_CR_BR_DELAY ? sizeof(int) : 0);
/* read break instruction */
instr = mips_ufetch32((void *)va);
+ insn.word = instr;
if (l->l_md.md_ss_addr != va || instr != MIPS_BREAK_SSTEP) {
+ bool advance_pc = false;
+
ksi.ksi_trap = type & ~T_USER;
ksi.ksi_signo = SIGTRAP;
ksi.ksi_addr = (void *)va;
ksi.ksi_code = TRAP_TRACE;
- /* we broke, skip it to avoid infinite loop */
- if (instr == MIPS_BREAK_INSTR)
+
+ if ((insn.JType.op == OP_SPECIAL) &&
+ (insn.RType.func == OP_BREAK)) {
+ int code = (insn.RType.rs << 5) | insn.RType.rt;
+ switch (code) {
+ case 0:
+ /* we broke, skip it to avoid infinite loop */
+ advance_pc = true;
+ break;
+ case MIPS_BREAK_INTOVERFLOW:
+ ksi.ksi_signo = SIGFPE;
+ ksi.ksi_code = FPE_INTOVF;
+ advance_pc = true;
+ break;
+ case MIPS_BREAK_INTDIVZERO:
+ ksi.ksi_signo = SIGFPE;
+ ksi.ksi_code = FPE_INTDIV;
+ advance_pc = true;
+ break;
+ default:
+ /* do nothing */
+ break;
+ }
+ }
+
+ if (advance_pc)
tf->tf_regs[_R_PC] += 4;
break;
}
@@ -627,12 +654,40 @@
userret(l);
return; /* GEN */
case T_OVFLOW+T_USER:
- case T_TRAP+T_USER:
+ case T_TRAP+T_USER: {
+ /* compute address of trap/faulting instruction */
+ vaddr_t va = pc + (cause & MIPS_CR_BR_DELAY ? sizeof(int) : 0);
+ bool advance_pc = false;
+
+ /* read break instruction */
+ instr = mips_ufetch32((void *)va);
+ insn.word = instr;
+
ksi.ksi_trap = type & ~T_USER;
ksi.ksi_signo = SIGFPE;
ksi.ksi_addr = (void *)(intptr_t)pc /*utf->tf_regs[_R_PC]*/;
ksi.ksi_code = FPE_FLTOVF; /* XXX */
+
+ if ((insn.JType.op == OP_SPECIAL) &&
+ (insn.RType.func == OP_TEQ)) {
+ int code = (insn.RType.rd << 5) | insn.RType.shamt;
+ switch (code) {
+ case MIPS_BREAK_INTOVERFLOW:
+ ksi.ksi_code = FPE_INTOVF;
+ advance_pc = true;
+ break;
+ case MIPS_BREAK_INTDIVZERO:
+ ksi.ksi_code = FPE_INTDIV;
+ advance_pc = true;
+ break;
+ }
+ }
+
+ /* XXX when else do we advance the PC? */
+ if (advance_pc)
+ tf->tf_regs[_R_PC] += 4;
break; /* SIGNAL */
+ }
}
utf->tf_regs[_R_CAUSE] = cause;
utf->tf_regs[_R_BADVADDR] = vaddr;
Home |
Main Index |
Thread Index |
Old Index