/src/ghostpdl/jbig2dec/jbig2_generic.c
Line | Count | Source (jump to first uncovered line) |
1 | | /* Copyright (C) 2001-2023 Artifex Software, Inc. |
2 | | All Rights Reserved. |
3 | | |
4 | | This software is provided AS-IS with no warranty, either express or |
5 | | implied. |
6 | | |
7 | | This software is distributed under license and may not be copied, |
8 | | modified or distributed except as expressly authorized under the terms |
9 | | of the license contained in the file LICENSE in this distribution. |
10 | | |
11 | | Refer to licensing information at http://www.artifex.com or contact |
12 | | Artifex Software, Inc., 39 Mesa Street, Suite 108A, San Francisco, |
13 | | CA 94129, USA, for further information. |
14 | | */ |
15 | | |
16 | | /* |
17 | | jbig2dec |
18 | | */ |
19 | | |
20 | | /** |
21 | | * Generic region handlers. |
22 | | **/ |
23 | | |
24 | | #ifdef HAVE_CONFIG_H |
25 | | #include "config.h" |
26 | | #endif |
27 | | #include "os_types.h" |
28 | | |
29 | | #include <stddef.h> |
30 | | #include <string.h> /* memcpy(), memset() */ |
31 | | |
32 | | #ifdef OUTPUT_PBM |
33 | | #include <stdio.h> |
34 | | #endif |
35 | | |
36 | | #include "jbig2.h" |
37 | | #include "jbig2_priv.h" |
38 | | #include "jbig2_arith.h" |
39 | | #include "jbig2_generic.h" |
40 | | #include "jbig2_image.h" |
41 | | #include "jbig2_mmr.h" |
42 | | #include "jbig2_page.h" |
43 | | #include "jbig2_segment.h" |
44 | | |
45 | | /* |
46 | | This is an explanation of the unoptimized and optimized generic |
47 | | region decoder implementations below, wherein we try to explain |
48 | | all the magic numbers. |
49 | | |
50 | | The generic region decoders decode the output pixels one row at a |
51 | | time, top to bottom. Within each row the pixels are decoded left |
52 | | to right. The input for the arithmetic integer decoder used to |
53 | | decode each pixel is a context consisting of up to 16 previously |
54 | | decoded pixels. These pixels are chosen according to a predefined |
55 | | template placed relative to the location of the pixel to be |
56 | | decoded (6.2.5.3 figures 3, 4, 5 and 6). There are four different |
57 | | template that may be used (6.2.5.3). The template to use is |
58 | | determined by GBTEMPLATE. GBTEMPLATE is set in the symbol |
59 | | dictionary (6.5.8.1), generic region (7.4.6.4), or when decoding |
60 | | a halftone region's gray-scale image (annex C.5). |
61 | | |
62 | | Most of the pixels in each template have fixed locations relative |
63 | | to the pixel to be decoded. However, all templates have at least |
64 | | one adaptive pixel. The adaptive pixels have nominal locations, |
65 | | but these locations may be changed by GBAT. GBAT is set in the |
66 | | symbol dictionary (7.4.2.1.2), generic region (7.4.6.1), or hard |
67 | | coded as for halftone patterns (6.7.5). |
68 | | |
69 | | Adaptive pixels are restricted to fall within a field of |
70 | | previously decoded pixels relative to the pixel to be decoded |
71 | | (figure 7). The relative Y-coordinate for these adaptive pixels |
72 | | may vary between -128 and 0. The relative X-coordinate may vary |
73 | | between -128 and +127 (however, if the Y-coordinate is 0 the |
74 | | range of the X-coordinate is further restricted to -128 to -1 |
75 | | since the pixels at locations 0 to +127 have not yet been |
76 | | decoded). If a template refers to a pixel location that reside |
77 | | outside of the image boundaries its value is assumed to be 0. |
78 | | |
79 | | UNOPTIMIZED DECODER |
80 | | |
81 | | The unoptimized decoders first check the contents of GBAT. If |
82 | | GBAT specifies that any of the adaptive pixels reside outside the |
83 | | allowed field the decoding is aborted. Next, each row is |
84 | | processed top to bottom, left to right, one pixel at a time. For |
85 | | each pixel a context is created containing the bit values of the |
86 | | pixels that fall inside the template. |
87 | | |
88 | | The order these bits are stored in the context is implementation |
89 | | dependent (6.2.5.3). We store the bit values in the CONTEXT |
90 | | variable from LSB to MSB, starting with the value of the pixel to |
91 | | the left of the current pixel, continuing right to left, bottom |
92 | | to top following the template. Using the CONTEXT created from |
93 | | these pixel values, the arithmetic integer decoder retrieves the |
94 | | pixel value, which is then written into the output image. |
95 | | |
96 | | Example when GBTEMPLATE is 2: |
97 | | |
98 | | The figure below represents a pixel grid of the output image. |
99 | | Each pixel is a single bit in the image. The pixel "OO" in the |
100 | | figure below is about to be decoded. The pixels "??" have not |
101 | | been decoded yet. The CONTEXT variable is constructed by |
102 | | combining the bit values from the pixels referred to by the |
103 | | template, shifted to their corresponding bit position. |
104 | | |
105 | | . . . . . . . . |
106 | | . . . . . . . . |
107 | | ...+----+----+----+----+----+----+----+... |
108 | | | | | X9 | X8 | X7 | | | |
109 | | ...+----+----+----+----+----+----+----+... |
110 | | | | X6 | X5 | X4 | X3 | A1 | | |
111 | | ...+----+----+----+----+----+----+----+... |
112 | | | | X2 | X1 | OO | ?? | ?? | ?? | |
113 | | ...+----+----+----+----+----+----+----+... |
114 | | . . . . . . . . |
115 | | . . . . . . . . |
116 | | |
117 | | In the table below pixel OO is assumed to be at coordinate (x, y). |
118 | | |
119 | | Bit 9: Pixel at location (x-1, y-2) (This is fixed pixel X9) |
120 | | Bit 8: Pixel at location (x , y-2) (This is fixed pixel X8) |
121 | | Bit 7: Pixel at location (x+1, y-2) (This is fixed pixel X7) |
122 | | Bit 6: Pixel at location (x-2, y-1) (This is fixed pixel X6) |
123 | | Bit 5: Pixel at location (x-1, y-1) (This is fixed pixel X5) |
124 | | Bit 4: Pixel at location (x , y-1) (This is fixed pixel X4) |
125 | | Bit 3: Pixel at location (x+1, y-1) (This is fixed pixel X3) |
126 | | Bit 2: Pixel at location (x+2, y-1) (This is adaptive pixel A1) |
127 | | Bit 1: Pixel at location (x-2, y ) (This is fixed pixel X2) |
128 | | Bit 0: Pixel at location (x-1, y ) (This is fixed pixel X1) |
129 | | |
130 | | The location of adaptive pixel A1 may not always be at the |
131 | | nominal location (x+2, y-1). It could be at any pixel location to |
132 | | the left or above OO as specified by GBAT, e.g. at the location |
133 | | (x-128, y+127). |
134 | | |
135 | | OPTIMIZED DECODER |
136 | | |
137 | | The optimized decoders work differently. They strive to avoid |
138 | | recreating the arithmetic integer decoder context from scratch |
139 | | for every pixel decoded. Instead they reuse part of the CONTEXT |
140 | | used to compute the previous pixel (the pixel to left of the one |
141 | | now being decoded). They also keep two sliding windows of pixel |
142 | | bit values from the two rows of pixels immediately above the |
143 | | pixel to be decoded. These are stored in the 32-bit variables |
144 | | line_m1 (row above the pixel to be decoded) and line_m2 (row |
145 | | above that of line_m1). These optimized decoders ONLY work for |
146 | | the nominal adaptive pixel locations since these locations are |
147 | | hard-coded into the implementation. |
148 | | |
149 | | The bit ordering in the CONTEXT variable is identical to the |
150 | | unoptimized case described above. |
151 | | |
152 | | The optimized decoders decode the output pixels one row at a |
153 | | time, top to bottom. Within each row the pixels are decoded in |
154 | | batches of up to eight pixels at a time (except possibly the |
155 | | right most batch which may be less than eight pixels). The |
156 | | batches in a row are decoded in sequence from left to right. |
157 | | Within each such batch the pixels are decoded in sequence from |
158 | | left to right. |
159 | | |
160 | | Before decoding the pixels in a row the two sliding windows of |
161 | | pixel values are reset. The first eight pixels of the row above |
162 | | the pixel to be decoded is stored in line_m1, while line_m2 |
163 | | stores the first eight pixels of the row above that of line_m1. |
164 | | |
165 | | The figure below illustrates the situation where the template has |
166 | | been placed so that the decoded pixel OO is the very first pixel |
167 | | of a row. It also gives labels to various pixels that we will |
168 | | refer to below. |
169 | | |
170 | | . . . . . . . . . . . |
171 | | | . . . . . . . . . . |
172 | | + + +----+----+----+----+----+----+----+----+----+----+... |
173 | | X9 | X8 | X7 | m1 | m2 | m3 | m4 | m5 | m6 | m7 | | |
174 | | + + +----+----+----+----+----+----+----+----+----+----+... |
175 | | X6 X5 | X4 | X3 | A1 | n1 | n2 | n3 | n4 | n5 | n6 | n7 | |
176 | | + + +----+----+----+----+----+----+----+----+----+----+... |
177 | | X2 X1 | OO | | | | | | | | | | |
178 | | + + +----+----+----+----+----+----+----+----+----+----+... |
179 | | | . . . . . . . . . . |
180 | | . . . . . . . . . . . |
181 | | |
182 | | The pixels X1, X2, X5, X6 and X9 all reside outside the left edge |
183 | | of the image. These pixels (like all others outside the image) |
184 | | can according to 6.2.5.2 be assumed to be 0. line_m1 stores n5 |
185 | | through n1 as well as A1, and X3 through X6. line_m2 stores m6 |
186 | | through m1 as well as X7 through X9. The bits in line_m2 are also |
187 | | shifted left four bits as seen below. |
188 | | |
189 | | 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0 | bit position |
190 | | ------------------------------------------------+------------------ |
191 | | 0 0 0 0 0 0 X6 X5 X4 X3 A1 n1 n2 n3 n4 n5 | line_m1 |
192 | | 0 0 0 X9 X8 X7 m1 m2 m3 m4 m5 m6 0 0 0 0 | line_m2 |
193 | | |
194 | | The way line_m1 and line_m2 are stored means we can simply shift |
195 | | them by the same amount to move the sliding window. |
196 | | |
197 | | The bit order in line_m1 and line_m2 matches the ordering in the |
198 | | CONTEXT variable. Each bit for the 'A' and 'X' pixels in line_m1 |
199 | | and line_m2 correspond to the equivalent bits in CONTEXT, only |
200 | | shifted right by 3 bits. Thus X3 is bit 3 in CONTEXT and bit 6 in |
201 | | line_m1, etc. |
202 | | |
203 | | The initial arithmetic integer decoder context is created and |
204 | | stored in the CONTEXT variable by masking, shifting, and bitwise |
205 | | ORing the contents of line_m1 and line_m2. The "CONTEXT contents" |
206 | | row is only shown for clarity, it is not present in the code. |
207 | | |
208 | | 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0 | bit position |
209 | | ------------------------------------------------+--------------------------- |
210 | | 0 0 0 0 0 0 0 0 0 X6 X5 X4 X3 A1 n1 n2 | line_m1 >> 3 |
211 | | 0 0 0 0 0 0 0 0 0 1 1 1 1 1 0 0 | mask for line_m1 (0x7c) |
212 | | 0 0 0 0 0 0 0 0 0 X6 X5 X4 X3 A1 0 0 | line_m1 AND mask |
213 | | ------------------------------------------------+--------------------------- |
214 | | 0 0 0 0 0 0 X9 X8 X7 m1 m2 m3 m4 m5 m6 0 | line_m2 >> 3 |
215 | | 0 0 0 0 0 0 1 1 1 0 0 0 0 0 0 0 | mask for line_m2 (0x380) |
216 | | 0 0 0 0 0 0 X9 X8 X7 0 0 0 0 0 0 0 | line_m2 AND mask |
217 | | ------------------------------------------------+--------------------------- |
218 | | 0 0 0 0 0 0 X9 X8 X7 X6 X5 X4 X3 A1 0 0 | CONTEXT = line_m1 OR line_m2 |
219 | | ------------------------------------------------+--------------------------- |
220 | | 0 0 0 0 0 0 X9 X8 X7 X6 X5 X4 X3 A1 X2 X1 | CONTEXT contents |
221 | | |
222 | | Each batch is normally 8 bits, but at the right edge of the image |
223 | | we may have fewer pixels to decode. The minor_width is how many |
224 | | pixels the current batch should decode, with a counter variable |
225 | | x_minor to keep track of the current pixel being decoded. |
226 | | |
227 | | In order to process a new batch of pixels, unless we're at the |
228 | | rightmost batch of pixels, we need to refill the sliding window |
229 | | variables with eight new bits. Looking at the diagram above we |
230 | | can see that in order to decode eight pixels starting with O0 |
231 | | we'll need to have bits up to pixel 'n7' for line_m1 and 'm7' for |
232 | | line_m2 available (A1 and X7 moved right 7 times). To do this |
233 | | simply and quickly, we shift line_m1 left by 8 bits, and OR in |
234 | | the next byte from corresponding row. Likewise for line_m2, but |
235 | | the next byte from the image is also shifted left by 4 bits to |
236 | | compensate for line_m2 having the four least significant bits |
237 | | unused. |
238 | | |
239 | | These new eight bits contain the bit values of the eight pixels |
240 | | to the right of those already present in line_m1 and line_m2. We |
241 | | call these new bits m7 through mE, and n6 through nD, as |
242 | | illustrated below. |
243 | | |
244 | | 23 22 21 20 19 18 17 16 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0 | bit position |
245 | | ------------------------------------------------------------------------+------------- |
246 | | 0 0 0 0 0 0 0 0 0 0 0 0 0 0 X6 X5 X4 X3 A1 n1 n2 n3 n4 n5 | original line_m1 |
247 | | 0 0 0 0 0 0 X6 X5 X4 X3 A1 n1 n2 n3 n4 n5 0 0 0 0 0 0 0 0 | line_m1 shifted left by 8 |
248 | | 0 0 0 0 0 0 X6 X5 X4 X3 A1 n1 n2 n3 n4 n5 n6 n7 n8 n9 nA nB nC nD | line_m1 with new bits ORed in |
249 | | ------------------------------------------------------------------------+------------- |
250 | | 0 0 0 0 0 0 0 0 0 0 0 X9 X8 X7 m1 m2 m3 m4 m5 m6 0 0 0 0 | original line_m2 |
251 | | 0 0 0 X9 X8 X7 m1 m2 m3 m4 m5 m6 0 0 0 0 0 0 0 0 0 0 0 0 | line_m2 shifted left by 8 |
252 | | 0 0 0 X9 X8 X7 m1 m2 m3 m4 m5 m6 m7 m8 m9 mA mB mC mD mE 0 0 0 0 | line_m2 with new bits ORed in |
253 | | |
254 | | . . . . . . . . . . . . . . . . . . . . |
255 | | | . . . . . . . . . . . . . . . . . . . |
256 | | + + +----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+... |
257 | | X9 | X8 | X7 | m1 | m2 | m3 | m4 | m5 | m6 | m7 | m8 | m9 | mA | mB | mC | mD | mE | | | | |
258 | | + + +----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+... |
259 | | X6 X5 | X4 | X3 | A1 | n1 | n2 | n3 | n4 | n5 | n6 | n7 | n8 | n9 | nA | nB | nC | nD | | | | |
260 | | + + +----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+... |
261 | | X2 X1 | OO | | | | | | | | | | | | | | | | | | | |
262 | | + + +----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+... |
263 | | | . . . . . . . . . . . . . . . . . . . |
264 | | . . . . . . . . . . . . . . . . . . . . |
265 | | |
266 | | CONTEXT, line_m1 and line_m2 now contain all necessary bits to |
267 | | decode a full batch of eight pixels. |
268 | | |
269 | | The first pixel in the batch is decoded using this CONTEXT. After |
270 | | that, for each following pixel we need to update the CONTEXT |
271 | | using both the last decoded pixel value and new bits from line_m1 |
272 | | and line_m2. |
273 | | |
274 | | . . . . . . . . . . . . . . . . . . . . |
275 | | | . . . . . . . . . . . . . . . . . . . |
276 | | + + +----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+... |
277 | | (X9)|_X8_|_X7_|>m1<| m2 | m3 | m4 | m5 | m6 | m7 | m8 | m9 | mA | mB | mC | mD | mE | | | | |
278 | | + + +----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+... |
279 | | (X6) _X5_|_X4_|_X3_|_A1_|>n1<| n2 | n3 | n4 | n5 | n6 | n7 | n8 | n9 | nA | nB | nC | nD | | | | |
280 | | + + +----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+... |
281 | | (X2) _X1_|>OO<| oo | | | | | | | | | | | | | | | | | | |
282 | | + + +----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+----+... |
283 | | | . . . . . . . . . . . . . . . . . . . |
284 | | . . . . . . . . . . . . . . . . . . . . |
285 | | |
286 | | This figure illustrates what happens when the same template is |
287 | | overlaid on itself shifted one pixel to the right in order to |
288 | | decode the next pixel. Pixels marked with _ _ are pixels that |
289 | | are present in both templates' CONTEXTs and can be reused. Pixels |
290 | | marked with ( ) are pixels from the first template that are no |
291 | | longer necessary and can be removed from CONTEXT. Pixels marked |
292 | | with > < are new pixels that were not part of the original |
293 | | CONTEXT, and so need to be moved into the CONTEXT at the |
294 | | appropriate locations. In general the leftmost pixels of each |
295 | | template row can be forgotten, while new pixels are needed at the |
296 | | right most location of each row. |
297 | | |
298 | | The CONTEXT corresponding to the current pixel OO and how it is |
299 | | masked is shown below. Note how the left most pixel of each row |
300 | | of the template is NOT propagated to the CONTEXT, these pixels |
301 | | are X2, X6 and X9. This is done by having the mask being 0 at the |
302 | | corresponding locations. |
303 | | |
304 | | 9 8 7 6 5 4 3 2 1 0 | bit position |
305 | | ------------------------------+------------- |
306 | | X9 X8 X7 X6 X5 X4 X3 A1 X2 X1 | pixel values from CONTEXT |
307 | | 0 1 1 0 1 1 1 1 0 1 | reused pixel bit value mask (0x1bd) |
308 | | 0 X8 X7 0 X5 X4 X3 A1 0 X1 | reused pixel values from CONTEXT |
309 | | |
310 | | Next the CONTEXT is shifted left by one bit to make it reference |
311 | | the next pixel to be decoded. The pixel bit value we just decoded |
312 | | is then written into the bit corresponding to X1. The sliding |
313 | | windows in line_m1 and line_m2 are both shifted (10 - x_minor) |
314 | | bits to the right to make the needed pixels' bit values appear at |
315 | | the correct positions to be ORed into CONTEXT. Note that this |
316 | | shift amount depends on which bit in the batch is currently being |
317 | | computed, as is given by the x_minor counter. In the example |
318 | | below we assume that x_minor is 0. |
319 | | |
320 | | 9 8 7 6 5 4 3 2 1 0 | bit position |
321 | | ------------------------------+-------------- |
322 | | 0 X8 X7 0 X5 X4 X3 A1 0 0 | reused pixels from CONTEXT |
323 | | X8 X7 0 X5 X4 X3 A1 0 0 0 | reused pixels shifted left 1 bit |
324 | | ------------------------------+-------------- |
325 | | X8 X7 0 X5 X4 X3 A1 0 X1 OO | new CONTEXT with current pixel at LSB |
326 | | ------------------------------+-------------- |
327 | | 0 0 X6 X5 X4 X3 A1 n1 n2 n3 | line_m1 shifted (10 - x_minor) bits to the right |
328 | | 0 0 0 0 0 0 0 1 0 0 | mask for new adaptive pixel one row above (0x4) |
329 | | X8 X7 0 X5 X4 X3 A1 n1 X1 OO | new CONTEXT with new adaptive pixel |
330 | | ------------------------------+-------------- |
331 | | X8 X7 m1 m2 m3 m4 m5 m6 m7 m8 | line_m2 with new bits ORed in |
332 | | 0 0 1 0 0 0 0 0 0 0 | mask for new pixel two rows above (0x80) |
333 | | X8 X7 m1 X5 X4 X3 A1 n1 X1 OO | new CONTEXT with new pixel |
334 | | |
335 | | This makes the computation of the new CONTEXT be: |
336 | | |
337 | | NEWCONTEXT = (CONTEXT & 0x1bd) << 1 |
338 | | NEWCONTEXT |= newbit; |
339 | | NEWCONTEXT |= (line_m1 >> (10-x_minor)) & 0x4; |
340 | | NEWCONTEXT |= (line_m2 >> (10-x_minor)) & 0x80; |
341 | | |
342 | | The optimized decoding functions for GBTEMPLATE 0, 1 and 3 all |
343 | | work similarly. */ |
344 | | |
345 | | /* Get a bit. No bounds checking. */ |
346 | | static inline int |
347 | | jbig2_image_get_pixel_fast(Jbig2Image *image, int x, int y) |
348 | 0 | { |
349 | 0 | const int byte = (x >> 3) + y * image->stride; |
350 | 0 | const int bit = 7 - (x & 7); |
351 | |
|
352 | 0 | return ((image->data[byte] >> bit) & 1); |
353 | 0 | } |
354 | | |
355 | | /* return the appropriate context size for the given template */ |
356 | | int |
357 | | jbig2_generic_stats_size(Jbig2Ctx *ctx, int template) |
358 | 452 | { |
359 | 452 | int stats_size = template == 0 ? 1 << 16 : template == 1 ? 1 << 13 : 1 << 10; |
360 | | |
361 | 452 | return stats_size; |
362 | 452 | } |
363 | | |
364 | | static int |
365 | | jbig2_decode_generic_template0(Jbig2Ctx *ctx, |
366 | | Jbig2Segment *segment, |
367 | | const Jbig2GenericRegionParams *params, Jbig2ArithState *as, |
368 | | Jbig2Image *image, Jbig2ArithCx *GB_stats) |
369 | 519 | { |
370 | 519 | const uint32_t GBW = image->width; |
371 | 519 | const uint32_t GBH = image->height; |
372 | 519 | const uint32_t rowstride = image->stride; |
373 | 519 | uint32_t x, y; |
374 | 519 | byte *line2 = NULL; |
375 | 519 | byte *line1 = NULL; |
376 | 519 | byte *gbreg_line = (byte *) image->data; |
377 | | |
378 | | #ifdef OUTPUT_PBM |
379 | | printf("P4\n%d %d\n", GBW, GBH); |
380 | | #endif |
381 | | |
382 | 519 | if (GBW <= 0) |
383 | 0 | return 0; |
384 | | |
385 | 1.55M | for (y = 0; y < GBH; y++) { |
386 | 1.55M | uint32_t CONTEXT; |
387 | 1.55M | uint32_t line_m1; |
388 | 1.55M | uint32_t line_m2; |
389 | 1.55M | uint32_t padded_width = (GBW + 7) & -8; |
390 | | |
391 | 1.55M | line_m1 = line1 ? line1[0] : 0; |
392 | 1.55M | line_m2 = line2 ? line2[0] << 6 : 0; |
393 | 1.55M | CONTEXT = (line_m1 & 0x7f0) | (line_m2 & 0xf800); |
394 | | |
395 | | /* 6.2.5.7 3d */ |
396 | 479M | for (x = 0; x < padded_width; x += 8) { |
397 | 477M | byte result = 0; |
398 | 477M | int x_minor; |
399 | 477M | int minor_width = GBW - x > 8 ? 8 : GBW - x; |
400 | | |
401 | 477M | if (line1) |
402 | 477M | line_m1 = (line_m1 << 8) | (x + 8 < GBW ? line1[(x >> 3) + 1] : 0); |
403 | | |
404 | 477M | if (line2) |
405 | 477M | line_m2 = (line_m2 << 8) | (x + 8 < GBW ? line2[(x >> 3) + 1] << 6 : 0); |
406 | | |
407 | | /* This is the speed-critical inner loop. */ |
408 | 4.29G | for (x_minor = 0; x_minor < minor_width; x_minor++) { |
409 | 3.82G | int bit; |
410 | | |
411 | 3.82G | bit = jbig2_arith_decode(ctx, as, &GB_stats[CONTEXT]); |
412 | 3.82G | if (bit < 0) |
413 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template0 optimized"); |
414 | 3.82G | result |= bit << (7 - x_minor); |
415 | 3.82G | CONTEXT = ((CONTEXT & 0x7bf7) << 1) | bit | ((line_m1 >> (7 - x_minor)) & 0x10) | ((line_m2 >> (7 - x_minor)) & 0x800); |
416 | 3.82G | } |
417 | 477M | gbreg_line[x >> 3] = result; |
418 | 477M | } |
419 | | #ifdef OUTPUT_PBM |
420 | | fwrite(gbreg_line, 1, rowstride, stdout); |
421 | | #endif |
422 | 1.55M | line2 = line1; |
423 | 1.55M | line1 = gbreg_line; |
424 | 1.55M | gbreg_line += rowstride; |
425 | 1.55M | } |
426 | | |
427 | 519 | return 0; |
428 | 519 | } |
429 | | |
430 | | #define pixel_outside_field(x, y) \ |
431 | 116 | ((y) < -128 || (y) > 0 || (x) < -128 || ((y) < 0 && (x) > 127) || ((y) == 0 && (x) >= 0)) |
432 | | |
433 | | static int |
434 | | jbig2_decode_generic_template0_unopt(Jbig2Ctx *ctx, |
435 | | Jbig2Segment *segment, |
436 | | const Jbig2GenericRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GB_stats) |
437 | 17 | { |
438 | 17 | const uint32_t GBW = image->width; |
439 | 17 | const uint32_t GBH = image->height; |
440 | 17 | uint32_t CONTEXT; |
441 | 17 | uint32_t x, y; |
442 | 17 | int bit; |
443 | | |
444 | 17 | if (pixel_outside_field(params->gbat[0], params->gbat[1]) || |
445 | 17 | pixel_outside_field(params->gbat[2], params->gbat[3]) || |
446 | 17 | pixel_outside_field(params->gbat[4], params->gbat[5]) || |
447 | 17 | pixel_outside_field(params->gbat[6], params->gbat[7])) |
448 | 1 | return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, |
449 | 1 | "adaptive template pixel is out of field"); |
450 | | |
451 | 4.01k | for (y = 0; y < GBH; y++) { |
452 | 4.00k | uint32_t out_byte = 0; |
453 | 4.00k | int out_bits_to_go_in_byte = 8; |
454 | 4.00k | uint8_t *d = &image->data[image->stride * y]; |
455 | 4.00k | uint32_t pd = 0; |
456 | 4.00k | uint32_t ppd = 0; |
457 | 4.00k | uint8_t *pline = NULL; |
458 | 4.00k | uint8_t *ppline = NULL; |
459 | 4.00k | if (y >= 1) |
460 | 3.98k | { |
461 | 3.98k | pline = &image->data[image->stride * (y-1)]; |
462 | 3.98k | pd = (*pline++ << 8); |
463 | 3.98k | if (GBW > 8) |
464 | 3.98k | pd |= *pline++; |
465 | 3.98k | } |
466 | 4.00k | if (y >= 2) { |
467 | 3.96k | ppline = &image->data[image->stride * (y-2)]; |
468 | 3.96k | ppd = (*ppline++ << 8); |
469 | 3.96k | if (GBW > 8) |
470 | 3.96k | ppd |= *ppline++; |
471 | 3.96k | } |
472 | 912k | for (x = 0; x < GBW; x++) { |
473 | 908k | if (params->USESKIP && jbig2_image_get_pixel(params->SKIP, x, y)) { |
474 | 0 | bit = 0; |
475 | 908k | } else { |
476 | 908k | CONTEXT = out_byte & 0x000F; /* First 4 pixels */ |
477 | 908k | CONTEXT |= jbig2_image_get_pixel(image, x + params->gbat[0], y + params->gbat[1]) << 4; |
478 | 908k | CONTEXT |= (pd>>8) & 0x03E0; /* Next 5 pixels */ |
479 | 908k | CONTEXT |= jbig2_image_get_pixel(image, x + params->gbat[2], y + params->gbat[3]) << 10; |
480 | 908k | CONTEXT |= jbig2_image_get_pixel(image, x + params->gbat[4], y + params->gbat[5]) << 11; |
481 | 908k | CONTEXT |= (ppd>>2) & 0x7000; /* Next 3 pixels */ |
482 | 908k | CONTEXT |= jbig2_image_get_pixel(image, x + params->gbat[6], y + params->gbat[7]) << 15; |
483 | 908k | bit = jbig2_arith_decode(ctx, as, &GB_stats[CONTEXT]); |
484 | 908k | if (bit < 0) |
485 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template0 unoptimized"); |
486 | 908k | } |
487 | 908k | pd = pd<<1; |
488 | 908k | ppd = ppd<<1; |
489 | 908k | out_byte = (out_byte<<1) | bit; |
490 | 908k | out_bits_to_go_in_byte--; |
491 | 908k | *d = out_byte<<out_bits_to_go_in_byte; |
492 | 908k | if (out_bits_to_go_in_byte == 0) { |
493 | 113k | out_bits_to_go_in_byte = 8; |
494 | 113k | d++; |
495 | 113k | if (x+9 < GBW && pline != NULL) { |
496 | 106k | pd |= *pline++; |
497 | 106k | if (ppline != NULL) |
498 | 105k | ppd |= *ppline++; |
499 | 106k | } |
500 | 113k | } |
501 | 908k | } |
502 | 4.00k | if (out_bits_to_go_in_byte != 8) |
503 | 1.50k | *d = (uint8_t)out_byte<<out_bits_to_go_in_byte; |
504 | 4.00k | } |
505 | 16 | return 0; |
506 | 16 | } |
507 | | |
508 | | static int |
509 | | jbig2_decode_generic_template1_unopt(Jbig2Ctx *ctx, |
510 | | Jbig2Segment *segment, |
511 | | const Jbig2GenericRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GB_stats) |
512 | 0 | { |
513 | 0 | const uint32_t GBW = image->width; |
514 | 0 | const uint32_t GBH = image->height; |
515 | 0 | uint32_t CONTEXT; |
516 | 0 | uint32_t x, y; |
517 | 0 | int bit; |
518 | |
|
519 | 0 | if (pixel_outside_field(params->gbat[0], params->gbat[1])) |
520 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, |
521 | 0 | "adaptive template pixel is out of field"); |
522 | | |
523 | 0 | for (y = 0; y < GBH; y++) { |
524 | 0 | uint32_t out_byte = 0; |
525 | 0 | int out_bits_to_go_in_byte = 8; |
526 | 0 | uint8_t *d = &image->data[image->stride * y]; |
527 | 0 | uint32_t pd = 0; |
528 | 0 | uint32_t ppd = 0; |
529 | 0 | uint8_t *pline = NULL; |
530 | 0 | uint8_t *ppline = NULL; |
531 | 0 | if (y >= 1) { |
532 | 0 | pline = &image->data[image->stride * (y-1)]; |
533 | 0 | pd = (*pline++ << 8); |
534 | 0 | if (GBW > 8) |
535 | 0 | pd |= *pline++; |
536 | 0 | } |
537 | 0 | if (y >= 2) { |
538 | 0 | ppline = &image->data[image->stride * (y-2)]; |
539 | 0 | ppd = (*ppline++ << 8); |
540 | 0 | if (GBW > 8) |
541 | 0 | ppd |= *ppline++; |
542 | 0 | } |
543 | 0 | for (x = 0; x < GBW; x++) { |
544 | 0 | if (params->USESKIP && jbig2_image_get_pixel(params->SKIP, x, y)) { |
545 | 0 | bit = 0; |
546 | 0 | } else { |
547 | 0 | CONTEXT = out_byte & 0x0007; /* First 3 pixels */ |
548 | 0 | CONTEXT |= jbig2_image_get_pixel(image, x + params->gbat[0], y + params->gbat[1]) << 3; |
549 | 0 | CONTEXT |= (pd>>9) & 0x01F0; /* Next 5 pixels */ |
550 | 0 | CONTEXT |= (ppd>>4) & 0x1E00; /* Next 4 pixels */ |
551 | 0 | bit = jbig2_arith_decode(ctx, as, &GB_stats[CONTEXT]); |
552 | 0 | if (bit < 0) |
553 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template1 unoptimized"); |
554 | 0 | } |
555 | 0 | pd = pd<<1; |
556 | 0 | ppd = ppd<<1; |
557 | 0 | out_byte = (out_byte<<1) | bit; |
558 | 0 | out_bits_to_go_in_byte--; |
559 | 0 | *d = out_byte<<out_bits_to_go_in_byte; |
560 | 0 | if (out_bits_to_go_in_byte == 0) { |
561 | 0 | out_bits_to_go_in_byte = 8; |
562 | 0 | d++; |
563 | 0 | if (x+9 < GBW && pline != NULL) { |
564 | 0 | pd |= *pline++; |
565 | 0 | if (ppline != NULL) |
566 | 0 | ppd |= *ppline++; |
567 | 0 | } |
568 | 0 | } |
569 | 0 | } |
570 | 0 | if (out_bits_to_go_in_byte != 8) |
571 | 0 | *d = (uint8_t)out_byte<<out_bits_to_go_in_byte; |
572 | 0 | } |
573 | 0 | return 0; |
574 | 0 | } |
575 | | |
576 | | static int |
577 | | jbig2_decode_generic_template1(Jbig2Ctx *ctx, |
578 | | Jbig2Segment *segment, |
579 | | const Jbig2GenericRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GB_stats) |
580 | 0 | { |
581 | 0 | const uint32_t GBW = image->width; |
582 | 0 | const uint32_t GBH = image->height; |
583 | 0 | const uint32_t rowstride = image->stride; |
584 | 0 | uint32_t x, y; |
585 | 0 | byte *line2 = NULL; |
586 | 0 | byte *line1 = NULL; |
587 | 0 | byte *gbreg_line = (byte *) image->data; |
588 | |
|
589 | | #ifdef OUTPUT_PBM |
590 | | printf("P4\n%d %d\n", GBW, GBH); |
591 | | #endif |
592 | |
|
593 | 0 | if (GBW <= 0) |
594 | 0 | return 0; |
595 | | |
596 | 0 | for (y = 0; y < GBH; y++) { |
597 | 0 | uint32_t CONTEXT; |
598 | 0 | uint32_t line_m1; |
599 | 0 | uint32_t line_m2; |
600 | 0 | uint32_t padded_width = (GBW + 7) & -8; |
601 | |
|
602 | 0 | line_m1 = line1 ? line1[0] : 0; |
603 | 0 | line_m2 = line2 ? line2[0] << 5 : 0; |
604 | 0 | CONTEXT = ((line_m1 >> 1) & 0x1f8) | ((line_m2 >> 1) & 0x1e00); |
605 | | |
606 | | /* 6.2.5.7 3d */ |
607 | 0 | for (x = 0; x < padded_width; x += 8) { |
608 | 0 | byte result = 0; |
609 | 0 | int x_minor; |
610 | 0 | int minor_width = GBW - x > 8 ? 8 : GBW - x; |
611 | |
|
612 | 0 | if (line1) |
613 | 0 | line_m1 = (line_m1 << 8) | (x + 8 < GBW ? line1[(x >> 3) + 1] : 0); |
614 | |
|
615 | 0 | if (line2) |
616 | 0 | line_m2 = (line_m2 << 8) | (x + 8 < GBW ? line2[(x >> 3) + 1] << 5 : 0); |
617 | | |
618 | | /* This is the speed-critical inner loop. */ |
619 | 0 | for (x_minor = 0; x_minor < minor_width; x_minor++) { |
620 | 0 | int bit; |
621 | |
|
622 | 0 | bit = jbig2_arith_decode(ctx, as, &GB_stats[CONTEXT]); |
623 | 0 | if (bit < 0) |
624 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template1 optimized"); |
625 | 0 | result |= bit << (7 - x_minor); |
626 | 0 | CONTEXT = ((CONTEXT & 0xefb) << 1) | bit | ((line_m1 >> (8 - x_minor)) & 0x8) | ((line_m2 >> (8 - x_minor)) & 0x200); |
627 | 0 | } |
628 | 0 | gbreg_line[x >> 3] = result; |
629 | 0 | } |
630 | | #ifdef OUTPUT_PBM |
631 | | fwrite(gbreg_line, 1, rowstride, stdout); |
632 | | #endif |
633 | 0 | line2 = line1; |
634 | 0 | line1 = gbreg_line; |
635 | 0 | gbreg_line += rowstride; |
636 | 0 | } |
637 | | |
638 | 0 | return 0; |
639 | 0 | } |
640 | | |
641 | | static int |
642 | | jbig2_decode_generic_template2_unopt(Jbig2Ctx *ctx, |
643 | | Jbig2Segment *segment, |
644 | | const Jbig2GenericRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GB_stats) |
645 | 0 | { |
646 | 0 | const uint32_t GBW = image->width; |
647 | 0 | const uint32_t GBH = image->height; |
648 | 0 | uint32_t CONTEXT; |
649 | 0 | uint32_t x, y; |
650 | 0 | int bit; |
651 | |
|
652 | 0 | if (pixel_outside_field(params->gbat[0], params->gbat[1])) |
653 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, |
654 | 0 | "adaptive template pixel is out of field"); |
655 | | |
656 | 0 | for (y = 0; y < GBH; y++) { |
657 | 0 | uint32_t out_byte = 0; |
658 | 0 | int out_bits_to_go_in_byte = 8; |
659 | 0 | uint8_t *d = &image->data[image->stride * y]; |
660 | 0 | uint8_t *pline = &image->data[image->stride * (y-1)]; |
661 | 0 | uint8_t *ppline = &image->data[image->stride * (y-2)]; |
662 | 0 | uint32_t pd = 0; |
663 | 0 | uint32_t ppd = 0; |
664 | 0 | if (y > 0) { |
665 | 0 | pd = (*pline++ << 8); |
666 | 0 | if (GBW > 8) |
667 | 0 | pd |= *pline++; |
668 | 0 | if (y > 1) { |
669 | 0 | ppd = (*ppline++ << 8); |
670 | 0 | if (GBW > 8) |
671 | 0 | ppd |= *ppline++; |
672 | 0 | } |
673 | 0 | } |
674 | 0 | for (x = 0; x < GBW; x++) { |
675 | 0 | if (params->USESKIP && jbig2_image_get_pixel(params->SKIP, x, y)) { |
676 | 0 | bit = 0; |
677 | 0 | } else { |
678 | 0 | CONTEXT = out_byte & 0x003; /* First 2 pixels */ |
679 | 0 | CONTEXT |= jbig2_image_get_pixel(image, x + params->gbat[0], y + params->gbat[1]) << 2; |
680 | 0 | CONTEXT |= (pd>>11) & 0x078; /* Next 4 pixels */ |
681 | 0 | CONTEXT |= (ppd>>7) & 0x380; /* Next 3 pixels */ |
682 | 0 | bit = jbig2_arith_decode(ctx, as, &GB_stats[CONTEXT]); |
683 | 0 | if (bit < 0) |
684 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template2 unoptimized"); |
685 | 0 | } |
686 | 0 | pd = pd<<1; |
687 | 0 | ppd = ppd<<1; |
688 | 0 | out_byte = (out_byte<<1) | bit; |
689 | 0 | out_bits_to_go_in_byte--; |
690 | 0 | *d = (uint8_t)out_byte<<out_bits_to_go_in_byte; |
691 | 0 | if (out_bits_to_go_in_byte == 0) { |
692 | 0 | out_bits_to_go_in_byte = 8; |
693 | 0 | d++; |
694 | 0 | if (x+9 < GBW && y > 0) { |
695 | 0 | pd |= *pline++; |
696 | 0 | if (y > 1) |
697 | 0 | ppd |= *ppline++; |
698 | 0 | } |
699 | 0 | } |
700 | 0 | } |
701 | 0 | if (out_bits_to_go_in_byte != 8) |
702 | 0 | *d = (uint8_t)out_byte<<out_bits_to_go_in_byte; |
703 | 0 | } |
704 | | |
705 | 0 | return 0; |
706 | 0 | } |
707 | | |
708 | | static int |
709 | | jbig2_decode_generic_template2(Jbig2Ctx *ctx, |
710 | | Jbig2Segment *segment, |
711 | | const Jbig2GenericRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GB_stats) |
712 | 296 | { |
713 | 296 | const uint32_t GBW = image->width; |
714 | 296 | const uint32_t GBH = image->height; |
715 | 296 | const uint32_t rowstride = image->stride; |
716 | 296 | uint32_t x, y; |
717 | 296 | byte *line2 = NULL; |
718 | 296 | byte *line1 = NULL; |
719 | 296 | byte *gbreg_line = (byte *) image->data; |
720 | | |
721 | | #ifdef OUTPUT_PBM |
722 | | printf("P4\n%d %d\n", GBW, GBH); |
723 | | #endif |
724 | | |
725 | 296 | if (GBW <= 0) |
726 | 0 | return 0; |
727 | | |
728 | 4.44k | for (y = 0; y < GBH; y++) { |
729 | 4.14k | uint32_t CONTEXT; |
730 | 4.14k | uint32_t line_m1; |
731 | 4.14k | uint32_t line_m2; |
732 | 4.14k | uint32_t padded_width = (GBW + 7) & -8; |
733 | | |
734 | 4.14k | line_m1 = line1 ? line1[0] : 0; |
735 | 4.14k | line_m2 = line2 ? line2[0] << 4 : 0; |
736 | 4.14k | CONTEXT = ((line_m1 >> 3) & 0x7c) | ((line_m2 >> 3) & 0x380); |
737 | | |
738 | | /* 6.2.5.7 3d */ |
739 | 12.4k | for (x = 0; x < padded_width; x += 8) { |
740 | 8.28k | byte result = 0; |
741 | 8.28k | int x_minor; |
742 | 8.28k | int minor_width = GBW - x > 8 ? 8 : GBW - x; |
743 | | |
744 | 8.28k | if (line1) |
745 | 7.69k | line_m1 = (line_m1 << 8) | (x + 8 < GBW ? line1[(x >> 3) + 1] : 0); |
746 | | |
747 | 8.28k | if (line2) |
748 | 7.10k | line_m2 = (line_m2 << 8) | (x + 8 < GBW ? line2[(x >> 3) + 1] << 4 : 0); |
749 | | |
750 | | /* This is the speed-critical inner loop. */ |
751 | 55.4k | for (x_minor = 0; x_minor < minor_width; x_minor++) { |
752 | 47.1k | int bit; |
753 | | |
754 | 47.1k | bit = jbig2_arith_decode(ctx, as, &GB_stats[CONTEXT]); |
755 | 47.1k | if (bit < 0) |
756 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template2 optimized"); |
757 | 47.1k | result |= bit << (7 - x_minor); |
758 | 47.1k | CONTEXT = ((CONTEXT & 0x1bd) << 1) | bit | ((line_m1 >> (10 - x_minor)) & 0x4) | ((line_m2 >> (10 - x_minor)) & 0x80); |
759 | 47.1k | } |
760 | 8.28k | gbreg_line[x >> 3] = result; |
761 | 8.28k | } |
762 | | #ifdef OUTPUT_PBM |
763 | | fwrite(gbreg_line, 1, rowstride, stdout); |
764 | | #endif |
765 | 4.14k | line2 = line1; |
766 | 4.14k | line1 = gbreg_line; |
767 | 4.14k | gbreg_line += rowstride; |
768 | 4.14k | } |
769 | | |
770 | 296 | return 0; |
771 | 296 | } |
772 | | |
773 | | static int |
774 | | jbig2_decode_generic_template3(Jbig2Ctx *ctx, |
775 | | Jbig2Segment *segment, |
776 | | const Jbig2GenericRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GB_stats) |
777 | 0 | { |
778 | 0 | const uint32_t GBW = image->width; |
779 | 0 | const uint32_t GBH = image->height; |
780 | 0 | const uint32_t rowstride = image->stride; |
781 | 0 | byte *line1 = NULL; |
782 | 0 | byte *gbreg_line = (byte *) image->data; |
783 | 0 | uint32_t x, y; |
784 | |
|
785 | | #ifdef OUTPUT_PBM |
786 | | printf("P4\n%d %d\n", GBW, GBH); |
787 | | #endif |
788 | |
|
789 | 0 | if (GBW <= 0) |
790 | 0 | return 0; |
791 | | |
792 | 0 | for (y = 0; y < GBH; y++) { |
793 | 0 | uint32_t CONTEXT; |
794 | 0 | uint32_t line_m1; |
795 | 0 | uint32_t padded_width = (GBW + 7) & -8; |
796 | |
|
797 | 0 | line_m1 = line1 ? line1[0] : 0; |
798 | 0 | CONTEXT = (line_m1 >> 1) & 0x3f0; |
799 | | |
800 | | /* 6.2.5.7 3d */ |
801 | 0 | for (x = 0; x < padded_width; x += 8) { |
802 | 0 | byte result = 0; |
803 | 0 | int x_minor; |
804 | 0 | int minor_width = GBW - x > 8 ? 8 : GBW - x; |
805 | |
|
806 | 0 | if (line1) |
807 | 0 | line_m1 = (line_m1 << 8) | (x + 8 < GBW ? line1[(x >> 3) + 1] : 0); |
808 | | |
809 | | /* This is the speed-critical inner loop. */ |
810 | 0 | for (x_minor = 0; x_minor < minor_width; x_minor++) { |
811 | 0 | int bit; |
812 | |
|
813 | 0 | bit = jbig2_arith_decode(ctx, as, &GB_stats[CONTEXT]); |
814 | 0 | if (bit < 0) |
815 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template3 optimized"); |
816 | 0 | result |= bit << (7 - x_minor); |
817 | 0 | CONTEXT = ((CONTEXT & 0x1f7) << 1) | bit | ((line_m1 >> (8 - x_minor)) & 0x10); |
818 | 0 | } |
819 | 0 | gbreg_line[x >> 3] = result; |
820 | 0 | } |
821 | | #ifdef OUTPUT_PBM |
822 | | fwrite(gbreg_line, 1, rowstride, stdout); |
823 | | #endif |
824 | 0 | line1 = gbreg_line; |
825 | 0 | gbreg_line += rowstride; |
826 | 0 | } |
827 | | |
828 | 0 | return 0; |
829 | 0 | } |
830 | | |
831 | | static int |
832 | | jbig2_decode_generic_template3_unopt(Jbig2Ctx *ctx, |
833 | | Jbig2Segment *segment, |
834 | | const Jbig2GenericRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GB_stats) |
835 | 0 | { |
836 | 0 | const uint32_t GBW = image->width; |
837 | 0 | const uint32_t GBH = image->height; |
838 | 0 | uint32_t CONTEXT; |
839 | 0 | uint32_t x, y; |
840 | 0 | int bit; |
841 | |
|
842 | 0 | if (pixel_outside_field(params->gbat[0], params->gbat[1])) |
843 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, |
844 | 0 | "adaptive template pixel is out of field"); |
845 | | |
846 | 0 | for (y = 0; y < GBH; y++) { |
847 | 0 | uint32_t out_byte = 0; |
848 | 0 | int out_bits_to_go_in_byte = 8; |
849 | 0 | uint8_t *d = &image->data[image->stride * y]; |
850 | 0 | uint8_t *pline = &image->data[image->stride * (y-1)]; |
851 | 0 | uint32_t pd = 0; |
852 | 0 | if (y > 0) { |
853 | 0 | pd = (*pline++ << 8); |
854 | 0 | if (GBW > 8) |
855 | 0 | pd |= *pline++; |
856 | 0 | } |
857 | 0 | for (x = 0; x < GBW; x++) { |
858 | 0 | if (params->USESKIP && jbig2_image_get_pixel(params->SKIP, x, y)) { |
859 | 0 | bit = 0; |
860 | 0 | } else { |
861 | 0 | CONTEXT = out_byte & 0x00F; /* First 4 pixels */ |
862 | 0 | CONTEXT |= jbig2_image_get_pixel(image, x + params->gbat[0], y + params->gbat[1]) << 4; |
863 | 0 | CONTEXT |= (pd>>9) & 0x3E0; /* Next 5 pixels */ |
864 | 0 | bit = jbig2_arith_decode(ctx, as, &GB_stats[CONTEXT]); |
865 | 0 | if (bit < 0) |
866 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template3 unoptimized"); |
867 | 0 | } |
868 | 0 | pd = pd<<1; |
869 | 0 | out_byte = (out_byte<<1) | bit; |
870 | 0 | out_bits_to_go_in_byte--; |
871 | 0 | *d = (uint8_t)out_byte<<out_bits_to_go_in_byte; |
872 | 0 | if (out_bits_to_go_in_byte == 0) { |
873 | 0 | out_bits_to_go_in_byte = 8; |
874 | 0 | d++; |
875 | 0 | if (x+9 < GBW && y > 0) |
876 | 0 | pd |= *pline++; |
877 | 0 | } |
878 | 0 | } |
879 | 0 | if (out_bits_to_go_in_byte != 8) |
880 | 0 | *d = (uint8_t)out_byte<<out_bits_to_go_in_byte; |
881 | 0 | } |
882 | 0 | return 0; |
883 | 0 | } |
884 | | |
885 | | static void |
886 | | copy_prev_row(Jbig2Image *image, int row) |
887 | 0 | { |
888 | 0 | if (!row) { |
889 | | /* no previous row */ |
890 | 0 | memset(image->data, 0, image->stride); |
891 | 0 | } else { |
892 | | /* duplicate data from the previous row */ |
893 | 0 | uint8_t *src = image->data + (row - 1) * image->stride; |
894 | |
|
895 | 0 | memcpy(src + image->stride, src, image->stride); |
896 | 0 | } |
897 | 0 | } |
898 | | |
899 | | static int |
900 | | jbig2_decode_generic_template0_TPGDON(Jbig2Ctx *ctx, |
901 | | Jbig2Segment *segment, |
902 | | const Jbig2GenericRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GB_stats) |
903 | 0 | { |
904 | 0 | const uint32_t GBW = image->width; |
905 | 0 | const uint32_t GBH = image->height; |
906 | 0 | uint32_t CONTEXT; |
907 | 0 | uint32_t x, y; |
908 | 0 | int LTP = 0; |
909 | 0 | int gmin, gmax; |
910 | 0 | uint32_t left, right, top; |
911 | |
|
912 | 0 | if (pixel_outside_field(params->gbat[0], params->gbat[1]) || |
913 | 0 | pixel_outside_field(params->gbat[2], params->gbat[3]) || |
914 | 0 | pixel_outside_field(params->gbat[4], params->gbat[5]) || |
915 | 0 | pixel_outside_field(params->gbat[6], params->gbat[7])) |
916 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, |
917 | 0 | "adaptive template pixel is out of field"); |
918 | | |
919 | | /* JBig2 has 'standard' values for gbat (see 6.2.5.4 of the spec). |
920 | | * Have an optimised version for those locations. This greatly |
921 | | * simplifies some of the fetches. It's almost like they thought |
922 | | * it through. */ |
923 | 0 | if (params->gbat[0] == 3 && params->gbat[1] == -1 && |
924 | 0 | params->gbat[2] == -3 && params->gbat[3] == -1 && |
925 | 0 | params->gbat[4] == 2 && params->gbat[5] == -2 && |
926 | 0 | params->gbat[6] == -2 && params->gbat[7] == -2) |
927 | 0 | { |
928 | 0 | for (y = 0; y < GBH; y++) { |
929 | 0 | int bit = jbig2_arith_decode(ctx, as, &GB_stats[0x9B25]); |
930 | 0 | if (bit < 0) |
931 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template0 TPGDON1"); |
932 | 0 | LTP ^= bit; |
933 | 0 | if (!LTP) { |
934 | 0 | uint32_t out_byte = 0; |
935 | 0 | int out_bits_to_go_in_byte = 8; |
936 | 0 | uint8_t *d = &image->data[image->stride * y]; |
937 | 0 | uint8_t *pline = &image->data[image->stride * (y-1)]; |
938 | 0 | uint8_t *ppline = &image->data[image->stride * (y-2)]; |
939 | 0 | uint32_t pd = 0; |
940 | 0 | uint32_t ppd = 0; |
941 | 0 | if (y > 0) { |
942 | 0 | pd = (*pline++ << 8); |
943 | 0 | if (GBW > 8) |
944 | 0 | pd |= *pline++; |
945 | 0 | if (y > 1) { |
946 | 0 | ppd = (*ppline++ << 8); |
947 | 0 | if (GBW > 8) |
948 | 0 | ppd |= *ppline++; |
949 | 0 | } |
950 | 0 | } |
951 | 0 | for (x = 0; x < GBW; x++) { |
952 | 0 | if (params->USESKIP && jbig2_image_get_pixel(params->SKIP, x, y)) { |
953 | 0 | bit = 0; |
954 | 0 | } else { |
955 | 0 | CONTEXT = out_byte & 0x00F; /* First 4 pixels */ |
956 | 0 | CONTEXT |= (pd>>8) & 0x7F0; /* Next 7 pixels */ |
957 | 0 | CONTEXT |= (ppd>>2) & 0xF800; /* Final 5 pixels */ |
958 | 0 | bit = jbig2_arith_decode(ctx, as, &GB_stats[CONTEXT]); |
959 | 0 | if (bit < 0) |
960 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template0 TPGDON2"); |
961 | 0 | } |
962 | 0 | pd = pd<<1; |
963 | 0 | ppd = ppd<<1; |
964 | 0 | out_byte = (out_byte<<1) | bit; |
965 | 0 | out_bits_to_go_in_byte--; |
966 | 0 | if (out_bits_to_go_in_byte == 0) { |
967 | 0 | out_bits_to_go_in_byte = 8; |
968 | 0 | *d++ = (uint8_t)out_byte; |
969 | 0 | if (x+9 < GBW && y > 0) { |
970 | 0 | pd |= *pline++; |
971 | 0 | if (y > 1) |
972 | 0 | ppd |= *ppline++; |
973 | 0 | } |
974 | 0 | } |
975 | 0 | } |
976 | 0 | if (out_bits_to_go_in_byte != 8) |
977 | 0 | *d = (uint8_t)out_byte<<out_bits_to_go_in_byte; |
978 | 0 | } else { |
979 | 0 | copy_prev_row(image, y); |
980 | 0 | } |
981 | 0 | } |
982 | 0 | return 0; |
983 | 0 | } |
984 | | |
985 | | /* We divide the width into 3 regions 0..left...right...GBW, |
986 | | * between left and right, we know that our accesses will never |
987 | | * step outside the image, enabling us to use faster accessors. */ |
988 | 0 | left = 4; |
989 | 0 | right = 2; |
990 | 0 | gmin = gmax = params->gbat[0]; |
991 | 0 | if (params->gbat[2] < gmin) |
992 | 0 | gmin = params->gbat[2]; |
993 | 0 | if (gmax < params->gbat[2]) |
994 | 0 | gmax = params->gbat[2]; |
995 | 0 | if (params->gbat[4] < gmin) |
996 | 0 | gmin = params->gbat[4]; |
997 | 0 | if (gmax < params->gbat[4]) |
998 | 0 | gmax = params->gbat[4]; |
999 | 0 | if (params->gbat[6] < gmin) |
1000 | 0 | gmin = params->gbat[6]; |
1001 | 0 | if (gmax < params->gbat[6]) |
1002 | 0 | gmax = params->gbat[6]; |
1003 | 0 | if ((int)left < -gmin) |
1004 | 0 | left = -gmin; |
1005 | 0 | if ((int)right < gmax) |
1006 | 0 | right = gmax; |
1007 | 0 | if (right > GBW) |
1008 | 0 | right = GBW; |
1009 | 0 | right = GBW - right; |
1010 | | /* So 0 <= x < left or right <= x < GBW needs bounds checking. */ |
1011 | | |
1012 | | /* Now we do the same for the height, but here there is no bottom |
1013 | | * region, as we only ever look up for y. */ |
1014 | 0 | top = 2; |
1015 | 0 | gmin = params->gbat[1]; |
1016 | 0 | if (params->gbat[3] < gmin) |
1017 | 0 | gmin = params->gbat[3]; |
1018 | 0 | if (params->gbat[5] < gmin) |
1019 | 0 | gmin = params->gbat[5]; |
1020 | 0 | if (params->gbat[7] < gmin) |
1021 | 0 | gmin = params->gbat[7]; |
1022 | 0 | if ((int)top < -gmin) |
1023 | 0 | top = -gmin; |
1024 | | /* So 0 <= y < top needs bounds checking. */ |
1025 | |
|
1026 | 0 | for (y = 0; y < GBH; y++) { |
1027 | 0 | int bit = jbig2_arith_decode(ctx, as, &GB_stats[0x9B25]); |
1028 | 0 | if (bit < 0) |
1029 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template0 TPGDON1"); |
1030 | 0 | LTP ^= bit; |
1031 | 0 | if (!LTP) { |
1032 | 0 | uint32_t out_byte = 0; |
1033 | 0 | int out_bits_to_go_in_byte = 8; |
1034 | 0 | uint8_t *d = &image->data[image->stride * y]; |
1035 | 0 | uint8_t *pline = &image->data[image->stride * (y-1)]; |
1036 | 0 | uint8_t *ppline = &image->data[image->stride * (y-2)]; |
1037 | 0 | uint32_t pd = 0; |
1038 | 0 | uint32_t ppd = 0; |
1039 | 0 | if (y > 0) { |
1040 | 0 | pd = (*pline++ << 8); |
1041 | 0 | if (GBW > 8) |
1042 | 0 | pd |= *pline++; |
1043 | 0 | if (y > 1) { |
1044 | 0 | ppd = (*ppline++ << 8); |
1045 | 0 | if (GBW > 8) |
1046 | 0 | ppd |= *ppline++; |
1047 | 0 | } |
1048 | 0 | } |
1049 | 0 | for (x = 0; x < GBW; x++) { |
1050 | 0 | if (params->USESKIP && jbig2_image_get_pixel(params->SKIP, x, y)) { |
1051 | 0 | bit = 0; |
1052 | 0 | } else { |
1053 | 0 | CONTEXT = out_byte & 0x000F; /* First 4 pixels */ |
1054 | 0 | CONTEXT |= (pd>>8) & 0x03E0; /* Skip one, next 5 pixels */ |
1055 | 0 | CONTEXT |= (ppd>>2) & 0x7000; /* Skip 2, next 3 pixels, skip one */ |
1056 | 0 | if (y >= top && x >= left && x < right) |
1057 | 0 | { |
1058 | 0 | CONTEXT |= jbig2_image_get_pixel_fast(image, x + params->gbat[0], y + params->gbat[1]) << 4; |
1059 | 0 | CONTEXT |= jbig2_image_get_pixel_fast(image, x + params->gbat[2], y + params->gbat[3]) << 10; |
1060 | 0 | CONTEXT |= jbig2_image_get_pixel_fast(image, x + params->gbat[4], y + params->gbat[5]) << 11; |
1061 | 0 | CONTEXT |= jbig2_image_get_pixel_fast(image, x + params->gbat[6], y + params->gbat[7]) << 15; |
1062 | 0 | } |
1063 | 0 | else |
1064 | 0 | { |
1065 | 0 | CONTEXT |= jbig2_image_get_pixel(image, x + params->gbat[0], y + params->gbat[1]) << 4; |
1066 | 0 | CONTEXT |= jbig2_image_get_pixel(image, x + params->gbat[2], y + params->gbat[3]) << 10; |
1067 | 0 | CONTEXT |= jbig2_image_get_pixel(image, x + params->gbat[4], y + params->gbat[5]) << 11; |
1068 | 0 | CONTEXT |= jbig2_image_get_pixel(image, x + params->gbat[6], y + params->gbat[7]) << 15; |
1069 | 0 | } |
1070 | 0 | bit = jbig2_arith_decode(ctx, as, &GB_stats[CONTEXT]); |
1071 | 0 | if (bit < 0) |
1072 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template0 TPGDON2"); |
1073 | 0 | } |
1074 | 0 | pd = pd<<1; |
1075 | 0 | ppd = ppd<<1; |
1076 | 0 | out_byte = (out_byte<<1) | bit; |
1077 | 0 | out_bits_to_go_in_byte--; |
1078 | 0 | *d = (uint8_t)out_byte<<out_bits_to_go_in_byte; |
1079 | 0 | if (out_bits_to_go_in_byte == 0) { |
1080 | 0 | out_bits_to_go_in_byte = 8; |
1081 | 0 | d++; |
1082 | 0 | if (x+9 < GBW && y > 0) { |
1083 | 0 | pd |= *pline++; |
1084 | 0 | if (y > 1) |
1085 | 0 | ppd |= *ppline++; |
1086 | 0 | } |
1087 | 0 | } |
1088 | 0 | } |
1089 | 0 | if (out_bits_to_go_in_byte != 8) |
1090 | 0 | *d = (uint8_t)out_byte<<out_bits_to_go_in_byte; |
1091 | 0 | } else { |
1092 | 0 | copy_prev_row(image, y); |
1093 | 0 | } |
1094 | 0 | } |
1095 | | |
1096 | 0 | return 0; |
1097 | 0 | } |
1098 | | |
1099 | | static int |
1100 | | jbig2_decode_generic_template1_TPGDON(Jbig2Ctx *ctx, |
1101 | | Jbig2Segment *segment, |
1102 | | const Jbig2GenericRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GB_stats) |
1103 | 0 | { |
1104 | 0 | const uint32_t GBW = image->width; |
1105 | 0 | const uint32_t GBH = image->height; |
1106 | 0 | uint32_t CONTEXT; |
1107 | 0 | uint32_t x, y; |
1108 | 0 | int LTP = 0; |
1109 | |
|
1110 | 0 | if (pixel_outside_field(params->gbat[0], params->gbat[1])) |
1111 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, |
1112 | 0 | "adaptive template pixel is out of field"); |
1113 | | |
1114 | 0 | for (y = 0; y < GBH; y++) { |
1115 | 0 | int bit = jbig2_arith_decode(ctx, as, &GB_stats[0x0795]); |
1116 | 0 | if (bit < 0) |
1117 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template1 TPGDON1"); |
1118 | 0 | LTP ^= bit; |
1119 | 0 | if (!LTP) { |
1120 | 0 | uint32_t out_byte = 0; |
1121 | 0 | int out_bits_to_go_in_byte = 8; |
1122 | 0 | uint8_t *d = &image->data[image->stride * y]; |
1123 | 0 | uint8_t *pline = &image->data[image->stride * (y-1)]; |
1124 | 0 | uint8_t *ppline = &image->data[image->stride * (y-2)]; |
1125 | 0 | uint32_t pd = 0; |
1126 | 0 | uint32_t ppd = 0; |
1127 | 0 | if (y > 0) { |
1128 | 0 | pd = (*pline++ << 8); |
1129 | 0 | if (GBW > 8) |
1130 | 0 | pd |= *pline++; |
1131 | 0 | if (y > 1) { |
1132 | 0 | ppd = (*ppline++ << 8); |
1133 | 0 | if (GBW > 8) |
1134 | 0 | ppd |= *ppline++; |
1135 | 0 | } |
1136 | 0 | } |
1137 | 0 | for (x = 0; x < GBW; x++) { |
1138 | 0 | if (params->USESKIP && jbig2_image_get_pixel(params->SKIP, x, y)) { |
1139 | 0 | bit = 0; |
1140 | 0 | } else { |
1141 | 0 | CONTEXT = out_byte & 0x0007; /* First 3 pixels */ |
1142 | 0 | CONTEXT |= jbig2_image_get_pixel(image, x + params->gbat[0], y + params->gbat[1]) << 3; |
1143 | 0 | CONTEXT |= (pd>>9) & 0x01F0; /* next 5 pixels */ |
1144 | 0 | CONTEXT |= (ppd>>4) & 0x1E00; /* next 4 pixels */ |
1145 | 0 | bit = jbig2_arith_decode(ctx, as, &GB_stats[CONTEXT]); |
1146 | 0 | if (bit < 0) |
1147 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template1 TPGDON2"); |
1148 | 0 | } |
1149 | 0 | pd = pd<<1; |
1150 | 0 | ppd = ppd<<1; |
1151 | 0 | out_byte = (out_byte<<1) | bit; |
1152 | 0 | out_bits_to_go_in_byte--; |
1153 | 0 | *d = (uint8_t)out_byte<<out_bits_to_go_in_byte; |
1154 | 0 | if (out_bits_to_go_in_byte == 0) { |
1155 | 0 | out_bits_to_go_in_byte = 8; |
1156 | 0 | d++; |
1157 | 0 | if (x+9 < GBW && y > 0) { |
1158 | 0 | pd |= *pline++; |
1159 | 0 | if (y > 1) |
1160 | 0 | ppd |= *ppline++; |
1161 | 0 | } |
1162 | 0 | } |
1163 | 0 | } |
1164 | 0 | if (out_bits_to_go_in_byte != 8) |
1165 | 0 | *d = (uint8_t)out_byte<<out_bits_to_go_in_byte; |
1166 | 0 | } else { |
1167 | 0 | copy_prev_row(image, y); |
1168 | 0 | } |
1169 | 0 | } |
1170 | | |
1171 | 0 | return 0; |
1172 | 0 | } |
1173 | | |
1174 | | static int |
1175 | | jbig2_decode_generic_template2_TPGDON(Jbig2Ctx *ctx, |
1176 | | Jbig2Segment *segment, |
1177 | | const Jbig2GenericRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GB_stats) |
1178 | 0 | { |
1179 | 0 | const uint32_t GBW = image->width; |
1180 | 0 | const uint32_t GBH = image->height; |
1181 | 0 | uint32_t CONTEXT; |
1182 | 0 | uint32_t x, y; |
1183 | 0 | int LTP = 0; |
1184 | |
|
1185 | 0 | if (pixel_outside_field(params->gbat[0], params->gbat[1])) |
1186 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, |
1187 | 0 | "adaptive template pixel is out of field"); |
1188 | | |
1189 | 0 | for (y = 0; y < GBH; y++) { |
1190 | 0 | int bit = jbig2_arith_decode(ctx, as, &GB_stats[0xE5]); |
1191 | 0 | if (bit < 0) |
1192 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template2 TPGDON1"); |
1193 | 0 | LTP ^= bit; |
1194 | 0 | if (!LTP) { |
1195 | 0 | uint32_t out_byte = 0; |
1196 | 0 | int out_bits_to_go_in_byte = 8; |
1197 | 0 | uint8_t *d = &image->data[image->stride * y]; |
1198 | 0 | uint8_t *pline = &image->data[image->stride * (y-1)]; |
1199 | 0 | uint8_t *ppline = &image->data[image->stride * (y-2)]; |
1200 | 0 | uint32_t pd = 0; |
1201 | 0 | uint32_t ppd = 0; |
1202 | 0 | if (y > 0) { |
1203 | 0 | pd = (*pline++ << 8); |
1204 | 0 | if (GBW > 8) |
1205 | 0 | pd |= *pline++; |
1206 | 0 | if (y > 1) { |
1207 | 0 | ppd = (*ppline++ << 8); |
1208 | 0 | if (GBW > 8) |
1209 | 0 | ppd |= *ppline++; |
1210 | 0 | } |
1211 | 0 | } |
1212 | 0 | for (x = 0; x < GBW; x++) { |
1213 | 0 | if (params->USESKIP && jbig2_image_get_pixel(params->SKIP, x, y)) { |
1214 | 0 | bit = 0; |
1215 | 0 | } else { |
1216 | 0 | CONTEXT = out_byte & 0x003; /* First 2 pixels */ |
1217 | 0 | CONTEXT |= jbig2_image_get_pixel(image, x + params->gbat[0], y + params->gbat[1]) << 2; |
1218 | 0 | CONTEXT |= (pd>>11) & 0x078; /* next 4 pixels */ |
1219 | 0 | CONTEXT |= (ppd>>7) & 0x380; /* next 3 pixels */ |
1220 | 0 | bit = jbig2_arith_decode(ctx, as, &GB_stats[CONTEXT]); |
1221 | 0 | if (bit < 0) |
1222 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template2 TPGDON2"); |
1223 | 0 | } |
1224 | 0 | pd = pd<<1; |
1225 | 0 | ppd = ppd<<1; |
1226 | 0 | out_byte = (out_byte<<1) | bit; |
1227 | 0 | out_bits_to_go_in_byte--; |
1228 | 0 | *d = (uint8_t)out_byte<<out_bits_to_go_in_byte; |
1229 | 0 | if (out_bits_to_go_in_byte == 0) { |
1230 | 0 | out_bits_to_go_in_byte = 8; |
1231 | 0 | d++; |
1232 | 0 | if (x+9 < GBW && y > 0) { |
1233 | 0 | pd |= *pline++; |
1234 | 0 | if (y > 1) |
1235 | 0 | ppd |= *ppline++; |
1236 | 0 | } |
1237 | 0 | } |
1238 | 0 | } |
1239 | 0 | if (out_bits_to_go_in_byte != 8) |
1240 | 0 | *d = (uint8_t)out_byte<<out_bits_to_go_in_byte; |
1241 | 0 | } else { |
1242 | 0 | copy_prev_row(image, y); |
1243 | 0 | } |
1244 | 0 | } |
1245 | | |
1246 | 0 | return 0; |
1247 | 0 | } |
1248 | | |
1249 | | static int |
1250 | | jbig2_decode_generic_template3_TPGDON(Jbig2Ctx *ctx, |
1251 | | Jbig2Segment *segment, |
1252 | | const Jbig2GenericRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GB_stats) |
1253 | 0 | { |
1254 | 0 | const uint32_t GBW = image->width; |
1255 | 0 | const uint32_t GBH = image->height; |
1256 | 0 | uint32_t CONTEXT; |
1257 | 0 | uint32_t x, y; |
1258 | 0 | int LTP = 0; |
1259 | |
|
1260 | 0 | if (pixel_outside_field(params->gbat[0], params->gbat[1])) |
1261 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, |
1262 | 0 | "adaptive template pixel is out of field"); |
1263 | | |
1264 | 0 | for (y = 0; y < GBH; y++) { |
1265 | 0 | int bit = jbig2_arith_decode(ctx, as, &GB_stats[0x0195]); |
1266 | 0 | if (bit < 0) |
1267 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template3 TPGDON1"); |
1268 | 0 | LTP ^= bit; |
1269 | 0 | if (!LTP) { |
1270 | 0 | uint32_t out_byte = 0; |
1271 | 0 | int out_bits_to_go_in_byte = 8; |
1272 | 0 | uint8_t *d = &image->data[image->stride * y]; |
1273 | 0 | uint8_t *pline = &image->data[image->stride * (y-1)]; |
1274 | 0 | uint32_t pd = 0; |
1275 | 0 | if (y > 0) { |
1276 | 0 | pd = (*pline++ << 8); |
1277 | 0 | if (GBW > 8) |
1278 | 0 | pd |= *pline++; |
1279 | 0 | } |
1280 | 0 | for (x = 0; x < GBW; x++) { |
1281 | 0 | if (params->USESKIP && jbig2_image_get_pixel(params->SKIP, x, y)) { |
1282 | 0 | bit = 0; |
1283 | 0 | } else { |
1284 | 0 | CONTEXT = out_byte & 0x0F; /* First 4 pixels */ |
1285 | 0 | CONTEXT |= jbig2_image_get_pixel(image, x + params->gbat[0], y + params->gbat[1]) << 4; |
1286 | 0 | CONTEXT |= (pd>>9) & 0x3E0; /* next 5 pixels */ |
1287 | 0 | bit = jbig2_arith_decode(ctx, as, &GB_stats[CONTEXT]); |
1288 | 0 | if (bit < 0) |
1289 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling generic template3 TPGDON2"); |
1290 | 0 | } |
1291 | 0 | pd = pd<<1; |
1292 | 0 | out_byte = (out_byte<<1) | bit; |
1293 | 0 | out_bits_to_go_in_byte--; |
1294 | 0 | *d = (uint8_t)out_byte<<out_bits_to_go_in_byte; |
1295 | 0 | if (out_bits_to_go_in_byte == 0) { |
1296 | 0 | out_bits_to_go_in_byte = 8; |
1297 | 0 | d++; |
1298 | 0 | if (x+9 < GBW && y > 0) |
1299 | 0 | pd |= *pline++; |
1300 | 0 | } |
1301 | 0 | } |
1302 | 0 | if (out_bits_to_go_in_byte != 8) |
1303 | 0 | *d = (uint8_t)out_byte<<out_bits_to_go_in_byte; |
1304 | 0 | } else { |
1305 | 0 | copy_prev_row(image, y); |
1306 | 0 | } |
1307 | 0 | } |
1308 | | |
1309 | 0 | return 0; |
1310 | 0 | } |
1311 | | |
1312 | | static int |
1313 | | jbig2_decode_generic_region_TPGDON(Jbig2Ctx *ctx, |
1314 | | Jbig2Segment *segment, |
1315 | | const Jbig2GenericRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GB_stats) |
1316 | 0 | { |
1317 | 0 | switch (params->GBTEMPLATE) { |
1318 | 0 | case 0: |
1319 | 0 | return jbig2_decode_generic_template0_TPGDON(ctx, segment, params, as, image, GB_stats); |
1320 | 0 | case 1: |
1321 | 0 | return jbig2_decode_generic_template1_TPGDON(ctx, segment, params, as, image, GB_stats); |
1322 | 0 | case 2: |
1323 | 0 | return jbig2_decode_generic_template2_TPGDON(ctx, segment, params, as, image, GB_stats); |
1324 | 0 | case 3: |
1325 | 0 | return jbig2_decode_generic_template3_TPGDON(ctx, segment, params, as, image, GB_stats); |
1326 | 0 | } |
1327 | | |
1328 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "unsupported GBTEMPLATE (%d)", params->GBTEMPLATE); |
1329 | 0 | } |
1330 | | |
1331 | | /** |
1332 | | * jbig2_decode_generic_region: Decode a generic region. |
1333 | | * @ctx: The context for allocation and error reporting. |
1334 | | * @segment: A segment reference for error reporting. |
1335 | | * @params: Decoding parameter set. |
1336 | | * @as: Arithmetic decoder state. |
1337 | | * @image: Where to store the decoded data. |
1338 | | * @GB_stats: Arithmetic stats. |
1339 | | * |
1340 | | * Decodes a generic region, according to section 6.2. The caller should |
1341 | | * pass an already allocated Jbig2Image object for @image |
1342 | | * |
1343 | | * Because this API is based on an arithmetic decoding state, it is |
1344 | | * not suitable for MMR decoding. |
1345 | | * |
1346 | | * Return code: 0 on success. |
1347 | | **/ |
1348 | | int |
1349 | | jbig2_decode_generic_region(Jbig2Ctx *ctx, |
1350 | | Jbig2Segment *segment, const Jbig2GenericRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GB_stats) |
1351 | 832 | { |
1352 | 832 | const int8_t *gbat = params->gbat; |
1353 | | |
1354 | 832 | if (!params->MMR && params->TPGDON) |
1355 | 0 | return jbig2_decode_generic_region_TPGDON(ctx, segment, params, as, image, GB_stats); |
1356 | | |
1357 | 832 | if (!params->MMR && params->GBTEMPLATE == 0) { |
1358 | 536 | if (!params->USESKIP && gbat[0] == +3 && gbat[1] == -1 && gbat[2] == -3 && gbat[3] == -1 && gbat[4] == +2 && gbat[5] == -2 && gbat[6] == -2 && gbat[7] == -2) |
1359 | 519 | return jbig2_decode_generic_template0(ctx, segment, params, as, image, GB_stats); |
1360 | 17 | else |
1361 | 17 | return jbig2_decode_generic_template0_unopt(ctx, segment, params, as, image, GB_stats); |
1362 | 536 | } else if (!params->MMR && params->GBTEMPLATE == 1) { |
1363 | 0 | if (!params->USESKIP && gbat[0] == +3 && gbat[1] == -1) |
1364 | 0 | return jbig2_decode_generic_template1(ctx, segment, params, as, image, GB_stats); |
1365 | 0 | else |
1366 | 0 | return jbig2_decode_generic_template1_unopt(ctx, segment, params, as, image, GB_stats); |
1367 | 0 | } |
1368 | 296 | else if (!params->MMR && params->GBTEMPLATE == 2) { |
1369 | 296 | if (!params->USESKIP && gbat[0] == 2 && gbat[1] == -1) |
1370 | 296 | return jbig2_decode_generic_template2(ctx, segment, params, as, image, GB_stats); |
1371 | 0 | else |
1372 | 0 | return jbig2_decode_generic_template2_unopt(ctx, segment, params, as, image, GB_stats); |
1373 | 296 | } else if (!params->MMR && params->GBTEMPLATE == 3) { |
1374 | 0 | if (!params->USESKIP && gbat[0] == 2 && gbat[1] == -1) |
1375 | 0 | return jbig2_decode_generic_template3(ctx, segment, params, as, image, GB_stats); |
1376 | 0 | else |
1377 | 0 | return jbig2_decode_generic_template3_unopt(ctx, segment, params, as, image, GB_stats); |
1378 | 0 | } |
1379 | | |
1380 | 0 | { |
1381 | 0 | int i; |
1382 | |
|
1383 | 0 | for (i = 0; i < 8; i++) |
1384 | 0 | jbig2_error(ctx, JBIG2_SEVERITY_DEBUG, segment->number, "gbat[%d] = %d", i, params->gbat[i]); |
1385 | 0 | } |
1386 | |
|
1387 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "unsupported generic region (MMR=%d, GBTEMPLATE=%d)", params->MMR, params->GBTEMPLATE); |
1388 | 832 | } |
1389 | | |
1390 | | /** |
1391 | | * Handler for immediate generic region segments |
1392 | | */ |
1393 | | int |
1394 | | jbig2_immediate_generic_region(Jbig2Ctx *ctx, Jbig2Segment *segment, const byte *segment_data) |
1395 | 451 | { |
1396 | 451 | Jbig2RegionSegmentInfo rsi; |
1397 | 451 | byte seg_flags; |
1398 | 451 | int8_t gbat[8]; |
1399 | 451 | int offset; |
1400 | 451 | uint32_t gbat_bytes = 0; |
1401 | 451 | Jbig2GenericRegionParams params; |
1402 | 451 | int code = 0; |
1403 | 451 | Jbig2Image *image = NULL; |
1404 | 451 | Jbig2WordStream *ws = NULL; |
1405 | 451 | Jbig2ArithState *as = NULL; |
1406 | 451 | Jbig2ArithCx *GB_stats = NULL; |
1407 | 451 | uint32_t height; |
1408 | 451 | Jbig2Page *page = &ctx->pages[ctx->current_page]; |
1409 | | |
1410 | | /* 7.4.6 */ |
1411 | 451 | if (segment->data_length < 18) |
1412 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "segment too short"); |
1413 | | |
1414 | 451 | jbig2_get_region_segment_info(&rsi, segment_data); |
1415 | 451 | jbig2_error(ctx, JBIG2_SEVERITY_INFO, segment->number, "generic region: %u x %u @ (%u, %u), flags = %02x", rsi.width, rsi.height, rsi.x, rsi.y, rsi.flags); |
1416 | | |
1417 | | /* 7.4.6.4 */ |
1418 | 451 | height = rsi.height; |
1419 | 451 | if (segment->rows != UINT32_MAX) { |
1420 | 0 | if (segment->rows > rsi.height) |
1421 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "segment contains more rows than stated in header"); |
1422 | 0 | height = segment->rows; |
1423 | 0 | } |
1424 | | |
1425 | | /* 7.4.6.2 */ |
1426 | 451 | seg_flags = segment_data[17]; |
1427 | 451 | jbig2_error(ctx, JBIG2_SEVERITY_INFO, segment->number, "segment flags = %02x", seg_flags); |
1428 | 451 | if ((seg_flags & 1) && (seg_flags & 6)) |
1429 | 0 | jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "MMR is 1, but GBTEMPLATE is not 0"); |
1430 | | |
1431 | | /* 7.4.6.3 */ |
1432 | 451 | if (!(seg_flags & 1)) { |
1433 | 451 | gbat_bytes = (seg_flags & 6) ? 2 : 8; |
1434 | 451 | if (18 + gbat_bytes > segment->data_length) |
1435 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "segment too short"); |
1436 | 451 | memcpy(gbat, segment_data + 18, gbat_bytes); |
1437 | 451 | jbig2_error(ctx, JBIG2_SEVERITY_INFO, segment->number, "gbat: %d, %d", gbat[0], gbat[1]); |
1438 | 451 | } |
1439 | | |
1440 | 451 | offset = 18 + gbat_bytes; |
1441 | | |
1442 | | /* Check for T.88 amendment 2 */ |
1443 | 451 | if ((seg_flags >> 5) & 1) |
1444 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "segment uses 12 adaptive template pixels (NYI)"); |
1445 | | |
1446 | | /* Table 34 */ |
1447 | 451 | params.MMR = seg_flags & 1; |
1448 | 451 | params.GBTEMPLATE = (seg_flags & 6) >> 1; |
1449 | 451 | params.TPGDON = (seg_flags & 8) >> 3; |
1450 | 451 | params.USESKIP = 0; |
1451 | 451 | memcpy(params.gbat, gbat, gbat_bytes); |
1452 | | |
1453 | 451 | if (page->height == 0xffffffff && page->striped && page->stripe_size > 0) { |
1454 | 0 | if (rsi.y >= page->end_row + page->stripe_size) { |
1455 | 0 | jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "ignoring %u x %u region at (%u, %u) outside of stripe at row %u covering %u rows, on page of height %u", rsi.width, rsi.height, rsi.x, rsi.y, page->end_row, page->stripe_size, page->image->height); |
1456 | 0 | return 0; |
1457 | 0 | } |
1458 | 0 | if (height > page->end_row + page->stripe_size) { |
1459 | 0 | height = page->end_row + page->stripe_size; |
1460 | 0 | } |
1461 | 451 | } else { |
1462 | 451 | if (rsi.y >= page->height) { |
1463 | 0 | jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "ignoring %u x %u region at (%u, %u) outside of page of height %u", rsi.width, rsi.height, rsi.x, rsi.y, page->height); |
1464 | 0 | return 0; |
1465 | 0 | } |
1466 | 451 | if (height > page->height - rsi .y) { |
1467 | 0 | height = page->height - rsi.y; |
1468 | 0 | } |
1469 | 451 | } |
1470 | 451 | if (height == 0) { |
1471 | 0 | jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "nothing remains of region, ignoring"); |
1472 | 0 | return 0; |
1473 | 0 | } |
1474 | | |
1475 | 451 | image = jbig2_image_new(ctx, rsi.width, height); |
1476 | 451 | if (image == NULL) |
1477 | 0 | return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to allocate generic image"); |
1478 | 451 | jbig2_error(ctx, JBIG2_SEVERITY_DEBUG, segment->number, "allocated %d x %d image buffer for region decode results", rsi.width, height); |
1479 | | |
1480 | 451 | if (params.MMR) { |
1481 | 0 | code = jbig2_decode_generic_mmr(ctx, segment, ¶ms, segment_data + offset, segment->data_length - offset, image); |
1482 | 0 | if (code < 0) { |
1483 | 0 | code = jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode MMR-coded generic region"); |
1484 | 0 | goto cleanup; |
1485 | 0 | } |
1486 | 451 | } else { |
1487 | 451 | int stats_size = jbig2_generic_stats_size(ctx, params.GBTEMPLATE); |
1488 | | |
1489 | 451 | GB_stats = jbig2_new(ctx, Jbig2ArithCx, stats_size); |
1490 | 451 | if (GB_stats == NULL) { |
1491 | 0 | code = jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "failed to allocate arithmetic decoder states when handling immediate generic region"); |
1492 | 0 | goto cleanup; |
1493 | 0 | } |
1494 | 451 | memset(GB_stats, 0, stats_size); |
1495 | | |
1496 | 451 | ws = jbig2_word_stream_buf_new(ctx, segment_data + offset, segment->data_length - offset); |
1497 | 451 | if (ws == NULL) { |
1498 | 0 | code = jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to allocated word stream when handling immediate generic region"); |
1499 | 0 | goto cleanup; |
1500 | 0 | } |
1501 | 451 | as = jbig2_arith_new(ctx, ws); |
1502 | 451 | if (as == NULL) { |
1503 | 0 | code = jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to allocate arithmetic coding state when handling immediate generic region"); |
1504 | 0 | goto cleanup; |
1505 | 0 | } |
1506 | 451 | code = jbig2_decode_generic_region(ctx, segment, ¶ms, as, image, GB_stats); |
1507 | 451 | if (code < 0) { |
1508 | 0 | code = jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode immediate generic region"); |
1509 | 0 | goto cleanup; |
1510 | 0 | } |
1511 | 451 | } |
1512 | | |
1513 | 451 | code = jbig2_page_add_result(ctx, &ctx->pages[ctx->current_page], image, rsi.x, rsi.y, rsi.op); |
1514 | 451 | if (code < 0) |
1515 | 0 | code = jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "unable to add result to page"); |
1516 | | |
1517 | 451 | cleanup: |
1518 | 451 | jbig2_free(ctx->allocator, as); |
1519 | 451 | jbig2_word_stream_buf_free(ctx, ws); |
1520 | 451 | jbig2_free(ctx->allocator, GB_stats); |
1521 | 451 | jbig2_image_release(ctx, image); |
1522 | | |
1523 | 451 | return code; |
1524 | 451 | } |