muldiv.c 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  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)
  4. *
  5. * Copyright (C) 1996 Jakub Jelinek (jj@sunsite.mff.cuni.cz)
  6. * Copyright (C) 1996 David S. Miller (davem@caip.rutgers.edu)
  7. *
  8. * 2004-12-25 Krzysztof Helt (krzysztof.h1@wp.pl)
  9. * - fixed registers constrains in inline assembly declarations
  10. */
  11. #include <linux/kernel.h>
  12. #include <linux/sched.h>
  13. #include <linux/mm.h>
  14. #include <asm/ptrace.h>
  15. #include <asm/processor.h>
  16. #include <asm/system.h>
  17. #include <asm/uaccess.h>
  18. /* #define DEBUG_MULDIV */
  19. static inline int has_imm13(int insn)
  20. {
  21. return (insn & 0x2000);
  22. }
  23. static inline int is_foocc(int insn)
  24. {
  25. return (insn & 0x800000);
  26. }
  27. static inline int sign_extend_imm13(int imm)
  28. {
  29. return imm << 19 >> 19;
  30. }
  31. static inline void advance(struct pt_regs *regs)
  32. {
  33. regs->pc = regs->npc;
  34. regs->npc += 4;
  35. }
  36. static inline void maybe_flush_windows(unsigned int rs1, unsigned int rs2,
  37. unsigned int rd)
  38. {
  39. if(rs2 >= 16 || rs1 >= 16 || rd >= 16) {
  40. /* Wheee... */
  41. __asm__ __volatile__("save %sp, -0x40, %sp\n\t"
  42. "save %sp, -0x40, %sp\n\t"
  43. "save %sp, -0x40, %sp\n\t"
  44. "save %sp, -0x40, %sp\n\t"
  45. "save %sp, -0x40, %sp\n\t"
  46. "save %sp, -0x40, %sp\n\t"
  47. "save %sp, -0x40, %sp\n\t"
  48. "restore; restore; restore; restore;\n\t"
  49. "restore; restore; restore;\n\t");
  50. }
  51. }
  52. #define fetch_reg(reg, regs) ({ \
  53. struct reg_window __user *win; \
  54. register unsigned long ret; \
  55. \
  56. if (!(reg)) ret = 0; \
  57. else if ((reg) < 16) { \
  58. ret = regs->u_regs[(reg)]; \
  59. } else { \
  60. /* Ho hum, the slightly complicated case. */ \
  61. win = (struct reg_window __user *)regs->u_regs[UREG_FP];\
  62. if (get_user (ret, &win->locals[(reg) - 16])) return -1;\
  63. } \
  64. ret; \
  65. })
  66. static inline int
  67. store_reg(unsigned int result, unsigned int reg, struct pt_regs *regs)
  68. {
  69. struct reg_window __user *win;
  70. if (!reg)
  71. return 0;
  72. if (reg < 16) {
  73. regs->u_regs[reg] = result;
  74. return 0;
  75. } else {
  76. /* need to use put_user() in this case: */
  77. win = (struct reg_window __user *) regs->u_regs[UREG_FP];
  78. return (put_user(result, &win->locals[reg - 16]));
  79. }
  80. }
  81. extern void handle_hw_divzero (struct pt_regs *regs, unsigned long pc,
  82. unsigned long npc, unsigned long psr);
  83. /* Should return 0 if mul/div emulation succeeded and SIGILL should
  84. * not be issued.
  85. */
  86. int do_user_muldiv(struct pt_regs *regs, unsigned long pc)
  87. {
  88. unsigned int insn;
  89. int inst;
  90. unsigned int rs1, rs2, rdv;
  91. if (!pc)
  92. return -1; /* This happens to often, I think */
  93. if (get_user (insn, (unsigned int __user *)pc))
  94. return -1;
  95. if ((insn & 0xc1400000) != 0x80400000)
  96. return -1;
  97. inst = ((insn >> 19) & 0xf);
  98. if ((inst & 0xe) != 10 && (inst & 0xe) != 14)
  99. return -1;
  100. /* Now we know we have to do something with umul, smul, udiv or sdiv */
  101. rs1 = (insn >> 14) & 0x1f;
  102. rs2 = insn & 0x1f;
  103. rdv = (insn >> 25) & 0x1f;
  104. if (has_imm13(insn)) {
  105. maybe_flush_windows(rs1, 0, rdv);
  106. rs2 = sign_extend_imm13(insn);
  107. } else {
  108. maybe_flush_windows(rs1, rs2, rdv);
  109. rs2 = fetch_reg(rs2, regs);
  110. }
  111. rs1 = fetch_reg(rs1, regs);
  112. switch (inst) {
  113. case 10: /* umul */
  114. #ifdef DEBUG_MULDIV
  115. printk ("unsigned muldiv: 0x%x * 0x%x = ", rs1, rs2);
  116. #endif
  117. __asm__ __volatile__ ("\n\t"
  118. "mov %0, %%o0\n\t"
  119. "call .umul\n\t"
  120. " mov %1, %%o1\n\t"
  121. "mov %%o0, %0\n\t"
  122. "mov %%o1, %1\n\t"
  123. : "=r" (rs1), "=r" (rs2)
  124. : "0" (rs1), "1" (rs2)
  125. : "o0", "o1", "o2", "o3", "o4", "o5", "o7", "cc");
  126. #ifdef DEBUG_MULDIV
  127. printk ("0x%x%08x\n", rs2, rs1);
  128. #endif
  129. if (store_reg(rs1, rdv, regs))
  130. return -1;
  131. regs->y = rs2;
  132. break;
  133. case 11: /* smul */
  134. #ifdef DEBUG_MULDIV
  135. printk ("signed muldiv: 0x%x * 0x%x = ", rs1, rs2);
  136. #endif
  137. __asm__ __volatile__ ("\n\t"
  138. "mov %0, %%o0\n\t"
  139. "call .mul\n\t"
  140. " mov %1, %%o1\n\t"
  141. "mov %%o0, %0\n\t"
  142. "mov %%o1, %1\n\t"
  143. : "=r" (rs1), "=r" (rs2)
  144. : "0" (rs1), "1" (rs2)
  145. : "o0", "o1", "o2", "o3", "o4", "o5", "o7", "cc");
  146. #ifdef DEBUG_MULDIV
  147. printk ("0x%x%08x\n", rs2, rs1);
  148. #endif
  149. if (store_reg(rs1, rdv, regs))
  150. return -1;
  151. regs->y = rs2;
  152. break;
  153. case 14: /* udiv */
  154. #ifdef DEBUG_MULDIV
  155. printk ("unsigned muldiv: 0x%x%08x / 0x%x = ", regs->y, rs1, rs2);
  156. #endif
  157. if (!rs2) {
  158. #ifdef DEBUG_MULDIV
  159. printk ("DIVISION BY ZERO\n");
  160. #endif
  161. handle_hw_divzero (regs, pc, regs->npc, regs->psr);
  162. return 0;
  163. }
  164. __asm__ __volatile__ ("\n\t"
  165. "mov %2, %%o0\n\t"
  166. "mov %0, %%o1\n\t"
  167. "mov %%g0, %%o2\n\t"
  168. "call __udivdi3\n\t"
  169. " mov %1, %%o3\n\t"
  170. "mov %%o1, %0\n\t"
  171. "mov %%o0, %1\n\t"
  172. : "=r" (rs1), "=r" (rs2)
  173. : "r" (regs->y), "0" (rs1), "1" (rs2)
  174. : "o0", "o1", "o2", "o3", "o4", "o5", "o7",
  175. "g1", "g2", "g3", "cc");
  176. #ifdef DEBUG_MULDIV
  177. printk ("0x%x\n", rs1);
  178. #endif
  179. if (store_reg(rs1, rdv, regs))
  180. return -1;
  181. break;
  182. case 15: /* sdiv */
  183. #ifdef DEBUG_MULDIV
  184. printk ("signed muldiv: 0x%x%08x / 0x%x = ", regs->y, rs1, rs2);
  185. #endif
  186. if (!rs2) {
  187. #ifdef DEBUG_MULDIV
  188. printk ("DIVISION BY ZERO\n");
  189. #endif
  190. handle_hw_divzero (regs, pc, regs->npc, regs->psr);
  191. return 0;
  192. }
  193. __asm__ __volatile__ ("\n\t"
  194. "mov %2, %%o0\n\t"
  195. "mov %0, %%o1\n\t"
  196. "mov %%g0, %%o2\n\t"
  197. "call __divdi3\n\t"
  198. " mov %1, %%o3\n\t"
  199. "mov %%o1, %0\n\t"
  200. "mov %%o0, %1\n\t"
  201. : "=r" (rs1), "=r" (rs2)
  202. : "r" (regs->y), "0" (rs1), "1" (rs2)
  203. : "o0", "o1", "o2", "o3", "o4", "o5", "o7",
  204. "g1", "g2", "g3", "cc");
  205. #ifdef DEBUG_MULDIV
  206. printk ("0x%x\n", rs1);
  207. #endif
  208. if (store_reg(rs1, rdv, regs))
  209. return -1;
  210. break;
  211. }
  212. if (is_foocc (insn)) {
  213. regs->psr &= ~PSR_ICC;
  214. if ((inst & 0xe) == 14) {
  215. /* ?div */
  216. if (rs2) regs->psr |= PSR_V;
  217. }
  218. if (!rs1) regs->psr |= PSR_Z;
  219. if (((int)rs1) < 0) regs->psr |= PSR_N;
  220. #ifdef DEBUG_MULDIV
  221. printk ("psr muldiv: %08x\n", regs->psr);
  222. #endif
  223. }
  224. advance(regs);
  225. return 0;
  226. }