Coverage Report

Created: 2026-04-09 07:06

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
494
ENUM_PTRS_WITH(image_enum_enum_ptrs, gx_image_enum *eptr)
64
182
{
65
182
    int bps;
66
182
    gs_ptr_type_t ret;
67
68
    /* Enumerate the used members of clues.dev_color. */
69
182
    index -= gx_image_enum_num_ptrs;
70
182
    bps = eptr->unpack_bps;
71
182
    if (eptr->spp != 1)
72
0
        bps = 8;
73
182
    else if (bps > 8 || eptr->unpack == sample_unpack_copy)
74
0
        bps = 1;
75
182
    if (index >= (1 << bps) * st_device_color_max_ptrs)         /* done */
76
26
        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
156
    if (eptr->spp == 1) {
80
156
        if (eptr->clues != NULL) {
81
156
            if (eptr->clues[(index/st_device_color_max_ptrs) *
82
156
                (255 / ((1 << bps) - 1))].dev_color.type != 0) {
83
156
                ret = ENUM_USING(st_device_color,
84
156
                                 &eptr->clues[(index / st_device_color_max_ptrs) *
85
156
                                 (255 / ((1 << bps) - 1))].dev_color,
86
156
                                 sizeof(eptr->clues[0].dev_color),
87
156
                                 index % st_device_color_max_ptrs);
88
156
            } else {
89
0
                ret = 0;
90
0
            }
91
156
        } else {
92
0
            ret = 0;
93
0
        }
94
156
    } else {
95
0
        ret = 0;
96
0
    }
97
156
    if (ret == 0)               /* don't stop early */
98
156
        ENUM_RETURN(0);
99
0
    return ret;
100
156
}
101
102
312
#define e1(i,elt) ENUM_PTR(i,gx_image_enum,elt);
103
494
gx_image_enum_do_ptrs(e1)
104
494
#undef e1
105
494
ENUM_PTRS_END
106
107
26
static RELOC_PTRS_WITH(image_enum_reloc_ptrs, gx_image_enum *eptr)
108
26
{
109
26
    int i;
110
111
312
#define r1(i,elt) RELOC_PTR(gx_image_enum,elt);
112
312
    gx_image_enum_do_ptrs(r1)
113
26
#undef r1
114
26
    {
115
26
        int bps = eptr->unpack_bps;
116
117
26
        if (eptr->spp != 1)
118
0
            bps = 8;
119
26
        else if (bps > 8 || eptr->unpack == sample_unpack_copy)
120
0
            bps = 1;
121
26
        if (eptr->spp == 1) {
122
78
        for (i = 0; i <= 255; i += 255 / ((1 << bps) - 1))
123
52
            RELOC_USING(st_device_color,
124
26
                        &eptr->clues[i].dev_color, sizeof(gx_device_color));
125
26
    }
126
26
}
127
26
}
128
26
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.55M
{
152
1.55M
    const gs_pixel_image_t *pim = (const gs_pixel_image_t *)pic;
153
1.55M
    int width = pim->Width, height = pim->Height;
154
1.55M
    int bpc = pim->BitsPerComponent;
155
1.55M
    gx_image_enum *penum;
156
157
1.55M
    if (width < 0 || height < 0)
158
0
        return_error(gs_error_rangecheck);
159
1.55M
    switch (pim->format) {
160
1.52M
    case gs_image_format_chunky:
161
1.55M
    case gs_image_format_component_planar:
162
1.55M
        switch (bpc) {
163
1.55M
        case 1: case 2: case 4: case 8: case 12: case 16: break;
164
2
        default: return_error(gs_error_rangecheck);
165
1.55M
        }
166
1.55M
        break;
167
1.55M
    case gs_image_format_bit_planar:
168
0
        if (bpc < 1 || bpc > 8)
169
0
            return_error(gs_error_rangecheck);
170
1.55M
    }
171
1.55M
    if (prect) {
172
450k
        if (prect->p.x < 0 || prect->p.y < 0 ||
173
450k
            prect->q.x < prect->p.x || prect->q.y < prect->p.y ||
174
450k
            prect->q.x > width || prect->q.y > height
175
450k
            )
176
0
            return_error(gs_error_rangecheck);
177
450k
    }
178
1.55M
    *ppenum = NULL;   /* in case alloc fails and caller doesn't check code */
179
1.55M
    penum = gs_alloc_struct(mem, gx_image_enum, &st_gx_image_enum,
180
1.55M
                            "gx_default_begin_image");
181
1.55M
    if (penum == 0)
182
0
        return_error(gs_error_VMerror);
183
1.55M
    memset(penum, 0, sizeof(gx_image_enum));  /* in case of failure, no dangling pointers */
184
1.55M
    if (prect) {
185
450k
        penum->rect.x = prect->p.x;
186
450k
        penum->rect.y = prect->p.y;
187
450k
        penum->rect.w = prect->q.x - prect->p.x;
188
450k
        penum->rect.h = prect->q.y - prect->p.y;
189
1.10M
    } else {
190
1.10M
        penum->rect.x = 0, penum->rect.y = 0;
191
1.10M
        penum->rect.w = width, penum->rect.h = height;
192
1.10M
    }
193
1.55M
    penum->rrect.x = penum->rect.x;
194
1.55M
    penum->rrect.y = penum->rect.y;
195
1.55M
    penum->rrect.w = penum->rect.w;
196
1.55M
    penum->rrect.h = penum->rect.h;
197
1.55M
    penum->drect.x = penum->rect.x;
198
1.55M
    penum->drect.y = penum->rect.y;
199
1.55M
    penum->drect.w = penum->rect.w;
200
1.55M
    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.55M
    *ppenum = penum;
210
1.55M
    return 0;
211
1.55M
}
212
213
/* Convert and restrict to a valid range. */
214
4.12M
static inline fixed float2fixed_rounded_boxed(double src) {
215
4.12M
    float v = floor(src*fixed_scale + 0.5);
216
217
4.12M
    if (v <= min_fixed)
218
4.39k
        return min_fixed;
219
4.12M
    else if (v >= max_fixed)
220
43.2k
        return max_fixed;
221
4.07M
    else
222
4.07M
        return  (fixed)v;
223
4.12M
}
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.29M
{
230
2.29M
    int code = 0;
231
232
2.29M
    if (pmat == 0)
233
2.24M
        pmat = &ctm_only(pgs);
234
2.29M
    if (ImageMatrix->xx == pmat->xx && ImageMatrix->xy == pmat->xy &&
235
891k
        ImageMatrix->yx == pmat->yx && ImageMatrix->yy == pmat->yy) {
236
        /* Process common special case separately to accept singular matrix. */
237
890k
        rmat->xx = rmat->yy = 1.;
238
890k
        rmat->xy = rmat->yx = 0.;
239
890k
        rmat->tx = pmat->tx - ImageMatrix->tx;
240
890k
        rmat->ty = pmat->ty - ImageMatrix->ty;
241
1.40M
    } else {
242
1.40M
        if ((code = gs_matrix_invert_to_double(ImageMatrix, rmat)) < 0 ||
243
1.40M
            (code = gs_matrix_multiply_double(rmat, pmat, rmat)) < 0
244
1.40M
            ) {
245
115
            return code;
246
115
        }
247
1.40M
    }
248
2.29M
    return code;
249
2.29M
}
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.55M
{
264
1.55M
    const gs_pixel_image_t *pim = (const gs_pixel_image_t *)pic;
265
1.55M
    gs_image_format_t format = pim->format;
266
1.55M
    const int width = pim->Width;
267
1.55M
    const int height = pim->Height;
268
1.55M
    const int bps = pim->BitsPerComponent;
269
1.55M
    bool masked = penum->masked;
270
1.55M
    const float *decode = pim->Decode;
271
1.55M
    gs_matrix_double mat;
272
1.55M
    int index_bps;
273
1.55M
    gs_color_space *pcs = pim->ColorSpace;
274
1.55M
    gs_logical_operation_t lop = (pgs ? pgs->log_op : lop_default);
275
1.55M
    int code;
276
1.55M
    int log2_xbytes = (bps <= 8 ? 0 : arch_log2_sizeof_frac);
277
1.55M
    int spp, nplanes, spread;
278
1.55M
    uint bsize;
279
1.55M
    byte *buffer = NULL;
280
1.55M
    fixed mtx, mty;
281
1.55M
    gs_fixed_point row_extent, col_extent, x_extent, y_extent;
282
1.55M
    bool device_color = true;
283
1.55M
    gs_fixed_rect obox, cbox;
284
1.55M
    bool gridfitimages = 0;
285
1.55M
    bool in_pattern_accumulator;
286
1.55M
    bool in_smask;
287
1.55M
    int orthogonal;
288
1.55M
    int force_interpolation = 0;
289
290
1.55M
    penum->pcs = NULL;
291
1.55M
    penum->clues = NULL;
292
1.55M
    penum->icc_setup.has_transfer = false;
293
1.55M
    penum->icc_setup.is_lab = false;
294
1.55M
    penum->icc_setup.must_halftone = false;
295
1.55M
    penum->icc_setup.need_decode = false;
296
1.55M
    penum->Width = width;
297
1.55M
    penum->Height = height;
298
299
1.55M
    if ((code = gx_image_compute_mat(pgs, pmat, &(pim->ImageMatrix), &mat)) < 0) {
300
92
        return code;
301
92
    }
302
1.55M
    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.55M
    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.55M
    if (mat.xy == 0 && mat.yx == 0)
326
1.37M
        orthogonal = 1;
327
179k
    else if (mat.xx == 0 && mat.yy == 0)
328
104k
        orthogonal = 2;
329
74.9k
    else
330
74.9k
        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.55M
    in_smask = (pim->override_in_smask ||
338
1.55M
                (dev_proc(dev, dev_spec_op)(dev, gxdso_in_smask, NULL, 0)) > 0);
339
1.55M
    gridfitimages = (in_smask || in_pattern_accumulator) && orthogonal;
340
341
1.55M
    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.22M
    } else if (!gridfitimages &&
345
939k
               (!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
773k
    } else if (gridfitimages && (penum->masked && penum->image_parent_type == 0)) {
349
        /* We don't gridfit imagemasks in a pattern accumulator */
350
447k
    } 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
443k
    } else if (orthogonal == 1) {
353
319k
        if (width == 1 || gridfitimages) {
354
278k
            if (mat.xx > 0) {
355
278k
                fixed ix0 = int2fixed(fixed2int(float2fixed(mat.tx)));
356
278k
                double x1 = mat.tx + mat.xx * width;
357
278k
                fixed ix1 = int2fixed(fixed2int_ceiling(float2fixed(x1)));
358
278k
                mat.tx = (double)fixed2float(ix0);
359
278k
                mat.xx = (double)(fixed2float(ix1 - ix0)/width);
360
278k
            } else if (mat.xx < 0) {
361
155
                fixed ix0 = int2fixed(fixed2int_ceiling(float2fixed(mat.tx)));
362
155
                double x1 = mat.tx + mat.xx * width;
363
155
                fixed ix1 = int2fixed(fixed2int(float2fixed(x1)));
364
155
                mat.tx = (double)fixed2float(ix0);
365
155
                mat.xx = (double)(fixed2float(ix1 - ix0)/width);
366
155
            }
367
278k
        }
368
319k
        if (height == 1 || gridfitimages) {
369
278k
            if (mat.yy > 0) {
370
252k
                fixed iy0 = int2fixed(fixed2int(float2fixed(mat.ty)));
371
252k
                double y1 = mat.ty + mat.yy * height;
372
252k
                fixed iy1 = int2fixed(fixed2int_ceiling(float2fixed(y1)));
373
252k
                mat.ty = (double)fixed2float(iy0);
374
252k
                mat.yy = (double)(fixed2float(iy1 - iy0)/height);
375
252k
            } else if (mat.yy < 0) {
376
25.9k
                fixed iy0 = int2fixed(fixed2int_ceiling(float2fixed(mat.ty)));
377
25.9k
                double y1 = mat.ty + mat.yy * height;
378
25.9k
                fixed iy1 = int2fixed(fixed2int(float2fixed(y1)));
379
25.9k
                mat.ty = (double)fixed2float(iy0);
380
25.9k
                mat.yy = ((double)fixed2float(iy1 - iy0)/height);
381
25.9k
            }
382
278k
        }
383
319k
    } else if (orthogonal == 2) {
384
98.6k
        if (height == 1 || gridfitimages) {
385
8
            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
0
                fixed ix0 = int2fixed(fixed2int_ceiling(float2fixed(mat.tx)));
393
0
                double x1 = mat.tx + mat.yx * height;
394
0
                fixed ix1 = int2fixed(fixed2int(float2fixed(x1)));
395
0
                mat.tx = (double)fixed2float(ix0);
396
0
                mat.yx = (double)(fixed2float(ix1 - ix0)/height);
397
0
            }
398
8
        }
