Coverage Report

Created: 2026-04-01 07:17

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ghostpdl/base/gximono.c
Line
Count
Source
1
/* Copyright (C) 2001-2026 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
/* General mono-component image rendering */
18
#include "gx.h"
19
#include "memory_.h"
20
#include "gpcheck.h"
21
#include "gserrors.h"
22
#include "gxfixed.h"
23
#include "gxarith.h"
24
#include "gxmatrix.h"
25
#include "gsccolor.h"
26
#include "gspaint.h"
27
#include "gsutil.h"
28
#include "gxdevice.h"
29
#include "gxcmap.h"
30
#include "gxdcolor.h"
31
#include "gxgstate.h"
32
#include "gxdevmem.h"
33
#include "gdevmem.h"            /* for mem_mono_device */
34
#include "gxcpath.h"
35
#include "gximage.h"
36
#include "gzht.h"
37
#include "gsicc.h"
38
#include "gsicc_cache.h"
39
#include "gsicc_cms.h"
40
#include "gxcie.h"
41
#include "gscie.h"
42
#include "gxht_thresh.h"
43
#include "gxdda.h"
44
#include "gxdevsop.h"
45
#ifdef WITH_CAL
46
#include "cal.h"
47
#endif
48
49
#define fastfloor(x) (((int)(x)) - (((x)<0) && ((x) != (float)(int)(x))))
50
51
/* Enable the following define to perform a little extra work to stop
52
 * spurious valgrind errors. The code should perform perfectly even without
53
 * this enabled, but enabling it makes debugging much easier.
54
 */
55
/* #define PACIFY_VALGRIND */
56
57
/* ------ Strategy procedure ------ */
58
59
/* Check the prototype. */
60
iclass_proc(gs_image_class_3_mono);
61
62
static irender_proc(image_render_mono);
63
#ifndef WITH_CAL
64
static irender_proc(image_render_mono_ht);
65
#else
66
static irender_proc(image_render_mono_ht_cal);
67
static int image_render_mono_ht_cal_skip_line(gx_image_enum *penum,
68
                gx_device *dev);
69
70
static void
71
halftone_callback(cal_halftone_data_t *ht, void *arg)
72
{
73
    gx_device *dev = arg;
74
    gx_color_index dev_white = gx_device_white(dev);
75
    gx_color_index dev_black = gx_device_black(dev);
76
77
    if (dev->num_planar_planes) {
78
        (*dev_proc(dev, copy_planes)) (dev, ht->data, ht->x + (ht->offset_x<<3), ht->raster,
79
            gx_no_bitmap_id, ht->x, ht->y, ht->w, ht->h,
80
            ht->plane_raster);
81
    } else {
82
        (*dev_proc(dev, copy_mono)) (dev, ht->data, ht->x + (ht->offset_x<<3), ht->raster,
83
            gx_no_bitmap_id, ht->x, ht->y, ht->w, ht->h, dev_white,
84
            dev_black);
85
    }
86
}
87
88
static cal_halftone*
89
halftone_init(gx_image_enum *penum)
90
{
91
    cal_halftone *cal_ht = NULL;
92
    gx_dda_fixed dda_ht;
93
    int k;
94
    gx_ht_order *d_order;
95
    int code;
96
    byte *cache = (penum->color_cache != NULL ? penum->color_cache->device_contone : NULL);
97
    cal_matrix matrix;
98
    int clip_x, clip_y;
99
    gx_device_halftone *pdht = gx_select_dev_ht(penum->pgs);
100
101
    if (!gx_device_must_halftone(penum->dev))
102
        return NULL;
103
104
    if (penum->pgs == NULL || pdht == NULL)
105
        return NULL;
106
107
    dda_ht = penum->dda.pixel0.x;
108
    if (penum->dxx > 0)
109
        dda_translate(dda_ht, -fixed_epsilon);
110
    matrix.xx = penum->matrix.xx;
111
    matrix.xy = penum->matrix.xy;
112
    matrix.yx = penum->matrix.yx;
113
    matrix.yy = penum->matrix.yy;
114
    matrix.tx = penum->matrix.tx + matrix.xx * penum->rect.x + matrix.yx * penum->rect.y;
115
    matrix.ty = penum->matrix.ty + matrix.xy * penum->rect.x + matrix.yy * penum->rect.y;
116
117
    clip_x = fixed2int(penum->clip_outer.p.x);
118
    clip_y = fixed2int(penum->clip_outer.p.y);
119
    cal_ht = cal_halftone_init(penum->memory->gs_lib_ctx->core->cal_ctx,
120
                               penum->memory->non_gc_memory,
121
                               penum->rect.w,
122
                               penum->rect.h,
123
                               &matrix,
124
                               penum->dev->color_info.num_components,
125
                               cache,
126
                               clip_x,
127
                               clip_y,
128
                               fixed2int_ceiling(penum->clip_outer.q.x) - clip_x,
129
                               fixed2int_ceiling(penum->clip_outer.q.y) - clip_y,
130
                               penum->adjust);
131
    if (cal_ht == NULL)
132
        goto fail;
133
134
    for (k = 0; k < pdht->num_comp; k++) {
135
        d_order = &(pdht->components[k].corder);
136
        code = gx_ht_construct_threshold(d_order, penum->dev, penum->pgs, k);
137
        if (code < 0)
138
            goto fail;
139
        if (cal_halftone_add_screen(penum->memory->gs_lib_ctx->core->cal_ctx,
140
                                    penum->memory->non_gc_memory,
141
                                    cal_ht,
142
                                    pdht->components[k].corder.threshold_inverted,
143
                                    pdht->components[k].corder.width,
144
                                    pdht->components[k].corder.full_height,
145
                                    penum->pgs->screen_phase[k].x,
146
                                    -penum->pgs->screen_phase[k].y,
147
                                    pdht->components[k].corder.threshold) < 0)
148
            goto fail;
149
    }
150
151
    return cal_ht;
152
153
fail:
154
    cal_halftone_fin(cal_ht, penum->memory->non_gc_memory);
155
    return NULL;
156
}
157
#endif
158
159
int
160
gs_image_class_3_mono(gx_image_enum * penum, irender_proc_t *render_fn)
161
1.31M
{
162
1.31M
#if USE_FAST_HT_CODE
163
1.31M
    bool use_fast_code = true;
164
#else
165
    bool use_fast_code = false;
166
#endif
167
1.31M
    int code = 0;
168
    /* Set up the link now */
169
1.31M
    const gs_color_space *pcs;
170
1.31M
    gsicc_rendering_param_t rendering_params;
171
1.31M
    cmm_dev_profile_t *dev_profile;
172
1.31M
    bool dev_color_ok = false;
173
1.31M
    bool is_planar_dev = penum->dev->num_planar_planes;
174
175
1.31M
    if (penum->spp == 1) {
176
        /* At this point in time, only use the ht approach if our device
177
           uses halftoning, and our source image is a reasonable size.  We
178
           probably don't want to do this if we have a bunch of tiny little
179
           images.  Then the rect fill approach is probably not all that bad.
180
           Also for now avoid images that include a type3 image mask.  Due
181
           to the limited precision and mismatch of the stepping space in which
182
           the interpolations occur this can cause a minor mismatch at large
183
           scalings */
184
185
        /* Allow this for CMYK planar and mono binary halftoned devices */
186
630k
        dev_color_ok = ((penum->dev->color_info.num_components == 1 &&
187
558k
                         penum->dev->color_info.depth == 1) ||
188
#if 0
189
                         /* Don't allow CMYK Planar devices just yet */
190
                         0);
191
#else
192
396k
                        (penum->dev->color_info.num_components == 4 &&
193
1.12k
                         penum->dev->color_info.depth == 4 && is_planar_dev));
194
630k
#endif
195
196
630k
        if (use_fast_code && penum->pcs != NULL && dev_color_ok &&
197
71.3k
            penum->bps == 8 && (penum->posture == image_portrait
198
34
            || penum->posture == image_landscape) &&
199
70.9k
            penum->image_parent_type == gs_image_type1 &&
200
70.2k
            gx_transfer_is_monotonic(penum->pgs, 0)) {
201
70.2k
            penum->icc_setup.need_decode = false;
202
            /* Check if we need to do any decoding.  */
203
70.2k
            if ( penum->map[0].decoding != sd_none ) {
204
1.49k
                if (!(penum->map[0].decoding == sd_compute
205
1.49k
                     && penum->map[0].decode_factor == 1.0 &&
206
1.48k
                        penum->map[0].decode_lookup[0] == 0.0)) {
207
14
                    penum->icc_setup.need_decode = true;
208
14
                }
209
1.49k
            }
210
70.2k
            code = dev_proc(penum->dev, get_profile)(penum->dev, &dev_profile);
211
70.2k
            if (code < 0)
212
0
                return code;
213
214
            /* Define the rendering intents */
215
70.2k
            rendering_params.black_point_comp = penum->pgs->blackptcomp;
216
70.2k
            rendering_params.graphics_type_tag = GS_IMAGE_TAG;
217
70.2k
            rendering_params.override_icc = false;
218
70.2k
            rendering_params.preserve_black = gsBKPRESNOTSPECIFIED;
219
70.2k
            rendering_params.rendering_intent = penum->pgs->renderingintent;
220
70.2k
            rendering_params.cmm = gsCMM_DEFAULT;
221
70.2k
            if (gs_color_space_get_index(penum->pcs) ==
222
70.2k
                gs_color_space_index_Indexed) {
223
1.49k
                pcs = penum->pcs->base_space;
224
68.7k
            } else {
225
68.7k
                pcs = penum->pcs;
226
68.7k
            }
227
70.2k
            if (gs_color_space_is_PSCIE(pcs) && pcs->icc_equivalent != NULL) {
228
0
                pcs = pcs->icc_equivalent;
229
0
            }
230
231
            /* The code below falls over if cmm->icc_profile_data is NULL.
232
             * For now, just drop out. Michael can review this when he
233
             * returns. */
234
70.2k
            if (pcs->cmm_icc_profile_data == NULL)
235
23
                goto not_fast_halftoning;
236
237
70.1k
            penum->icc_setup.is_lab = pcs->cmm_icc_profile_data->islab;
238
70.1k
            penum->icc_setup.must_halftone = gx_device_must_halftone(penum->dev);
239
            /* The effective transfer is built into the threshold array */
240
70.1k
            penum->icc_setup.has_transfer = false;
241
70.1k
            if (penum->icc_setup.is_lab)
242
38
                penum->icc_setup.need_decode = false;
243
70.1k
            if (penum->icc_link == NULL) {
244
70.1k
                penum->icc_link = gsicc_get_link(penum->pgs, penum->dev, pcs, NULL,
245
70.1k
                    &rendering_params, penum->memory);
246
70.1k
            }
247
            /* PS CIE color spaces may have addition decoding that needs to
248
               be performed to ensure that the range of 0 to 1 is provided
249
               to the CMM since ICC profiles are restricted to that range
250
               but the PS color spaces are not. */
251
70.1k
            penum->use_cie_range = false;
252
70.1k
            if (gs_color_space_is_PSCIE(penum->pcs) &&
253
0
                penum->pcs->icc_equivalent != NULL) {
254
                /* We have a PS CIE space.  Check the range */
255
0
                if ( !check_cie_range(penum->pcs) ) {
256
                    /* It is not 0 to 1.  We will be doing decode
257
                       plus an additional linear adjustment */
258
0
                    penum->use_cie_range = (get_cie_range(penum->pcs) != NULL);
259
0
                }
260
0
            }
261
            /* If the image has more than 256 pixels then go ahead and
262
               precompute the contone device colors for all of our 256 source
263
               values.  We should not be taking this path for cases where
264
               we have lots of tiny little images.  Mark those that are
265
               transparent or masked also at this time.  Since halftoning will
266
               be done via thresholding we will keep clues in continuous tone */
267
70.1k
            code = image_init_color_cache(penum, penum->bps, penum->spp);
268
70.1k
            if (code >= 0) {
269
#ifdef WITH_CAL
270
                penum->cal_ht = halftone_init(penum);
271
                if (penum->cal_ht != NULL)
272
                {
273
                    penum->skip_next_line = image_render_mono_ht_cal_skip_line;
274
                    *render_fn = &image_render_mono_ht_cal;
275
                    return code;
276
                }
277
#else
278
70.1k
                code = gxht_thresh_image_init(penum);
279
70.1k
                if (code >= 0) {
280
70.1k
                    *render_fn = &image_render_mono_ht;
281
70.1k
                    return code;
282
70.1k
                }
283
70.1k
#endif
284
70.1k
            }
285
70.1k
        }
286
560k
not_fast_halftoning:
287
        /*
288
         * Use the slow loop for imagemask with a halftone or a non-default
289
         * logical operation.
290
         */
291
560k
        penum->slow_loop =
292
560k
            (penum->masked && !color_is_pure(penum->icolor0)) ||
293
560k
            penum->use_rop;
294
        /* We can bypass X clipping for portrait mono-component images. */
295
560k
        if (!(penum->slow_loop || penum->posture != image_portrait))
296
460k
            penum->clip_image &= ~(image_clip_xmin | image_clip_xmax);
297
560k
        if_debug0m('b', penum->memory, "[b]render=mono\n");
298
        /* Precompute values needed for rasterizing. */
299
560k
        penum->dxx =
300
560k
            float2fixed(penum->matrix.xx + fixed2float(fixed_epsilon) / 2);
301
        /*
302
         * Scale the mask colors to match the scaling of each sample to a
303
         * full byte.  Also, if black or white is transparent, reset icolor0
304
         * or icolor1, which are used directly in the fast case loop.
305
         */
306
560k
        if (penum->use_mask_color) {
307
940
            gx_image_scale_mask_colors(penum, 0);
308
940
            if (penum->mask_color.values[0] <= 0)
309
930
                color_set_null(penum->icolor0);
310
940
            if (penum->mask_color.values[1] >= 255)
311
0
                color_set_null(penum->icolor1);
312
940
        }
313
        /* Reset the clues here, rather than in image_render_mono as
314
         * previously. Even doing so this often may be overzealous. */
315
560k
        image_init_clues(penum, penum->bps, penum->spp);
316
560k
        *render_fn = &image_render_mono;
317
560k
    }
