Coverage Report

Created: 2025-12-19 06:58

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/jbig2dec/jbig2_halftone.c
Line
Count
Source
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
/* JBIG2 Pattern Dictionary and Halftone Region decoding */
21
22
#ifdef HAVE_CONFIG_H
23
#include "config.h"
24
#endif
25
#include "os_types.h"
26
27
#include <string.h>             /* memset() */
28
29
#include "jbig2.h"
30
#include "jbig2_priv.h"
31
#include "jbig2_arith.h"
32
#include "jbig2_generic.h"
33
#include "jbig2_image.h"
34
#include "jbig2_halftone.h"
35
#include "jbig2_mmr.h"
36
#include "jbig2_page.h"
37
#include "jbig2_segment.h"
38
39
/**
40
 * jbig2_hd_new: create a new dictionary from a collective bitmap
41
 */
42
static Jbig2PatternDict *
43
jbig2_hd_new(Jbig2Ctx *ctx, const Jbig2PatternDictParams *params, Jbig2Image *image)
44
4.07k
{
45
4.07k
    Jbig2PatternDict *new;
46
4.07k
    const uint32_t N = params->GRAYMAX + 1;
47
4.07k
    const uint32_t HPW = params->HDPW;
48
4.07k
    const uint32_t HPH = params->HDPH;
49
4.07k
    int code;
50
4.07k
    uint32_t i, j;
51
52
4.07k
    if (N == 0) {
53
        /* We've wrapped. */
54
0
        jbig2_error(ctx, JBIG2_SEVERITY_WARNING, JBIG2_UNKNOWN_SEGMENT_NUMBER, "params->GRAYMAX out of range");
55
0
        return NULL;
56
0
    }
57
58
    /* allocate a new struct */
59
4.07k
    new = jbig2_new(ctx, Jbig2PatternDict, 1);
60
4.07k
    if (new != NULL) {
61
4.07k
        new->patterns = jbig2_new(ctx, Jbig2Image *, N);
62
4.07k
        if (new->patterns == NULL) {
63
987
            jbig2_error(ctx, JBIG2_SEVERITY_FATAL, JBIG2_UNKNOWN_SEGMENT_NUMBER, "failed to allocate pattern in collective bitmap dictionary");
64
987
            jbig2_free(ctx->allocator, new);
65
987
            return NULL;
66
987
        }
67
3.08k
        new->n_patterns = N;
68
3.08k
        new->HPW = HPW;
69
3.08k
        new->HPH = HPH;
70
71
        /* 6.7.5(4) - copy out the individual pattern images */
72
24.9M
        for (i = 0; i < N; i++) {
73
24.9M
            new->patterns[i] = jbig2_image_new(ctx, HPW, HPH);
74
24.9M
            if (new->patterns[i] == NULL) {
75
148
                jbig2_error(ctx, JBIG2_SEVERITY_WARNING, JBIG2_UNKNOWN_SEGMENT_NUMBER, "failed to allocate pattern element image");
76
                /* new->patterns[i] above did not succeed, so releasing patterns 0..i-1 is enough */
77
10.5M
                for (j = 0; j < i; j++)
78
10.5M
                    jbig2_image_release(ctx, new->patterns[j]);
79
148
                jbig2_free(ctx->allocator, new->patterns);
80
148
                jbig2_free(ctx->allocator, new);
81
148
                return NULL;
82
148
            }
83
            /* compose with the REPLACE operator; the source
84
               will be clipped to the destination, selecting the
85
               proper sub image */
86
24.9M
            code = jbig2_image_compose(ctx, new->patterns[i], image, -i * (int32_t) HPW, 0, JBIG2_COMPOSE_REPLACE);
87
24.9M
            if (code < 0) {
88
0
                jbig2_error(ctx, JBIG2_SEVERITY_WARNING, JBIG2_UNKNOWN_SEGMENT_NUMBER, "failed to compose image into collective bitmap dictionary");
89
                /* new->patterns[i] above succeeded, so release all patterns 0..i */
90
0
                for (j = 0; j <= i; j++)
91
0
                    jbig2_image_release(ctx, new->patterns[j]);
92
0
                jbig2_free(ctx->allocator, new->patterns);
93
0
                jbig2_free(ctx->allocator, new);
94
0
                return NULL;
95
0
            }
96
24.9M
        }
97
3.08k
    } else {
98
2
        jbig2_error(ctx, JBIG2_SEVERITY_FATAL, JBIG2_UNKNOWN_SEGMENT_NUMBER, "failed to allocate collective bitmap dictionary");
99
2
    }
100
101
2.94k
    return new;
102
4.07k
}
103
104
/**
105
 * jbig2_hd_release: release a pattern dictionary
106
 */