399
98.6k
        if (width == 1 || gridfitimages) {
400
0
            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
0
            } 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
0
        }
414
98.6k
    }
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.55M
    if (in_pattern_accumulator) {
420
6.14k
        double ome = ((double)(fixed_1 - fixed_epsilon)) / (double)fixed_1; /* One Minus Epsilon */
421
422
6.14k
        if (orthogonal == 1) {
423
1.91k
            if ((mat.xx > -ome && mat.xx < ome) || (mat.yy > -ome && mat.yy < ome)) {
424
280
                force_interpolation = true;
425
280
            }
426
4.23k
        } else if (orthogonal == 2) {
427
4.22k
            if ((mat.xy > -ome && mat.xy < ome) || (mat.yx > -ome && mat.yx < ome)) {
428
0
                force_interpolation = true;
429
0
            }
430
4.22k
        }
431
6.14k
    }
432
433
    /* Can we restrict the amount of image we need? */
434
1.55M
    while (!pim->imagematrices_are_untrustworthy) /* So we can break out of it */
435
1.55M
    {
436
1.55M
        gs_rect rect, rect_src;
437
1.55M
        gs_matrix mi;
438
1.55M
        const gs_matrix *m = pgs != NULL ? &ctm_only(pgs) : NULL;
439
1.55M
        gs_int_rect irect;
440
1.55M
        if (m == NULL || (code = gs_matrix_invert(m, &mi)) < 0 ||
441
1.55M
            (code = gs_matrix_multiply(&mi, &pic->ImageMatrix, &mi)) < 0) {
442
            /* Give up trying to shrink the render box, but continue processing */
443
6.69k
            break;
444
6.69k
        }
445
1.55M
        if (pcpath)
446
638k
        {
447
638k
            gs_fixed_rect obox;
448
638k
            gx_cpath_outer_box(pcpath, &obox);
449
638k
            rect.p.x = fixed2float(obox.p.x);
450
638k
            rect.p.y = fixed2float(obox.p.y);
451
638k
            rect.q.x = fixed2float(obox.q.x);
452
638k
            rect.q.y = fixed2float(obox.q.y);
453
638k
        }
454
914k
        else
455
914k
        {
456
914k
            rect.p.x = 0;
457
914k
            rect.p.y = 0;
458
914k
            rect.q.x = dev->width;
459
914k
            rect.q.y = dev->height;
460
914k
        }
461
        /* rect is in destination space. Calculate rect_src, in source space. */
462
1.55M
        code = gs_bbox_transform(&rect, &mi, &rect_src);
463
1.55M
        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.55M
        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.55M
        irect.p.x = (int)floor(rect_src.p.x);
495
1.55M
        irect.p.y = (int)floor(rect_src.p.y);
496
1.55M
        irect.q.x = (int)ceil(rect_src.q.x);
497
1.55M
        irect.q.y = (int)ceil(rect_src.q.y);
498
        /* We therefore only need to render within irect. Restrict rrect to this. */
499
1.55M
        if (penum->rrect.x < irect.p.x) {
500
63.4k
            penum->rrect.w -= irect.p.x - penum->rrect.x;
501
63.4k
            if (penum->rrect.w < 0)
502
55.6k
               penum->rrect.w = 0;
503
63.4k
            penum->rrect.x = irect.p.x;
504
63.4k
        }
505
1.55M
        if (penum->rrect.x + penum->rrect.w > irect.q.x) {
506
121k
            penum->rrect.w = irect.q.x - penum->rrect.x;
507
121k
            if (penum->rrect.w < 0)
508
94.7k
                penum->rrect.w = 0;
509
121k
        }
510
1.55M
        if (penum->rrect.y < irect.p.y) {
511
129k
            penum->rrect.h -= irect.p.y - penum->rrect.y;
512
129k
            if (penum->rrect.h < 0)
513
53.3k
                penum->rrect.h = 0;
514
129k
            penum->rrect.y = irect.p.y;
515
129k
        }
516
1.55M
        if (penum->rrect.y + penum->rrect.h > irect.q.y) {
517
103k
            penum->rrect.h = irect.q.y - penum->rrect.y;
518
103k
            if (penum->rrect.h < 0)
519
14.5k
                penum->rrect.h = 0;
520
103k
        }
521
1.55M
        if (penum->drect.x < irect.p.x) {
522
63.4k
            penum->drect.w -= irect.p.x - penum->drect.x;
523
63.4k
            if (penum->drect.w < 0)
524
55.6k
               penum->drect.w = 0;
525
63.4k
            penum->drect.x = irect.p.x;
526
63.4k
        }
527
1.55M
        if (penum->drect.x + penum->drect.w > irect.q.x) {
528
121k
            penum->drect.w = irect.q.x - penum->drect.x;
529
121k
            if (penum->drect.w < 0)
530
94.7k
                penum->drect.w = 0;
531
121k
        }
532
1.55M
        if (penum->drect.y < irect.p.y) {
533
129k
            penum->drect.h -= irect.p.y - penum->drect.y;
534
129k
            if (penum->drect.h < 0)
535
53.3k
                penum->drect.h = 0;
536
129k
            penum->drect.y = irect.p.y;
537
129k
        }
538
1.55M
        if (penum->drect.y + penum->drect.h > irect.q.y) {
539
103k
            penum->drect.h = irect.q.y - penum->drect.y;
540
103k
            if (penum->drect.h < 0)
541
14.5k
                penum->drect.h = 0;
542
103k
        }
543
1.55M
        break; /* Out of the while */
544
1.55M
    }