318
1.24M
    return 0;
319
1.31M
}
320
321
#define USE_SET_GRAY_FUNCTION 0
322
#if USE_SET_GRAY_FUNCTION
323
/* Temporary function to make it easier to debug the uber-macro below */
324
static inline int
325
image_set_gray(byte sample_value, const bool masked, uint mask_base,
326
                uint mask_limit, gx_device_color **ppdevc, gs_client_color *cc,
327
                const gs_color_space *pcs, const gs_gstate *pgs,
328
                gx_device * dev, gs_color_select_t gs_color_select_source,
329
                gx_image_enum * penum)
330
{
331
   cs_proc_remap_color((*remap_color));
332
   int code;
333
   gx_device_color *pdevc;
334
   pdevc = *ppdevc = &penum->clues[sample_value].dev_color;
335
336
   if (!color_is_set(pdevc)) {
337
       if ((uint)(sample_value - mask_base) < mask_limit) {
338
           color_set_null(pdevc);
339
       } else {
340
            switch ( penum->map[0].decoding )
341
            {
342
            case sd_none:
343
            cc->paint.values[0] = (sample_value) * (1.0f / 255.0f);  /* faster than / */
344
            break;
345
            case sd_lookup:     /* <= 4 significant bits */
346
            cc->paint.values[0] =
347
              penum->map[0].decode_lookup[(sample_value) >> 4];
348
            break;
349
            case sd_compute:
350
            cc->paint.values[0] =
351
              penum->map[0].decode_base + (sample_value) * penum->map[0].decode_factor;
352
            }
353
            remap_color = pcs->type->remap_color;
354
            code = (*remap_color)(cc, pcs, pdevc, pgs, dev, gs_color_select_source);
355
            return(code);
356
        }
357
    } else if (!color_is_pure(pdevc)) {
358
        code = gx_color_load_select(pdevc, pgs, dev, gs_color_select_source);
359
        if (code < 0)
360
            return(code);
361
    }
362
    return(0);
363
}
364
#endif
365
366
/*
367
 * Rendering procedure for general mono-component images, dealing with
368
 * multiple bit-per-sample images, general transformations, arbitrary
369
 * single-component color spaces (DeviceGray, DevicePixel, CIEBasedA,
370
 * Separation, Indexed), and color masking. This procedure handles a
371
 * single scan line.
372
 */
373
static int
374
image_render_mono(gx_image_enum * penum, const byte * buffer, int data_x,
375
                  uint w, int h, gx_device * dev)