107
void
108
jbig2_hd_release(Jbig2Ctx *ctx, Jbig2PatternDict *dict)
109
2.94k
{
110
2.94k
    int i;
111
112
2.94k
    if (dict == NULL)
113
0
        return;
114
2.94k
    if (dict->patterns != NULL)
115
14.4M
        for (i = 0; i < dict->n_patterns; i++)
116
14.4M
            jbig2_image_release(ctx, dict->patterns[i]);
117
2.94k
    jbig2_free(ctx->allocator, dict->patterns);
118
2.94k
    jbig2_free(ctx->allocator, dict);
119
2.94k
}
120
121
/**
122
 * jbig2_decode_pattern_dict: decode pattern dictionary data
123
 *
124
 * @ctx: jbig2 decoder context
125
 * @segment: jbig2 segment (header) structure
126
 * @params: parameters from the pattern dictionary header
127
 * @data: pointer to text region data to be decoded
128
 * @size: length of text region data
129
 * @GB_stats: arithmetic coding context to use
130
 *
131
 * Implements the pattern dictionary decoding procedure
132
 * described in section 6.7 of the JBIG2 spec.
133
 *
134
 * returns: a pointer to the resulting dictionary on success
135
 * returns: 0 on failure
136
 **/
137
static Jbig2PatternDict *
138
jbig2_decode_pattern_dict(Jbig2Ctx *ctx, Jbig2Segment *segment,
139
                          const Jbig2PatternDictParams *params, const byte *data, const size_t size, Jbig2ArithCx *GB_stats)
140
4.31k
{
141
4.31k
    Jbig2PatternDict *hd = NULL;
142
4.31k
    Jbig2Image *image = NULL;
143
4.31k
    Jbig2GenericRegionParams rparams;
144
4.31k
    int code = 0;
145
146
    /* allocate the collective image */
147
4.31k
    image = jbig2_image_new(ctx, params->HDPW * (params->GRAYMAX + 1), params->HDPH);
148
4.31k
    if (image == NULL) {
149
61
        jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to allocate collective bitmap for halftone dictionary");
150
61
        return NULL;
151
61
    }
152
153
    /* fill out the generic region decoder parameters */
154
4.25k
    rparams.MMR = params->HDMMR;
155
4.25k
    rparams.GBTEMPLATE = params->HDTEMPLATE;
156
4.25k
    rparams.TPGDON = 0;         /* not used if HDMMR = 1 */
157
4.25k
    rparams.USESKIP = 0;
158
4.25k
    rparams.gbat[0] = -(int8_t) params->HDPW;
159
4.25k
    rparams.gbat[1] = 0;
160
4.25k
    rparams.gbat[2] = -3;
161
4.25k
    rparams.gbat[3] = -1;
162
4.25k
    rparams.gbat[4] = 2;
163
4.25k
    rparams.gbat[5] = -2;
164
4.25k
    rparams.gbat[6] = -2;
165
4.25k
    rparams.gbat[7] = -2;
166
167
4.25k
    if (params->HDMMR) {
168
3.85k
        code = jbig2_decode_generic_mmr(ctx, segment, &rparams, data, size, image);
169
3.85k
    } else {
170
404
        Jbig2WordStream *ws = jbig2_word_stream_buf_new(ctx, data, size);
171
172
404
        if (ws != NULL) {
173
403
            Jbig2ArithState *as = jbig2_arith_new(ctx, ws);
174
175
403
            if (as != NULL) {
176
401
                code = jbig2_decode_generic_region(ctx, segment, &rparams, as, image, GB_stats);
177
401
            } else {
178
2
                code = jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to allocate arithmetic coding state when handling halftone dictionary");
179
2
            }
180
181
403
            jbig2_free(ctx->allocator, as);
182
403
            jbig2_word_stream_buf_free(ctx, ws);
183
403
        } else {
184
1
            code = jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to allocate word stream when handling halftone dictionary");
185
1
        }
186
404
    }
187
188
4.25k
    if (code == 0)
189
4.07k
        hd = jbig2_hd_new(ctx, params, image);
190
181
    else
191
181
        jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode immediate generic region");
192
4.25k
    jbig2_image_release(ctx, image);
193
194
4.25k
    return hd;
195
4.31k
}
196
197
/* 7.4.4 */
198
int
199
jbig2_pattern_dictionary(Jbig2Ctx *ctx, Jbig2Segment *segment, const byte *segment_data)
200
4.33k
{
201
4.33k
    Jbig2PatternDictParams params;
202
4.33k
    Jbig2ArithCx *GB_stats = NULL;
203
4.33k
    byte flags;
204
4.33k
    int offset = 0;
205
206
    /* 7.4.4.1 - Data header */
207
4.33k
    if (segment->data_length < 7) {
208
17
        return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "segment too short");
209
17
    }