545
    /* Check for the intersection being null */
546
1.55M
    if (penum->drect.x + penum->drect.w <= penum->rect.x  ||
547
1.46M
        penum->rect.x  + penum->rect.w  <= penum->drect.x ||
548
1.40M
        penum->drect.y + penum->drect.h <= penum->rect.y  ||
549
1.39M
        penum->rect.y  + penum->rect.h  <= penum->drect.y)
550
186k
    {
551
          /* Something may have gone wrong with the floating point above.
552
           * set the region to something sane. */
553
186k
        penum->drect.x = penum->rect.x;
554
186k
        penum->drect.y = penum->rect.y;
555
186k
        penum->drect.w = 0;
556
186k
        penum->drect.h = 0;
557
186k
    }
558
1.55M
    if (penum->rrect.x + penum->rrect.w <= penum->drect.x  ||
559
1.46M
        penum->drect.x + penum->drect.w  <= penum->rrect.x ||
560
1.37M
        penum->rrect.y + penum->rrect.h <= penum->drect.y  ||
561
1.37M
        penum->drect.y + penum->drect.h  <= penum->rrect.y)
562
186k
    {
563
          /* Something may have gone wrong with the floating point above.
564
           * set the region to something sane. */
565
186k
        penum->rrect.x = penum->drect.x;
566
186k
        penum->rrect.y = penum->drect.y;
567
186k
        penum->rrect.w = 0;
568
186k
        penum->rrect.h = 0;
569
186k
    }
570
571
    /*penum->matrix = mat;*/
572
1.55M
    penum->matrix.xx = mat.xx;
573
1.55M
    penum->matrix.xy = mat.xy;
574
1.55M
    penum->matrix.yx = mat.yx;
575
1.55M
    penum->matrix.yy = mat.yy;
576
1.55M
    penum->matrix.tx = mat.tx;
577
1.55M
    penum->matrix.ty = mat.ty;
578
1.55M
    if_debug6m('b', mem, " [%g %g %g %g %g %g]\n",
579
1.55M
              mat.xx, mat.xy, mat.yx, mat.yy, mat.tx, mat.ty);
580
    /* following works for 1, 2, 4, 8, 12, 16 */
581
1.55M
    index_bps = (bps < 8 ? bps >> 1 : (bps >> 2) + 1);
582
    /*
583
     * Compute extents with distance transformation.
584
     */
585
1.55M
    if (mat.tx > 0)
586
1.27M
        mtx = float2fixed(mat.tx);
587
285k
    else { /* Use positive values to ensure round down. */
588
285k
        int f = (int)-mat.tx + 1;
589
590
285k
        mtx = float2fixed(mat.tx + f) - int2fixed(f);
591
285k
    }