376
9.28M
{
377
9.28M
    const gs_gstate *pgs = penum->pgs;
378
9.28M
    gs_logical_operation_t lop = penum->log_op;
379
9.28M
    const bool masked = penum->masked;
380
9.28M
    const gs_color_space *pcs = NULL;   /* only set for non-masks */
381
9.28M
    cs_proc_remap_color((*remap_color)) = NULL; /* ditto */
382
9.28M
    gs_client_color cc;
383
9.28M
    gx_device_color *pdevc = penum->icolor1;    /* color for masking */
384
9.28M
    uint mask_base =            /* : 0 to pacify Valgrind */
385
9.28M
        (penum->use_mask_color ? penum->mask_color.values[0] : 0);
386
9.28M
    uint mask_limit =
387
9.28M
        (penum->use_mask_color ?
388
9.27M
         penum->mask_color.values[1] - mask_base + 1 : 0);
389
/*
390
 * Free variables of IMAGE_SET_GRAY:
391
 *   Read: penum, pgs, dev, mask_base, mask_limit
392
 *   Set: pdevc, code, cc
393
 */
394
9.28M
#define IMAGE_SET_GRAY(sample_value)\
395
63.6M
  BEGIN\
396
63.6M
    pdevc = &penum->clues[sample_value].dev_color;\
397
63.6M
    if (!color_is_set(pdevc)) {\
398
4.79M
        if ((uint)(sample_value - mask_base) < mask_limit)\
399
4.79M
            color_set_null(pdevc);\
400
4.79M
        else {\
401
4.79M
            decode_sample(sample_value, cc, 0);\
402
4.79M
            code = (*remap_color)(&cc, pcs, pdevc, pgs, dev, gs_color_select_source);\
403
4.79M
            if (code < 0)\
404
4.79M
                goto err;\
405
4.79M
            pdevc->tag = device_current_tag(dev);\
406
4.79M
        }\
407
55.8M
    } else if (!color_is_pure(pdevc)) {\
408
16.9M
        code = gx_color_load_select(pdevc, pgs, dev, gs_color_select_source);\
409
16.9M
        if (code < 0)\
410
16.9M
            goto err;\
411
16.9M
    }\
412
63.6M
  END
413
9.28M
    gx_dda_fixed_point next;    /* (y not used in fast loop) */
414
9.28M
    gx_dda_step_fixed dxx2, dxx3, dxx4;         /* (not used in all loops) */
415
9.28M
    const byte *psrc_initial = buffer + data_x;
416
9.28M
    const byte *psrc = psrc_initial;
417
9.28M
    const byte *rsrc = psrc;    /* psrc at start of run */
418
9.28M
    const byte *endp = psrc + w;
419
9.28M
    const byte *stop = endp;
420
9.28M
    fixed xrun;                 /* x at start of run */
421
9.28M
    byte run;           /* run value */
422
9.28M
    int htrun = (masked ? 255 : -2);            /* halftone run value */
423
9.28M
    int i, code = 0;
424
425
9.28M
    if (h == 0)
426
1.09M
        return 0;
427
    /*
428
     * Make sure the cache setup matches the graphics state.  Also determine
429
     * whether all tiles fit in the cache.  We may bypass the latter check
430
     * for masked images with a pure color.
431
     */
432
433
8.19M
    next = penum->dda.pixel0;
434
8.19M
    xrun = dda_current(next.x);
435
8.19M
    if (!masked) {
436
2.66M
        pcs = penum->pcs;       /* (may not be set for masks) */
437
2.66M
        remap_color = pcs->type->remap_color;
438
2.66M
    }
439
8.19M
    run = *psrc;
440
    /* Find the last transition in the input. */
441
8.19M
    if (masked &&
442
5.52M
        (penum->posture != image_portrait) &&
443
3.47M
        (penum->posture != image_landscape)) {
444
        /* No need to calculate stop */
445
4.72M
    } else {
446
4.72M
        byte last = stop[-1];
447
448
608M
        while (stop > psrc && stop[-1] == last)
449
603M
            --stop;
450
4.72M
    }
451
8.19M
    if (penum->slow_loop || penum->posture != image_portrait) {
452
453
        /**************************************************************
454
         * Slow case (skewed, rotated, or imagemask with a halftone). *
455
         **************************************************************/
456
457
3.49M
        fixed yrun;
458
3.49M
        const fixed pdyx = dda_current(penum->dda.row.x) - penum->cur.x;
459
3.49M
        const fixed pdyy = dda_current(penum->dda.row.y) - penum->cur.y;
460
3.49M
        dev_proc_fill_parallelogram((*fill_pgram)) =
461
3.49M
            dev_proc(dev, fill_parallelogram);
462
463
138M
#define xl dda_current(next.x)
464
141M
#define ytf dda_current(next.y)
465
3.49M
        yrun = ytf;
466
3.49M
        if (masked) {
467
468
            /**********************
469
             * Slow case, masked. *
470
             **********************/
471
472
3.47M
            pdevc = penum->icolor1;
473
3.47M
            code = gx_color_load(pdevc, pgs, dev);
474
3.47M
            if (code < 0)
475
0
                return code;
476
3.47M
            if ((stop <= psrc) && (penum->adjust == 0) &&
477
0
                ((penum->posture == image_portrait) ||
478
0
                 (penum->posture == image_landscape)))
479
0
                goto last;
480
3.47M
            if (penum->posture == image_portrait) {
481
482
                /********************************
483
                 * Slow case, masked, portrait. *
484
                 ********************************/
485
486
                /*
487
                 * We don't have to worry about the Y DDA, and the fill
488
                 * regions are rectangles.  Calculate multiples of the DDA
489
                 * step.
490
                 */
491
0
                fixed ax =
492
0
                    (penum->matrix.xx < 0 ? -penum->adjust : penum->adjust);
493
0
                fixed ay =
494
0
                    (pdyy < 0 ? -penum->adjust : penum->adjust);
495
0
                fixed dyy = pdyy + (ay << 1);
496
497
0
                yrun -= ay;
498
0
                dda_translate(next.x, -ax);
499
0
                ax <<= 1;
500
0
                dxx2 = next.x.step;
501
0
                dda_step_add(dxx2, next.x.step);
502
0
                dxx3 = dxx2;
503
0
                dda_step_add(dxx3, next.x.step);
504
0
                dxx4 = dxx3;
505
0
                dda_step_add(dxx4, next.x.step);
506
0
                for (;;) {      /* Skip a run of zeros. */
507
0
                    while (!psrc[0])
508
0
                        if (psrc + 4 <= endp) {
509
                            /* We can use fast skipping */
510
0
                            if (!psrc[1]) {
511
0
                                if (!psrc[2]) {
512
0
                                    if (!psrc[3]) {
513
0
                                        psrc += 4;
514
0
                                        dda_state_next(next.x.state, dxx4);
515
0
                                        if (psrc >= endp)
516
0
                                            break;
517
0
                                        continue;
518
0
                                    }
519
0
                                    psrc += 3;
520
0
                                    dda_state_next(next.x.state, dxx3);
521
0
                                    break;
522
0
                                }
523
0
                                psrc += 2;
524
0
                                dda_state_next(next.x.state, dxx2);
525
0
                                break;
526
0
                            } else {
527
0
                                ++psrc;
528
0
                                dda_next(next.x);
529
0
                                break;
530
0
                            }
531
0
                        } else {
532
                            /* We're too close to the end - skip 1 at a time */
533
0
                            while (!psrc[0]) {
534
0
                                ++psrc;
535
0
                                dda_next(next.x);
536
0
                                if (psrc >= endp)
537
0
                                    break;
538
0
                            }
539
0
                            if (psrc >= endp)
540
0
                                break;
541
0
                        }
542
0
                    xrun = xl;
543
0
                    if (psrc >= stop)
544
0
                        break;
545
0
                    for (; psrc < endp && *psrc; ++psrc)
546
0
                        dda_next(next.x);
547
0
                    code = (*fill_pgram)(dev, xrun, yrun,
548
0
                                         xl - xrun + ax, fixed_0, fixed_0, dyy,
549
0
                                         pdevc, lop);
550
0
                    if (code < 0)
551
0
                        goto err;
552
0
                    rsrc = psrc;
553
0
                    if (psrc >= stop)
554
0
                        break;
555
0
                }
556
557
3.47M
            } else if (penum->posture == image_landscape) {
558
559
                /*********************************
560
                 * Slow case, masked, landscape. *
561
                 *********************************/
562
563
                /*
564
                 * We don't have to worry about the X DDA.  However, we do
565
                 * have to take adjustment into account.  We don't bother to
566
                 * optimize this as heavily as the portrait case.
567
                 */
568
2
                fixed ax =
569
2
                    (pdyx < 0 ? -penum->adjust : penum->adjust);
570
2
                fixed dyx = pdyx + (ax << 1);
571
2
                fixed ay =
572
2
                    (penum->matrix.xy < 0 ? -penum->adjust : penum->adjust);
573
574
2
                xrun -= ax;
575
2
                dda_translate(next.y, -ay);
576
2
                ay <<= 1;
577
3
                for (;;) {
578
41
                    for (; psrc < endp && !*psrc; ++psrc)
579
38
                        dda_next(next.y);
580
3
                    yrun = ytf;
581
3
                    if (psrc >= stop)
582
1
                        break;
583
14
                    for (; psrc < endp && *psrc; ++psrc)
584
12
                        dda_next(next.y);
585
2
                    code = (*fill_pgram)(dev, xrun, yrun, fixed_0,
586
2
                                         ytf - yrun + ay, dyx, fixed_0,
587
2
                                         pdevc, lop);
588
2
                    if (code < 0)
589
0
                        goto err;
590
2
                    rsrc = psrc;
591
2
                    if (psrc >= stop)
592
1
                        break;
593
2
                }
594
595
3.47M
            } else {
596
597
                /**************************************
598
                 * Slow case, masked, not orthogonal. *
599
                 **************************************/
600
                /* FIXME: RJW: This code doesn't do adjustment. Should it?
601
                 * In the grand scheme of things it almost certainly doesn't
602
                 * matter (as adjust should only be used when plotting chars,
603
                 * and we use freetype now), but we record the fact that this
604
                 * may be a defect here. */
605
                /* Previous code here used to skip blocks of matching pixels
606
                 * and plot them all in one go. We can't do that, as it can
607
                 * cause rounding errors and mismatches with the image pixels
608
                 * that will be plotted after us. */
609
3.47M
                stop = endp;
610
14.4M
                for (;;) {
611
                    /* skip forward until we find a 1 bit */
612
101M
                    for (; psrc < stop && !*psrc; ++psrc) {
613
87.1M
                        dda_next(next.x);
614
87.1M
                        dda_next(next.y);
615
87.1M
                    }
616
14.4M
                    if (psrc >= endp) /* Note, endp NOT stop! */
617
3.47M
                        break;
618
                    /* Then draw the pgram and step forward until we find a
619
                     * 0 bit. */
620
47.5M
                    do {
621
47.5M
                        yrun = ytf;
622
47.5M
                        xrun = xl;
623
47.5M
                        dda_next(next.x);
624
47.5M
                        dda_next(next.y);
625
47.5M
                        code = (*fill_pgram)(dev, xrun, yrun, xl - xrun,
626
47.5M
                                             ytf - yrun, pdyx, pdyy, pdevc, lop);
627
47.5M
                        if (code < 0)
628
0
                            goto err;
629
47.5M
                        psrc++;
630
47.5M
                        rsrc = psrc;
631
47.5M
                        if (psrc >= stop)
632
1.16M
                            break;
633
47.5M
                    } while (*psrc);
634
10.9M
                }
635
636
3.47M
            }
637
638
3.47M
        } else if (penum->posture == image_portrait) {
639
            /**************************************
640
             * Slow case, not masked, portrait. *
641
             **************************************/
642
1.99k
            dev_proc_fill_rectangle((*fill_proc)) =
643
1.99k
                dev_proc(dev, fill_rectangle);
644
1.99k
            int iy = fixed2int_pixround(yrun);
645
1.99k
            int ih = fixed2int_pixround(yrun + pdyy) - iy;
646
1.99k
            if (ih < 0)
647
13
                iy += ih, ih = -ih;
648
649
            /* In this case, we can fill runs quickly. */
650
            /****** DOESN'T DO ADJUSTMENT ******/
651
1.99k
            if (stop <= psrc)
652
561
                goto last;
653
358k
            for (;;) {
654
358k
                byte c = *psrc++;
655
358k
                if (c != run) {
656
39.6k
                    int ix = fixed2int_pixround(xrun);
657
39.6k
                    int iw = fixed2int_pixround(xl) - ix;
658
39.6k
                    if (iw < 0)
659
116
                        ix += iw, iw = -iw;
660
39.6k
                    switch (run)
661
39.6k
                    {
662
19.7k
                    case 0:
663
19.7k
                        if (!color_is_pure(penum->icolor0))
664
19.7k
                            goto ht_port;
665
0
                        code = (*fill_proc) (dev, ix, iy, iw, ih,
666
0
                                             penum->icolor0->colors.pure);
667
0
                        break;
668
4.77k
                    case 0xff:
669
4.77k
                        if (!color_is_pure(penum->icolor1))
670
0
                            goto ht_port;
671
4.77k
                        code = (*fill_proc) (dev, ix, iy, iw, ih,
672
4.77k
                                             penum->icolor1->colors.pure);
673
4.77k
                        break;
674
15.0k
                    default:
675
34.8k
                        ht_port:
676
34.8k
                        if (run != htrun) {
677
30.3k
                            htrun = run;
678
#if USE_SET_GRAY_FUNCTION
679
                            code = image_set_gray(run,masked,mask_base,mask_limit,&pdevc,
680
                                    &cc,pcs,pgs,dev,gs_color_select_source,penum);
681
                            if (code < 0)
682
                                goto err;
683
#else
684
30.3k
                            IMAGE_SET_GRAY(run);
685
30.3k
#endif
686
30.3k
                        }
687
34.8k
                        code = gx_fill_rectangle_device_rop(ix, iy, iw, ih,
688
39.6k
                                                            pdevc, dev, lop);
689
39.6k
                    }
690
39.6k
                    if (code < 0)
691
0
                        goto err;
692
39.6k
                    xrun = xl;
693
39.6k
                    rsrc = psrc;
694
39.6k
                    if (psrc > stop)
695
1.43k
                        break;
696
38.2k
                    run = c;
697
38.2k
                }
698
356k
                dda_next(next.x);
699
356k
                if (psrc >= endp)
700
0
                    break;
701
356k
            }
702
19.9k
        } else if (penum->posture == image_landscape) {
703
704
            /**************************************
705
             * Slow case, not masked, landscape. *
706
             **************************************/
707
1.35k
            dev_proc_fill_rectangle((*fill_proc)) =
708
1.35k
                dev_proc(dev, fill_rectangle);
709
1.35k
            int ix = fixed2int_pixround(xrun);
710
1.35k
            int iw = fixed2int_pixround(xrun + pdyx) - ix;
711
1.35k
            if (iw < 0)
712
394
                ix += iw, iw = -iw;
713
714
            /* In this case, we can fill runs quickly. */
715
            /****** DOESN'T DO ADJUSTMENT ******/
716
1.35k
            if (stop <= psrc)
717
117
                goto last;
718
225k
            for (;;) {
719
225k
                byte c = *psrc++;
720
225k
                if (c != run) {
721
54.2k
                    int iy = fixed2int_pixround(yrun);
722
54.2k
                    int ih = fixed2int_pixround(ytf) - iy;
723
54.2k
                    if (ih < 0)
724
19.4k
                        iy += ih, ih = -ih;
725
54.2k
                    switch (run)
726
54.2k
                    {
727
10.8k
                    case 0:
728
10.8k
                        if (!color_is_pure(penum->icolor0))
729
10.8k
                            goto ht_land;
730
42
                        code = (*fill_proc) (dev, ix, iy, iw, ih,
731
42
                                             penum->icolor0->colors.pure);
732
42
                        break;
733
9.29k
                    case 0xff:
734
9.29k
                        if (!color_is_pure(penum->icolor1))
735
6
                            goto ht_land;
736
9.29k
                        code = (*fill_proc) (dev, ix, iy, iw, ih,
737
9.29k
                                             penum->icolor1->colors.pure);
738
9.29k
                        break;
739
34.1k
                    default:
740
44.9k
                        ht_land:
741
44.9k
                        if (run != htrun) {
742
36.2k
                            htrun = run;
743
#if USE_SET_GRAY_FUNCTION
744
                            code = image_set_gray(run,masked,mask_base,mask_limit,&pdevc,
745
                                    &cc,pcs,pgs,dev,gs_color_select_source,penum);
746
                            if (code < 0)
747
                                goto err;
748
#else
749
36.2k
                            IMAGE_SET_GRAY(run);
750
36.2k
#endif
751
36.2k
                        }
752
44.9k
                        code = gx_fill_rectangle_device_rop(ix, iy, iw, ih,
753
54.2k
                                                            pdevc, dev, lop);
754
54.2k
                    }
755
54.2k
                    if (code < 0)
756
0
                        goto err;
757
54.2k
                    yrun = ytf;
758
54.2k
                    rsrc = psrc;
759
54.2k
                    if (psrc > stop)
760
1.23k
                        break;
761
53.0k
                    run = c;
762
53.0k
                }
763
223k
                dda_next(next.y);
764
223k
                if (psrc >= endp)
765
0
                    break;
766
223k
            }
767
18.5k
        } else {
768
769
            /******************************************
770
             * Slow case, not masked, not orthogonal. *
771
             ******************************************/
772
773
            /*
774
             * Since we have to check for the end after every pixel
775
             * anyway, we may as well avoid the last-run code.
776
             */
777
18.5k
            stop = endp;
778
21.6M
            for (;;) {
779
                /* We can't skip large constant regions quickly, */
780
                /* because this leads to rounding errors. */
781
                /* Just fill the region between xrun and xl. */
782
21.6M
                if (run != htrun) {
783
253k
                    htrun = run;
784
#if USE_SET_GRAY_FUNCTION
785
                    code = image_set_gray(run,masked,mask_base,mask_limit,&pdevc,
786
                        &cc,pcs,pgs,dev,gs_color_select_source,penum);
787
                    if (code < 0)
788
                        goto err;
789
#else
790
253k
                    IMAGE_SET_GRAY(run);
791
253k
#endif
792
253k
                }
793
21.6M
                code = (*fill_pgram) (dev, xrun, yrun, xl - xrun,
794
21.6M
                                      ytf - yrun, pdyx, pdyy, pdevc, lop);
795
21.6M
                if (code < 0)
796
0
                    goto err;
797
21.6M
                yrun = ytf;
798
21.6M
                xrun = xl;
799
21.6M
                rsrc = psrc;
800
21.6M
                if (psrc >= stop)
801
18.5k
                    break;
802
21.6M
                run = *psrc++;
803
21.6M
                dda_next(next.x);
804
21.6M
                dda_next(next.y);       /* harmless if no skew */
805
21.6M
            }
806
807
18.5k
        }
808
        /* Fill the last run. */
809
3.49M
      last:if (stop < endp && (*stop || !masked)) {
810
3.34k
            if (!masked) {
811
#if USE_SET_GRAY_FUNCTION
812
                    code = image_set_gray(*stop, masked,mask_base,mask_limit,&pdevc,
813
                        &cc,pcs,pgs,dev,gs_color_select_source,penum);
814
                    if (code < 0)
815
                        goto err;
816
#else
817
3.34k
                    IMAGE_SET_GRAY(*stop);
818
3.34k
#endif
819
3.34k
            }
820
3.34k
            dda_advance(next.x, endp - stop);
821
3.34k
            dda_advance(next.y, endp - stop);
822
3.34k
            code = (*fill_pgram) (dev, xrun, yrun, xl - xrun,
823
3.34k
                                  ytf - yrun, pdyx, pdyy, pdevc, lop);
824
3.34k
        }
825
3.49M
#undef xl
826
3.49M
#undef ytf
827
828
4.69M
    } else {
829
830
        /**********************************************************
831
         * Fast case: no skew, and not imagemask with a halftone. *
832
         **********************************************************/
833
834
4.69M
        const fixed adjust = penum->adjust;
835
4.69M
        const fixed dxx = penum->dxx;
836
4.69M
        fixed xa = (dxx >= 0 ? adjust : -adjust);
837
4.69M
        const int yt = penum->yci, iht = penum->hci;
838
839
4.69M
        dev_proc_fill_rectangle((*fill_proc)) =
840
4.69M
            dev_proc(dev, fill_rectangle);
841
4.69M
        int xmin = fixed2int_pixround(penum->clip_outer.p.x);
842
4.69M
        int xmax = fixed2int_pixround(penum->clip_outer.q.x);
843
105M
#define xl dda_current(next.x)
844
845
4.69M
        if_debug2m('b', penum->memory, "[b]image y=%d  dda.y.Q=%lg\n",
846
4.69M
                   penum->y + penum->rect.y, penum->dda.row.y.state.Q / 256.);
847
        /* Fold the adjustment into xrun and xl, */
848
        /* including the +0.5-epsilon for rounding. */
849
4.69M
        xrun = xrun - xa + (fixed_half - fixed_epsilon);
850
4.69M
        dda_translate(next.x, xa + (fixed_half - fixed_epsilon));
851
4.69M
        xa <<= 1;
852
        /* Calculate multiples of the DDA step. */
853
4.69M
        dxx2 = next.x.step;
854
4.69M
        dda_step_add(dxx2, next.x.step);
855
4.69M
        dxx3 = dxx2;
856
4.69M
        dda_step_add(dxx3, next.x.step);
857
4.69M
        dxx4 = dxx3;
858
4.69M
        dda_step_add(dxx4, next.x.step);
859
4.69M
        if (stop > psrc)
860
105M
            for (;;) {          /* Skip large constant regions quickly, */
861
                /* but don't slow down transitions too much. */
862
246M
              skf:if (psrc[0] == run) {
863
187M
                    if (psrc[1] == run) {
864
164M
                        if (psrc[2] == run) {
865
149M
                            if (psrc[3] == run) {
866
140M
                                psrc += 4;
867
140M
                                dda_state_next(next.x.state, dxx4);
868
140M
                                if (psrc >= endp)
869
0
                                    break;
870
140M
                                goto skf;
871
140M
                            } else {
872
9.05M
                                psrc += 4;
873
9.05M
                                dda_state_next(next.x.state, dxx3);
874
9.05M
                            }
875
149M
                        } else {
876
14.9M
                            psrc += 3;
877
14.9M
                            dda_state_next(next.x.state, dxx2);
878
14.9M
                        }
879
164M
                    } else {
880
22.2M
                        psrc += 2;
881
22.2M
                        dda_next(next.x);
882
22.2M
                    }
883
187M
                } else
884
59.0M
                    psrc++;
885
105M
                {               /* Now fill the region between xrun and xl. */
886
105M
                    int xi = fixed2int_var(xrun);
887
105M
                    int wi = fixed2int_var(xl) - xi;
888
105M
                    int xei;
889
890
105M
                    if (wi <= 0) {
891
2.69M
                        if (wi == 0)
892
2.63M
                            goto mt;
893
56.6k
                        xi += wi, wi = -wi;
894
56.6k
                    }
895
102M
                    if ((xei = xi + wi) > xmax || xi < xmin) {  /* Do X clipping */
896
77.5k
                        if (xi < xmin)
897
23.1k
                            wi -= xmin - xi, xi = xmin;
898
77.5k
                        if (xei > xmax)
899
55.0k
                            wi -= xei - xmax;
900
77.5k
                        if (wi <= 0)
901
56.0k
                            goto mt;
902
77.5k
                    }
903
102M
                    switch (run) {
904
22.6M
                        case 0:
905
22.6M
                            if (masked)
906
3.18M
                                goto mt;
907
19.4M
                            if (!color_is_pure(penum->icolor0))
908
378k
                                goto ht;
909
19.0M
                            code = (*fill_proc) (dev, xi, yt, wi, iht,
910
19.0M
                                                 penum->icolor0->colors.pure);
911
19.0M
                            break;
912
21.5M
                        case 255:       /* just for speed */
913
21.5M
                            if (!color_is_pure(penum->icolor1))
914
361k
                                goto ht;
915
21.1M
                            code = (*fill_proc) (dev, xi, yt, wi, iht,
916
21.1M
                                                 penum->icolor1->colors.pure);
917
21.1M
                            break;
918
58.4M
                        default:
919
59.1M
                          ht:   /* Use halftone if needed */
920
59.1M
                            if (run != htrun) {
921
#if USE_SET_GRAY_FUNCTION
922
                                code = image_set_gray(run, masked,mask_base,mask_limit,&pdevc,
923
                                    &cc,pcs,pgs,dev,gs_color_select_source,penum);
924
                                if (code < 0)
925
                                    goto err;
926
#else
927
57.3M
                                IMAGE_SET_GRAY(run);
928
57.3M
#endif
929
57.3M
                                htrun = run;
930
57.3M
                            }
931
59.1M
                            code = gx_fill_rectangle_device_rop(xi, yt, wi, iht,
932
102M
                                                                 pdevc, dev, lop);
933
102M
                    }
934
99.3M
                    if (code < 0)
935
0
                        goto err;
936
105M
                  mt:xrun = xl - xa;    /* original xa << 1 */
937
105M
                    rsrc = psrc - 1;
938
105M
                    if (psrc > stop) {
939
3.38M
                        --psrc;
940
3.38M
                        break;
941
3.38M
                    }
942
101M
                    run = psrc[-1];
943
101M
                }
944
101M
                dda_next(next.x);
945
101M
            }
946
        /* Fill the last run. */
947
4.69M
        if (*stop != 0 || !masked) {
948
3.07M
            int xi = fixed2int_var(xrun);
949
3.07M
            int wi, xei;
950
951
3.07M
            dda_advance(next.x, endp - stop);
952
3.07M
            wi = fixed2int_var(xl) - xi;
953
3.07M
            if (wi <= 0) {
954
84.5k
                if (wi == 0)
955
71.7k
                    goto lmt;
956
12.8k
                xi += wi, wi = -wi;
957
12.8k
            }
958
3.00M
            if ((xei = xi + wi) > xmax || xi < xmin) {  /* Do X clipping */
959
104k
                if (xi < xmin)
960
4.26k
                    wi -= xmin - xi, xi = xmin;
961
104k
                if (xei > xmax)
962
100k
                    wi -= xei - xmax;
963
104k
                if (wi <= 0)
964
12.5k
                    goto lmt;
965
104k
            }
966
#if USE_SET_GRAY_FUNCTION
967
            code = image_set_gray(*stop, masked,mask_base,mask_limit,&pdevc,
968
                &cc,pcs,pgs,dev,gs_color_select_source,penum);
969
            if (code < 0)
970
                goto err;
971
#else
972
3.00M
            IMAGE_SET_GRAY(*stop);
973
2.99M
#endif
974
2.99M
            code = gx_fill_rectangle_device_rop(xi, yt, wi, iht,
975
2.99M
                                                pdevc, dev, lop);
976
3.07M
          lmt:;
977
3.07M
        }
978
979
4.69M
    }
980
8.19M
#undef xl
981
8.19M
    if (code >= 0) {
982
8.19M
        code = 1;
983
8.19M
        goto done;
984
8.19M
    }
985
    /* Save position if error, in case we resume. */
986
0
err:
987
0
    penum->used.x = rsrc - psrc_initial;
988
0
    penum->used.y = 0;
989
8.19M
done:
990
    /* Since dev_color.binary.b_tile is just a pointer to an entry in the halftone tile "cache"
991
       we cannot leave it set in case the interpeter changes the halftone
992
       (whilst it shouldn't do it, it is possible for Postscript image data source
993
       procedure to execute setcreen or similar). This also doesn't really adversely
994
       affect performance, as we'll call gx_color_load_select() for all the samples
995
       from the next buffer, which will NULL the b_tile pointer anyway (it only gets
996
       set when we actually try to draw with it.
997
    */
998
2.10G
    for (i = 0; i < 256; i++) {
999
2.09G
        penum->clues[i].dev_color.colors.binary.b_tile = NULL;
1000
2.09G
    }
1001
8.19M
    return code;
1002
0
}
1003
1004
#ifdef WITH_CAL
1005
static int
1006
image_render_mono_ht_cal_skip_line(gx_image_enum *penum,
1007
                                   gx_device     *dev)