210
4.32k
    flags = segment_data[0];
211
4.32k
    params.HDMMR = flags & 1;
212
4.32k
    params.HDTEMPLATE = (flags & 6) >> 1;
213
4.32k
    params.HDPW = segment_data[1];
214
4.32k
    params.HDPH = segment_data[2];
215
4.32k
    params.GRAYMAX = jbig2_get_uint32(segment_data + 3);
216
4.32k
    offset += 7;
217
218
4.32k
    jbig2_error(ctx, JBIG2_SEVERITY_INFO, segment->number,
219
4.32k
                "pattern dictionary, flags=%02x, %d grays (%dx%d cell)", flags, params.GRAYMAX + 1, params.HDPW, params.HDPH);
220
221
4.32k
    if (params.HDMMR && params.HDTEMPLATE) {
222
3.07k
        jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "HDTEMPLATE is %d when HDMMR is %d, contrary to spec", params.HDTEMPLATE, params.HDMMR);
223
3.07k
    }
224
4.32k
    if (flags & 0xf8) {
225
3.58k
        jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "reserved flag bits non-zero");
226
3.58k
    }
227
228
    /* 7.4.4.2 */
229
4.32k
    if (!params.HDMMR) {
230
        /* allocate and zero arithmetic coding stats */
231
422
        int stats_size = jbig2_generic_stats_size(ctx, params.HDTEMPLATE);
232
233
422
        GB_stats = jbig2_new(ctx, Jbig2ArithCx, stats_size);
234
422
        if (GB_stats == NULL)
235
1
            return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to allocate arithmetic coding state when handling pattern dictionary");
236
421
        memset(GB_stats, 0, stats_size);
237
421
    }
238
239
4.31k
    segment->result = jbig2_decode_pattern_dict(ctx, segment, &params, segment_data + offset, segment->data_length - offset, GB_stats);
240
241
    /* todo: retain GB_stats? */
242
4.31k
    if (!params.HDMMR) {
243
421
        jbig2_free(ctx->allocator, GB_stats);
244
421
    }
245
246
4.31k
    return (segment->result != NULL) ? 0 : -1;
247
4.32k
}
248
249
/**
250
 * jbig2_decode_gray_scale_image: decode gray-scale image
251
 *
252
 * @ctx: jbig2 decoder context
253
 * @segment: jbig2 segment (header) structure
254
 * @data: pointer to text region data to be decoded
255
 * @size: length of text region data
256
 * @GSMMR: if MMR is used
257
 * @GSW: width of gray-scale image
258
 * @GSH: height of gray-scale image
259
 * @GSBPP: number of bitplanes/Jbig2Images to use
260
 * @GSKIP: mask indicating which values should be skipped
261
 * @GSTEMPLATE: template used to code the gray-scale bitplanes
262
 * @GB_stats: arithmetic coding context to use
263
 *
264
 * Implements the decoding a gray-scale image described in
265
 * annex C.5. This is part of the halftone region decoding.
266
 *
267
 * returns: array of gray-scale values with GSW x GSH width/height
268
 *          0 on failure
269
 **/
