dec_msa.c 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. // Copyright 2016 Google Inc. All Rights Reserved.
  2. //
  3. // Use of this source code is governed by a BSD-style license
  4. // that can be found in the COPYING file in the root of the source
  5. // tree. An additional intellectual property rights grant can be found
  6. // in the file PATENTS. All contributing project authors may
  7. // be found in the AUTHORS file in the root of the source tree.
  8. // -----------------------------------------------------------------------------
  9. //
  10. // MSA version of dsp functions
  11. //
  12. // Author(s): Prashant Patil ([email protected])
  13. #include "./dsp.h"
  14. #if defined(WEBP_USE_MSA)
  15. #include "./msa_macro.h"
  16. //------------------------------------------------------------------------------
  17. // Transforms
  18. #define IDCT_1D_W(in0, in1, in2, in3, out0, out1, out2, out3) { \
  19. v4i32 a1_m, b1_m, c1_m, d1_m; \
  20. v4i32 c_tmp1_m, c_tmp2_m, d_tmp1_m, d_tmp2_m; \
  21. const v4i32 cospi8sqrt2minus1 = __msa_fill_w(20091); \
  22. const v4i32 sinpi8sqrt2 = __msa_fill_w(35468); \
  23. \
  24. a1_m = in0 + in2; \
  25. b1_m = in0 - in2; \
  26. c_tmp1_m = (in1 * sinpi8sqrt2) >> 16; \
  27. c_tmp2_m = in3 + ((in3 * cospi8sqrt2minus1) >> 16); \
  28. c1_m = c_tmp1_m - c_tmp2_m; \
  29. d_tmp1_m = in1 + ((in1 * cospi8sqrt2minus1) >> 16); \
  30. d_tmp2_m = (in3 * sinpi8sqrt2) >> 16; \
  31. d1_m = d_tmp1_m + d_tmp2_m; \
  32. BUTTERFLY_4(a1_m, b1_m, c1_m, d1_m, out0, out1, out2, out3); \
  33. }
  34. #define MULT1(a) ((((a) * 20091) >> 16) + (a))
  35. #define MULT2(a) (((a) * 35468) >> 16)
  36. static void TransformOne(const int16_t* in, uint8_t* dst) {
  37. v8i16 input0, input1;
  38. v4i32 in0, in1, in2, in3, hz0, hz1, hz2, hz3, vt0, vt1, vt2, vt3;
  39. v4i32 res0, res1, res2, res3;
  40. const v16i8 zero = { 0 };
  41. v16i8 dest0, dest1, dest2, dest3;
  42. LD_SH2(in, 8, input0, input1);
  43. UNPCK_SH_SW(input0, in0, in1);
  44. UNPCK_SH_SW(input1, in2, in3);
  45. IDCT_1D_W(in0, in1, in2, in3, hz0, hz1, hz2, hz3);
  46. TRANSPOSE4x4_SW_SW(hz0, hz1, hz2, hz3, hz0, hz1, hz2, hz3);
  47. IDCT_1D_W(hz0, hz1, hz2, hz3, vt0, vt1, vt2, vt3);
  48. SRARI_W4_SW(vt0, vt1, vt2, vt3, 3);
  49. TRANSPOSE4x4_SW_SW(vt0, vt1, vt2, vt3, vt0, vt1, vt2, vt3);
  50. LD_SB4(dst, BPS, dest0, dest1, dest2, dest3);
  51. ILVR_B4_SW(zero, dest0, zero, dest1, zero, dest2, zero, dest3,
  52. res0, res1, res2, res3);
  53. ILVR_H4_SW(zero, res0, zero, res1, zero, res2, zero, res3,
  54. res0, res1, res2, res3);
  55. ADD4(res0, vt0, res1, vt1, res2, vt2, res3, vt3, res0, res1, res2, res3);
  56. CLIP_SW4_0_255(res0, res1, res2, res3);
  57. PCKEV_B2_SW(res0, res1, res2, res3, vt0, vt1);
  58. res0 = (v4i32)__msa_pckev_b((v16i8)vt0, (v16i8)vt1);
  59. ST4x4_UB(res0, res0, 3, 2, 1, 0, dst, BPS);
  60. }
  61. static void TransformTwo(const int16_t* in, uint8_t* dst, int do_two) {
  62. TransformOne(in, dst);
  63. if (do_two) {
  64. TransformOne(in + 16, dst + 4);
  65. }
  66. }
  67. static void TransformWHT(const int16_t* in, int16_t* out) {
  68. v8i16 input0, input1;
  69. const v8i16 mask0 = { 0, 1, 2, 3, 8, 9, 10, 11 };
  70. const v8i16 mask1 = { 4, 5, 6, 7, 12, 13, 14, 15 };
  71. const v8i16 mask2 = { 0, 4, 8, 12, 1, 5, 9, 13 };
  72. const v8i16 mask3 = { 3, 7, 11, 15, 2, 6, 10, 14 };
  73. v8i16 tmp0, tmp1, tmp2, tmp3;
  74. v8i16 out0, out1;
  75. LD_SH2(in, 8, input0, input1);
  76. input1 = SLDI_SH(input1, input1, 8);
  77. tmp0 = input0 + input1;
  78. tmp1 = input0 - input1;
  79. VSHF_H2_SH(tmp0, tmp1, tmp0, tmp1, mask0, mask1, tmp2, tmp3);
  80. out0 = tmp2 + tmp3;
  81. out1 = tmp2 - tmp3;
  82. VSHF_H2_SH(out0, out1, out0, out1, mask2, mask3, input0, input1);
  83. tmp0 = input0 + input1;
  84. tmp1 = input0 - input1;
  85. VSHF_H2_SH(tmp0, tmp1, tmp0, tmp1, mask0, mask1, tmp2, tmp3);
  86. tmp0 = tmp2 + tmp3;
  87. tmp1 = tmp2 - tmp3;
  88. ADDVI_H2_SH(tmp0, 3, tmp1, 3, out0, out1);
  89. SRAI_H2_SH(out0, out1, 3);
  90. out[0] = __msa_copy_s_h(out0, 0);
  91. out[16] = __msa_copy_s_h(out0, 4);
  92. out[32] = __msa_copy_s_h(out1, 0);
  93. out[48] = __msa_copy_s_h(out1, 4);
  94. out[64] = __msa_copy_s_h(out0, 1);
  95. out[80] = __msa_copy_s_h(out0, 5);
  96. out[96] = __msa_copy_s_h(out1, 1);
  97. out[112] = __msa_copy_s_h(out1, 5);
  98. out[128] = __msa_copy_s_h(out0, 2);
  99. out[144] = __msa_copy_s_h(out0, 6);
  100. out[160] = __msa_copy_s_h(out1, 2);
  101. out[176] = __msa_copy_s_h(out1, 6);
  102. out[192] = __msa_copy_s_h(out0, 3);
  103. out[208] = __msa_copy_s_h(out0, 7);
  104. out[224] = __msa_copy_s_h(out1, 3);
  105. out[240] = __msa_copy_s_h(out1, 7);
  106. }
  107. static void TransformDC(const int16_t* in, uint8_t* dst) {
  108. const int DC = (in[0] + 4) >> 3;
  109. const v8i16 tmp0 = __msa_fill_h(DC);
  110. ADDBLK_ST4x4_UB(tmp0, tmp0, tmp0, tmp0, dst, BPS);
  111. }
  112. static void TransformAC3(const int16_t* in, uint8_t* dst) {
  113. const int a = in[0] + 4;
  114. const int c4 = MULT2(in[4]);
  115. const int d4 = MULT1(in[4]);
  116. const int in2 = MULT2(in[1]);
  117. const int in3 = MULT1(in[1]);
  118. v4i32 tmp0 = { 0 };
  119. v4i32 out0 = __msa_fill_w(a + d4);
  120. v4i32 out1 = __msa_fill_w(a + c4);
  121. v4i32 out2 = __msa_fill_w(a - c4);
  122. v4i32 out3 = __msa_fill_w(a - d4);
  123. v4i32 res0, res1, res2, res3;
  124. const v4i32 zero = { 0 };
  125. v16u8 dest0, dest1, dest2, dest3;
  126. INSERT_W4_SW(in3, in2, -in2, -in3, tmp0);
  127. ADD4(out0, tmp0, out1, tmp0, out2, tmp0, out3, tmp0,
  128. out0, out1, out2, out3);
  129. SRAI_W4_SW(out0, out1, out2, out3, 3);
  130. LD_UB4(dst, BPS, dest0, dest1, dest2, dest3);
  131. ILVR_B4_SW(zero, dest0, zero, dest1, zero, dest2, zero, dest3,
  132. res0, res1, res2, res3);
  133. ILVR_H4_SW(zero, res0, zero, res1, zero, res2, zero, res3,
  134. res0, res1, res2, res3);
  135. ADD4(res0, out0, res1, out1, res2, out2, res3, out3, res0, res1, res2, res3);
  136. CLIP_SW4_0_255(res0, res1, res2, res3);
  137. PCKEV_B2_SW(res0, res1, res2, res3, out0, out1);
  138. res0 = (v4i32)__msa_pckev_b((v16i8)out0, (v16i8)out1);
  139. ST4x4_UB(res0, res0, 3, 2, 1, 0, dst, BPS);
  140. }
  141. //------------------------------------------------------------------------------
  142. // Entry point
  143. extern void VP8DspInitMSA(void);
  144. WEBP_TSAN_IGNORE_FUNCTION void VP8DspInitMSA(void) {
  145. VP8TransformWHT = TransformWHT;
  146. VP8Transform = TransformTwo;
  147. VP8TransformDC = TransformDC;
  148. VP8TransformAC3 = TransformAC3;
  149. }
  150. #else // !WEBP_USE_MSA
  151. WEBP_DSP_INIT_STUB(VP8DspInitMSA)
  152. #endif // WEBP_USE_MSA