xref: /aosp_15_r20/external/webp/src/dsp/rescaler_mips32.c (revision b2055c353e87c8814eb2b6b1b11112a1562253bd)
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 rescaling functions
11*b2055c35SXin Li //
12*b2055c35SXin Li // Author(s): Djordje Pesut ([email protected])
13*b2055c35SXin Li 
14*b2055c35SXin Li #include "src/dsp/dsp.h"
15*b2055c35SXin Li 
16*b2055c35SXin Li #if defined(WEBP_USE_MIPS32) && !defined(WEBP_REDUCE_SIZE)
17*b2055c35SXin Li 
18*b2055c35SXin Li #include <assert.h>
19*b2055c35SXin Li #include "src/utils/rescaler_utils.h"
20*b2055c35SXin Li 
21*b2055c35SXin Li //------------------------------------------------------------------------------
22*b2055c35SXin Li // Row import
23*b2055c35SXin Li 
ImportRowShrink_MIPS32(WebPRescaler * const wrk,const uint8_t * src)24*b2055c35SXin Li static void ImportRowShrink_MIPS32(WebPRescaler* const wrk,
25*b2055c35SXin Li                                    const uint8_t* src) {
26*b2055c35SXin Li   const int x_stride = wrk->num_channels;
27*b2055c35SXin Li   const int x_out_max = wrk->dst_width * wrk->num_channels;
28*b2055c35SXin Li   const int fx_scale = wrk->fx_scale;
29*b2055c35SXin Li   const int x_add = wrk->x_add;
30*b2055c35SXin Li   const int x_sub = wrk->x_sub;
31*b2055c35SXin Li   const int x_stride1 = x_stride << 2;
32*b2055c35SXin Li   int channel;
33*b2055c35SXin Li   assert(!wrk->x_expand);
34*b2055c35SXin Li   assert(!WebPRescalerInputDone(wrk));
35*b2055c35SXin Li 
36*b2055c35SXin Li   for (channel = 0; channel < x_stride; ++channel) {
37*b2055c35SXin Li     const uint8_t* src1 = src + channel;
38*b2055c35SXin Li     rescaler_t* frow = wrk->frow + channel;
39*b2055c35SXin Li     int temp1, temp2, temp3;
40*b2055c35SXin Li     int base, frac, sum;
41*b2055c35SXin Li     int accum, accum1;
42*b2055c35SXin Li     int loop_c = x_out_max - channel;
43*b2055c35SXin Li 
44*b2055c35SXin Li     __asm__ volatile (
45*b2055c35SXin Li       "li     %[temp1],   0x8000                    \n\t"
46*b2055c35SXin Li       "li     %[temp2],   0x10000                   \n\t"
47*b2055c35SXin Li       "li     %[sum],     0                         \n\t"
48*b2055c35SXin Li       "li     %[accum],   0                         \n\t"
49*b2055c35SXin Li     "1:                                             \n\t"
50*b2055c35SXin Li       "addu   %[accum],   %[accum],   %[x_add]      \n\t"
51*b2055c35SXin Li       "li     %[base],    0                         \n\t"
52*b2055c35SXin Li       "blez   %[accum],   3f                        \n\t"
53*b2055c35SXin Li     "2:                                             \n\t"
54*b2055c35SXin Li       "lbu    %[base],    0(%[src1])                \n\t"
55*b2055c35SXin Li       "subu   %[accum],   %[accum],   %[x_sub]      \n\t"
56*b2055c35SXin Li       "addu   %[src1],    %[src1],    %[x_stride]   \n\t"
57*b2055c35SXin Li       "addu   %[sum],     %[sum],     %[base]       \n\t"
58*b2055c35SXin Li       "bgtz   %[accum],   2b                        \n\t"
59*b2055c35SXin Li     "3:                                             \n\t"
60*b2055c35SXin Li       "negu   %[accum1],  %[accum]                  \n\t"
61*b2055c35SXin Li       "mul    %[frac],    %[base],    %[accum1]     \n\t"
62*b2055c35SXin Li       "mul    %[temp3],   %[sum],     %[x_sub]      \n\t"
63*b2055c35SXin Li       "subu   %[loop_c],  %[loop_c],  %[x_stride]   \n\t"
64*b2055c35SXin Li       "mult   %[temp1],   %[temp2]                  \n\t"
65*b2055c35SXin Li       "maddu  %[frac],    %[fx_scale]               \n\t"
66*b2055c35SXin Li       "mfhi   %[sum]                                \n\t"
67*b2055c35SXin Li       "subu   %[temp3],   %[temp3],   %[frac]       \n\t"
68*b2055c35SXin Li       "sw     %[temp3],   0(%[frow])                \n\t"
69*b2055c35SXin Li       "addu   %[frow],    %[frow],    %[x_stride1]  \n\t"
70*b2055c35SXin Li       "bgtz   %[loop_c],  1b                        \n\t"
71*b2055c35SXin Li       : [accum]"=&r"(accum), [src1]"+r"(src1), [temp3]"=&r"(temp3),
72*b2055c35SXin Li         [sum]"=&r"(sum), [base]"=&r"(base), [frac]"=&r"(frac),
73*b2055c35SXin Li         [frow]"+r"(frow), [accum1]"=&r"(accum1),
74*b2055c35SXin Li         [temp2]"=&r"(temp2), [temp1]"=&r"(temp1)
75*b2055c35SXin Li       : [x_stride]"r"(x_stride), [fx_scale]"r"(fx_scale),
76*b2055c35SXin Li         [x_sub]"r"(x_sub), [x_add]"r"(x_add),
77*b2055c35SXin Li         [loop_c]"r"(loop_c), [x_stride1]"r"(x_stride1)
78*b2055c35SXin Li       : "memory", "hi", "lo"
79*b2055c35SXin Li     );
80*b2055c35SXin Li     assert(accum == 0);
81*b2055c35SXin Li   }
82*b2055c35SXin Li }
83*b2055c35SXin Li 
ImportRowExpand_MIPS32(WebPRescaler * const wrk,const uint8_t * src)84*b2055c35SXin Li static void ImportRowExpand_MIPS32(WebPRescaler* const wrk,
85*b2055c35SXin Li                                    const uint8_t* src) {
86*b2055c35SXin Li   const int x_stride = wrk->num_channels;
87*b2055c35SXin Li   const int x_out_max = wrk->dst_width * wrk->num_channels;
88*b2055c35SXin Li   const int x_add = wrk->x_add;
89*b2055c35SXin Li   const int x_sub = wrk->x_sub;
90*b2055c35SXin Li   const int src_width = wrk->src_width;
91*b2055c35SXin Li   const int x_stride1 = x_stride << 2;
92*b2055c35SXin Li   int channel;
93*b2055c35SXin Li   assert(wrk->x_expand);
94*b2055c35SXin Li   assert(!WebPRescalerInputDone(wrk));
95*b2055c35SXin Li 
96*b2055c35SXin Li   for (channel = 0; channel < x_stride; ++channel) {
97*b2055c35SXin Li     const uint8_t* src1 = src + channel;
98*b2055c35SXin Li     rescaler_t* frow = wrk->frow + channel;
99*b2055c35SXin Li     int temp1, temp2, temp3, temp4;
100*b2055c35SXin Li     int frac;
101*b2055c35SXin Li     int accum;
102*b2055c35SXin Li     int x_out = channel;
103*b2055c35SXin Li 
104*b2055c35SXin Li     __asm__ volatile (
105*b2055c35SXin Li       "addiu  %[temp3],   %[src_width], -1            \n\t"
106*b2055c35SXin Li       "lbu    %[temp2],   0(%[src1])                  \n\t"
107*b2055c35SXin Li       "addu   %[src1],    %[src1],      %[x_stride]   \n\t"
108*b2055c35SXin Li       "bgtz   %[temp3],   0f                          \n\t"
109*b2055c35SXin Li       "addiu  %[temp1],   %[temp2],     0             \n\t"
110*b2055c35SXin Li       "b      3f                                      \n\t"
111*b2055c35SXin Li     "0:                                               \n\t"
112*b2055c35SXin Li       "lbu    %[temp1],   0(%[src1])                  \n\t"
113*b2055c35SXin Li     "3:                                               \n\t"
114*b2055c35SXin Li       "addiu  %[accum],   %[x_add],     0             \n\t"
115*b2055c35SXin Li     "1:                                               \n\t"
116*b2055c35SXin Li       "subu   %[temp3],   %[temp2],     %[temp1]      \n\t"
117*b2055c35SXin Li       "mul    %[temp3],   %[temp3],     %[accum]      \n\t"
118*b2055c35SXin Li       "mul    %[temp4],   %[temp1],     %[x_add]      \n\t"
119*b2055c35SXin Li       "addu   %[temp3],   %[temp4],     %[temp3]      \n\t"
120*b2055c35SXin Li       "sw     %[temp3],   0(%[frow])                  \n\t"
121*b2055c35SXin Li       "addu   %[frow],    %[frow],      %[x_stride1]  \n\t"
122*b2055c35SXin Li       "addu   %[x_out],   %[x_out],     %[x_stride]   \n\t"
123*b2055c35SXin Li       "subu   %[temp3],   %[x_out],     %[x_out_max]  \n\t"
124*b2055c35SXin Li       "bgez   %[temp3],   2f                          \n\t"
125*b2055c35SXin Li       "subu   %[accum],   %[accum],     %[x_sub]      \n\t"
126*b2055c35SXin Li       "bgez   %[accum],   4f                          \n\t"
127*b2055c35SXin Li       "addiu  %[temp2],   %[temp1],     0             \n\t"
128*b2055c35SXin Li       "addu   %[src1],    %[src1],      %[x_stride]   \n\t"
129*b2055c35SXin Li       "lbu    %[temp1],   0(%[src1])                  \n\t"
130*b2055c35SXin Li       "addu   %[accum],   %[accum],     %[x_add]      \n\t"
131*b2055c35SXin Li     "4:                                               \n\t"
132*b2055c35SXin Li       "b      1b                                      \n\t"
133*b2055c35SXin Li     "2:                                               \n\t"
134*b2055c35SXin Li       : [src1]"+r"(src1), [accum]"=&r"(accum), [temp1]"=&r"(temp1),
135*b2055c35SXin Li         [temp2]"=&r"(temp2), [temp3]"=&r"(temp3), [temp4]"=&r"(temp4),
136*b2055c35SXin Li         [x_out]"+r"(x_out), [frac]"=&r"(frac), [frow]"+r"(frow)
137*b2055c35SXin Li       : [x_stride]"r"(x_stride), [x_add]"r"(x_add), [x_sub]"r"(x_sub),
138*b2055c35SXin Li         [x_stride1]"r"(x_stride1), [src_width]"r"(src_width),
139*b2055c35SXin Li         [x_out_max]"r"(x_out_max)
140*b2055c35SXin Li       : "memory", "hi", "lo"
141*b2055c35SXin Li     );
142*b2055c35SXin Li     assert(wrk->x_sub == 0 /* <- special case for src_width=1 */ || accum == 0);
143*b2055c35SXin Li   }
144*b2055c35SXin Li }
145*b2055c35SXin Li 
146*b2055c35SXin Li //------------------------------------------------------------------------------
147*b2055c35SXin Li // Row export
148*b2055c35SXin Li 
ExportRowExpand_MIPS32(WebPRescaler * const wrk)149*b2055c35SXin Li static void ExportRowExpand_MIPS32(WebPRescaler* const wrk) {
150*b2055c35SXin Li   uint8_t* dst = wrk->dst;
151*b2055c35SXin Li   rescaler_t* irow = wrk->irow;
152*b2055c35SXin Li   const int x_out_max = wrk->dst_width * wrk->num_channels;
153*b2055c35SXin Li   const rescaler_t* frow = wrk->frow;
154*b2055c35SXin Li   int temp0, temp1, temp3, temp4, temp5, loop_end;
155*b2055c35SXin Li   const int temp2 = (int)wrk->fy_scale;
156*b2055c35SXin Li   const int temp6 = x_out_max << 2;
157*b2055c35SXin Li   assert(!WebPRescalerOutputDone(wrk));
158*b2055c35SXin Li   assert(wrk->y_accum <= 0);
159*b2055c35SXin Li   assert(wrk->y_expand);
160*b2055c35SXin Li   assert(wrk->y_sub != 0);
161*b2055c35SXin Li   if (wrk->y_accum == 0) {
162*b2055c35SXin Li     __asm__ volatile (
163*b2055c35SXin Li       "li       %[temp3],    0x10000                    \n\t"
164*b2055c35SXin Li       "li       %[temp4],    0x8000                     \n\t"
165*b2055c35SXin Li       "addu     %[loop_end], %[frow],     %[temp6]      \n\t"
166*b2055c35SXin Li     "1:                                                 \n\t"
167*b2055c35SXin Li       "lw       %[temp0],    0(%[frow])                 \n\t"
168*b2055c35SXin Li       "addiu    %[dst],      %[dst],      1             \n\t"
169*b2055c35SXin Li       "addiu    %[frow],     %[frow],     4             \n\t"
170*b2055c35SXin Li       "mult     %[temp3],    %[temp4]                   \n\t"
171*b2055c35SXin Li       "maddu    %[temp0],    %[temp2]                   \n\t"
172*b2055c35SXin Li       "mfhi     %[temp5]                                \n\t"
173*b2055c35SXin Li       "sb       %[temp5],    -1(%[dst])                 \n\t"
174*b2055c35SXin Li       "bne      %[frow],     %[loop_end], 1b            \n\t"
175*b2055c35SXin Li       : [temp0]"=&r"(temp0), [temp1]"=&r"(temp1), [temp3]"=&r"(temp3),
176*b2055c35SXin Li         [temp4]"=&r"(temp4), [temp5]"=&r"(temp5), [frow]"+r"(frow),
177*b2055c35SXin Li         [dst]"+r"(dst), [loop_end]"=&r"(loop_end)
178*b2055c35SXin Li       : [temp2]"r"(temp2), [temp6]"r"(temp6)
179*b2055c35SXin Li       : "memory", "hi", "lo"
180*b2055c35SXin Li     );
181*b2055c35SXin Li   } else {
182*b2055c35SXin Li     const uint32_t B = WEBP_RESCALER_FRAC(-wrk->y_accum, wrk->y_sub);
183*b2055c35SXin Li     const uint32_t A = (uint32_t)(WEBP_RESCALER_ONE - B);
184*b2055c35SXin Li     __asm__ volatile (
185*b2055c35SXin Li       "li       %[temp3],    0x10000                    \n\t"
186*b2055c35SXin Li       "li       %[temp4],    0x8000                     \n\t"
187*b2055c35SXin Li       "addu     %[loop_end], %[frow],     %[temp6]      \n\t"
188*b2055c35SXin Li     "1:                                                 \n\t"
189*b2055c35SXin Li       "lw       %[temp0],    0(%[frow])                 \n\t"
190*b2055c35SXin Li       "lw       %[temp1],    0(%[irow])                 \n\t"
191*b2055c35SXin Li       "addiu    %[dst],      %[dst],      1             \n\t"
192*b2055c35SXin Li       "mult     %[temp3],    %[temp4]                   \n\t"
193*b2055c35SXin Li       "maddu    %[A],        %[temp0]                   \n\t"
194*b2055c35SXin Li       "maddu    %[B],        %[temp1]                   \n\t"
195*b2055c35SXin Li       "addiu    %[frow],     %[frow],     4             \n\t"
196*b2055c35SXin Li       "addiu    %[irow],     %[irow],     4             \n\t"
197*b2055c35SXin Li       "mfhi     %[temp5]                                \n\t"
198*b2055c35SXin Li       "mult     %[temp3],    %[temp4]                   \n\t"
199*b2055c35SXin Li       "maddu    %[temp5],    %[temp2]                   \n\t"
200*b2055c35SXin Li       "mfhi     %[temp5]                                \n\t"
201*b2055c35SXin Li       "sb       %[temp5],    -1(%[dst])                 \n\t"
202*b2055c35SXin Li       "bne      %[frow],     %[loop_end], 1b            \n\t"
203*b2055c35SXin Li       : [temp0]"=&r"(temp0), [temp1]"=&r"(temp1), [temp3]"=&r"(temp3),
204*b2055c35SXin Li         [temp4]"=&r"(temp4), [temp5]"=&r"(temp5), [frow]"+r"(frow),
205*b2055c35SXin Li         [irow]"+r"(irow), [dst]"+r"(dst), [loop_end]"=&r"(loop_end)
206*b2055c35SXin Li       : [temp2]"r"(temp2), [temp6]"r"(temp6), [A]"r"(A), [B]"r"(B)
207*b2055c35SXin Li       : "memory", "hi", "lo"
208*b2055c35SXin Li     );
209*b2055c35SXin Li   }
210*b2055c35SXin Li }
211*b2055c35SXin Li 
212*b2055c35SXin Li #if 0  // disabled for now. TODO(skal): make match the C-code
213*b2055c35SXin Li static void ExportRowShrink_MIPS32(WebPRescaler* const wrk) {
214*b2055c35SXin Li   const int x_out_max = wrk->dst_width * wrk->num_channels;
215*b2055c35SXin Li   uint8_t* dst = wrk->dst;
216*b2055c35SXin Li   rescaler_t* irow = wrk->irow;
217*b2055c35SXin Li   const rescaler_t* frow = wrk->frow;
218*b2055c35SXin Li   const int yscale = wrk->fy_scale * (-wrk->y_accum);
219*b2055c35SXin Li   int temp0, temp1, temp3, temp4, temp5, loop_end;
220*b2055c35SXin Li   const int temp2 = (int)wrk->fxy_scale;
221*b2055c35SXin Li   const int temp6 = x_out_max << 2;
222*b2055c35SXin Li 
223*b2055c35SXin Li   assert(!WebPRescalerOutputDone(wrk));
224*b2055c35SXin Li   assert(wrk->y_accum <= 0);
225*b2055c35SXin Li   assert(!wrk->y_expand);
226*b2055c35SXin Li   assert(wrk->fxy_scale != 0);
227*b2055c35SXin Li   if (yscale) {
228*b2055c35SXin Li     __asm__ volatile (
229*b2055c35SXin Li       "li       %[temp3],    0x10000                    \n\t"
230*b2055c35SXin Li       "li       %[temp4],    0x8000                     \n\t"
231*b2055c35SXin Li       "addu     %[loop_end], %[frow],     %[temp6]      \n\t"
232*b2055c35SXin Li     "1:                                                 \n\t"
233*b2055c35SXin Li       "lw       %[temp0],    0(%[frow])                 \n\t"
234*b2055c35SXin Li       "mult     %[temp3],    %[temp4]                   \n\t"
235*b2055c35SXin Li       "addiu    %[frow],     %[frow],     4             \n\t"
236*b2055c35SXin Li       "maddu    %[temp0],    %[yscale]                  \n\t"
237*b2055c35SXin Li       "mfhi     %[temp1]                                \n\t"
238*b2055c35SXin Li       "lw       %[temp0],    0(%[irow])                 \n\t"
239*b2055c35SXin Li       "addiu    %[dst],      %[dst],      1             \n\t"
240*b2055c35SXin Li       "addiu    %[irow],     %[irow],     4             \n\t"
241*b2055c35SXin Li       "subu     %[temp0],    %[temp0],    %[temp1]      \n\t"
242*b2055c35SXin Li       "mult     %[temp3],    %[temp4]                   \n\t"
243*b2055c35SXin Li       "maddu    %[temp0],    %[temp2]                   \n\t"
244*b2055c35SXin Li       "mfhi     %[temp5]                                \n\t"
245*b2055c35SXin Li       "sw       %[temp1],    -4(%[irow])                \n\t"
246*b2055c35SXin Li       "sb       %[temp5],    -1(%[dst])                 \n\t"
247*b2055c35SXin Li       "bne      %[frow],     %[loop_end], 1b            \n\t"
248*b2055c35SXin Li       : [temp0]"=&r"(temp0), [temp1]"=&r"(temp1), [temp3]"=&r"(temp3),
249*b2055c35SXin Li         [temp4]"=&r"(temp4), [temp5]"=&r"(temp5), [frow]"+r"(frow),
250*b2055c35SXin Li         [irow]"+r"(irow), [dst]"+r"(dst), [loop_end]"=&r"(loop_end)
251*b2055c35SXin Li       : [temp2]"r"(temp2), [yscale]"r"(yscale), [temp6]"r"(temp6)
252*b2055c35SXin Li       : "memory", "hi", "lo"
253*b2055c35SXin Li     );
254*b2055c35SXin Li   } else {
255*b2055c35SXin Li     __asm__ volatile (
256*b2055c35SXin Li       "li       %[temp3],    0x10000                    \n\t"
257*b2055c35SXin Li       "li       %[temp4],    0x8000                     \n\t"
258*b2055c35SXin Li       "addu     %[loop_end], %[irow],     %[temp6]      \n\t"
259*b2055c35SXin Li     "1:                                                 \n\t"
260*b2055c35SXin Li       "lw       %[temp0],    0(%[irow])                 \n\t"
261*b2055c35SXin Li       "addiu    %[dst],      %[dst],      1             \n\t"
262*b2055c35SXin Li       "addiu    %[irow],     %[irow],     4             \n\t"
263*b2055c35SXin Li       "mult     %[temp3],    %[temp4]                   \n\t"
264*b2055c35SXin Li       "maddu    %[temp0],    %[temp2]                   \n\t"
265*b2055c35SXin Li       "mfhi     %[temp5]                                \n\t"
266*b2055c35SXin Li       "sw       $zero,       -4(%[irow])                \n\t"
267*b2055c35SXin Li       "sb       %[temp5],    -1(%[dst])                 \n\t"
268*b2055c35SXin Li       "bne      %[irow],     %[loop_end], 1b            \n\t"
269*b2055c35SXin Li       : [temp0]"=&r"(temp0), [temp1]"=&r"(temp1), [temp3]"=&r"(temp3),
270*b2055c35SXin Li         [temp4]"=&r"(temp4), [temp5]"=&r"(temp5), [irow]"+r"(irow),
271*b2055c35SXin Li         [dst]"+r"(dst), [loop_end]"=&r"(loop_end)
272*b2055c35SXin Li       : [temp2]"r"(temp2), [temp6]"r"(temp6)
273*b2055c35SXin Li       : "memory", "hi", "lo"
274*b2055c35SXin Li     );
275*b2055c35SXin Li   }
276*b2055c35SXin Li }
277*b2055c35SXin Li #endif  // 0
278*b2055c35SXin Li 
279*b2055c35SXin Li //------------------------------------------------------------------------------
280*b2055c35SXin Li // Entry point
281*b2055c35SXin Li 
282*b2055c35SXin Li extern void WebPRescalerDspInitMIPS32(void);
283*b2055c35SXin Li 
WebPRescalerDspInitMIPS32(void)284*b2055c35SXin Li WEBP_TSAN_IGNORE_FUNCTION void WebPRescalerDspInitMIPS32(void) {
285*b2055c35SXin Li   WebPRescalerImportRowExpand = ImportRowExpand_MIPS32;
286*b2055c35SXin Li   WebPRescalerImportRowShrink = ImportRowShrink_MIPS32;
287*b2055c35SXin Li   WebPRescalerExportRowExpand = ExportRowExpand_MIPS32;
288*b2055c35SXin Li //  WebPRescalerExportRowShrink = ExportRowShrink_MIPS32;
289*b2055c35SXin Li }
290*b2055c35SXin Li 
291*b2055c35SXin Li #else  // !WEBP_USE_MIPS32
292*b2055c35SXin Li 
293*b2055c35SXin Li WEBP_DSP_INIT_STUB(WebPRescalerDspInitMIPS32)
294*b2055c35SXin Li 
295*b2055c35SXin Li #endif  // WEBP_USE_MIPS32
296