Coverage Report

Created: 2025-12-31 07:31

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