Imported Upstream version 3.0.1
[platform/upstream/libjpeg-turbo.git] / jddiffct.c
1 /*
2  * jddiffct.c
3  *
4  * This file was part of the Independent JPEG Group's software:
5  * Copyright (C) 1994-1997, Thomas G. Lane.
6  * Lossless JPEG Modifications:
7  * Copyright (C) 1999, Ken Murchison.
8  * libjpeg-turbo Modifications:
9  * Copyright (C) 2022, D. R. Commander.
10  * For conditions of distribution and use, see the accompanying README.ijg
11  * file.
12  *
13  * This file contains the [un]difference buffer controller for decompression.
14  * This controller is the top level of the lossless JPEG decompressor proper.
15  * The difference buffer lies between the entropy decoding and
16  * prediction/undifferencing steps.  The undifference buffer lies between the
17  * prediction/undifferencing and scaling steps.
18  *
19  * In buffered-image mode, this controller is the interface between
20  * input-oriented processing and output-oriented processing.
21  */
22
23 #define JPEG_INTERNALS
24 #include "jinclude.h"
25 #include "jpeglib.h"
26 #include "jlossls.h"            /* Private declarations for lossless codec */
27
28
29 #ifdef D_LOSSLESS_SUPPORTED
30
31 /* Private buffer controller object */
32
33 typedef struct {
34   struct jpeg_d_coef_controller pub; /* public fields */
35
36   /* These variables keep track of the current location of the input side. */
37   /* cinfo->input_iMCU_row is also used for this. */
38   JDIMENSION MCU_ctr;           /* counts MCUs processed in current row */
39   unsigned int restart_rows_to_go;      /* MCU rows left in this restart
40                                            interval */
41   unsigned int MCU_vert_offset;         /* counts MCU rows within iMCU row */
42   unsigned int MCU_rows_per_iMCU_row;   /* number of such rows needed */
43
44   /* The output side's location is represented by cinfo->output_iMCU_row. */
45
46   JDIFFARRAY diff_buf[MAX_COMPONENTS];  /* iMCU row of differences */
47   JDIFFARRAY undiff_buf[MAX_COMPONENTS]; /* iMCU row of undiff'd samples */
48
49 #ifdef D_MULTISCAN_FILES_SUPPORTED
50   /* In multi-pass modes, we need a virtual sample array for each component. */
51   jvirt_sarray_ptr whole_image[MAX_COMPONENTS];
52 #endif
53 } my_diff_controller;
54
55 typedef my_diff_controller *my_diff_ptr;
56
57 /* Forward declarations */
58 METHODDEF(int) decompress_data(j_decompress_ptr cinfo, _JSAMPIMAGE output_buf);
59 #ifdef D_MULTISCAN_FILES_SUPPORTED
60 METHODDEF(int) output_data(j_decompress_ptr cinfo, _JSAMPIMAGE output_buf);
61 #endif
62
63
64 LOCAL(void)
65 start_iMCU_row(j_decompress_ptr cinfo)
66 /* Reset within-iMCU-row counters for a new row (input side) */
67 {
68   my_diff_ptr diff = (my_diff_ptr)cinfo->coef;
69
70   /* In an interleaved scan, an MCU row is the same as an iMCU row.
71    * In a noninterleaved scan, an iMCU row has v_samp_factor MCU rows.
72    * But at the bottom of the image, process only what's left.
73    */
74   if (cinfo->comps_in_scan > 1) {
75     diff->MCU_rows_per_iMCU_row = 1;
76   } else {
77     if (cinfo->input_iMCU_row < (cinfo->total_iMCU_rows-1))
78       diff->MCU_rows_per_iMCU_row = cinfo->cur_comp_info[0]->v_samp_factor;
79     else
80       diff->MCU_rows_per_iMCU_row = cinfo->cur_comp_info[0]->last_row_height;
81   }
82
83   diff->MCU_ctr = 0;
84   diff->MCU_vert_offset = 0;
85 }
86
87
88 /*
89  * Initialize for an input processing pass.
90  */
91
92 METHODDEF(void)
93 start_input_pass(j_decompress_ptr cinfo)
94 {
95   my_diff_ptr diff = (my_diff_ptr)cinfo->coef;
96
97   /* Because it is hitching a ride on the jpeg_inverse_dct struct,
98    * start_pass_lossless() will be called at the start of the output pass.
99    * This ensures that it will be called at the start of the input pass as
100    * well.
101    */
102   (*cinfo->idct->start_pass) (cinfo);
103
104   /* Check that the restart interval is an integer multiple of the number
105    * of MCUs in an MCU row.
106    */
107   if (cinfo->restart_interval % cinfo->MCUs_per_row != 0)
108     ERREXIT2(cinfo, JERR_BAD_RESTART,
109              cinfo->restart_interval, cinfo->MCUs_per_row);
110
111   /* Initialize restart counter */
112   diff->restart_rows_to_go = cinfo->restart_interval / cinfo->MCUs_per_row;
113
114   cinfo->input_iMCU_row = 0;
115   start_iMCU_row(cinfo);
116 }
117
118
119 /*
120  * Check for a restart marker & resynchronize decoder, undifferencer.
121  * Returns FALSE if must suspend.
122  */
123
124 METHODDEF(boolean)
125 process_restart(j_decompress_ptr cinfo)
126 {
127   my_diff_ptr diff = (my_diff_ptr)cinfo->coef;
128
129   if (!(*cinfo->entropy->process_restart) (cinfo))
130     return FALSE;
131
132   (*cinfo->idct->start_pass) (cinfo);
133
134   /* Reset restart counter */
135   diff->restart_rows_to_go = cinfo->restart_interval / cinfo->MCUs_per_row;
136
137   return TRUE;
138 }
139
140
141 /*
142  * Initialize for an output processing pass.
143  */
144
145 METHODDEF(void)
146 start_output_pass(j_decompress_ptr cinfo)
147 {
148   cinfo->output_iMCU_row = 0;
149 }
150
151
152 /*
153  * Decompress and return some data in the supplied buffer.
154  * Always attempts to emit one fully interleaved MCU row ("iMCU" row).
155  * Input and output must run in lockstep since we have only a one-MCU buffer.
156  * Return value is JPEG_ROW_COMPLETED, JPEG_SCAN_COMPLETED, or JPEG_SUSPENDED.
157  *
158  * NB: output_buf contains a plane for each component in image,
159  * which we index according to the component's SOF position.
160  */
161
162 METHODDEF(int)
163 decompress_data(j_decompress_ptr cinfo, _JSAMPIMAGE output_buf)
164 {
165   my_diff_ptr diff = (my_diff_ptr)cinfo->coef;
166   lossless_decomp_ptr losslessd = (lossless_decomp_ptr)cinfo->idct;
167   JDIMENSION MCU_col_num;       /* index of current MCU within row */
168   JDIMENSION MCU_count;         /* number of MCUs decoded */
169   JDIMENSION last_iMCU_row = cinfo->total_iMCU_rows - 1;
170   int ci, compi, row, prev_row;
171   unsigned int yoffset;
172   jpeg_component_info *compptr;
173
174   /* Loop to process as much as one whole iMCU row */
175   for (yoffset = diff->MCU_vert_offset; yoffset < diff->MCU_rows_per_iMCU_row;
176        yoffset++) {
177
178     /* Process restart marker if needed; may have to suspend */
179     if (cinfo->restart_interval) {
180       if (diff->restart_rows_to_go == 0)
181         if (!process_restart(cinfo))
182           return JPEG_SUSPENDED;
183     }
184
185     MCU_col_num = diff->MCU_ctr;
186     /* Try to fetch an MCU row (or remaining portion of suspended MCU row). */
187     MCU_count =
188       (*cinfo->entropy->decode_mcus) (cinfo,
189                                       diff->diff_buf, yoffset, MCU_col_num,
190                                       cinfo->MCUs_per_row - MCU_col_num);
191     if (MCU_count != cinfo->MCUs_per_row - MCU_col_num) {
192       /* Suspension forced; update state counters and exit */
193       diff->MCU_vert_offset = yoffset;
194       diff->MCU_ctr += MCU_count;
195       return JPEG_SUSPENDED;
196     }
197
198     /* Account for restart interval (no-op if not using restarts) */
199     if (cinfo->restart_interval)
200       diff->restart_rows_to_go--;
201
202     /* Completed an MCU row, but perhaps not an iMCU row */
203     diff->MCU_ctr = 0;
204   }
205
206   /*
207    * Undifference and scale each scanline of the disassembled MCU row
208    * separately.  We do not process dummy samples at the end of a scanline
209    * or dummy rows at the end of the image.
210    */
211   for (ci = 0; ci < cinfo->comps_in_scan; ci++) {
212     compptr = cinfo->cur_comp_info[ci];
213     compi = compptr->component_index;
214     for (row = 0, prev_row = compptr->v_samp_factor - 1;
215          row < (cinfo->input_iMCU_row == last_iMCU_row ?
216                 compptr->last_row_height : compptr->v_samp_factor);
217          prev_row = row, row++) {
218       (*losslessd->predict_undifference[compi])
219         (cinfo, compi, diff->diff_buf[compi][row],
220           diff->undiff_buf[compi][prev_row], diff->undiff_buf[compi][row],
221           compptr->width_in_blocks);
222       (*losslessd->scaler_scale) (cinfo, diff->undiff_buf[compi][row],
223                                   output_buf[compi][row],
224                                   compptr->width_in_blocks);
225     }
226   }
227
228   /* Completed the iMCU row, advance counters for next one.
229    *
230    * NB: output_data will increment output_iMCU_row.
231    * This counter is not needed for the single-pass case
232    * or the input side of the multi-pass case.
233    */
234   if (++(cinfo->input_iMCU_row) < cinfo->total_iMCU_rows) {
235     start_iMCU_row(cinfo);
236     return JPEG_ROW_COMPLETED;
237   }
238   /* Completed the scan */
239   (*cinfo->inputctl->finish_input_pass) (cinfo);
240   return JPEG_SCAN_COMPLETED;
241 }
242
243
244 /*
245  * Dummy consume-input routine for single-pass operation.
246  */
247
248 METHODDEF(int)
249 dummy_consume_data(j_decompress_ptr cinfo)
250 {
251   return JPEG_SUSPENDED;        /* Always indicate nothing was done */
252 }
253
254
255 #ifdef D_MULTISCAN_FILES_SUPPORTED
256
257 /*
258  * Consume input data and store it in the full-image sample buffer.
259  * We read as much as one fully interleaved MCU row ("iMCU" row) per call,
260  * ie, v_samp_factor rows for each component in the scan.
261  * Return value is JPEG_ROW_COMPLETED, JPEG_SCAN_COMPLETED, or JPEG_SUSPENDED.
262  */
263
264 METHODDEF(int)
265 consume_data(j_decompress_ptr cinfo)
266 {
267   my_diff_ptr diff = (my_diff_ptr)cinfo->coef;
268   int ci, compi;
269   _JSAMPARRAY buffer[MAX_COMPS_IN_SCAN];
270   jpeg_component_info *compptr;
271
272   /* Align the virtual buffers for the components used in this scan. */
273   for (ci = 0; ci < cinfo->comps_in_scan; ci++) {
274     compptr = cinfo->cur_comp_info[ci];
275     compi = compptr->component_index;
276     buffer[compi] = (_JSAMPARRAY)(*cinfo->mem->access_virt_sarray)
277       ((j_common_ptr)cinfo, diff->whole_image[compi],
278        cinfo->input_iMCU_row * compptr->v_samp_factor,
279        (JDIMENSION)compptr->v_samp_factor, TRUE);
280   }
281
282   return decompress_data(cinfo, buffer);
283 }
284
285
286 /*
287  * Output some data from the full-image sample buffer in the multi-pass case.
288  * Always attempts to emit one fully interleaved MCU row ("iMCU" row).
289  * Return value is JPEG_ROW_COMPLETED, JPEG_SCAN_COMPLETED, or JPEG_SUSPENDED.
290  *
291  * NB: output_buf contains a plane for each component in image.
292  */
293
294 METHODDEF(int)
295 output_data(j_decompress_ptr cinfo, _JSAMPIMAGE output_buf)
296 {
297   my_diff_ptr diff = (my_diff_ptr)cinfo->coef;
298   JDIMENSION last_iMCU_row = cinfo->total_iMCU_rows - 1;
299   int ci, samp_rows, row;
300   _JSAMPARRAY buffer;
301   jpeg_component_info *compptr;
302
303   /* Force some input to be done if we are getting ahead of the input. */
304   while (cinfo->input_scan_number < cinfo->output_scan_number ||
305          (cinfo->input_scan_number == cinfo->output_scan_number &&
306           cinfo->input_iMCU_row <= cinfo->output_iMCU_row)) {
307     if ((*cinfo->inputctl->consume_input) (cinfo) == JPEG_SUSPENDED)
308       return JPEG_SUSPENDED;
309   }
310
311   /* OK, output from the virtual arrays. */
312   for (ci = 0, compptr = cinfo->comp_info; ci < cinfo->num_components;
313        ci++, compptr++) {
314     /* Align the virtual buffer for this component. */
315     buffer = (_JSAMPARRAY)(*cinfo->mem->access_virt_sarray)
316       ((j_common_ptr)cinfo, diff->whole_image[ci],
317        cinfo->output_iMCU_row * compptr->v_samp_factor,
318        (JDIMENSION)compptr->v_samp_factor, FALSE);
319
320     if (cinfo->output_iMCU_row < last_iMCU_row)
321       samp_rows = compptr->v_samp_factor;
322     else {
323       /* NB: can't use last_row_height here; it is input-side-dependent! */
324       samp_rows = (int)(compptr->height_in_blocks % compptr->v_samp_factor);
325       if (samp_rows == 0) samp_rows = compptr->v_samp_factor;
326     }
327
328     for (row = 0; row < samp_rows; row++) {
329       memcpy(output_buf[ci][row], buffer[row],
330              compptr->width_in_blocks * sizeof(_JSAMPLE));
331     }
332   }
333
334   if (++(cinfo->output_iMCU_row) < cinfo->total_iMCU_rows)
335     return JPEG_ROW_COMPLETED;
336   return JPEG_SCAN_COMPLETED;
337 }
338
339 #endif /* D_MULTISCAN_FILES_SUPPORTED */
340
341
342 /*
343  * Initialize difference buffer controller.
344  */
345
346 GLOBAL(void)
347 _jinit_d_diff_controller(j_decompress_ptr cinfo, boolean need_full_buffer)
348 {
349   my_diff_ptr diff;
350   int ci;
351   jpeg_component_info *compptr;
352
353   diff = (my_diff_ptr)
354     (*cinfo->mem->alloc_small) ((j_common_ptr)cinfo, JPOOL_IMAGE,
355                                 sizeof(my_diff_controller));
356   cinfo->coef = (struct jpeg_d_coef_controller *)diff;
357   diff->pub.start_input_pass = start_input_pass;
358   diff->pub.start_output_pass = start_output_pass;
359
360   /* Create the [un]difference buffers. */
361   for (ci = 0, compptr = cinfo->comp_info; ci < cinfo->num_components;
362        ci++, compptr++) {
363     diff->diff_buf[ci] =
364       ALLOC_DARRAY(JPOOL_IMAGE,
365                    (JDIMENSION)jround_up((long)compptr->width_in_blocks,
366                                          (long)compptr->h_samp_factor),
367                    (JDIMENSION)compptr->v_samp_factor);
368     diff->undiff_buf[ci] =
369       ALLOC_DARRAY(JPOOL_IMAGE,
370                    (JDIMENSION)jround_up((long)compptr->width_in_blocks,
371                                          (long)compptr->h_samp_factor),
372                    (JDIMENSION)compptr->v_samp_factor);
373   }
374
375   if (need_full_buffer) {
376 #ifdef D_MULTISCAN_FILES_SUPPORTED
377     /* Allocate a full-image virtual array for each component. */
378     int access_rows;
379
380     for (ci = 0, compptr = cinfo->comp_info; ci < cinfo->num_components;
381          ci++, compptr++) {
382       access_rows = compptr->v_samp_factor;
383       diff->whole_image[ci] = (*cinfo->mem->request_virt_sarray)
384         ((j_common_ptr)cinfo, JPOOL_IMAGE, FALSE,
385          (JDIMENSION)jround_up((long)compptr->width_in_blocks,
386                                (long)compptr->h_samp_factor),
387          (JDIMENSION)jround_up((long)compptr->height_in_blocks,
388                                (long)compptr->v_samp_factor),
389          (JDIMENSION)access_rows);
390     }
391     diff->pub.consume_data = consume_data;
392     diff->pub._decompress_data = output_data;
393 #else
394     ERREXIT(cinfo, JERR_NOT_COMPILED);
395 #endif
396   } else {
397     diff->pub.consume_data = dummy_consume_data;
398     diff->pub._decompress_data = decompress_data;
399     diff->whole_image[0] = NULL; /* flag for no virtual arrays */
400   }
401 }
402
403 #endif /* D_LOSSLESS_SUPPORTED */