Coverage Report

Created: 2026-02-14 07:09

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ghostpdl/base/gxipixel.c
Line
Count
Source
1
/* Copyright (C) 2001-2025 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
/* Common code for ImageType 1 and 4 initialization */
18
#include "gx.h"
19
#include "math_.h"
20
#include "memory_.h"
21
#include "gpcheck.h"
22
#include "gscdefs.h"            /* for image class table */
23
#include "gserrors.h"
24
#include "gsstruct.h"
25
#include "gsutil.h"
26
#include "gxfixed.h"
27
#include "gxfrac.h"
28
#include "gxarith.h"
29
#include "gxmatrix.h"
30
#include "gsccolor.h"
31
#include "gspaint.h"
32
#include "gzstate.h"
33
#include "gxdevice.h"
34
#include "gzpath.h"
35
#include "gzcpath.h"
36
#include "gxdevmem.h"
37
#include "gximage.h"
38
#include "gxiparam.h"
39
#include "gdevmrop.h"
40
#include "gscspace.h"
41
#include "gscindex.h"
42
#include "gsicc_cache.h"
43
#include "gsicc_cms.h"
44
#include "gsicc_manage.h"
45
#include "gxdevsop.h"
46
47
/* Structure descriptors */
48
private_st_gx_image_enum();
49
50
/* Image class procedures */
51
extern_gx_image_class_table();
52
53
/* Enumerator procedures */
54
static const gx_image_enum_procs_t image1_enum_procs = {
55
    gx_image1_plane_data, gx_image1_end_image, gx_image1_flush
56
};
57
58
/* GC procedures */
59
gs_private_st_ptrs2(st_color_cache, gx_image_color_cache_t, "gx_image_color_cache",
60
                    color_cache_enum_ptrs, color_cache_reloc_ptrs,
61
                    is_transparent, device_contone);
62
static
63
589
ENUM_PTRS_WITH(image_enum_enum_ptrs, gx_image_enum *eptr)
64
217
{
65
217
    int bps;
66
217
    gs_ptr_type_t ret;
67
68
    /* Enumerate the used members of clues.dev_color. */
69
217
    index -= gx_image_enum_num_ptrs;
70
217
    bps = eptr->unpack_bps;
71
217
    if (eptr->spp != 1)
72
0
        bps = 8;
73
217
    else if (bps > 8 || eptr->unpack == sample_unpack_copy)
74
0
        bps = 1;
75
217
    if (index >= (1 << bps) * st_device_color_max_ptrs)         /* done */
76
31
        return 0;
77
    /* the clues may have been cleared by gx_image_free_enum, but not freed in that */
78
    /* function due to being at a different save level. Only trace if dev_color.type != 0. */
79
186
    if (eptr->spp == 1) {
80
186
        if (eptr->clues != NULL) {
81
186
            if (eptr->clues[(index/st_device_color_max_ptrs) *
82
186
                (255 / ((1 << bps) - 1))].dev_color.type != 0) {
83
186
                ret = ENUM_USING(st_device_color,
84
186
                                 &eptr->clues[(index / st_device_color_max_ptrs) *
85
186
                                 (255 / ((1 << bps) - 1))].dev_color,
86
186
                                 sizeof(eptr->clues[0].dev_color),
87
186
                                 index % st_device_color_max_ptrs);
88
186
            } else {
89
0
                ret = 0;
90
0
            }
91
186
        } else {
92
0
            ret = 0;
93
0
        }
94
186
    } else {
95
0
        ret = 0;
96
0
    }
97
186
    if (ret == 0)               /* don't stop early */
98
186
        ENUM_RETURN(0);
99
0
    return ret;
100
186
}
101
102
372
#define e1(i,elt) ENUM_PTR(i,gx_image_enum,elt);
103
589
gx_image_enum_do_ptrs(e1)
104
589
#undef e1
105
589
ENUM_PTRS_END
106
107
31
static RELOC_PTRS_WITH(image_enum_reloc_ptrs, gx_image_enum *eptr)
108
31
{
109
31
    int i;
110
111
372
#define r1(i,elt) RELOC_PTR(gx_image_enum,elt);
112
372
    gx_image_enum_do_ptrs(r1)
113
31
#undef r1
114
31
    {
115
31
        int bps = eptr->unpack_bps;
116
117
31
        if (eptr->spp != 1)
118
0
            bps = 8;
119
31
        else if (bps > 8 || eptr->unpack == sample_unpack_copy)
120
0
            bps = 1;
121
31
        if (eptr->spp == 1) {
122
93
        for (i = 0; i <= 255; i += 255 / ((1 << bps) - 1))
123
62
            RELOC_USING(st_device_color,
124
31
                        &eptr->clues[i].dev_color, sizeof(gx_device_color));
125
31
    }
126
31
}
127
31
}
128
31
RELOC_PTRS_END
129
130
/* Forward declarations */
131
static int color_draws_b_w(gx_device * dev,
132
                            const gx_drawing_color * pdcolor);
133
static int image_init_colors(gx_image_enum * penum, int bps, int spp,
134
                               gs_image_format_t format,
135
                               const float *decode,
136
                               const gs_gstate * pgs, gx_device * dev,
137
                               const gs_color_space * pcs, bool * pdcb);
138
139
/* Procedures for unpacking the input data into bytes or fracs. */
140
/*extern SAMPLE_UNPACK_PROC(sample_unpack_copy); *//* declared above */
141
142
/*
143
 * Do common initialization for processing an ImageType 1 or 4 image.
144
 * Allocate the enumerator and fill in the following members:
145
 *      rect
146
 */
147
int
148
gx_image_enum_alloc(const gs_image_common_t * pic,
149
                    const gs_int_rect * prect, gs_memory_t * mem,
150
                    gx_image_enum **ppenum)
151
1.58M
{
152
1.58M
    const gs_pixel_image_t *pim = (const gs_pixel_image_t *)pic;
153
1.58M
    int width = pim->Width, height = pim->Height;
154
1.58M
    int bpc = pim->BitsPerComponent;
155
1.58M
    gx_image_enum *penum;
156
157
1.58M
    if (width < 0 || height < 0)
158
0
        return_error(gs_error_rangecheck);
159
1.58M
    switch (pim->format) {
160
1.56M
    case gs_image_format_chunky:
161
1.58M
    case gs_image_format_component_planar:
162
1.58M
        switch (bpc) {
163
1.58M
        case 1: case 2: case 4: case 8: case 12: case 16: break;
164
2
        default: return_error(gs_error_rangecheck);
165
1.58M
        }
166
1.58M
        break;
167
1.58M
    case gs_image_format_bit_planar:
168
0
        if (bpc < 1 || bpc > 8)
169
0
            return_error(gs_error_rangecheck);
170
1.58M
    }
171
1.58M
    if (prect) {
172
543k
        if (prect->p.x < 0 || prect->p.y < 0 ||
173
543k
            prect->q.x < prect->p.x || prect->q.y < prect->p.y ||
174
543k
            prect->q.x > width || prect->q.y > height
175
543k
            )
176
0
            return_error(gs_error_rangecheck);
177
543k
    }
178
1.58M
    *ppenum = NULL;   /* in case alloc fails and caller doesn't check code */
179
1.58M
    penum = gs_alloc_struct(mem, gx_image_enum, &st_gx_image_enum,
180
1.58M
                            "gx_default_begin_image");
181
1.58M
    if (penum == 0)
182
0
        return_error(gs_error_VMerror);
183
1.58M
    memset(penum, 0, sizeof(gx_image_enum));  /* in case of failure, no dangling pointers */
184
1.58M
    if (prect) {
185
543k
        penum->rect.x = prect->p.x;
186
543k
        penum->rect.y = prect->p.y;
187
543k
        penum->rect.w = prect->q.x - prect->p.x;
188
543k
        penum->rect.h = prect->q.y - prect->p.y;
189
1.04M
    } else {
190
1.04M
        penum->rect.x = 0, penum->rect.y = 0;
191
1.04M
        penum->rect.w = width, penum->rect.h = height;
192
1.04M
    }
193
1.58M
    penum->rrect.x = penum->rect.x;
194
1.58M
    penum->rrect.y = penum->rect.y;
195
1.58M
    penum->rrect.w = penum->rect.w;
196
1.58M
    penum->rrect.h = penum->rect.h;
197
1.58M
    penum->drect.x = penum->rect.x;
198
1.58M
    penum->drect.y = penum->rect.y;
199
1.58M
    penum->drect.w = penum->rect.w;
200
1.58M
    penum->drect.h = penum->rect.h;
201
#ifdef DEBUG
202
    if (gs_debug_c('b')) {
203
        dmlprintf2(mem, "[b]Image: w=%d h=%d", width, height);
204
        if (prect)
205
            dmprintf4(mem, " ((%d,%d),(%d,%d))",
206
                     prect->p.x, prect->p.y, prect->q.x, prect->q.y);
207
    }
208
#endif
209
1.58M
    *ppenum = penum;
210
1.58M
    return 0;
211
1.58M
}
212
213
/* Convert and restrict to a valid range. */
214
4.38M
static inline fixed float2fixed_rounded_boxed(double src) {
215
4.38M
    float v = floor(src*fixed_scale + 0.5);
216
217
4.38M
    if (v <= min_fixed)
218
10.4k
        return min_fixed;
219
4.37M
    else if (v >= max_fixed)
220
46.5k
        return max_fixed;
221
4.32M
    else
222
4.32M
        return  (fixed)v;
223
4.38M
}
224
225
/* Compute the image matrix combining the ImageMatrix with either the pmat or the pgs ctm */
226
int
227
gx_image_compute_mat(const gs_gstate *pgs, const gs_matrix *pmat, const gs_matrix *ImageMatrix,
228
                     gs_matrix_double *rmat)
