diff options
Diffstat (limited to 'arch/sparc64/kernel/traps.c')
-rw-r--r-- | arch/sparc64/kernel/traps.c | 18 |
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 | |||
2264 | extern int handle_popc(u32 insn, struct pt_regs *regs); | 2267 | extern int handle_popc(u32 insn, struct pt_regs *regs); |
2265 | extern int handle_ldf_stq(u32 insn, struct pt_regs *regs); | 2268 | extern int handle_ldf_stq(u32 insn, struct pt_regs *regs); |
2269 | extern int vis_emul(struct pt_regs *, unsigned int); | ||
2266 | 2270 | ||
2267 | void do_illegal_instruction(struct pt_regs *regs) | 2271 | void 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; |