unaligned.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766
  1. /*
  2. * Unaligned memory access handler
  3. *
  4. * Copyright (C) 2001 Randolph Chung <tausq@debian.org>
  5. * Significantly tweaked by LaMont Jones <lamont@debian.org>
  6. *
  7. * This program is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License as published by
  9. * the Free Software Foundation; either version 2, or (at your option)
  10. * any later version.
  11. *
  12. * This program is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this program; if not, write to the Free Software
  19. * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  20. *
  21. */
  22. #include <linux/jiffies.h>
  23. #include <linux/kernel.h>
  24. #include <linux/module.h>
  25. #include <linux/sched/signal.h>
  26. #include <linux/sched/debug.h>
  27. #include <linux/signal.h>
  28. #include <linux/ratelimit.h>
  29. #include <linux/uaccess.h>
  30. #include <asm/hardirq.h>
  31. #include <asm/traps.h>
  32. /* #define DEBUG_UNALIGNED 1 */
  33. #ifdef DEBUG_UNALIGNED
  34. #define DPRINTF(fmt, args...) do { printk(KERN_DEBUG "%s:%d:%s ", __FILE__, __LINE__, __func__ ); printk(KERN_DEBUG fmt, ##args ); } while (0)
  35. #else
  36. #define DPRINTF(fmt, args...)
  37. #endif
  38. #ifdef CONFIG_64BIT
  39. #define RFMT "%016lx"
  40. #else
  41. #define RFMT "%08lx"
  42. #endif
  43. #define FIXUP_BRANCH(lbl) \
  44. "\tldil L%%" #lbl ", %%r1\n" \
  45. "\tldo R%%" #lbl "(%%r1), %%r1\n" \
  46. "\tbv,n %%r0(%%r1)\n"
  47. /* If you use FIXUP_BRANCH, then you must list this clobber */
  48. #define FIXUP_BRANCH_CLOBBER "r1"
  49. /* 1111 1100 0000 0000 0001 0011 1100 0000 */
  50. #define OPCODE1(a,b,c) ((a)<<26|(b)<<12|(c)<<6)
  51. #define OPCODE2(a,b) ((a)<<26|(b)<<1)
  52. #define OPCODE3(a,b) ((a)<<26|(b)<<2)
  53. #define OPCODE4(a) ((a)<<26)
  54. #define OPCODE1_MASK OPCODE1(0x3f,1,0xf)
  55. #define OPCODE2_MASK OPCODE2(0x3f,1)
  56. #define OPCODE3_MASK OPCODE3(0x3f,1)
  57. #define OPCODE4_MASK OPCODE4(0x3f)
  58. /* skip LDB - never unaligned (index) */
  59. #define OPCODE_LDH_I OPCODE1(0x03,0,0x1)
  60. #define OPCODE_LDW_I OPCODE1(0x03,0,0x2)
  61. #define OPCODE_LDD_I OPCODE1(0x03,0,0x3)
  62. #define OPCODE_LDDA_I OPCODE1(0x03,0,0x4)
  63. #define OPCODE_LDCD_I OPCODE1(0x03,0,0x5)
  64. #define OPCODE_LDWA_I OPCODE1(0x03,0,0x6)
  65. #define OPCODE_LDCW_I OPCODE1(0x03,0,0x7)
  66. /* skip LDB - never unaligned (short) */
  67. #define OPCODE_LDH_S OPCODE1(0x03,1,0x1)
  68. #define OPCODE_LDW_S OPCODE1(0x03,1,0x2)
  69. #define OPCODE_LDD_S OPCODE1(0x03,1,0x3)
  70. #define OPCODE_LDDA_S OPCODE1(0x03,1,0x4)
  71. #define OPCODE_LDCD_S OPCODE1(0x03,1,0x5)
  72. #define OPCODE_LDWA_S OPCODE1(0x03,1,0x6)
  73. #define OPCODE_LDCW_S OPCODE1(0x03,1,0x7)
  74. /* skip STB - never unaligned */
  75. #define OPCODE_STH OPCODE1(0x03,1,0x9)
  76. #define OPCODE_STW OPCODE1(0x03,1,0xa)
  77. #define OPCODE_STD OPCODE1(0x03,1,0xb)
  78. /* skip STBY - never unaligned */
  79. /* skip STDBY - never unaligned */
  80. #define OPCODE_STWA OPCODE1(0x03,1,0xe)
  81. #define OPCODE_STDA OPCODE1(0x03,1,0xf)
  82. #define OPCODE_FLDWX OPCODE1(0x09,0,0x0)
  83. #define OPCODE_FLDWXR OPCODE1(0x09,0,0x1)
  84. #define OPCODE_FSTWX OPCODE1(0x09,0,0x8)
  85. #define OPCODE_FSTWXR OPCODE1(0x09,0,0x9)
  86. #define OPCODE_FLDWS OPCODE1(0x09,1,0x0)
  87. #define OPCODE_FLDWSR OPCODE1(0x09,1,0x1)
  88. #define OPCODE_FSTWS OPCODE1(0x09,1,0x8)
  89. #define OPCODE_FSTWSR OPCODE1(0x09,1,0x9)
  90. #define OPCODE_FLDDX OPCODE1(0x0b,0,0x0)
  91. #define OPCODE_FSTDX OPCODE1(0x0b,0,0x8)
  92. #define OPCODE_FLDDS OPCODE1(0x0b,1,0x0)
  93. #define OPCODE_FSTDS OPCODE1(0x0b,1,0x8)
  94. #define OPCODE_LDD_L OPCODE2(0x14,0)
  95. #define OPCODE_FLDD_L OPCODE2(0x14,1)
  96. #define OPCODE_STD_L OPCODE2(0x1c,0)
  97. #define OPCODE_FSTD_L OPCODE2(0x1c,1)
  98. #define OPCODE_LDW_M OPCODE3(0x17,1)
  99. #define OPCODE_FLDW_L OPCODE3(0x17,0)
  100. #define OPCODE_FSTW_L OPCODE3(0x1f,0)
  101. #define OPCODE_STW_M OPCODE3(0x1f,1)
  102. #define OPCODE_LDH_L OPCODE4(0x11)
  103. #define OPCODE_LDW_L OPCODE4(0x12)
  104. #define OPCODE_LDWM OPCODE4(0x13)
  105. #define OPCODE_STH_L OPCODE4(0x19)
  106. #define OPCODE_STW_L OPCODE4(0x1A)
  107. #define OPCODE_STWM OPCODE4(0x1B)
  108. #define MAJOR_OP(i) (((i)>>26)&0x3f)
  109. #define R1(i) (((i)>>21)&0x1f)
  110. #define R2(i) (((i)>>16)&0x1f)
  111. #define R3(i) ((i)&0x1f)
  112. #define FR3(i) ((((i)<<1)&0x1f)|(((i)>>6)&1))
  113. #define IM(i,n) (((i)>>1&((1<<(n-1))-1))|((i)&1?((0-1L)<<(n-1)):0))
  114. #define IM5_2(i) IM((i)>>16,5)
  115. #define IM5_3(i) IM((i),5)
  116. #define IM14(i) IM((i),14)
  117. #define ERR_NOTHANDLED -1
  118. #define ERR_PAGEFAULT -2
  119. int unaligned_enabled __read_mostly = 1;
  120. static int emulate_ldh(struct pt_regs *regs, int toreg)
  121. {
  122. unsigned long saddr = regs->ior;
  123. unsigned long val = 0;
  124. int ret;
  125. DPRINTF("load " RFMT ":" RFMT " to r%d for 2 bytes\n",
  126. regs->isr, regs->ior, toreg);
  127. __asm__ __volatile__ (
  128. " mtsp %4, %%sr1\n"
  129. "1: ldbs 0(%%sr1,%3), %%r20\n"
  130. "2: ldbs 1(%%sr1,%3), %0\n"
  131. " depw %%r20, 23, 24, %0\n"
  132. " copy %%r0, %1\n"
  133. "3: \n"
  134. " .section .fixup,\"ax\"\n"
  135. "4: ldi -2, %1\n"
  136. FIXUP_BRANCH(3b)
  137. " .previous\n"
  138. ASM_EXCEPTIONTABLE_ENTRY(1b, 4b)
  139. ASM_EXCEPTIONTABLE_ENTRY(2b, 4b)
  140. : "=r" (val), "=r" (ret)
  141. : "0" (val), "r" (saddr), "r" (regs->isr)
  142. : "r20", FIXUP_BRANCH_CLOBBER );
  143. DPRINTF("val = 0x" RFMT "\n", val);
  144. if (toreg)
  145. regs->gr[toreg] = val;
  146. return ret;
  147. }
  148. static int emulate_ldw(struct pt_regs *regs, int toreg, int flop)
  149. {
  150. unsigned long saddr = regs->ior;
  151. unsigned long val = 0;
  152. int ret;
  153. DPRINTF("load " RFMT ":" RFMT " to r%d for 4 bytes\n",
  154. regs->isr, regs->ior, toreg);
  155. __asm__ __volatile__ (
  156. " zdep %3,28,2,%%r19\n" /* r19=(ofs&3)*8 */
  157. " mtsp %4, %%sr1\n"
  158. " depw %%r0,31,2,%3\n"
  159. "1: ldw 0(%%sr1,%3),%0\n"
  160. "2: ldw 4(%%sr1,%3),%%r20\n"
  161. " subi 32,%%r19,%%r19\n"
  162. " mtctl %%r19,11\n"
  163. " vshd %0,%%r20,%0\n"
  164. " copy %%r0, %1\n"
  165. "3: \n"
  166. " .section .fixup,\"ax\"\n"
  167. "4: ldi -2, %1\n"
  168. FIXUP_BRANCH(3b)
  169. " .previous\n"
  170. ASM_EXCEPTIONTABLE_ENTRY(1b, 4b)
  171. ASM_EXCEPTIONTABLE_ENTRY(2b, 4b)
  172. : "=r" (val), "=r" (ret)
  173. : "0" (val), "r" (saddr), "r" (regs->isr)
  174. : "r19", "r20", FIXUP_BRANCH_CLOBBER );
  175. DPRINTF("val = 0x" RFMT "\n", val);
  176. if (flop)
  177. ((__u32*)(regs->fr))[toreg] = val;
  178. else if (toreg)
  179. regs->gr[toreg] = val;
  180. return ret;
  181. }
  182. static int emulate_ldd(struct pt_regs *regs, int toreg, int flop)
  183. {
  184. unsigned long saddr = regs->ior;
  185. __u64 val = 0;
  186. int ret;
  187. DPRINTF("load " RFMT ":" RFMT " to r%d for 8 bytes\n",
  188. regs->isr, regs->ior, toreg);
  189. #ifdef CONFIG_PA20
  190. #ifndef CONFIG_64BIT
  191. if (!flop)
  192. return -1;
  193. #endif
  194. __asm__ __volatile__ (
  195. " depd,z %3,60,3,%%r19\n" /* r19=(ofs&7)*8 */
  196. " mtsp %4, %%sr1\n"
  197. " depd %%r0,63,3,%3\n"
  198. "1: ldd 0(%%sr1,%3),%0\n"
  199. "2: ldd 8(%%sr1,%3),%%r20\n"
  200. " subi 64,%%r19,%%r19\n"
  201. " mtsar %%r19\n"
  202. " shrpd %0,%%r20,%%sar,%0\n"
  203. " copy %%r0, %1\n"
  204. "3: \n"
  205. " .section .fixup,\"ax\"\n"
  206. "4: ldi -2, %1\n"
  207. FIXUP_BRANCH(3b)
  208. " .previous\n"
  209. ASM_EXCEPTIONTABLE_ENTRY(1b,4b)
  210. ASM_EXCEPTIONTABLE_ENTRY(2b,4b)
  211. : "=r" (val), "=r" (ret)
  212. : "0" (val), "r" (saddr), "r" (regs->isr)
  213. : "r19", "r20", FIXUP_BRANCH_CLOBBER );
  214. #else
  215. {
  216. unsigned long valh=0,vall=0;
  217. __asm__ __volatile__ (
  218. " zdep %5,29,2,%%r19\n" /* r19=(ofs&3)*8 */
  219. " mtsp %6, %%sr1\n"
  220. " dep %%r0,31,2,%5\n"
  221. "1: ldw 0(%%sr1,%5),%0\n"
  222. "2: ldw 4(%%sr1,%5),%1\n"
  223. "3: ldw 8(%%sr1,%5),%%r20\n"
  224. " subi 32,%%r19,%%r19\n"
  225. " mtsar %%r19\n"
  226. " vshd %0,%1,%0\n"
  227. " vshd %1,%%r20,%1\n"
  228. " copy %%r0, %2\n"
  229. "4: \n"
  230. " .section .fixup,\"ax\"\n"
  231. "5: ldi -2, %2\n"
  232. FIXUP_BRANCH(4b)
  233. " .previous\n"
  234. ASM_EXCEPTIONTABLE_ENTRY(1b,5b)
  235. ASM_EXCEPTIONTABLE_ENTRY(2b,5b)
  236. ASM_EXCEPTIONTABLE_ENTRY(3b,5b)
  237. : "=r" (valh), "=r" (vall), "=r" (ret)
  238. : "0" (valh), "1" (vall), "r" (saddr), "r" (regs->isr)
  239. : "r19", "r20", FIXUP_BRANCH_CLOBBER );
  240. val=((__u64)valh<<32)|(__u64)vall;
  241. }
  242. #endif
  243. DPRINTF("val = 0x%llx\n", val);
  244. if (flop)
  245. regs->fr[toreg] = val;
  246. else if (toreg)
  247. regs->gr[toreg] = val;
  248. return ret;
  249. }
  250. static int emulate_sth(struct pt_regs *regs, int frreg)
  251. {
  252. unsigned long val = regs->gr[frreg];
  253. int ret;
  254. if (!frreg)
  255. val = 0;
  256. DPRINTF("store r%d (0x" RFMT ") to " RFMT ":" RFMT " for 2 bytes\n", frreg,
  257. val, regs->isr, regs->ior);
  258. __asm__ __volatile__ (
  259. " mtsp %3, %%sr1\n"
  260. " extrw,u %1, 23, 8, %%r19\n"
  261. "1: stb %1, 1(%%sr1, %2)\n"
  262. "2: stb %%r19, 0(%%sr1, %2)\n"
  263. " copy %%r0, %0\n"
  264. "3: \n"
  265. " .section .fixup,\"ax\"\n"
  266. "4: ldi -2, %0\n"
  267. FIXUP_BRANCH(3b)
  268. " .previous\n"
  269. ASM_EXCEPTIONTABLE_ENTRY(1b,4b)
  270. ASM_EXCEPTIONTABLE_ENTRY(2b,4b)
  271. : "=r" (ret)
  272. : "r" (val), "r" (regs->ior), "r" (regs->isr)
  273. : "r19", FIXUP_BRANCH_CLOBBER );
  274. return ret;
  275. }
  276. static int emulate_stw(struct pt_regs *regs, int frreg, int flop)
  277. {
  278. unsigned long val;
  279. int ret;
  280. if (flop)
  281. val = ((__u32*)(regs->fr))[frreg];
  282. else if (frreg)
  283. val = regs->gr[frreg];
  284. else
  285. val = 0;
  286. DPRINTF("store r%d (0x" RFMT ") to " RFMT ":" RFMT " for 4 bytes\n", frreg,
  287. val, regs->isr, regs->ior);
  288. __asm__ __volatile__ (
  289. " mtsp %3, %%sr1\n"
  290. " zdep %2, 28, 2, %%r19\n"
  291. " dep %%r0, 31, 2, %2\n"
  292. " mtsar %%r19\n"
  293. " depwi,z -2, %%sar, 32, %%r19\n"
  294. "1: ldw 0(%%sr1,%2),%%r20\n"
  295. "2: ldw 4(%%sr1,%2),%%r21\n"
  296. " vshd %%r0, %1, %%r22\n"
  297. " vshd %1, %%r0, %%r1\n"
  298. " and %%r20, %%r19, %%r20\n"
  299. " andcm %%r21, %%r19, %%r21\n"
  300. " or %%r22, %%r20, %%r20\n"
  301. " or %%r1, %%r21, %%r21\n"
  302. " stw %%r20,0(%%sr1,%2)\n"
  303. " stw %%r21,4(%%sr1,%2)\n"
  304. " copy %%r0, %0\n"
  305. "3: \n"
  306. " .section .fixup,\"ax\"\n"
  307. "4: ldi -2, %0\n"
  308. FIXUP_BRANCH(3b)
  309. " .previous\n"
  310. ASM_EXCEPTIONTABLE_ENTRY(1b,4b)
  311. ASM_EXCEPTIONTABLE_ENTRY(2b,4b)
  312. : "=r" (ret)
  313. : "r" (val), "r" (regs->ior), "r" (regs->isr)
  314. : "r19", "r20", "r21", "r22", "r1", FIXUP_BRANCH_CLOBBER );
  315. return 0;
  316. }
  317. static int emulate_std(struct pt_regs *regs, int frreg, int flop)
  318. {
  319. __u64 val;
  320. int ret;
  321. if (flop)
  322. val = regs->fr[frreg];
  323. else if (frreg)
  324. val = regs->gr[frreg];
  325. else
  326. val = 0;
  327. DPRINTF("store r%d (0x%016llx) to " RFMT ":" RFMT " for 8 bytes\n", frreg,
  328. val, regs->isr, regs->ior);
  329. #ifdef CONFIG_PA20
  330. #ifndef CONFIG_64BIT
  331. if (!flop)
  332. return -1;
  333. #endif
  334. __asm__ __volatile__ (
  335. " mtsp %3, %%sr1\n"
  336. " depd,z %2, 60, 3, %%r19\n"
  337. " depd %%r0, 63, 3, %2\n"
  338. " mtsar %%r19\n"
  339. " depdi,z -2, %%sar, 64, %%r19\n"
  340. "1: ldd 0(%%sr1,%2),%%r20\n"
  341. "2: ldd 8(%%sr1,%2),%%r21\n"
  342. " shrpd %%r0, %1, %%sar, %%r22\n"
  343. " shrpd %1, %%r0, %%sar, %%r1\n"
  344. " and %%r20, %%r19, %%r20\n"
  345. " andcm %%r21, %%r19, %%r21\n"
  346. " or %%r22, %%r20, %%r20\n"
  347. " or %%r1, %%r21, %%r21\n"
  348. "3: std %%r20,0(%%sr1,%2)\n"
  349. "4: std %%r21,8(%%sr1,%2)\n"
  350. " copy %%r0, %0\n"
  351. "5: \n"
  352. " .section .fixup,\"ax\"\n"
  353. "6: ldi -2, %0\n"
  354. FIXUP_BRANCH(5b)
  355. " .previous\n"
  356. ASM_EXCEPTIONTABLE_ENTRY(1b,6b)
  357. ASM_EXCEPTIONTABLE_ENTRY(2b,6b)
  358. ASM_EXCEPTIONTABLE_ENTRY(3b,6b)
  359. ASM_EXCEPTIONTABLE_ENTRY(4b,6b)
  360. : "=r" (ret)
  361. : "r" (val), "r" (regs->ior), "r" (regs->isr)
  362. : "r19", "r20", "r21", "r22", "r1", FIXUP_BRANCH_CLOBBER );
  363. #else
  364. {
  365. unsigned long valh=(val>>32),vall=(val&0xffffffffl);
  366. __asm__ __volatile__ (
  367. " mtsp %4, %%sr1\n"
  368. " zdep %2, 29, 2, %%r19\n"
  369. " dep %%r0, 31, 2, %2\n"
  370. " mtsar %%r19\n"
  371. " zvdepi -2, 32, %%r19\n"
  372. "1: ldw 0(%%sr1,%3),%%r20\n"
  373. "2: ldw 8(%%sr1,%3),%%r21\n"
  374. " vshd %1, %2, %%r1\n"
  375. " vshd %%r0, %1, %1\n"
  376. " vshd %2, %%r0, %2\n"
  377. " and %%r20, %%r19, %%r20\n"
  378. " andcm %%r21, %%r19, %%r21\n"
  379. " or %1, %%r20, %1\n"
  380. " or %2, %%r21, %2\n"
  381. "3: stw %1,0(%%sr1,%1)\n"
  382. "4: stw %%r1,4(%%sr1,%3)\n"
  383. "5: stw %2,8(%%sr1,%3)\n"
  384. " copy %%r0, %0\n"
  385. "6: \n"
  386. " .section .fixup,\"ax\"\n"
  387. "7: ldi -2, %0\n"
  388. FIXUP_BRANCH(6b)
  389. " .previous\n"
  390. ASM_EXCEPTIONTABLE_ENTRY(1b,7b)
  391. ASM_EXCEPTIONTABLE_ENTRY(2b,7b)
  392. ASM_EXCEPTIONTABLE_ENTRY(3b,7b)
  393. ASM_EXCEPTIONTABLE_ENTRY(4b,7b)
  394. ASM_EXCEPTIONTABLE_ENTRY(5b,7b)
  395. : "=r" (ret)
  396. : "r" (valh), "r" (vall), "r" (regs->ior), "r" (regs->isr)
  397. : "r19", "r20", "r21", "r1", FIXUP_BRANCH_CLOBBER );
  398. }
  399. #endif
  400. return ret;
  401. }
  402. void handle_unaligned(struct pt_regs *regs)
  403. {
  404. static DEFINE_RATELIMIT_STATE(ratelimit, 5 * HZ, 5);
  405. unsigned long newbase = R1(regs->iir)?regs->gr[R1(regs->iir)]:0;
  406. int modify = 0;
  407. int ret = ERR_NOTHANDLED;
  408. struct siginfo si;
  409. register int flop=0; /* true if this is a flop */
  410. __inc_irq_stat(irq_unaligned_count);
  411. /* log a message with pacing */
  412. if (user_mode(regs)) {
  413. if (current->thread.flags & PARISC_UAC_SIGBUS) {
  414. goto force_sigbus;
  415. }
  416. if (!(current->thread.flags & PARISC_UAC_NOPRINT) &&
  417. __ratelimit(&ratelimit)) {
  418. char buf[256];
  419. sprintf(buf, "%s(%d): unaligned access to 0x" RFMT " at ip=0x" RFMT "\n",
  420. current->comm, task_pid_nr(current), regs->ior, regs->iaoq[0]);
  421. printk(KERN_WARNING "%s", buf);
  422. #ifdef DEBUG_UNALIGNED
  423. show_regs(regs);
  424. #endif
  425. }
  426. if (!unaligned_enabled)
  427. goto force_sigbus;
  428. }
  429. /* handle modification - OK, it's ugly, see the instruction manual */
  430. switch (MAJOR_OP(regs->iir))
  431. {
  432. case 0x03:
  433. case 0x09:
  434. case 0x0b:
  435. if (regs->iir&0x20)
  436. {
  437. modify = 1;
  438. if (regs->iir&0x1000) /* short loads */
  439. if (regs->iir&0x200)
  440. newbase += IM5_3(regs->iir);
  441. else
  442. newbase += IM5_2(regs->iir);
  443. else if (regs->iir&0x2000) /* scaled indexed */
  444. {
  445. int shift=0;
  446. switch (regs->iir & OPCODE1_MASK)
  447. {
  448. case OPCODE_LDH_I:
  449. shift= 1; break;
  450. case OPCODE_LDW_I:
  451. shift= 2; break;
  452. case OPCODE_LDD_I:
  453. case OPCODE_LDDA_I:
  454. shift= 3; break;
  455. }
  456. newbase += (R2(regs->iir)?regs->gr[R2(regs->iir)]:0)<<shift;
  457. } else /* simple indexed */
  458. newbase += (R2(regs->iir)?regs->gr[R2(regs->iir)]:0);
  459. }
  460. break;
  461. case 0x13:
  462. case 0x1b:
  463. modify = 1;
  464. newbase += IM14(regs->iir);
  465. break;
  466. case 0x14:
  467. case 0x1c:
  468. if (regs->iir&8)
  469. {
  470. modify = 1;
  471. newbase += IM14(regs->iir&~0xe);
  472. }
  473. break;
  474. case 0x16:
  475. case 0x1e:
  476. modify = 1;
  477. newbase += IM14(regs->iir&6);
  478. break;
  479. case 0x17:
  480. case 0x1f:
  481. if (regs->iir&4)
  482. {
  483. modify = 1;
  484. newbase += IM14(regs->iir&~4);
  485. }
  486. break;
  487. }
  488. /* TODO: make this cleaner... */
  489. switch (regs->iir & OPCODE1_MASK)
  490. {
  491. case OPCODE_LDH_I:
  492. case OPCODE_LDH_S:
  493. ret = emulate_ldh(regs, R3(regs->iir));
  494. break;
  495. case OPCODE_LDW_I:
  496. case OPCODE_LDWA_I:
  497. case OPCODE_LDW_S:
  498. case OPCODE_LDWA_S:
  499. ret = emulate_ldw(regs, R3(regs->iir),0);
  500. break;
  501. case OPCODE_STH:
  502. ret = emulate_sth(regs, R2(regs->iir));
  503. break;
  504. case OPCODE_STW:
  505. case OPCODE_STWA:
  506. ret = emulate_stw(regs, R2(regs->iir),0);
  507. break;
  508. #ifdef CONFIG_PA20
  509. case OPCODE_LDD_I:
  510. case OPCODE_LDDA_I:
  511. case OPCODE_LDD_S:
  512. case OPCODE_LDDA_S:
  513. ret = emulate_ldd(regs, R3(regs->iir),0);
  514. break;
  515. case OPCODE_STD:
  516. case OPCODE_STDA:
  517. ret = emulate_std(regs, R2(regs->iir),0);
  518. break;
  519. #endif
  520. case OPCODE_FLDWX:
  521. case OPCODE_FLDWS:
  522. case OPCODE_FLDWXR:
  523. case OPCODE_FLDWSR:
  524. flop=1;
  525. ret = emulate_ldw(regs,FR3(regs->iir),1);
  526. break;
  527. case OPCODE_FLDDX:
  528. case OPCODE_FLDDS:
  529. flop=1;
  530. ret = emulate_ldd(regs,R3(regs->iir),1);
  531. break;
  532. case OPCODE_FSTWX:
  533. case OPCODE_FSTWS:
  534. case OPCODE_FSTWXR:
  535. case OPCODE_FSTWSR:
  536. flop=1;
  537. ret = emulate_stw(regs,FR3(regs->iir),1);
  538. break;
  539. case OPCODE_FSTDX:
  540. case OPCODE_FSTDS:
  541. flop=1;
  542. ret = emulate_std(regs,R3(regs->iir),1);
  543. break;
  544. case OPCODE_LDCD_I:
  545. case OPCODE_LDCW_I:
  546. case OPCODE_LDCD_S:
  547. case OPCODE_LDCW_S:
  548. ret = ERR_NOTHANDLED; /* "undefined", but lets kill them. */
  549. break;
  550. }
  551. #ifdef CONFIG_PA20
  552. switch (regs->iir & OPCODE2_MASK)
  553. {
  554. case OPCODE_FLDD_L:
  555. flop=1;
  556. ret = emulate_ldd(regs,R2(regs->iir),1);
  557. break;
  558. case OPCODE_FSTD_L:
  559. flop=1;
  560. ret = emulate_std(regs, R2(regs->iir),1);
  561. break;
  562. case OPCODE_LDD_L:
  563. ret = emulate_ldd(regs, R2(regs->iir),0);
  564. break;
  565. case OPCODE_STD_L:
  566. ret = emulate_std(regs, R2(regs->iir),0);
  567. break;
  568. }
  569. #endif
  570. switch (regs->iir & OPCODE3_MASK)
  571. {
  572. case OPCODE_FLDW_L:
  573. flop=1;
  574. ret = emulate_ldw(regs, R2(regs->iir),0);
  575. break;
  576. case OPCODE_LDW_M:
  577. ret = emulate_ldw(regs, R2(regs->iir),1);
  578. break;
  579. case OPCODE_FSTW_L:
  580. flop=1;
  581. ret = emulate_stw(regs, R2(regs->iir),1);
  582. break;
  583. case OPCODE_STW_M:
  584. ret = emulate_stw(regs, R2(regs->iir),0);
  585. break;
  586. }
  587. switch (regs->iir & OPCODE4_MASK)
  588. {
  589. case OPCODE_LDH_L:
  590. ret = emulate_ldh(regs, R2(regs->iir));
  591. break;
  592. case OPCODE_LDW_L:
  593. case OPCODE_LDWM:
  594. ret = emulate_ldw(regs, R2(regs->iir),0);
  595. break;
  596. case OPCODE_STH_L:
  597. ret = emulate_sth(regs, R2(regs->iir));
  598. break;
  599. case OPCODE_STW_L:
  600. case OPCODE_STWM:
  601. ret = emulate_stw(regs, R2(regs->iir),0);
  602. break;
  603. }
  604. if (ret == 0 && modify && R1(regs->iir))
  605. regs->gr[R1(regs->iir)] = newbase;
  606. if (ret == ERR_NOTHANDLED)
  607. printk(KERN_CRIT "Not-handled unaligned insn 0x%08lx\n", regs->iir);
  608. DPRINTF("ret = %d\n", ret);
  609. if (ret)
  610. {
  611. /*
  612. * The unaligned handler failed.
  613. * If we were called by __get_user() or __put_user() jump
  614. * to it's exception fixup handler instead of crashing.
  615. */
  616. if (!user_mode(regs) && fixup_exception(regs))
  617. return;
  618. printk(KERN_CRIT "Unaligned handler failed, ret = %d\n", ret);
  619. die_if_kernel("Unaligned data reference", regs, 28);
  620. if (ret == ERR_PAGEFAULT)
  621. {
  622. si.si_signo = SIGSEGV;
  623. si.si_errno = 0;
  624. si.si_code = SEGV_MAPERR;
  625. si.si_addr = (void __user *)regs->ior;
  626. force_sig_info(SIGSEGV, &si, current);
  627. }
  628. else
  629. {
  630. force_sigbus:
  631. /* couldn't handle it ... */
  632. si.si_signo = SIGBUS;
  633. si.si_errno = 0;
  634. si.si_code = BUS_ADRALN;
  635. si.si_addr = (void __user *)regs->ior;
  636. force_sig_info(SIGBUS, &si, current);
  637. }
  638. return;
  639. }
  640. /* else we handled it, let life go on. */
  641. regs->gr[0]|=PSW_N;
  642. }
  643. /*
  644. * NB: check_unaligned() is only used for PCXS processors right
  645. * now, so we only check for PA1.1 encodings at this point.
  646. */
  647. int
  648. check_unaligned(struct pt_regs *regs)
  649. {
  650. unsigned long align_mask;
  651. /* Get alignment mask */
  652. align_mask = 0UL;
  653. switch (regs->iir & OPCODE1_MASK) {
  654. case OPCODE_LDH_I:
  655. case OPCODE_LDH_S:
  656. case OPCODE_STH:
  657. align_mask = 1UL;
  658. break;
  659. case OPCODE_LDW_I:
  660. case OPCODE_LDWA_I:
  661. case OPCODE_LDW_S:
  662. case OPCODE_LDWA_S:
  663. case OPCODE_STW:
  664. case OPCODE_STWA:
  665. align_mask = 3UL;
  666. break;
  667. default:
  668. switch (regs->iir & OPCODE4_MASK) {
  669. case OPCODE_LDH_L:
  670. case OPCODE_STH_L:
  671. align_mask = 1UL;
  672. break;
  673. case OPCODE_LDW_L:
  674. case OPCODE_LDWM:
  675. case OPCODE_STW_L:
  676. case OPCODE_STWM:
  677. align_mask = 3UL;
  678. break;
  679. }
  680. break;
  681. }
  682. return (int)(regs->ior & align_mask);
  683. }