aboutsummaryrefslogtreecommitdiffstats
path: root/arch/sparc64/kernel/traps.c
diff options
context:
space:
mode:
Diffstat (limited to 'arch/sparc64/kernel/traps.c')
-rw-r--r--arch/sparc64/kernel/traps.c18
1 files changed, 15 insertions, 3 deletions
diff --git a/arch/sparc64/kernel/traps.c b/arch/sparc64/kernel/traps.c
index fe1796c939c3..ec7a60142bec 100644
--- a/arch/sparc64/kernel/traps.c
+++ b/arch/sparc64/kernel/traps.c
@@ -2261,8 +2261,12 @@ void die_if_kernel(char *str, struct pt_regs *regs)
2261 do_exit(SIGSEGV); 2261 do_exit(SIGSEGV);
2262} 2262}
2263 2263
2264#define VIS_OPCODE_MASK ((0x3 << 30) | (0x3f << 19))
2265#define VIS_OPCODE_VAL ((0x2 << 30) | (0x36 << 19))
2266
2264extern int handle_popc(u32 insn, struct pt_regs *regs); 2267extern int handle_popc(u32 insn, struct pt_regs *regs);
2265extern int handle_ldf_stq(u32 insn, struct pt_regs *regs); 2268extern int handle_ldf_stq(u32 insn, struct pt_regs *regs);
2269extern int vis_emul(struct pt_regs *, unsigned int);
2266 2270
2267void do_illegal_instruction(struct pt_regs *regs) 2271void do_illegal_instruction(struct pt_regs *regs)
2268{ 2272{
@@ -2287,10 +2291,18 @@ void do_illegal_instruction(struct pt_regs *regs)
2287 if (handle_ldf_stq(insn, regs)) 2291 if (handle_ldf_stq(insn, regs))
2288 return; 2292 return;
2289 } else if (tlb_type == hypervisor) { 2293 } else if (tlb_type == hypervisor) {
2290 extern int vis_emul(struct pt_regs *, unsigned int); 2294 if ((insn & VIS_OPCODE_MASK) == VIS_OPCODE_VAL) {
2295 if (!vis_emul(regs, insn))
2296 return;
2297 } else {
2298 struct fpustate *f = FPUSTATE;
2291 2299
2292 if (!vis_emul(regs, insn)) 2300 /* XXX maybe verify XFSR bits like
2293 return; 2301 * XXX do_fpother() does?
2302 */
2303 if (do_mathemu(regs, f))
2304 return;
2305 }
2294 } 2306 }
2295 } 2307 }
2296 info.si_signo = SIGILL; 2308 info.si_signo = SIGILL;