592
1.55M
    if (mat.ty > 0)
593
582k
        mty = float2fixed(mat.ty);
594
976k
    else {  /* Use positive values to ensure round down. */
595
976k
        int f = (int)-mat.ty + 1;
596
597
976k
        mty = float2fixed(mat.ty + f) - int2fixed(f);
598
976k
    }
599
600
1.55M
    row_extent.x = float2fixed_rounded_boxed(width * mat.xx);
601
1.55M
    row_extent.y =
602
1.55M
        (is_fzero(mat.xy) ? fixed_0 :
603
1.55M
         float2fixed_rounded_boxed(width * mat.xy));
604
1.55M
    col_extent.x =
605
1.55M
        (is_fzero(mat.yx) ? fixed_0 :
606
1.55M
         float2fixed_rounded_boxed(height * mat.yx));
607
1.55M
    col_extent.y = float2fixed_rounded_boxed(height * mat.yy);
608
1.55M
    gx_image_enum_common_init((gx_image_enum_common_t *)penum,
609
1.55M
                              (const gs_data_image_t *)pim,
610
1.55M
                              &image1_enum_procs, dev,
611
1.55M
                              (masked ? 1 : (penum->alpha ? cs_num_components(pcs)+1 : cs_num_components(pcs))),
612
1.55M
                              format);
613
1.55M
    if (penum->rect.w == width && penum->rect.h == height) {
614
1.21M
        x_extent = row_extent;
615
1.21M
        y_extent = col_extent;
616
1.21M
    } else {
617
349k
        int rw = penum->rect.w, rh = penum->rect.h;
618
619
349k
        x_extent.x = float2fixed_rounded_boxed(rw * mat.xx);
620
349k
        x_extent.y =
621
349k
            (is_fzero(mat.xy) ? fixed_0 :
622
349k
             float2fixed_rounded_boxed(rw * mat.xy));
623
349k
        y_extent.x =
624
349k
            (is_fzero(mat.yx) ? fixed_0 :
625
349k
             float2fixed_rounded_boxed(rh * mat.yx));
626
349k
        y_extent.y = float2fixed_rounded_boxed(rh * mat.yy);
627
349k
    }
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.55M
    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
863k
        penum->clues = (gx_image_clue*) gs_alloc_bytes(mem, sizeof(gx_image_clue)*256,
636
863k
                             "gx_image_enum_begin");
637
863k
        if (penum->clues == NULL) {
638
0
            code = gs_error_VMerror;
639
0
            goto fail;
640
0
        }
641
863k
        penum->icolor0 = &(penum->clues[0].dev_color);
642
863k
        penum->icolor1 = &(penum->clues[255].dev_color);
643
863k
    } else {
644
696k
        penum->icolor0 = &(penum->icolor0_val);
645
696k
        penum->icolor1 = &(penum->icolor1_val);
646
696k
    }
647
1.55M
    penum->icolor0->tag = penum->icolor1->tag = device_current_tag(dev);
648
649
1.55M
    if (masked) {       /* This is imagemask. */
650
504k
        if (bps != 1 || pcs != NULL || penum->alpha || decode[0] == decode[1]) {
651
48
            code = gs_error_rangecheck;
652
48
            goto fail;
653
48
        }
654
        /* Initialize color entries 0 and 255. */
655
504k
        set_nonclient_dev_color(penum->icolor0, gx_no_color_index);
656
504k
        set_nonclient_dev_color(penum->icolor1, gx_no_color_index);
657
504k
        *(penum->icolor1) = *pdcolor;
658
504k
        memcpy(&penum->map[0].table.lookup4x1to32[0],
659
504k
               (decode[0] < decode[1] ? lookup4x1to32_inverted :
660
504k
                lookup4x1to32_identity),
661
504k
               16 * 4);
662
504k
        penum->map[0].decoding = sd_none;
663
504k
        spp = 1;
664
504k
        lop = rop3_know_S_0(lop);
665
1.05M
    } else {                    /* This is image, not imagemask. */
666
1.05M
        const gs_color_space_type *pcst = pcs->type;
667
1.05M
        int b_w_color;
668
669
1.05M
        spp = cs_num_components(pcs);
670
1.05M
        if (spp < 0) {          /* Pattern not allowed */
671
0
            code = gs_error_rangecheck;
672
0
            goto fail;
673
0
        }
674
1.05M
        if (penum->alpha)
675
0
            ++spp;
676
        /* Use a less expensive format if possible. */
677
1.05M
        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
29.8k
        case gs_image_format_component_planar:
683
29.8k
            if (spp == 1)
684
0
                format = gs_image_format_chunky;
685
1.05M
        default:                /* chunky */
686
1.05M
            break;
687
1.05M
        }
688
689
1.05M
        if (pcs->cmm_icc_profile_data != NULL) {
690
1.02M
            device_color = false;
691
1.02M
        } else {
692
25.0k
            device_color = (*pcst->concrete_space) (pcs, pgs) == pcs;
693
25.0k
        }
694
695
1.05M
        code = image_init_colors(penum, bps, spp, format, decode, pgs, dev,
696
1.05M
                          pcs, &device_color);
697
1.05M
        if (code < 0) {
698
5
            gs_free_object(mem, penum->clues, "gx_image_enum_begin");
699
5
            gs_free_object(mem, penum, "gx_default_begin_image");
700
5
            return gs_throw(code, "Image colors initialization failed");
701
5
        }
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.05M
        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.05M
        if (!pim->CombineWithColor)
723
1.05M
            lop = rop3_know_T_0(lop);
724
4.43k
        else if ((rop3_uses_T(lop) && color_draws_b_w(dev, pdcolor) == 0))
725
4.24k
            lop = rop3_know_T_0(lop);
726
727
1.05M
        if (lop != rop3_S &&    /* if best case, no more work needed */
728
1.05M
            !rop3_uses_T(lop) && bps == 1 && spp == 1 &&
729
396
            (b_w_color =
730
396
             color_draws_b_w(dev, penum->icolor0)) >= 0 &&
731
396
            color_draws_b_w(dev, penum->icolor1) == (b_w_color ^ 1)
732
1.05M
            ) {
733
396
            if (b_w_color) {    /* Swap the colors and invert the RasterOp source. */
734
396
                gx_device_color dcolor;
735
736
396
                dcolor = *(penum->icolor0);
737
396
                *(penum->icolor0) = *(penum->icolor1);
738
396
                *(penum->icolor1) = dcolor;
739
396
                lop = rop3_invert_S(lop);
740
396
            }
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
396
            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
396
                default:
771
396
                    ;
772
396
            }
773
396
        }
774
1.05M
    }
775
1.55M
    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.55M
    bsize = ((bps > 8 ? width * 2 : width) + 15) * spp;
782
1.55M
    buffer = gs_alloc_bytes(mem, bsize, "image buffer");
783
1.55M
    if (buffer == 0) {
784
5
        code = gs_error_VMerror;
785
5
        goto fail;
786
5
    }
787
1.55M
    penum->bps = bps;
788
1.55M
    penum->unpack_bps = bps;