270
static uint16_t **
271
jbig2_decode_gray_scale_image(Jbig2Ctx *ctx, Jbig2Segment *segment,
272
                              const byte *data, const size_t size,
273
                              bool GSMMR, uint32_t GSW, uint32_t GSH,
274
                              uint32_t GSBPP, bool GSUSESKIP, Jbig2Image *GSKIP, int GSTEMPLATE, Jbig2ArithCx *GB_stats)
275
2.09k
{
276
2.09k
    uint16_t **GSVALS = NULL;
277
2.09k
    size_t consumed_bytes = 0;
278
2.09k
    uint32_t i, j, stride, x, y;
279
2.09k
    int code;
280
2.09k
    Jbig2Image **GSPLANES;
281
2.09k
    Jbig2GenericRegionParams rparams;
282
2.09k
    Jbig2WordStream *ws = NULL;
283
2.09k
    Jbig2ArithState *as = NULL;
284
285
    /* allocate GSPLANES */
286
2.09k
    GSPLANES = jbig2_new(ctx, Jbig2Image *, GSBPP);
287
2.09k
    if (GSPLANES == NULL) {
288
3
        jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "failed to allocate %d bytes for GSPLANES", GSBPP);
289
3
        return NULL;
290
3
    }
291
292
20.7k
    for (i = 0; i < GSBPP; ++i) {
293
18.6k
        GSPLANES[i] = jbig2_image_new(ctx, GSW, GSH);
294
18.6k
        if (GSPLANES[i] == NULL) {
295
20
            jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to allocate %dx%d image for GSPLANES", GSW, GSH);
296
            /* free already allocated */
297
106
            for (j = i; j > 0;)
298
86
                jbig2_image_release(ctx, GSPLANES[--j]);
299
20
            jbig2_free(ctx->allocator, GSPLANES);
300
20
            return NULL;
301
20
        }
302
18.6k
    }
303
304
    /* C.5 step 1. Decode GSPLANES[GSBPP-1] */
305
    /* fill generic region decoder parameters */
306
2.07k
    rparams.MMR = GSMMR;
307
2.07k
    rparams.GBTEMPLATE = GSTEMPLATE;
308
2.07k
    rparams.TPGDON = 0;
309
2.07k
    rparams.USESKIP = GSUSESKIP;
310
2.07k
    rparams.SKIP = GSKIP;
311
2.07k
    rparams.gbat[0] = (GSTEMPLATE <= 1 ? 3 : 2);
312
2.07k
    rparams.gbat[1] = -1;
313
2.07k
    rparams.gbat[2] = -3;
314
2.07k
    rparams.gbat[3] = -1;
315
2.07k
    rparams.gbat[4] = 2;
316
2.07k
    rparams.gbat[5] = -2;
317
2.07k
    rparams.gbat[6] = -2;
318
2.07k
    rparams.gbat[7] = -2;
319
320
2.07k
    if (GSMMR) {
321
1.81k
        code = jbig2_decode_halftone_mmr(ctx, &rparams, data, size, GSPLANES[GSBPP - 1], &consumed_bytes);
322
1.81k
    } else {
323
261
        ws = jbig2_word_stream_buf_new(ctx, data, size);
324
261
        if (ws == NULL) {
325
1
            jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to allocate word stream when decoding gray scale image");
326
1
            goto cleanup;
327
1
        }
328
329
260
        as = jbig2_arith_new(ctx, ws);
330
260
        if (as == NULL) {
331
5
            jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to allocate arithmetic coding state when decoding gray scale image");
332
5
            goto cleanup;
333
5
        }
334
335
255
        code = jbig2_decode_generic_region(ctx, segment, &rparams, as, GSPLANES[GSBPP - 1], GB_stats);
336
255
    }
337
2.07k
    if (code < 0) {
338
3
        jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "error decoding GSPLANES for halftone image");
