1*b2055c35SXin Li // Copyright 2014 Google Inc. All Rights Reserved.
2*b2055c35SXin Li //
3*b2055c35SXin Li // Use of this source code is governed by a BSD-style license
4*b2055c35SXin Li // that can be found in the COPYING file in the root of the source
5*b2055c35SXin Li // tree. An additional intellectual property rights grant can be found
6*b2055c35SXin Li // in the file PATENTS. All contributing project authors may
7*b2055c35SXin Li // be found in the AUTHORS file in the root of the source tree.
8*b2055c35SXin Li // -----------------------------------------------------------------------------
9*b2055c35SXin Li //
10*b2055c35SXin Li // MIPS version of dsp functions
11*b2055c35SXin Li //
12*b2055c35SXin Li // Author(s): Djordje Pesut ([email protected])
13*b2055c35SXin Li // Jovan Zelincevic ([email protected])
14*b2055c35SXin Li
15*b2055c35SXin Li #include "src/dsp/dsp.h"
16*b2055c35SXin Li
17*b2055c35SXin Li #if defined(WEBP_USE_MIPS32)
18*b2055c35SXin Li
19*b2055c35SXin Li #include "src/dsp/mips_macro.h"
20*b2055c35SXin Li
21*b2055c35SXin Li static const int kC1 = WEBP_TRANSFORM_AC3_C1;
22*b2055c35SXin Li static const int kC2 = WEBP_TRANSFORM_AC3_C2;
23*b2055c35SXin Li
abs_mips32(int x)24*b2055c35SXin Li static WEBP_INLINE int abs_mips32(int x) {
25*b2055c35SXin Li const int sign = x >> 31;
26*b2055c35SXin Li return (x ^ sign) - sign;
27*b2055c35SXin Li }
28*b2055c35SXin Li
29*b2055c35SXin Li // 4 pixels in, 2 pixels out
do_filter2(uint8_t * p,int step)30*b2055c35SXin Li static WEBP_INLINE void do_filter2(uint8_t* p, int step) {
31*b2055c35SXin Li const int p1 = p[-2 * step], p0 = p[-step], q0 = p[0], q1 = p[step];
32*b2055c35SXin Li const int a = 3 * (q0 - p0) + VP8ksclip1[p1 - q1];
33*b2055c35SXin Li const int a1 = VP8ksclip2[(a + 4) >> 3];
34*b2055c35SXin Li const int a2 = VP8ksclip2[(a + 3) >> 3];
35*b2055c35SXin Li p[-step] = VP8kclip1[p0 + a2];
36*b2055c35SXin Li p[ 0] = VP8kclip1[q0 - a1];
37*b2055c35SXin Li }
38*b2055c35SXin Li
39*b2055c35SXin Li // 4 pixels in, 4 pixels out
do_filter4(uint8_t * p,int step)40*b2055c35SXin Li static WEBP_INLINE void do_filter4(uint8_t* p, int step) {
41*b2055c35SXin Li const int p1 = p[-2 * step], p0 = p[-step], q0 = p[0], q1 = p[step];
42*b2055c35SXin Li const int a = 3 * (q0 - p0);
43*b2055c35SXin Li const int a1 = VP8ksclip2[(a + 4) >> 3];
44*b2055c35SXin Li const int a2 = VP8ksclip2[(a + 3) >> 3];
45*b2055c35SXin Li const int a3 = (a1 + 1) >> 1;
46*b2055c35SXin Li p[-2 * step] = VP8kclip1[p1 + a3];
47*b2055c35SXin Li p[- step] = VP8kclip1[p0 + a2];
48*b2055c35SXin Li p[ 0] = VP8kclip1[q0 - a1];
49*b2055c35SXin Li p[ step] = VP8kclip1[q1 - a3];
50*b2055c35SXin Li }
51*b2055c35SXin Li
52*b2055c35SXin Li // 6 pixels in, 6 pixels out
do_filter6(uint8_t * p,int step)53*b2055c35SXin Li static WEBP_INLINE void do_filter6(uint8_t* p, int step) {
54*b2055c35SXin Li const int p2 = p[-3 * step], p1 = p[-2 * step], p0 = p[-step];
55*b2055c35SXin Li const int q0 = p[0], q1 = p[step], q2 = p[2 * step];
56*b2055c35SXin Li const int a = VP8ksclip1[3 * (q0 - p0) + VP8ksclip1[p1 - q1]];
57*b2055c35SXin Li // a is in [-128,127], a1 in [-27,27], a2 in [-18,18] and a3 in [-9,9]
58*b2055c35SXin Li const int a1 = (27 * a + 63) >> 7; // eq. to ((3 * a + 7) * 9) >> 7
59*b2055c35SXin Li const int a2 = (18 * a + 63) >> 7; // eq. to ((2 * a + 7) * 9) >> 7
60*b2055c35SXin Li const int a3 = (9 * a + 63) >> 7; // eq. to ((1 * a + 7) * 9) >> 7
61*b2055c35SXin Li p[-3 * step] = VP8kclip1[p2 + a3];
62*b2055c35SXin Li p[-2 * step] = VP8kclip1[p1 + a2];
63*b2055c35SXin Li p[- step] = VP8kclip1[p0 + a1];
64*b2055c35SXin Li p[ 0] = VP8kclip1[q0 - a1];
65*b2055c35SXin Li p[ step] = VP8kclip1[q1 - a2];
66*b2055c35SXin Li p[ 2 * step] = VP8kclip1[q2 - a3];
67*b2055c35SXin Li }
68*b2055c35SXin Li
hev(const uint8_t * p,int step,int thresh)69*b2055c35SXin Li static WEBP_INLINE int hev(const uint8_t* p, int step, int thresh) {
70*b2055c35SXin Li const int p1 = p[-2 * step], p0 = p[-step], q0 = p[0], q1 = p[step];
71*b2055c35SXin Li return (abs_mips32(p1 - p0) > thresh) || (abs_mips32(q1 - q0) > thresh);
72*b2055c35SXin Li }
73*b2055c35SXin Li
needs_filter(const uint8_t * p,int step,int t)74*b2055c35SXin Li static WEBP_INLINE int needs_filter(const uint8_t* p, int step, int t) {
75*b2055c35SXin Li const int p1 = p[-2 * step], p0 = p[-step], q0 = p[0], q1 = p[step];
76*b2055c35SXin Li return ((4 * abs_mips32(p0 - q0) + abs_mips32(p1 - q1)) <= t);
77*b2055c35SXin Li }
78*b2055c35SXin Li
needs_filter2(const uint8_t * p,int step,int t,int it)79*b2055c35SXin Li static WEBP_INLINE int needs_filter2(const uint8_t* p,
80*b2055c35SXin Li int step, int t, int it) {
81*b2055c35SXin Li const int p3 = p[-4 * step], p2 = p[-3 * step];
82*b2055c35SXin Li const int p1 = p[-2 * step], p0 = p[-step];
83*b2055c35SXin Li const int q0 = p[0], q1 = p[step], q2 = p[2 * step], q3 = p[3 * step];
84*b2055c35SXin Li if ((4 * abs_mips32(p0 - q0) + abs_mips32(p1 - q1)) > t) {
85*b2055c35SXin Li return 0;
86*b2055c35SXin Li }
87*b2055c35SXin Li return abs_mips32(p3 - p2) <= it && abs_mips32(p2 - p1) <= it &&
88*b2055c35SXin Li abs_mips32(p1 - p0) <= it && abs_mips32(q3 - q2) <= it &&
89*b2055c35SXin Li abs_mips32(q2 - q1) <= it && abs_mips32(q1 - q0) <= it;
90*b2055c35SXin Li }
91*b2055c35SXin Li
FilterLoop26(uint8_t * p,int hstride,int vstride,int size,int thresh,int ithresh,int hev_thresh)92*b2055c35SXin Li static WEBP_INLINE void FilterLoop26(uint8_t* p,
93*b2055c35SXin Li int hstride, int vstride, int size,
94*b2055c35SXin Li int thresh, int ithresh, int hev_thresh) {
95*b2055c35SXin Li const int thresh2 = 2 * thresh + 1;
96*b2055c35SXin Li while (size-- > 0) {
97*b2055c35SXin Li if (needs_filter2(p, hstride, thresh2, ithresh)) {
98*b2055c35SXin Li if (hev(p, hstride, hev_thresh)) {
99*b2055c35SXin Li do_filter2(p, hstride);
100*b2055c35SXin Li } else {
101*b2055c35SXin Li do_filter6(p, hstride);
102*b2055c35SXin Li }
103*b2055c35SXin Li }
104*b2055c35SXin Li p += vstride;
105*b2055c35SXin Li }
106*b2055c35SXin Li }
107*b2055c35SXin Li
FilterLoop24(uint8_t * p,int hstride,int vstride,int size,int thresh,int ithresh,int hev_thresh)108*b2055c35SXin Li static WEBP_INLINE void FilterLoop24(uint8_t* p,
109*b2055c35SXin Li int hstride, int vstride, int size,
110*b2055c35SXin Li int thresh, int ithresh, int hev_thresh) {
111*b2055c35SXin Li const int thresh2 = 2 * thresh + 1;
112*b2055c35SXin Li while (size-- > 0) {
113*b2055c35SXin Li if (needs_filter2(p, hstride, thresh2, ithresh)) {
114*b2055c35SXin Li if (hev(p, hstride, hev_thresh)) {
115*b2055c35SXin Li do_filter2(p, hstride);
116*b2055c35SXin Li } else {
117*b2055c35SXin Li do_filter4(p, hstride);
118*b2055c35SXin Li }
119*b2055c35SXin Li }
120*b2055c35SXin Li p += vstride;
121*b2055c35SXin Li }
122*b2055c35SXin Li }
123*b2055c35SXin Li
124*b2055c35SXin Li // on macroblock edges
VFilter16(uint8_t * p,int stride,int thresh,int ithresh,int hev_thresh)125*b2055c35SXin Li static void VFilter16(uint8_t* p, int stride,
126*b2055c35SXin Li int thresh, int ithresh, int hev_thresh) {
127*b2055c35SXin Li FilterLoop26(p, stride, 1, 16, thresh, ithresh, hev_thresh);
128*b2055c35SXin Li }
129*b2055c35SXin Li
HFilter16(uint8_t * p,int stride,int thresh,int ithresh,int hev_thresh)130*b2055c35SXin Li static void HFilter16(uint8_t* p, int stride,
131*b2055c35SXin Li int thresh, int ithresh, int hev_thresh) {
132*b2055c35SXin Li FilterLoop26(p, 1, stride, 16, thresh, ithresh, hev_thresh);
133*b2055c35SXin Li }
134*b2055c35SXin Li
135*b2055c35SXin Li // 8-pixels wide variant, for chroma filtering
VFilter8(uint8_t * u,uint8_t * v,int stride,int thresh,int ithresh,int hev_thresh)136*b2055c35SXin Li static void VFilter8(uint8_t* u, uint8_t* v, int stride,
137*b2055c35SXin Li int thresh, int ithresh, int hev_thresh) {
138*b2055c35SXin Li FilterLoop26(u, stride, 1, 8, thresh, ithresh, hev_thresh);
139*b2055c35SXin Li FilterLoop26(v, stride, 1, 8, thresh, ithresh, hev_thresh);
140*b2055c35SXin Li }
141*b2055c35SXin Li
HFilter8(uint8_t * u,uint8_t * v,int stride,int thresh,int ithresh,int hev_thresh)142*b2055c35SXin Li static void HFilter8(uint8_t* u, uint8_t* v, int stride,
143*b2055c35SXin Li int thresh, int ithresh, int hev_thresh) {
144*b2055c35SXin Li FilterLoop26(u, 1, stride, 8, thresh, ithresh, hev_thresh);
145*b2055c35SXin Li FilterLoop26(v, 1, stride, 8, thresh, ithresh, hev_thresh);
146*b2055c35SXin Li }
147*b2055c35SXin Li
VFilter8i(uint8_t * u,uint8_t * v,int stride,int thresh,int ithresh,int hev_thresh)148*b2055c35SXin Li static void VFilter8i(uint8_t* u, uint8_t* v, int stride,
149*b2055c35SXin Li int thresh, int ithresh, int hev_thresh) {
150*b2055c35SXin Li FilterLoop24(u + 4 * stride, stride, 1, 8, thresh, ithresh, hev_thresh);
151*b2055c35SXin Li FilterLoop24(v + 4 * stride, stride, 1, 8, thresh, ithresh, hev_thresh);
152*b2055c35SXin Li }
153*b2055c35SXin Li
HFilter8i(uint8_t * u,uint8_t * v,int stride,int thresh,int ithresh,int hev_thresh)154*b2055c35SXin Li static void HFilter8i(uint8_t* u, uint8_t* v, int stride,
155*b2055c35SXin Li int thresh, int ithresh, int hev_thresh) {
156*b2055c35SXin Li FilterLoop24(u + 4, 1, stride, 8, thresh, ithresh, hev_thresh);
157*b2055c35SXin Li FilterLoop24(v + 4, 1, stride, 8, thresh, ithresh, hev_thresh);
158*b2055c35SXin Li }
159*b2055c35SXin Li
160*b2055c35SXin Li // on three inner edges
VFilter16i(uint8_t * p,int stride,int thresh,int ithresh,int hev_thresh)161*b2055c35SXin Li static void VFilter16i(uint8_t* p, int stride,
162*b2055c35SXin Li int thresh, int ithresh, int hev_thresh) {
163*b2055c35SXin Li int k;
164*b2055c35SXin Li for (k = 3; k > 0; --k) {
165*b2055c35SXin Li p += 4 * stride;
166*b2055c35SXin Li FilterLoop24(p, stride, 1, 16, thresh, ithresh, hev_thresh);
167*b2055c35SXin Li }
168*b2055c35SXin Li }
169*b2055c35SXin Li
HFilter16i(uint8_t * p,int stride,int thresh,int ithresh,int hev_thresh)170*b2055c35SXin Li static void HFilter16i(uint8_t* p, int stride,
171*b2055c35SXin Li int thresh, int ithresh, int hev_thresh) {
172*b2055c35SXin Li int k;
173*b2055c35SXin Li for (k = 3; k > 0; --k) {
174*b2055c35SXin Li p += 4;
175*b2055c35SXin Li FilterLoop24(p, 1, stride, 16, thresh, ithresh, hev_thresh);
176*b2055c35SXin Li }
177*b2055c35SXin Li }
178*b2055c35SXin Li
179*b2055c35SXin Li //------------------------------------------------------------------------------
180*b2055c35SXin Li // Simple In-loop filtering (Paragraph 15.2)
181*b2055c35SXin Li
SimpleVFilter16(uint8_t * p,int stride,int thresh)182*b2055c35SXin Li static void SimpleVFilter16(uint8_t* p, int stride, int thresh) {
183*b2055c35SXin Li int i;
184*b2055c35SXin Li const int thresh2 = 2 * thresh + 1;
185*b2055c35SXin Li for (i = 0; i < 16; ++i) {
186*b2055c35SXin Li if (needs_filter(p + i, stride, thresh2)) {
187*b2055c35SXin Li do_filter2(p + i, stride);
188*b2055c35SXin Li }
189*b2055c35SXin Li }
190*b2055c35SXin Li }
191*b2055c35SXin Li
SimpleHFilter16(uint8_t * p,int stride,int thresh)192*b2055c35SXin Li static void SimpleHFilter16(uint8_t* p, int stride, int thresh) {
193*b2055c35SXin Li int i;
194*b2055c35SXin Li const int thresh2 = 2 * thresh + 1;
195*b2055c35SXin Li for (i = 0; i < 16; ++i) {
196*b2055c35SXin Li if (needs_filter(p + i * stride, 1, thresh2)) {
197*b2055c35SXin Li do_filter2(p + i * stride, 1);
198*b2055c35SXin Li }
199*b2055c35SXin Li }
200*b2055c35SXin Li }
201*b2055c35SXin Li
SimpleVFilter16i(uint8_t * p,int stride,int thresh)202*b2055c35SXin Li static void SimpleVFilter16i(uint8_t* p, int stride, int thresh) {
203*b2055c35SXin Li int k;
204*b2055c35SXin Li for (k = 3; k > 0; --k) {
205*b2055c35SXin Li p += 4 * stride;
206*b2055c35SXin Li SimpleVFilter16(p, stride, thresh);
207*b2055c35SXin Li }
208*b2055c35SXin Li }
209*b2055c35SXin Li
SimpleHFilter16i(uint8_t * p,int stride,int thresh)210*b2055c35SXin Li static void SimpleHFilter16i(uint8_t* p, int stride, int thresh) {
211*b2055c35SXin Li int k;
212*b2055c35SXin Li for (k = 3; k > 0; --k) {
213*b2055c35SXin Li p += 4;
214*b2055c35SXin Li SimpleHFilter16(p, stride, thresh);
215*b2055c35SXin Li }
216*b2055c35SXin Li }
217*b2055c35SXin Li
TransformOne(const int16_t * in,uint8_t * dst)218*b2055c35SXin Li static void TransformOne(const int16_t* in, uint8_t* dst) {
219*b2055c35SXin Li int temp0, temp1, temp2, temp3, temp4;
220*b2055c35SXin Li int temp5, temp6, temp7, temp8, temp9;
221*b2055c35SXin Li int temp10, temp11, temp12, temp13, temp14;
222*b2055c35SXin Li int temp15, temp16, temp17, temp18, temp19;
223*b2055c35SXin Li int16_t* p_in = (int16_t*)in;
224*b2055c35SXin Li
225*b2055c35SXin Li // loops unrolled and merged to avoid usage of tmp buffer
226*b2055c35SXin Li // and to reduce number of stalls. MUL macro is written
227*b2055c35SXin Li // in assembler and inlined
228*b2055c35SXin Li __asm__ volatile(
229*b2055c35SXin Li "lh %[temp0], 0(%[in]) \n\t"
230*b2055c35SXin Li "lh %[temp8], 16(%[in]) \n\t"
231*b2055c35SXin Li "lh %[temp4], 8(%[in]) \n\t"
232*b2055c35SXin Li "lh %[temp12], 24(%[in]) \n\t"
233*b2055c35SXin Li "addu %[temp16], %[temp0], %[temp8] \n\t"
234*b2055c35SXin Li "subu %[temp0], %[temp0], %[temp8] \n\t"
235*b2055c35SXin Li "mul %[temp8], %[temp4], %[kC2] \n\t"
236*b2055c35SXin Li MUL_SHIFT_C1(temp17, temp12)
237*b2055c35SXin Li MUL_SHIFT_C1_IO(temp4, temp19)
238*b2055c35SXin Li "mul %[temp12], %[temp12], %[kC2] \n\t"
239*b2055c35SXin Li "lh %[temp1], 2(%[in]) \n\t"
240*b2055c35SXin Li "lh %[temp5], 10(%[in]) \n\t"
241*b2055c35SXin Li "lh %[temp9], 18(%[in]) \n\t"
242*b2055c35SXin Li "lh %[temp13], 26(%[in]) \n\t"
243*b2055c35SXin Li "sra %[temp8], %[temp8], 16 \n\t"
244*b2055c35SXin Li "sra %[temp12], %[temp12], 16 \n\t"
245*b2055c35SXin Li "lh %[temp2], 4(%[in]) \n\t"
246*b2055c35SXin Li "lh %[temp6], 12(%[in]) \n\t"
247*b2055c35SXin Li "lh %[temp10], 20(%[in]) \n\t"
248*b2055c35SXin Li "lh %[temp14], 28(%[in]) \n\t"
249*b2055c35SXin Li "subu %[temp17], %[temp8], %[temp17] \n\t"
250*b2055c35SXin Li "addu %[temp4], %[temp4], %[temp12] \n\t"
251*b2055c35SXin Li "addu %[temp8], %[temp16], %[temp4] \n\t"
252*b2055c35SXin Li "subu %[temp4], %[temp16], %[temp4] \n\t"
253*b2055c35SXin Li "addu %[temp16], %[temp1], %[temp9] \n\t"
254*b2055c35SXin Li "subu %[temp1], %[temp1], %[temp9] \n\t"
255*b2055c35SXin Li "lh %[temp3], 6(%[in]) \n\t"
256*b2055c35SXin Li "lh %[temp7], 14(%[in]) \n\t"
257*b2055c35SXin Li "lh %[temp11], 22(%[in]) \n\t"
258*b2055c35SXin Li "lh %[temp15], 30(%[in]) \n\t"
259*b2055c35SXin Li "addu %[temp12], %[temp0], %[temp17] \n\t"
260*b2055c35SXin Li "subu %[temp0], %[temp0], %[temp17] \n\t"
261*b2055c35SXin Li "mul %[temp9], %[temp5], %[kC2] \n\t"
262*b2055c35SXin Li MUL_SHIFT_C1(temp17, temp13)
263*b2055c35SXin Li MUL_SHIFT_C1_IO(temp5, temp19)
264*b2055c35SXin Li "mul %[temp13], %[temp13], %[kC2] \n\t"
265*b2055c35SXin Li "sra %[temp9], %[temp9], 16 \n\t"
266*b2055c35SXin Li "subu %[temp17], %[temp9], %[temp17] \n\t"
267*b2055c35SXin Li "sra %[temp13], %[temp13], 16 \n\t"
268*b2055c35SXin Li "addu %[temp5], %[temp5], %[temp13] \n\t"
269*b2055c35SXin Li "addu %[temp13], %[temp1], %[temp17] \n\t"
270*b2055c35SXin Li "subu %[temp1], %[temp1], %[temp17] \n\t"
271*b2055c35SXin Li MUL_SHIFT_C1(temp17, temp14)
272*b2055c35SXin Li "mul %[temp14], %[temp14], %[kC2] \n\t"
273*b2055c35SXin Li "addu %[temp9], %[temp16], %[temp5] \n\t"
274*b2055c35SXin Li "subu %[temp5], %[temp16], %[temp5] \n\t"
275*b2055c35SXin Li "addu %[temp16], %[temp2], %[temp10] \n\t"
276*b2055c35SXin Li "subu %[temp2], %[temp2], %[temp10] \n\t"
277*b2055c35SXin Li "mul %[temp10], %[temp6], %[kC2] \n\t"
278*b2055c35SXin Li MUL_SHIFT_C1_IO(temp6, temp19)
279*b2055c35SXin Li "sra %[temp14], %[temp14], 16 \n\t"
280*b2055c35SXin Li "sra %[temp10], %[temp10], 16 \n\t"
281*b2055c35SXin Li "subu %[temp17], %[temp10], %[temp17] \n\t"
282*b2055c35SXin Li "addu %[temp6], %[temp6], %[temp14] \n\t"
283*b2055c35SXin Li "addu %[temp10], %[temp16], %[temp6] \n\t"
284*b2055c35SXin Li "subu %[temp6], %[temp16], %[temp6] \n\t"
285*b2055c35SXin Li "addu %[temp14], %[temp2], %[temp17] \n\t"
286*b2055c35SXin Li "subu %[temp2], %[temp2], %[temp17] \n\t"
287*b2055c35SXin Li MUL_SHIFT_C1(temp17, temp15)
288*b2055c35SXin Li "mul %[temp15], %[temp15], %[kC2] \n\t"
289*b2055c35SXin Li "addu %[temp16], %[temp3], %[temp11] \n\t"
290*b2055c35SXin Li "subu %[temp3], %[temp3], %[temp11] \n\t"
291*b2055c35SXin Li "mul %[temp11], %[temp7], %[kC2] \n\t"
292*b2055c35SXin Li MUL_SHIFT_C1_IO(temp7, temp19)
293*b2055c35SXin Li "addiu %[temp8], %[temp8], 4 \n\t"
294*b2055c35SXin Li "addiu %[temp12], %[temp12], 4 \n\t"
295*b2055c35SXin Li "addiu %[temp0], %[temp0], 4 \n\t"
296*b2055c35SXin Li "addiu %[temp4], %[temp4], 4 \n\t"
297*b2055c35SXin Li "sra %[temp15], %[temp15], 16 \n\t"
298*b2055c35SXin Li "sra %[temp11], %[temp11], 16 \n\t"
299*b2055c35SXin Li "subu %[temp17], %[temp11], %[temp17] \n\t"
300*b2055c35SXin Li "addu %[temp7], %[temp7], %[temp15] \n\t"
301*b2055c35SXin Li "addu %[temp15], %[temp3], %[temp17] \n\t"
302*b2055c35SXin Li "subu %[temp3], %[temp3], %[temp17] \n\t"
303*b2055c35SXin Li "addu %[temp11], %[temp16], %[temp7] \n\t"
304*b2055c35SXin Li "subu %[temp7], %[temp16], %[temp7] \n\t"
305*b2055c35SXin Li "addu %[temp16], %[temp8], %[temp10] \n\t"
306*b2055c35SXin Li "subu %[temp8], %[temp8], %[temp10] \n\t"
307*b2055c35SXin Li "mul %[temp10], %[temp9], %[kC2] \n\t"
308*b2055c35SXin Li MUL_SHIFT_C1(temp17, temp11)
309*b2055c35SXin Li MUL_SHIFT_C1_IO(temp9, temp19)
310*b2055c35SXin Li "mul %[temp11], %[temp11], %[kC2] \n\t"
311*b2055c35SXin Li "sra %[temp10], %[temp10], 16 \n\t"
312*b2055c35SXin Li "sra %[temp11], %[temp11], 16 \n\t"
313*b2055c35SXin Li "subu %[temp17], %[temp10], %[temp17] \n\t"
314*b2055c35SXin Li "addu %[temp11], %[temp9], %[temp11] \n\t"
315*b2055c35SXin Li "addu %[temp10], %[temp12], %[temp14] \n\t"
316*b2055c35SXin Li "subu %[temp12], %[temp12], %[temp14] \n\t"
317*b2055c35SXin Li "mul %[temp14], %[temp13], %[kC2] \n\t"
318*b2055c35SXin Li MUL_SHIFT_C1(temp9, temp15)
319*b2055c35SXin Li MUL_SHIFT_C1_IO(temp13, temp19)
320*b2055c35SXin Li "mul %[temp15], %[temp15], %[kC2] \n\t"
321*b2055c35SXin Li "sra %[temp14], %[temp14], 16 \n\t"
322*b2055c35SXin Li "sra %[temp15], %[temp15], 16 \n\t"
323*b2055c35SXin Li "subu %[temp9], %[temp14], %[temp9] \n\t"
324*b2055c35SXin Li "addu %[temp15], %[temp13], %[temp15] \n\t"
325*b2055c35SXin Li "addu %[temp14], %[temp0], %[temp2] \n\t"
326*b2055c35SXin Li "subu %[temp0], %[temp0], %[temp2] \n\t"
327*b2055c35SXin Li "mul %[temp2], %[temp1], %[kC2] \n\t"
328*b2055c35SXin Li MUL_SHIFT_C1(temp13, temp3)
329*b2055c35SXin Li MUL_SHIFT_C1_IO(temp1, temp19)
330*b2055c35SXin Li "mul %[temp3], %[temp3], %[kC2] \n\t"
331*b2055c35SXin Li "sra %[temp2], %[temp2], 16 \n\t"
332*b2055c35SXin Li "sra %[temp3], %[temp3], 16 \n\t"
333*b2055c35SXin Li "subu %[temp13], %[temp2], %[temp13] \n\t"
334*b2055c35SXin Li "addu %[temp3], %[temp1], %[temp3] \n\t"
335*b2055c35SXin Li "addu %[temp2], %[temp4], %[temp6] \n\t"
336*b2055c35SXin Li "subu %[temp4], %[temp4], %[temp6] \n\t"
337*b2055c35SXin Li "mul %[temp6], %[temp5], %[kC2] \n\t"
338*b2055c35SXin Li MUL_SHIFT_C1(temp1, temp7)
339*b2055c35SXin Li MUL_SHIFT_C1_IO(temp5, temp19)
340*b2055c35SXin Li "mul %[temp7], %[temp7], %[kC2] \n\t"
341*b2055c35SXin Li "sra %[temp6], %[temp6], 16 \n\t"
342*b2055c35SXin Li "sra %[temp7], %[temp7], 16 \n\t"
343*b2055c35SXin Li "subu %[temp1], %[temp6], %[temp1] \n\t"
344*b2055c35SXin Li "addu %[temp7], %[temp5], %[temp7] \n\t"
345*b2055c35SXin Li "addu %[temp5], %[temp16], %[temp11] \n\t"
346*b2055c35SXin Li "subu %[temp16], %[temp16], %[temp11] \n\t"
347*b2055c35SXin Li "addu %[temp11], %[temp8], %[temp17] \n\t"
348*b2055c35SXin Li "subu %[temp8], %[temp8], %[temp17] \n\t"
349*b2055c35SXin Li "sra %[temp5], %[temp5], 3 \n\t"
350*b2055c35SXin Li "sra %[temp16], %[temp16], 3 \n\t"
351*b2055c35SXin Li "sra %[temp11], %[temp11], 3 \n\t"
352*b2055c35SXin Li "sra %[temp8], %[temp8], 3 \n\t"
353*b2055c35SXin Li "addu %[temp17], %[temp10], %[temp15] \n\t"
354*b2055c35SXin Li "subu %[temp10], %[temp10], %[temp15] \n\t"
355*b2055c35SXin Li "addu %[temp15], %[temp12], %[temp9] \n\t"
356*b2055c35SXin Li "subu %[temp12], %[temp12], %[temp9] \n\t"
357*b2055c35SXin Li "sra %[temp17], %[temp17], 3 \n\t"
358*b2055c35SXin Li "sra %[temp10], %[temp10], 3 \n\t"
359*b2055c35SXin Li "sra %[temp15], %[temp15], 3 \n\t"
360*b2055c35SXin Li "sra %[temp12], %[temp12], 3 \n\t"
361*b2055c35SXin Li "addu %[temp9], %[temp14], %[temp3] \n\t"
362*b2055c35SXin Li "subu %[temp14], %[temp14], %[temp3] \n\t"
363*b2055c35SXin Li "addu %[temp3], %[temp0], %[temp13] \n\t"
364*b2055c35SXin Li "subu %[temp0], %[temp0], %[temp13] \n\t"
365*b2055c35SXin Li "sra %[temp9], %[temp9], 3 \n\t"
366*b2055c35SXin Li "sra %[temp14], %[temp14], 3 \n\t"
367*b2055c35SXin Li "sra %[temp3], %[temp3], 3 \n\t"
368*b2055c35SXin Li "sra %[temp0], %[temp0], 3 \n\t"
369*b2055c35SXin Li "addu %[temp13], %[temp2], %[temp7] \n\t"
370*b2055c35SXin Li "subu %[temp2], %[temp2], %[temp7] \n\t"
371*b2055c35SXin Li "addu %[temp7], %[temp4], %[temp1] \n\t"
372*b2055c35SXin Li "subu %[temp4], %[temp4], %[temp1] \n\t"
373*b2055c35SXin Li "sra %[temp13], %[temp13], 3 \n\t"
374*b2055c35SXin Li "sra %[temp2], %[temp2], 3 \n\t"
375*b2055c35SXin Li "sra %[temp7], %[temp7], 3 \n\t"
376*b2055c35SXin Li "sra %[temp4], %[temp4], 3 \n\t"
377*b2055c35SXin Li "addiu %[temp6], $zero, 255 \n\t"
378*b2055c35SXin Li "lbu %[temp1], 0+0*" XSTR(BPS) "(%[dst]) \n\t"
379*b2055c35SXin Li "addu %[temp1], %[temp1], %[temp5] \n\t"
380*b2055c35SXin Li "sra %[temp5], %[temp1], 8 \n\t"
381*b2055c35SXin Li "sra %[temp18], %[temp1], 31 \n\t"
382*b2055c35SXin Li "beqz %[temp5], 1f \n\t"
383*b2055c35SXin Li "xor %[temp1], %[temp1], %[temp1] \n\t"
384*b2055c35SXin Li "movz %[temp1], %[temp6], %[temp18] \n\t"
385*b2055c35SXin Li "1: \n\t"
386*b2055c35SXin Li "lbu %[temp18], 1+0*" XSTR(BPS) "(%[dst]) \n\t"
387*b2055c35SXin Li "sb %[temp1], 0+0*" XSTR(BPS) "(%[dst]) \n\t"
388*b2055c35SXin Li "addu %[temp18], %[temp18], %[temp11] \n\t"
389*b2055c35SXin Li "sra %[temp11], %[temp18], 8 \n\t"
390*b2055c35SXin Li "sra %[temp1], %[temp18], 31 \n\t"
391*b2055c35SXin Li "beqz %[temp11], 2f \n\t"
392*b2055c35SXin Li "xor %[temp18], %[temp18], %[temp18] \n\t"
393*b2055c35SXin Li "movz %[temp18], %[temp6], %[temp1] \n\t"
394*b2055c35SXin Li "2: \n\t"
395*b2055c35SXin Li "lbu %[temp1], 2+0*" XSTR(BPS) "(%[dst]) \n\t"
396*b2055c35SXin Li "sb %[temp18], 1+0*" XSTR(BPS) "(%[dst]) \n\t"
397*b2055c35SXin Li "addu %[temp1], %[temp1], %[temp8] \n\t"
398*b2055c35SXin Li "sra %[temp8], %[temp1], 8 \n\t"
399*b2055c35SXin Li "sra %[temp18], %[temp1], 31 \n\t"
400*b2055c35SXin Li "beqz %[temp8], 3f \n\t"
401*b2055c35SXin Li "xor %[temp1], %[temp1], %[temp1] \n\t"
402*b2055c35SXin Li "movz %[temp1], %[temp6], %[temp18] \n\t"
403*b2055c35SXin Li "3: \n\t"
404*b2055c35SXin Li "lbu %[temp18], 3+0*" XSTR(BPS) "(%[dst]) \n\t"
405*b2055c35SXin Li "sb %[temp1], 2+0*" XSTR(BPS) "(%[dst]) \n\t"
406*b2055c35SXin Li "addu %[temp18], %[temp18], %[temp16] \n\t"
407*b2055c35SXin Li "sra %[temp16], %[temp18], 8 \n\t"
408*b2055c35SXin Li "sra %[temp1], %[temp18], 31 \n\t"
409*b2055c35SXin Li "beqz %[temp16], 4f \n\t"
410*b2055c35SXin Li "xor %[temp18], %[temp18], %[temp18] \n\t"
411*b2055c35SXin Li "movz %[temp18], %[temp6], %[temp1] \n\t"
412*b2055c35SXin Li "4: \n\t"
413*b2055c35SXin Li "sb %[temp18], 3+0*" XSTR(BPS) "(%[dst]) \n\t"
414*b2055c35SXin Li "lbu %[temp5], 0+1*" XSTR(BPS) "(%[dst]) \n\t"
415*b2055c35SXin Li "lbu %[temp8], 1+1*" XSTR(BPS) "(%[dst]) \n\t"
416*b2055c35SXin Li "lbu %[temp11], 2+1*" XSTR(BPS) "(%[dst]) \n\t"
417*b2055c35SXin Li "lbu %[temp16], 3+1*" XSTR(BPS) "(%[dst]) \n\t"
418*b2055c35SXin Li "addu %[temp5], %[temp5], %[temp17] \n\t"
419*b2055c35SXin Li "addu %[temp8], %[temp8], %[temp15] \n\t"
420*b2055c35SXin Li "addu %[temp11], %[temp11], %[temp12] \n\t"
421*b2055c35SXin Li "addu %[temp16], %[temp16], %[temp10] \n\t"
422*b2055c35SXin Li "sra %[temp18], %[temp5], 8 \n\t"
423*b2055c35SXin Li "sra %[temp1], %[temp5], 31 \n\t"
424*b2055c35SXin Li "beqz %[temp18], 5f \n\t"
425*b2055c35SXin Li "xor %[temp5], %[temp5], %[temp5] \n\t"
426*b2055c35SXin Li "movz %[temp5], %[temp6], %[temp1] \n\t"
427*b2055c35SXin Li "5: \n\t"
428*b2055c35SXin Li "sra %[temp18], %[temp8], 8 \n\t"
429*b2055c35SXin Li "sra %[temp1], %[temp8], 31 \n\t"
430*b2055c35SXin Li "beqz %[temp18], 6f \n\t"
431*b2055c35SXin Li "xor %[temp8], %[temp8], %[temp8] \n\t"
432*b2055c35SXin Li "movz %[temp8], %[temp6], %[temp1] \n\t"
433*b2055c35SXin Li "6: \n\t"
434*b2055c35SXin Li "sra %[temp18], %[temp11], 8 \n\t"
435*b2055c35SXin Li "sra %[temp1], %[temp11], 31 \n\t"
436*b2055c35SXin Li "sra %[temp17], %[temp16], 8 \n\t"
437*b2055c35SXin Li "sra %[temp15], %[temp16], 31 \n\t"
438*b2055c35SXin Li "beqz %[temp18], 7f \n\t"
439*b2055c35SXin Li "xor %[temp11], %[temp11], %[temp11] \n\t"
440*b2055c35SXin Li "movz %[temp11], %[temp6], %[temp1] \n\t"
441*b2055c35SXin Li "7: \n\t"
442*b2055c35SXin Li "beqz %[temp17], 8f \n\t"
443*b2055c35SXin Li "xor %[temp16], %[temp16], %[temp16] \n\t"
444*b2055c35SXin Li "movz %[temp16], %[temp6], %[temp15] \n\t"
445*b2055c35SXin Li "8: \n\t"
446*b2055c35SXin Li "sb %[temp5], 0+1*" XSTR(BPS) "(%[dst]) \n\t"
447*b2055c35SXin Li "sb %[temp8], 1+1*" XSTR(BPS) "(%[dst]) \n\t"
448*b2055c35SXin Li "sb %[temp11], 2+1*" XSTR(BPS) "(%[dst]) \n\t"
449*b2055c35SXin Li "sb %[temp16], 3+1*" XSTR(BPS) "(%[dst]) \n\t"
450*b2055c35SXin Li "lbu %[temp5], 0+2*" XSTR(BPS) "(%[dst]) \n\t"
451*b2055c35SXin Li "lbu %[temp8], 1+2*" XSTR(BPS) "(%[dst]) \n\t"
452*b2055c35SXin Li "lbu %[temp11], 2+2*" XSTR(BPS) "(%[dst]) \n\t"
453*b2055c35SXin Li "lbu %[temp16], 3+2*" XSTR(BPS) "(%[dst]) \n\t"
454*b2055c35SXin Li "addu %[temp5], %[temp5], %[temp9] \n\t"
455*b2055c35SXin Li "addu %[temp8], %[temp8], %[temp3] \n\t"
456*b2055c35SXin Li "addu %[temp11], %[temp11], %[temp0] \n\t"
457*b2055c35SXin Li "addu %[temp16], %[temp16], %[temp14] \n\t"
458*b2055c35SXin Li "sra %[temp18], %[temp5], 8 \n\t"
459*b2055c35SXin Li "sra %[temp1], %[temp5], 31 \n\t"
460*b2055c35SXin Li "sra %[temp17], %[temp8], 8 \n\t"
461*b2055c35SXin Li "sra %[temp15], %[temp8], 31 \n\t"
462*b2055c35SXin Li "sra %[temp12], %[temp11], 8 \n\t"
463*b2055c35SXin Li "sra %[temp10], %[temp11], 31 \n\t"
464*b2055c35SXin Li "sra %[temp9], %[temp16], 8 \n\t"
465*b2055c35SXin Li "sra %[temp3], %[temp16], 31 \n\t"
466*b2055c35SXin Li "beqz %[temp18], 9f \n\t"
467*b2055c35SXin Li "xor %[temp5], %[temp5], %[temp5] \n\t"
468*b2055c35SXin Li "movz %[temp5], %[temp6], %[temp1] \n\t"
469*b2055c35SXin Li "9: \n\t"
470*b2055c35SXin Li "beqz %[temp17], 10f \n\t"
471*b2055c35SXin Li "xor %[temp8], %[temp8], %[temp8] \n\t"
472*b2055c35SXin Li "movz %[temp8], %[temp6], %[temp15] \n\t"
473*b2055c35SXin Li "10: \n\t"
474*b2055c35SXin Li "beqz %[temp12], 11f \n\t"
475*b2055c35SXin Li "xor %[temp11], %[temp11], %[temp11] \n\t"
476*b2055c35SXin Li "movz %[temp11], %[temp6], %[temp10] \n\t"
477*b2055c35SXin Li "11: \n\t"
478*b2055c35SXin Li "beqz %[temp9], 12f \n\t"
479*b2055c35SXin Li "xor %[temp16], %[temp16], %[temp16] \n\t"
480*b2055c35SXin Li "movz %[temp16], %[temp6], %[temp3] \n\t"
481*b2055c35SXin Li "12: \n\t"
482*b2055c35SXin Li "sb %[temp5], 0+2*" XSTR(BPS) "(%[dst]) \n\t"
483*b2055c35SXin Li "sb %[temp8], 1+2*" XSTR(BPS) "(%[dst]) \n\t"
484*b2055c35SXin Li "sb %[temp11], 2+2*" XSTR(BPS) "(%[dst]) \n\t"
485*b2055c35SXin Li "sb %[temp16], 3+2*" XSTR(BPS) "(%[dst]) \n\t"
486*b2055c35SXin Li "lbu %[temp5], 0+3*" XSTR(BPS) "(%[dst]) \n\t"
487*b2055c35SXin Li "lbu %[temp8], 1+3*" XSTR(BPS) "(%[dst]) \n\t"
488*b2055c35SXin Li "lbu %[temp11], 2+3*" XSTR(BPS) "(%[dst]) \n\t"
489*b2055c35SXin Li "lbu %[temp16], 3+3*" XSTR(BPS) "(%[dst]) \n\t"
490*b2055c35SXin Li "addu %[temp5], %[temp5], %[temp13] \n\t"
491*b2055c35SXin Li "addu %[temp8], %[temp8], %[temp7] \n\t"
492*b2055c35SXin Li "addu %[temp11], %[temp11], %[temp4] \n\t"
493*b2055c35SXin Li "addu %[temp16], %[temp16], %[temp2] \n\t"
494*b2055c35SXin Li "sra %[temp18], %[temp5], 8 \n\t"
495*b2055c35SXin Li "sra %[temp1], %[temp5], 31 \n\t"
496*b2055c35SXin Li "sra %[temp17], %[temp8], 8 \n\t"
497*b2055c35SXin Li "sra %[temp15], %[temp8], 31 \n\t"
498*b2055c35SXin Li "sra %[temp12], %[temp11], 8 \n\t"
499*b2055c35SXin Li "sra %[temp10], %[temp11], 31 \n\t"
500*b2055c35SXin Li "sra %[temp9], %[temp16], 8 \n\t"
501*b2055c35SXin Li "sra %[temp3], %[temp16], 31 \n\t"
502*b2055c35SXin Li "beqz %[temp18], 13f \n\t"
503*b2055c35SXin Li "xor %[temp5], %[temp5], %[temp5] \n\t"
504*b2055c35SXin Li "movz %[temp5], %[temp6], %[temp1] \n\t"
505*b2055c35SXin Li "13: \n\t"
506*b2055c35SXin Li "beqz %[temp17], 14f \n\t"
507*b2055c35SXin Li "xor %[temp8], %[temp8], %[temp8] \n\t"
508*b2055c35SXin Li "movz %[temp8], %[temp6], %[temp15] \n\t"
509*b2055c35SXin Li "14: \n\t"
510*b2055c35SXin Li "beqz %[temp12], 15f \n\t"
511*b2055c35SXin Li "xor %[temp11], %[temp11], %[temp11] \n\t"
512*b2055c35SXin Li "movz %[temp11], %[temp6], %[temp10] \n\t"
513*b2055c35SXin Li "15: \n\t"
514*b2055c35SXin Li "beqz %[temp9], 16f \n\t"
515*b2055c35SXin Li "xor %[temp16], %[temp16], %[temp16] \n\t"
516*b2055c35SXin Li "movz %[temp16], %[temp6], %[temp3] \n\t"
517*b2055c35SXin Li "16: \n\t"
518*b2055c35SXin Li "sb %[temp5], 0+3*" XSTR(BPS) "(%[dst]) \n\t"
519*b2055c35SXin Li "sb %[temp8], 1+3*" XSTR(BPS) "(%[dst]) \n\t"
520*b2055c35SXin Li "sb %[temp11], 2+3*" XSTR(BPS) "(%[dst]) \n\t"
521*b2055c35SXin Li "sb %[temp16], 3+3*" XSTR(BPS) "(%[dst]) \n\t"
522*b2055c35SXin Li
523*b2055c35SXin Li : [temp0]"=&r"(temp0), [temp1]"=&r"(temp1), [temp2]"=&r"(temp2),
524*b2055c35SXin Li [temp3]"=&r"(temp3), [temp4]"=&r"(temp4), [temp5]"=&r"(temp5),
525*b2055c35SXin Li [temp6]"=&r"(temp6), [temp7]"=&r"(temp7), [temp8]"=&r"(temp8),
526*b2055c35SXin Li [temp9]"=&r"(temp9), [temp10]"=&r"(temp10), [temp11]"=&r"(temp11),
527*b2055c35SXin Li [temp12]"=&r"(temp12), [temp13]"=&r"(temp13), [temp14]"=&r"(temp14),
528*b2055c35SXin Li [temp15]"=&r"(temp15), [temp16]"=&r"(temp16), [temp17]"=&r"(temp17),
529*b2055c35SXin Li [temp18]"=&r"(temp18), [temp19]"=&r"(temp19)
530*b2055c35SXin Li : [in]"r"(p_in), [kC1]"r"(kC1), [kC2]"r"(kC2), [dst]"r"(dst)
531*b2055c35SXin Li : "memory", "hi", "lo"
532*b2055c35SXin Li );
533*b2055c35SXin Li }
534*b2055c35SXin Li
TransformTwo(const int16_t * in,uint8_t * dst,int do_two)535*b2055c35SXin Li static void TransformTwo(const int16_t* in, uint8_t* dst, int do_two) {
536*b2055c35SXin Li TransformOne(in, dst);
537*b2055c35SXin Li if (do_two) {
538*b2055c35SXin Li TransformOne(in + 16, dst + 4);
539*b2055c35SXin Li }
540*b2055c35SXin Li }
541*b2055c35SXin Li
542*b2055c35SXin Li //------------------------------------------------------------------------------
543*b2055c35SXin Li // Entry point
544*b2055c35SXin Li
545*b2055c35SXin Li extern void VP8DspInitMIPS32(void);
546*b2055c35SXin Li
VP8DspInitMIPS32(void)547*b2055c35SXin Li WEBP_TSAN_IGNORE_FUNCTION void VP8DspInitMIPS32(void) {
548*b2055c35SXin Li VP8InitClipTables();
549*b2055c35SXin Li
550*b2055c35SXin Li VP8Transform = TransformTwo;
551*b2055c35SXin Li
552*b2055c35SXin Li VP8VFilter16 = VFilter16;
553*b2055c35SXin Li VP8HFilter16 = HFilter16;
554*b2055c35SXin Li VP8VFilter8 = VFilter8;
555*b2055c35SXin Li VP8HFilter8 = HFilter8;
556*b2055c35SXin Li VP8VFilter16i = VFilter16i;
557*b2055c35SXin Li VP8HFilter16i = HFilter16i;
558*b2055c35SXin Li VP8VFilter8i = VFilter8i;
559*b2055c35SXin Li VP8HFilter8i = HFilter8i;
560*b2055c35SXin Li
561*b2055c35SXin Li VP8SimpleVFilter16 = SimpleVFilter16;
562*b2055c35SXin Li VP8SimpleHFilter16 = SimpleHFilter16;
563*b2055c35SXin Li VP8SimpleVFilter16i = SimpleVFilter16i;
564*b2055c35SXin Li VP8SimpleHFilter16i = SimpleHFilter16i;
565*b2055c35SXin Li }
566*b2055c35SXin Li
567*b2055c35SXin Li #else // !WEBP_USE_MIPS32
568*b2055c35SXin Li
569*b2055c35SXin Li WEBP_DSP_INIT_STUB(VP8DspInitMIPS32)
570*b2055c35SXin Li
571*b2055c35SXin Li #endif // WEBP_USE_MIPS32
572