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 __user *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 __user *)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 __user *win;
 
  83                 regs->u_regs[reg] = result;
 
  86                 /* need to use put_user() in this case: */
 
  87                 win = (struct reg_window __user *) 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
 
  98 int do_user_muldiv(struct pt_regs *regs, unsigned long pc)
 
 102         unsigned int rs1, rs2, rdv;
 
 105                 return -1; /* This happens to often, I think */
 
 106         if (get_user (insn, (unsigned int __user *)pc))
 
 108         if ((insn & 0xc1400000) != 0x80400000)
 
 110         inst = ((insn >> 19) & 0xf);
 
 111         if ((inst & 0xe) != 10 && (inst & 0xe) != 14)
 
 114         /* Now we know we have to do something with umul, smul, udiv or sdiv */
 
 115         rs1 = (insn >> 14) & 0x1f;
 
 117         rdv = (insn >> 25) & 0x1f;
 
 118         if (has_imm13(insn)) {
 
 119                 maybe_flush_windows(rs1, 0, rdv);
 
 120                 rs2 = sign_extend_imm13(insn);
 
 122                 maybe_flush_windows(rs1, rs2, rdv);
 
 123                 rs2 = fetch_reg(rs2, regs);
 
 125         rs1 = fetch_reg(rs1, regs);
 
 129                 printk ("unsigned muldiv: 0x%x * 0x%x = ", rs1, rs2);
 
 131                 __asm__ __volatile__ ("\n\t"
 
 137                         : "=r" (rs1), "=r" (rs2)
 
 138                         : "0" (rs1), "1" (rs2)
 
 139                         : "o0", "o1", "o2", "o3", "o4", "o5", "o7", "cc");
 
 141                 printk ("0x%x%08x\n", rs2, rs1);
 
 143                 if (store_reg(rs1, rdv, regs))
 
 149                 printk ("signed muldiv: 0x%x * 0x%x = ", rs1, rs2);
 
 151                 __asm__ __volatile__ ("\n\t"
 
 157                         : "=r" (rs1), "=r" (rs2)
 
 158                         : "0" (rs1), "1" (rs2)
 
 159                         : "o0", "o1", "o2", "o3", "o4", "o5", "o7", "cc");
 
 161                 printk ("0x%x%08x\n", rs2, rs1);
 
 163                 if (store_reg(rs1, rdv, regs))
 
 169                 printk ("unsigned muldiv: 0x%x%08x / 0x%x = ", regs->y, rs1, rs2);
 
 173                         printk ("DIVISION BY ZERO\n");
 
 175                         handle_hw_divzero (regs, pc, regs->npc, regs->psr);
 
 178                 __asm__ __volatile__ ("\n\t"
 
 186                         : "=r" (rs1), "=r" (rs2)
 
 187                         : "r" (regs->y), "0" (rs1), "1" (rs2)
 
 188                         : "o0", "o1", "o2", "o3", "o4", "o5", "o7",
 
 189                           "g1", "g2", "g3", "cc");
 
 191                 printk ("0x%x\n", rs1);
 
 193                 if (store_reg(rs1, rdv, regs))
 
 198                 printk ("signed muldiv: 0x%x%08x / 0x%x = ", regs->y, rs1, rs2);
 
 202                         printk ("DIVISION BY ZERO\n");
 
 204                         handle_hw_divzero (regs, pc, regs->npc, regs->psr);
 
 207                 __asm__ __volatile__ ("\n\t"
 
 215                         : "=r" (rs1), "=r" (rs2)
 
 216                         : "r" (regs->y), "0" (rs1), "1" (rs2)
 
 217                         : "o0", "o1", "o2", "o3", "o4", "o5", "o7",
 
 218                           "g1", "g2", "g3", "cc");
 
 220                 printk ("0x%x\n", rs1);
 
 222                 if (store_reg(rs1, rdv, regs))
 
 226         if (is_foocc (insn)) {
 
 227                 regs->psr &= ~PSR_ICC;
 
 228                 if ((inst & 0xe) == 14) {
 
 230                         if (rs2) regs->psr |= PSR_V;
 
 232                 if (!rs1) regs->psr |= PSR_Z;
 
 233                 if (((int)rs1) < 0) regs->psr |= PSR_N;
 
 235                 printk ("psr muldiv: %08x\n", regs->psr);