339
3
        goto cleanup;
340
3
    }
341
342
    /* C.5 step 2. Set j = GSBPP-2 */
343
2.06k
    j = GSBPP - 1;
344
    /* C.5 step 3. decode loop */
345
18.4k
    while (j > 0) {
346
16.4k
        j--;
347
        /*  C.5 step 3. (a) */
348
16.4k
        if (GSMMR) {
349
14.6k
            code = jbig2_decode_halftone_mmr(ctx, &rparams, data + consumed_bytes, size - consumed_bytes, GSPLANES[j], &consumed_bytes);
350
14.6k
        } else {
351
1.77k
            code = jbig2_decode_generic_region(ctx, segment, &rparams, as, GSPLANES[j], GB_stats);
352
1.77k
        }
353
16.4k
        if (code < 0) {
354
13
            jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode GSPLANES for halftone image");
355
13
            goto cleanup;
356
13
        }
357
358
        /* C.5 step 3. (b):
359
         * for each [x,y]
360
         * GSPLANES[j][x][y] = GSPLANES[j+1][x][y] XOR GSPLANES[j][x][y] */
361
16.4k
        stride = GSPLANES[j]->stride;
362
365M
        for (i = 0; i < stride * GSH; ++i)
363
365M
            GSPLANES[j]->data[i] ^= GSPLANES[j + 1]->data[i];
364
365
        /*  C.5 step 3. (c) */
366
16.4k
    }
367
368
    /* allocate GSVALS */
369
2.05k
    GSVALS = jbig2_new(ctx, uint16_t *, GSW);
370
2.05k
    if (GSVALS == NULL) {
371
11
        jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "failed to allocate GSVALS: %d bytes", GSW);
372
11
        goto cleanup;
373
11
    }
374
5.31M
    for (i = 0; i < GSW; ++i) {
375
5.31M
        GSVALS[i] = jbig2_new(ctx, uint16_t, GSH);
376
5.31M
        if (GSVALS[i] == NULL) {
377
35
            jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "failed to allocate GSVALS: %d bytes", GSH * GSW);
378
            /* free already allocated */
379
2.34M
            for (j = i; j > 0;)
380
2.34M
                jbig2_free(ctx->allocator, GSVALS[--j]);
381
35
            jbig2_free(ctx->allocator, GSVALS);
382
35
            GSVALS = NULL;
383
35
            goto cleanup;
384
35
        }
385
5.31M
    }
386
387
    /*  C.5 step 4.  */
388
2.96M
    for (x = 0; x < GSW; ++x) {
389
148M
        for (y = 0; y < GSH; ++y) {
390
145M
            GSVALS[x][y] = 0;
391
392
766M
            for (j = 0; j < GSBPP; ++j)
393
621M
                GSVALS[x][y] += jbig2_image_get_pixel(GSPLANES[j], x, y) << j;
394
145M
        }
395
2.96M
    }
396
397
2.07k
cleanup:
398
    /* free memory */
399
2.07k
    if (!GSMMR) {
400
261
        jbig2_free(ctx->allocator, as);
401
261
        jbig2_word_stream_buf_free(ctx, ws);
402
261
    }
403
20.6k
    for (i = 0; i < GSBPP; ++i)
404
18.5k
        jbig2_image_release(ctx, GSPLANES[i]);
405
406
2.07k
    jbig2_free(ctx->allocator, GSPLANES);
407
408
2.07k
    return GSVALS;
409
2.00k
}
410
411
/**
412
 * jbig2_decode_ht_region_get_hpats: get pattern dictionary
413
 *
414
 * @ctx: jbig2 decoder context
415
 * @segment: jbig2 halftone region segment
416
 *
417
 * Returns the first referred pattern dictionary of segment
418
 *
419
 * returns: pattern dictionary
420
 *          0 if search failed
421
 **/
