1 /* $Id: muldiv.c,v 1.5 1997/12/15 20:07:20 ecd Exp $
2 * muldiv.c: Hardware multiply/division illegal instruction trap
3 * for sun4c/sun4 (which do not have those instructions)
5 * Copyright (C) 1996 Jakub Jelinek (jj@sunsite.mff.cuni.cz)
6 * Copyright (C) 1996 David S. Miller (davem@caip.rutgers.edu)
8 * 2004-12-25 Krzysztof Helt (krzysztof.h1@wp.pl)
9 * - fixed registers constrains in inline assembly declarations
12 #include <linux/kernel.h>
13 #include <linux/sched.h>
15 #include <asm/ptrace.h>
16 #include <asm/processor.h>
17 #include <asm/system.h>
18 #include <asm/uaccess.h>
20 /* #define DEBUG_MULDIV */
22 static inline int has_imm13(int insn)
24 return (insn & 0x2000);
27 static inline int is_foocc(int insn)
29 return (insn & 0x800000);
32 static inline int sign_extend_imm13(int imm)
34 return imm << 19 >> 19;
37 static inline void advance(struct pt_regs *regs)
43 static inline void maybe_flush_windows(unsigned int rs1, unsigned int rs2,
46 if(rs2 >= 16 || rs1 >= 16 || rd >= 16) {
48 __asm__ __volatile__("save %sp, -0x40, %sp\n\t"
49 "save %sp, -0x40, %sp\n\t"
50 "save %sp, -0x40, %sp\n\t"
51 "save %sp, -0x40, %sp\n\t"
52 "save %sp, -0x40, %sp\n\t"
53 "save %sp, -0x40, %sp\n\t"
54 "save %sp, -0x40, %sp\n\t"
55 "restore; restore; restore; restore;\n\t"
56 "restore; restore; restore;\n\t");
60 #define fetch_reg(reg, regs) ({ \
61 struct reg_window *win; \
62 register unsigned long ret; \
64 if (!(reg)) ret = 0; \
65 else if((reg) < 16) { \
66 ret = regs->u_regs[(reg)]; \
68 /* Ho hum, the slightly complicated case. */ \
69 win = (struct reg_window *)regs->u_regs[UREG_FP]; \
70 if (get_user (ret, &win->locals[(reg) - 16])) return -1; \
76 store_reg(unsigned int result, unsigned int reg, struct pt_regs *regs)
78 struct reg_window *win;
83 regs->u_regs[reg] = result;
86 /* need to use put_user() in this case: */
87 win = (struct reg_window *)regs->u_regs[UREG_FP];
88 return (put_user(result, &win->locals[reg - 16]));
92 extern void handle_hw_divzero (struct pt_regs *regs, unsigned long pc,
93 unsigned long npc, unsigned long psr);
95 /* Should return 0 if mul/div emulation succeeded and SIGILL should not be issued */
96 int do_user_muldiv(struct pt_regs *regs, unsigned long pc)
100 unsigned int rs1, rs2, rdv;
102 if (!pc) return -1; /* This happens to often, I think */
103 if (get_user (insn, (unsigned int *)pc)) return -1;
104 if ((insn & 0xc1400000) != 0x80400000) return -1;
105 inst = ((insn >> 19) & 0xf);
106 if ((inst & 0xe) != 10 && (inst & 0xe) != 14) return -1;
107 /* Now we know we have to do something with umul, smul, udiv or sdiv */
108 rs1 = (insn >> 14) & 0x1f;
110 rdv = (insn >> 25) & 0x1f;
111 if(has_imm13(insn)) {
112 maybe_flush_windows(rs1, 0, rdv);
113 rs2 = sign_extend_imm13(insn);
115 maybe_flush_windows(rs1, rs2, rdv);
116 rs2 = fetch_reg(rs2, regs);
118 rs1 = fetch_reg(rs1, regs);
122 printk ("unsigned muldiv: 0x%x * 0x%x = ", rs1, rs2);
124 __asm__ __volatile__ ("\n\t"
130 : "=r" (rs1), "=r" (rs2)
131 : "0" (rs1), "1" (rs2)
132 : "o0", "o1", "o2", "o3", "o4", "o5", "o7", "cc");
134 printk ("0x%x%08x\n", rs2, rs1);
136 if (store_reg(rs1, rdv, regs))
142 printk ("signed muldiv: 0x%x * 0x%x = ", rs1, rs2);
144 __asm__ __volatile__ ("\n\t"
150 : "=r" (rs1), "=r" (rs2)
151 : "0" (rs1), "1" (rs2)
152 : "o0", "o1", "o2", "o3", "o4", "o5", "o7", "cc");
154 printk ("0x%x%08x\n", rs2, rs1);
156 if (store_reg(rs1, rdv, regs))
162 printk ("unsigned muldiv: 0x%x%08x / 0x%x = ", regs->y, rs1, rs2);
166 printk ("DIVISION BY ZERO\n");
168 handle_hw_divzero (regs, pc, regs->npc, regs->psr);
171 __asm__ __volatile__ ("\n\t"
179 : "=r" (rs1), "=r" (rs2)
180 : "r" (regs->y), "0" (rs1), "1" (rs2)
181 : "o0", "o1", "o2", "o3", "o4", "o5", "o7",
182 "g1", "g2", "g3", "cc");
184 printk ("0x%x\n", rs1);
186 if (store_reg(rs1, rdv, regs))
191 printk ("signed muldiv: 0x%x%08x / 0x%x = ", regs->y, rs1, rs2);
195 printk ("DIVISION BY ZERO\n");
197 handle_hw_divzero (regs, pc, regs->npc, regs->psr);
200 __asm__ __volatile__ ("\n\t"
208 : "=r" (rs1), "=r" (rs2)
209 : "r" (regs->y), "0" (rs1), "1" (rs2)
210 : "o0", "o1", "o2", "o3", "o4", "o5", "o7",
211 "g1", "g2", "g3", "cc");
213 printk ("0x%x\n", rs1);
215 if (store_reg(rs1, rdv, regs))
219 if (is_foocc (insn)) {
220 regs->psr &= ~PSR_ICC;
221 if ((inst & 0xe) == 14) {
223 if (rs2) regs->psr |= PSR_V;
225 if (!rs1) regs->psr |= PSR_Z;
226 if (((int)rs1) < 0) regs->psr |= PSR_N;
228 printk ("psr muldiv: %08x\n", regs->psr);