xref: /aosp_15_r20/external/pdfium/third_party/libopenjpeg/sparse_array.c (revision 3ac0a46f773bac49fa9476ec2b1cf3f8da5ec3a4)
1*3ac0a46fSAndroid Build Coastguard Worker /*
2*3ac0a46fSAndroid Build Coastguard Worker  * The copyright in this software is being made available under the 2-clauses
3*3ac0a46fSAndroid Build Coastguard Worker  * BSD License, included below. This software may be subject to other third
4*3ac0a46fSAndroid Build Coastguard Worker  * party and contributor rights, including patent rights, and no such rights
5*3ac0a46fSAndroid Build Coastguard Worker  * are granted under this license.
6*3ac0a46fSAndroid Build Coastguard Worker  *
7*3ac0a46fSAndroid Build Coastguard Worker  * Copyright (c) 2017, IntoPix SA <[email protected]>
8*3ac0a46fSAndroid Build Coastguard Worker  * All rights reserved.
9*3ac0a46fSAndroid Build Coastguard Worker  *
10*3ac0a46fSAndroid Build Coastguard Worker  * Redistribution and use in source and binary forms, with or without
11*3ac0a46fSAndroid Build Coastguard Worker  * modification, are permitted provided that the following conditions
12*3ac0a46fSAndroid Build Coastguard Worker  * are met:
13*3ac0a46fSAndroid Build Coastguard Worker  * 1. Redistributions of source code must retain the above copyright
14*3ac0a46fSAndroid Build Coastguard Worker  *    notice, this list of conditions and the following disclaimer.
15*3ac0a46fSAndroid Build Coastguard Worker  * 2. Redistributions in binary form must reproduce the above copyright
16*3ac0a46fSAndroid Build Coastguard Worker  *    notice, this list of conditions and the following disclaimer in the
17*3ac0a46fSAndroid Build Coastguard Worker  *    documentation and/or other materials provided with the distribution.
18*3ac0a46fSAndroid Build Coastguard Worker  *
19*3ac0a46fSAndroid Build Coastguard Worker  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS `AS IS'
20*3ac0a46fSAndroid Build Coastguard Worker  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21*3ac0a46fSAndroid Build Coastguard Worker  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22*3ac0a46fSAndroid Build Coastguard Worker  * ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
23*3ac0a46fSAndroid Build Coastguard Worker  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24*3ac0a46fSAndroid Build Coastguard Worker  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25*3ac0a46fSAndroid Build Coastguard Worker  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26*3ac0a46fSAndroid Build Coastguard Worker  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27*3ac0a46fSAndroid Build Coastguard Worker  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28*3ac0a46fSAndroid Build Coastguard Worker  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29*3ac0a46fSAndroid Build Coastguard Worker  * POSSIBILITY OF SUCH DAMAGE.
30*3ac0a46fSAndroid Build Coastguard Worker  */
31*3ac0a46fSAndroid Build Coastguard Worker 
32*3ac0a46fSAndroid Build Coastguard Worker #include "opj_includes.h"
33*3ac0a46fSAndroid Build Coastguard Worker 
34*3ac0a46fSAndroid Build Coastguard Worker 
35*3ac0a46fSAndroid Build Coastguard Worker struct opj_sparse_array_int32 {
36*3ac0a46fSAndroid Build Coastguard Worker     OPJ_UINT32 width;
37*3ac0a46fSAndroid Build Coastguard Worker     OPJ_UINT32 height;
38*3ac0a46fSAndroid Build Coastguard Worker     OPJ_UINT32 block_width;
39*3ac0a46fSAndroid Build Coastguard Worker     OPJ_UINT32 block_height;
40*3ac0a46fSAndroid Build Coastguard Worker     OPJ_UINT32 block_count_hor;
41*3ac0a46fSAndroid Build Coastguard Worker     OPJ_UINT32 block_count_ver;
42*3ac0a46fSAndroid Build Coastguard Worker     OPJ_INT32** data_blocks;
43*3ac0a46fSAndroid Build Coastguard Worker };
44*3ac0a46fSAndroid Build Coastguard Worker 
opj_sparse_array_int32_create(OPJ_UINT32 width,OPJ_UINT32 height,OPJ_UINT32 block_width,OPJ_UINT32 block_height)45*3ac0a46fSAndroid Build Coastguard Worker opj_sparse_array_int32_t* opj_sparse_array_int32_create(OPJ_UINT32 width,
46*3ac0a46fSAndroid Build Coastguard Worker         OPJ_UINT32 height,
47*3ac0a46fSAndroid Build Coastguard Worker         OPJ_UINT32 block_width,
48*3ac0a46fSAndroid Build Coastguard Worker         OPJ_UINT32 block_height)
49*3ac0a46fSAndroid Build Coastguard Worker {
50*3ac0a46fSAndroid Build Coastguard Worker     opj_sparse_array_int32_t* sa;
51*3ac0a46fSAndroid Build Coastguard Worker 
52*3ac0a46fSAndroid Build Coastguard Worker     if (width == 0 || height == 0 || block_width == 0 || block_height == 0) {
53*3ac0a46fSAndroid Build Coastguard Worker         return NULL;
54*3ac0a46fSAndroid Build Coastguard Worker     }
55*3ac0a46fSAndroid Build Coastguard Worker     if (block_width > ((OPJ_UINT32)~0U) / block_height / sizeof(OPJ_INT32)) {
56*3ac0a46fSAndroid Build Coastguard Worker         return NULL;
57*3ac0a46fSAndroid Build Coastguard Worker     }
58*3ac0a46fSAndroid Build Coastguard Worker 
59*3ac0a46fSAndroid Build Coastguard Worker     sa = (opj_sparse_array_int32_t*) opj_calloc(1,
60*3ac0a46fSAndroid Build Coastguard Worker             sizeof(opj_sparse_array_int32_t));
61*3ac0a46fSAndroid Build Coastguard Worker     sa->width = width;
62*3ac0a46fSAndroid Build Coastguard Worker     sa->height = height;
63*3ac0a46fSAndroid Build Coastguard Worker     sa->block_width = block_width;
64*3ac0a46fSAndroid Build Coastguard Worker     sa->block_height = block_height;
65*3ac0a46fSAndroid Build Coastguard Worker     sa->block_count_hor = opj_uint_ceildiv(width, block_width);
66*3ac0a46fSAndroid Build Coastguard Worker     sa->block_count_ver = opj_uint_ceildiv(height, block_height);
67*3ac0a46fSAndroid Build Coastguard Worker     if (sa->block_count_hor > ((OPJ_UINT32)~0U) / sa->block_count_ver) {
68*3ac0a46fSAndroid Build Coastguard Worker         opj_free(sa);
69*3ac0a46fSAndroid Build Coastguard Worker         return NULL;
70*3ac0a46fSAndroid Build Coastguard Worker     }
71*3ac0a46fSAndroid Build Coastguard Worker     sa->data_blocks = (OPJ_INT32**) opj_calloc(sizeof(OPJ_INT32*),
72*3ac0a46fSAndroid Build Coastguard Worker                       (size_t) sa->block_count_hor * sa->block_count_ver);
73*3ac0a46fSAndroid Build Coastguard Worker     if (sa->data_blocks == NULL) {
74*3ac0a46fSAndroid Build Coastguard Worker         opj_free(sa);
75*3ac0a46fSAndroid Build Coastguard Worker         return NULL;
76*3ac0a46fSAndroid Build Coastguard Worker     }
77*3ac0a46fSAndroid Build Coastguard Worker 
78*3ac0a46fSAndroid Build Coastguard Worker     return sa;
79*3ac0a46fSAndroid Build Coastguard Worker }
80*3ac0a46fSAndroid Build Coastguard Worker 
opj_sparse_array_int32_free(opj_sparse_array_int32_t * sa)81*3ac0a46fSAndroid Build Coastguard Worker void opj_sparse_array_int32_free(opj_sparse_array_int32_t* sa)
82*3ac0a46fSAndroid Build Coastguard Worker {
83*3ac0a46fSAndroid Build Coastguard Worker     if (sa) {
84*3ac0a46fSAndroid Build Coastguard Worker         OPJ_UINT32 i;
85*3ac0a46fSAndroid Build Coastguard Worker         for (i = 0; i < sa->block_count_hor * sa->block_count_ver; i++) {
86*3ac0a46fSAndroid Build Coastguard Worker             if (sa->data_blocks[i]) {
87*3ac0a46fSAndroid Build Coastguard Worker                 opj_free(sa->data_blocks[i]);
88*3ac0a46fSAndroid Build Coastguard Worker             }
89*3ac0a46fSAndroid Build Coastguard Worker         }
90*3ac0a46fSAndroid Build Coastguard Worker         opj_free(sa->data_blocks);
91*3ac0a46fSAndroid Build Coastguard Worker         opj_free(sa);
92*3ac0a46fSAndroid Build Coastguard Worker     }
93*3ac0a46fSAndroid Build Coastguard Worker }
94*3ac0a46fSAndroid Build Coastguard Worker 
opj_sparse_array_is_region_valid(const opj_sparse_array_int32_t * sa,OPJ_UINT32 x0,OPJ_UINT32 y0,OPJ_UINT32 x1,OPJ_UINT32 y1)95*3ac0a46fSAndroid Build Coastguard Worker OPJ_BOOL opj_sparse_array_is_region_valid(const opj_sparse_array_int32_t* sa,
96*3ac0a46fSAndroid Build Coastguard Worker         OPJ_UINT32 x0,
97*3ac0a46fSAndroid Build Coastguard Worker         OPJ_UINT32 y0,
98*3ac0a46fSAndroid Build Coastguard Worker         OPJ_UINT32 x1,
99*3ac0a46fSAndroid Build Coastguard Worker         OPJ_UINT32 y1)
100*3ac0a46fSAndroid Build Coastguard Worker {
101*3ac0a46fSAndroid Build Coastguard Worker     return !(x0 >= sa->width || x1 <= x0 || x1 > sa->width ||
102*3ac0a46fSAndroid Build Coastguard Worker              y0 >= sa->height || y1 <= y0 || y1 > sa->height);
103*3ac0a46fSAndroid Build Coastguard Worker }
104*3ac0a46fSAndroid Build Coastguard Worker 
opj_sparse_array_int32_read_or_write(const opj_sparse_array_int32_t * sa,OPJ_UINT32 x0,OPJ_UINT32 y0,OPJ_UINT32 x1,OPJ_UINT32 y1,OPJ_INT32 * buf,OPJ_UINT32 buf_col_stride,OPJ_UINT32 buf_line_stride,OPJ_BOOL forgiving,OPJ_BOOL is_read_op)105*3ac0a46fSAndroid Build Coastguard Worker static OPJ_BOOL opj_sparse_array_int32_read_or_write(
106*3ac0a46fSAndroid Build Coastguard Worker     const opj_sparse_array_int32_t* sa,
107*3ac0a46fSAndroid Build Coastguard Worker     OPJ_UINT32 x0,
108*3ac0a46fSAndroid Build Coastguard Worker     OPJ_UINT32 y0,
109*3ac0a46fSAndroid Build Coastguard Worker     OPJ_UINT32 x1,
110*3ac0a46fSAndroid Build Coastguard Worker     OPJ_UINT32 y1,
111*3ac0a46fSAndroid Build Coastguard Worker     OPJ_INT32* buf,
112*3ac0a46fSAndroid Build Coastguard Worker     OPJ_UINT32 buf_col_stride,
113*3ac0a46fSAndroid Build Coastguard Worker     OPJ_UINT32 buf_line_stride,
114*3ac0a46fSAndroid Build Coastguard Worker     OPJ_BOOL forgiving,
115*3ac0a46fSAndroid Build Coastguard Worker     OPJ_BOOL is_read_op)
116*3ac0a46fSAndroid Build Coastguard Worker {
117*3ac0a46fSAndroid Build Coastguard Worker     OPJ_UINT32 y, block_y;
118*3ac0a46fSAndroid Build Coastguard Worker     OPJ_UINT32 y_incr = 0;
119*3ac0a46fSAndroid Build Coastguard Worker     const OPJ_UINT32 block_width = sa->block_width;
120*3ac0a46fSAndroid Build Coastguard Worker 
121*3ac0a46fSAndroid Build Coastguard Worker     if (!opj_sparse_array_is_region_valid(sa, x0, y0, x1, y1)) {
122*3ac0a46fSAndroid Build Coastguard Worker         return forgiving;
123*3ac0a46fSAndroid Build Coastguard Worker     }
124*3ac0a46fSAndroid Build Coastguard Worker 
125*3ac0a46fSAndroid Build Coastguard Worker     block_y = y0 / sa->block_height;
126*3ac0a46fSAndroid Build Coastguard Worker     for (y = y0; y < y1; block_y ++, y += y_incr) {
127*3ac0a46fSAndroid Build Coastguard Worker         OPJ_UINT32 x, block_x;
128*3ac0a46fSAndroid Build Coastguard Worker         OPJ_UINT32 x_incr = 0;
129*3ac0a46fSAndroid Build Coastguard Worker         OPJ_UINT32 block_y_offset;
130*3ac0a46fSAndroid Build Coastguard Worker         y_incr = (y == y0) ? sa->block_height - (y0 % sa->block_height) :
131*3ac0a46fSAndroid Build Coastguard Worker                  sa->block_height;
132*3ac0a46fSAndroid Build Coastguard Worker         block_y_offset = sa->block_height - y_incr;
133*3ac0a46fSAndroid Build Coastguard Worker         y_incr = opj_uint_min(y_incr, y1 - y);
134*3ac0a46fSAndroid Build Coastguard Worker         block_x = x0 / block_width;
135*3ac0a46fSAndroid Build Coastguard Worker         for (x = x0; x < x1; block_x ++, x += x_incr) {
136*3ac0a46fSAndroid Build Coastguard Worker             OPJ_UINT32 j;
137*3ac0a46fSAndroid Build Coastguard Worker             OPJ_UINT32 block_x_offset;
138*3ac0a46fSAndroid Build Coastguard Worker             OPJ_INT32* src_block;
139*3ac0a46fSAndroid Build Coastguard Worker             x_incr = (x == x0) ? block_width - (x0 % block_width) : block_width;
140*3ac0a46fSAndroid Build Coastguard Worker             block_x_offset = block_width - x_incr;
141*3ac0a46fSAndroid Build Coastguard Worker             x_incr = opj_uint_min(x_incr, x1 - x);
142*3ac0a46fSAndroid Build Coastguard Worker             src_block = sa->data_blocks[block_y * sa->block_count_hor + block_x];
143*3ac0a46fSAndroid Build Coastguard Worker             if (is_read_op) {
144*3ac0a46fSAndroid Build Coastguard Worker                 if (src_block == NULL) {
145*3ac0a46fSAndroid Build Coastguard Worker                     if (buf_col_stride == 1) {
146*3ac0a46fSAndroid Build Coastguard Worker                         OPJ_INT32* dest_ptr = buf + (y - y0) * (OPJ_SIZE_T)buf_line_stride +
147*3ac0a46fSAndroid Build Coastguard Worker                                               (x - x0) * buf_col_stride;
148*3ac0a46fSAndroid Build Coastguard Worker                         for (j = 0; j < y_incr; j++) {
149*3ac0a46fSAndroid Build Coastguard Worker                             memset(dest_ptr, 0, sizeof(OPJ_INT32) * x_incr);
150*3ac0a46fSAndroid Build Coastguard Worker                             dest_ptr += buf_line_stride;
151*3ac0a46fSAndroid Build Coastguard Worker                         }
152*3ac0a46fSAndroid Build Coastguard Worker                     } else {
153*3ac0a46fSAndroid Build Coastguard Worker                         OPJ_INT32* dest_ptr = buf + (y - y0) * (OPJ_SIZE_T)buf_line_stride +
154*3ac0a46fSAndroid Build Coastguard Worker                                               (x - x0) * buf_col_stride;
155*3ac0a46fSAndroid Build Coastguard Worker                         for (j = 0; j < y_incr; j++) {
156*3ac0a46fSAndroid Build Coastguard Worker                             OPJ_UINT32 k;
157*3ac0a46fSAndroid Build Coastguard Worker                             for (k = 0; k < x_incr; k++) {
158*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr[k * buf_col_stride] = 0;
159*3ac0a46fSAndroid Build Coastguard Worker                             }
160*3ac0a46fSAndroid Build Coastguard Worker                             dest_ptr += buf_line_stride;
161*3ac0a46fSAndroid Build Coastguard Worker                         }
162*3ac0a46fSAndroid Build Coastguard Worker                     }
163*3ac0a46fSAndroid Build Coastguard Worker                 } else {
164*3ac0a46fSAndroid Build Coastguard Worker                     const OPJ_INT32* OPJ_RESTRICT src_ptr = src_block + block_y_offset *
165*3ac0a46fSAndroid Build Coastguard Worker                                                             (OPJ_SIZE_T)block_width + block_x_offset;
166*3ac0a46fSAndroid Build Coastguard Worker                     if (buf_col_stride == 1) {
167*3ac0a46fSAndroid Build Coastguard Worker                         OPJ_INT32* OPJ_RESTRICT dest_ptr = buf + (y - y0) * (OPJ_SIZE_T)buf_line_stride
168*3ac0a46fSAndroid Build Coastguard Worker                                                            +
169*3ac0a46fSAndroid Build Coastguard Worker                                                            (x - x0) * buf_col_stride;
170*3ac0a46fSAndroid Build Coastguard Worker                         if (x_incr == 4) {
171*3ac0a46fSAndroid Build Coastguard Worker                             /* Same code as general branch, but the compiler */
172*3ac0a46fSAndroid Build Coastguard Worker                             /* can have an efficient memcpy() */
173*3ac0a46fSAndroid Build Coastguard Worker                             (void)(x_incr); /* trick to silent cppcheck duplicateBranch warning */
174*3ac0a46fSAndroid Build Coastguard Worker                             for (j = 0; j < y_incr; j++) {
175*3ac0a46fSAndroid Build Coastguard Worker                                 memcpy(dest_ptr, src_ptr, sizeof(OPJ_INT32) * x_incr);
176*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr += buf_line_stride;
177*3ac0a46fSAndroid Build Coastguard Worker                                 src_ptr += block_width;
178*3ac0a46fSAndroid Build Coastguard Worker                             }
179*3ac0a46fSAndroid Build Coastguard Worker                         } else {
180*3ac0a46fSAndroid Build Coastguard Worker                             for (j = 0; j < y_incr; j++) {
181*3ac0a46fSAndroid Build Coastguard Worker                                 memcpy(dest_ptr, src_ptr, sizeof(OPJ_INT32) * x_incr);
182*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr += buf_line_stride;
183*3ac0a46fSAndroid Build Coastguard Worker                                 src_ptr += block_width;
184*3ac0a46fSAndroid Build Coastguard Worker                             }
185*3ac0a46fSAndroid Build Coastguard Worker                         }
186*3ac0a46fSAndroid Build Coastguard Worker                     } else {
187*3ac0a46fSAndroid Build Coastguard Worker                         OPJ_INT32* OPJ_RESTRICT dest_ptr = buf + (y - y0) * (OPJ_SIZE_T)buf_line_stride
188*3ac0a46fSAndroid Build Coastguard Worker                                                            +
189*3ac0a46fSAndroid Build Coastguard Worker                                                            (x - x0) * buf_col_stride;
190*3ac0a46fSAndroid Build Coastguard Worker                         if (x_incr == 1) {
191*3ac0a46fSAndroid Build Coastguard Worker                             for (j = 0; j < y_incr; j++) {
192*3ac0a46fSAndroid Build Coastguard Worker                                 *dest_ptr = *src_ptr;
193*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr += buf_line_stride;
194*3ac0a46fSAndroid Build Coastguard Worker                                 src_ptr += block_width;
195*3ac0a46fSAndroid Build Coastguard Worker                             }
196*3ac0a46fSAndroid Build Coastguard Worker                         } else if (y_incr == 1 && buf_col_stride == 2) {
197*3ac0a46fSAndroid Build Coastguard Worker                             OPJ_UINT32 k;
198*3ac0a46fSAndroid Build Coastguard Worker                             for (k = 0; k < (x_incr & ~3U); k += 4) {
199*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr[k * buf_col_stride] = src_ptr[k];
200*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr[(k + 1) * buf_col_stride] = src_ptr[k + 1];
201*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr[(k + 2) * buf_col_stride] = src_ptr[k + 2];
202*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr[(k + 3) * buf_col_stride] = src_ptr[k + 3];
203*3ac0a46fSAndroid Build Coastguard Worker                             }
204*3ac0a46fSAndroid Build Coastguard Worker                             for (; k < x_incr; k++) {
205*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr[k * buf_col_stride] = src_ptr[k];
206*3ac0a46fSAndroid Build Coastguard Worker                             }
207*3ac0a46fSAndroid Build Coastguard Worker                         } else if (x_incr >= 8 && buf_col_stride == 8) {
208*3ac0a46fSAndroid Build Coastguard Worker                             for (j = 0; j < y_incr; j++) {
209*3ac0a46fSAndroid Build Coastguard Worker                                 OPJ_UINT32 k;
210*3ac0a46fSAndroid Build Coastguard Worker                                 for (k = 0; k < (x_incr & ~3U); k += 4) {
211*3ac0a46fSAndroid Build Coastguard Worker                                     dest_ptr[k * buf_col_stride] = src_ptr[k];
212*3ac0a46fSAndroid Build Coastguard Worker                                     dest_ptr[(k + 1) * buf_col_stride] = src_ptr[k + 1];
213*3ac0a46fSAndroid Build Coastguard Worker                                     dest_ptr[(k + 2) * buf_col_stride] = src_ptr[k + 2];
214*3ac0a46fSAndroid Build Coastguard Worker                                     dest_ptr[(k + 3) * buf_col_stride] = src_ptr[k + 3];
215*3ac0a46fSAndroid Build Coastguard Worker                                 }
216*3ac0a46fSAndroid Build Coastguard Worker                                 for (; k < x_incr; k++) {
217*3ac0a46fSAndroid Build Coastguard Worker                                     dest_ptr[k * buf_col_stride] = src_ptr[k];
218*3ac0a46fSAndroid Build Coastguard Worker                                 }
219*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr += buf_line_stride;
220*3ac0a46fSAndroid Build Coastguard Worker                                 src_ptr += block_width;
221*3ac0a46fSAndroid Build Coastguard Worker                             }
222*3ac0a46fSAndroid Build Coastguard Worker                         } else {
223*3ac0a46fSAndroid Build Coastguard Worker                             /* General case */
224*3ac0a46fSAndroid Build Coastguard Worker                             for (j = 0; j < y_incr; j++) {
225*3ac0a46fSAndroid Build Coastguard Worker                                 OPJ_UINT32 k;
226*3ac0a46fSAndroid Build Coastguard Worker                                 for (k = 0; k < x_incr; k++) {
227*3ac0a46fSAndroid Build Coastguard Worker                                     dest_ptr[k * buf_col_stride] = src_ptr[k];
228*3ac0a46fSAndroid Build Coastguard Worker                                 }
229*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr += buf_line_stride;
230*3ac0a46fSAndroid Build Coastguard Worker                                 src_ptr += block_width;
231*3ac0a46fSAndroid Build Coastguard Worker                             }
232*3ac0a46fSAndroid Build Coastguard Worker                         }
233*3ac0a46fSAndroid Build Coastguard Worker                     }
234*3ac0a46fSAndroid Build Coastguard Worker                 }
235*3ac0a46fSAndroid Build Coastguard Worker             } else {
236*3ac0a46fSAndroid Build Coastguard Worker                 if (src_block == NULL) {
237*3ac0a46fSAndroid Build Coastguard Worker                     src_block = (OPJ_INT32*) opj_calloc(1,
238*3ac0a46fSAndroid Build Coastguard Worker                                                         (size_t) sa->block_width * sa->block_height * sizeof(OPJ_INT32));
239*3ac0a46fSAndroid Build Coastguard Worker                     if (src_block == NULL) {
240*3ac0a46fSAndroid Build Coastguard Worker                         return OPJ_FALSE;
241*3ac0a46fSAndroid Build Coastguard Worker                     }
242*3ac0a46fSAndroid Build Coastguard Worker                     sa->data_blocks[block_y * sa->block_count_hor + block_x] = src_block;
243*3ac0a46fSAndroid Build Coastguard Worker                 }
244*3ac0a46fSAndroid Build Coastguard Worker 
245*3ac0a46fSAndroid Build Coastguard Worker                 if (buf_col_stride == 1) {
246*3ac0a46fSAndroid Build Coastguard Worker                     OPJ_INT32* OPJ_RESTRICT dest_ptr = src_block + block_y_offset *
247*3ac0a46fSAndroid Build Coastguard Worker                                                        (OPJ_SIZE_T)block_width + block_x_offset;
248*3ac0a46fSAndroid Build Coastguard Worker                     const OPJ_INT32* OPJ_RESTRICT src_ptr = buf + (y - y0) *
249*3ac0a46fSAndroid Build Coastguard Worker                                                             (OPJ_SIZE_T)buf_line_stride + (x - x0) * buf_col_stride;
250*3ac0a46fSAndroid Build Coastguard Worker                     if (x_incr == 4) {
251*3ac0a46fSAndroid Build Coastguard Worker                         /* Same code as general branch, but the compiler */
252*3ac0a46fSAndroid Build Coastguard Worker                         /* can have an efficient memcpy() */
253*3ac0a46fSAndroid Build Coastguard Worker                         (void)(x_incr); /* trick to silent cppcheck duplicateBranch warning */
254*3ac0a46fSAndroid Build Coastguard Worker                         for (j = 0; j < y_incr; j++) {
255*3ac0a46fSAndroid Build Coastguard Worker                             memcpy(dest_ptr, src_ptr, sizeof(OPJ_INT32) * x_incr);
256*3ac0a46fSAndroid Build Coastguard Worker                             dest_ptr += block_width;
257*3ac0a46fSAndroid Build Coastguard Worker                             src_ptr += buf_line_stride;
258*3ac0a46fSAndroid Build Coastguard Worker                         }
259*3ac0a46fSAndroid Build Coastguard Worker                     } else {
260*3ac0a46fSAndroid Build Coastguard Worker                         for (j = 0; j < y_incr; j++) {
261*3ac0a46fSAndroid Build Coastguard Worker                             memcpy(dest_ptr, src_ptr, sizeof(OPJ_INT32) * x_incr);
262*3ac0a46fSAndroid Build Coastguard Worker                             dest_ptr += block_width;
263*3ac0a46fSAndroid Build Coastguard Worker                             src_ptr += buf_line_stride;
264*3ac0a46fSAndroid Build Coastguard Worker                         }
265*3ac0a46fSAndroid Build Coastguard Worker                     }
266*3ac0a46fSAndroid Build Coastguard Worker                 } else {
267*3ac0a46fSAndroid Build Coastguard Worker                     OPJ_INT32* OPJ_RESTRICT dest_ptr = src_block + block_y_offset *
268*3ac0a46fSAndroid Build Coastguard Worker                                                        (OPJ_SIZE_T)block_width + block_x_offset;
269*3ac0a46fSAndroid Build Coastguard Worker                     const OPJ_INT32* OPJ_RESTRICT src_ptr = buf + (y - y0) *
270*3ac0a46fSAndroid Build Coastguard Worker                                                             (OPJ_SIZE_T)buf_line_stride + (x - x0) * buf_col_stride;
271*3ac0a46fSAndroid Build Coastguard Worker                     if (x_incr == 1) {
272*3ac0a46fSAndroid Build Coastguard Worker                         for (j = 0; j < y_incr; j++) {
273*3ac0a46fSAndroid Build Coastguard Worker                             *dest_ptr = *src_ptr;
274*3ac0a46fSAndroid Build Coastguard Worker                             src_ptr += buf_line_stride;
275*3ac0a46fSAndroid Build Coastguard Worker                             dest_ptr += block_width;
276*3ac0a46fSAndroid Build Coastguard Worker                         }
277*3ac0a46fSAndroid Build Coastguard Worker                     } else if (x_incr >= 8 && buf_col_stride == 8) {
278*3ac0a46fSAndroid Build Coastguard Worker                         for (j = 0; j < y_incr; j++) {
279*3ac0a46fSAndroid Build Coastguard Worker                             OPJ_UINT32 k;
280*3ac0a46fSAndroid Build Coastguard Worker                             for (k = 0; k < (x_incr & ~3U); k += 4) {
281*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr[k] = src_ptr[k * buf_col_stride];
282*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr[k + 1] = src_ptr[(k + 1) * buf_col_stride];
283*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr[k + 2] = src_ptr[(k + 2) * buf_col_stride];
284*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr[k + 3] = src_ptr[(k + 3) * buf_col_stride];
285*3ac0a46fSAndroid Build Coastguard Worker                             }
286*3ac0a46fSAndroid Build Coastguard Worker                             for (; k < x_incr; k++) {
287*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr[k] = src_ptr[k * buf_col_stride];
288*3ac0a46fSAndroid Build Coastguard Worker                             }
289*3ac0a46fSAndroid Build Coastguard Worker                             src_ptr += buf_line_stride;
290*3ac0a46fSAndroid Build Coastguard Worker                             dest_ptr += block_width;
291*3ac0a46fSAndroid Build Coastguard Worker                         }
292*3ac0a46fSAndroid Build Coastguard Worker                     } else {
293*3ac0a46fSAndroid Build Coastguard Worker                         /* General case */
294*3ac0a46fSAndroid Build Coastguard Worker                         for (j = 0; j < y_incr; j++) {
295*3ac0a46fSAndroid Build Coastguard Worker                             OPJ_UINT32 k;
296*3ac0a46fSAndroid Build Coastguard Worker                             for (k = 0; k < x_incr; k++) {
297*3ac0a46fSAndroid Build Coastguard Worker                                 dest_ptr[k] = src_ptr[k * buf_col_stride];
298*3ac0a46fSAndroid Build Coastguard Worker                             }
299*3ac0a46fSAndroid Build Coastguard Worker                             src_ptr += buf_line_stride;
300*3ac0a46fSAndroid Build Coastguard Worker                             dest_ptr += block_width;
301*3ac0a46fSAndroid Build Coastguard Worker                         }
302*3ac0a46fSAndroid Build Coastguard Worker                     }
303*3ac0a46fSAndroid Build Coastguard Worker                 }
304*3ac0a46fSAndroid Build Coastguard Worker             }
305*3ac0a46fSAndroid Build Coastguard Worker         }
306*3ac0a46fSAndroid Build Coastguard Worker     }
307*3ac0a46fSAndroid Build Coastguard Worker 
308*3ac0a46fSAndroid Build Coastguard Worker     return OPJ_TRUE;
309*3ac0a46fSAndroid Build Coastguard Worker }
310*3ac0a46fSAndroid Build Coastguard Worker 
opj_sparse_array_int32_read(const opj_sparse_array_int32_t * sa,OPJ_UINT32 x0,OPJ_UINT32 y0,OPJ_UINT32 x1,OPJ_UINT32 y1,OPJ_INT32 * dest,OPJ_UINT32 dest_col_stride,OPJ_UINT32 dest_line_stride,OPJ_BOOL forgiving)311*3ac0a46fSAndroid Build Coastguard Worker OPJ_BOOL opj_sparse_array_int32_read(const opj_sparse_array_int32_t* sa,
312*3ac0a46fSAndroid Build Coastguard Worker                                      OPJ_UINT32 x0,
313*3ac0a46fSAndroid Build Coastguard Worker                                      OPJ_UINT32 y0,
314*3ac0a46fSAndroid Build Coastguard Worker                                      OPJ_UINT32 x1,
315*3ac0a46fSAndroid Build Coastguard Worker                                      OPJ_UINT32 y1,
316*3ac0a46fSAndroid Build Coastguard Worker                                      OPJ_INT32* dest,
317*3ac0a46fSAndroid Build Coastguard Worker                                      OPJ_UINT32 dest_col_stride,
318*3ac0a46fSAndroid Build Coastguard Worker                                      OPJ_UINT32 dest_line_stride,
319*3ac0a46fSAndroid Build Coastguard Worker                                      OPJ_BOOL forgiving)
320*3ac0a46fSAndroid Build Coastguard Worker {
321*3ac0a46fSAndroid Build Coastguard Worker     return opj_sparse_array_int32_read_or_write(
322*3ac0a46fSAndroid Build Coastguard Worker                (opj_sparse_array_int32_t*)sa, x0, y0, x1, y1,
323*3ac0a46fSAndroid Build Coastguard Worker                dest,
324*3ac0a46fSAndroid Build Coastguard Worker                dest_col_stride,
325*3ac0a46fSAndroid Build Coastguard Worker                dest_line_stride,
326*3ac0a46fSAndroid Build Coastguard Worker                forgiving,
327*3ac0a46fSAndroid Build Coastguard Worker                OPJ_TRUE);
328*3ac0a46fSAndroid Build Coastguard Worker }
329*3ac0a46fSAndroid Build Coastguard Worker 
opj_sparse_array_int32_write(opj_sparse_array_int32_t * sa,OPJ_UINT32 x0,OPJ_UINT32 y0,OPJ_UINT32 x1,OPJ_UINT32 y1,const OPJ_INT32 * src,OPJ_UINT32 src_col_stride,OPJ_UINT32 src_line_stride,OPJ_BOOL forgiving)330*3ac0a46fSAndroid Build Coastguard Worker OPJ_BOOL opj_sparse_array_int32_write(opj_sparse_array_int32_t* sa,
331*3ac0a46fSAndroid Build Coastguard Worker                                       OPJ_UINT32 x0,
332*3ac0a46fSAndroid Build Coastguard Worker                                       OPJ_UINT32 y0,
333*3ac0a46fSAndroid Build Coastguard Worker                                       OPJ_UINT32 x1,
334*3ac0a46fSAndroid Build Coastguard Worker                                       OPJ_UINT32 y1,
335*3ac0a46fSAndroid Build Coastguard Worker                                       const OPJ_INT32* src,
336*3ac0a46fSAndroid Build Coastguard Worker                                       OPJ_UINT32 src_col_stride,
337*3ac0a46fSAndroid Build Coastguard Worker                                       OPJ_UINT32 src_line_stride,
338*3ac0a46fSAndroid Build Coastguard Worker                                       OPJ_BOOL forgiving)
339*3ac0a46fSAndroid Build Coastguard Worker {
340*3ac0a46fSAndroid Build Coastguard Worker     return opj_sparse_array_int32_read_or_write(sa, x0, y0, x1, y1,
341*3ac0a46fSAndroid Build Coastguard Worker             (OPJ_INT32*)src,
342*3ac0a46fSAndroid Build Coastguard Worker             src_col_stride,
343*3ac0a46fSAndroid Build Coastguard Worker             src_line_stride,
344*3ac0a46fSAndroid Build Coastguard Worker             forgiving,
345*3ac0a46fSAndroid Build Coastguard Worker             OPJ_FALSE);
346*3ac0a46fSAndroid Build Coastguard Worker }
347