2015-05-08 08:53:27 +02:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2015 The WebM project authors. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license
|
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
|
* tree. An additional intellectual property rights grant can be found
|
|
|
|
* in the file PATENTS. All contributing project authors may
|
|
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
|
|
*/
|
|
|
|
|
2015-06-02 08:46:28 +02:00
|
|
|
#include <assert.h>
|
2015-07-31 19:53:25 +02:00
|
|
|
|
2015-06-01 05:49:01 +02:00
|
|
|
#include "vp9/common/mips/msa/vp9_idct_msa.h"
|
2015-07-31 19:53:25 +02:00
|
|
|
#include "vp9/common/vp9_enums.h"
|
2015-05-08 08:53:27 +02:00
|
|
|
|
2015-06-01 05:49:01 +02:00
|
|
|
void vp9_idct8x8_64_add_msa(const int16_t *input, uint8_t *dst,
|
|
|
|
int32_t dst_stride) {
|
2015-05-08 08:53:27 +02:00
|
|
|
v8i16 in0, in1, in2, in3, in4, in5, in6, in7;
|
|
|
|
|
|
|
|
/* load vector elements of 8x8 block */
|
2015-06-01 05:49:01 +02:00
|
|
|
LD_SH8(input, 8, in0, in1, in2, in3, in4, in5, in6, in7);
|
2015-05-08 08:53:27 +02:00
|
|
|
|
|
|
|
/* rows transform */
|
2015-06-01 05:49:01 +02:00
|
|
|
TRANSPOSE8x8_SH_SH(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
2015-05-08 08:53:27 +02:00
|
|
|
/* 1D idct8x8 */
|
|
|
|
VP9_IDCT8x8_1D(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
|
|
|
/* columns transform */
|
2015-06-01 05:49:01 +02:00
|
|
|
TRANSPOSE8x8_SH_SH(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
2015-05-08 08:53:27 +02:00
|
|
|
/* 1D idct8x8 */
|
|
|
|
VP9_IDCT8x8_1D(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
|
|
|
/* final rounding (add 2^4, divide by 2^5) and shift */
|
2015-06-01 05:49:01 +02:00
|
|
|
SRARI_H4_SH(in0, in1, in2, in3, 5);
|
|
|
|
SRARI_H4_SH(in4, in5, in6, in7, 5);
|
2015-05-08 08:53:27 +02:00
|
|
|
/* add block and store 8x8 */
|
2015-06-01 05:49:01 +02:00
|
|
|
VP9_ADDBLK_ST8x4_UB(dst, dst_stride, in0, in1, in2, in3);
|
|
|
|
dst += (4 * dst_stride);
|
|
|
|
VP9_ADDBLK_ST8x4_UB(dst, dst_stride, in4, in5, in6, in7);
|
2015-05-08 08:53:27 +02:00
|
|
|
}
|
|
|
|
|
2015-06-01 05:49:01 +02:00
|
|
|
void vp9_idct8x8_12_add_msa(const int16_t *input, uint8_t *dst,
|
|
|
|
int32_t dst_stride) {
|
2015-05-08 08:53:27 +02:00
|
|
|
v8i16 in0, in1, in2, in3, in4, in5, in6, in7;
|
2015-06-01 05:49:01 +02:00
|
|
|
v8i16 s0, s1, s2, s3, s4, s5, s6, s7, k0, k1, k2, k3, m0, m1, m2, m3;
|
2015-05-08 08:53:27 +02:00
|
|
|
v4i32 tmp0, tmp1, tmp2, tmp3;
|
|
|
|
v8i16 zero = { 0 };
|
|
|
|
|
|
|
|
/* load vector elements of 8x8 block */
|
2015-06-01 05:49:01 +02:00
|
|
|
LD_SH8(input, 8, in0, in1, in2, in3, in4, in5, in6, in7);
|
|
|
|
TRANSPOSE8X4_SH_SH(in0, in1, in2, in3, in0, in1, in2, in3);
|
2015-05-08 08:53:27 +02:00
|
|
|
|
|
|
|
/* stage1 */
|
2015-06-01 05:49:01 +02:00
|
|
|
ILVL_H2_SH(in3, in0, in2, in1, s0, s1);
|
|
|
|
k0 = VP9_SET_COSPI_PAIR(cospi_28_64, -cospi_4_64);
|
|
|
|
k1 = VP9_SET_COSPI_PAIR(cospi_4_64, cospi_28_64);
|
|
|
|
k2 = VP9_SET_COSPI_PAIR(-cospi_20_64, cospi_12_64);
|
|
|
|
k3 = VP9_SET_COSPI_PAIR(cospi_12_64, cospi_20_64);
|
|
|
|
DOTP_SH4_SW(s0, s0, s1, s1, k0, k1, k2, k3, tmp0, tmp1, tmp2, tmp3);
|
|
|
|
SRARI_W4_SW(tmp0, tmp1, tmp2, tmp3, DCT_CONST_BITS);
|
|
|
|
PCKEV_H2_SH(zero, tmp0, zero, tmp1, s0, s1);
|
|
|
|
PCKEV_H2_SH(zero, tmp2, zero, tmp3, s2, s3);
|
2015-05-08 08:53:27 +02:00
|
|
|
BUTTERFLY_4(s0, s1, s3, s2, s4, s7, s6, s5);
|
|
|
|
|
|
|
|
/* stage2 */
|
2015-06-01 05:49:01 +02:00
|
|
|
ILVR_H2_SH(in3, in1, in2, in0, s1, s0);
|
|
|
|
k0 = VP9_SET_COSPI_PAIR(cospi_16_64, cospi_16_64);
|
|
|
|
k1 = VP9_SET_COSPI_PAIR(cospi_16_64, -cospi_16_64);
|
|
|
|
k2 = VP9_SET_COSPI_PAIR(cospi_24_64, -cospi_8_64);
|
|
|
|
k3 = VP9_SET_COSPI_PAIR(cospi_8_64, cospi_24_64);
|
|
|
|
DOTP_SH4_SW(s0, s0, s1, s1, k0, k1, k2, k3, tmp0, tmp1, tmp2, tmp3);
|
|
|
|
SRARI_W4_SW(tmp0, tmp1, tmp2, tmp3, DCT_CONST_BITS);
|
|
|
|
PCKEV_H2_SH(zero, tmp0, zero, tmp1, s0, s1);
|
|
|
|
PCKEV_H2_SH(zero, tmp2, zero, tmp3, s2, s3);
|
2015-05-08 08:53:27 +02:00
|
|
|
BUTTERFLY_4(s0, s1, s2, s3, m0, m1, m2, m3);
|
|
|
|
|
|
|
|
/* stage3 */
|
|
|
|
s0 = __msa_ilvr_h(s6, s5);
|
|
|
|
|
2015-06-01 05:49:01 +02:00
|
|
|
k1 = VP9_SET_COSPI_PAIR(-cospi_16_64, cospi_16_64);
|
|
|
|
DOTP_SH2_SW(s0, s0, k1, k0, tmp0, tmp1);
|
|
|
|
SRARI_W2_SW(tmp0, tmp1, DCT_CONST_BITS);
|
|
|
|
PCKEV_H2_SH(zero, tmp0, zero, tmp1, s2, s3);
|
2015-05-08 08:53:27 +02:00
|
|
|
|
|
|
|
/* stage4 */
|
|
|
|
BUTTERFLY_8(m0, m1, m2, m3, s4, s2, s3, s7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
2015-06-01 05:49:01 +02:00
|
|
|
TRANSPOSE4X8_SH_SH(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
2015-05-08 08:53:27 +02:00
|
|
|
VP9_IDCT8x8_1D(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
|
|
|
|
|
|
|
/* final rounding (add 2^4, divide by 2^5) and shift */
|
2015-06-01 05:49:01 +02:00
|
|
|
SRARI_H4_SH(in0, in1, in2, in3, 5);
|
|
|
|
SRARI_H4_SH(in4, in5, in6, in7, 5);
|
2015-05-08 08:53:27 +02:00
|
|
|
|
|
|
|
/* add block and store 8x8 */
|
2015-06-01 05:49:01 +02:00
|
|
|
VP9_ADDBLK_ST8x4_UB(dst, dst_stride, in0, in1, in2, in3);
|
|
|
|
dst += (4 * dst_stride);
|
|
|
|
VP9_ADDBLK_ST8x4_UB(dst, dst_stride, in4, in5, in6, in7);
|
2015-05-08 08:53:27 +02:00
|
|
|
}
|
|
|
|
|
2015-06-01 05:49:01 +02:00
|
|
|
void vp9_idct8x8_1_add_msa(const int16_t *input, uint8_t *dst,
|
|
|
|
int32_t dst_stride) {
|
2015-05-08 08:53:27 +02:00
|
|
|
int16_t out;
|
2015-06-01 05:49:01 +02:00
|
|
|
int32_t val;
|
|
|
|
v8i16 vec;
|
2015-05-08 08:53:27 +02:00
|
|
|
|
2015-06-01 05:49:01 +02:00
|
|
|
out = ROUND_POWER_OF_TWO((input[0] * cospi_16_64), DCT_CONST_BITS);
|
|
|
|
out = ROUND_POWER_OF_TWO((out * cospi_16_64), DCT_CONST_BITS);
|
|
|
|
val = ROUND_POWER_OF_TWO(out, 5);
|
|
|
|
vec = __msa_fill_h(val);
|
2015-05-08 08:53:27 +02:00
|
|
|
|
2015-06-01 05:49:01 +02:00
|
|
|
VP9_ADDBLK_ST8x4_UB(dst, dst_stride, vec, vec, vec, vec);
|
|
|
|
dst += (4 * dst_stride);
|
|
|
|
VP9_ADDBLK_ST8x4_UB(dst, dst_stride, vec, vec, vec, vec);
|
2015-05-08 08:53:27 +02:00
|
|
|
}
|
|
|
|
|
2015-06-01 05:49:01 +02:00
|
|
|
void vp9_iht8x8_64_add_msa(const int16_t *input, uint8_t *dst,
|
|
|
|
int32_t dst_stride, int32_t tx_type) {
|
2015-05-08 08:53:27 +02:00
|
|
|
v8i16 in0, in1, in2, in3, in4, in5, in6, in7;
|
|
|
|
|
|
|
|
/* load vector elements of 8x8 block */
|
2015-06-01 05:49:01 +02:00
|
|
|
LD_SH8(input, 8, in0, in1, in2, in3, in4, in5, in6, in7);
|
2015-05-08 08:53:27 +02:00
|
|
|
|
2015-06-01 05:49:01 +02:00
|
|
|
TRANSPOSE8x8_SH_SH(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
2015-05-08 08:53:27 +02:00
|
|
|
|
|
|
|
switch (tx_type) {
|
|
|
|
case DCT_DCT:
|
|
|
|
/* DCT in horizontal */
|
|
|
|
VP9_IDCT8x8_1D(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
|
|
|
/* DCT in vertical */
|
2015-06-01 05:49:01 +02:00
|
|
|
TRANSPOSE8x8_SH_SH(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
2015-05-08 08:53:27 +02:00
|
|
|
VP9_IDCT8x8_1D(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
|
|
|
break;
|
|
|
|
case ADST_DCT:
|
|
|
|
/* DCT in horizontal */
|
|
|
|
VP9_IDCT8x8_1D(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
|
|
|
/* ADST in vertical */
|
2015-06-01 05:49:01 +02:00
|
|
|
TRANSPOSE8x8_SH_SH(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
2015-05-08 08:53:27 +02:00
|
|
|
VP9_ADST8(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
|
|
|
break;
|
|
|
|
case DCT_ADST:
|
|
|
|
/* ADST in horizontal */
|
2015-06-01 05:49:01 +02:00
|
|
|
VP9_ADST8(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
2015-05-08 08:53:27 +02:00
|
|
|
/* DCT in vertical */
|
2015-06-01 05:49:01 +02:00
|
|
|
TRANSPOSE8x8_SH_SH(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
2015-05-08 08:53:27 +02:00
|
|
|
VP9_IDCT8x8_1D(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
|
|
|
break;
|
|
|
|
case ADST_ADST:
|
|
|
|
/* ADST in horizontal */
|
|
|
|
VP9_ADST8(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
|
|
|
/* ADST in vertical */
|
2015-06-01 05:49:01 +02:00
|
|
|
TRANSPOSE8x8_SH_SH(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
2015-05-08 08:53:27 +02:00
|
|
|
VP9_ADST8(in0, in1, in2, in3, in4, in5, in6, in7,
|
|
|
|
in0, in1, in2, in3, in4, in5, in6, in7);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
assert(0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* final rounding (add 2^4, divide by 2^5) and shift */
|
2015-06-01 05:49:01 +02:00
|
|
|
SRARI_H4_SH(in0, in1, in2, in3, 5);
|
|
|
|
SRARI_H4_SH(in4, in5, in6, in7, 5);
|
2015-05-08 08:53:27 +02:00
|
|
|
|
|
|
|
/* add block and store 8x8 */
|
2015-06-01 05:49:01 +02:00
|
|
|
VP9_ADDBLK_ST8x4_UB(dst, dst_stride, in0, in1, in2, in3);
|
|
|
|
dst += (4 * dst_stride);
|
|
|
|
VP9_ADDBLK_ST8x4_UB(dst, dst_stride, in4, in5, in6, in7);
|
2015-05-08 08:53:27 +02:00
|
|
|
}
|