789
1.55M
    penum->log2_xbytes = log2_xbytes;
790
1.55M
    penum->spp = spp;
791
1.55M
    switch (format) {
792
1.52M
    case gs_image_format_chunky:
793
1.52M
        nplanes = 1;
794
1.52M
        spread = 1 << log2_xbytes;
795
1.52M
        break;
796
29.8k
    case gs_image_format_component_planar:
797
29.8k
        nplanes = spp;
798
29.8k
        spread = spp << log2_xbytes;
799
29.8k
        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.55M
    }
808
1.55M
    penum->num_planes = nplanes;
809
1.55M
    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.55M
    penum->interpolate = force_interpolation ? interp_force : pim->Interpolate ? interp_on : interp_off;
817
1.55M
    penum->x_extent = x_extent;
818
1.55M
    penum->y_extent = y_extent;
819
1.55M
    penum->posture =
820
1.55M
        ((x_extent.y | y_extent.x) == 0 ? image_portrait :
821
1.55M
         (x_extent.x | y_extent.y) == 0 ? image_landscape :
822
179k
         image_skewed);
823
1.55M
    penum->pgs = pgs;
824
1.55M
    if (pgs != NULL)
825
1.55M
        penum->pgs_level = pgs->level;
826
1.55M
    penum->pcs = pcs;
827
1.55M
    rc_increment_cs(pcs); /* Grab a ref (will decrement in gx_image1_end_image() */
828
1.55M
    penum->memory = mem;
829
1.55M
    penum->buffer = buffer;
830
1.55M
    penum->buffer_size = bsize;
831
1.55M
    penum->line = NULL;
832
1.55M
    penum->icc_link = NULL;
833
1.55M
    penum->color_cache = NULL;
834
1.55M
    penum->ht_buffer = NULL;
835
1.55M
    penum->thresh_buffer = NULL;
836
1.55M
    penum->use_cie_range = false;
837
1.55M
    penum->line_size = 0;
838
1.55M
    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.55M
    penum->slow_loop = 0;
852
1.55M
    if (pcpath == 0) {
853
914k
        (*dev_proc(dev, get_clipping_box)) (dev, &obox);
854
914k
        cbox = obox;
855
914k
        penum->clip_image = 0;
856
914k
    } else
857
644k
        penum->clip_image =
858
644k
            (gx_cpath_outer_box(pcpath, &obox) |        /* not || */
859
644k
             gx_cpath_inner_box(pcpath, &cbox) ?
860
643k
             0 : image_clip_region);
861
1.55M
    penum->clip_outer = obox;
862
1.55M
    penum->clip_inner = cbox;
863
1.55M
    penum->log_op = rop3_T;     /* rop device takes care of this */
864
1.55M
    penum->clip_dev = 0;        /* in case we bail out */
865
1.55M
    penum->rop_dev = 0;         /* ditto */
866
1.55M
    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.55M
    {
874
1.55M
        fixed
875
1.55M
            epx = min(row_extent.x, 0) + min(col_extent.x, 0),
876
1.55M
            eqx = max(row_extent.x, 0) + max(col_extent.x, 0),
877
1.55M
            epy = min(row_extent.y, 0) + min(col_extent.y, 0),
878
1.55M
            eqy = max(row_extent.y, 0) + max(col_extent.y, 0);
879
880
1.55M
        {
881
1.55M
            int hwx, hwy;
882
883
1.55M
            switch (penum->posture) {
884
1.37M
                case image_portrait:
885
1.37M
                    hwx = width, hwy = height;
886
1.37M
                    break;
887
104k
                case image_landscape:
888
104k
                    hwx = height, hwy = width;
889
104k
                    break;
890
74.8k
                default:
891
74.8k
                    hwx = hwy = 0;
892
1.55M
            }
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.55M
            if (hwx == 1 && eqx - epx < fixed_1) {
903
16
                fixed diff =
904
16
                arith_rshift_1(row_extent.x + col_extent.x);
905
906
16
                mtx = (((mtx + diff) | fixed_half) & -fixed_half) - diff;
907
16
            }
908
1.55M
            if (hwy == 1 && eqy - epy < fixed_1) {
909
21
                fixed diff =
910
21
                arith_rshift_1(row_extent.y + col_extent.y);
911
912
21
                mty = (((mty + diff) | fixed_half) & -fixed_half) - diff;
913
21
            }
914
1.55M
        }
915
1.55M
        if_debug5m('b', mem, "[b]Image: %sspp=%d, bps=%d, mt=(%g,%g)\n",
916
1.55M
                   (masked? "masked, " : ""), spp, bps,
917
1.55M
                   fixed2float(mtx), fixed2float(mty));
918
1.55M
        if_debug9m('b', mem,
919
1.55M
                   "[b]   cbox=(%g,%g),(%g,%g), obox=(%g,%g),(%g,%g), clip_image=0x%x\n",
920
1.55M
                   fixed2float(cbox.p.x), fixed2float(cbox.p.y),
921
1.55M
                   fixed2float(cbox.q.x), fixed2float(cbox.q.y),
922
1.55M
                   fixed2float(obox.p.x), fixed2float(obox.p.y),
923
1.55M
                   fixed2float(obox.q.x), fixed2float(obox.q.y),
924
1.55M
                   penum->clip_image);
925
        /* These DDAs enumerate the starting position of each source pixel
926
         * row in device space. */
927
1.55M
        dda_init(penum->dda.row.x, mtx, col_extent.x, height);
928
1.55M
        dda_init(penum->dda.row.y, mty, col_extent.y, height);
929
1.55M
        if (dda_will_overflow(penum->dda.row.x) ||
930
1.55M
            dda_will_overflow(penum->dda.row.y))
931
0
        {
932
0
            code = gs_error_rangecheck;
933
0
            goto fail;
934
0
        }
935
1.55M
        if (penum->posture == image_portrait) {
936
1.37M
            penum->dst_width = row_extent.x;
937
1.37M
            penum->dst_height = col_extent.y;
938
1.37M
        } else {
939
179k
            penum->dst_width = col_extent.x;
940
179k
            penum->dst_height = row_extent.y;
941
179k
        }
942
        /* For gs_image_class_0_interpolate. */
943
1.55M
        penum->yi0 = fixed2int_pixround_perfect(dda_current(penum->dda.row.y)); /* For gs_image_class_0_interpolate. */
944
1.55M
        if (penum->rect.y) {
945
304k
            int y = penum->rect.y;
946
947
36.4M
            while (y--) {
948
36.1M
                dda_next(penum->dda.row.x);
949
36.1M
                dda_next(penum->dda.row.y);
950
36.1M
            }
951
304k
        }
952
1.55M
        penum->cur.x = penum->prev.x = dda_current(penum->dda.row.x);
953
1.55M
        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.55M
        dda_init(penum->dda.strip.x, penum->cur.x, row_extent.x, width);
958
1.55M
        dda_init(penum->dda.strip.y, penum->cur.y, row_extent.y, width);
959
1.55M
        if (dda_will_overflow(penum->dda.strip.x) ||
960
1.55M
            dda_will_overflow(penum->dda.strip.y))
961
0
        {
962
0
            code = gs_error_rangecheck;
963
0
            goto fail;
964
0
        }
965
1.55M
        if (penum->rect.x) {
966
7.01k
            dda_advance(penum->dda.strip.x, penum->rect.x);
967
7.01k
            dda_advance(penum->dda.strip.y, penum->rect.x);
968
7.01k
        }
969
1.55M
        {
970
1.55M
            fixed ox = dda_current(penum->dda.strip.x);
971
1.55M
            fixed oy = dda_current(penum->dda.strip.y);
972
973
1.55M
            if (!penum->clip_image)     /* i.e., not clip region */
974
1.55M
                penum->clip_image =
975
1.55M
                    (fixed_pixround(ox + epx) < fixed_pixround(cbox.p.x) ?
976
1.48M
                     image_clip_xmin : 0) +
977
1.55M
                    (fixed_pixround(ox + eqx) >= fixed_pixround(cbox.q.x) ?
978
1.18M
                     image_clip_xmax : 0) +
979
1.55M
                    (fixed_pixround(oy + epy) < fixed_pixround(cbox.p.y) ?
980
1.23M
                     image_clip_ymin : 0) +
981
1.55M
                    (fixed_pixround(oy + eqy) >= fixed_pixround(cbox.q.y) ?
982
1.08M
                     image_clip_ymax : 0);
983
1.55M
        }
984
1.55M
    }