422
static Jbig2PatternDict *
423
jbig2_decode_ht_region_get_hpats(Jbig2Ctx *ctx, Jbig2Segment *segment)
424
2.12k
{
425
2.12k
    int index = 0;
426
2.12k
    Jbig2PatternDict *pattern_dict = NULL;
427
2.12k
    Jbig2Segment *rsegment = NULL;
428
429
    /* loop through all referred segments */
430
2.80k
    while (!pattern_dict && segment->referred_to_segment_count > index) {
431
2.79k
        rsegment = jbig2_find_segment(ctx, segment->referred_to_segments[index]);
432
2.79k
        if (rsegment) {
433
            /* segment type is pattern dictionary and result is not empty */
434
2.44k
            if ((rsegment->flags & 0x3f) == 16 && rsegment->result) {
435
2.10k
                pattern_dict = (Jbig2PatternDict *) rsegment->result;
436
2.10k
                return pattern_dict;
437
2.10k
            }
438
2.44k
        }
439
687
        index++;
440
687
    }
441
17
    return pattern_dict;
442
2.12k
}
443
444
/**
445
 * jbig2_decode_halftone_region: decode a halftone region
446
 *
447
 * @ctx: jbig2 decoder context
448
 * @segment: jbig2 halftone region segment
449
 * @params: parameters
450
 * @data: pointer to halftone region data to be decoded
451
 * @size: length of halftone region data
452
 * @GB_stats: arithmetic coding context to use
453
 *
454
 * Implements the halftone region decoding procedure
455
 * described in section 6.6.5 of the JBIG2 spec.
456
 *
457
 * returns: 0 on success
458
 *         <0 on failure
459
 **/
460
static int
461
jbig2_decode_halftone_region(Jbig2Ctx *ctx, Jbig2Segment *segment,
462
                             Jbig2HalftoneRegionParams *params, const byte *data, const size_t size, Jbig2Image *image, Jbig2ArithCx *GB_stats)
463
2.12k
{
464
2.12k
    uint32_t HBPP;
465
2.12k
    uint32_t HNUMPATS;
466
2.12k
    uint16_t **GI = NULL;
467
2.12k
    Jbig2Image *HSKIP = NULL;
468
2.12k
    Jbig2PatternDict *HPATS;
469
2.12k
    uint32_t i;
470
2.12k
    uint32_t mg, ng;
471
2.12k
    uint16_t gray_val;
472
2.12k
    int code = 0;
473
474
    /* We need the patterns used in this region, get them from the referred pattern dictionary */
475
2.12k
    HPATS = jbig2_decode_ht_region_get_hpats(ctx, segment);
476
2.12k
    if (!HPATS) {
477
17
        code = jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "no pattern dictionary found, skipping halftone image");
478
17
        goto cleanup;
479
17
    }
480
481
    /* 6.6.5 point 1. Fill bitmap with HDEFPIXEL */
482
2.10k
    memset(image->data, params->HDEFPIXEL, image->stride * image->height);
483
484
    /* 6.6.5 point 2. compute HSKIP according to 6.6.5.1 */
485
2.10k
    if (params->HENABLESKIP == 1) {
486
352
        HSKIP = jbig2_image_new(ctx, params->HGW, params->HGH);
487
352
        if (HSKIP == NULL)
488
3
            return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to allocate skip image");
489
490
283M
        for (mg = 0; mg < params->HGH; ++mg) {
491
1.75G
            for (ng = 0; ng < params->HGW; ++ng) {
492
1.46G
                int64_t x = ((int64_t) params->HGX + mg * params->HRY + ng * params->HRX) >> 8;
493
1.46G
                int64_t y = ((int64_t) params->HGY + mg * params->HRX - ng * params->HRY) >> 8;
494
495
1.46G
                if (x + HPATS->HPW <= 0 || x >= image->width || y + HPATS->HPH <= 0 || y >= image->height) {
496
1.46G
                    jbig2_image_set_pixel(HSKIP, ng, mg, 1);
497
1.46G
                } else {
498
1.00M
                    jbig2_image_set_pixel(HSKIP, ng, mg, 0);
499
1.00M
                }
500
1.46G
            }
501
283M
        }
502
349
    }
503
504
    /* 6.6.5 point 3. set HBPP to ceil(log2(HNUMPATS)): */
505
2.10k
    HNUMPATS = HPATS->n_patterns;