229
2.51M
{
230
2.51M
    int code = 0;
231
232
2.51M
    if (pmat == 0)
233
2.45M
        pmat = &ctm_only(pgs);
234
2.51M
    if (ImageMatrix->xx == pmat->xx && ImageMatrix->xy == pmat->xy &&
235
979k
        ImageMatrix->yx == pmat->yx && ImageMatrix->yy == pmat->yy) {
236
        /* Process common special case separately to accept singular matrix. */
237
978k
        rmat->xx = rmat->yy = 1.;
238
978k
        rmat->xy = rmat->yx = 0.;
239
978k
        rmat->tx = pmat->tx - ImageMatrix->tx;
240
978k
        rmat->ty = pmat->ty - ImageMatrix->ty;
241
1.54M
    } else {
242
1.54M
        if ((code = gs_matrix_invert_to_double(ImageMatrix, rmat)) < 0 ||
243
1.53M
            (code = gs_matrix_multiply_double(rmat, pmat, rmat)) < 0
244
1.54M
            ) {
245
147
            return code;
246
147
        }
247
1.54M
    }
248
2.51M
    return code;
249
2.51M
}
250
251
/*
252
 * Finish initialization for processing an ImageType 1 or 4 image.
253
 * Assumes the following members of *penum are set in addition to those
254
 * set by gx_image_enum_alloc:
255
 *      alpha, use_mask_color, mask_color (if use_mask_color is true),
256
 *      masked, adjust
257
 */
258
int
259
gx_image_enum_begin(gx_device * dev, const gs_gstate * pgs,
260
                    const gs_matrix *pmat, const gs_image_common_t * pic,
261
                const gx_drawing_color * pdcolor, const gx_clip_path * pcpath,
262
                gs_memory_t * mem, gx_image_enum *penum)
