You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

490 lines
11KB

  1. ;******************************************************************************
  2. ;* 32 point SSE-optimized DCT transform
  3. ;* Copyright (c) 2010 Vitor Sessak
  4. ;*
  5. ;* This file is part of Libav.
  6. ;*
  7. ;* Libav is free software; you can redistribute it and/or
  8. ;* modify it under the terms of the GNU Lesser General Public
  9. ;* License as published by the Free Software Foundation; either
  10. ;* version 2.1 of the License, or (at your option) any later version.
  11. ;*
  12. ;* Libav 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 GNU
  15. ;* Lesser General Public License for more details.
  16. ;*
  17. ;* You should have received a copy of the GNU Lesser General Public
  18. ;* License along with Libav; if not, write to the Free Software
  19. ;* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  20. ;******************************************************************************
  21. %include "libavutil/x86/x86util.asm"
  22. SECTION_RODATA 32
  23. ps_p1p1m1m1: dd 0, 0, 0x80000000, 0x80000000, 0, 0, 0x80000000, 0x80000000
  24. ps_cos_vec: dd 0.500603, 0.505471, 0.515447, 0.531043
  25. dd 0.553104, 0.582935, 0.622504, 0.674808
  26. dd -10.190008, -3.407609, -2.057781, -1.484165
  27. dd -1.169440, -0.972568, -0.839350, -0.744536
  28. dd 0.502419, 0.522499, 0.566944, 0.646822
  29. dd 0.788155, 1.060678, 1.722447, 5.101149
  30. dd 0.509796, 0.601345, 0.899976, 2.562916
  31. dd 0.509796, 0.601345, 0.899976, 2.562916
  32. dd 1.000000, 1.000000, 1.306563, 0.541196
  33. dd 1.000000, 1.000000, 1.306563, 0.541196
  34. dd 1.000000, 0.707107, 1.000000, -0.707107
  35. dd 1.000000, 0.707107, 1.000000, -0.707107
  36. dd 0.707107, 0.707107, 0.707107, 0.707107
  37. %macro BUTTERFLY 4
  38. subps %4, %1, %2
  39. addps %2, %2, %1
  40. mulps %1, %4, %3
  41. %endmacro
  42. %macro BUTTERFLY0 5
  43. %if cpuflag(sse2) && notcpuflag(avx)
  44. pshufd %4, %1, %5
  45. xorps %1, %2
  46. addps %1, %4
  47. mulps %1, %3
  48. %else
  49. shufps %4, %1, %1, %5
  50. xorps %1, %1, %2
  51. addps %4, %4, %1
  52. mulps %1, %4, %3
  53. %endif
  54. %endmacro
  55. %macro BUTTERFLY2 4
  56. BUTTERFLY0 %1, %2, %3, %4, 0x1b
  57. %endmacro
  58. %macro BUTTERFLY3 4
  59. BUTTERFLY0 %1, %2, %3, %4, 0xb1
  60. %endmacro
  61. %macro BUTTERFLY3V 5
  62. movaps m%5, m%1
  63. addps m%1, m%2
  64. subps m%5, m%2
  65. SWAP %2, %5
  66. mulps m%2, [ps_cos_vec+192]
  67. movaps m%5, m%3
  68. addps m%3, m%4
  69. subps m%4, m%5
  70. mulps m%4, [ps_cos_vec+192]
  71. %endmacro
  72. %macro PASS6_AND_PERMUTE 0
  73. mov tmpd, [outq+4]
  74. movss m7, [outq+72]
  75. addss m7, [outq+76]
  76. movss m3, [outq+56]
  77. addss m3, [outq+60]
  78. addss m4, m3
  79. movss m2, [outq+52]
  80. addss m2, m3
  81. movss m3, [outq+104]
  82. addss m3, [outq+108]
  83. addss m1, m3
  84. addss m5, m4
  85. movss [outq+ 16], m1
  86. movss m1, [outq+100]
  87. addss m1, m3
  88. movss m3, [outq+40]
  89. movss [outq+ 48], m1
  90. addss m3, [outq+44]
  91. movss m1, [outq+100]
  92. addss m4, m3
  93. addss m3, m2
  94. addss m1, [outq+108]
  95. movss [outq+ 40], m3
  96. addss m2, [outq+36]
  97. movss m3, [outq+8]
  98. movss [outq+ 56], m2
  99. addss m3, [outq+12]
  100. movss [outq+ 32], m3
  101. movss m3, [outq+80]
  102. movss [outq+ 8], m5
  103. movss [outq+ 80], m1
  104. movss m2, [outq+52]
  105. movss m5, [outq+120]
  106. addss m5, [outq+124]
  107. movss m1, [outq+64]
  108. addss m2, [outq+60]
  109. addss m0, m5
  110. addss m5, [outq+116]
  111. mov [outq+64], tmpd
  112. addss m6, m0
  113. addss m1, m6
  114. mov tmpd, [outq+12]
  115. mov [outq+ 96], tmpd
  116. movss [outq+ 4], m1
  117. movss m1, [outq+24]
  118. movss [outq+ 24], m4
  119. movss m4, [outq+88]
  120. addss m4, [outq+92]
  121. addss m3, m4
  122. addss m4, [outq+84]
  123. mov tmpd, [outq+108]
  124. addss m1, [outq+28]
  125. addss m0, m1
  126. addss m1, m5
  127. addss m6, m3
  128. addss m3, m0
  129. addss m0, m7
  130. addss m5, [outq+20]
  131. addss m7, m1
  132. movss [outq+ 12], m6
  133. mov [outq+112], tmpd
  134. movss m6, [outq+28]
  135. movss [outq+ 28], m0
  136. movss m0, [outq+36]
  137. movss [outq+ 36], m7
  138. addss m1, m4
  139. movss m7, [outq+116]
  140. addss m0, m2
  141. addss m7, [outq+124]
  142. movss [outq+ 72], m0
  143. movss m0, [outq+44]
  144. addss m2, m0
  145. movss [outq+ 44], m1
  146. movss [outq+ 88], m2
  147. addss m0, [outq+60]
  148. mov tmpd, [outq+60]
  149. mov [outq+120], tmpd
  150. movss [outq+104], m0
  151. addss m4, m5
  152. addss m5, [outq+68]
  153. movss [outq+52], m4
  154. movss [outq+60], m5
  155. movss m4, [outq+68]
  156. movss m5, [outq+20]
  157. movss [outq+ 20], m3
  158. addss m5, m7
  159. addss m7, m6
  160. addss m4, m5
  161. movss m2, [outq+84]
  162. addss m2, [outq+92]
  163. addss m5, m2
  164. movss [outq+ 68], m4
  165. addss m2, m7
  166. movss m4, [outq+76]
  167. movss [outq+ 84], m2
  168. movss [outq+ 76], m5
  169. addss m7, m4
  170. addss m6, [outq+124]
  171. addss m4, m6
  172. addss m6, [outq+92]
  173. movss [outq+100], m4
  174. movss [outq+108], m6
  175. movss m6, [outq+92]
  176. movss [outq+92], m7
  177. addss m6, [outq+124]
  178. movss [outq+116], m6
  179. %endmacro
  180. INIT_YMM avx
  181. SECTION .text
  182. ; void ff_dct32_float_avx(FFTSample *out, const FFTSample *in)
  183. cglobal dct32_float, 2,3,8, out, in, tmp
  184. ; pass 1
  185. vmovaps m4, [inq+0]
  186. vinsertf128 m5, m5, [inq+96], 1
  187. vinsertf128 m5, m5, [inq+112], 0
  188. vshufps m5, m5, m5, 0x1b
  189. BUTTERFLY m4, m5, [ps_cos_vec], m6
  190. vmovaps m2, [inq+64]
  191. vinsertf128 m6, m6, [inq+32], 1
  192. vinsertf128 m6, m6, [inq+48], 0
  193. vshufps m6, m6, m6, 0x1b
  194. BUTTERFLY m2, m6, [ps_cos_vec+32], m0
  195. ; pass 2
  196. BUTTERFLY m5, m6, [ps_cos_vec+64], m0
  197. BUTTERFLY m4, m2, [ps_cos_vec+64], m7
  198. ; pass 3
  199. vperm2f128 m3, m6, m4, 0x31
  200. vperm2f128 m1, m6, m4, 0x20
  201. vshufps m3, m3, m3, 0x1b
  202. BUTTERFLY m1, m3, [ps_cos_vec+96], m6
  203. vperm2f128 m4, m5, m2, 0x20
  204. vperm2f128 m5, m5, m2, 0x31
  205. vshufps m5, m5, m5, 0x1b
  206. BUTTERFLY m4, m5, [ps_cos_vec+96], m6
  207. ; pass 4
  208. vmovaps m6, [ps_p1p1m1m1+0]
  209. vmovaps m2, [ps_cos_vec+128]
  210. BUTTERFLY2 m5, m6, m2, m7
  211. BUTTERFLY2 m4, m6, m2, m7
  212. BUTTERFLY2 m1, m6, m2, m7
  213. BUTTERFLY2 m3, m6, m2, m7
  214. ; pass 5
  215. vshufps m6, m6, m6, 0xcc
  216. vmovaps m2, [ps_cos_vec+160]
  217. BUTTERFLY3 m5, m6, m2, m7
  218. BUTTERFLY3 m4, m6, m2, m7
  219. BUTTERFLY3 m1, m6, m2, m7
  220. BUTTERFLY3 m3, m6, m2, m7
  221. vperm2f128 m6, m3, m3, 0x31
  222. vmovaps [outq], m3
  223. vextractf128 [outq+64], m5, 1
  224. vextractf128 [outq+32], m5, 0
  225. vextractf128 [outq+80], m4, 1
  226. vextractf128 [outq+48], m4, 0
  227. vperm2f128 m0, m1, m1, 0x31
  228. vmovaps [outq+96], m1
  229. vzeroupper
  230. ; pass 6, no SIMD...
  231. INIT_XMM
  232. PASS6_AND_PERMUTE
  233. RET
  234. %if ARCH_X86_64
  235. %define SPILL SWAP
  236. %define UNSPILL SWAP
  237. %macro PASS5 0
  238. nop ; FIXME code alignment
  239. SWAP 5, 8
  240. SWAP 4, 12
  241. SWAP 6, 14
  242. SWAP 7, 13
  243. SWAP 0, 15
  244. PERMUTE 9,10, 10,12, 11,14, 12,9, 13,11, 14,13
  245. TRANSPOSE4x4PS 8, 9, 10, 11, 0
  246. BUTTERFLY3V 8, 9, 10, 11, 0
  247. addps m10, m11
  248. TRANSPOSE4x4PS 12, 13, 14, 15, 0
  249. BUTTERFLY3V 12, 13, 14, 15, 0
  250. addps m14, m15
  251. addps m12, m14
  252. addps m14, m13
  253. addps m13, m15
  254. %endmacro
  255. %macro PASS6 0
  256. SWAP 9, 12
  257. SWAP 11, 14
  258. movss [outq+0x00], m8
  259. pshuflw m0, m8, 0xe
  260. movss [outq+0x10], m9
  261. pshuflw m1, m9, 0xe
  262. movss [outq+0x20], m10
  263. pshuflw m2, m10, 0xe
  264. movss [outq+0x30], m11
  265. pshuflw m3, m11, 0xe
  266. movss [outq+0x40], m12
  267. pshuflw m4, m12, 0xe
  268. movss [outq+0x50], m13
  269. pshuflw m5, m13, 0xe
  270. movss [outq+0x60], m14
  271. pshuflw m6, m14, 0xe
  272. movaps [outq+0x70], m15
  273. pshuflw m7, m15, 0xe
  274. addss m0, m1
  275. addss m1, m2
  276. movss [outq+0x08], m0
  277. addss m2, m3
  278. movss [outq+0x18], m1
  279. addss m3, m4
  280. movss [outq+0x28], m2
  281. addss m4, m5
  282. movss [outq+0x38], m3
  283. addss m5, m6
  284. movss [outq+0x48], m4
  285. addss m6, m7
  286. movss [outq+0x58], m5
  287. movss [outq+0x68], m6
  288. movss [outq+0x78], m7
  289. PERMUTE 1,8, 3,9, 5,10, 7,11, 9,12, 11,13, 13,14, 8,1, 10,3, 12,5, 14,7
  290. movhlps m0, m1
  291. pshufd m1, m1, 3
  292. SWAP 0, 2, 4, 6, 8, 10, 12, 14
  293. SWAP 1, 3, 5, 7, 9, 11, 13, 15
  294. %rep 7
  295. movhlps m0, m1
  296. pshufd m1, m1, 3
  297. addss m15, m1
  298. SWAP 0, 2, 4, 6, 8, 10, 12, 14
  299. SWAP 1, 3, 5, 7, 9, 11, 13, 15
  300. %endrep
  301. %assign i 4
  302. %rep 15
  303. addss m0, m1
  304. movss [outq+i], m0
  305. SWAP 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15
  306. %assign i i+8
  307. %endrep
  308. %endmacro
  309. %else ; ARCH_X86_32
  310. %macro SPILL 2 ; xmm#, mempos
  311. movaps [outq+(%2-8)*16], m%1
  312. %endmacro
  313. %macro UNSPILL 2
  314. movaps m%1, [outq+(%2-8)*16]
  315. %endmacro
  316. %define PASS6 PASS6_AND_PERMUTE
  317. %macro PASS5 0
  318. movaps m2, [ps_cos_vec+160]
  319. shufps m3, m3, 0xcc
  320. BUTTERFLY3 m5, m3, m2, m1
  321. SPILL 5, 8
  322. UNSPILL 1, 9
  323. BUTTERFLY3 m1, m3, m2, m5
  324. SPILL 1, 14
  325. BUTTERFLY3 m4, m3, m2, m5
  326. SPILL 4, 12
  327. BUTTERFLY3 m7, m3, m2, m5
  328. SPILL 7, 13
  329. UNSPILL 5, 10
  330. BUTTERFLY3 m5, m3, m2, m7
  331. SPILL 5, 10
  332. UNSPILL 4, 11
  333. BUTTERFLY3 m4, m3, m2, m7
  334. SPILL 4, 11
  335. BUTTERFLY3 m6, m3, m2, m7
  336. SPILL 6, 9
  337. BUTTERFLY3 m0, m3, m2, m7
  338. SPILL 0, 15
  339. %endmacro
  340. %endif
  341. ; void ff_dct32_float_sse(FFTSample *out, const FFTSample *in)
  342. %macro DCT32_FUNC 0
  343. cglobal dct32_float, 2, 3, 16, out, in, tmp
  344. ; pass 1
  345. movaps m0, [inq+0]
  346. LOAD_INV m1, [inq+112]
  347. BUTTERFLY m0, m1, [ps_cos_vec], m3
  348. movaps m7, [inq+64]
  349. LOAD_INV m4, [inq+48]
  350. BUTTERFLY m7, m4, [ps_cos_vec+32], m3
  351. ; pass 2
  352. movaps m2, [ps_cos_vec+64]
  353. BUTTERFLY m1, m4, m2, m3
  354. SPILL 1, 11
  355. SPILL 4, 8
  356. ; pass 1
  357. movaps m1, [inq+16]
  358. LOAD_INV m6, [inq+96]
  359. BUTTERFLY m1, m6, [ps_cos_vec+16], m3
  360. movaps m4, [inq+80]
  361. LOAD_INV m5, [inq+32]
  362. BUTTERFLY m4, m5, [ps_cos_vec+48], m3
  363. ; pass 2
  364. BUTTERFLY m0, m7, m2, m3
  365. movaps m2, [ps_cos_vec+80]
  366. BUTTERFLY m6, m5, m2, m3
  367. BUTTERFLY m1, m4, m2, m3
  368. ; pass 3
  369. movaps m2, [ps_cos_vec+96]
  370. shufps m1, m1, 0x1b
  371. BUTTERFLY m0, m1, m2, m3
  372. SPILL 0, 15
  373. SPILL 1, 14
  374. UNSPILL 0, 8
  375. shufps m5, m5, 0x1b
  376. BUTTERFLY m0, m5, m2, m3
  377. UNSPILL 1, 11
  378. shufps m6, m6, 0x1b
  379. BUTTERFLY m1, m6, m2, m3
  380. SPILL 1, 11
  381. shufps m4, m4, 0x1b
  382. BUTTERFLY m7, m4, m2, m3
  383. ; pass 4
  384. movaps m3, [ps_p1p1m1m1+0]
  385. movaps m2, [ps_cos_vec+128]
  386. BUTTERFLY2 m5, m3, m2, m1
  387. BUTTERFLY2 m0, m3, m2, m1
  388. SPILL 0, 9
  389. BUTTERFLY2 m6, m3, m2, m1
  390. SPILL 6, 10
  391. UNSPILL 0, 11
  392. BUTTERFLY2 m0, m3, m2, m1
  393. SPILL 0, 11
  394. BUTTERFLY2 m4, m3, m2, m1
  395. BUTTERFLY2 m7, m3, m2, m1
  396. UNSPILL 6, 14
  397. BUTTERFLY2 m6, m3, m2, m1
  398. UNSPILL 0, 15
  399. BUTTERFLY2 m0, m3, m2, m1
  400. PASS5
  401. PASS6
  402. RET
  403. %endmacro
  404. %macro LOAD_INV 2
  405. %if cpuflag(sse2)
  406. pshufd %1, %2, 0x1b
  407. %elif cpuflag(sse)
  408. movaps %1, %2
  409. shufps %1, %1, 0x1b
  410. %endif
  411. %endmacro
  412. %if ARCH_X86_32
  413. INIT_XMM sse
  414. DCT32_FUNC
  415. %endif
  416. INIT_XMM sse2
  417. DCT32_FUNC