985
0
    penum->y = 0;
986
1.55M
    penum->used.x = 0;
987
1.55M
    penum->used.y = 0;
988
1.55M
    if (penum->clip_image && pcpath) {  /* Set up the clipping device. */
989
335k
        gx_device_clip *cdev =
990
335k
            gs_alloc_struct(mem, gx_device_clip,
991
335k
                            &st_device_clip, "image clipper");
992
993
335k
        if (cdev == NULL) {
994
0
            code = gs_error_VMerror;
995
0
            goto fail;
996
0
        }
997
335k
        gx_make_clip_device_in_heap(cdev, pcpath, dev, mem);
998
335k
        penum->clip_dev = cdev;
999
335k
        penum->dev = (gx_device *)cdev; /* Will restore this in a mo. Hacky! */
1000
335k
    }
1001
1.55M
    if (penum->use_rop) {       /* Set up the RasterOp source device. */
1002
278
        gx_device_rop_texture *rtdev;
1003
1004
278
        code = gx_alloc_rop_texture_device(&rtdev, mem,
1005
278
                                           "image RasterOp");
1006
278
        if (code < 0)
1007
0
            goto fail;
1008
        /* The 'target' must not be NULL for gx_make_rop_texture_device */
1009
278
        if (!penum->clip_dev && !dev)
1010
0
            return_error(gs_error_undefined);
1011
1012
278
        gx_make_rop_texture_device(rtdev,
1013
278
                                   (penum->clip_dev != 0 ?
1014
266
                                    (gx_device *) penum->clip_dev :
1015
278
                                    dev), lop, pdcolor);
1016
278
        gx_device_retain((gx_device *)rtdev, true);
1017
278
        penum->rop_dev = rtdev;
1018
278
        penum->dev = (gx_device *)rtdev; /* Will restore this in a mo. Hacky! */
1019
278
    }
1020
1.55M
    {
1021
1.55M
        static sample_unpack_proc_t procs[2][6] = {
1022
1.55M
        {   sample_unpack_1, sample_unpack_2,
1023
1.55M
            sample_unpack_4, sample_unpack_8,
1024
1.55M
            sample_unpack_12, sample_unpack_16
1025
1.55M
        },
1026
1.55M
        {   sample_unpack_1_interleaved, sample_unpack_2_interleaved,
1027
1.55M
            sample_unpack_4_interleaved, sample_unpack_8_interleaved,
1028
1.55M
            sample_unpack_12, sample_unpack_16
1029
1.55M
        }};
1030
1.55M
        int num_planes = penum->num_planes;
1031
1.55M
        bool interleaved = (num_planes == 1 && penum->plane_depths[0] != penum->bps);
1032
1.55M
        irender_proc_t render_fn = NULL;
1033
1.55M
        int i;
1034
1035
1.55M
        if (interleaved) {
1036
666k
            int num_components = penum->plane_depths[0] / penum->bps;
1037
1038
2.02M
            for (i = 1; i < num_components; i++) {
1039
1.35M
                if (decode[0] != decode[i * 2 + 0] ||
1040
1.35M
                    decode[1] != decode[i * 2 + 1])
1041
2
                    break;
1042
1.35M
            }
1043
666k
            if (i == num_components)
1044
666k
                interleaved = false; /* Use single table. */
1045
666k
        }
1046
1.55M
        penum->unpack = procs[interleaved][index_bps];
1047
1048
1.55M
        if_debug1m('b', mem, "[b]unpack=%d\n", bps);
1049
        /* Set up pixel0 for image class procedures. */
1050
1.55M
        penum->dda.pixel0 = penum->dda.strip;
1051
1.55M
        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.55M
                penum->render = render_fn;
1059
1.55M
                break;
1060
1.55M
            }
1061
6.39M
        }
1062
1.55M
        penum->dev = dev; /* Restore this (in case it was changed to cdev or rtdev) */
1063
1.55M
        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.55M
    }
1068
1.55M
    return 0;
1069
1070
53
fail:
1071
53
    gs_free_object(mem, buffer, "image buffer");
1072
53
    gs_free_object(mem, penum->clues, "gx_image_enum_begin");
1073
53
    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
53
    gs_free_object(mem, penum->clip_dev, "image clipper");
1078
53
    rc_decrement_cs(penum->pcs, "error in gx_begin_image1");
1079
53
    penum->pcs = NULL;
1080
53
    gs_free_object(mem, penum, "gx_begin_image1");
1081
53
    return code;
1082
1.55M
}
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
5.03k
{
1089
5.03k
    if (color_is_pure(pdcolor)) {
1090
5.03k
        gx_color_value rgb[3];
1091
1092
5.03k
        (*dev_proc(dev, map_color_rgb)) (dev, gx_dc_pure_color(pdcolor),
1093
5.03k
                                         rgb);
1094
5.03k
        if (!(rgb[0] | rgb[1] | rgb[2]))
1095
4.63k
            return 0;
1096
396
        if ((rgb[0] & rgb[1] & rgb[2]) == gx_max_color_value)
1097
396
            return 1;
1098
396
    }
1099
0
    return -1;
1100
5.03k
}
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
70.0k
{
1156
70.0k
    int num_des_comp = penum->dev->color_info.num_components;
1157
70.0k
    int num_src_comp;
1158
70.0k
    int num_entries = 1 << bps;
1159
70.0k
    bool need_decode = penum->icc_setup.need_decode;
1160
70.0k
    bool has_transfer = penum->icc_setup.has_transfer;
1161
70.0k
    byte value;
1162
70.0k
    bool decode_scale = true;
1163
70.0k
    int k, kk;
1164
70.0k
    byte psrc[4];
1165
70.0k
    byte *temp_buffer;
1166
70.0k
    byte *byte_ptr;
1167
70.0k
    bool is_indexed = (gs_color_space_get_index(penum->pcs) ==
1168
70.0k
                                            gs_color_space_index_Indexed);
1169
70.0k
    bool free_temp_buffer = true;
1170
70.0k
    gsicc_bufferdesc_t input_buff_desc;
1171
70.0k
    gsicc_bufferdesc_t output_buff_desc;
1172
70.0k
    gx_color_value conc[GX_DEVICE_COLOR_MAX_COMPONENTS];
1173
70.0k
    int code;
1174
1175
70.0k
    if (penum->icc_link == NULL) {
1176
0
        return gs_rethrow(-1, "ICC Link not created during image render color");
1177
0
    }
1178
70.0k
    if (is_indexed) {
1179
1.46k
        num_src_comp = gs_color_space_num_components(penum->pcs->base_space);
1180
68.6k
    } 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
68.6k
        if (penum->icc_link->is_identity && !need_decode && !has_transfer) {
1185
68.3k
            return 0;
1186
68.3k
        }
1187
250
        num_src_comp = 1;
1188
250
    }