506
2.10k
    HBPP = 0;
507
18.7k
    while (HNUMPATS > (1U << ++HBPP));
508
2.10k
    if (HBPP > 16) {
509
2
        code = jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "HBPP is larger than supported (%u)", HBPP);
510
2
        goto cleanup;
511
2
    }
512
513
    /* 6.6.5 point 4. decode gray-scale image as mentioned in annex C */
514
2.09k
    GI = jbig2_decode_gray_scale_image(ctx, segment, data, size,
515
2.09k
                                       params->HMMR, params->HGW, params->HGH, HBPP, params->HENABLESKIP, HSKIP, params->HTEMPLATE, GB_stats);
516
2.09k
    if (!GI) {
517
91
        code = jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "unable to acquire gray-scale image, skipping halftone image");
518
91
        goto cleanup;
519
91
    }
520
521
    /* 6.6.5 point 5. place patterns with procedure mentioned in 6.6.5.2 */
522
49.1M
    for (mg = 0; mg < params->HGH; ++mg) {
523
194M
        for (ng = 0; ng < params->HGW; ++ng) {
524
145M
            int64_t x = ((int64_t) params->HGX + mg * params->HRY + ng * params->HRX) >> 8;
525
145M
            int64_t y = ((int64_t) params->HGY + mg * params->HRX - ng * params->HRY) >> 8;
526
527
            /* prevent pattern index >= HNUMPATS */
528
145M
            gray_val = GI[ng][mg];
529
145M
            if (gray_val >= HNUMPATS) {
530
3.17M
                jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "gray-scale index %d out of range, using largest index", gray_val);
531
                /* use highest available pattern */
532
3.17M
                gray_val = HNUMPATS - 1;
533
3.17M
            }
534
145M
            code = jbig2_image_compose(ctx, image, HPATS->patterns[gray_val], x, y, params->HCOMBOP);
535
145M
            if (code < 0) {
536
0
                code = jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to compose pattern with gray-scale image");
537
0
                goto cleanup;
538
0
            }
539
145M
        }
540
49.1M
    }
541
542
2.11k
cleanup:
543
2.11k
    if (GI) {
544
2.96M
        for (i = 0; i < params->HGW; ++i) {
545
2.96M
            jbig2_free(ctx->allocator, GI[i]);
546
2.96M
        }
547
2.00k
    }
548
2.11k
    jbig2_free(ctx->allocator, GI);
549
2.11k
    jbig2_image_release(ctx, HSKIP);
550
551
2.11k
    return code;
552
2.00k
}
553
554
/**
555
 * jbig2_halftone_region: read a halftone region segment header
556
 **/
557
int
558
jbig2_halftone_region(Jbig2Ctx *ctx, Jbig2Segment *segment, const byte *segment_data)
559
2.15k
{
560
2.15k
    int offset = 0;
561
2.15k
    Jbig2RegionSegmentInfo region_info;
562
2.15k
    Jbig2HalftoneRegionParams params;
563
2.15k
    Jbig2Image *image = NULL;
564
2.15k
    Jbig2ArithCx *GB_stats = NULL;
565
2.15k
    int code = 0;
566
567
    /* 7.4.5.1 */
568
2.15k
    if (segment->data_length < 17)
569
6
        goto too_short;
570
2.14k
    jbig2_get_region_segment_info(&region_info, segment_data);
571
2.14k
    offset += 17;
572
573
2.14k
    if (segment->data_length < 18)
574
1
        goto too_short;
575
576
    /* 7.4.5.1.1 Figure 42 */
577
2.14k
    params.flags = segment_data[offset];
578
2.14k
    params.HMMR = params.flags & 1;
579
2.14k
    params.HTEMPLATE = (params.flags & 6) >> 1;
580
2.14k
    params.HENABLESKIP = (params.flags & 8) >> 3;
581
2.14k
    params.HCOMBOP = (Jbig2ComposeOp)((params.flags & 0x70) >> 4);
582
2.14k
    params.HDEFPIXEL = (params.flags & 0x80) >> 7;
583
2.14k
    offset += 1;
584
585
2.14k
    jbig2_error(ctx, JBIG2_SEVERITY_INFO, segment->number,
586
2.14k
                "halftone region: %u x %u @ (%u, %u), flags = %02x", region_info.width, region_info.height, region_info.x, region_info.y, params.flags);
587
588
2.14k
    if (params.HMMR && params.HTEMPLATE) {
589
141
        jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "HTEMPLATE is %d when HMMR is %d, contrary to spec", params.HTEMPLATE, params.HMMR);
590
141
    }