263
1.58M
{
264
1.58M
    const gs_pixel_image_t *pim = (const gs_pixel_image_t *)pic;
265
1.58M
    gs_image_format_t format = pim->format;
266
1.58M
    const int width = pim->Width;
267
1.58M
    const int height = pim->Height;
268
1.58M
    const int bps = pim->BitsPerComponent;
269
1.58M
    bool masked = penum->masked;
270
1.58M
    const float *decode = pim->Decode;
271
1.58M
    gs_matrix_double mat;
272
1.58M
    int index_bps;
273
1.58M
    gs_color_space *pcs = pim->ColorSpace;
274
1.58M
    gs_logical_operation_t lop = (pgs ? pgs->log_op : lop_default);
275
1.58M
    int code;
276
1.58M
    int log2_xbytes = (bps <= 8 ? 0 : arch_log2_sizeof_frac);
277
1.58M
    int spp, nplanes, spread;
278
1.58M
    uint bsize;
279
1.58M
    byte *buffer = NULL;
280
1.58M
    fixed mtx, mty;
281
1.58M
    gs_fixed_point row_extent, col_extent, x_extent, y_extent;
282
1.58M
    bool device_color = true;
283
1.58M
    gs_fixed_rect obox, cbox;
284
1.58M
    bool gridfitimages = 0;
285
1.58M
    bool in_pattern_accumulator;
286
1.58M
    bool in_smask;
287
1.58M
    int orthogonal;
288
1.58M
    int force_interpolation = 0;
289
290
1.58M
    penum->pcs = NULL;
291
1.58M
    penum->clues = NULL;
292
1.58M
    penum->icc_setup.has_transfer = false;
293
1.58M
    penum->icc_setup.is_lab = false;
294
1.58M
    penum->icc_setup.must_halftone = false;
295
1.58M
    penum->icc_setup.need_decode = false;
296
1.58M
    penum->Width = width;
297
1.58M
    penum->Height = height;
298
299
1.58M
    if ((code = gx_image_compute_mat(pgs, pmat, &(pim->ImageMatrix), &mat)) < 0) {
300
112
        return code;
301
112
    }
302
1.58M
    lop = lop_sanitize(lop);
303
    /* Grid fit: A common construction in postscript/PDF files is for images
304
     * to be constructed as a series of 'stacked' 1 pixel high images.
305
     * Furthermore, many of these are implemented as an imagemask plotted on
306
     * top of thin rectangles. The different fill rules for images and line
307
     * art produces problems; line art fills a pixel if any part of it is
308
     * touched - images only fill a pixel if the centre of the pixel is
309
     * covered. Bug 692666 is such a problem.
310
     *
311
     * As a workaround for this problem, the code below was introduced. The
312
     * concept is that orthogonal images can be 'grid fitted' (or 'stretch')
313
     * to entirely cover pixels that they touch. Initially I had this working
314
     * for all images regardless of type, but as testing has proceeded, this
315
     * showed more and more regressions, so I've cut the cases back in which
316
     * this code is used until it now only triggers on imagemasks that are
317
     * either 1 pixel high, or wide, and then not if we are rendering a
318
     * glyph (such as from a type3 font).
319
     */
320
321
    /* Ask the device if we are in a pattern accumulator */
322
1.58M
    in_pattern_accumulator = ((dev_proc(dev, dev_spec_op)(dev, gxdso_in_pattern_accumulator, NULL, 0)) > 0);
323
324
    /* Figure out if we are orthogonal */
325
1.58M
    if (mat.xy == 0 && mat.yx == 0)
326
1.35M
        orthogonal = 1;
327
233k
    else if (mat.xx == 0 && mat.yy == 0)
328
133k
        orthogonal = 2;
329
99.3k
    else
330
99.3k
        orthogonal = 0;
331
332
    /* If we are in a pattern accumulator, we choose to always grid fit
333
     * orthogonal images. We do this by asking the device whether we
334
     * should grid fit. This allows us to avoid nasty blank lines around
335
     * the edges of cells. Similarly, for smasks.
336
     */
337
1.58M
    in_smask = (pim->override_in_smask ||
338
1.58M
                (dev_proc(dev, dev_spec_op)(dev, gxdso_in_smask, NULL, 0)) > 0);
339
1.58M
    gridfitimages = (in_smask || in_pattern_accumulator) && orthogonal;
340
341
1.58M
    if (pgs != NULL && pgs->show_gstate != NULL) {
342
        /* If we're a graphics state, and we're in a text object, then we
343
         * must be in a type3 font. Don't fiddle with it. */
344
1.33M
    } else if (!gridfitimages &&
345
1.02M
               (!penum->masked || penum->image_parent_type != 0)) {
346
        /* Other than for images we are specifically looking to grid fit (such as
347
         * ones in a pattern device), we only grid fit imagemasks */
348
812k
    } else if (gridfitimages && (penum->masked && penum->image_parent_type == 0)) {
349
        /* We don't gridfit imagemasks in a pattern accumulator */
350
526k
    } else if (pgs != NULL && pgs->fill_adjust.x == 0 && pgs->fill_adjust.y == 0) {
351
        /* If fill adjust is disabled, so is grid fitting */
352
526k
    } else if (orthogonal == 1) {
353
368k
        if (width == 1 || gridfitimages) {
354
318k
            if (mat.xx > 0) {
355
318k
                fixed ix0 = int2fixed(fixed2int(float2fixed(mat.tx)));
356
318k
                double x1 = mat.tx + mat.xx * width;
357
318k
                fixed ix1 = int2fixed(fixed2int_ceiling(float2fixed(x1)));
358
318k
                mat.tx = (double)fixed2float(ix0);
359
318k
                mat.xx = (double)(fixed2float(ix1 - ix0)/width);
360
318k
            } else if (mat.xx < 0) {
361
215
                fixed ix0 = int2fixed(fixed2int_ceiling(float2fixed(mat.tx)));
362
215
                double x1 = mat.tx + mat.xx * width;
363
215
                fixed ix1 = int2fixed(fixed2int(float2fixed(x1)));
364
215
                mat.tx = (double)fixed2float(ix0);
365
215
                mat.xx = (double)(fixed2float(ix1 - ix0)/width);
366
215
            }
367
318k
        }
368
368k
        if (height == 1 || gridfitimages) {
369
318k
            if (mat.yy > 0) {
370
287k
                fixed iy0 = int2fixed(fixed2int(float2fixed(mat.ty)));
371
287k
                double y1 = mat.ty + mat.yy * height;
372
287k
                fixed iy1 = int2fixed(fixed2int_ceiling(float2fixed(y1)));
373
287k
                mat.ty = (double)fixed2float(iy0);
374
287k
                mat.yy = (double)(fixed2float(iy1 - iy0)/height);
375
287k
            } else if (mat.yy < 0) {
376
31.3k
                fixed iy0 = int2fixed(fixed2int_ceiling(float2fixed(mat.ty)));
377
31.3k
                double y1 = mat.ty + mat.yy * height;
378
31.3k
                fixed iy1 = int2fixed(fixed2int(float2fixed(y1)));
379
31.3k
                mat.ty = (double)fixed2float(iy0);
380
31.3k
                mat.yy = ((double)fixed2float(iy1 - iy0)/height);
381
31.3k
            }
382
318k
        }
383
368k
    } else if (orthogonal == 2) {
384
131k
        if (height == 1 || gridfitimages) {
385
11
            if (mat.yx > 0) {
386
8
                fixed ix0 = int2fixed(fixed2int(float2fixed(mat.tx)));
387
8
                double x1 = mat.tx + mat.yx * height;
388
8
                fixed ix1 = int2fixed(fixed2int_ceiling(float2fixed(x1)));
389
8
                mat.tx = (double)fixed2float(ix0);
390
8
                mat.yx = (double)(fixed2float(ix1 - ix0)/height);
391
8
            } else if (mat.yx < 0) {
392
3
                fixed ix0 = int2fixed(fixed2int_ceiling(float2fixed(mat.tx)));
393
3
                double x1 = mat.tx + mat.yx * height;
394
3
                fixed ix1 = int2fixed(fixed2int(float2fixed(x1)));
395
3
                mat.tx = (double)fixed2float(ix0);
396
3
                mat.yx = (double)(fixed2float(ix1 - ix0)/height);
397
3
            }
398
11
        }
399
131k
        if (width == 1 || gridfitimages) {
400
3
            if (mat.xy > 0) {
401
0
                fixed iy0 = int2fixed(fixed2int(float2fixed(mat.ty)));
402
0
                double y1 = mat.ty + mat.xy * width;
403
0
                fixed iy1 = int2fixed(fixed2int_ceiling(float2fixed(y1)));
404
0
                mat.ty = (double)fixed2float(iy0);
405
0
                mat.xy = (double)(fixed2float(iy1 - iy0)/width);
406
3
            } else if (mat.xy < 0) {
407
0
                fixed iy0 = int2fixed(fixed2int_ceiling(float2fixed(mat.ty)));
408
0
                double y1 = mat.ty + mat.xy * width;
409
0
                fixed iy1 = int2fixed(fixed2int(float2fixed(y1)));
410
0
                mat.ty = (double)fixed2float(iy0);
411
0
                mat.xy = ((double)fixed2float(iy1 - iy0)/width);
412
0
            }
413
3
        }
414
131k
    }
415
416
    /* When rendering to a pattern accumulator, if we are downscaling
417
     * then enable interpolation, as otherwise dropouts can cause
418
     * serious problems. */
419
1.58M
    if (in_pattern_accumulator) {
420
552
        double ome = ((double)(fixed_1 - fixed_epsilon)) / (double)fixed_1; /* One Minus Epsilon */
421
422
552
        if (orthogonal == 1) {
423
535
            if ((mat.xx > -ome && mat.xx < ome) || (mat.yy > -ome && mat.yy < ome)) {
424
350
                force_interpolation = true;
425
350
            }
426
535
        } else if (orthogonal == 2) {
427
0
            if ((mat.xy > -ome && mat.xy < ome) || (mat.yx > -ome && mat.yx < ome)) {
428
0
                force_interpolation = true;
429
0
            }
430
0
        }
431
552
    }
432
433
    /* Can we restrict the amount of image we need? */
434
1.58M
    while (!pim->imagematrices_are_untrustworthy) /* So we can break out of it */
435
1.58M
    {
436
1.58M
        gs_rect rect, rect_src;
437
1.58M
        gs_matrix mi;
438
1.58M
        const gs_matrix *m = pgs != NULL ? &ctm_only(pgs) : NULL;
439
1.58M
        gs_int_rect irect;
440
1.58M
        if (m == NULL || (code = gs_matrix_invert(m, &mi)) < 0 ||
441
1.57M
            (code = gs_matrix_multiply(&mi, &pic->ImageMatrix, &mi)) < 0) {
442
            /* Give up trying to shrink the render box, but continue processing */
443
7.60k
            break;
444
7.60k
        }
445
1.57M
        if (pcpath)
446
593k
        {
447
593k
            gs_fixed_rect obox;
448
593k
            gx_cpath_outer_box(pcpath, &obox);
449
593k
            rect.p.x = fixed2float(obox.p.x);
450
593k
            rect.p.y = fixed2float(obox.p.y);
451
593k
            rect.q.x = fixed2float(obox.q.x);
452
593k
            rect.q.y = fixed2float(obox.q.y);
453
593k
        }
454
985k
        else
455
985k
        {
456
985k
            rect.p.x = 0;
457
985k
            rect.p.y = 0;
458
985k
            rect.q.x = dev->width;
459
985k
            rect.q.y = dev->height;
460
985k
        }
461
        /* rect is in destination space. Calculate rect_src, in source space. */
462
1.57M
        code = gs_bbox_transform(&rect, &mi, &rect_src);
463
1.57M
        if (code < 0) {
464
            /* Give up trying to shrink the render/decode boxes, but continue processing */
465
0
            break;
466
0
        }
467
        /* Need to expand the region to allow for the fact that the mitchell
468
         * scaler reads multiple pixels in. */
469
        /* If mi.{xx,yy} > 1 then we are downscaling. During downscaling,
470
         * the support increases to ensure that we don't lose pixels contributions
471
         * entirely. */
472
1.57M
        if (pim->Interpolate)
473
36
        {
474
36
            float support = any_abs(mi.xx);
475
36
            int isupport;
476
36
            if (any_abs(mi.yy) > support)
477
0
                support = any_abs(mi.yy);
478
36
            if (any_abs(mi.xy) > support)
479
0
                support = any_abs(mi.xy);
480
36
            if (any_abs(mi.yx) > support)
481
0
                support = any_abs(mi.yx);
482
            /* If upscaling (support < 1) then we need 2 extra lines on each side of the source region
483
             * (2 being the maximum support for mitchell scaling).
484
             * If downscaling, then the number of lines is increased to avoid individual
485
             * contributions dropping out. */
486
36
            isupport = 2; /* Mitchell support. */
487
36
            if (support > 1)
488
1
                isupport = (int)ceil(isupport * support);
489
36
            rect_src.p.x -= isupport;
490
36
            rect_src.p.y -= isupport;
491
36
            rect_src.q.x += isupport;
492
36
            rect_src.q.y += isupport+1; /* +1 is a fudge! */
493
36
        }
494
1.57M
        irect.p.x = (int)floor(rect_src.p.x);
495
1.57M
        irect.p.y = (int)floor(rect_src.p.y);
496
1.57M
        irect.q.x = (int)ceil(rect_src.q.x);
497
1.57M
        irect.q.y = (int)ceil(rect_src.q.y);
498
        /* We therefore only need to render within irect. Restrict rrect to this. */
499
1.57M
        if (penum->rrect.x < irect.p.x) {
500
80.6k
            penum->rrect.w -= irect.p.x - penum->rrect.x;
501
80.6k
            if (penum->rrect.w < 0)
502
71.4k
               penum->rrect.w = 0;
503
80.6k
            penum->rrect.x = irect.p.x;
504
80.6k
        }
505
1.57M
        if (penum->rrect.x + penum->rrect.w > irect.q.x) {
506
186k
            penum->rrect.w = irect.q.x - penum->rrect.x;
507
186k
            if (penum->rrect.w < 0)
508
154k
                penum->rrect.w = 0;
509
186k
        }
510
1.57M
        if (penum->rrect.y < irect.p.y) {
511
182k
            penum->rrect.h -= irect.p.y - penum->rrect.y;
512
182k
            if (penum->rrect.h < 0)
513
96.3k
                penum->rrect.h = 0;
514
182k
            penum->rrect.y = irect.p.y;
515
182k
        }
516
1.57M
        if (penum->rrect.y + penum->rrect.h > irect.q.y) {
517
124k
            penum->rrect.h = irect.q.y - penum->rrect.y;
518
124k
            if (penum->rrect.h < 0)
519
24.1k
                penum->rrect.h = 0;
520
124k
        }
521
1.57M
        if (penum->drect.x < irect.p.x) {
522
80.6k
            penum->drect.w -= irect.p.x - penum->drect.x;
523
80.6k
            if (penum->drect.w < 0)
524
71.4k
               penum->drect.w = 0;
525
80.6k
            penum->drect.x = irect.p.x;
526
80.6k
        }
527
1.57M
        if (penum->drect.x + penum->drect.w > irect.q.x) {
528
186k
            penum->drect.w = irect.q.x - penum->drect.x;
529
186k
            if (penum->drect.w < 0)
530
154k
                penum->drect.w = 0;
531
186k
        }
532
1.57M
        if (penum->drect.y < irect.p.y) {
533
182k
            penum->drect.h -= irect.p.y - penum->drect.y;
534
182k
            if (penum->drect.h < 0)
535
96.3k
                penum->drect.h = 0;
536
182k
            penum->drect.y = irect.p.y;
537
182k
        }
538
1.57M
        if (penum->drect.y + penum->drect.h > irect.q.y) {
539
124k
            penum->drect.h = irect.q.y - penum->drect.y;
540
124k
            if (penum->drect.h < 0)
541
24.1k
                penum->drect.h = 0;
542
124k
        }
543
1.57M
        break; /* Out of the while */
544
1.57M
    }
545
    /* Check for the intersection being null */
546
1.58M
    if (penum->drect.x + penum->drect.w <= penum->rect.x  ||
547
1.42M
        penum->rect.x  + penum->rect.w  <= penum->drect.x ||
548
1.35M
        penum->drect.y + penum->drect.h <= penum->rect.y  ||
549
1.35M
        penum->rect.y  + penum->rect.h  <= penum->drect.y)
550
253k
    {
551
          /* Something may have gone wrong with the floating point above.
552
           * set the region to something sane. */
553
253k
        penum->drect.x = penum->rect.x;
554
253k
        penum->drect.y = penum->rect.y;
555
253k
        penum->drect.w = 0;
556
253k
        penum->drect.h = 0;
557
253k
    }
558
1.58M
    if (penum->rrect.x + penum->rrect.w <= penum->drect.x  ||
559
1.42M
        penum->drect.x + penum->drect.w  <= penum->rrect.x ||
560
1.33M
        penum->rrect.y + penum->rrect.h <= penum->drect.y  ||
561
1.33M
        penum->drect.y + penum->drect.h  <= penum->rrect.y)
562
253k
    {
563
          /* Something may have gone wrong with the floating point above.
564
           * set the region to something sane. */
565
253k
        penum->rrect.x = penum->drect.x;
566
253k
        penum->rrect.y = penum->drect.y;
567
253k
        penum->rrect.w = 0;
568
253k
        penum->rrect.h = 0;
569
253k
    }
570
571
    /*penum->matrix = mat;*/
572
1.58M
    penum->matrix.xx = mat.xx;
573
1.58M
    penum->matrix.xy = mat.xy;
574
1.58M
    penum->matrix.yx = mat.yx;
575
1.58M
    penum->matrix.yy = mat.yy;
576
1.58M
    penum->matrix.tx = mat.tx;
577
1.58M
    penum->matrix.ty = mat.ty;
578
1.58M
    if_debug6m('b', mem, " [%g %g %g %g %g %g]\n",
579
1.58M
              mat.xx, mat.xy, mat.yx, mat.yy, mat.tx, mat.ty);
580
    /* following works for 1, 2, 4, 8, 12, 16 */
581
1.58M
    index_bps = (bps < 8 ? bps >> 1 : (bps >> 2) + 1);
582
    /*
583
     * Compute extents with distance transformation.
584
     */
585
1.58M
    if (mat.tx > 0)
586
1.30M
        mtx = float2fixed(mat.tx);
587
282k
    else { /* Use positive values to ensure round down. */
588
282k
        int f = (int)-mat.tx + 1;
589
590
282k
        mtx = float2fixed(mat.tx + f) - int2fixed(f);
591
282k
    }
592
1.58M
    if (mat.ty > 0)
593
495k
        mty = float2fixed(mat.ty);
594
1.09M
    else {  /* Use positive values to ensure round down. */
595
1.09M
        int f = (int)-mat.ty + 1;
596
597
1.09M
        mty = float2fixed(mat.ty + f) - int2fixed(f);
598
1.09M
    }
599
600
1.58M
    row_extent.x = float2fixed_rounded_boxed(width * mat.xx);
601
1.58M
    row_extent.y =
602
1.58M
        (is_fzero(mat.xy) ? fixed_0 :
603
1.58M
         float2fixed_rounded_boxed(width * mat.xy));
604
1.58M
    col_extent.x =
605
1.58M
        (is_fzero(mat.yx) ? fixed_0 :
606
1.58M
         float2fixed_rounded_boxed(height * mat.yx));
607
1.58M
    col_extent.y = float2fixed_rounded_boxed(height * mat.yy);
608
1.58M
    gx_image_enum_common_init((gx_image_enum_common_t *)penum,
609
1.58M
                              (const gs_data_image_t *)pim,
610
1.58M
                              &image1_enum_procs, dev,
611
1.58M
                              (masked ? 1 : (penum->alpha ? cs_num_components(pcs)+1 : cs_num_components(pcs))),
612
1.58M
                              format);
613
1.58M
    if (penum->rect.w == width && penum->rect.h == height) {
614
1.18M
        x_extent = row_extent;
615
1.18M
        y_extent = col_extent;
616
1.18M
    } else {
617
406k
        int rw = penum->rect.w, rh = penum->rect.h;
618
619
406k
        x_extent.x = float2fixed_rounded_boxed(rw * mat.xx);
620
406k
        x_extent.y =
621
406k
            (is_fzero(mat.xy) ? fixed_0 :
622
406k
             float2fixed_rounded_boxed(rw * mat.xy));
623
406k
        y_extent.x =
624
406k
            (is_fzero(mat.yx) ? fixed_0 :
625
406k
             float2fixed_rounded_boxed(rh * mat.yx));
626
406k
        y_extent.y = float2fixed_rounded_boxed(rh * mat.yy);
627
406k
    }
628
    /* Set icolor0 and icolor1 to point to image clues locations if we have
629
       1spp or an imagemask, otherwise image clues is not used and
630
       we have these values point to other member variables */
631
1.58M
    if (masked || cs_num_components(pcs) == 1) {
632
        /* Go ahead and allocate now if not already done.  For a mask
633
           we really should only do 2 values. For now, the goal is to
634
           eliminate the 256 bytes for the >8bpp image enumerator */
635
864k
        penum->clues = (gx_image_clue*) gs_alloc_bytes(mem, sizeof(gx_image_clue)*256,
636
864k
                             "gx_image_enum_begin");
637
864k
        if (penum->clues == NULL) {
638
0
            code = gs_error_VMerror;
639
0
            goto fail;
640
0
        }
641
864k
        penum->icolor0 = &(penum->clues[0].dev_color);
642
864k
        penum->icolor1 = &(penum->clues[255].dev_color);
643
864k
    } else {
644
721k
        penum->icolor0 = &(penum->icolor0_val);
645
721k
        penum->icolor1 = &(penum->icolor1_val);
646
721k
    }
647
1.58M
    penum->icolor0->tag = penum->icolor1->tag = device_current_tag(dev);
648
649
1.58M
    if (masked) {       /* This is imagemask. */
650
455k
        if (bps != 1 || pcs != NULL || penum->alpha || decode[0] == decode[1]) {
651
54
            code = gs_error_rangecheck;
652
54
            goto fail;
653
54
        }
654
        /* Initialize color entries 0 and 255. */
655
455k
        set_nonclient_dev_color(penum->icolor0, gx_no_color_index);
656
455k
        set_nonclient_dev_color(penum->icolor1, gx_no_color_index);
657
455k
        *(penum->icolor1) = *pdcolor;
658
455k
        memcpy(&penum->map[0].table.lookup4x1to32[0],
659
455k
               (decode[0] < decode[1] ? lookup4x1to32_inverted :
660
455k
                lookup4x1to32_identity),
661
455k
               16 * 4);
662
455k
        penum->map[0].decoding = sd_none;
663
455k
        spp = 1;
664
455k
        lop = rop3_know_S_0(lop);
665
1.13M
    } else {                    /* This is image, not imagemask. */
666
1.13M
        const gs_color_space_type *pcst = pcs->type;
667
1.13M
        int b_w_color;
668
669
1.13M
        spp = cs_num_components(pcs);
670
1.13M
        if (spp < 0) {          /* Pattern not allowed */
671
0
            code = gs_error_rangecheck;
672
0
            goto fail;
673
0
        }
674
1.13M
        if (penum->alpha)
675
0
            ++spp;
676
        /* Use a less expensive format if possible. */
677
1.13M
        switch (format) {
678
0
        case gs_image_format_bit_planar:
679
0
            if (bps > 1)
680
0
                break;
681
0
            format = gs_image_format_component_planar;
682
22.1k
        case gs_image_format_component_planar:
683
22.1k
            if (spp == 1)
684
0
                format = gs_image_format_chunky;
685
1.13M
        default:                /* chunky */
686
1.13M
            break;
687
1.13M
        }
688
689
1.13M
        if (pcs->cmm_icc_profile_data != NULL) {
690
1.11M
            device_color = false;
691
1.11M
        } else {
692
19.2k
            device_color = (*pcst->concrete_space) (pcs, pgs) == pcs;
693
19.2k
        }
694
695
1.13M
        code = image_init_colors(penum, bps, spp, format, decode, pgs, dev,
696
1.13M
                          pcs, &device_color);
697
1.13M
        if (code < 0) {
698
9
            gs_free_object(mem, penum->clues, "gx_image_enum_begin");
699
9
            gs_free_object(mem, penum, "gx_default_begin_image");
700
9
            return gs_throw(code, "Image colors initialization failed");
701
9
        }
702
        /* If we have a CIE based color space and the icc equivalent profile
703
           is not yet set, go ahead and handle that now.  It may already
704
           be done due to the above init_colors which may go through remap. */
705
1.13M
        if (gs_color_space_is_PSCIE(pcs) && pcs->icc_equivalent == NULL) {
706
0
            code = gs_colorspace_set_icc_equivalent((gs_color_space *)pcs, &(penum->icc_setup.is_lab),
707
0
                                                pgs->memory);
708
0
            if (code < 0)
709
0
                goto fail;
710
0
            if (penum->icc_setup.is_lab) {
711
                /* Free what ever profile was created and use the icc manager's
712
                   cielab profile */
713
0
                gs_color_space *curr_pcs = (gs_color_space *)pcs;
714
0
                rc_decrement(curr_pcs->icc_equivalent,"gx_image_enum_begin");
715
0
                gsicc_adjust_profile_rc(curr_pcs->cmm_icc_profile_data, -1,"gx_image_enum_begin");
716
0
                curr_pcs->cmm_icc_profile_data = pgs->icc_manager->lab_profile;
717
0
                gsicc_adjust_profile_rc(curr_pcs->cmm_icc_profile_data, 1,"gx_image_enum_begin");
718
0
            }
719
0
        }
720
        /* Try to transform non-default RasterOps to something */
721
        /* that we implement less expensively. */
722
1.13M
        if (!pim->CombineWithColor)
723
1.13M
            lop = rop3_know_T_0(lop);
724
0
        else if ((rop3_uses_T(lop) && color_draws_b_w(dev, pdcolor) == 0))
725
0
            lop = rop3_know_T_0(lop);
726
727
1.13M
        if (lop != rop3_S &&    /* if best case, no more work needed */
728
1.13M
            !rop3_uses_T(lop) && bps == 1 && spp == 1 &&
729
0
            (b_w_color =
730
0
             color_draws_b_w(dev, penum->icolor0)) >= 0 &&
731
0
            color_draws_b_w(dev, penum->icolor1) == (b_w_color ^ 1)
732
1.13M
            ) {
733
0
            if (b_w_color) {    /* Swap the colors and invert the RasterOp source. */
734
0
                gx_device_color dcolor;
735
736
0
                dcolor = *(penum->icolor0);
737
0
                *(penum->icolor0) = *(penum->icolor1);
738
0
                *(penum->icolor1) = dcolor;
739
0
                lop = rop3_invert_S(lop);
740
0
            }
741
            /*
742
             * At this point, we know that the source pixels
743
             * correspond directly to the S input for the raster op,
744
             * i.e., icolor0 is black and icolor1 is white.
745
             */
746
0
            switch (lop) {
747
0
                case rop3_D & rop3_S:
748
                    /* Implement this as an inverted mask writing 0s. */
749
0
                    *(penum->icolor1) = *(penum->icolor0);
750
                    /* (falls through) */
751
0
                case rop3_D | rop3_not(rop3_S):
752
                    /* Implement this as an inverted mask writing 1s. */
753
0
                    memcpy(&penum->map[0].table.lookup4x1to32[0],
754
0
                           lookup4x1to32_inverted, 16 * 4);
755
0
                  rmask:        /* Fill in the remaining parameters for a mask. */
756
0
                    penum->masked = masked = true;
757
0
                    set_nonclient_dev_color(penum->icolor0, gx_no_color_index);
758
0
                    penum->map[0].decoding = sd_none;
759
0
                    lop = rop3_T;
760
0
                    break;
761
0
                case rop3_D & rop3_not(rop3_S):
762
                    /* Implement this as a mask writing 0s. */
763
0
                    *(penum->icolor1) = *(penum->icolor0);
764
                    /* (falls through) */
765
0
                case rop3_D | rop3_S:
766
                    /* Implement this as a mask writing 1s. */
767
0
                    memcpy(&penum->map[0].table.lookup4x1to32[0],
768
0
                           lookup4x1to32_identity, 16 * 4);
769
0
                    goto rmask;
770
0
                default:
771
0
                    ;
772
0
            }
773
0
        }
774
1.13M
    }
775
1.58M
    penum->device_color = device_color;
776
    /*
777
     * Adjust width upward for unpacking up to 7 trailing bits in
778
     * the row, plus 1 byte for end-of-run, plus up to 7 leading
779
     * bits for data_x offset within a packed byte.
780
     */
781
1.58M
    bsize = ((bps > 8 ? width * 2 : width) + 15) * spp;
782
1.58M
    buffer = gs_alloc_bytes(mem, bsize, "image buffer");
783
1.58M
    if (buffer == 0) {
784
7
        code = gs_error_VMerror;
785
7
        goto fail;
786
7
    }
787
1.58M
    penum->bps = bps;
788
1.58M
    penum->unpack_bps = bps;
789
1.58M
    penum->log2_xbytes = log2_xbytes;
790
1.58M
    penum->spp = spp;
791
1.58M
    switch (format) {
792
1.56M
    case gs_image_format_chunky:
793
1.56M
        nplanes = 1;
794
1.56M
        spread = 1 << log2_xbytes;
795
1.56M
        break;
796
22.1k
    case gs_image_format_component_planar:
797
22.1k
        nplanes = spp;
798
22.1k
        spread = spp << log2_xbytes;
799
22.1k
        break;
800
0
    case gs_image_format_bit_planar:
801
0
        nplanes = spp * bps;
802
0
        spread = spp << log2_xbytes;
803
0
        break;
804
0
    default:
805
        /* No other cases are possible (checked by gx_image_enum_alloc). */
806
0
        return_error(gs_error_Fatal);
807
1.58M
    }
808
1.58M
    penum->num_planes = nplanes;
809
1.58M
    penum->spread = spread;
810
    /*
811
     * If we're asked to interpolate in a partial image, we have to
812
     * assume that the client either really only is interested in
813
     * the given sub-image, or else is constructing output out of
814
     * overlapping pieces.
815
     */
816
1.58M
    penum->interpolate = force_interpolation ? interp_force : pim->Interpolate ? interp_on : interp_off;
817
1.58M
    penum->x_extent = x_extent;
818
1.58M
    penum->y_extent = y_extent;
819
1.58M
    penum->posture =
820
1.58M
        ((x_extent.y | y_extent.x) == 0 ? image_portrait :
821
1.58M
         (x_extent.x | y_extent.y) == 0 ? image_landscape :
822
232k
         image_skewed);
823
1.58M
    penum->pgs = pgs;
824
1.58M
    if (pgs != NULL)
825
1.58M
        penum->pgs_level = pgs->level;
826
1.58M
    penum->pcs = pcs;
827
1.58M
    rc_increment_cs(pcs); /* Grab a ref (will decrement in gx_image1_end_image() */
828
1.58M
    penum->memory = mem;
829
1.58M
    penum->buffer = buffer;
830
1.58M
    penum->buffer_size = bsize;
831
1.58M
    penum->line = NULL;
832
1.58M
    penum->icc_link = NULL;
833
1.58M
    penum->color_cache = NULL;
834
1.58M
    penum->ht_buffer = NULL;
835
1.58M
    penum->thresh_buffer = NULL;
836
1.58M
    penum->use_cie_range = false;
837
1.58M
    penum->line_size = 0;
838
1.58M
    penum->use_rop = lop != (masked ? rop3_T : rop3_S);
839
#ifdef DEBUG
840
    if (gs_debug_c('*')) {
841
        if (penum->use_rop)
842
            dmprintf1(mem, "[%03x]", lop);
843
        dmprintf5(mem, "%c%d%c%dx%d ",
844
                 (masked ? (color_is_pure(pdcolor) ? 'm' : 'h') : 'i'),
845
                 bps,
846
                 (penum->posture == image_portrait ? ' ' :
847
                  penum->posture == image_landscape ? 'L' : 'T'),
848
                 width, height);
849
    }
850
#endif
851
1.58M
    penum->slow_loop = 0;
852
1.58M
    if (pcpath == 0) {
853
985k
        (*dev_proc(dev, get_clipping_box)) (dev, &obox);
854
985k
        cbox = obox;
855
985k
        penum->clip_image = 0;
856
985k
    } else
857
600k
        penum->clip_image =
858
600k
            (gx_cpath_outer_box(pcpath, &obox) |        /* not || */
859
600k
             gx_cpath_inner_box(pcpath, &cbox) ?
860
598k
             0 : image_clip_region);
861
1.58M
    penum->clip_outer = obox;
862
1.58M
    penum->clip_inner = cbox;
863
1.58M
    penum->log_op = rop3_T;     /* rop device takes care of this */
864
1.58M
    penum->clip_dev = 0;        /* in case we bail out */
865
1.58M
    penum->rop_dev = 0;         /* ditto */
866
1.58M
    penum->scaler = 0;          /* ditto */
867
    /*
868
     * If all four extrema of the image fall within the clipping
869
     * rectangle, clipping is never required.  When making this check,
870
     * we must carefully take into account the fact that we only care
871
     * about pixel centers.
872
     */
873
1.58M
    {
874
1.58M
        fixed
875
1.58M
            epx = min(row_extent.x, 0) + min(col_extent.x, 0),
876
1.58M
            eqx = max(row_extent.x, 0) + max(col_extent.x, 0),
877
1.58M
            epy = min(row_extent.y, 0) + min(col_extent.y, 0),
878
1.58M
            eqy = max(row_extent.y, 0) + max(col_extent.y, 0);
879
880
1.58M
        {
881
1.58M
            int hwx, hwy;
882
883
1.58M
            switch (penum->posture) {
884
1.35M
                case image_portrait:
885
1.35M
                    hwx = width, hwy = height;
886
1.35M
                    break;
887
133k
                case image_landscape:
888
133k
                    hwx = height, hwy = width;
889
133k
                    break;
890
99.2k
                default:
891
99.2k
                    hwx = hwy = 0;
892
1.58M
            }
893
            /*
894
             * If the image is only 1 sample wide or high,
895
             * and is less than 1 device pixel wide or high,
896
             * move it slightly so that it covers pixel centers.
897
             * This is a hack to work around a bug in some old
898
             * versions of TeX/dvips, which use 1-bit-high images
899
             * to draw horizontal and vertical lines without
900
             * positioning them properly.
901
             */
902
1.58M
            if (hwx == 1 && eqx - epx < fixed_1) {
903
22
                fixed diff =
904
22
                arith_rshift_1(row_extent.x + col_extent.x);
905
906
22
                mtx = (((mtx + diff) | fixed_half) & -fixed_half) - diff;
907
22
            }
908
1.58M
            if (hwy == 1 && eqy - epy < fixed_1) {
909
18
                fixed diff =
910
18
                arith_rshift_1(row_extent.y + col_extent.y);
911
912
18
                mty = (((mty + diff) | fixed_half) & -fixed_half) - diff;
913
18
            }
914
1.58M
        }
915
1.58M
        if_debug5m('b', mem, "[b]Image: %sspp=%d, bps=%d, mt=(%g,%g)\n",
916
1.58M
                   (masked? "masked, " : ""), spp, bps,
917
1.58M
                   fixed2float(mtx), fixed2float(mty));
918
1.58M
        if_debug9m('b', mem,
919
1.58M
                   "[b]   cbox=(%g,%g),(%g,%g), obox=(%g,%g),(%g,%g), clip_image=0x%x\n",
920
1.58M
                   fixed2float(cbox.p.x), fixed2float(cbox.p.y),
921
1.58M
                   fixed2float(cbox.q.x), fixed2float(cbox.q.y),
922
1.58M
                   fixed2float(obox.p.x), fixed2float(obox.p.y),
923
1.58M
                   fixed2float(obox.q.x), fixed2float(obox.q.y),
924
1.58M
                   penum->clip_image);
925
        /* These DDAs enumerate the starting position of each source pixel
926
         * row in device space. */
927
1.58M
        dda_init(penum->dda.row.x, mtx, col_extent.x, height);
928
1.58M
        dda_init(penum->dda.row.y, mty, col_extent.y, height);
929
1.58M
        if (dda_will_overflow(penum->dda.row.x) ||
930
1.58M
            dda_will_overflow(penum->dda.row.y))
931
0
        {
932
0
            code = gs_error_rangecheck;
933
0
            goto fail;
934
0
        }
935
1.58M
        if (penum->posture == image_portrait) {
936
1.35M
            penum->dst_width = row_extent.x;
937
1.35M
            penum->dst_height = col_extent.y;
938
1.35M
        } else {
939
232k
            penum->dst_width = col_extent.x;
940
232k
            penum->dst_height = row_extent.y;
941
232k
        }
942
        /* For gs_image_class_0_interpolate. */
943
1.58M
        penum->yi0 = fixed2int_pixround_perfect(dda_current(penum->dda.row.y)); /* For gs_image_class_0_interpolate. */
944
1.58M
        if (penum->rect.y) {
945
351k
            int y = penum->rect.y;
946
947
41.8M
            while (y--) {
948
41.4M
                dda_next(penum->dda.row.x);
949
41.4M
                dda_next(penum->dda.row.y);
950
41.4M
            }
951
351k
        }
952
1.58M
        penum->cur.x = penum->prev.x = dda_current(penum->dda.row.x);
953
1.58M
        penum->cur.y = penum->prev.y = dda_current(penum->dda.row.y);
954
        /* These DDAs enumerate the starting positions of each row of our
955
         * source pixel data, in the subrectangle ('strip') that we are
956
         * actually rendering. */
957
1.58M
        dda_init(penum->dda.strip.x, penum->cur.x, row_extent.x, width);
958
1.58M
        dda_init(penum->dda.strip.y, penum->cur.y, row_extent.y, width);
959
1.58M
        if (dda_will_overflow(penum->dda.strip.x) ||
960
1.58M
            dda_will_overflow(penum->dda.strip.y))
961
0
        {
962
0
            code = gs_error_rangecheck;
963
0
            goto fail;
964
0
        }
965
1.58M
        if (penum->rect.x) {
966
8.56k
            dda_advance(penum->dda.strip.x, penum->rect.x);
967
8.56k
            dda_advance(penum->dda.strip.y, penum->rect.x);
968
8.56k
        }
969
1.58M
        {
970
1.58M
            fixed ox = dda_current(penum->dda.strip.x);
971
1.58M
            fixed oy = dda_current(penum->dda.strip.y);
972
973
1.58M
            if (!penum->clip_image)     /* i.e., not clip region */
974
1.58M
                penum->clip_image =
975
1.58M
                    (fixed_pixround(ox + epx) < fixed_pixround(cbox.p.x) ?
976
1.48M
                     image_clip_xmin : 0) +
977
1.58M
                    (fixed_pixround(ox + eqx) >= fixed_pixround(cbox.q.x) ?
978
1.16M
                     image_clip_xmax : 0) +
979
1.58M
                    (fixed_pixround(oy + epy) < fixed_pixround(cbox.p.y) ?
980
1.20M
                     image_clip_ymin : 0) +
981
1.58M
                    (fixed_pixround(oy + eqy) >= fixed_pixround(cbox.q.y) ?
982
1.17M
                     image_clip_ymax : 0);
983
1.58M
        }
984
1.58M
    }
985
0
    penum->y = 0;
986
1.58M
    penum->used.x = 0;
987
1.58M
    penum->used.y = 0;
988
1.58M
    if (penum->clip_image && pcpath) {  /* Set up the clipping device. */
989
419k
        gx_device_clip *cdev =
990
419k
            gs_alloc_struct(mem, gx_device_clip,
991
419k
                            &st_device_clip, "image clipper");
992
993
419k
        if (cdev == NULL) {
994
0
            code = gs_error_VMerror;
995
0
            goto fail;
996
0
        }
997
419k
        gx_make_clip_device_in_heap(cdev, pcpath, dev, mem);
998
419k
        penum->clip_dev = cdev;
999
419k
        penum->dev = (gx_device *)cdev; /* Will restore this in a mo. Hacky! */
1000
419k
    }
1001
1.58M
    if (penum->use_rop) {       /* Set up the RasterOp source device. */
1002
0
        gx_device_rop_texture *rtdev;
1003
1004
0
        code = gx_alloc_rop_texture_device(&rtdev, mem,
1005
0
                                           "image RasterOp");
1006
0
        if (code < 0)
1007
0
            goto fail;
1008
        /* The 'target' must not be NULL for gx_make_rop_texture_device */
1009
0
        if (!penum->clip_dev && !dev)
1010
0
            return_error(gs_error_undefined);
1011
1012
0
        gx_make_rop_texture_device(rtdev,
1013
0
                                   (penum->clip_dev != 0 ?
1014
0
                                    (gx_device *) penum->clip_dev :
1015
0
                                    dev), lop, pdcolor);
1016
0
        gx_device_retain((gx_device *)rtdev, true);
1017
0
        penum->rop_dev = rtdev;
1018
0
        penum->dev = (gx_device *)rtdev; /* Will restore this in a mo. Hacky! */
1019
0
    }
1020
1.58M
    {
1021
1.58M
        static sample_unpack_proc_t procs[2][6] = {
1022
1.58M
        {   sample_unpack_1, sample_unpack_2,
1023
1.58M
            sample_unpack_4, sample_unpack_8,
1024
1.58M
            sample_unpack_12, sample_unpack_16
1025
1.58M
        },
1026
1.58M
        {   sample_unpack_1_interleaved, sample_unpack_2_interleaved,
1027
1.58M
            sample_unpack_4_interleaved, sample_unpack_8_interleaved,
1028
1.58M
            sample_unpack_12, sample_unpack_16
1029
1.58M
        }};
1030
1.58M
        int num_planes = penum->num_planes;
1031
1.58M
        bool interleaved = (num_planes == 1 && penum->plane_depths[0] != penum->bps);
1032
1.58M
        irender_proc_t render_fn = NULL;
1033
1.58M
        int i;
1034
1035
1.58M
        if (interleaved) {
1036
699k
            int num_components = penum->plane_depths[0] / penum->bps;
1037
1038
2.10M
            for (i = 1; i < num_components; i++) {
1039
1.40M
                if (decode[0] != decode[i * 2 + 0] ||
1040
1.40M
                    decode[1] != decode[i * 2 + 1])
1041
16
                    break;
1042
1.40M
            }
1043
699k
            if (i == num_components)
1044
699k
                interleaved = false; /* Use single table. */
1045
699k
        }
1046
1.58M
        penum->unpack = procs[interleaved][index_bps];
1047
1048
1.58M
        if_debug1m('b', mem, "[b]unpack=%d\n", bps);
1049
        /* Set up pixel0 for image class procedures. */
1050
1.58M
        penum->dda.pixel0 = penum->dda.strip;
1051
1.58M
        penum->skip_next_line = NULL;
1052
6.39M
        for (i = 0; i < gx_image_class_table_count; ++i) {
1053
6.39M
            code = gx_image_class_table[i](penum, &render_fn);
1054
6.39M
            if (code < 0)
1055
0
                goto fail;
1056
1057
6.39M
            if (render_fn != NULL) {
1058
1.58M
                penum->render = render_fn;
1059
1.58M
                break;
1060
1.58M
            }
1061
6.39M
        }
1062
1.58M
        penum->dev = dev; /* Restore this (in case it was changed to cdev or rtdev) */
1063
1.58M
        if (i == gx_image_class_table_count) {
1064
            /* No available class can handle this image. */
1065
0
            return_error(gs_error_rangecheck);
1066
0
        }
1067
1.58M
    }
1068
1.58M
    return 0;
1069
1070
61
fail:
1071
61
    gs_free_object(mem, buffer, "image buffer");
1072
61
    gs_free_object(mem, penum->clues, "gx_image_enum_begin");
1073
61
    if (penum->clip_dev != NULL) {
1074
0
        rc_decrement(penum->clip_dev, "error in gx_begin_image1");
1075
0
        penum->clip_dev = NULL;
1076
0
    }
1077
61
    gs_free_object(mem, penum->clip_dev, "image clipper");
1078
61
    rc_decrement_cs(penum->pcs, "error in gx_begin_image1");
1079
61
    penum->pcs = NULL;
1080
61
    gs_free_object(mem, penum, "gx_begin_image1");
1081
61
    return code;
1082
1.58M
}
1083
1084
/* If a drawing color is black or white, return 0 or 1 respectively, */
1085
/* otherwise return -1. */
1086
static int
1087
color_draws_b_w(gx_device * dev, const gx_drawing_color * pdcolor)
1088
0
{
1089
0
    if (color_is_pure(pdcolor)) {
1090
0
        gx_color_value rgb[3];
1091
1092
0
        (*dev_proc(dev, map_color_rgb)) (dev, gx_dc_pure_color(pdcolor),
1093
0
                                         rgb);
1094
0
        if (!(rgb[0] | rgb[1] | rgb[2]))
1095
0
            return 0;
1096
0
        if ((rgb[0] & rgb[1] & rgb[2]) == gx_max_color_value)
1097
0
            return 1;
1098
0
    }
1099
0
    return -1;
1100
0
}
1101
1102
1103
static void
1104
image_cache_decode(gx_image_enum *penum, byte input, byte *output, bool scale)
1105
0
{
1106
0
    float temp;
1107
1108
0
    switch ( penum->map[0].decoding ) {
1109
0
        case sd_none:
1110
0
            *output = input;
1111
0
            break;
1112
0
        case sd_lookup:
1113
0
            temp = penum->map[0].decode_lookup[input >> 4]*255.0f;
1114
0
            if (temp > 255) temp = 255;
1115
0
            if (temp < 0 ) temp = 0;
1116
0
            *output = (unsigned char) temp;
1117
0
            break;
1118
0
        case sd_compute:
1119
0
            temp = penum->map[0].decode_base +
1120
0
                (float) input * penum->map[0].decode_factor;
1121
0
            if (scale) {
1122
0
                temp = temp * 255.0;
1123
0
            }
1124
0
            if (temp > 255) temp = 255;
1125
0
            if (temp < 0 ) temp = 0;
1126
0
            *output = (unsigned char) temp;
1127
0
            break;
1128
0
        default:
1129
0
            *output = 0;
1130
0
            break;
1131
0
    }
1132
0
}
1133
1134
static bool
1135
decode_range_needed(gx_image_enum *penum)
1136
0
{
1137
0
    bool scale = true;
1138
1139
0
    if (penum->map[0].decoding == sd_compute) {
1140
0
        if (!(gs_color_space_is_ICC(penum->pcs) ||
1141
0
            gs_color_space_is_PSCIE(penum->pcs))) {
1142
0
            scale = false;
1143
0
        }
1144
0
    }
1145
0
    return scale;
1146
0
}
1147
1148
/* A special case where we go ahead and initialize the whole index cache with
1149
   contone.  Device colors.  If we are halftoning we will then go ahead and
1150
   apply the thresholds to the device contone values.  Only used for gray,
1151
   rgb or cmyk source colors (No DeviceN for now) */