1008
{
1009
    return !cal_halftone_next_line_required(penum->cal_ht);
1010
}
1011
1012
static int
1013
image_render_mono_ht_cal(gx_image_enum * penum, const byte * buffer, int data_x,
1014
    uint w, int h, gx_device * dev)
1015
{
1016
    const byte *input = buffer + data_x;
1017
1018
    if (buffer == NULL)
1019
        return 0;
1020
1021
    return cal_halftone_process_planar(penum->cal_ht, penum->memory->non_gc_memory,
1022
                                       (const byte * const *)&input, halftone_callback, dev);
1023
}
1024
#else
1025
/*
1026
 An image render case where the source color is monochrome or indexed and
1027
 the output is to be halftoned.  If the source color requires decoding,
1028
 an index look-up or ICC color managment, these operations have already been
1029
 performed on the index values and we are ready now to halftone */
1030
static int
1031
image_render_mono_ht(gx_image_enum * penum_orig, const byte * buffer, int data_x,
1032
                  uint w, int h, gx_device * dev)
1033
1.05M
{
1034
1.05M
    gx_image_enum *penum = penum_orig; /* const within proc */
1035
1.05M
    image_posture posture = penum->posture;
1036
1.05M
    int vdi;  /* amounts to replicate */
1037
1.05M
    fixed xrun;
1038
1.05M
    byte *thresh_align;
1039
1.05M
    int spp_out = penum->dev->color_info.num_components;
1040
1.05M
    byte *devc_contone[GX_DEVICE_COLOR_MAX_COMPONENTS];
1041
1.05M
    byte *devc_contone_gray;
1042
1.05M
    const byte *psrc = buffer + data_x;
1043
1.05M
    int dest_width, dest_height, data_length;
1044
1.05M
    byte *color_cache;
1045
1.05M
    int position, k, j;
1046
1.05M
    int offset_bits = penum->ht_offset_bits;
1047
1.05M
    int contone_stride = 0;  /* Not used in landscape case */
1048
1.05M
    fixed offset;
1049
1.05M
    int src_size;
1050
1.05M
    bool flush_buff = false;
1051
1.05M
    int offset_contone[GX_DEVICE_COLOR_MAX_COMPONENTS];    /* to ensure 128 bit boundary */
1052
1.05M
    int offset_threshold;  /* to ensure 128 bit boundary */
1053
1.05M
    gx_dda_fixed dda_ht;
1054
1.05M
    int xn, xr;   /* destination position (pixel, not contone buffer offset) */
1055
1.05M
    int code = 0;
1056
1.05M
    byte *dev_value;
1057
1058
1.05M
    if (h == 0 || penum->line_size == 0) {      /* line_size == 0, nothing to do */
1059
140k
        if (penum->ht_landscape.count == 0 || posture == image_portrait) {
1060
140k
            return 0;
1061
140k
        } else {
1062
            /* Need to flush the buffer */
1063
1
            offset_bits = penum->ht_landscape.count;
1064
1
            penum->ht_offset_bits = offset_bits;
1065
1
            penum->ht_landscape.offset_set = true;
1066
1
            flush_buff = true;
1067
1
        }
1068
140k
    }
1069
916k
    src_size = penum->rect.w;
1070
1071
    /* Set up the dda.  We could move this out but the cost is pretty small */
1072
916k
    dda_ht = (posture == image_portrait) ? penum->dda.pixel0.x : penum->dda.pixel0.y;
1073
916k
    if (penum->dxx > 0)
1074
916k
        dda_translate(dda_ht, -fixed_epsilon);      /* to match rounding in non-fast code */
1075
1076
916k
    switch (posture) {
1077
916k
        case image_portrait:
1078
            /* Figure out our offset in the contone and threshold data
1079
               buffers so that we ensure that we are on the 128bit
1080
               memory boundaries when we get offset_bits into the data. */
1081
            /* Can't do this earlier, as GC might move the buffers. */
1082
916k
            xrun = dda_current(dda_ht);
1083
916k
            dest_width = gxht_dda_length(&dda_ht, src_size);
1084
916k
            if (penum->x_extent.x < 0)
1085
0
                xrun += penum->x_extent.x;
1086
916k
            vdi = penum->hci;
1087
916k
            contone_stride = penum->line_size;
1088
916k
            offset_threshold = (- (((int)(intptr_t)(penum->thresh_buffer)) +
1089
916k
                                   penum->ht_offset_bits)) & 15;
1090
1.89M
            for (k = 0; k < spp_out; k ++) {
1091
981k
                offset_contone[k]   = (- (((int)(intptr_t)(penum->line)) +
1092
981k
                                           contone_stride * k +
1093
981k
                                           penum->ht_offset_bits)) & 15;
1094
981k
            }
1095
916k
            data_length = dest_width;
1096
916k
            dest_height = fixed2int_var_rounded(any_abs(penum->y_extent.y));
1097
#ifdef DEBUG
1098
            /* Help in spotting problems */
1099
            memset(penum->ht_buffer, 0x00, penum->ht_stride * vdi * spp_out);
1100
#endif
1101
916k
            break;
1102
2
        case image_landscape:
1103
2
        default:
1104
            /* Figure out our offset in the contone and threshold data buffers
1105
               so that we ensure that we are on the 128bit memory boundaries.
1106
               Can't do this earlier as GC may move the buffers.
1107
             */
1108
2
            vdi = penum->wci;
1109
2
            contone_stride = penum->line_size;
1110
2
            dest_width = fixed2int_var_rounded(any_abs(penum->y_extent.x));
1111
            /* match height in gxht_thresh.c dev_width calculation */
1112
2
            xrun = dda_current(dda_ht);            /* really yrun, but just used here for landscape */
1113
2
            dest_height = gxht_dda_length(&dda_ht, src_size);
1114
2
            data_length = dest_height;
1115
2
            offset_threshold = (-(int)(intptr_t)(penum->thresh_buffer)) & 15;
1116
4
            for (k = 0; k < spp_out; k ++) {
1117
2
                offset_contone[k] = (- ((int)(intptr_t)(penum->line) +
1118
2
                                        contone_stride * k)) & 15;
1119
2
            }
1120
            /* In the landscaped case, we want to accumulate multiple columns
1121
               of data before sending to the device.  We want to have a full
1122
               byte of HT data in one write.  This may not be possible at the
1123
               left or right and for those and for those we have so send partial
1124
               chunks */
1125
            /* Initialize our xstart and compute our partial bit chunk so
1126
               that we get in sync with the 1 bit mem device 16 bit positions
1127
               for the rest of the chunks */
1128
2
            if (penum->ht_landscape.count == 0) {
1129
                /* In the landscape case, the size depends upon
1130
                   if we are moving left to right or right to left with
1131
                   the image data.  This offset is to ensure that we  get
1132
                   aligned in our chunks along 16 bit boundaries */
1133
1
                penum->ht_landscape.offset_set = true;
1134
1
                if (penum->ht_landscape.index < 0) {
1135
0
                    penum->ht_landscape.xstart = penum->xci + vdi - 1;
1136
0
                    offset_bits = (penum->ht_landscape.xstart % 16) + 1;
1137
                    /* xci can be negative, so allow for that */
1138
0
                    if (offset_bits <= 0) offset_bits += 16;
1139
1
                } else {
1140
1
                    penum->ht_landscape.xstart = penum->xci;
1141
                    /* xci can be negative, see Bug 692569. */
1142
1
                    offset_bits = 16 - penum->xci % 16;
1143
1
                    if (offset_bits >= 16) offset_bits -= 16;
1144
1
                }
1145
1
                if (offset_bits == 0 || offset_bits == 16) {
1146
1
                    penum->ht_landscape.offset_set = false;
1147
1
                    penum->ht_offset_bits = 0;
1148
1
                } else {
1149
0
                    penum->ht_offset_bits = offset_bits;
1150
0
                }
1151
1
            }
1152
2
            break;
1153
916k
    }
1154
916k
    if (flush_buff)
1155
1
        goto flush;  /* All done */
1156
1157
    /* Get the pointers to our buffers */
1158
1.89M
    for (k = 0; k < spp_out; k++) {
1159
981k
        if (posture == image_portrait) {
1160
981k
            devc_contone[k] = penum->line + contone_stride * k +
1161
981k
                              offset_contone[k];
1162
981k
        } else {
1163
1
            devc_contone[k] = penum->line + offset_contone[k] +
1164
1
                              LAND_BITS * k * contone_stride;
1165
1
        }
1166
981k
    }
1167
916k
    xr = fixed2int_var_rounded(dda_current(dda_ht));  /* indexes in the destination (contone) */
1168
1169
916k
    devc_contone_gray = devc_contone[0];
1170
916k
    if (penum->color_cache == NULL) {
1171
        /* No look-up in the cache to fill the source buffer. Still need to
1172
           have the data at device resolution.  Do these in quick small
1173
           loops.  This likely could be a vectorized function.  Note that
1174
           since the color_cache is NULL we must be in a case where we
1175
           are going to a monochrome device. */
1176
757k
        switch (posture) {
1177
757k
            case image_portrait:
1178
757k
                if (penum->dst_width > 0) {
1179
757k
                    if (src_size == dest_width) {
1180
749k
                        memcpy(devc_contone_gray, psrc, data_length);
1181
749k
                    } else if (src_size * 2 == dest_width) {
1182
0
                        const byte *psrc_temp = psrc;
1183
0
                        for (k = 0; k < data_length; k+=2, devc_contone_gray+=2,
1184
0
                             psrc_temp++) {
1185
0
                            *devc_contone_gray = *(devc_contone_gray+1) = *psrc_temp;
1186
0
                        }
1187
7.26k
                    } else {
1188
                        /* Mono case, forward */
1189
1.46M
                        for (k=0; k<src_size; k++) {
1190
1.45M
                            byte c = *psrc++;
1191
1.45M
                            dda_next(dda_ht);
1192
1.45M
                            xn = fixed2int_var_rounded(dda_current(dda_ht));
1193
4.71M
                            while (xr < xn) {
1194
3.25M
                                *devc_contone_gray++ = c;
1195
3.25M
                                xr++;
1196
3.25M
                            }           /* at loop exit xn will be >= xr */
1197
1.45M
                        }
1198
7.26k
                    }
1199
757k
                } else {
1200
                    /* Mono case, backwards */
1201
0
                    devc_contone_gray += (data_length - 1);
1202
0
                    for (k=0; k<src_size; k++) {
1203
0
                        byte c = *psrc++;
1204
0
                        dda_next(dda_ht);
1205
0
                        xn = fixed2int_var_rounded(dda_current(dda_ht));
1206
0
                        while (xr > xn) {
1207
0
                            *devc_contone_gray-- = c;
1208
0
                            xr--;
1209
0
                        }           /* at loop exit xn will be >= xr */
1210
0
                    }
1211
0
                }
1212
757k
                break;
1213
1
            case image_landscape:
1214
                /* We store the data at this point into a column. Depending
1215
                   upon our landscape direction we may be going left to right
1216
                   or right to left. */
1217
1
                if (penum->ht_landscape.flipy) {
1218
0
                    position = penum->ht_landscape.curr_pos +
1219
0
                                LAND_BITS * (data_length - 1);
1220
0
                    for (k=0; k<src_size; k++) {
1221
0
                        byte c = *psrc++;
1222
0
                        dda_next(dda_ht);
1223
0
                        xn = fixed2int_var_rounded(dda_current(dda_ht));
1224
0
                        while (xr > xn) {
1225
0
                            devc_contone_gray[position] = c;
1226
0
                            position -= LAND_BITS;
1227
0
                            xr--;
1228
0
                        }           /* at loop exit xn will be <= xr */
1229
0
                    }
1230
1
                } else {
1231
1
                    position = penum->ht_landscape.curr_pos;
1232
                    /* Code up special cases for when we have no scaling
1233
                       and 2x scaling which we will run into in 300 and
1234
                       600dpi devices and content */
1235
1
                    if (src_size == dest_height) {
1236
0
                        for (k = 0; k < data_length; k++) {
1237
0
                            devc_contone_gray[position] = psrc[k];
1238
0
                            position += LAND_BITS;
1239
0
                        }
1240
1
                    } else if (src_size*2 == dest_height) {
1241
0
                        for (k = 0; k < data_length; k+=2) {
1242
0
                            offset = fixed2int_rounded(fixed_half * k);
1243
0
                            devc_contone_gray[position] =
1244
0
                                devc_contone_gray[position + LAND_BITS] = psrc[offset];
1245
0
                            position += LAND_BITS*2;
1246
0
                        }
1247
1
                    } else {
1248
                        /* use dda */
1249
257
                        for (k=0; k<src_size; k++) {
1250
256
                            byte c = *psrc++;
1251
256
                            dda_next(dda_ht);
1252
256
                            xn = fixed2int_var_rounded(dda_current(dda_ht));
1253
259
                            while (xr < xn) {
1254
3
                                devc_contone_gray[position] = c;
1255
3
                                position += LAND_BITS;
1256
3
                                xr++;
1257
3
                            }           /* at loop exit xn will be >= xr */
1258
256
                        }
1259
1
                    }
1260
1
                }
1261
                /* Store the width information and update our counts */
1262
1
                penum->ht_landscape.count += vdi;
1263
1
                penum->ht_landscape.widths[penum->ht_landscape.curr_pos] = vdi;
1264
1
                penum->ht_landscape.curr_pos += penum->ht_landscape.index;
1265
1
                penum->ht_landscape.num_contones++;
1266
1
                break;
1267
0
            default:
1268
                /* error not allowed */
1269
0
                break;
1270
757k
            }
1271
757k
    } else {
1272
        /* look-up in the cache to fill the source buffer.  If our spp_out
1273
           is 4 then we need to write out the pixels in the color_cache into
1274
           the planes */
1275
159k
        color_cache = penum->color_cache->device_contone;
1276
159k
        switch (posture) {
1277
159k
            case image_portrait:
1278
159k
                if (penum->dst_width > 0) {
1279
                    /* loop filling contone values selected from the source */
1280
159k
                    if (spp_out == 1) {
1281
                        /* Mono case, forward */
1282
29.8M
                        for (k=0; k<src_size; k++) {
1283
29.7M
                            byte c = color_cache[*psrc++];
1284
29.7M
                            dda_next(dda_ht);
1285
29.7M
                            xn = fixed2int_var_rounded(dda_current(dda_ht));
1286
85.7M
                            while (xr < xn) {
1287
56.0M
                                *devc_contone_gray++ = c;
1288
56.0M
                                xr++;
1289
56.0M
                            }           /* at loop exit xn will be >= xr */
1290
29.7M
                        }
1291
137k
                    } else {
1292
                        /* CMYK case, forward */
1293
6.24M
                        for (k=0; k<src_size; k++) {
1294
6.22M
                            dev_value = &(color_cache[*psrc++ * spp_out]);
1295
6.22M
                            dda_next(dda_ht);
1296
6.22M
                            xn = fixed2int_var_rounded(dda_current(dda_ht));
1297
23.7M
                            while (xr < xn) {
1298
87.8M
                                for (j = 0; j < spp_out; j++) {
1299
70.2M
                                    *(devc_contone[j])++ = dev_value[j];
1300
70.2M
                                }
1301
17.5M
                                xr++;
1302
17.5M
                            }           /* at loop exit xn will be >= xr */
1303
6.22M
                        }
1304
21.5k
                    }
1305
159k
                } else {
1306
0
                    if (spp_out == 1) {
1307
                        /* Mono case, backwards */
1308
0
                        devc_contone_gray += data_length - 1;   /* move to end */
1309
0
                        for (k=0; k<src_size; k++) {
1310
0
                            byte c = color_cache[*psrc++];
1311
0
                            dda_next(dda_ht);
1312
0
                            xn = fixed2int_var_rounded(dda_current(dda_ht));
1313
0
                            while (xr > xn) {
1314
0
                                *devc_contone_gray-- = c;
1315
0
                                xr--;
1316
0
                            }           /* at loop exit xn will be <= xr */
1317
0
                        }
1318
0
                    } else {
1319
                        /* CMYK case, backwards */
1320
                        /* Move to the other end and we will decrement */
1321
0
                        for (j = 0; j < spp_out; j++) {
1322
0
                            devc_contone[j] = devc_contone[j] + data_length - 1;
1323
0
                        }
1324
0
                        for (k=0; k<src_size; k++) {
1325
0
                            dev_value = &(color_cache[*psrc++ * spp_out]);
1326
0
                            dda_next(dda_ht);
1327
0
                            xn = fixed2int_var_rounded(dda_current(dda_ht));
1328
0
                            while (xr > xn) {
1329
0
                                for (j = 0; j < spp_out; j++) {
1330
0
                                    *(devc_contone[j])-- = dev_value[j];
1331
0
                                }
1332
0
                                xr--;
1333
0
                            }           /* at loop exit xn will be <= xr */
1334
0
                        }
1335
0
                    }
1336
0
                }
1337
159k
                break;
1338
0
            case image_landscape:
1339
                /* We store the data at this point into a column. Depending
1340
                   upon our landscape direction we may be going left to right
1341
                   or right to left. */
1342
0
                if (penum->ht_landscape.flipy) {
1343
0
                    position = penum->ht_landscape.curr_pos +
1344
0
                                LAND_BITS * (data_length - 1);
1345
                    /* use dda */
1346
0
                    if (spp_out == 1) {
1347
                        /* Mono case */
1348
                        /* loop filling contone values selected from the source */
1349
0
                        for (k=0; k<src_size; k++) {
1350
0
                            byte c = color_cache[*psrc++];
1351
0
                            dda_next(dda_ht);
1352
0
                            xn = fixed2int_var_rounded(dda_current(dda_ht));
1353
0
                            while (xr > xn) {
1354
0
                                devc_contone_gray[position] = c;
1355
0
                                position -= LAND_BITS;
1356
0
                                xr--;
1357
0
                            }           /* at loop exit xn will be <= xr */
1358
0
                        }
1359
0
                    } else {
1360
                        /* CMYK case */
1361
0
                        for (k=0; k<src_size; k++) {
1362
0
                            dev_value = &(color_cache[*psrc++ * spp_out]);
1363
0
                            dda_next(dda_ht);
1364
0
                            xn = fixed2int_var_rounded(dda_current(dda_ht));
1365
0
                            while (xr > xn) {
1366
0
                                for (j = 0; j < spp_out; j++) {
1367
0
                                    *(devc_contone[j] + position) = dev_value[j];
1368
0
                                }
1369
0
                                position -= LAND_BITS;
1370
0
                                xr--;
1371
0
                            }           /* at loop exit xn will be <= xr */
1372
0
                        }
1373
0
                    }
1374
0
                } else {  /* Not flipped in Y */
1375
0
                    position = penum->ht_landscape.curr_pos;
1376
                    /* use dda */
1377
0
                    if (spp_out == 1) {
1378
                        /* Mono case */
1379
                        /* loop filling contone values selected from the source */
1380
0
                        for (k=0; k<src_size; k++) {
1381
0
                            byte c = color_cache[*psrc++];
1382
0
                            dda_next(dda_ht);
1383
0
                            xn = fixed2int_var_rounded(dda_current(dda_ht));
1384
0
                            while (xr < xn) {
1385
0
                                devc_contone_gray[position] = c;
1386
0
                                position += LAND_BITS;
1387
0
                                xr++;
1388
0
                            }           /* at loop exit xn will be >= xr */
1389
0
                        }
1390
0
                    } else {
1391
                        /* CMYK case */
1392
0
                        for (k=0; k<src_size; k++) {
1393
0
                            dev_value = &(color_cache[*psrc++ * spp_out]);
1394
0
                            dda_next(dda_ht);
1395
0
                            xn = fixed2int_var_rounded(dda_current(dda_ht));
1396
0
                            while (xr < xn) {
1397
0
                                for (j = 0; j < spp_out; j++) {
1398
0
                                    *(devc_contone[j] + position) = dev_value[j];
1399
0
                                }
1400
0
                                position += LAND_BITS;
1401
0
                                xr++;
1402
0
                            }           /* at loop exit xn will be >= xr */
1403
0
                        }
1404
0
                    }
1405
0
                }
1406
                /* Store the width information and update our counts */
1407
0
                penum->ht_landscape.count += vdi;
1408
0
                penum->ht_landscape.widths[penum->ht_landscape.curr_pos] = vdi;
1409
0
                penum->ht_landscape.curr_pos += penum->ht_landscape.index;
1410
0
                penum->ht_landscape.num_contones++;
1411
0
                break;
1412
0
            default:
1413
                /* error not allowed */
1414
0
                break;
1415
159k
        }
1416
159k
    }
1417
    /* Apply threshold array to image data */
1418
916k
flush:
1419
916k
    thresh_align = penum->thresh_buffer + offset_threshold;
1420
916k
    code = gxht_thresh_planes(penum, xrun, dest_width, dest_height,
1421
916k
                              thresh_align, dev, offset_contone,
1422
916k
                              contone_stride);
1423
916k
    return code;
1424
916k
}
1425
#endif