| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549 |
- /* Copyright (C) 2001-2023 Artifex Software, Inc.
- All Rights Reserved.
- This software is provided AS-IS with no warranty, either express or
- implied.
- This software is distributed under license and may not be copied,
- modified or distributed except as expressly authorized under the terms
- of the license contained in the file LICENSE in this distribution.
- Refer to licensing information at http://www.artifex.com or contact
- Artifex Software, Inc., 39 Mesa Street, Suite 108A, San Francisco,
- CA 94129, USA, for further information.
- */
- /*
- jbig2dec
- */
- /**
- * Generic Refinement region handlers.
- **/
- #ifdef HAVE_CONFIG_H
- #include "config.h"
- #endif
- #include "os_types.h"
- #include <stddef.h>
- #include <string.h> /* memcpy(), memset() */
- #include <stdio.h>
- #include "jbig2.h"
- #include "jbig2_priv.h"
- #include "jbig2_arith.h"
- #include "jbig2_generic.h"
- #include "jbig2_image.h"
- #include "jbig2_page.h"
- #include "jbig2_refinement.h"
- #include "jbig2_segment.h"
- #define pixel_outside_field(x, y) \
- ((y) < -128 || (y) > 0 || (x) < -128 || ((y) < 0 && (x) > 127) || ((y) == 0 && (x) >= 0))
- #define refpixel_outside_field(x, y) \
- ((y) < -128 || (y) > 127 || (x) < -128 || (x) > 127)
- static int
- jbig2_decode_refinement_template0_unopt(Jbig2Ctx *ctx,
- Jbig2Segment *segment,
- const Jbig2RefinementRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GR_stats)
- {
- const int GRW = image->width;
- const int GRH = image->height;
- Jbig2Image *ref = params->GRREFERENCE;
- const int dx = params->GRREFERENCEDX;
- const int dy = params->GRREFERENCEDY;
- uint32_t CONTEXT;
- int x, y;
- int bit;
- if (pixel_outside_field(params->grat[0], params->grat[1]) ||
- refpixel_outside_field(params->grat[2], params->grat[3]))
- return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number,
- "adaptive template pixel is out of field");
- for (y = 0; y < GRH; y++) {
- for (x = 0; x < GRW; x++) {
- CONTEXT = 0;
- CONTEXT |= jbig2_image_get_pixel(image, x - 1, y + 0) << 0;
- CONTEXT |= jbig2_image_get_pixel(image, x + 1, y - 1) << 1;
- CONTEXT |= jbig2_image_get_pixel(image, x + 0, y - 1) << 2;
- CONTEXT |= jbig2_image_get_pixel(image, x + params->grat[0], y + params->grat[1]) << 3;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 1, y - dy + 1) << 4;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 0, y - dy + 1) << 5;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx - 1, y - dy + 1) << 6;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 1, y - dy + 0) << 7;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 0, y - dy + 0) << 8;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx - 1, y - dy + 0) << 9;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 1, y - dy - 1) << 10;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 0, y - dy - 1) << 11;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + params->grat[2], y - dy + params->grat[3]) << 12;
- bit = jbig2_arith_decode(ctx, as, &GR_stats[CONTEXT]);
- if (bit < 0)
- return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling refinement template0");
- jbig2_image_set_pixel(image, x, y, bit);
- }
- }
- #ifdef JBIG2_DEBUG_DUMP
- {
- static count = 0;
- char name[32];
- int code;
- snprintf(name, 32, "refin-%d.pbm", count);
- code = jbig2_image_write_pbm_file(ref, name);
- if (code < 0)
- return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed write refinement input");
- snprintf(name, 32, "refout-%d.pbm", count);
- code = jbig2_image_write_pbm_file(image, name);
- if (code < 0)
- return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed write refinement output");
- count++;
- }
- #endif
- return 0;
- }
- static int
- jbig2_decode_refinement_template1_unopt(Jbig2Ctx *ctx,
- Jbig2Segment *segment,
- const Jbig2RefinementRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GR_stats)
- {
- const int GRW = image->width;
- const int GRH = image->height;
- Jbig2Image *ref = params->GRREFERENCE;
- const int dx = params->GRREFERENCEDX;
- const int dy = params->GRREFERENCEDY;
- uint32_t CONTEXT;
- int x, y;
- int bit;
- for (y = 0; y < GRH; y++) {
- for (x = 0; x < GRW; x++) {
- CONTEXT = 0;
- CONTEXT |= jbig2_image_get_pixel(image, x - 1, y + 0) << 0;
- CONTEXT |= jbig2_image_get_pixel(image, x + 1, y - 1) << 1;
- CONTEXT |= jbig2_image_get_pixel(image, x + 0, y - 1) << 2;
- CONTEXT |= jbig2_image_get_pixel(image, x - 1, y - 1) << 3;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 1, y - dy + 1) << 4;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 0, y - dy + 1) << 5;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 1, y - dy + 0) << 6;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 0, y - dy + 0) << 7;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx - 1, y - dy + 0) << 8;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 0, y - dy - 1) << 9;
- bit = jbig2_arith_decode(ctx, as, &GR_stats[CONTEXT]);
- if (bit < 0)
- return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling refinement template0");
- jbig2_image_set_pixel(image, x, y, bit);
- }
- }
- #ifdef JBIG2_DEBUG_DUMP
- {
- static count = 0;
- char name[32];
- snprintf(name, 32, "refin-%d.pbm", count);
- code = jbig2_image_write_pbm_file(ref, name);
- if (code < 0)
- return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to write refinement input");
- snprintf(name, 32, "refout-%d.pbm", count);
- code = jbig2_image_write_pbm_file(image, name);
- if (code < 0)
- return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to write refinement output");
- count++;
- }
- #endif
- return 0;
- }
- #if 0 /* currently not used */
- static int
- jbig2_decode_refinement_template1(Jbig2Ctx *ctx,
- Jbig2Segment *segment,
- const Jbig2RefinementRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GR_stats)
- {
- const int GRW = image->width;
- const int GRH = image->height;
- const int stride = image->stride;
- const int refstride = params->reference->stride;
- const int dy = params->DY;
- byte *grreg_line = (byte *) image->data;
- byte *grref_line = (byte *) params->reference->data;
- int x, y;
- for (y = 0; y < GRH; y++) {
- const int padded_width = (GRW + 7) & -8;
- uint32_t CONTEXT;
- uint32_t refline_m1; /* previous line of the reference bitmap */
- uint32_t refline_0; /* current line of the reference bitmap */
- uint32_t refline_1; /* next line of the reference bitmap */
- uint32_t line_m1; /* previous line of the decoded bitmap */
- line_m1 = (y >= 1) ? grreg_line[-stride] : 0;
- refline_m1 = ((y - dy) >= 1) ? grref_line[(-1 - dy) * stride] << 2 : 0;
- refline_0 = (((y - dy) > 0) && ((y - dy) < GRH)) ? grref_line[(0 - dy) * stride] << 4 : 0;
- refline_1 = (y < GRH - 1) ? grref_line[(+1 - dy) * stride] << 7 : 0;
- CONTEXT = ((line_m1 >> 5) & 0x00e) | ((refline_1 >> 5) & 0x030) | ((refline_0 >> 5) & 0x1c0) | ((refline_m1 >> 5) & 0x200);
- for (x = 0; x < padded_width; x += 8) {
- byte result = 0;
- int x_minor;
- const int minor_width = GRW - x > 8 ? 8 : GRW - x;
- if (y >= 1) {
- line_m1 = (line_m1 << 8) | (x + 8 < GRW ? grreg_line[-stride + (x >> 3) + 1] : 0);
- refline_m1 = (refline_m1 << 8) | (x + 8 < GRW ? grref_line[-refstride + (x >> 3) + 1] << 2 : 0);
- }
- refline_0 = (refline_0 << 8) | (x + 8 < GRW ? grref_line[(x >> 3) + 1] << 4 : 0);
- if (y < GRH - 1)
- refline_1 = (refline_1 << 8) | (x + 8 < GRW ? grref_line[+refstride + (x >> 3) + 1] << 7 : 0);
- else
- refline_1 = 0;
- /* this is the speed critical inner-loop */
- for (x_minor = 0; x_minor < minor_width; x_minor++) {
- int bit;
- bit = jbig2_arith_decode(ctx, as, &GR_stats[CONTEXT]);
- if (bit < 0)
- return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode arithmetic code when handling refinement template1");
- result |= bit << (7 - x_minor);
- CONTEXT = ((CONTEXT & 0x0d6) << 1) | bit |
- ((line_m1 >> (9 - x_minor)) & 0x002) |
- ((refline_1 >> (9 - x_minor)) & 0x010) | ((refline_0 >> (9 - x_minor)) & 0x040) | ((refline_m1 >> (9 - x_minor)) & 0x200);
- }
- grreg_line[x >> 3] = result;
- }
- grreg_line += stride;
- grref_line += refstride;
- }
- return 0;
- }
- #endif
- typedef uint32_t(*ContextBuilder)(const Jbig2RefinementRegionParams *, Jbig2Image *, int, int);
- static int
- implicit_value(const Jbig2RefinementRegionParams *params, Jbig2Image *image, int x, int y)
- {
- Jbig2Image *ref = params->GRREFERENCE;
- int i = x - params->GRREFERENCEDX;
- int j = y - params->GRREFERENCEDY;
- int m = jbig2_image_get_pixel(ref, i, j);
- return ((jbig2_image_get_pixel(ref, i - 1, j - 1) == m) &&
- (jbig2_image_get_pixel(ref, i, j - 1) == m) &&
- (jbig2_image_get_pixel(ref, i + 1, j - 1) == m) &&
- (jbig2_image_get_pixel(ref, i - 1, j) == m) &&
- (jbig2_image_get_pixel(ref, i + 1, j) == m) &&
- (jbig2_image_get_pixel(ref, i - 1, j + 1) == m) &&
- (jbig2_image_get_pixel(ref, i, j + 1) == m) &&
- (jbig2_image_get_pixel(ref, i + 1, j + 1) == m)
- )? m : -1;
- }
- static uint32_t
- mkctx0(const Jbig2RefinementRegionParams *params, Jbig2Image *image, int x, int y)
- {
- Jbig2Image *ref = params->GRREFERENCE;
- const int dx = params->GRREFERENCEDX;
- const int dy = params->GRREFERENCEDY;
- uint32_t CONTEXT;
- CONTEXT = jbig2_image_get_pixel(image, x - 1, y + 0);
- CONTEXT |= jbig2_image_get_pixel(image, x + 1, y - 1) << 1;
- CONTEXT |= jbig2_image_get_pixel(image, x + 0, y - 1) << 2;
- CONTEXT |= jbig2_image_get_pixel(image, x + params->grat[0], y + params->grat[1]) << 3;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 1, y - dy + 1) << 4;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 0, y - dy + 1) << 5;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx - 1, y - dy + 1) << 6;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 1, y - dy + 0) << 7;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 0, y - dy + 0) << 8;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx - 1, y - dy + 0) << 9;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 1, y - dy - 1) << 10;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 0, y - dy - 1) << 11;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + params->grat[2], y - dy + params->grat[3]) << 12;
- return CONTEXT;
- }
- static uint32_t
- mkctx1(const Jbig2RefinementRegionParams *params, Jbig2Image *image, int x, int y)
- {
- Jbig2Image *ref = params->GRREFERENCE;
- const int dx = params->GRREFERENCEDX;
- const int dy = params->GRREFERENCEDY;
- uint32_t CONTEXT;
- CONTEXT = jbig2_image_get_pixel(image, x - 1, y + 0);
- CONTEXT |= jbig2_image_get_pixel(image, x + 1, y - 1) << 1;
- CONTEXT |= jbig2_image_get_pixel(image, x + 0, y - 1) << 2;
- CONTEXT |= jbig2_image_get_pixel(image, x - 1, y - 1) << 3;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 1, y - dy + 1) << 4;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 0, y - dy + 1) << 5;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 1, y - dy + 0) << 6;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 0, y - dy + 0) << 7;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx - 1, y - dy + 0) << 8;
- CONTEXT |= jbig2_image_get_pixel(ref, x - dx + 0, y - dy - 1) << 9;
- return CONTEXT;
- }
- static int
- jbig2_decode_refinement_TPGRON(Jbig2Ctx *ctx, const Jbig2RefinementRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GR_stats)
- {
- const int GRW = image->width;
- const int GRH = image->height;
- int x, y, iv, LTP = 0;
- uint32_t start_context = (params->GRTEMPLATE ? 0x40 : 0x100);
- ContextBuilder mkctx = (params->GRTEMPLATE ? mkctx1 : mkctx0);
- if (params->GRTEMPLATE == 0 &&
- (pixel_outside_field(params->grat[0], params->grat[1]) ||
- refpixel_outside_field(params->grat[2], params->grat[3])))
- return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, JBIG2_UNKNOWN_SEGMENT_NUMBER,
- "adaptive template pixel is out of field");
- for (y = 0; y < GRH; y++) {
- int bit = jbig2_arith_decode(ctx, as, &GR_stats[start_context]);
- if (bit < 0)
- return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, JBIG2_UNKNOWN_SEGMENT_NUMBER, "failed to decode arithmetic code when handling refinement TPGRON1");
- LTP ^= bit;
- if (!LTP) {
- for (x = 0; x < GRW; x++) {
- bit = jbig2_arith_decode(ctx, as, &GR_stats[mkctx(params, image, x, y)]);
- if (bit < 0)
- return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, JBIG2_UNKNOWN_SEGMENT_NUMBER, "failed to decode arithmetic code when handling refinement TPGRON1");
- jbig2_image_set_pixel(image, x, y, bit);
- }
- } else {
- for (x = 0; x < GRW; x++) {
- iv = implicit_value(params, image, x, y);
- if (iv < 0) {
- int bit = jbig2_arith_decode(ctx, as, &GR_stats[mkctx(params, image, x, y)]);
- if (bit < 0)
- return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, JBIG2_UNKNOWN_SEGMENT_NUMBER, "failed to decode arithmetic code when handling refinement TPGRON1");
- jbig2_image_set_pixel(image, x, y, bit);
- } else
- jbig2_image_set_pixel(image, x, y, iv);
- }
- }
- }
- return 0;
- }
- /**
- * jbig2_decode_refinement_region: Decode a generic refinement region.
- * @ctx: The context for allocation and error reporting.
- * @segment: A segment reference for error reporting.
- * @params: Decoding parameter set.
- * @as: Arithmetic decoder state.
- * @image: Where to store the decoded image.
- * @GR_stats: Arithmetic stats.
- *
- * Decodes a generic refinement region, according to section 6.3.
- * an already allocated Jbig2Image object in @image for the result.
- *
- * Because this API is based on an arithmetic decoding state, it is
- * not suitable for MMR decoding.
- *
- * Return code: 0 on success.
- **/
- int
- jbig2_decode_refinement_region(Jbig2Ctx *ctx,
- Jbig2Segment *segment,
- const Jbig2RefinementRegionParams *params, Jbig2ArithState *as, Jbig2Image *image, Jbig2ArithCx *GR_stats)
- {
- jbig2_error(ctx, JBIG2_SEVERITY_DEBUG, segment->number,
- "decoding generic refinement region with offset %d,%x, GRTEMPLATE=%d, TPGRON=%d",
- params->GRREFERENCEDX, params->GRREFERENCEDY, params->GRTEMPLATE, params->TPGRON);
- if (params->TPGRON)
- return jbig2_decode_refinement_TPGRON(ctx, params, as, image, GR_stats);
- if (params->GRTEMPLATE)
- return jbig2_decode_refinement_template1_unopt(ctx, segment, params, as, image, GR_stats);
- else
- return jbig2_decode_refinement_template0_unopt(ctx, segment, params, as, image, GR_stats);
- }
- /**
- * Find the first referred-to intermediate region segment
- * with a non-NULL result for use as a reference image
- */
- static Jbig2Segment *
- jbig2_region_find_referred(Jbig2Ctx *ctx, Jbig2Segment *segment)
- {
- const int nsegments = segment->referred_to_segment_count;
- Jbig2Segment *rsegment;
- int index;
- for (index = 0; index < nsegments; index++) {
- rsegment = jbig2_find_segment(ctx, segment->referred_to_segments[index]);
- if (rsegment == NULL) {
- jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to find referred to segment %d", segment->referred_to_segments[index]);
- continue;
- }
- switch (rsegment->flags & 63) {
- case 4: /* intermediate text region */
- case 20: /* intermediate halftone region */
- case 36: /* intermediate generic region */
- case 40: /* intermediate generic refinement region */
- if (rsegment->result)
- return rsegment;
- break;
- default: /* keep looking */
- break;
- }
- }
- /* no appropriate reference was found. */
- return NULL;
- }
- /**
- * Handler for generic refinement region segments
- */
- int
- jbig2_refinement_region(Jbig2Ctx *ctx, Jbig2Segment *segment, const byte *segment_data)
- {
- Jbig2RefinementRegionParams params;
- Jbig2RegionSegmentInfo rsi;
- int offset = 0;
- byte seg_flags;
- int code = 0;
- /* 7.4.7 */
- if (segment->data_length < 18)
- return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "segment too short");
- jbig2_get_region_segment_info(&rsi, segment_data);
- 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);
- /* 7.4.7.2 */
- seg_flags = segment_data[17];
- params.GRTEMPLATE = seg_flags & 0x01;
- params.TPGRON = seg_flags & 0x02 ? 1 : 0;
- jbig2_error(ctx, JBIG2_SEVERITY_INFO, segment->number,
- "segment flags = %02x %s%s", seg_flags, params.GRTEMPLATE ? " GRTEMPLATE" : "", params.TPGRON ? " TPGRON" : "");
- if (seg_flags & 0xFC)
- jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "reserved segment flag bits are non-zero");
- offset += 18;
- /* 7.4.7.3 */
- if (!params.GRTEMPLATE) {
- if (segment->data_length < 22)
- return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "segment too short");
- params.grat[0] = segment_data[offset + 0];
- params.grat[1] = segment_data[offset + 1];
- params.grat[2] = segment_data[offset + 2];
- params.grat[3] = segment_data[offset + 3];
- jbig2_error(ctx, JBIG2_SEVERITY_INFO, segment->number,
- "grat1: (%d, %d) grat2: (%d, %d)", params.grat[0], params.grat[1], params.grat[2], params.grat[3]);
- offset += 4;
- }
- /* 7.4.7.4 - set up the reference image */
- if (segment->referred_to_segment_count) {
- Jbig2Segment *ref;
- ref = jbig2_region_find_referred(ctx, segment);
- if (ref == NULL)
- return jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to find reference bitmap");
- if (ref->result == NULL)
- return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "reference bitmap has no decoded image");
- /* the reference bitmap is the result of a previous
- intermediate region segment; the reference selection
- rules say to use the first one available, and not to
- reuse any intermediate result, so we simply take another
- reference to it and free the original to keep track of this. */
- params.GRREFERENCE = jbig2_image_reference(ctx, (Jbig2Image *) ref->result);
- jbig2_image_release(ctx, (Jbig2Image *) ref->result);
- ref->result = NULL;
- jbig2_error(ctx, JBIG2_SEVERITY_DEBUG, segment->number, "found reference bitmap in segment %d", ref->number);
- } else {
- /* the reference is just (a subset of) the page buffer */
- if (ctx->pages[ctx->current_page].image == NULL)
- return jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "reference page bitmap has no decoded image");
- params.GRREFERENCE = jbig2_image_reference(ctx, ctx->pages[ctx->current_page].image);
- /* TODO: subset the image if appropriate */
- }
- /* 7.4.7.5 */
- params.GRREFERENCEDX = 0;
- params.GRREFERENCEDY = 0;
- {
- Jbig2WordStream *ws = NULL;
- Jbig2ArithState *as = NULL;
- Jbig2ArithCx *GR_stats = NULL;
- int stats_size;
- Jbig2Image *image = NULL;
- image = jbig2_image_new(ctx, rsi.width, rsi.height);
- if (image == NULL) {
- code = jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to allocate refinement image");
- goto cleanup;
- }
- jbig2_error(ctx, JBIG2_SEVERITY_DEBUG, segment->number, "allocated %d x %d image buffer for region decode results", rsi.width, rsi.height);
- stats_size = params.GRTEMPLATE ? 1 << 10 : 1 << 13;
- GR_stats = jbig2_new(ctx, Jbig2ArithCx, stats_size);
- if (GR_stats == NULL) {
- code = jbig2_error(ctx, JBIG2_SEVERITY_FATAL, segment->number, "failed to allocate arithmetic decoder state for generic refinement regions");
- goto cleanup;
- }
- memset(GR_stats, 0, stats_size);
- ws = jbig2_word_stream_buf_new(ctx, segment_data + offset, segment->data_length - offset);
- if (ws == NULL) {
- code = jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to allocate word stream when handling refinement region");
- goto cleanup;
- }
- as = jbig2_arith_new(ctx, ws);
- if (as == NULL) {
- code = jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to allocate arithmetic coding state when handling refinement region");
- goto cleanup;
- }
- code = jbig2_decode_refinement_region(ctx, segment, ¶ms, as, image, GR_stats);
- if (code < 0) {
- jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "failed to decode refinement region");
- goto cleanup;
- }
- if ((segment->flags & 63) == 40) {
- /* intermediate region. save the result for later */
- segment->result = jbig2_image_reference(ctx, image);
- } else {
- /* immediate region. composite onto the page */
- jbig2_error(ctx, JBIG2_SEVERITY_DEBUG, segment->number,
- "composing %dx%d decoded refinement region onto page at (%d, %d)", rsi.width, rsi.height, rsi.x, rsi.y);
- code = jbig2_page_add_result(ctx, &ctx->pages[ctx->current_page], image, rsi.x, rsi.y, rsi.op);
- if (code < 0) {
- jbig2_error(ctx, JBIG2_SEVERITY_WARNING, segment->number, "unable to add refinement region to page");
- goto cleanup;
- }
- }
- cleanup:
- jbig2_image_release(ctx, image);
- jbig2_image_release(ctx, params.GRREFERENCE);
- jbig2_free(ctx->allocator, as);
- jbig2_word_stream_buf_free(ctx, ws);
- jbig2_free(ctx->allocator, GR_stats);
- }
- return code;
- }
|