ldstfp.S 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373
  1. /*
  2. * Floating-point, VMX/Altivec and VSX loads and stores
  3. * for use in instruction emulation.
  4. *
  5. * Copyright 2010 Paul Mackerras, IBM Corp. <paulus@au1.ibm.com>
  6. *
  7. * This program is free software; you can redistribute it and/or
  8. * modify it under the terms of the GNU General Public License
  9. * as published by the Free Software Foundation; either version
  10. * 2 of the License, or (at your option) any later version.
  11. */
  12. #include <asm/processor.h>
  13. #include <asm/ppc_asm.h>
  14. #include <asm/ppc-opcode.h>
  15. #include <asm/reg.h>
  16. #include <asm/asm-offsets.h>
  17. #include <linux/errno.h>
  18. #ifdef CONFIG_PPC_FPU
  19. #define STKFRM (PPC_MIN_STKFRM + 16)
  20. .macro inst32 op
  21. reg = 0
  22. .rept 32
  23. 20: \op reg,0,r4
  24. b 3f
  25. EX_TABLE(20b,99f)
  26. reg = reg + 1
  27. .endr
  28. .endm
  29. /* Get the contents of frN into fr0; N is in r3. */
  30. _GLOBAL(get_fpr)
  31. mflr r0
  32. rlwinm r3,r3,3,0xf8
  33. bcl 20,31,1f
  34. blr /* fr0 is already in fr0 */
  35. nop
  36. reg = 1
  37. .rept 31
  38. fmr fr0,reg
  39. blr
  40. reg = reg + 1
  41. .endr
  42. 1: mflr r5
  43. add r5,r3,r5
  44. mtctr r5
  45. mtlr r0
  46. bctr
  47. /* Put the contents of fr0 into frN; N is in r3. */
  48. _GLOBAL(put_fpr)
  49. mflr r0
  50. rlwinm r3,r3,3,0xf8
  51. bcl 20,31,1f
  52. blr /* fr0 is already in fr0 */
  53. nop
  54. reg = 1
  55. .rept 31
  56. fmr reg,fr0
  57. blr
  58. reg = reg + 1
  59. .endr
  60. 1: mflr r5
  61. add r5,r3,r5
  62. mtctr r5
  63. mtlr r0
  64. bctr
  65. /* Load FP reg N from float at *p. N is in r3, p in r4. */
  66. _GLOBAL(do_lfs)
  67. PPC_STLU r1,-STKFRM(r1)
  68. mflr r0
  69. PPC_STL r0,STKFRM+PPC_LR_STKOFF(r1)
  70. mfmsr r6
  71. ori r7,r6,MSR_FP
  72. cmpwi cr7,r3,0
  73. MTMSRD(r7)
  74. isync
  75. beq cr7,1f
  76. stfd fr0,STKFRM-16(r1)
  77. 1: li r9,-EFAULT
  78. 2: lfs fr0,0(r4)
  79. li r9,0
  80. 3: bl put_fpr
  81. beq cr7,4f
  82. lfd fr0,STKFRM-16(r1)
  83. 4: PPC_LL r0,STKFRM+PPC_LR_STKOFF(r1)
  84. mtlr r0
  85. MTMSRD(r6)
  86. isync
  87. mr r3,r9
  88. addi r1,r1,STKFRM
  89. blr
  90. EX_TABLE(2b,3b)
  91. /* Load FP reg N from double at *p. N is in r3, p in r4. */
  92. _GLOBAL(do_lfd)
  93. PPC_STLU r1,-STKFRM(r1)
  94. mflr r0
  95. PPC_STL r0,STKFRM+PPC_LR_STKOFF(r1)
  96. mfmsr r6
  97. ori r7,r6,MSR_FP
  98. cmpwi cr7,r3,0
  99. MTMSRD(r7)
  100. isync
  101. beq cr7,1f
  102. stfd fr0,STKFRM-16(r1)
  103. 1: li r9,-EFAULT
  104. 2: lfd fr0,0(r4)
  105. li r9,0
  106. 3: beq cr7,4f
  107. bl put_fpr
  108. lfd fr0,STKFRM-16(r1)
  109. 4: PPC_LL r0,STKFRM+PPC_LR_STKOFF(r1)
  110. mtlr r0
  111. MTMSRD(r6)
  112. isync
  113. mr r3,r9
  114. addi r1,r1,STKFRM
  115. blr
  116. EX_TABLE(2b,3b)
  117. /* Store FP reg N to float at *p. N is in r3, p in r4. */
  118. _GLOBAL(do_stfs)
  119. PPC_STLU r1,-STKFRM(r1)
  120. mflr r0
  121. PPC_STL r0,STKFRM+PPC_LR_STKOFF(r1)
  122. mfmsr r6
  123. ori r7,r6,MSR_FP
  124. cmpwi cr7,r3,0
  125. MTMSRD(r7)
  126. isync
  127. beq cr7,1f
  128. stfd fr0,STKFRM-16(r1)
  129. bl get_fpr
  130. 1: li r9,-EFAULT
  131. 2: stfs fr0,0(r4)
  132. li r9,0
  133. 3: beq cr7,4f
  134. lfd fr0,STKFRM-16(r1)
  135. 4: PPC_LL r0,STKFRM+PPC_LR_STKOFF(r1)
  136. mtlr r0
  137. MTMSRD(r6)
  138. isync
  139. mr r3,r9
  140. addi r1,r1,STKFRM
  141. blr
  142. EX_TABLE(2b,3b)
  143. /* Store FP reg N to double at *p. N is in r3, p in r4. */
  144. _GLOBAL(do_stfd)
  145. PPC_STLU r1,-STKFRM(r1)
  146. mflr r0
  147. PPC_STL r0,STKFRM+PPC_LR_STKOFF(r1)
  148. mfmsr r6
  149. ori r7,r6,MSR_FP
  150. cmpwi cr7,r3,0
  151. MTMSRD(r7)
  152. isync
  153. beq cr7,1f
  154. stfd fr0,STKFRM-16(r1)
  155. bl get_fpr
  156. 1: li r9,-EFAULT
  157. 2: stfd fr0,0(r4)
  158. li r9,0
  159. 3: beq cr7,4f
  160. lfd fr0,STKFRM-16(r1)
  161. 4: PPC_LL r0,STKFRM+PPC_LR_STKOFF(r1)
  162. mtlr r0
  163. MTMSRD(r6)
  164. isync
  165. mr r3,r9
  166. addi r1,r1,STKFRM
  167. blr
  168. EX_TABLE(2b,3b)
  169. #ifdef CONFIG_ALTIVEC
  170. /* Get the contents of vrN into v0; N is in r3. */
  171. _GLOBAL(get_vr)
  172. mflr r0
  173. rlwinm r3,r3,3,0xf8
  174. bcl 20,31,1f
  175. blr /* v0 is already in v0 */
  176. nop
  177. reg = 1
  178. .rept 31
  179. vor v0,reg,reg /* assembler doesn't know vmr? */
  180. blr
  181. reg = reg + 1
  182. .endr
  183. 1: mflr r5
  184. add r5,r3,r5
  185. mtctr r5
  186. mtlr r0
  187. bctr
  188. /* Put the contents of v0 into vrN; N is in r3. */
  189. _GLOBAL(put_vr)
  190. mflr r0
  191. rlwinm r3,r3,3,0xf8
  192. bcl 20,31,1f
  193. blr /* v0 is already in v0 */
  194. nop
  195. reg = 1
  196. .rept 31
  197. vor reg,v0,v0
  198. blr
  199. reg = reg + 1
  200. .endr
  201. 1: mflr r5
  202. add r5,r3,r5
  203. mtctr r5
  204. mtlr r0
  205. bctr
  206. /* Load vector reg N from *p. N is in r3, p in r4. */
  207. _GLOBAL(do_lvx)
  208. PPC_STLU r1,-STKFRM(r1)
  209. mflr r0
  210. PPC_STL r0,STKFRM+PPC_LR_STKOFF(r1)
  211. mfmsr r6
  212. oris r7,r6,MSR_VEC@h
  213. cmpwi cr7,r3,0
  214. li r8,STKFRM-16
  215. MTMSRD(r7)
  216. isync
  217. beq cr7,1f
  218. stvx v0,r1,r8
  219. 1: li r9,-EFAULT
  220. 2: lvx v0,0,r4
  221. li r9,0
  222. 3: beq cr7,4f
  223. bl put_vr
  224. lvx v0,r1,r8
  225. 4: PPC_LL r0,STKFRM+PPC_LR_STKOFF(r1)
  226. mtlr r0
  227. MTMSRD(r6)
  228. isync
  229. mr r3,r9
  230. addi r1,r1,STKFRM
  231. blr
  232. EX_TABLE(2b,3b)
  233. /* Store vector reg N to *p. N is in r3, p in r4. */
  234. _GLOBAL(do_stvx)
  235. PPC_STLU r1,-STKFRM(r1)
  236. mflr r0
  237. PPC_STL r0,STKFRM+PPC_LR_STKOFF(r1)
  238. mfmsr r6
  239. oris r7,r6,MSR_VEC@h
  240. cmpwi cr7,r3,0
  241. li r8,STKFRM-16
  242. MTMSRD(r7)
  243. isync
  244. beq cr7,1f
  245. stvx v0,r1,r8
  246. bl get_vr
  247. 1: li r9,-EFAULT
  248. 2: stvx v0,0,r4
  249. li r9,0
  250. 3: beq cr7,4f
  251. lvx v0,r1,r8
  252. 4: PPC_LL r0,STKFRM+PPC_LR_STKOFF(r1)
  253. mtlr r0
  254. MTMSRD(r6)
  255. isync
  256. mr r3,r9
  257. addi r1,r1,STKFRM
  258. blr
  259. EX_TABLE(2b,3b)
  260. #endif /* CONFIG_ALTIVEC */
  261. #ifdef CONFIG_VSX
  262. /* Get the contents of vsN into vs0; N is in r3. */
  263. _GLOBAL(get_vsr)
  264. mflr r0
  265. rlwinm r3,r3,3,0x1f8
  266. bcl 20,31,1f
  267. blr /* vs0 is already in vs0 */
  268. nop
  269. reg = 1
  270. .rept 63
  271. XXLOR(0,reg,reg)
  272. blr
  273. reg = reg + 1
  274. .endr
  275. 1: mflr r5
  276. add r5,r3,r5
  277. mtctr r5
  278. mtlr r0
  279. bctr
  280. /* Put the contents of vs0 into vsN; N is in r3. */
  281. _GLOBAL(put_vsr)
  282. mflr r0
  283. rlwinm r3,r3,3,0x1f8
  284. bcl 20,31,1f
  285. blr /* v0 is already in v0 */
  286. nop
  287. reg = 1
  288. .rept 63
  289. XXLOR(reg,0,0)
  290. blr
  291. reg = reg + 1
  292. .endr
  293. 1: mflr r5
  294. add r5,r3,r5
  295. mtctr r5
  296. mtlr r0
  297. bctr
  298. /* Load VSX reg N from vector doubleword *p. N is in r3, p in r4. */
  299. _GLOBAL(do_lxvd2x)
  300. PPC_STLU r1,-STKFRM(r1)
  301. mflr r0
  302. PPC_STL r0,STKFRM+PPC_LR_STKOFF(r1)
  303. mfmsr r6
  304. oris r7,r6,MSR_VSX@h
  305. cmpwi cr7,r3,0
  306. li r8,STKFRM-16
  307. MTMSRD(r7)
  308. isync
  309. beq cr7,1f
  310. STXVD2X(0,R1,R8)
  311. 1: li r9,-EFAULT
  312. 2: LXVD2X(0,R0,R4)
  313. li r9,0
  314. 3: beq cr7,4f
  315. bl put_vsr
  316. LXVD2X(0,R1,R8)
  317. 4: PPC_LL r0,STKFRM+PPC_LR_STKOFF(r1)
  318. mtlr r0
  319. MTMSRD(r6)
  320. isync
  321. mr r3,r9
  322. addi r1,r1,STKFRM
  323. blr
  324. EX_TABLE(2b,3b)
  325. /* Store VSX reg N to vector doubleword *p. N is in r3, p in r4. */
  326. _GLOBAL(do_stxvd2x)
  327. PPC_STLU r1,-STKFRM(r1)
  328. mflr r0
  329. PPC_STL r0,STKFRM+PPC_LR_STKOFF(r1)
  330. mfmsr r6
  331. oris r7,r6,MSR_VSX@h
  332. cmpwi cr7,r3,0
  333. li r8,STKFRM-16
  334. MTMSRD(r7)
  335. isync
  336. beq cr7,1f
  337. STXVD2X(0,R1,R8)
  338. bl get_vsr
  339. 1: li r9,-EFAULT
  340. 2: STXVD2X(0,R0,R4)
  341. li r9,0
  342. 3: beq cr7,4f
  343. LXVD2X(0,R1,R8)
  344. 4: PPC_LL r0,STKFRM+PPC_LR_STKOFF(r1)
  345. mtlr r0
  346. MTMSRD(r6)
  347. isync
  348. mr r3,r9
  349. addi r1,r1,STKFRM
  350. blr
  351. EX_TABLE(2b,3b)
  352. #endif /* CONFIG_VSX */
  353. #endif /* CONFIG_PPC_FPU */