1189
    /* Allocate cache of device contone values */
1190
1.71k
    penum->color_cache = gs_alloc_struct(penum->memory, gx_image_color_cache_t,
1191
1.71k
                                         &st_color_cache,
1192
1.71k
                                         "image_init_color_cache");
1193
1.71k
    if (penum->color_cache == NULL)
1194
0
        return_error(gs_error_VMerror);
1195
1196
1.71k
    penum->color_cache->device_contone = (byte*) gs_alloc_bytes(penum->memory,
1197
1.71k
                   (size_t)num_des_comp * num_entries * sizeof(byte), "image_init_color_cache");
1198
1.71k
    penum->color_cache->is_transparent = (bool*) gs_alloc_bytes(penum->memory,
1199
1.71k
             (size_t)num_entries * sizeof(bool), "image_init_color_cache");
1200
1.71k
    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.71k
    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.71k
    if (need_decode) {
1218
0
        decode_scale = decode_range_needed(penum);
1219
0
    }
1220
1.71k
    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.71k
    } 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.71k
        temp_buffer = (byte*) gs_alloc_bytes(penum->memory,
1266
1.71k
                                             (size_t)num_entries * num_src_comp,
1267
1.71k
                                             "image_init_color_cache");
1268
1.71k
        if (temp_buffer == NULL)
1269
0
            return_error(gs_error_VMerror);
1270
1271
1.71k
        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.71k
        } else {
1286
            /* No Decode */
1287
1.71k
            if (is_indexed) {
1288
                /* If index uses a num_entries sized table then just use its pointer */
1289
1.46k
                if (penum->pcs->params.indexed.use_proc ||
1290
1.46k
                    penum->pcs->params.indexed.hival < (num_entries - 1)) {
1291
                    /* Have to do the slow way */
1292
6.07k
                    for (k = 0; k <= penum->pcs->params.indexed.hival; k++) {
1293
5.90k
                        gs_cspace_indexed_lookup_bytes(penum->pcs, (float)k, psrc);
1294
5.90k
                        memcpy(&(temp_buffer[k * num_src_comp]), psrc, num_src_comp);
1295
5.90k
                    }
1296
                    /* just use psrc results from converting 'hival' to fill the remaining slots */
1297
36.4k
                    for (; k < num_entries; k++) {
1298
36.3k
                        memcpy(&(temp_buffer[k * num_src_comp]), psrc, num_src_comp);
1299
36.3k
                    }
1300
1.30k
                } else {
1301
                    /* Use the index table directly. */
1302
1.30k
                    gs_free_object(penum->memory, temp_buffer, "image_init_color_cache");
1303
1.30k
                    free_temp_buffer = false;
1304
1.30k
                    temp_buffer = (byte *)(penum->pcs->params.indexed.lookup.table.data);
1305
1.30k
                }
1306
1.46k
            } else {
1307
                /* CM only */
1308
64.2k
                for (k = 0; k < num_entries; k++) {
1309
64.0k
                    temp_buffer[k] = k;
1310
64.0k
                }
1311
250
            }
1312
1.71k
        }
1313
        /* Set up the buffer descriptors. */
1314
1.71k
        gsicc_init_buffer(&input_buff_desc, num_src_comp, 1, false, false, false,
1315
1.71k
                          0, num_entries * num_src_comp, 1, num_entries);
1316
1.71k
        gsicc_init_buffer(&output_buff_desc, num_des_comp, 1, false, false, false,
1317
1.71k
                          0, num_entries * num_des_comp,
1318
1.71k
                      1, num_entries);
1319
1.71k
        code = (penum->icc_link->procs.map_buffer)(penum->dev, penum->icc_link,
1320
1.71k
                                            &input_buff_desc, &output_buff_desc,
1321
1.71k
                                            (void*) temp_buffer,
1322
1.71k
                                            (void*) penum->color_cache->device_contone);
1323
1.71k
        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.71k
        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.71k
        if (free_temp_buffer)
1341
415
            gs_free_object(penum->memory, temp_buffer, "image_init_color_cache");
1342
1.71k
    }
1343
1.71k
    return 0;