591
2.14k
    if (params.HMMR && params.HENABLESKIP) {
592
113
        jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "HENABLESKIP is %d when HMMR is %d, contrary to spec", params.HENABLESKIP, params.HMMR);
593
113
    }
594
595
    /* 7.4.5.1.2 Figure 43 */
596
2.14k
    if (segment->data_length - offset < 16)
597
6
        goto too_short;
598
2.13k
    params.HGW = jbig2_get_uint32(segment_data + offset);
599
2.13k
    params.HGH = jbig2_get_uint32(segment_data + offset + 4);
600
2.13k
    params.HGX = jbig2_get_int32(segment_data + offset + 8);
601
2.13k
    params.HGY = jbig2_get_int32(segment_data + offset + 12);
602
2.13k
    offset += 16;
603
604
    /* 7.4.5.1.3 Figure 44 */
605
2.13k
    if (segment->data_length - offset < 4)
606
1
        goto too_short;
607
2.13k
    params.HRX = jbig2_get_uint16(segment_data + offset);
608
2.13k
    params.HRY = jbig2_get_uint16(segment_data + offset + 2);
609
2.13k
    offset += 4;
610
611
2.13k
    jbig2_error(ctx, JBIG2_SEVERITY_INFO, segment->number,
612
2.13k
                "grid %d x %d @ (%d.%d,%d.%d) vector (%d.%d,%d.%d)",
613
2.13k
                params.HGW, params.HGH,
614
2.13k
                params.HGX >> 8, params.HGX & 0xff,
615
2.13k
                params.HGY >> 8, params.HGY & 0xff,
616
2.13k
                params.HRX >> 8, params.HRX & 0xff,
617
2.13k
                params.HRY >> 8, params.HRY & 0xff);
618
619
    /* 7.4.5.2 */
620
2.13k
    if (!params.HMMR) {
621
        /* allocate and zero arithmetic coding stats */
622
290
        int stats_size = jbig2_generic_stats_size(ctx, params.HTEMPLATE);
623
624
290
        GB_stats = jbig2_new(ctx, Jbig2ArithCx, stats_size);
625
290
        if (GB_stats == NULL) {
626
1
            return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "failed to allocate arithmetic decoder states in halftone region");
627
1
        }
628
289
        memset(GB_stats, 0, stats_size);
629
289
    }
630
631
2.13k
    image = jbig2_image_new(ctx, region_info.width, region_info.height);
632
2.13k
    if (image == NULL) {
633
14
        jbig2_free(ctx->allocator, GB_stats);
634
14
        return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to allocate halftone image");
635
14
    }
636
637
2.12k
    code = jbig2_decode_halftone_region(ctx, segment, &params, segment_data + offset, segment->data_length - offset, image, GB_stats);
638
2.12k
    if (code < 0) {
639
113
        jbig2_image_release(ctx, image);
640
113
        jbig2_free(ctx->allocator, GB_stats);
641
113
        return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode halftone region");
642
113
    }
643
644
    /* todo: retain GB_stats? */
645
2.00k
    if (!params.HMMR) {
646
243
        jbig2_free(ctx->allocator, GB_stats);
647
243
    }
648
649
2.00k
    code = jbig2_page_add_result(ctx, &ctx->pages[ctx->current_page], image, region_info.x, region_info.y, region_info.op);
650
2.00k
    if (code < 0) {
651
190
        jbig2_image_release(ctx, image);
652
190
        return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "unable to add halftone region to page");
653
190
    }
654
655
1.81k
    jbig2_image_release(ctx, image);
656
657
1.81k
    return code;
658
659
14
too_short:
660
14
    return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "segment too short");
661
2.00k
}