1152
/* TO DO  Add in PSCIE decoder */
1153
int
1154
image_init_color_cache(gx_image_enum * penum, int bps, int spp)
1155
79.3k
{
1156
79.3k
    int num_des_comp = penum->dev->color_info.num_components;
1157
79.3k
    int num_src_comp;
1158
79.3k
    int num_entries = 1 << bps;
1159
79.3k
    bool need_decode = penum->icc_setup.need_decode;
1160
79.3k
    bool has_transfer = penum->icc_setup.has_transfer;
1161
79.3k
    byte value;
1162
79.3k
    bool decode_scale = true;
1163
79.3k
    int k, kk;
1164
79.3k
    byte psrc[4];
1165
79.3k
    byte *temp_buffer;
1166
79.3k
    byte *byte_ptr;
1167
79.3k
    bool is_indexed = (gs_color_space_get_index(penum->pcs) ==
1168
79.3k
                                            gs_color_space_index_Indexed);
1169
79.3k
    bool free_temp_buffer = true;
1170
79.3k
    gsicc_bufferdesc_t input_buff_desc;
1171
79.3k
    gsicc_bufferdesc_t output_buff_desc;
1172
79.3k
    gx_color_value conc[GX_DEVICE_COLOR_MAX_COMPONENTS];
1173
79.3k
    int code;
1174
1175
79.3k
    if (penum->icc_link == NULL) {
1176
0
        return gs_rethrow(-1, "ICC Link not created during image render color");
1177
0
    }
1178
79.3k
    if (is_indexed) {
1179
1.52k
        num_src_comp = gs_color_space_num_components(penum->pcs->base_space);
1180
77.7k
    } else {
1181
        /* Detect case where cache is not needed.  Colors are already in the
1182
           device space.  Need to fast track this one and halftone row directly.
1183
           Detected in gximono.c by looking if penum->color_cache is NULL */
1184
77.7k
        if (penum->icc_link->is_identity && !need_decode && !has_transfer) {
1185
77.5k
            return 0;
1186
77.5k
        }
1187
251
        num_src_comp = 1;
1188
251
    }
1189
    /* Allocate cache of device contone values */
1190
1.77k
    penum->color_cache = gs_alloc_struct(penum->memory, gx_image_color_cache_t,
1191
1.77k
                                         &st_color_cache,
1192
1.77k
                                         "image_init_color_cache");
1193
1.77k
    if (penum->color_cache == NULL)
1194
0
        return_error(gs_error_VMerror);
1195
1196
1.77k
    penum->color_cache->device_contone = (byte*) gs_alloc_bytes(penum->memory,
1197
1.77k
                   (size_t)num_des_comp * num_entries * sizeof(byte), "image_init_color_cache");
1198
1.77k
    penum->color_cache->is_transparent = (bool*) gs_alloc_bytes(penum->memory,
1199
1.77k
             (size_t)num_entries * sizeof(bool), "image_init_color_cache");
1200
1.77k
    if (penum->color_cache->device_contone == NULL || penum->color_cache->is_transparent == NULL) {
1201
0
        gs_free_object(penum->memory, penum->color_cache->device_contone, "image_init_color_cache");
1202
0
        gs_free_object(penum->memory, penum->color_cache->is_transparent, "image_init_color_cache");
1203
0
        gs_free_object(penum->memory, penum->color_cache, "image_init_color_cache");
1204
0
        penum->color_cache = NULL;
1205
0
        return_error(gs_error_VMerror);
1206
0
    }
1207
    /* Initialize */
1208
1.77k
    memset(penum->color_cache->is_transparent,0,num_entries * sizeof(bool));
1209
    /* Depending upon if we need decode and ICC CM, fill the cache a couple
1210
       different ways. If the link is the identity, then we don't need to do any
1211
       color conversions except for potentially a decode.  This is written in
1212
       the manner shown below so that the common case of no decode and indexed
1213
       image with a look-up-table uses the table data directly or does as many
1214
       operations with memcpy as we can */
1215
    /* Need to check the decode output range so we know how we need to scale.
1216
       We want 8 bit output */
1217
1.77k
    if (need_decode) {
1218
0
        decode_scale = decode_range_needed(penum);
1219
0
    }
1220
1.77k
    if (penum->icc_link->is_identity) {
1221
        /* No CM needed.  */
1222
0
        if (need_decode || has_transfer) {
1223
            /* Slower case.  This could be sped up later to avoid the tests
1224
               within the loop by use of specialized loops.  */
1225
0
            for (k = 0; k < num_entries; k++) {
1226
                /* Data is in k */
1227
0
                if (need_decode) {
1228
0
                    image_cache_decode(penum, k, &value, decode_scale);
1229
0
                } else {
1230
0
                    value = k;
1231
0
                }
1232
                /* Data is in value */
1233
0
                if (is_indexed) {
1234
0
                    gs_cspace_indexed_lookup_bytes(penum->pcs, value, psrc);
1235
0
                } else {
1236
0
                    psrc[0] = value;
1237
0
                }
1238
                /* Data is in psrc */
1239
                /* These silly transforms need to go away. ToDo. */
1240
0
                if (has_transfer) {
1241
0
                    for (kk = 0; kk < num_des_comp; kk++) {
1242
0
                        conc[kk] = gx_color_value_from_byte(psrc[kk]);
1243
0
                    }
1244
0
                    cmap_transfer(&(conc[0]), penum->pgs, penum->dev);
1245
0
                    for (kk = 0; kk < num_des_comp; kk++) {
1246
0
                        psrc[kk] = gx_color_value_to_byte(conc[kk]);
1247
0
                    }
1248
0
                }
1249
0
                memcpy(&(penum->color_cache->device_contone[k * num_des_comp]),
1250
0
                               psrc, num_des_comp);
1251
0
            }
1252
0
        } else {
1253
            /* Indexing only.  No CM, decode or transfer functions. */
1254
0
            for (k = 0; k < num_entries; k++) {
1255
0
                gs_cspace_indexed_lookup_bytes(penum->pcs, (float)k, psrc);
1256
0
                memcpy(&(penum->color_cache->device_contone[k * num_des_comp]),
1257
0
                           psrc, num_des_comp);
1258
0
            }
1259
0
        }
1260
1.77k
    } else {
1261
        /* Need CM */
1262
        /* We need to worry about if the source is indexed and if we need
1263
           to decode first.  Then we can apply CM. Create a temp buffer in
1264
           the source space and then transform it with one call */
1265
1.77k
        temp_buffer = (byte*) gs_alloc_bytes(penum->memory,
1266
1.77k
                                             (size_t)num_entries * num_src_comp,
1267
1.77k
                                             "image_init_color_cache");
1268
1.77k
        if (temp_buffer == NULL)
1269
0
            return_error(gs_error_VMerror);
1270
1271
1.77k
        if (need_decode) {
1272
0
            if (is_indexed) {
1273
                /* Decode and lookup in index */
1274
0
                for (k = 0; k < num_entries; k++) {
1275
0
                    image_cache_decode(penum, k, &value, decode_scale);
1276
0
                    gs_cspace_indexed_lookup_bytes(penum->pcs, value, psrc);
1277
0
                    memcpy(&(temp_buffer[k * num_src_comp]), psrc, num_src_comp);
1278
0
                }
1279
0
            } else {
1280
                /* Decode only */
1281
0
                for (k = 0; k < num_entries; k++) {
1282
0
                    image_cache_decode(penum, k, &(temp_buffer[k]), decode_scale);
1283
0
                }
1284
0
            }
1285
1.77k
        } else {
1286
            /* No Decode */
1287
1.77k
            if (is_indexed) {
1288
                /* If index uses a num_entries sized table then just use its pointer */
1289
1.52k
                if (penum->pcs->params.indexed.use_proc ||
1290
1.52k
                    penum->pcs->params.indexed.hival < (num_entries - 1)) {
1291
                    /* Have to do the slow way */
1292
6.12k
                    for (k = 0; k <= penum->pcs->params.indexed.hival; k++) {
1293
5.96k
                        gs_cspace_indexed_lookup_bytes(penum->pcs, (float)k, psrc);
1294
5.96k
                        memcpy(&(temp_buffer[k * num_src_comp]), psrc, num_src_comp);
1295
5.96k
                    }
1296
                    /* just use psrc results from converting 'hival' to fill the remaining slots */
1297
34.3k
                    for (; k < num_entries; k++) {
1298
34.2k
                        memcpy(&(temp_buffer[k * num_src_comp]), psrc, num_src_comp);
1299
34.2k
                    }
1300
1.37k
                } else {
1301
                    /* Use the index table directly. */
1302
1.37k
                    gs_free_object(penum->memory, temp_buffer, "image_init_color_cache");
1303
1.37k
                    free_temp_buffer = false;
1304
1.37k
                    temp_buffer = (byte *)(penum->pcs->params.indexed.lookup.table.data);
1305
1.37k
                }
1306
1.52k
            } else {
1307
                /* CM only */
1308
64.5k
                for (k = 0; k < num_entries; k++) {
1309
64.2k
                    temp_buffer[k] = k;
1310
64.2k
                }
1311
251
            }
1312
1.77k
        }
1313
        /* Set up the buffer descriptors. */
1314
1.77k
        gsicc_init_buffer(&input_buff_desc, num_src_comp, 1, false, false, false,
1315
1.77k
                          0, num_entries * num_src_comp, 1, num_entries);
1316
1.77k
        gsicc_init_buffer(&output_buff_desc, num_des_comp, 1, false, false, false,
1317
1.77k
                          0, num_entries * num_des_comp,
1318
1.77k
                      1, num_entries);
1319
1.77k
        code = (penum->icc_link->procs.map_buffer)(penum->dev, penum->icc_link,
1320
1.77k
                                            &input_buff_desc, &output_buff_desc,
1321
1.77k
                                            (void*) temp_buffer,
1322
1.77k
                                            (void*) penum->color_cache->device_contone);
1323
1.77k
        if (code < 0)
1324
0
            return gs_rethrow(code, "Failure to map color buffer");
1325
1326
        /* Check if we need to apply any transfer functions.  If so then do it now */
1327
1.77k
        if (has_transfer) {
1328
0
            for (k = 0; k < num_entries; k++) {
1329
0
                byte_ptr =
1330
0
                    &(penum->color_cache->device_contone[k * num_des_comp]);
1331
0
                for (kk = 0; kk < num_des_comp; kk++) {
1332
0
                    conc[kk] = gx_color_value_from_byte(byte_ptr[kk]);
1333
0
                }
1334
0
                cmap_transfer(&(conc[0]), penum->pgs, penum->dev);
1335
0
                for (kk = 0; kk < num_des_comp; kk++) {
1336
0
                    byte_ptr[kk] = gx_color_value_to_byte(conc[kk]);
1337
0
                }
1338
0
            }
1339
0
        }
1340
1.77k
        if (free_temp_buffer)
1341
408
            gs_free_object(penum->memory, temp_buffer, "image_init_color_cache");
1342
1.77k
    }
1343
1.77k
    return 0;
1344
1.77k
}
1345
1346
/* Export this for use by image_render_ functions */
1347
void
1348
image_init_clues(gx_image_enum * penum, int bps, int spp)
1349
858k
{
1350
    /* Initialize the color table */
1351
858k
#define ictype(i)\
1352
858k
  penum->clues[i].dev_color.type
1353
1354
858k
    switch ((spp == 1 ? bps : 8)) {
1355
704k
        case 8:         /* includes all color images */
1356
704k
            {
1357
704k
                register gx_image_clue *pcht = &penum->clues[0];
1358
704k
                register int n = 64;    /* 8 bits means 256 clues, do   */
1359
                                        /* 4 at a time for efficiency   */
1360
45.0M
                do {
1361
45.0M
                    pcht[0].dev_color.type =
1362
45.0M
                        pcht[1].dev_color.type =
1363
45.0M
                        pcht[2].dev_color.type =
1364
45.0M
                        pcht[3].dev_color.type =
1365
45.0M
                        gx_dc_type_none;
1366
45.0M
                    pcht[0].key = pcht[1].key =
1367
45.0M
                        pcht[2].key = pcht[3].key = 0;
1368
45.0M
                    pcht += 4;
1369
45.0M
                }
1370
45.0M
                while (--n > 0);
1371
704k
                penum->clues[0].key = 1;        /* guarantee no hit */
1372
704k
                break;
1373
0
            }
1374
700
        case 4:
1375
700
            ictype(17) = ictype(2 * 17) = ictype(3 * 17) =
1376
700
                ictype(4 * 17) = ictype(6 * 17) = ictype(7 * 17) =
1377
700
                ictype(8 * 17) = ictype(9 * 17) = ictype(11 * 17) =
1378
700
                ictype(12 * 17) = ictype(13 * 17) = ictype(14 * 17) =
1379
700
                gx_dc_type_none;
1380
            /* falls through */
1381
1.75k
        case 2:
1382
1.75k
            ictype(5 * 17) = ictype(10 * 17) = gx_dc_type_none;
1383
858k
#undef ictype
1384
858k
    }
1385
858k
}
1386
1387
/* Initialize the color mapping tables for a non-mask image. */
1388
static int
1389
image_init_colors(gx_image_enum * penum, int bps, int spp,
1390
                  gs_image_format_t format, const float *decode /*[spp*2] */ ,
1391
                  const gs_gstate * pgs, gx_device * dev,
1392
                  const gs_color_space * pcs, bool * pdcb)