1344
1.71k
}
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
879k
{
1350
    /* Initialize the color table */
1351
879k
#define ictype(i)\
1352
879k
  penum->clues[i].dev_color.type
1353
1354
879k
    switch ((spp == 1 ? bps : 8)) {
1355
601k
        case 8:         /* includes all color images */
1356
601k
            {
1357
601k
                register gx_image_clue *pcht = &penum->clues[0];
1358
601k
                register int n = 64;    /* 8 bits means 256 clues, do   */
1359
                                        /* 4 at a time for efficiency   */
1360
38.5M
                do {
1361
38.5M
                    pcht[0].dev_color.type =
1362
38.5M
                        pcht[1].dev_color.type =
1363
38.5M
                        pcht[2].dev_color.type =
1364
38.5M
                        pcht[3].dev_color.type =
1365
38.5M
                        gx_dc_type_none;
1366
38.5M
                    pcht[0].key = pcht[1].key =
1367
38.5M
                        pcht[2].key = pcht[3].key = 0;
1368
38.5M
                    pcht += 4;
1369
38.5M
                }
1370
38.5M
                while (--n > 0);
1371
601k
                penum->clues[0].key = 1;        /* guarantee no hit */
1372
601k
                break;
1373
0
            }
1374
594
        case 4:
1375
594
            ictype(17) = ictype(2 * 17) = ictype(3 * 17) =
1376
594
                ictype(4 * 17) = ictype(6 * 17) = ictype(7 * 17) =
1377
594
                ictype(8 * 17) = ictype(9 * 17) = ictype(11 * 17) =
1378
594
                ictype(12 * 17) = ictype(13 * 17) = ictype(14 * 17) =
1379
594
                gx_dc_type_none;
1380
            /* falls through */
1381
1.60k
        case 2:
1382
1.60k
            ictype(5 * 17) = ictype(10 * 17) = gx_dc_type_none;
1383
879k
#undef ictype
1384
879k
    }
1385
879k
}
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.05M
{
1394
1.05M
    int ci, decode_type, code;
1395
1.05M
    static const float default_decode[] = {
1396
1.05M
        0.0, 1.0, 0.0, 1.0, 0.0, 1.0, 0.0, 1.0, 0.0, 1.0
1397
1.05M
    };
1398
1399
    /* Clues are only used with image_mono_render */
1400
1.05M
    if (spp == 1) {
1401
358k
        image_init_clues(penum, bps, spp);
1402
358k
    }
1403
1.05M
    decode_type = 3; /* 0=custom, 1=identity, 2=inverted, 3=impossible */
1404
2.80M
    for (ci = 0; ci < spp; ci +=2 ) {
1405
1.75M
        decode_type &= (decode[ci] == 0. && decode[ci + 1] == 1.) |
1406
1.75M
                       (decode[ci] == 1. && decode[ci + 1] == 0.) << 1;
1407
1.75M
    }
1408
1409
    /* Initialize the maps from samples to intensities. */
1410
3.55M
    for (ci = 0; ci < spp; ci++) {
1411
2.50M
        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.50M
        const float *this_decode = &decode[ci * 2];
1418
2.50M
        const float *map_decode;        /* decoding used to */
1419
                                        /* construct the expansion map */
1420
2.50M
        const float *real_decode;       /* decoding for expanded samples */
1421
1422
2.50M
        map_decode = real_decode = this_decode;
1423
2.50M
        if (!(decode_type & 1)) {
1424
5.71k
            if ((decode_type & 2) && bps <= 8) {
1425
419
                real_decode = default_decode;
1426
5.29k
            } else {
1427
5.29k
                *pdcb = false;
1428
5.29k
                map_decode = default_decode;
1429
5.29k
            }
1430
5.71k
        }
1431
2.50M
        if (bps > 2 || format != gs_image_format_chunky) {
1432
2.47M
            if (bps <= 8)
1433
2.47M
                image_init_map(&pmap->table.lookup8[0], 1 << bps,
1434
2.47M
                               map_decode);
1435
2.47M
        } else {                /* The map index encompasses more than one pixel. */
1436
22.5k
            byte map[4];
1437
22.5k
            register int i;
1438
1439
22.5k
            image_init_map(&map[0], 1 << bps, map_decode);
1440
22.5k
            switch (bps) {
1441
22.0k
                case 1:
1442
22.0k
                    {
1443
22.0k
                        register bits32 *p = &pmap->table.lookup4x1to32[0];
1444
1445
22.0k
                        if (map[0] == 0 && map[1] == 0xff)
1446
21.9k
                            memcpy((byte *) p, lookup4x1to32_identity, 16 * 4);
1447
35
                        else if (map[0] == 0xff && map[1] == 0)
1448
35
                            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
22.0k
                    }
1456
22.0k
                    break;
1457
505
                case 2:
1458
505
                    {
1459
505
                        register bits16 *p = &pmap->table.lookup2x2to16[0];
1460
1461
8.58k
                        for (i = 0; i < 16; i++, p++)
1462
8.08k
                            ((byte *) p)[0] = map[i >> 2],
1463
8.08k
                                ((byte *) p)[1] = map[i & 3];
1464
505
                    }
1465
505
                    break;
1466
22.5k
            }
1467
22.5k
        }
1468
2.50M
        pmap->decode_base /* = decode_lookup[0] */  = real_decode[0];
1469
2.50M
        pmap->decode_factor =
1470
2.50M
            (real_decode[1] - real_decode[0]) /
1471
2.50M
            (bps <= 8 ? 255.0 : (float)frac_1);
1472
2.50M
        pmap->decode_max /* = decode_lookup[15] */  = real_decode[1];
1473
2.50M
        if (decode_type) {
1474
2.49M
            pmap->decoding = sd_none;
1475
2.49M
            pmap->inverted = map_decode[0] != 0;
1476
2.49M
        } else if (bps <= 4) {
1477
555
            int step = 15 / ((1 << bps) - 1);
1478
555
            int i;
1479
1480
555
            pmap->decoding = sd_lookup;
1481
2.94k
            for (i = 15 - step; i > 0; i -= step)
1482
2.38k
                pmap->decode_lookup[i] = pmap->decode_base +
1483
2.38k
                    i * (255.0 / 15) * pmap->decode_factor;
1484
555
            pmap->inverted = 0;
1485
4.73k
        } else {
1486
4.73k
            pmap->decoding = sd_compute;
1487
4.73k
            pmap->inverted = 0;
1488
4.73k
        }
1489
2.50M
        if (spp == 1) {         /* and ci == 0 *//* Pre-map entries 0 and 255. */
1490
358k
            gs_client_color cc;
1491
1492
            /* Image clues are used in this case */
1493
358k
            cc.paint.values[0] = real_decode[0];
1494
358k
            code = (*pcs->type->remap_color) (&cc, pcs, penum->icolor0,
1495
358k
                                       pgs, dev, gs_color_select_source);
1496
358k
            if (code < 0)
1497
5
                return code;
1498
358k
            cc.paint.values[0] = real_decode[1];
1499
358k
            code = (*pcs->type->remap_color) (&cc, pcs, penum->icolor1,
1500
358k
                                       pgs, dev, gs_color_select_source);
1501
358k
            if (code < 0)
1502
0
                return code;
1503
358k
        }
1504
2.50M
    }
1505
1.05M
    return 0;
1506
1.05M
}
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.50M
{
1513
2.50M
    float min_v = decode[0];
1514
2.50M
    float diff_v = decode[1] - min_v;
1515
1516
2.50M
    if (diff_v == 1 || diff_v == -1) {  /* We can do the stepping with integers, without overflow. */
1517
2.50M
        byte *limit = map + map_size;
1518
2.50M
        uint value = (uint)(min_v * 0xffffL);
1519
2.50M
        int diff = (int)(diff_v * (0xffffL / (map_size - 1)));
1520
1521
638M
        for (; map != limit; map++, value += diff)
1522
636M
            *map = value >> 8;
1523
2.50M
    } 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.50M
}
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
1.42k
{
1542
1.42k
    uint scale = 255 / ((1 << penum->bps) - 1);
1543
1.42k
    uint *values = &penum->mask_color.values[component_index * 2];
1544
1.42k
    uint v0 = values[0] *= scale;
1545
1.42k
    uint v1 = values[1] *= scale;
1546
1547
1.42k
    if (penum->map[component_index].decoding == sd_none &&
1548
697
        penum->map[component_index].inverted
1549
1.42k
        ) {
1550
0
        values[0] = 255 - v1;
1551
0
        values[1] = 255 - v0;
1552
0
    }
1553
1.42k
}
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
732k
{
1559
732k
    int k;
1560
1561
1.92M
    for (k = 0; k < num_comps; k++) {
1562
1.49M
        if (pgs->effective_transfer[k]->proc != gs_identity_transfer) {
1563
301k
            return(true);
1564
301k
        }
1565
1.49M
    }
1566
430k
    return(false);
1567
732k
}