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.

169 lines
6.2KB

  1. /*
  2. * SIMD-optimized IDCT-related routines
  3. * Copyright (c) 2000, 2001 Fabrice Bellard
  4. * Copyright (c) 2002-2004 Michael Niedermayer <michaelni@gmx.at>
  5. *
  6. * MMX optimization by Nick Kurshev <nickols_k@mail.ru>
  7. *
  8. * This file is part of Libav.
  9. *
  10. * Libav is free software; you can redistribute it and/or
  11. * modify it under the terms of the GNU Lesser General Public
  12. * License as published by the Free Software Foundation; either
  13. * version 2.1 of the License, or (at your option) any later version.
  14. *
  15. * Libav is distributed in the hope that it will be useful,
  16. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  18. * Lesser General Public License for more details.
  19. *
  20. * You should have received a copy of the GNU Lesser General Public
  21. * License along with Libav; if not, write to the Free Software
  22. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  23. */
  24. #include "config.h"
  25. #include "libavutil/cpu.h"
  26. #include "libavutil/x86/asm.h"
  27. #include "idctdsp.h"
  28. #include "inline_asm.h"
  29. #if HAVE_INLINE_ASM
  30. void ff_put_pixels_clamped_mmx(const int16_t *block, uint8_t *pixels,
  31. int line_size)
  32. {
  33. const int16_t *p;
  34. uint8_t *pix;
  35. /* read the pixels */
  36. p = block;
  37. pix = pixels;
  38. /* unrolled loop */
  39. __asm__ volatile (
  40. "movq (%3), %%mm0 \n\t"
  41. "movq 8(%3), %%mm1 \n\t"
  42. "movq 16(%3), %%mm2 \n\t"
  43. "movq 24(%3), %%mm3 \n\t"
  44. "movq 32(%3), %%mm4 \n\t"
  45. "movq 40(%3), %%mm5 \n\t"
  46. "movq 48(%3), %%mm6 \n\t"
  47. "movq 56(%3), %%mm7 \n\t"
  48. "packuswb %%mm1, %%mm0 \n\t"
  49. "packuswb %%mm3, %%mm2 \n\t"
  50. "packuswb %%mm5, %%mm4 \n\t"
  51. "packuswb %%mm7, %%mm6 \n\t"
  52. "movq %%mm0, (%0) \n\t"
  53. "movq %%mm2, (%0, %1) \n\t"
  54. "movq %%mm4, (%0, %1, 2) \n\t"
  55. "movq %%mm6, (%0, %2) \n\t"
  56. :: "r" (pix), "r" ((x86_reg) line_size), "r" ((x86_reg) line_size * 3),
  57. "r" (p)
  58. : "memory");
  59. pix += line_size * 4;
  60. p += 32;
  61. // if here would be an exact copy of the code above
  62. // compiler would generate some very strange code
  63. // thus using "r"
  64. __asm__ volatile (
  65. "movq (%3), %%mm0 \n\t"
  66. "movq 8(%3), %%mm1 \n\t"
  67. "movq 16(%3), %%mm2 \n\t"
  68. "movq 24(%3), %%mm3 \n\t"
  69. "movq 32(%3), %%mm4 \n\t"
  70. "movq 40(%3), %%mm5 \n\t"
  71. "movq 48(%3), %%mm6 \n\t"
  72. "movq 56(%3), %%mm7 \n\t"
  73. "packuswb %%mm1, %%mm0 \n\t"
  74. "packuswb %%mm3, %%mm2 \n\t"
  75. "packuswb %%mm5, %%mm4 \n\t"
  76. "packuswb %%mm7, %%mm6 \n\t"
  77. "movq %%mm0, (%0) \n\t"
  78. "movq %%mm2, (%0, %1) \n\t"
  79. "movq %%mm4, (%0, %1, 2) \n\t"
  80. "movq %%mm6, (%0, %2) \n\t"
  81. :: "r" (pix), "r" ((x86_reg) line_size), "r" ((x86_reg) line_size * 3),
  82. "r" (p)
  83. : "memory");
  84. }
  85. #define put_signed_pixels_clamped_mmx_half(off) \
  86. "movq "#off"(%2), %%mm1 \n\t" \
  87. "movq 16 + "#off"(%2), %%mm2 \n\t" \
  88. "movq 32 + "#off"(%2), %%mm3 \n\t" \
  89. "movq 48 + "#off"(%2), %%mm4 \n\t" \
  90. "packsswb 8 + "#off"(%2), %%mm1 \n\t" \
  91. "packsswb 24 + "#off"(%2), %%mm2 \n\t" \
  92. "packsswb 40 + "#off"(%2), %%mm3 \n\t" \
  93. "packsswb 56 + "#off"(%2), %%mm4 \n\t" \
  94. "paddb %%mm0, %%mm1 \n\t" \
  95. "paddb %%mm0, %%mm2 \n\t" \
  96. "paddb %%mm0, %%mm3 \n\t" \
  97. "paddb %%mm0, %%mm4 \n\t" \
  98. "movq %%mm1, (%0) \n\t" \
  99. "movq %%mm2, (%0, %3) \n\t" \
  100. "movq %%mm3, (%0, %3, 2) \n\t" \
  101. "movq %%mm4, (%0, %1) \n\t"
  102. void ff_put_signed_pixels_clamped_mmx(const int16_t *block, uint8_t *pixels,
  103. int line_size)
  104. {
  105. x86_reg line_skip = line_size;
  106. x86_reg line_skip3;
  107. __asm__ volatile (
  108. "movq "MANGLE(ff_pb_80)", %%mm0 \n\t"
  109. "lea (%3, %3, 2), %1 \n\t"
  110. put_signed_pixels_clamped_mmx_half(0)
  111. "lea (%0, %3, 4), %0 \n\t"
  112. put_signed_pixels_clamped_mmx_half(64)
  113. : "+&r" (pixels), "=&r" (line_skip3)
  114. : "r" (block), "r" (line_skip)
  115. : "memory");
  116. }
  117. void ff_add_pixels_clamped_mmx(const int16_t *block, uint8_t *pixels,
  118. int line_size)
  119. {
  120. const int16_t *p;
  121. uint8_t *pix;
  122. int i;
  123. /* read the pixels */
  124. p = block;
  125. pix = pixels;
  126. MOVQ_ZERO(mm7);
  127. i = 4;
  128. do {
  129. __asm__ volatile (
  130. "movq (%2), %%mm0 \n\t"
  131. "movq 8(%2), %%mm1 \n\t"
  132. "movq 16(%2), %%mm2 \n\t"
  133. "movq 24(%2), %%mm3 \n\t"
  134. "movq %0, %%mm4 \n\t"
  135. "movq %1, %%mm6 \n\t"
  136. "movq %%mm4, %%mm5 \n\t"
  137. "punpcklbw %%mm7, %%mm4 \n\t"
  138. "punpckhbw %%mm7, %%mm5 \n\t"
  139. "paddsw %%mm4, %%mm0 \n\t"
  140. "paddsw %%mm5, %%mm1 \n\t"
  141. "movq %%mm6, %%mm5 \n\t"
  142. "punpcklbw %%mm7, %%mm6 \n\t"
  143. "punpckhbw %%mm7, %%mm5 \n\t"
  144. "paddsw %%mm6, %%mm2 \n\t"
  145. "paddsw %%mm5, %%mm3 \n\t"
  146. "packuswb %%mm1, %%mm0 \n\t"
  147. "packuswb %%mm3, %%mm2 \n\t"
  148. "movq %%mm0, %0 \n\t"
  149. "movq %%mm2, %1 \n\t"
  150. : "+m" (*pix), "+m" (*(pix + line_size))
  151. : "r" (p)
  152. : "memory");
  153. pix += line_size * 2;
  154. p += 16;
  155. } while (--i);
  156. }
  157. #endif /* HAVE_INLINE_ASM */