1393
1.13M
{
1394
1.13M
    int ci, decode_type, code;
1395
1.13M
    static const float default_decode[] = {
1396
1.13M
        0.0, 1.0, 0.0, 1.0, 0.0, 1.0, 0.0, 1.0, 0.0, 1.0
1397
1.13M
    };
1398
1399
    /* Clues are only used with image_mono_render */
1400
1.13M
    if (spp == 1) {
1401
409k
        image_init_clues(penum, bps, spp);
1402
409k
    }
1403
1.13M
    decode_type = 3; /* 0=custom, 1=identity, 2=inverted, 3=impossible */
1404
2.98M
    for (ci = 0; ci < spp; ci +=2 ) {
1405
1.85M
        decode_type &= (decode[ci] == 0. && decode[ci + 1] == 1.) |
1406
1.85M
                       (decode[ci] == 1. && decode[ci + 1] == 0.) << 1;
1407
1.85M
    }
1408
1409
    /* Initialize the maps from samples to intensities. */
1410
3.73M
    for (ci = 0; ci < spp; ci++) {
1411
2.60M
        sample_map *pmap = &penum->map[ci];
1412
1413
        /* If the decoding is [0 1] or [1 0], we can fold it */
1414
        /* into the expansion of the sample values; */
1415
        /* otherwise, we have to use the floating point method. */
1416
1417
2.60M
        const float *this_decode = &decode[ci * 2];
1418
2.60M
        const float *map_decode;        /* decoding used to */
1419
                                        /* construct the expansion map */
1420
2.60M
        const float *real_decode;       /* decoding for expanded samples */
1421
1422
2.60M
        map_decode = real_decode = this_decode;
1423
2.60M
        if (!(decode_type & 1)) {
1424
5.59k
            if ((decode_type & 2) && bps <= 8) {
1425
665
                real_decode = default_decode;
1426
4.92k
            } else {
1427
4.92k
                *pdcb = false;
1428
4.92k
                map_decode = default_decode;
1429
4.92k
            }
1430
5.59k
        }
1431
2.60M
        if (bps > 2 || format != gs_image_format_chunky) {
1432
2.58M
            if (bps <= 8)
1433
2.58M
                image_init_map(&pmap->table.lookup8[0], 1 << bps,
1434
2.58M
                               map_decode);
1435
2.58M
        } else {                /* The map index encompasses more than one pixel. */
1436
17.7k
            byte map[4];
1437
17.7k
            register int i;
1438
1439
17.7k
            image_init_map(&map[0], 1 << bps, map_decode);
1440
17.7k
            switch (bps) {
1441
17.2k
                case 1:
1442
17.2k
                    {
1443
17.2k
                        register bits32 *p = &pmap->table.lookup4x1to32[0];
1444
1445
17.2k
                        if (map[0] == 0 && map[1] == 0xff)
1446
17.1k
                            memcpy((byte *) p, lookup4x1to32_identity, 16 * 4);
1447
58
                        else if (map[0] == 0xff && map[1] == 0)
1448
58
                            memcpy((byte *) p, lookup4x1to32_inverted, 16 * 4);
1449
0
                        else
1450
0
                            for (i = 0; i < 16; i++, p++)
1451
0
                                ((byte *) p)[0] = map[i >> 3],
1452
0
                                    ((byte *) p)[1] = map[(i >> 2) & 1],
1453
0
                                    ((byte *) p)[2] = map[(i >> 1) & 1],
1454
0
                                    ((byte *) p)[3] = map[i & 1];
1455
17.2k
                    }
1456
17.2k
                    break;
1457
531
                case 2:
1458
531
                    {
1459
531
                        register bits16 *p = &pmap->table.lookup2x2to16[0];
1460
1461
9.02k
                        for (i = 0; i < 16; i++, p++)
1462
8.49k
                            ((byte *) p)[0] = map[i >> 2],
1463
8.49k
                                ((byte *) p)[1] = map[i & 3];
1464
531
                    }
1465
531
                    break;
1466
17.7k
            }
1467
17.7k
        }
1468
2.60M
        pmap->decode_base /* = decode_lookup[0] */  = real_decode[0];
1469
2.60M
        pmap->decode_factor =
1470
2.60M
            (real_decode[1] - real_decode[0]) /
1471
2.60M
            (bps <= 8 ? 255.0 : (float)frac_1);
1472
2.60M
        pmap->decode_max /* = decode_lookup[15] */  = real_decode[1];
1473
2.60M
        if (decode_type) {
1474
2.60M
            pmap->decoding = sd_none;
1475
2.60M
            pmap->inverted = map_decode[0] != 0;
1476
2.60M
        } else if (bps <= 4) {
1477
597
            int step = 15 / ((1 << bps) - 1);
1478
597
            int i;
1479
1480
597
            pmap->decoding = sd_lookup;
1481
3.42k
            for (i = 15 - step; i > 0; i -= step)
1482
2.82k
                pmap->decode_lookup[i] = pmap->decode_base +
1483
2.82k
                    i * (255.0 / 15) * pmap->decode_factor;
1484
597
            pmap->inverted = 0;
1485
4.33k
        } else {
1486
4.33k
            pmap->decoding = sd_compute;
1487
4.33k
            pmap->inverted = 0;
1488
4.33k
        }
1489
2.60M
        if (spp == 1) {         /* and ci == 0 *//* Pre-map entries 0 and 255. */
1490
409k
            gs_client_color cc;
1491
1492
            /* Image clues are used in this case */
1493
409k
            cc.paint.values[0] = real_decode[0];
1494
409k
            code = (*pcs->type->remap_color) (&cc, pcs, penum->icolor0,
1495
409k
                                       pgs, dev, gs_color_select_source);
1496
409k
            if (code < 0)
1497
9
                return code;
1498
409k
            cc.paint.values[0] = real_decode[1];
1499
409k
            code = (*pcs->type->remap_color) (&cc, pcs, penum->icolor1,
1500
409k
                                       pgs, dev, gs_color_select_source);
1501
409k
            if (code < 0)
1502
0
                return code;
1503
409k
        }
1504
2.60M
    }
1505
1.13M
    return 0;
1506
1.13M
}
1507
/* Construct a mapping table for sample values. */
1508
/* map_size is 2, 4, 16, or 256.  Note that 255 % (map_size - 1) == 0, */
1509
/* so the division 0xffffL / (map_size - 1) is always exact. */
1510
void
1511
image_init_map(byte * map, int map_size, const float *decode)
1512
2.61M
{
1513
2.61M
    float min_v = decode[0];
1514
2.61M
    float diff_v = decode[1] - min_v;
1515
1516
2.61M
    if (diff_v == 1 || diff_v == -1) {  /* We can do the stepping with integers, without overflow. */
1517
2.61M
        byte *limit = map + map_size;
1518
2.61M
        uint value = (uint)(min_v * 0xffffL);
1519
2.61M
        int diff = (int)(diff_v * (0xffffL / (map_size - 1)));
1520
1521
666M
        for (; map != limit; map++, value += diff)
1522
663M
            *map = value >> 8;
1523
2.61M
    } else {                    /* Step in floating point, with clamping. */
1524
0
        int i;
1525
1526
0
        for (i = 0; i < map_size; ++i) {
1527
0
            int value = (int)((min_v + diff_v * i / (map_size - 1)) * 255);
1528
1529
0
            map[i] = (value < 0 ? 0 : value > 255 ? 255 : value);
1530
0
        }
1531
0
    }
1532
2.61M
}
1533
1534
/*
1535
 * Scale a pair of mask_color values to match the scaling of each sample to
1536
 * a full byte, and complement and swap them if the map incorporates
1537
 * a Decode = [1 0] inversion.
1538
 */
1539
void
1540
gx_image_scale_mask_colors(gx_image_enum *penum, int component_index)
1541
567
{
1542
567
    uint scale = 255 / ((1 << penum->bps) - 1);
1543
567
    uint *values = &penum->mask_color.values[component_index * 2];
1544
567
    uint v0 = values[0] *= scale;
1545
567
    uint v1 = values[1] *= scale;
1546
1547
567
    if (penum->map[component_index].decoding == sd_none &&
1548
555
        penum->map[component_index].inverted
1549
567
        ) {
1550
0
        values[0] = 255 - v1;
1551
0
        values[1] = 255 - v0;
1552
0
    }
1553
567
}
1554
1555
/* Used to indicate for ICC procesing if we have decoding to do */
1556
bool
1557
gx_has_transfer(const gs_gstate *pgs, int num_comps)
1558
751k
{
1559
751k
    int k;
1560
1561
1.94M
    for (k = 0; k < num_comps; k++) {
1562
1.51M
        if (pgs->effective_transfer[k]->proc != gs_identity_transfer) {
1563
316k
            return(true);
1564
316k
        }
1565
1.51M
    }
1566
435k
    return(false);
1567
751k
}