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