xref: /aosp_15_r20/external/libpng/mips/mips_init.c (revision a67afe4df73cf47866eedc69947994b8ff839aba)
1*a67afe4dSAndroid Build Coastguard Worker 
2*a67afe4dSAndroid Build Coastguard Worker /* mips_init.c - MSA optimised filter functions
3*a67afe4dSAndroid Build Coastguard Worker  *
4*a67afe4dSAndroid Build Coastguard Worker  * Copyright (c) 2018-2024 Cosmin Truta
5*a67afe4dSAndroid Build Coastguard Worker  * Copyright (c) 2016 Glenn Randers-Pehrson
6*a67afe4dSAndroid Build Coastguard Worker  * Written by Mandar Sahastrabuddhe, 2016
7*a67afe4dSAndroid Build Coastguard Worker  * Updated by guxiwei, 2023
8*a67afe4dSAndroid Build Coastguard Worker  *
9*a67afe4dSAndroid Build Coastguard Worker  * This code is released under the libpng license.
10*a67afe4dSAndroid Build Coastguard Worker  * For conditions of distribution and use, see the disclaimer
11*a67afe4dSAndroid Build Coastguard Worker  * and license in png.h
12*a67afe4dSAndroid Build Coastguard Worker  */
13*a67afe4dSAndroid Build Coastguard Worker 
14*a67afe4dSAndroid Build Coastguard Worker /* Below, after checking __linux__, various non-C90 POSIX 1003.1 functions are
15*a67afe4dSAndroid Build Coastguard Worker  * called.
16*a67afe4dSAndroid Build Coastguard Worker  */
17*a67afe4dSAndroid Build Coastguard Worker #define _POSIX_SOURCE 1
18*a67afe4dSAndroid Build Coastguard Worker 
19*a67afe4dSAndroid Build Coastguard Worker #include <stdio.h>
20*a67afe4dSAndroid Build Coastguard Worker #include "../pngpriv.h"
21*a67afe4dSAndroid Build Coastguard Worker 
22*a67afe4dSAndroid Build Coastguard Worker #ifdef PNG_READ_SUPPORTED
23*a67afe4dSAndroid Build Coastguard Worker 
24*a67afe4dSAndroid Build Coastguard Worker #if PNG_MIPS_MSA_IMPLEMENTATION == 1 || PNG_MIPS_MMI_IMPLEMENTATION > 0
25*a67afe4dSAndroid Build Coastguard Worker 
26*a67afe4dSAndroid Build Coastguard Worker #ifdef PNG_MIPS_MSA_CHECK_SUPPORTED /* Do MIPS MSA run-time checks */
27*a67afe4dSAndroid Build Coastguard Worker /* WARNING: it is strongly recommended that you do not build libpng with
28*a67afe4dSAndroid Build Coastguard Worker  * run-time checks for CPU features if at all possible.  In the case of the MIPS
29*a67afe4dSAndroid Build Coastguard Worker  * MSA instructions there is no processor-specific way of detecting the
30*a67afe4dSAndroid Build Coastguard Worker  * presence of the required support, therefore run-time detection is extremely
31*a67afe4dSAndroid Build Coastguard Worker  * OS specific.
32*a67afe4dSAndroid Build Coastguard Worker  *
33*a67afe4dSAndroid Build Coastguard Worker  * You may set the macro PNG_MIPS_MSA_FILE to the file name of file containing
34*a67afe4dSAndroid Build Coastguard Worker  * a fragment of C source code which defines the png_have_msa function.  There
35*a67afe4dSAndroid Build Coastguard Worker  * are a number of implementations in contrib/mips-msa, but the only one that
36*a67afe4dSAndroid Build Coastguard Worker  * has partial support is contrib/mips-msa/linux.c - a generic Linux
37*a67afe4dSAndroid Build Coastguard Worker  * implementation which reads /proc/cpufino.
38*a67afe4dSAndroid Build Coastguard Worker  */
39*a67afe4dSAndroid Build Coastguard Worker #ifndef PNG_MIPS_MSA_FILE
40*a67afe4dSAndroid Build Coastguard Worker #  ifdef __linux__
41*a67afe4dSAndroid Build Coastguard Worker #     define PNG_MIPS_MSA_FILE "contrib/mips-msa/linux.c"
42*a67afe4dSAndroid Build Coastguard Worker #  endif
43*a67afe4dSAndroid Build Coastguard Worker #endif
44*a67afe4dSAndroid Build Coastguard Worker 
45*a67afe4dSAndroid Build Coastguard Worker #ifdef PNG_MIPS_MSA_FILE
46*a67afe4dSAndroid Build Coastguard Worker 
47*a67afe4dSAndroid Build Coastguard Worker #include <signal.h> /* for sig_atomic_t */
48*a67afe4dSAndroid Build Coastguard Worker static int png_have_msa(png_structp png_ptr);
49*a67afe4dSAndroid Build Coastguard Worker #include PNG_MIPS_MSA_FILE
50*a67afe4dSAndroid Build Coastguard Worker 
51*a67afe4dSAndroid Build Coastguard Worker #else  /* PNG_MIPS_MSA_FILE */
52*a67afe4dSAndroid Build Coastguard Worker #  error "PNG_MIPS_MSA_FILE undefined: no support for run-time MIPS MSA checks"
53*a67afe4dSAndroid Build Coastguard Worker #endif /* PNG_MIPS_MSA_FILE */
54*a67afe4dSAndroid Build Coastguard Worker #endif /* PNG_MIPS_MSA_CHECK_SUPPORTED */
55*a67afe4dSAndroid Build Coastguard Worker 
56*a67afe4dSAndroid Build Coastguard Worker #ifdef PNG_MIPS_MMI_CHECK_SUPPORTED /* Do MIPS MMI run-times checks */
57*a67afe4dSAndroid Build Coastguard Worker #ifndef PNG_MIPS_MMI_FILE
58*a67afe4dSAndroid Build Coastguard Worker #  ifdef __linux__
59*a67afe4dSAndroid Build Coastguard Worker #     define PNG_MIPS_MMI_FILE "contrib/mips-mmi/linux.c"
60*a67afe4dSAndroid Build Coastguard Worker #  endif
61*a67afe4dSAndroid Build Coastguard Worker #endif
62*a67afe4dSAndroid Build Coastguard Worker 
63*a67afe4dSAndroid Build Coastguard Worker #ifdef PNG_MIPS_MMI_FILE
64*a67afe4dSAndroid Build Coastguard Worker 
65*a67afe4dSAndroid Build Coastguard Worker #include <signal.h> /* for sig_atomic_t */
66*a67afe4dSAndroid Build Coastguard Worker static int png_have_mmi();
67*a67afe4dSAndroid Build Coastguard Worker #include PNG_MIPS_MMI_FILE
68*a67afe4dSAndroid Build Coastguard Worker 
69*a67afe4dSAndroid Build Coastguard Worker #else  /* PNG_MIPS_MMI_FILE */
70*a67afe4dSAndroid Build Coastguard Worker #  error "PNG_MIPS_MMI_FILE undefined: no support for run-time MIPS MMI checks"
71*a67afe4dSAndroid Build Coastguard Worker #endif /* PNG_MIPS_MMI_FILE */
72*a67afe4dSAndroid Build Coastguard Worker #endif /* PNG_MIPS_MMI_CHECK_SUPPORTED*/
73*a67afe4dSAndroid Build Coastguard Worker 
74*a67afe4dSAndroid Build Coastguard Worker #ifndef PNG_ALIGNED_MEMORY_SUPPORTED
75*a67afe4dSAndroid Build Coastguard Worker #  error "ALIGNED_MEMORY is required; set: -DPNG_ALIGNED_MEMORY_SUPPORTED"
76*a67afe4dSAndroid Build Coastguard Worker #endif
77*a67afe4dSAndroid Build Coastguard Worker 
78*a67afe4dSAndroid Build Coastguard Worker /* MIPS supports two optimizations: MMI and MSA. The appropriate
79*a67afe4dSAndroid Build Coastguard Worker  * optimization is chosen at runtime
80*a67afe4dSAndroid Build Coastguard Worker  */
81*a67afe4dSAndroid Build Coastguard Worker void
png_init_filter_functions_mips(png_structp pp,unsigned int bpp)82*a67afe4dSAndroid Build Coastguard Worker png_init_filter_functions_mips(png_structp pp, unsigned int bpp)
83*a67afe4dSAndroid Build Coastguard Worker {
84*a67afe4dSAndroid Build Coastguard Worker #if PNG_MIPS_MMI_IMPLEMENTATION  > 0
85*a67afe4dSAndroid Build Coastguard Worker #ifdef PNG_MIPS_MMI_API_SUPPORTED
86*a67afe4dSAndroid Build Coastguard Worker    switch ((pp->options >> PNG_MIPS_MMI) & 3)
87*a67afe4dSAndroid Build Coastguard Worker    {
88*a67afe4dSAndroid Build Coastguard Worker       case PNG_OPTION_UNSET:
89*a67afe4dSAndroid Build Coastguard Worker #endif /* PNG_MIPS_MMI_API_SUPPORTED */
90*a67afe4dSAndroid Build Coastguard Worker #ifdef PNG_MIPS_MMI_CHECK_SUPPORTED
91*a67afe4dSAndroid Build Coastguard Worker          {
92*a67afe4dSAndroid Build Coastguard Worker             static volatile sig_atomic_t no_mmi = -1; /* not checked */
93*a67afe4dSAndroid Build Coastguard Worker 
94*a67afe4dSAndroid Build Coastguard Worker             if (no_mmi < 0)
95*a67afe4dSAndroid Build Coastguard Worker                no_mmi = !png_have_mmi();
96*a67afe4dSAndroid Build Coastguard Worker 
97*a67afe4dSAndroid Build Coastguard Worker             if (no_mmi)
98*a67afe4dSAndroid Build Coastguard Worker               goto MIPS_MSA_INIT;
99*a67afe4dSAndroid Build Coastguard Worker          }
100*a67afe4dSAndroid Build Coastguard Worker #ifdef PNG_MIPS_MMI_API_SUPPORTED
101*a67afe4dSAndroid Build Coastguard Worker          break;
102*a67afe4dSAndroid Build Coastguard Worker #endif
103*a67afe4dSAndroid Build Coastguard Worker #endif /* PNG_MIPS_MMI_CHECK_SUPPORTED */
104*a67afe4dSAndroid Build Coastguard Worker 
105*a67afe4dSAndroid Build Coastguard Worker #ifdef PNG_MIPS_MMI_API_SUPPORTED
106*a67afe4dSAndroid Build Coastguard Worker       default: /* OFF or INVALID */
107*a67afe4dSAndroid Build Coastguard Worker          goto MIPS_MSA_INIT;
108*a67afe4dSAndroid Build Coastguard Worker 
109*a67afe4dSAndroid Build Coastguard Worker       case PNG_OPTION_ON:
110*a67afe4dSAndroid Build Coastguard Worker          /* Option turned on */
111*a67afe4dSAndroid Build Coastguard Worker          break;
112*a67afe4dSAndroid Build Coastguard Worker    }
113*a67afe4dSAndroid Build Coastguard Worker #endif
114*a67afe4dSAndroid Build Coastguard Worker    pp->read_filter[PNG_FILTER_VALUE_UP-1] = png_read_filter_row_up_mmi;
115*a67afe4dSAndroid Build Coastguard Worker    if (bpp == 3)
116*a67afe4dSAndroid Build Coastguard Worker    {
117*a67afe4dSAndroid Build Coastguard Worker       pp->read_filter[PNG_FILTER_VALUE_SUB-1] = png_read_filter_row_sub3_mmi;
118*a67afe4dSAndroid Build Coastguard Worker       pp->read_filter[PNG_FILTER_VALUE_AVG-1] = png_read_filter_row_avg3_mmi;
119*a67afe4dSAndroid Build Coastguard Worker       pp->read_filter[PNG_FILTER_VALUE_PAETH-1] =
120*a67afe4dSAndroid Build Coastguard Worker          png_read_filter_row_paeth3_mmi;
121*a67afe4dSAndroid Build Coastguard Worker    }
122*a67afe4dSAndroid Build Coastguard Worker    else if (bpp == 4)
123*a67afe4dSAndroid Build Coastguard Worker    {
124*a67afe4dSAndroid Build Coastguard Worker       pp->read_filter[PNG_FILTER_VALUE_SUB-1] = png_read_filter_row_sub4_mmi;
125*a67afe4dSAndroid Build Coastguard Worker       pp->read_filter[PNG_FILTER_VALUE_AVG-1] = png_read_filter_row_avg4_mmi;
126*a67afe4dSAndroid Build Coastguard Worker       pp->read_filter[PNG_FILTER_VALUE_PAETH-1] =
127*a67afe4dSAndroid Build Coastguard Worker           png_read_filter_row_paeth4_mmi;
128*a67afe4dSAndroid Build Coastguard Worker    }
129*a67afe4dSAndroid Build Coastguard Worker #endif /* PNG_MIPS_MMI_IMPLEMENTATION > 0 */
130*a67afe4dSAndroid Build Coastguard Worker 
131*a67afe4dSAndroid Build Coastguard Worker MIPS_MSA_INIT:
132*a67afe4dSAndroid Build Coastguard Worker #if PNG_MIPS_MSA_IMPLEMENTATION == 1
133*a67afe4dSAndroid Build Coastguard Worker    /* The switch statement is compiled in for MIPS_MSA_API, the call to
134*a67afe4dSAndroid Build Coastguard Worker     * png_have_msa is compiled in for MIPS_MSA_CHECK. If both are defined
135*a67afe4dSAndroid Build Coastguard Worker     * the check is only performed if the API has not set the MSA option on
136*a67afe4dSAndroid Build Coastguard Worker     * or off explicitly. In this case the check controls what happens.
137*a67afe4dSAndroid Build Coastguard Worker     */
138*a67afe4dSAndroid Build Coastguard Worker 
139*a67afe4dSAndroid Build Coastguard Worker #ifdef PNG_MIPS_MSA_API_SUPPORTED
140*a67afe4dSAndroid Build Coastguard Worker    switch ((pp->options >> PNG_MIPS_MSA) & 3)
141*a67afe4dSAndroid Build Coastguard Worker    {
142*a67afe4dSAndroid Build Coastguard Worker       case PNG_OPTION_UNSET:
143*a67afe4dSAndroid Build Coastguard Worker          /* Allow the run-time check to execute if it has been enabled -
144*a67afe4dSAndroid Build Coastguard Worker           * thus both API and CHECK can be turned on.  If it isn't supported
145*a67afe4dSAndroid Build Coastguard Worker           * this case will fall through to the 'default' below, which just
146*a67afe4dSAndroid Build Coastguard Worker           * returns.
147*a67afe4dSAndroid Build Coastguard Worker           */
148*a67afe4dSAndroid Build Coastguard Worker #endif /* PNG_MIPS_MSA_API_SUPPORTED */
149*a67afe4dSAndroid Build Coastguard Worker #ifdef PNG_MIPS_MSA_CHECK_SUPPORTED
150*a67afe4dSAndroid Build Coastguard Worker          {
151*a67afe4dSAndroid Build Coastguard Worker             static volatile sig_atomic_t no_msa = -1; /* not checked */
152*a67afe4dSAndroid Build Coastguard Worker 
153*a67afe4dSAndroid Build Coastguard Worker             if (no_msa < 0)
154*a67afe4dSAndroid Build Coastguard Worker                no_msa = !png_have_msa(pp);
155*a67afe4dSAndroid Build Coastguard Worker 
156*a67afe4dSAndroid Build Coastguard Worker             if (no_msa)
157*a67afe4dSAndroid Build Coastguard Worker                return;
158*a67afe4dSAndroid Build Coastguard Worker          }
159*a67afe4dSAndroid Build Coastguard Worker #ifdef PNG_MIPS_MSA_API_SUPPORTED
160*a67afe4dSAndroid Build Coastguard Worker          break;
161*a67afe4dSAndroid Build Coastguard Worker #endif
162*a67afe4dSAndroid Build Coastguard Worker #endif /* PNG_MIPS_MSA_CHECK_SUPPORTED */
163*a67afe4dSAndroid Build Coastguard Worker 
164*a67afe4dSAndroid Build Coastguard Worker #ifdef PNG_MIPS_MSA_API_SUPPORTED
165*a67afe4dSAndroid Build Coastguard Worker       default: /* OFF or INVALID */
166*a67afe4dSAndroid Build Coastguard Worker          return;
167*a67afe4dSAndroid Build Coastguard Worker 
168*a67afe4dSAndroid Build Coastguard Worker       case PNG_OPTION_ON:
169*a67afe4dSAndroid Build Coastguard Worker          /* Option turned on */
170*a67afe4dSAndroid Build Coastguard Worker          break;
171*a67afe4dSAndroid Build Coastguard Worker    }
172*a67afe4dSAndroid Build Coastguard Worker #endif
173*a67afe4dSAndroid Build Coastguard Worker 
174*a67afe4dSAndroid Build Coastguard Worker    /* IMPORTANT: any new external functions used here must be declared using
175*a67afe4dSAndroid Build Coastguard Worker     * PNG_INTERNAL_FUNCTION in ../pngpriv.h.  This is required so that the
176*a67afe4dSAndroid Build Coastguard Worker     * 'prefix' option to configure works:
177*a67afe4dSAndroid Build Coastguard Worker     *
178*a67afe4dSAndroid Build Coastguard Worker     *    ./configure --with-libpng-prefix=foobar_
179*a67afe4dSAndroid Build Coastguard Worker     *
180*a67afe4dSAndroid Build Coastguard Worker     * Verify you have got this right by running the above command, doing a build
181*a67afe4dSAndroid Build Coastguard Worker     * and examining pngprefix.h; it must contain a #define for every external
182*a67afe4dSAndroid Build Coastguard Worker     * function you add.  (Notice that this happens automatically for the
183*a67afe4dSAndroid Build Coastguard Worker     * initialization function.)
184*a67afe4dSAndroid Build Coastguard Worker     */
185*a67afe4dSAndroid Build Coastguard Worker    pp->read_filter[PNG_FILTER_VALUE_UP-1] = png_read_filter_row_up_msa;
186*a67afe4dSAndroid Build Coastguard Worker 
187*a67afe4dSAndroid Build Coastguard Worker    if (bpp == 3)
188*a67afe4dSAndroid Build Coastguard Worker    {
189*a67afe4dSAndroid Build Coastguard Worker       pp->read_filter[PNG_FILTER_VALUE_SUB-1] = png_read_filter_row_sub3_msa;
190*a67afe4dSAndroid Build Coastguard Worker       pp->read_filter[PNG_FILTER_VALUE_AVG-1] = png_read_filter_row_avg3_msa;
191*a67afe4dSAndroid Build Coastguard Worker       pp->read_filter[PNG_FILTER_VALUE_PAETH-1] = png_read_filter_row_paeth3_msa;
192*a67afe4dSAndroid Build Coastguard Worker    }
193*a67afe4dSAndroid Build Coastguard Worker 
194*a67afe4dSAndroid Build Coastguard Worker    else if (bpp == 4)
195*a67afe4dSAndroid Build Coastguard Worker    {
196*a67afe4dSAndroid Build Coastguard Worker       pp->read_filter[PNG_FILTER_VALUE_SUB-1] = png_read_filter_row_sub4_msa;
197*a67afe4dSAndroid Build Coastguard Worker       pp->read_filter[PNG_FILTER_VALUE_AVG-1] = png_read_filter_row_avg4_msa;
198*a67afe4dSAndroid Build Coastguard Worker       pp->read_filter[PNG_FILTER_VALUE_PAETH-1] = png_read_filter_row_paeth4_msa;
199*a67afe4dSAndroid Build Coastguard Worker    }
200*a67afe4dSAndroid Build Coastguard Worker #endif /* PNG_MIPS_MSA_IMPLEMENTATION == 1 */
201*a67afe4dSAndroid Build Coastguard Worker    return;
202*a67afe4dSAndroid Build Coastguard Worker }
203*a67afe4dSAndroid Build Coastguard Worker #endif /* PNG_MIPS_MSA_IMPLEMENTATION == 1 || PNG_MIPS_MMI_IMPLEMENTATION > 0 */
204*a67afe4dSAndroid Build Coastguard Worker #endif /* READ */
205