Coverage Report

Created: 2026-02-14 07:09

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ghostpdl/devices/vector/gdevpx.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
/* H-P PCL XL driver */
18
#include "math_.h"
19
#include "memory_.h"
20
#include "gx.h"
21
#include "gserrors.h"
22
#include "gsccolor.h"
23
#include "gsdcolor.h"
24
#include "gxiparam.h"
25
#include "gxcspace.h"           /* for color mapping for images */
26
#include "gxdevice.h"
27
#include "gxpath.h"
28
#include "gdevvec.h"
29
#include "gdevmrop.h"
30
#include "strimpl.h"
31
#include "srlx.h"
32
#include "jpeglib_.h"
33
#include "sdct.h"
34
#include "sjpeg.h"
35
#include "gdevpxat.h"
36
#include "gdevpxen.h"
37
#include "gdevpxop.h"
38
#include "gdevpxut.h"
39
#include "gxlum.h"
40
#include "gdevpcl.h"            /* for gdev_pcl_mode3compress() */
41
#include "gsicc_manage.h"
42
#include "gsicc_cache.h"
43
#include <stdlib.h>             /* abs() */
44
45
#include "gxfcache.h"
46
#include "gxfont.h"
47
48
/* ---------------- Device definition ---------------- */
49
50
/* Define the default resolution. */
51
#ifndef X_DPI
52
#  define X_DPI 600
53
#endif
54
#ifndef Y_DPI
55
#  define Y_DPI 600
56
#endif
57
58
/* Structure definition */
59
57.2M
#define NUM_POINTS 40           /* must be >= 3 and <= 255 */
60
typedef enum {
61
    POINTS_NONE,
62
    POINTS_LINES,
63
    POINTS_CURVES
64
} point_type_t;
65
typedef struct gx_device_pclxl_s {
66
    gx_device_vector_common;
67
    /* Additional state information */
68
    pxeMediaSize_t media_size;
69
    bool ManualFeed;            /* map ps setpage commands to pxl */
70
    bool ManualFeed_set;
71
    int MediaPosition_old;      /* old Position attribute - for duplex detection */
72
    int MediaPosition;          /* MediaPosition attribute */
73
    int MediaPosition_set;
74
    char MediaType_old[64];     /* old MediaType attribute - for duplex detection */
75
    char MediaType[64];         /* MediaType attribute */
76
    int MediaType_set;
77
    int page;                   /* Page number starting at 0 */
78
    bool Duplex;                /* Duplex attribute */
79
    bool Staple;                /* Staple attribute */
80
    bool Tumble;                /* Tumble attribute */
81
    gx_path_type_t fill_rule;   /* ...winding_number or ...even_odd  */
82
    gx_path_type_t clip_rule;   /* ditto */
83
    pxeColorSpace_t color_space;
84
    struct pal_ {
85
        int size;               /* # of bytes */
86
        byte data[256 * 3];     /* up to 8-bit samples */
87
    } palette;
88
    struct pts_ {
89
        /* buffer for accumulating path points */
90
        gs_int_point current;   /* current point as of start of data */
91
        point_type_t type;
92
        int count;
93
        gs_int_point data[NUM_POINTS];
94
    } points;
95
    struct ch_ {
96
    /* cache for downloaded characters */
97
1.06k
#define MAX_CACHED_CHARS 400
98
1.04k
#define MAX_CHAR_DATA 500000
99
17.1k
#define MAX_CHAR_SIZE 5000
100
17.6k
#define CHAR_HASH_FACTOR 247
101
        ushort table[MAX_CACHED_CHARS * 3 / 2];
102
        struct cd_ {
103
            gs_id id;           /* key */
104
            uint size;
105
        } data[MAX_CACHED_CHARS];
106
        int next_in;            /* next data element to fill in */
107
        int next_out;           /* next data element to discard */
108
        int count;              /* of occupied data elements */
109
        ulong used;
110
    } chars;
111
    bool font_set;
112
    int state_rotated;          /* 0, 1, 2, -1, mutiple of 90 deg */
113
    int CompressMode;           /* std PXL enum: None=0, RLE=1, JPEG=2, DeltaRow=3 */
114
    bool scaled;
115
    double x_scale;             /* chosen so that max(x) is scaled to 0x7FFF, to give max distinction between x values */
116
    double y_scale;
117
    bool pen_null;
118
    bool brush_null;
119
    bool iccTransform;
120
} gx_device_pclxl;
121
122
gs_public_st_suffix_add0_final(st_device_pclxl, gx_device_pclxl,
123
                               "gx_device_pclxl",
124
                               device_pclxl_enum_ptrs,
125
                               device_pclxl_reloc_ptrs, gx_device_finalize,
126
                               st_device_vector);
127
128
#define pclxl_device_body(dname, depth, init)\
129
  std_device_dci_type_body(gx_device_pclxl, init, dname, &st_device_pclxl,\
130
                           DEFAULT_WIDTH_10THS * X_DPI / 10,\
131
                           DEFAULT_HEIGHT_10THS * Y_DPI / 10,\
132
                           X_DPI, Y_DPI,\
133
                           (depth > 8 ? 3 : 1), depth,\
134
                           (depth > 1 ? 255 : 1), (depth > 8 ? 255 : 0),\
135
                           (depth > 1 ? 256 : 2), (depth > 8 ? 256 : 1))
136
137
/* Driver procedures */
138
static dev_proc_open_device(pclxl_open_device);
139
static dev_proc_output_page(pclxl_output_page);
140
static dev_proc_close_device(pclxl_close_device);
141
static dev_proc_copy_mono(pclxl_copy_mono);
142
static dev_proc_copy_color(pclxl_copy_color);
143
static dev_proc_fill_mask(pclxl_fill_mask);
144
145
static dev_proc_get_params(pclxl_get_params);
146
static dev_proc_put_params(pclxl_put_params);
147
static dev_proc_text_begin(pclxl_text_begin);
148
149
/*static dev_proc_draw_thin_line(pclxl_draw_thin_line); */
150
static dev_proc_begin_typed_image(pclxl_begin_typed_image);
151
static dev_proc_strip_copy_rop2(pclxl_strip_copy_rop2);
152
153
static void
154
pclxl_initialize_device_procs(gx_device *dev,
155
                              dev_proc_map_rgb_color(map_rgb_color),
156
                              dev_proc_map_color_rgb(map_color_rgb))
157
25.2k
{
158
25.2k
    set_dev_proc(dev, open_device, pclxl_open_device);
159
25.2k
    set_dev_proc(dev, output_page, pclxl_output_page);
160
25.2k
    set_dev_proc(dev, close_device, pclxl_close_device);
161
25.2k
    set_dev_proc(dev, map_rgb_color, map_rgb_color);
162
25.2k
    set_dev_proc(dev, map_color_rgb, map_color_rgb);
163
25.2k
    set_dev_proc(dev, fill_rectangle, gdev_vector_fill_rectangle);
164
25.2k
    set_dev_proc(dev, copy_mono, pclxl_copy_mono);
165
25.2k
    set_dev_proc(dev, copy_color, pclxl_copy_color);
166
25.2k
    set_dev_proc(dev, get_params, pclxl_get_params);
167
25.2k
    set_dev_proc(dev, put_params, pclxl_put_params);
168
25.2k
    set_dev_proc(dev, get_page_device, gx_page_device_get_page_device);
169
25.2k
    set_dev_proc(dev, fill_path, gdev_vector_fill_path);
170
25.2k
    set_dev_proc(dev, stroke_path, gdev_vector_stroke_path);
171
25.2k
    set_dev_proc(dev, fill_mask, pclxl_fill_mask);
172
25.2k
    set_dev_proc(dev, fill_trapezoid, gdev_vector_fill_trapezoid);
173
25.2k
    set_dev_proc(dev, fill_parallelogram, gdev_vector_fill_parallelogram);
174
25.2k
    set_dev_proc(dev, fill_triangle, gdev_vector_fill_triangle);
175
25.2k
    set_dev_proc(dev, begin_typed_image, pclxl_begin_typed_image);
176
25.2k
    set_dev_proc(dev, strip_copy_rop2, pclxl_strip_copy_rop2);
177
25.2k
    set_dev_proc(dev, text_begin, pclxl_text_begin);
178
25.2k
}
179
180
static void
181
pxlmono_initialize_device_procs(gx_device *dev)
182
12.3k
{
183
12.3k
    pclxl_initialize_device_procs(dev,
184
12.3k
                                  gx_default_gray_map_rgb_color,
185
12.3k
                                  gx_default_gray_map_color_rgb);
186
12.3k
}
187
188
static void
189
pxlcolor_initialize_device_procs(gx_device *dev)
190
12.9k
{
191
12.9k
    pclxl_initialize_device_procs(dev,
192
12.9k
                                  gx_default_rgb_map_rgb_color,
193
12.9k
                                  gx_default_rgb_map_color_rgb);
194
12.9k
}
195
196
const gx_device_pclxl gs_pxlmono_device = {
197
    pclxl_device_body("pxlmono", 8, pxlmono_initialize_device_procs)
198
};
199
200
const gx_device_pclxl gs_pxlcolor_device = {
201
    pclxl_device_body("pxlcolor", 24, pxlcolor_initialize_device_procs)
202
};
203
204
/* ---------------- Other utilities ---------------- */
205
206
static inline stream *
207
pclxl_stream(gx_device_pclxl * xdev)
208
178M
{
209
178M
    return gdev_vector_stream((gx_device_vector *) xdev);
210
178M
}
211
212
/* Initialize for a page. */
213
static void
214
pclxl_page_init(gx_device_pclxl * xdev)
215
45.3k
{
216
45.3k
    gdev_vector_init((gx_device_vector *) xdev);
217
45.3k
    xdev->in_page = false;
218
45.3k
    xdev->fill_rule = gx_path_type_winding_number;
219
45.3k
    xdev->clip_rule = gx_path_type_winding_number;
220
45.3k
    xdev->color_space = eNoColorSpace;
221
45.3k
    xdev->palette.size = 0;
222
45.3k
    xdev->font_set = false;
223
45.3k
    xdev->state_rotated = 0;
224
45.3k
    xdev->scaled = false;
225
45.3k
    xdev->x_scale = 1;
226
45.3k
    xdev->y_scale = 1;
227
45.3k
    xdev->pen_null = false;
228
45.3k
    xdev->brush_null = false;
229
45.3k
}
230
231
/* Test whether a RGB color is actually a gray shade. */
232
14.9M
#define RGB_IS_GRAY(ci) ((ci) >> 8 == ((ci) & 0xffff))
233
234
/* Set the color space and (optionally) palette. */
235
static void
236
pclxl_set_color_space(gx_device_pclxl * xdev, pxeColorSpace_t color_space)
237
22.7M
{
238
22.7M
    if (xdev->color_space != color_space) {
239
781k
        stream *s = pclxl_stream(xdev);
240
241
781k
        px_put_ub(s, (byte) color_space);
242
781k
        px_put_ac(s, pxaColorSpace, pxtSetColorSpace);
243
781k
        xdev->color_space = color_space;
244
781k
        xdev->palette.size = 0; /* purge the cached palette */
245
781k
    }
246
22.7M
}
247
static void
248
pclxl_set_color_palette(gx_device_pclxl * xdev, pxeColorSpace_t color_space,
249
                        const byte * palette, uint palette_size)
250
167k
{
251
167k
    if (xdev->color_space != color_space ||
252
166k
        xdev->palette.size != palette_size ||
253
165k
        memcmp(xdev->palette.data, palette, palette_size)
254
167k
        ) {
255
2.32k
        stream *s = pclxl_stream(xdev);
256
257
2.32k
        static const byte csp_[] = {
258
2.32k
            DA(pxaColorSpace),
259
2.32k
            DUB(e8Bit), DA(pxaPaletteDepth),
260
2.32k
            pxt_ubyte_array
261
2.32k
        };
262
263
2.32k
        px_put_ub(s, (byte) color_space);
264
2.32k
        PX_PUT_LIT(s, csp_);
265
2.32k
        px_put_u(s, palette_size);
266
2.32k
        px_put_bytes(s, palette, palette_size);
267
2.32k
        px_put_ac(s, pxaPaletteData, pxtSetColorSpace);
268
2.32k
        xdev->color_space = color_space;
269
2.32k
        xdev->palette.size = palette_size;
270
2.32k
        memcpy(xdev->palette.data, palette, palette_size);
271
2.32k
    }
272
167k
}
273
274
/* For caching either NullPen or NullBrush, which happens a lot for
275
 * drawing masks in the PS3 CET test set.
276
 *
277
 * The expected null_source/op combos are:
278
 * pxaNullPen/pxtSetPenSource and pxaNullBrush/pxtSetBrushSource
279
 */
280
static int
281
pclxl_set_cached_nulls(gx_device_pclxl * xdev, px_attribute_t null_source,
282
                       px_tag_t op)
283
44.0M
{
284
44.0M
    stream *s = pclxl_stream(xdev);
285
286
44.0M
    if (op == pxtSetPenSource) {
287
43.5M
        if (xdev->pen_null)
288
43.4M
            return 0;
289
135k
        else
290
135k
            xdev->pen_null = true;
291
43.5M
    }
292
637k
    if (op == pxtSetBrushSource) {
293
502k
        if (xdev->brush_null)
294
385k
            return 0;
295
117k
        else
296
117k
            xdev->brush_null = true;
297
502k
    }
298
252k
    px_put_uba(s, 0, (byte) null_source);
299
252k
    spputc(s, (byte) op);
300
252k
    return 0;
301
637k
}
302
303
/* Set a drawing RGB color. */
304
static int
305
pclxl_set_color(gx_device_pclxl * xdev, const gx_drawing_color * pdc,
306
                px_attribute_t null_source, px_tag_t op)
307
21.9M
{
308
21.9M
    stream *s = pclxl_stream(xdev);
309
310
21.9M
    if (gx_dc_is_pure(pdc)) {
311
21.9M
        gx_color_index color = gx_dc_pure_color(pdc);
312
313
21.9M
        if (op == pxtSetPenSource)
314
117k
            xdev->pen_null = false;
315
21.9M
        if (op == pxtSetBrushSource)
316
21.8M
            xdev->brush_null = false;
317
318
21.9M
        if (xdev->color_info.num_components == 1 || RGB_IS_GRAY(color)) {
319
13.1M
            pclxl_set_color_space(xdev, eGray);
320
13.1M
            px_put_uba(s, (byte) color, pxaGrayLevel);
321
13.1M
        } else {
322
8.75M
            pclxl_set_color_space(xdev, eRGB);
323
8.75M
            spputc(s, pxt_ubyte_array);
324
8.75M
            px_put_ub(s, 3);
325
8.75M
            spputc(s, (byte) (color >> 16));
326
8.75M
            spputc(s, (byte) (color >> 8));
327
8.75M
            spputc(s, (byte) color);
328
8.75M
            px_put_a(s, pxaRGBColor);
329
8.75M
        }
330
21.9M
    } else if (gx_dc_is_null(pdc) || !color_is_set(pdc)) {
331
2.93k
        if (op == pxtSetPenSource || op == pxtSetBrushSource)
332
2.93k
            return pclxl_set_cached_nulls(xdev, null_source, op);
333
0
        else
334
0
            px_put_uba(s, 0, null_source);
335
2.93k
    } else
336
20.5k
        return_error(gs_error_rangecheck);
337
21.9M
    spputc(s, (byte) op);
338
21.9M
    return 0;
339
21.9M
}
340
341
/* Test whether we can handle a given color space in an image. */
342
/* We cannot handle ICCBased color spaces. */
343
static bool
344
pclxl_can_handle_color_space(const gs_color_space * pcs)
345
10.5k
{
346
10.5k
    gs_color_space_index index;
347
348
    /* an image with no colorspace info arrived; cannot handle */
349
10.5k
    if (!pcs)
350
0
        return false;
351
10.5k
    index = gs_color_space_get_index(pcs);
352
353
10.5k
    if (index == gs_color_space_index_Indexed) {
354
319
        if (pcs->params.indexed.use_proc)
355
0
            return false;
356
319
        index =
357
319
            gs_color_space_get_index(gs_color_space_indexed_base_space(pcs));
358
10.1k
    } else if (index == gs_color_space_index_ICC) {
359
10.1k
        index = gsicc_get_default_type(pcs->cmm_icc_profile_data);
360
10.1k
        return ((index < gs_color_space_index_DevicePixel) ? true : false);
361
10.1k
    }
362
363
348
    return !(index == gs_color_space_index_Separation ||
364
335
             index == gs_color_space_index_Pattern ||
365
335
             index == gs_color_space_index_DeviceN ||
366
313
             index == gs_color_space_index_ICC ||
367
0
             (index <= gs_color_space_index_CIEA &&
368
0
             index >= gs_color_space_index_CIEDEFG)
369
348
             );
370
10.5k
}
371
372
/* Test whether we can icclink-transform an image. */
373
static bool
374
pclxl_can_icctransform(const gs_image_t * pim)
375
1.74k
{
376
1.74k
    const gs_color_space *pcs = pim->ColorSpace;
377
1.74k
    int bits_per_pixel;
378
379
    /* an image with no colorspace info arrived; cannot transform */
380
1.74k
    if (!pcs)
381
0
        return false;
382
1.74k
    bits_per_pixel =
383
1.74k
        (pim->ImageMask ? 1 :
384
1.74k
         pim->BitsPerComponent * gs_color_space_num_components(pcs));
385
386
1.74k
    if ((gs_color_space_get_index(pcs) == gs_color_space_index_ICC)
387
1.39k
        && (bits_per_pixel == 24 || bits_per_pixel == 32))
388
639
        return true;
389
390
1.10k
    return false;
391
1.74k
}
392
393
/*
394
 * Avoid PXL high level images if a transfer function has been set.
395
 * Allow the graphics library to render to a lower level
396
 * representation with the function applied to the colors.
397
 */
398
399
static bool
400
pclxl_nontrivial_transfer(const gs_gstate * pgs)
401
234k
{
402
234k
    gx_transfer_map *red = pgs->set_transfer.red;
403
234k
    gx_transfer_map *green = pgs->set_transfer.green;
404
234k
    gx_transfer_map *blue = pgs->set_transfer.blue;
405
406
234k
    return (red || green || blue);
407
408
234k
}
409
/* Set brush, pen, and mode for painting a path. */
410
static void
411
pclxl_set_paints(gx_device_pclxl * xdev, gx_path_type_t type)
412
44.0M
{
413
44.0M
    stream *s = pclxl_stream(xdev);
414
44.0M
    gx_path_type_t rule = type & gx_path_type_rule;
415
416
44.0M
    if (!(type & gx_path_type_fill) &&
417
499k
        (color_is_set(&xdev->saved_fill_color.saved_dev_color) ||
418
0
         !gx_dc_is_null(&xdev->saved_fill_color.saved_dev_color)
419
499k
        )
420
44.0M
        ) {
421
499k
        pclxl_set_cached_nulls(xdev, pxaNullBrush, pxtSetBrushSource);
422
499k
        color_set_null(&xdev->saved_fill_color.saved_dev_color);
423
499k
        if (rule != xdev->fill_rule) {
424
44
            px_put_ub(s, (byte) (rule == gx_path_type_even_odd ? eEvenOdd :
425
44
                                 eNonZeroWinding));
426
44
            px_put_ac(s, pxaFillMode, pxtSetFillMode);
427
44
            xdev->fill_rule = rule;
428
44
        }
429
499k
    }
430
44.0M
    if (!(type & gx_path_type_stroke) &&
431
43.5M
        (color_is_set(&xdev->saved_stroke_color.saved_dev_color) ||
432
0
         !gx_dc_is_null(&xdev->saved_stroke_color.saved_dev_color)
433
43.5M
        )
434
44.0M
        ) {
435
43.5M
        pclxl_set_cached_nulls(xdev, pxaNullPen, pxtSetPenSource);
436
43.5M
        color_set_null(&xdev->saved_stroke_color.saved_dev_color);
437
43.5M
    }
438
44.0M
}
439
440
static void
441
pclxl_set_page_origin(stream * s, int x, int y)
442
81.9k
{
443
81.9k
    px_put_ssp(s, x, y);
444
81.9k
    px_put_ac(s, pxaPageOrigin, pxtSetPageOrigin);
445
81.9k
    return;
446
81.9k
}
447
448
static void
449
pclxl_set_page_scale(gx_device_pclxl * xdev, double x_scale, double y_scale)
450
10.0M
{
451
10.0M
    stream *s = pclxl_stream(xdev);
452
453
10.0M
    if (xdev->scaled) {
454
2.84M
        xdev->x_scale = x_scale;
455
2.84M
        xdev->y_scale = y_scale;
456
2.84M
        px_put_rp(s, x_scale, y_scale);
457
2.84M
        px_put_ac(s, pxaPageScale, pxtSetPageScale);
458
2.84M
    }
459
10.0M
    return;
460
10.0M
}
461
462
static void
463
pclxl_unset_page_scale(gx_device_pclxl * xdev)
464
28.1M
{
465
28.1M
    stream *s = pclxl_stream(xdev);
466
467
28.1M
    if (xdev->scaled) {
468
2.84M
        px_put_rp(s, 1 / xdev->x_scale, 1 / xdev->y_scale);
469
2.84M
        px_put_ac(s, pxaPageScale, pxtSetPageScale);
470
2.84M
        xdev->scaled = false;
471
2.84M
        xdev->x_scale = 1;
472
2.84M
        xdev->y_scale = 1;
473
2.84M
    }
474
28.1M
    return;
475
28.1M
}
476
477
/* Set the cursor. */
478
static int
479
pclxl_set_cursor(gx_device_pclxl * xdev, int x, int y)
480
8.42M
{
481
8.42M
    stream *s = pclxl_stream(xdev);
482
8.42M
    double x_scale = 1;
483
8.42M
    double y_scale = 1;
484
485
    /* Points must be one of ubyte/uint16/sint16;
486
       Here we play with PageScale (one of ubyte/uint16/real32_xy) to go higher.
487
       This gives us 32768 x 3.4e38 in UnitsPerMeasure.
488
       If we ever need to go higher, we play with UnitsPerMeasure. */
489
8.42M
    if (abs(x) > 0x7FFF) {
490
711k
        x_scale = ((double)abs(x)) / 0x7FFF;
491
711k
        x = (x > 0 ? 0x7FFF : -0x7FFF);
492
711k
        xdev->scaled = true;
493
711k
    }
494
8.42M
    if (abs(y) > 0x7FFF) {
495
1.00M
        y_scale = ((double)abs(y)) / 0x7FFF;
496
1.00M
        y = (y > 0 ? 0x7FFF : -0x7FFF);
497
1.00M
        xdev->scaled = true;
498
1.00M
    }
499
8.42M
    pclxl_set_page_scale(xdev, x_scale, y_scale);
500
8.42M
    px_put_ssp(s, x, y);
501
8.42M
    px_put_ac(s, pxaPoint, pxtSetCursor);
502
8.42M
    pclxl_unset_page_scale(xdev);
503
8.42M
    return 0;
504
8.42M
}
505
506
/* ------ Paths ------ */
507
508
/* Flush any buffered path points. */
509
static void
510
px_put_np(stream * s, int count, pxeDataType_t dtype)
511
13.3M
{
512
13.3M
    px_put_uba(s, (byte) count, pxaNumberOfPoints);
513
13.3M
    px_put_uba(s, (byte) dtype, pxaPointType);
514
13.3M
}
515
static int
516
pclxl_flush_points(gx_device_pclxl * xdev)
517
29.5M
{
518
29.5M
    int count = xdev->points.count;
519
520
29.5M
    if (count) {
521
19.7M
        stream *s = pclxl_stream(xdev);
522
19.7M
        px_tag_t op;
523
19.7M
        int x = xdev->points.current.x, y = xdev->points.current.y;
524
19.7M
        int uor = 0, sor = 0;
525
19.7M
        pxeDataType_t data_type;
526
19.7M
        int i, di;
527
19.7M
        byte diffs[NUM_POINTS * 2];
528
19.7M
        double x_scale = 1;
529
19.7M
        double y_scale = 1;
530
19.7M
        int temp_origin_x = 0, temp_origin_y = 0;
531
19.7M
        int count_smalls = 0;
532
533
19.7M
        if (xdev->points.type != POINTS_NONE) {
534
128M
            for (i = 0; i < count; ++i) {
535
108M
                if ((abs(xdev->points.data[i].x) > 0x7FFF)
536
103M
                    || (abs(xdev->points.data[i].y) > 0x7FFF))
537
7.49M
                    xdev->scaled = true;
538
108M
                if ((abs(xdev->points.data[i].x) < 0x8000)
539
103M
                    && (abs(xdev->points.data[i].y) < 0x8000)) {
540
100M
                    if ((temp_origin_x != xdev->points.data[i].x)
541
89.0M
                        || (temp_origin_y != xdev->points.data[i].y)) {
542
89.0M
                        temp_origin_x = xdev->points.data[i].x;
543
89.0M
                        temp_origin_y = xdev->points.data[i].y;
544
89.0M
                        count_smalls++;
545
89.0M
                    }
546
100M
                }
547
108M
            }
548
19.7M
            if (xdev->scaled) {
549
                /* if there are some points with small co-ordinates, we set origin to it
550
                   before scaling, an unset afterwards. This works around problems
551
                   for small co-ordinates being moved snapped to 32767 x 32767 grid points;
552
                   if there are more than 1, the other points
553
                   will be in-accurate, unfortunately */
554
1.64M
                if (count_smalls) {
555
40.9k
                    pclxl_set_page_origin(s, temp_origin_x, temp_origin_y);
556
40.9k
                }
557
9.27M
                for (i = 0; i < count; ++i) {
558
7.62M
                    x_scale = max(((double)
559
7.62M
                                   abs(xdev->points.data[i].x -
560
7.62M
                                       temp_origin_x)) / 0x7FFF, x_scale);
561
7.62M
                    y_scale = max(((double)
562
7.62M
                                   abs(xdev->points.data[i].y -
563
7.62M
                                       temp_origin_y)) / 0x7FFF, y_scale);
564
7.62M
                }
565
9.27M
                for (i = 0; i < count; ++i) {
566
7.62M
                    xdev->points.data[i].x =
567
7.62M
                        (int)((xdev->points.data[i].x -
568
7.62M
                               temp_origin_x) / x_scale + 0.5);
569
7.62M
                    xdev->points.data[i].y =
570
7.62M
                        (int)((xdev->points.data[i].y -
571
7.62M
                               temp_origin_y) / y_scale + 0.5);
572
7.62M
                }
573
1.64M
                x = (int)((x - temp_origin_x) / x_scale + 0.5);
574
1.64M
                y = (int)((y - temp_origin_y) / y_scale + 0.5);
575
1.64M
                pclxl_set_page_scale(xdev, x_scale, y_scale);
576
18.1M
            } else {
577
                /* don't reset origin if we did not scale */
578
18.1M
                count_smalls = 0;
579
18.1M
            }
580
19.7M
        }
581
        /*
582
         * Writing N lines using a point list requires 11 + 4*N or 11 +
583
         * 2*N bytes, as opposed to 8*N bytes using separate commands;
584
         * writing N curves requires 11 + 12*N or 11 + 6*N bytes
585
         * vs. 22*N.  So it's always shorter to write curves with a
586
         * list (except for N = 1 with full-size coordinates, but since
587
         * the difference is only 1 byte, we don't bother to ever use
588
         * the non-list form), but lines are shorter only if N >= 3
589
         * (again, with a 1-byte difference if N = 2 and byte
590
         * coordinates).
591
         */
592
19.7M
        switch (xdev->points.type) {
593
0
            case POINTS_NONE:
594
0
                return 0;
595
12.2M
            case POINTS_LINES:
596
12.2M
                op = pxtLinePath;
597
12.2M
                if (count < 3) {
598
14.8M
                    for (i = 0; i < count; ++i) {
599
8.51M
                        px_put_ssp(s, xdev->points.data[i].x,
600
8.51M
                                   xdev->points.data[i].y);
601
8.51M
                        px_put_a(s, pxaEndPoint);
602
8.51M
                        spputc(s, (byte) op);
603
8.51M
                    }
604
6.37M
                    pclxl_unset_page_scale(xdev);
605
6.37M
                    if (count_smalls)
606
17.8k
                        pclxl_set_page_origin(s, -temp_origin_x,
607
17.8k
                                              -temp_origin_y);
608
6.37M
                    goto zap;
609
6.37M
                }
610
                /* See if we can use byte values. */
611
28.6M
                for (i = di = 0; i < count; ++i, di += 2) {
612
22.8M
                    int dx = xdev->points.data[i].x - x;
613
22.8M
                    int dy = xdev->points.data[i].y - y;
614
615
22.8M
                    diffs[di] = (byte) dx;
616
22.8M
                    diffs[di + 1] = (byte) dy;
617
22.8M
                    uor |= dx | dy;
618
22.8M
                    sor |= (dx + 0x80) | (dy + 0x80);
619
22.8M
                    x += dx, y += dy;
620
22.8M
                }
621
5.85M
                if (!(uor & ~0xff))
622
347k
                    data_type = eUByte;
623
5.50M
                else if (!(sor & ~0xff))
624
4.16M
                    data_type = eSByte;
625
1.33M
                else
626
1.33M
                    break;
627
4.51M
                op = pxtLineRelPath;
628
                /* Use byte values. */
629
11.7M
              useb:px_put_np(s, count, data_type);
630
11.7M
                spputc(s, (byte) op);
631
11.7M
                px_put_data_length(s, count * 2);       /* 2 bytes per point */
632
11.7M
                px_put_bytes(s, diffs, count * 2);
633
11.7M
                pclxl_unset_page_scale(xdev);
634
11.7M
                if (count_smalls)
635
71
                    pclxl_set_page_origin(s, -temp_origin_x, -temp_origin_y);
636
11.7M
                goto zap;
637
7.53M
            case POINTS_CURVES:
638
7.53M
                op = pxtBezierPath;
639
                /* See if we can use byte values. */
640
33.2M
                for (i = di = 0; i < count; i += 3, di += 6) {
641
25.6M
                    int dx1 = xdev->points.data[i].x - x;
642
25.6M
                    int dy1 = xdev->points.data[i].y - y;
643
25.6M
                    int dx2 = xdev->points.data[i + 1].x - x;
644
25.6M
                    int dy2 = xdev->points.data[i + 1].y - y;
645
25.6M
                    int dx = xdev->points.data[i + 2].x - x;
646
25.6M
                    int dy = xdev->points.data[i + 2].y - y;
647
648
25.6M
                    diffs[di] = (byte) dx1;
649
25.6M
                    diffs[di + 1] = (byte) dy1;
650
25.6M
                    diffs[di + 2] = (byte) dx2;
651
25.6M
                    diffs[di + 3] = (byte) dy2;
652
25.6M
                    diffs[di + 4] = (byte) dx;
653
25.6M
                    diffs[di + 5] = (byte) dy;
654
25.6M
                    uor |= dx1 | dy1 | dx2 | dy2 | dx | dy;
655
25.6M
                    sor |= (dx1 + 0x80) | (dy1 + 0x80) |
656
25.6M
                        (dx2 + 0x80) | (dy2 + 0x80) |
657
25.6M
                        (dx + 0x80) | (dy + 0x80);
658
25.6M
                    x += dx, y += dy;
659
25.6M
                }
660
7.53M
                if (!(uor & ~0xff))
661
890k
                    data_type = eUByte;
662
6.64M
                else if (!(sor & ~0xff))
663
6.38M
                    data_type = eSByte;
664
263k
                else
665
263k
                    break;
666
7.27M
                op = pxtBezierRelPath;
667
7.27M
                goto useb;
668
0
            default:           /* can't happen */
669
0
                return_error(gs_error_unknownerror);
670
19.7M
        }
671
1.60M
        px_put_np(s, count, eSInt16);
672
1.60M
        spputc(s, (byte) op);
673
1.60M
        px_put_data_length(s, count * 4);       /* 2 UInt16s per point */
674
9.46M
        for (i = 0; i < count; ++i) {
675
7.85M
            px_put_s(s, xdev->points.data[i].x);
676
7.85M
            px_put_s(s, xdev->points.data[i].y);
677
7.85M
        }
678
1.60M
        pclxl_unset_page_scale(xdev);
679
1.60M
        if (count_smalls)
680
23.0k
            pclxl_set_page_origin(s, -temp_origin_x, -temp_origin_y);
681
19.7M
      zap:xdev->points.type = POINTS_NONE;
682
19.7M
        xdev->points.count = 0;
683
19.7M
    }
684
29.5M
    return 0;
685
29.5M
}
686
687
/* ------ Images ------ */
688
689
static image_enum_proc_plane_data(pclxl_image_plane_data);
690
static image_enum_proc_end_image(pclxl_image_end_image);
691
692
static const gx_image_enum_procs_t pclxl_image_enum_procs = {
693
    pclxl_image_plane_data, pclxl_image_end_image
694
};
695
696
/* Begin an image. */
697
static void
698
pclxl_write_begin_image(gx_device_pclxl * xdev, uint width, uint height,
699
                        uint dest_width, uint dest_height)
700
137k
{
701
137k
    stream *s = pclxl_stream(xdev);
702
703
137k
    px_put_usa(s, width, pxaSourceWidth);
704
137k
    px_put_usa(s, height, pxaSourceHeight);
705
137k
    px_put_usp(s, dest_width, dest_height);
706
137k
    px_put_ac(s, pxaDestinationSize, pxtBeginImage);
707
137k
}
708
709
/* Write rows of an image. */
710
/****** IGNORES data_bit ******/
711
/* 2009: we try to cope with the case of data_bit being multiple of 8 now */
712
/* RLE version */
713
static void
714
pclxl_write_image_data_RLE(gx_device_pclxl * xdev, const byte * base,
715
                           int data_bit, uint raster, uint width_bits, int y,
716
                           int height)
717
137k
{
718
137k
    stream *s = pclxl_stream(xdev);
719
137k
    uint width_bytes = (width_bits + 7) >> 3;
720
137k
    uint num_bytes = ROUND_UP(width_bytes, 4) * height;
721
137k
    bool compress = num_bytes >= 8;
722
137k
    int i;
723
137k
    int code;
724
725
    /* cannot handle data_bit not multiple of 8, but we don't invoke this routine that way */
726
137k
    int offset = data_bit >> 3;
727
137k
    const byte *data = base + offset;
728
729
137k
    px_put_usa(s, y, pxaStartLine);
730
137k
    px_put_usa(s, height, pxaBlockHeight);
731
137k
    if (compress) {
732
137k
        stream_RLE_state rlstate;
733
137k
        stream_cursor_write w;
734
137k
        stream_cursor_read r;
735
736
        /*
737
         * H-P printers require that all the data for an operator be
738
         * contained in a single data block.  Thus, we must allocate a
739
         * temporary buffer for the compressed data.  Currently we don't go
740
         * to the trouble of doing two passes if we can't allocate a buffer
741
         * large enough for the entire transfer.
742
         */
743
137k
        byte *buf = gs_alloc_bytes(xdev->v_memory, num_bytes,
744
137k
                                   "pclxl_write_image_data");
745
746
137k
        if (buf == 0)
747
0
            goto nc;
748
137k
        s_RLE_set_defaults_inline(&rlstate);
749
137k
        rlstate.EndOfData = false;
750
137k
        rlstate.omitEOD = true;
751
137k
        s_RLE_init_inline(&rlstate);
752
137k
        w.ptr = buf - 1;
753
137k
        w.limit = w.ptr + num_bytes;
754
        /*
755
         * If we ever overrun the buffer, it means that the compressed
756
         * data was larger than the uncompressed.  If this happens,
757
         * write the data uncompressed.
758
         */
759
9.05M
        for (i = 0; i < height; ++i) {
760
8.92M
            r.ptr = data + i * raster - 1;
761
8.92M
            r.limit = r.ptr + width_bytes;
762
8.92M
            if ((*s_RLE_template.process)
763
8.92M
                ((stream_state *) & rlstate, &r, &w, false) != 0 ||
764
8.91M
                r.ptr != r.limit)
765
14.1k
                goto ncfree;
766
8.91M
            r.ptr = (const byte *)"\000\000\000\000\000";
767
8.91M
            r.limit = r.ptr + (-(int)width_bytes & 3);
768
8.91M
            if ((*s_RLE_template.process)
769
8.91M
                ((stream_state *) & rlstate, &r, &w, false) != 0 ||
770
8.91M
                r.ptr != r.limit)
771
9
                goto ncfree;
772
8.91M
        }
773
123k
        r.ptr = r.limit;
774
123k
        code = (*s_RLE_template.process)
775
123k
            ((stream_state *) & rlstate, &r, &w, true);
776
123k
        if (code != EOFC && code != 0)
777
43.4k
            goto ncfree;
778
79.6k
        {
779
79.6k
            uint count = w.ptr + 1 - buf;
780
781
79.6k
            px_put_ub(s, eRLECompression);
782
79.6k
            px_put_ac(s, pxaCompressMode, pxtReadImage);
783
79.6k
            px_put_data_length(s, count);
784
79.6k
            px_put_bytes(s, buf, count);
785
79.6k
        }
786
79.6k
        gs_free_object(xdev->v_memory, buf, "pclxl_write_image_data");
787
79.6k
        return;
788
57.6k
      ncfree:gs_free_object(xdev->v_memory, buf,
789
57.6k
                       "pclxl_write_image_data");
790
57.6k
    }
791
57.6k
  nc:
792
    /* Write the data uncompressed. */
793
57.6k
    px_put_ub(s, eNoCompression);
794
57.6k
    px_put_ac(s, pxaCompressMode, pxtReadImage);
795
57.6k
    px_put_data_length(s, num_bytes);
796
3.46M
    for (i = 0; i < height; ++i) {
797
3.40M
        px_put_bytes(s, data + i * raster, width_bytes);
798
3.40M
        px_put_bytes(s, (const byte *)"\000\000\000\000",
799
3.40M
                     -(int)width_bytes & 3);
800
3.40M
    }
801
57.6k
}
802
803
static void
804
pclxl_write_image_data_JPEG(gx_device_pclxl * xdev, const byte * base,
805
                            int data_bit, uint raster, uint width_bits, int y,
806
                            int height)
807
0
{
808
0
    stream *s = pclxl_stream(xdev);
809
0
    uint width_bytes = (width_bits + 7) >> 3;
810
0
    int i;
811
0
    int count;
812
0
    int code;
813
814
    /* cannot handle data_bit not multiple of 8, but we don't invoke this routine that way */
815
0
    int offset = data_bit >> 3;
816
0
    const byte *data = base + offset;
817
0
    jpeg_compress_data *jcdp =
818
0
        gs_alloc_struct_immovable(xdev->v_memory, jpeg_compress_data,
819
0
                                  &st_jpeg_compress_data,
820
0
                                  "pclxl_write_image_data_JPEG(jpeg_compress_data)");
821
0
    stream_DCT_state state;
822
0
    stream_cursor_read r;
823
0
    stream_cursor_write w;
824
825
    /* Approx. The worse case is ~ header + width_bytes * height.
826
       Apparently minimal SOI/DHT/DQT/SOS/EOI is 341 bytes. TO CHECK. */
827
0
    size_t buffersize = 341 + width_bytes * height;
828
829
0
    byte *buf = gs_alloc_bytes(xdev->v_memory, buffersize,
830
0
                               "pclxl_write_image_data_JPEG(buf)");
831
832
    /* RLE can write uncompressed without extra-allocation */
833
0
    if ((buf == 0) || (jcdp == 0)) {
834
0
        goto failed_so_use_rle_instead;
835
0
    }
836
    /* Create the DCT encoder state. */
837
0
    jcdp->templat = s_DCTE_template;
838
0
    s_init_state((stream_state *) & state, &jcdp->templat, 0);
839
0
    if (state.templat->set_defaults) {
840
0
        state.memory = xdev->v_memory;
841
0
        (*state.templat->set_defaults) ((stream_state *) & state);
842
0
        state.memory = NULL;
843
0
    }
844
0
    state.ColorTransform = (xdev->color_info.num_components == 3 ? 1 : 0);
845
0
    state.data.compress = jcdp;
846
0
    state.icc_profile = NULL;
847
    /* state.memory needs set for creation..... */
848
0
    state.memory = jcdp->memory = state.jpeg_memory = xdev->v_memory;
849
0
    if ((code = gs_jpeg_create_compress(&state)) < 0)
850
0
        goto cleanup_and_use_rle;
851
    /* .... and NULL after, so we don't try to free the stack based "state" */
852
0
    state.memory = NULL;
853
    /* image-specific info */
854
0
    jcdp->cinfo.image_width = width_bytes / xdev->color_info.num_components;
855
0
    jcdp->cinfo.image_height = height;
856
0
    switch (xdev->color_info.num_components) {
857
0
        case 3:
858
0
            jcdp->cinfo.input_components = 3;
859
0
            jcdp->cinfo.in_color_space = JCS_RGB;
860
0
            break;
861
0
        case 1:
862
0
            jcdp->cinfo.input_components = 1;
863
0
            jcdp->cinfo.in_color_space = JCS_GRAYSCALE;
864
0
            break;
865
0
        default:
866
0
            goto cleanup_and_use_rle;
867
0
            break;
868
0
    }
869
    /* Set compression parameters. */
870
0
    if ((code = gs_jpeg_set_defaults(&state)) < 0)
871
0
        goto cleanup_and_use_rle;
872
873
0
    if (state.templat->init)
874
0
        (*state.templat->init) ((stream_state *) & state);
875
0
    state.scan_line_size = jcdp->cinfo.input_components *
876
0
        jcdp->cinfo.image_width;
877
0
    jcdp->templat.min_in_size =
878
0
        max(s_DCTE_template.min_in_size, state.scan_line_size);
879
0
    jcdp->templat.min_out_size =
880
0
        max(s_DCTE_template.min_out_size, state.Markers.size);
881
882
0
    w.ptr = buf - 1;
883
0
    w.limit = w.ptr + buffersize;
884
0
    for (i = 0; i < height; ++i) {
885
0
        r.ptr = data + i * raster - 1;
886
0
        r.limit = r.ptr + width_bytes;
887
0
        if (((code = (*state.templat->process)
888
0
              ((stream_state *) & state, &r, &w, false)) != 0 && code != EOFC)
889
0
            || r.ptr != r.limit)
890
0
            goto cleanup_and_use_rle;
891
0
    }
892
0
    count = w.ptr + 1 - buf;
893
0
    px_put_usa(s, y, pxaStartLine);
894
0
    px_put_usa(s, height, pxaBlockHeight);
895
0
    px_put_ub(s, eJPEGCompression);
896
0
    px_put_ac(s, pxaCompressMode, pxtReadImage);
897
0
    px_put_data_length(s, count);
898
0
    px_put_bytes(s, buf, count);
899
900
0
    gs_free_object(xdev->v_memory, buf, "pclxl_write_image_data_JPEG(buf)");
901
0
    if (jcdp)
902
0
        gs_jpeg_destroy(&state);        /* frees *jcdp */
903
0
    return;
904
905
0
  cleanup_and_use_rle:
906
    /* cleans up - something went wrong after allocation */
907
0
    gs_free_object(xdev->v_memory, buf, "pclxl_write_image_data_JPEG(buf)");
908
0
    if (jcdp)
909
0
        gs_jpeg_destroy(&state);        /* frees *jcdp */
910
    /* fall through to redo in RLE */
911
0
  failed_so_use_rle_instead:
912
    /* the RLE routine can write without new allocation - use as fallback. */
913
0
    pclxl_write_image_data_RLE(xdev, data, data_bit, raster, width_bits, y,
914
0
                               height);
915
0
    return;
916
0
}
917
918
/* DeltaRow compression (also called "mode 3"):
919
   drawn heavily from gdevcljc.c:cljc_print_page(),
920
   This is simplier since PCL XL does not allow
921
   compression mix-and-match.
922
923
   Worse case of RLE is + 1/128, but worse case of DeltaRow is + 1/8
924
 */
925
static void
926
pclxl_write_image_data_DeltaRow(gx_device_pclxl * xdev, const byte * base,
927
                                int data_bit, uint raster, uint width_bits,
928
                                int y, int height)
929
0
{
930
0
    stream *s = pclxl_stream(xdev);
931
0
    uint width_bytes = (width_bits + 7) >> 3;
932
0
    int worst_case_comp_size = width_bytes + (width_bytes / 8) + 1;
933
0
    byte *cdata = 0;
934
0
    byte *prow = 0;
935
0
    int i;
936
0
    int count;
937
938
    /* cannot handle data_bit not multiple of 8, but we don't invoke this routine that way */
939
0
    int offset = data_bit >> 3;
940
0
    const byte *data = base + offset;
941
942
    /* allocate the worst case scenario; PCL XL has an extra 2 byte per row compared to PCL5 */
943
0
    byte *buf =
944
0
        gs_alloc_bytes(xdev->v_memory, (worst_case_comp_size + 2) * (size_t)height,
945
0
                       "pclxl_write_image_data_DeltaRow(buf)");
946
947
0
    prow =
948
0
        gs_alloc_bytes(xdev->v_memory, width_bytes,
949
0
                       "pclxl_write_image_data_DeltaRow(prow)");
950
    /* the RLE routine can write uncompressed without extra-allocation */
951
0
    if ((buf == 0) || (prow == 0)) {
952
0
        pclxl_write_image_data_RLE(xdev, data, data_bit, raster, width_bits,
953
0
                                   y, height);
954
0
        return;
955
0
    }
956
    /* initialize the seed row */
957
0
    memset(prow, 0, width_bytes);
958
0
    cdata = buf;
959
0
    for (i = 0; i < height; i++) {
960
0
        int compressed_size =
961
0
            gdev_pcl_mode3compress(width_bytes, data + i * raster, prow,
962
0
                                   cdata + 2);
963
964
        /* PCL XL prepends row data with byte count */
965
0
        *cdata = compressed_size & 0xff;
966
0
        *(cdata + 1) = compressed_size >> 8;
967
0
        cdata += compressed_size + 2;
968
0
    }
969
0
    px_put_usa(s, y, pxaStartLine);
970
0
    px_put_usa(s, height, pxaBlockHeight);
971
0
    px_put_ub(s, eDeltaRowCompression);
972
0
    px_put_ac(s, pxaCompressMode, pxtReadImage);
973
0
    count = cdata - buf;
974
0
    px_put_data_length(s, count);
975
0
    px_put_bytes(s, buf, count);
976
977
0
    gs_free_object(xdev->v_memory, buf,
978
0
                   "pclxl_write_image_data_DeltaRow(buf)");
979
0
    gs_free_object(xdev->v_memory, prow,
980
0
                   "pclxl_write_image_data_DeltaRow(prow)");
981
0
    return;
982
0
}
983
984
/* calling from copy_mono/copy_color/fill_mask should never do lossy compression */
985
static void
986
pclxl_write_image_data(gx_device_pclxl * xdev, const byte * data,
987
                       int data_bit, uint raster, uint width_bits, int y,
988
                       int height, bool allow_lossy)
989
137k
{
990
    /* If we only have 1 line, it does not make sense to do JPEG/DeltaRow */
991
137k
    if (height < 2) {
992
102
        pclxl_write_image_data_RLE(xdev, data, data_bit, raster, width_bits,
993
102
                                   y, height);
994
102
        return;
995
102
    }
996
997
137k
    switch (xdev->CompressMode) {
998
0
        case eDeltaRowCompression:
999
0
            pclxl_write_image_data_DeltaRow(xdev, data, data_bit, raster,
1000
0
                                            width_bits, y, height);
1001
0
            break;
1002
0
        case eJPEGCompression:
1003
            /* JPEG should not be used for mask or other data */
1004
0
            if (allow_lossy)
1005
0
                pclxl_write_image_data_JPEG(xdev, data, data_bit, raster,
1006
0
                                            width_bits, y, height);
1007
0
            else
1008
0
                pclxl_write_image_data_RLE(xdev, data, data_bit, raster,
1009
0
                                           width_bits, y, height);
1010
0
            break;
1011
0
        case eRLECompression:
1012
137k
        default:
1013
137k
            pclxl_write_image_data_RLE(xdev, data, data_bit, raster,
1014
137k
                                       width_bits, y, height);
1015
137k
            break;
1016
137k
    }
1017
137k
}
1018
1019
/* End an image. */
1020
static void
1021
pclxl_write_end_image(gx_device_pclxl * xdev)
1022
137k
{
1023
137k
    spputc(xdev->strm, pxtEndImage);
1024
137k
}
1025
1026
/* ------ Fonts ------ */
1027
1028
/* Write a string (single- or double-byte). */
1029
static void
1030
px_put_string(stream * s, const byte * data, uint len, bool wide)
1031
17.7k
{
1032
17.7k
    if (wide) {
1033
0
        spputc(s, pxt_uint16_array);
1034
0
        px_put_u(s, len);
1035
0
        px_put_bytes(s, data, len * 2);
1036
17.7k
    } else {
1037
17.7k
        spputc(s, pxt_ubyte_array);
1038
17.7k
        px_put_u(s, len);
1039
17.7k
        px_put_bytes(s, data, len);
1040
17.7k
    }
1041
17.7k
}
1042
1043
/* Write a 16-bit big-endian value. */
1044
static void
1045
px_put_us_be(stream * s, uint i)
1046
1.08k
{
1047
1.08k
    spputc(s, (byte) (i >> 8));
1048
1.08k
    spputc(s, (byte) i);
1049
1.08k
}
1050
1051
/* Define a bitmap font.  The client must call px_put_string */
1052
/* with the font name immediately before calling this procedure. */
1053
static void
1054
pclxl_define_bitmap_font(gx_device_pclxl * xdev)
1055
21
{
1056
21
    stream *s = pclxl_stream(xdev);
1057
1058
21
    static const byte bfh_[] = {
1059
21
        DA(pxaFontName), DUB(0), DA(pxaFontFormat),
1060
21
        pxtBeginFontHeader,
1061
21
        DUS(8 + 6 + 4 + 6), DA(pxaFontHeaderLength),
1062
21
        pxtReadFontHeader,
1063
21
        pxt_dataLengthByte, 8 + 6 + 4 + 6,
1064
21
        0, 0, 0, 0,
1065
21
        254, 0, (MAX_CACHED_CHARS + 255) >> 8, 0,
1066
21
        'B', 'R', 0, 0, 0, 4
1067
21
    };
1068
21
    static const byte efh_[] = {
1069
21
        0xff, 0xff, 0, 0, 0, 0,
1070
21
        pxtEndFontHeader
1071
21
    };
1072
1073
21
    PX_PUT_LIT(s, bfh_);
1074
21
    px_put_us_be(s, (uint) (xdev->HWResolution[0] + 0.5));
1075
21
    px_put_us_be(s, (uint) (xdev->HWResolution[1] + 0.5));
1076
21
    PX_PUT_LIT(s, efh_);
1077
21
}
1078
1079
/* Set the font.  The client must call px_put_string */
1080
/* with the font name immediately before calling this procedure. */
1081
static void
1082
pclxl_set_font(gx_device_pclxl * xdev)
1083
29
{
1084
29
    stream *s = pclxl_stream(xdev);
1085
1086
29
    static const byte sf_[] = {
1087
29
        DA(pxaFontName), DUB(1), DA(pxaCharSize), DUS(0), DA(pxaSymbolSet),
1088
29
        pxtSetFont
1089
29
    };
1090
1091
29
    PX_PUT_LIT(s, sf_);
1092
29
}
1093
1094
/* Define a character in a bitmap font.  The client must call px_put_string */
1095
/* with the font name immediately before calling this procedure. */
1096
static void
1097
pclxl_define_bitmap_char(gx_device_pclxl * xdev, uint ccode,
1098
                         const byte * data, uint raster, uint width_bits,
1099
                         uint height)
1100
523
{
1101
523
    stream *s = pclxl_stream(xdev);
1102
523
    uint width_bytes = (width_bits + 7) >> 3;
1103
523
    uint size = 10 + width_bytes * height;
1104
523
    uint i;
1105
1106
523
    px_put_ac(s, pxaFontName, pxtBeginChar);
1107
523
    px_put_u(s, ccode);
1108
523
    px_put_a(s, pxaCharCode);
1109
523
    if (size > 0xffff) {
1110
0
        spputc(s, pxt_uint32);
1111
0
        px_put_l(s, (ulong) size);
1112
0
    } else
1113
523
        px_put_us(s, size);
1114
523
    px_put_ac(s, pxaCharDataSize, pxtReadChar);
1115
523
    px_put_data_length(s, size);
1116
523
    px_put_bytes(s, (const byte *)"\000\000\000\000\000\000", 6);
1117
523
    px_put_us_be(s, width_bits);
1118
523
    px_put_us_be(s, height);
1119
10.1k
    for (i = 0; i < height; ++i)
1120
9.62k
        px_put_bytes(s, data + i * raster, width_bytes);
1121
523
    spputc(s, pxtEndChar);
1122
523
}
1123
1124
/* Write the name of the only font we define. */
1125
static void
1126
pclxl_write_font_name(gx_device_pclxl * xdev)
1127
573
{
1128
573
    stream *s = pclxl_stream(xdev);
1129
1130
573
    px_put_string(s, (const byte *)"@", 1, false);
1131
573
}
1132
1133
/* Look up a bitmap id, return the index in the character table. */
1134
/* If the id is missing, return an index for inserting. */
1135
static int
1136
pclxl_char_index(gx_device_pclxl * xdev, gs_id id)
1137
17.6k
{
1138
17.6k
    int i, i_empty = -1;
1139
17.6k
    uint ccode;
1140
1141
17.6k
    for (i = (id * CHAR_HASH_FACTOR) % countof(xdev->chars.table);;
1142
17.6k
         i = (i == 0 ? countof(xdev->chars.table) : i) - 1) {
1143
17.6k
        ccode = xdev->chars.table[i];
1144
17.6k
        if (ccode == 0)
1145
1.04k
            return (i_empty >= 0 ? i_empty : i);
1146
16.6k
        else if (ccode == 1) {
1147
0
            if (i_empty < 0)
1148
0
                i_empty = i;
1149
0
            else if (i == i_empty)      /* full table */
1150
0
                return i;
1151
16.6k
        } else if (xdev->chars.data[ccode].id == id)
1152
16.6k
            return i;
1153
17.6k
    }
1154
17.6k
}
1155
1156
/* Remove the character table entry at a given index. */
1157
static void
1158
pclxl_remove_char(gx_device_pclxl * xdev, int index)
1159
0
{
1160
0
    uint ccode = xdev->chars.table[index];
1161
0
    int i;
1162
1163
0
    if (ccode < 2)
1164
0
        return;
1165
0
    xdev->chars.count--;
1166
0
    xdev->chars.used -= xdev->chars.data[ccode].size;
1167
0
    xdev->chars.table[index] = 1;       /* mark as deleted */
1168
0
    i = (index == 0 ? countof(xdev->chars.table) : index) - 1;
1169
0
    if (xdev->chars.table[i] == 0) {
1170
        /* The next slot in probe order is empty. */
1171
        /* Mark this slot and any deleted predecessors as empty. */
1172
0
        for (i = index; xdev->chars.table[i] == 1;
1173
0
             i = (i == countof(xdev->chars.table) - 1 ? 0 : i + 1)
1174
0
            )
1175
0
            xdev->chars.table[i] = 0;
1176
0
    }
1177
0
}
1178
1179
/* Write a bitmap as a text character if possible. */
1180
/* The caller must set the color, cursor, and RasterOp. */
1181
/* We know id != gs_no_id. */
1182
static int
1183
pclxl_copy_text_char(gx_device_pclxl * xdev, const byte * data,
1184
                     int raster, gx_bitmap_id id, int w, int h)
1185
17.1k
{
1186
17.1k
    uint width_bytes = (w + 7) >> 3;
1187
17.1k
    uint size = width_bytes * h;
1188
17.1k
    int index;
1189
17.1k
    uint ccode;
1190
17.1k
    stream *s = pclxl_stream(xdev);
1191
1192
17.1k
    if (size > MAX_CHAR_SIZE)
1193
0
        return -1;
1194
17.1k
    index = pclxl_char_index(xdev, id);
1195
17.1k
    if ((ccode = xdev->chars.table[index]) < 2) {
1196
        /* Enter the character in the table. */
1197
523
        while (xdev->chars.used + size > MAX_CHAR_DATA ||
1198
523
               xdev->chars.count >= MAX_CACHED_CHARS - 2) {
1199
0
            ccode = xdev->chars.next_out;
1200
0
            index = pclxl_char_index(xdev, xdev->chars.data[ccode].id);
1201
0
            pclxl_remove_char(xdev, index);
1202
0
            xdev->chars.next_out =
1203
0
                (ccode == MAX_CACHED_CHARS - 1 ? 2 : ccode + 1);
1204
0
        }
1205
523
        index = pclxl_char_index(xdev, id);
1206
523
        ccode = xdev->chars.next_in;
1207
523
        xdev->chars.data[ccode].id = id;
1208
523
        xdev->chars.data[ccode].size = size;
1209
523
        xdev->chars.table[index] = ccode;
1210
523
        xdev->chars.next_in = (ccode == MAX_CACHED_CHARS - 1 ? 2 : ccode + 1);
1211
523
        if (!xdev->chars.count++) {
1212
            /* This is the very first character. */
1213
21
            pclxl_write_font_name(xdev);
1214
21
            pclxl_define_bitmap_font(xdev);
1215
21
        }
1216
523
        xdev->chars.used += size;
1217
523
        pclxl_write_font_name(xdev);
1218
523
        pclxl_define_bitmap_char(xdev, ccode, data, raster, w, h);
1219
523
    }
1220
17.1k
    if (!xdev->font_set) {
1221
29
        pclxl_write_font_name(xdev);
1222
29
        pclxl_set_font(xdev);
1223
29
        xdev->font_set = true;
1224
29
    }
1225
17.1k
    {
1226
17.1k
        byte cc_bytes[2];
1227
1228
17.1k
        cc_bytes[0] = (byte) ccode;
1229
17.1k
        cc_bytes[1] = ccode >> 8;
1230
17.1k
        px_put_string(s, cc_bytes, 1, cc_bytes[1] != 0);
1231
17.1k
    }
1232
17.1k
    px_put_ac(s, pxaTextData, pxtText);
1233
17.1k
    return 0;
1234
17.1k
}
1235
1236
/* ---------------- Vector implementation procedures ---------------- */
1237
1238
static int
1239
pclxl_beginpage(gx_device_vector * vdev)
1240
21.5k
{
1241
21.5k
    gx_device_pclxl *const xdev = (gx_device_pclxl *) vdev;
1242
1243
    /*
1244
     * We can't use gdev_vector_stream here, because this may be called
1245
     * from there before in_page is set.
1246
     */
1247
21.5k
    stream *s = vdev->strm;
1248
21.5k
    byte media_source = eAutoSelect;    /* default */
1249
1250
21.5k
    xdev->page++;               /* even/odd for duplex front/back */
1251
1252
/*
1253
    errprintf(vdev->memory, "PAGE: %d %d\n", xdev->page, xdev->NumCopies);
1254
    errprintf(vdev->memory, "INFO: Printing page %d...\n", xdev->page);
1255
    errflush(vdev->memory);
1256
*/
1257
1258
21.5k
    px_write_page_header(s, (const gx_device *)vdev);
1259
1260
21.5k
    if (xdev->ManualFeed_set && xdev->ManualFeed)
1261
0
        media_source = 2;
1262
21.5k
    else if (xdev->MediaPosition_set && xdev->MediaPosition >= 0)
1263
21.5k
        media_source = xdev->MediaPosition;
1264
1265
21.5k
    px_write_select_media(s, (const gx_device *)vdev, &xdev->media_size,
1266
21.5k
                          &media_source,
1267
21.5k
                          xdev->page, xdev->Duplex, xdev->Tumble,
1268
21.5k
                          xdev->MediaType_set, xdev->MediaType);
1269
1270
21.5k
    spputc(s, pxtBeginPage);
1271
21.5k
    return 0;
1272
21.5k
}
1273
1274
static int
1275
pclxl_setlinewidth(gx_device_vector * vdev, double width)
1276
9.56k
{
1277
9.56k
    stream *s = gdev_vector_stream(vdev);
1278
1279
9.56k
    px_put_us(s, (uint) (width + 0.5));
1280
9.56k
    px_put_ac(s, pxaPenWidth, pxtSetPenWidth);
1281
9.56k
    return 0;
1282
9.56k
}
1283
1284
static int
1285
pclxl_setlinecap(gx_device_vector * vdev, gs_line_cap cap)
1286
1.81k
{
1287
1.81k
    stream *s = gdev_vector_stream(vdev);
1288
1289
    /* The PCL XL cap styles just happen to be identical to PostScript. */
1290
1.81k
    px_put_ub(s, (byte) cap);
1291
1.81k
    px_put_ac(s, pxaLineCapStyle, pxtSetLineCap);
1292
1.81k
    return 0;
1293
1.81k
}
1294
1295
static int
1296
pclxl_setlinejoin(gx_device_vector * vdev, gs_line_join join)
1297
711
{
1298
711
    stream *s = gdev_vector_stream(vdev);
1299
1300
711
    if (((int)join < 0) || ((int)join > 3)) {
1301
4
        emprintf1(vdev->memory,
1302
4
                  "Igoring invalid linejoin enumerator %d\n", join);
1303
4
        return 0;
1304
4
    }
1305
    /* The PCL XL join styles just happen to be identical to PostScript. */
1306
707
    px_put_ub(s, (byte) join);
1307
707
    px_put_ac(s, pxaLineJoinStyle, pxtSetLineJoin);
1308
707
    return 0;
1309
711
}
1310
1311
static int
1312
pclxl_setmiterlimit(gx_device_vector * vdev, double limit)
1313
483
{
1314
483
    stream *s = gdev_vector_stream(vdev);
1315
1316
    /*
1317
     * Amazingly enough, the PCL XL specification doesn't allow real
1318
     * numbers for the miter limit.
1319
     */
1320
483
    int i_limit = (int)(limit + 0.5);
1321
1322
483
    px_put_u(s, max(i_limit, 1));
1323
483
    px_put_ac(s, pxaMiterLength, pxtSetMiterLimit);
1324
483
    return 0;
1325
483
}
1326
1327
/*
1328
 * The number of elements in the dash pattern array is device
1329
 * dependent but a maximum of 20 has been observed on several HP
1330
 * printers.
1331
 */
1332
1333
491
#define MAX_DASH_ELEMENTS 20
1334
1335
static int
1336
pclxl_setdash(gx_device_vector * vdev, const float *pattern, uint count,
1337
              double offset)
1338
766
{
1339
766
    stream *s = gdev_vector_stream(vdev);
1340
1341
766
    if (count == 0) {
1342
275
        static const byte nac_[] = {
1343
275
            DUB(0), DA(pxaSolidLine)
1344
275
        };
1345
1346
275
        PX_PUT_LIT(s, nac_);
1347
491
    } else if (count > MAX_DASH_ELEMENTS)
1348
0
        return_error(gs_error_limitcheck);
1349
491
    else {
1350
491
        uint i;
1351
491
        uint pattern_length = 0;
1352
        /*
1353
         * Astoundingly, PCL XL doesn't allow real numbers here.
1354
         * Do the best we can.
1355
         */
1356
1357
        /* check if the resulting total pattern length will be 0 */
1358
1.34k
        for (i = 0; i < count; ++i)
1359
855
            pattern_length += (uint) (pattern[i]);
1360
491
        if (pattern_length == 0)
1361
0
            return_error(gs_error_rangecheck);
1362
1363
491
        spputc(s, pxt_uint16_array);
1364
491
        px_put_ub(s, (byte) count);
1365
1.34k
        for (i = 0; i < count; ++i)
1366
855
            px_put_s(s, (uint) pattern[i]);
1367
491
        px_put_a(s, pxaLineDashStyle);
1368
491
        if (offset != 0)
1369
2
            px_put_usa(s, (uint) offset, pxaDashOffset);
1370
491
    }
1371
766
    spputc(s, pxtSetLineDash);
1372
766
    return 0;
1373
766
}
1374
1375
static int
1376
pclxl_setlogop(gx_device_vector * vdev, gs_logical_operation_t lop,
1377
               gs_logical_operation_t diff)
1378
460k
{
1379
460k
    stream *s = gdev_vector_stream(vdev);
1380
1381
460k
    if (diff & lop_S_transparent) {
1382
6.44k
        px_put_ub(s, (byte) (lop & lop_S_transparent ? 1 : 0));
1383
6.44k
        px_put_ac(s, pxaTxMode, pxtSetSourceTxMode);
1384
6.44k
    }
1385
460k
    if (diff & lop_T_transparent) {
1386
544
        px_put_ub(s, (byte) (lop & lop_T_transparent ? 1 : 0));
1387
544
        px_put_ac(s, pxaTxMode, pxtSetPaintTxMode);
1388
544
    }
1389
460k
    if (lop_rop(diff)) {
1390
455k
        px_put_ub(s, (byte) lop_rop(lop));
1391
455k
        px_put_ac(s, pxaROP3, pxtSetROP);
1392
455k
    }
1393
460k
    return 0;
1394
460k
}
1395
1396
static bool
1397
pclxl_can_handle_hl_color(gx_device_vector * vdev, const gs_gstate * pgs,
1398
                          const gx_drawing_color * pdc)
1399
44.4M
{
1400
44.4M
    return false;
1401
44.4M
}
1402
1403
static int
1404
pclxl_setfillcolor(gx_device_vector * vdev, const gs_gstate * pgs,
1405
                   const gx_drawing_color * pdc)
1406
21.8M
{
1407
21.8M
    gx_device_pclxl *const xdev = (gx_device_pclxl *) vdev;
1408
1409
21.8M
    return pclxl_set_color(xdev, pdc, pxaNullBrush, pxtSetBrushSource);
1410
21.8M
}
1411
1412
static int
1413
pclxl_setstrokecolor(gx_device_vector * vdev, const gs_gstate * pgs,
1414
                     const gx_drawing_color * pdc)
1415
117k
{
1416
117k
    gx_device_pclxl *const xdev = (gx_device_pclxl *) vdev;
1417
1418
117k
    return pclxl_set_color(xdev, pdc, pxaNullPen, pxtSetPenSource);
1419
117k
}
1420
1421
static int
1422
pclxl_dorect(gx_device_vector * vdev, fixed x0, fixed y0, fixed x1,
1423
             fixed y1, gx_path_type_t type)
1424
37.2M
{
1425
37.2M
    gx_device_pclxl *const xdev = (gx_device_pclxl *) vdev;
1426
37.2M
    stream *s = gdev_vector_stream(vdev);
1427
1428
    /* Check for out-of-range points. */
1429
260M
#define OUT_OF_RANGE(v) (v < 0 || v >= int2fixed(0x10000))
1430
37.2M
    if (OUT_OF_RANGE(x0) || OUT_OF_RANGE(y0) ||
1431
37.1M
        OUT_OF_RANGE(x1) || OUT_OF_RANGE(y1)
1432
37.2M
        )
1433
108k
        return_error(gs_error_rangecheck);
1434
37.1M
#undef OUT_OF_RANGE
1435
37.1M
    if (type & (gx_path_type_fill | gx_path_type_stroke)) {
1436
37.0M
        pclxl_set_paints(xdev, type);
1437
37.0M
        px_put_usq_fixed(s, x0, y0, x1, y1);
1438
37.0M
        px_put_ac(s, pxaBoundingBox, pxtRectangle);
1439
37.0M
    }
1440
37.1M
    if (type & gx_path_type_clip) {
1441
50.8k
        static const byte cr_[] = {
1442
50.8k
            DA(pxaBoundingBox),
1443
50.8k
            DUB(eInterior), DA(pxaClipRegion),
1444
50.8k
            pxtSetClipRectangle
1445
50.8k
        };
1446
1447
50.8k
        px_put_usq_fixed(s, x0, y0, x1, y1);
1448
50.8k
        PX_PUT_LIT(s, cr_);
1449
50.8k
    }
1450
37.1M
    return 0;
1451
37.2M
}
1452
1453
static int
1454
pclxl_beginpath(gx_device_vector * vdev, gx_path_type_t type)
1455
7.51M
{
1456
7.51M
    gx_device_pclxl *const xdev = (gx_device_pclxl *) vdev;
1457
7.51M
    stream *s = gdev_vector_stream(vdev);
1458
1459
7.51M
    spputc(s, pxtNewPath);
1460
7.51M
    xdev->points.type = POINTS_NONE;
1461
7.51M
    xdev->points.count = 0;
1462
7.51M
    return 0;
1463
7.51M
}
1464
1465
static int
1466
pclxl_moveto(gx_device_vector * vdev, double x0, double y0, double x,
1467
             double y, gx_path_type_t type)
1468
8.26M
{
1469
8.26M
    gx_device_pclxl *const xdev = (gx_device_pclxl *) vdev;
1470
8.26M
    int code = pclxl_flush_points(xdev);
1471
1472
8.26M
    if (code < 0)
1473
0
        return code;
1474
8.26M
    return pclxl_set_cursor(xdev,
1475
8.26M
                            xdev->points.current.x = (int)(x + 0.5),
1476
8.26M
                            xdev->points.current.y = (int)(y + 0.5));
1477
8.26M
}
1478
1479
static int
1480
pclxl_lineto(gx_device_vector * vdev, double x0, double y0, double x,
1481
             double y, gx_path_type_t type)
1482
31.3M
{
1483
31.3M
    gx_device_pclxl *const xdev = (gx_device_pclxl *) vdev;
1484
1485
31.3M
    if (xdev->points.type != POINTS_LINES || xdev->points.count >= NUM_POINTS - 2) {
1486
12.2M
        if (xdev->points.type != POINTS_NONE) {
1487
5.94M
            int code = pclxl_flush_points(xdev);
1488
1489
5.94M
            if (code < 0)
1490
0
                return code;
1491
5.94M
        }
1492
12.2M
        xdev->points.current.x = (int)(x0 + 0.5);
1493
12.2M
        xdev->points.current.y = (int)(y0 + 0.5);
1494
12.2M
        xdev->points.type = POINTS_LINES;
1495
1496
        /* This can only happen if we get two 'POINTS_NONE' in a row, in which case
1497
         * just overwrite the previous one below.
1498
         */
1499
12.2M
        if (xdev->points.count >= NUM_POINTS - 1)
1500
0
            xdev->points.count--;
1501
12.2M
    }
1502
31.3M
    {
1503
31.3M
        gs_int_point *ppt = &xdev->points.data[xdev->points.count++];
1504
1505
31.3M
        ppt->x = (int)(x + 0.5), ppt->y = (int)(y + 0.5);
1506
31.3M
    }
1507
31.3M
    return 0;
1508
31.3M
}
1509
1510
static int
1511
pclxl_curveto(gx_device_vector * vdev, double x0, double y0,
1512
              double x1, double y1, double x2, double y2, double x3,
1513
              double y3, gx_path_type_t type)
1514
25.6M
{
1515
25.6M
    gx_device_pclxl *const xdev = (gx_device_pclxl *) vdev;
1516
1517
25.6M
    if (xdev->points.type != POINTS_CURVES ||
1518
18.3M
        xdev->points.count >= NUM_POINTS - 4) {
1519
7.53M
        if (xdev->points.type != POINTS_NONE) {
1520
5.57M
            int code = pclxl_flush_points(xdev);
1521
1522
5.57M
            if (code < 0)
1523
0
                return code;
1524
5.57M
        }
1525
7.53M
        xdev->points.current.x = (int)(x0 + 0.5);
1526
7.53M
        xdev->points.current.y = (int)(y0 + 0.5);
1527
7.53M
        xdev->points.type = POINTS_CURVES;
1528
1529
        /* This can only happen if we get multiple 'POINTS_NONE' in a row, in which case
1530
         * just overwrite the previous one below.
1531
         */
1532
7.53M
        if (xdev->points.count >= NUM_POINTS - 3)
1533
0
            xdev->points.count -= 3;
1534
7.53M
    }
1535
25.6M
    {
1536
25.6M
        gs_int_point *ppt = &xdev->points.data[xdev->points.count];
1537
1538
25.6M
        ppt->x = (int)(x1 + 0.5), ppt->y = (int)(y1 + 0.5), ++ppt;
1539
25.6M
        ppt->x = (int)(x2 + 0.5), ppt->y = (int)(y2 + 0.5), ++ppt;
1540
25.6M
        ppt->x = (int)(x3 + 0.5), ppt->y = (int)(y3 + 0.5);
1541
25.6M
    }
1542
25.6M
    xdev->points.count += 3;
1543
25.6M
    return 0;
1544
25.6M
}
1545
1546
static int
1547
pclxl_closepath(gx_device_vector * vdev, double x, double y,
1548
                double x_start, double y_start, gx_path_type_t type)
1549
2.25M
{
1550
2.25M
    gx_device_pclxl *const xdev = (gx_device_pclxl *) vdev;
1551
2.25M
    stream *s = gdev_vector_stream(vdev);
1552
2.25M
    int code = pclxl_flush_points(xdev);
1553
1554
2.25M
    if (code < 0)
1555
0
        return code;
1556
2.25M
    spputc(s, pxtCloseSubPath);
1557
2.25M
    xdev->points.current.x = (int)(x_start + 0.5);
1558
2.25M
    xdev->points.current.y = (int)(y_start + 0.5);
1559
2.25M
    return 0;
1560
2.25M
}
1561
1562
static int
1563
pclxl_endpath(gx_device_vector * vdev, gx_path_type_t type)
1564
7.51M
{
1565
7.51M
    gx_device_pclxl *const xdev = (gx_device_pclxl *) vdev;
1566
7.51M
    stream *s = gdev_vector_stream(vdev);
1567
7.51M
    int code = pclxl_flush_points(xdev);
1568
7.51M
    gx_path_type_t rule = type & gx_path_type_rule;
1569
1570
7.51M
    if (code < 0)
1571
0
        return code;
1572
7.51M
    if (type & (gx_path_type_fill | gx_path_type_stroke)) {
1573
7.00M
        if (rule != xdev->fill_rule) {
1574
4.25k
            px_put_ub(s, (byte) (rule == gx_path_type_even_odd ? eEvenOdd :
1575
4.25k
                                 eNonZeroWinding));
1576
4.25k
            px_put_ac(s, pxaFillMode, pxtSetFillMode);
1577
4.25k
            xdev->fill_rule = rule;
1578
4.25k
        }
1579
7.00M
        pclxl_set_paints(xdev, type);
1580
7.00M
        spputc(s, pxtPaintPath);
1581
7.00M
    }
1582
7.51M
    if (type & gx_path_type_clip) {
1583
505k
        static const byte scr_[] = {
1584
505k
            DUB(eInterior), DA(pxaClipRegion), pxtSetClipReplace
1585
505k
        };
1586
1587
505k
        if (rule != xdev->clip_rule) {
1588
378
            px_put_ub(s, (byte) (rule == gx_path_type_even_odd ? eEvenOdd :
1589
378
                                 eNonZeroWinding));
1590
378
            px_put_ac(s, pxaClipMode, pxtSetClipMode);
1591
378
            xdev->clip_rule = rule;
1592
378
        }
1593
505k
        PX_PUT_LIT(s, scr_);
1594
505k
    }
1595
7.51M
    return 0;
1596
7.51M
}
1597
1598
/* Vector implementation procedures */
1599
1600
static const gx_device_vector_procs pclxl_vector_procs = {
1601
    /* Page management */
1602
    pclxl_beginpage,
1603
    /* Imager state */
1604
    pclxl_setlinewidth,
1605
    pclxl_setlinecap,
1606
    pclxl_setlinejoin,
1607
    pclxl_setmiterlimit,
1608
    pclxl_setdash,
1609
    gdev_vector_setflat,
1610
    pclxl_setlogop,
1611
    /* Other state */
1612
    pclxl_can_handle_hl_color,
1613
    pclxl_setfillcolor,
1614
    pclxl_setstrokecolor,
1615
    /* Paths */
1616
    gdev_vector_dopath,
1617
    pclxl_dorect,
1618
    pclxl_beginpath,
1619
    pclxl_moveto,
1620
    pclxl_lineto,
1621
    pclxl_curveto,
1622
    pclxl_closepath,
1623
    pclxl_endpath
1624
};
1625
1626
/* ---------------- Driver procedures ---------------- */
1627
1628
/* ------ Open/close/page ------ */
1629
1630
/* Open the device. */
1631
static int
1632
pclxl_open_device(gx_device * dev)
1633
25.2k
{
1634
25.2k
    gx_device_vector *vdev = (gx_device_vector *) dev;
1635
25.2k
    gx_device_pclxl *xdev = (gx_device_pclxl *) dev;
1636
25.2k
    int code;
1637
1638
25.2k
    vdev->v_memory = dev->memory->stable_memory;
1639
25.2k
    vdev->vec_procs = &pclxl_vector_procs;
1640
25.2k
    code = gdev_vector_open_file_options(vdev, 512,
1641
25.2k
                                         VECTOR_OPEN_FILE_SEQUENTIAL);
1642
25.2k
    if (code < 0)
1643
0
        return code;
1644
1645
25.2k
    while (dev->child)
1646
0
        dev = dev->child;
1647
25.2k
    vdev = (gx_device_vector *) dev;
1648
25.2k
    xdev = (gx_device_pclxl *) dev;
1649
1650
25.2k
    pclxl_page_init(xdev);
1651
25.2k
    px_write_file_header(vdev->strm, dev, xdev->Staple);
1652
25.2k
    xdev->media_size = pxeMediaSize_next;       /* no size selected */
1653
25.2k
    memset(&xdev->chars, 0, sizeof(xdev->chars));
1654
25.2k
    xdev->chars.next_in = xdev->chars.next_out = 2;
1655
    /* xdev->iccTransform = false; *//* set true/false here to ignore command line */
1656
25.2k
    return 0;
1657
25.2k
}
1658
1659
/* Wrap up ("output") a page. */
1660
/* We only support flush = true */
1661
static int
1662
pclxl_output_page(gx_device * dev, int num_copies, int flush)
1663
20.1k
{
1664
20.1k
    gx_device_pclxl *const xdev = (gx_device_pclxl *) dev;
1665
20.1k
    stream *s;
1666
20.1k
    int code;
1667
1668
    /* Note that unlike close_device, end_page must not omit blank pages. */
1669
20.1k
    if (!xdev->in_page)
1670
0
        pclxl_beginpage((gx_device_vector *) dev);
1671
20.1k
    s = xdev->strm;
1672
20.1k
    px_put_usa(s, (uint) num_copies, pxaPageCopies);    /* num_copies */
1673
20.1k
    spputc(s, pxtEndPage);
1674
20.1k
    sflush(s);
1675
20.1k
    pclxl_page_init(xdev);
1676
20.1k
    if (gp_ferror(xdev->file))
1677
0
        return_error(gs_error_ioerror);
1678
20.1k
    if ((code = gx_finish_output_page(dev, num_copies, flush)) < 0)
1679
0
        return code;
1680
    /* Check if we need to change the output file for separate pages */
1681
20.1k
    if (gx_outputfile_is_separate_pages
1682
20.1k
        (((gx_device_vector *) dev)->fname, dev->memory)) {
1683
0
        if ((code = pclxl_close_device(dev)) < 0)
1684
0
            return code;
1685
0
        code = pclxl_open_device(dev);
1686
0
    }
1687
20.1k
    return code;
1688
20.1k
}
1689
1690
/* Close the device. */
1691
/* Note that if this is being called as a result of finalization, */
1692
/* the stream may no longer exist. */
1693
static int
1694
pclxl_close_device(gx_device * dev)
1695
25.2k
{
1696
25.2k
    gx_device_pclxl *const xdev = (gx_device_pclxl *) dev;
1697
25.2k
    gp_file *file = xdev->file;
1698
1699
25.2k
    if (xdev->strm != NULL)
1700
25.2k
        sflush(xdev->strm);
1701
25.2k
    if (xdev->in_page)
1702
1.43k
        gp_fputc(pxtEndPage, file);
1703
25.2k
    px_write_file_trailer(file);
1704
25.2k
    return gdev_vector_close_file((gx_device_vector *) dev);
1705
25.2k
}
1706
1707
/* ------ One-for-one images ------ */
1708
1709
static const byte eBit_values[] = {
1710
    0, e1Bit, 0, 0, e4Bit, 0, 0, 0, e8Bit
1711
};
1712
1713
/* Copy a monochrome bitmap. */
1714
static int
1715
pclxl_copy_mono(gx_device * dev, const byte * data, int data_x, int raster,
1716
                gx_bitmap_id id, int x, int y, int w, int h,
1717
                gx_color_index zero, gx_color_index one)
1718
128k
{
1719
128k
    gx_device_vector *const vdev = (gx_device_vector *) dev;
1720
128k
    gx_device_pclxl *const xdev = (gx_device_pclxl *) dev;
1721
128k
    int code;
1722
128k
    stream *s;
1723
128k
    gx_color_index color0 = zero, color1 = one;
1724
128k
    gx_color_index white = ((gx_color_index)1 << dev->color_info.depth) - 1;
1725
128k
    gx_color_index black = 0;
1726
128k
    gs_logical_operation_t lop;
1727
128k
    byte palette[2 * 3];
1728
128k
    int palette_size;
1729
128k
    pxeColorSpace_t color_space;
1730
1731
128k
    fit_copy(dev, data, data_x, raster, id, x, y, w, h);
1732
126k
    code = gdev_vector_update_clip_path(vdev, NULL);
1733
126k
    if (code < 0)
1734
0
        return code;
1735
1736
    /* write_image_data() needs byte-alignment,
1737
     * and gx_default_copy_* is more efficient than pxlcl_*
1738
     * for small rasters. See details in copy_color().
1739
     */
1740
126k
    if (((data_x & 7) != 0) || (h == 1) || (w == 1))
1741
121k
        return gx_default_copy_mono(dev, data, data_x, raster, id,
1742
121k
                                    x, y, w, h, zero, one);
1743
1744
4.35k
    pclxl_set_cursor(xdev, x, y);
1745
4.35k
    if (id != gs_no_id && zero == gx_no_color_index &&
1746
2.15k
        one != gx_no_color_index && data_x == 0) {
1747
2.15k
        gx_drawing_color dcolor;
1748
1749
2.15k
        code = gdev_vector_update_log_op(vdev, rop3_T | lop_T_transparent);
1750
2.15k
        if (code < 0)
1751
0
            return 0;
1752
1753
2.15k
        set_nonclient_dev_color(&dcolor, one);
1754
2.15k
        pclxl_setfillcolor(vdev, NULL, &dcolor);
1755
2.15k
        if (pclxl_copy_text_char(xdev, data, raster, id, w, h) >= 0)
1756
2.15k
            return 0;
1757
2.15k
    }
1758
    /*
1759
     * The following doesn't work if we're writing white with a mask.
1760
     * We'll fix it eventually.
1761
     *
1762
     * The logic goes like this: non-white + mask (transparent)
1763
     * works by setting the mask color to white and also declaring
1764
     * white-is-transparent. This doesn't work for drawing white + mask,
1765
     * since everything is then white+white-and-transparent. So instead
1766
     * we set mask color to black, invert and draw the destination/background
1767
     * through it, as well as drawing the white color.
1768
     *
1769
     * In rop3 terms, this is (D & ~S) | S
1770
     *
1771
     * This also only works in the case of the drawing color is white,
1772
     * because we need the inversion to not draw anything, (especially
1773
     * not the complimentary color/shade). So we have two different code paths,
1774
     * white + mask and non-whites + mask.
1775
     *
1776
     * There is a further refinement to this algorithm - it appears that
1777
     * black+mask is treated specially by the vector driver core (rendered
1778
     * as transparent on white), and does not work as non-white + mask.
1779
     * So Instead we set mask color to white and do (S & D) (i.e. draw
1780
     * background on mask, instead of transparent on mask).
1781
     *
1782
     */
1783
2.20k
    if (zero == gx_no_color_index) {
1784
1.86k
        if (one == gx_no_color_index)
1785
0
            return 0;
1786
1.86k
        if (one != white) {
1787
1.80k
            if (one == black) {
1788
936
                lop = (rop3_S & rop3_D);
1789
936
            } else {
1790
873
                lop = rop3_S | lop_S_transparent;
1791
873
            }
1792
1.80k
            color0 = white;
1793
1.80k
        } else {
1794
60
            lop = rop3_S | (rop3_D & rop3_not(rop3_S));
1795
60
            color0 = black;
1796
60
        }
1797
1.86k
    } else if (one == gx_no_color_index) {
1798
0
        if (zero != white) {
1799
0
            if (zero == black) {
1800
0
                lop = (rop3_S & rop3_D);
1801
0
            } else {
1802
0
                lop = rop3_S | lop_S_transparent;
1803
0
            }
1804
0
            color1 = white;
1805
0
        } else {
1806
0
            lop = rop3_S | (rop3_D & rop3_not(rop3_S));
1807
0
            color1 = black;
1808
0
        }
1809
334
    } else {
1810
334
        lop = rop3_S;
1811
334
    }
1812
1813
2.20k
    if (dev->color_info.num_components == 1 ||
1814
1.41k
        (RGB_IS_GRAY(color0) && RGB_IS_GRAY(color1))
1815
2.20k
        ) {
1816
1.59k
        palette[0] = (byte) color0;
1817
1.59k
        palette[1] = (byte) color1;
1818
1.59k
        palette_size = 2;
1819
1.59k
        color_space = eGray;
1820
1.59k
        if_debug2m('b', dev->memory, "color palette %02X %02X\n",
1821
1.59k
                   palette[0], palette[1]);
1822
1.59k
    } else {
1823
605
        palette[0] = (byte) (color0 >> 16);
1824
605
        palette[1] = (byte) (color0 >> 8);
1825
605
        palette[2] = (byte) color0;
1826
605
        palette[3] = (byte) (color1 >> 16);
1827
605
        palette[4] = (byte) (color1 >> 8);
1828
605
        palette[5] = (byte) color1;
1829
605
        palette_size = 6;
1830
605
        color_space = eRGB;
1831
605
    }
1832
2.20k
    code = gdev_vector_update_log_op(vdev, lop);
1833
2.20k
    if (code < 0)
1834
0
        return 0;
1835
2.20k
    pclxl_set_color_palette(xdev, color_space, palette, palette_size);
1836
2.20k
    s = pclxl_stream(xdev);
1837
2.20k
    {
1838
2.20k
        static const byte mi_[] = {
1839
2.20k
            DUB(e1Bit), DA(pxaColorDepth),
1840
2.20k
            DUB(eIndexedPixel), DA(pxaColorMapping)
1841
2.20k
        };
1842
1843
2.20k
        PX_PUT_LIT(s, mi_);
1844
2.20k
    }
1845
2.20k
    pclxl_write_begin_image(xdev, w, h, w, h);
1846
2.20k
    pclxl_write_image_data(xdev, data, data_x, raster, w, 0, h, false);
1847
2.20k
    pclxl_write_end_image(xdev);
1848
2.20k
    return 0;
1849
2.20k
}
1850
1851
/* Copy a color bitmap. */
1852
static int
1853
pclxl_copy_color(gx_device * dev,
1854
                 const byte * base, int sourcex, int raster, gx_bitmap_id id,
1855
                 int x, int y, int w, int h)
1856
768k
{
1857
768k
    gx_device_vector *const vdev = (gx_device_vector *) dev;
1858
768k
    gx_device_pclxl *const xdev = (gx_device_pclxl *) dev;
1859
768k
    stream *s;
1860
768k
    uint source_bit;
1861
768k
    int code;
1862
1863
768k
    fit_copy(dev, base, sourcex, raster, id, x, y, w, h);
1864
768k
    code = gdev_vector_update_clip_path(vdev, NULL);
1865
768k
    if (code < 0)
1866
0
        return code;
1867
1868
768k
    source_bit = sourcex * dev->color_info.depth;
1869
1870
    /* side-effect from fill/stroke may have set color space to eGray */
1871
768k
    if (dev->color_info.num_components == 3)
1872
274k
        pclxl_set_color_space(xdev, eRGB);
1873
494k
    else if (dev->color_info.num_components == 1)
1874
494k
        pclxl_set_color_space(xdev, eGray);
1875
1876
    /* write_image_data() needs byte-alignment,
1877
     * and gx_default_copy_* is more efficient than pxlcl_*
1878
     * for small rasters.
1879
     *
1880
     * SetBrushSource + Rectangle = 21 byte for 1x1 RGB
1881
     * SetCursor+ BeginImage + ReadImage + EndImage = 56 bytes for 1x1 RGB
1882
     * 3x1 RGB at 3 different colors takes 62 bytes for pxlcl_*
1883
     * but gx_default_copy_* uses 63 bytes. Below 3 pixels, gx_default_copy_*
1884
     * is better than pxlcl_*; above 3 pixels, it is less clear;
1885
     * in practice, single-lines seems better coded as painted rectangles
1886
     * than images.
1887
     */
1888
768k
    if (((source_bit & 7) != 0) || (w == 1) || (h == 1))
1889
756k
        return gx_default_copy_color(dev, base, sourcex, raster, id,
1890
756k
                                     x, y, w, h);
1891
12.5k
    code = gdev_vector_update_log_op(vdev, rop3_S);
1892
12.5k
    if (code < 0)
1893
0
        return 0;
1894
12.5k
    pclxl_set_cursor(xdev, x, y);
1895
12.5k
    s = pclxl_stream(xdev);
1896
12.5k
    {
1897
12.5k
        static const byte ci_[] = {
1898
12.5k
            DA(pxaColorDepth),
1899
12.5k
            DUB(eDirectPixel), DA(pxaColorMapping)
1900
12.5k
        };
1901
1902
12.5k
        px_put_ub(s, eBit_values[dev->color_info.depth /
1903
12.5k
                                 dev->color_info.num_components]);
1904
12.5k
        PX_PUT_LIT(s, ci_);
1905
12.5k
    }
1906
12.5k
    pclxl_write_begin_image(xdev, w, h, w, h);
1907
12.5k
    pclxl_write_image_data(xdev, base, source_bit, raster,
1908
12.5k
                           w * dev->color_info.depth, 0, h, false);
1909
12.5k
    pclxl_write_end_image(xdev);
1910
12.5k
    return 0;
1911
12.5k
}
1912
1913
/* Fill a mask. */
1914
static int
1915
pclxl_fill_mask(gx_device * dev,
1916
                const byte * data, int data_x, int raster, gx_bitmap_id id,
1917
                int x, int y, int w, int h,
1918
                const gx_drawing_color * pdcolor, int depth,
1919
                gs_logical_operation_t lop, const gx_clip_path * pcpath)
1920
50.2k
{
1921
50.2k
    gx_device_vector *const vdev = (gx_device_vector *) dev;
1922
50.2k
    gx_device_pclxl *const xdev = (gx_device_pclxl *) dev;
1923
50.2k
    int code;
1924
50.2k
    stream *s;
1925
50.2k
    gx_color_index foreground;
1926
1927
50.2k
    fit_copy(dev, data, data_x, raster, id, x, y, w, h);
1928
    /* write_image_data() needs byte-alignment,
1929
     * and gx_default_copy_* is more efficient than pxlcl_*
1930
     * for small rasters. See details in copy_color().
1931
     */
1932
15.0k
    if ((data_x & 7) != 0 || !gx_dc_is_pure(pdcolor) || depth > 1 || (w == 1)
1933
15.0k
        || (h == 1))
1934
4
        return gx_default_fill_mask(dev, data, data_x, raster, id, x, y, w, h,
1935
4
                                    pdcolor, depth, lop, pcpath);
1936
15.0k
    code = gdev_vector_update_clip_path(vdev, pcpath);
1937
15.0k
    foreground = gx_dc_pure_color(pdcolor);
1938
15.0k
    if (code < 0)
1939
0
        return code;
1940
15.0k
    code = gdev_vector_update_fill_color(vdev, NULL, pdcolor);
1941
15.0k
    if (code < 0)
1942
0
        return 0;
1943
15.0k
    pclxl_set_cursor(xdev, x, y);
1944
15.0k
    if (id != gs_no_id && data_x == 0) {
1945
15.0k
        code = gdev_vector_update_log_op(vdev, lop);
1946
15.0k
        if (code < 0)
1947
0
            return 0;
1948
15.0k
        if (pclxl_copy_text_char(xdev, data, raster, id, w, h) >= 0)
1949
15.0k
            return 0;
1950
15.0k
    }
1951
    /* This is similiar to the copy_mono white-on-mask,
1952
     * except we are drawing white on the black of a black/white mask,
1953
     * so we invert source, compared to copy_mono */
1954
0
    if (foreground == ((gx_color_index)1 << dev->color_info.depth) - 1) {       /* white */
1955
0
        lop = rop3_not(rop3_S) | (rop3_D & rop3_S);
1956
0
    } else if (foreground == 0) {       /* black */
1957
0
        lop = (rop3_S & rop3_D);
1958
0
    } else
1959
0
        lop |= rop3_S | lop_S_transparent;
1960
1961
0
    code = gdev_vector_update_log_op(vdev, lop);
1962
0
    if (code < 0)
1963
0
        return 0;
1964
0
    pclxl_set_color_palette(xdev, eGray, (const byte *)"\377\000", 2);
1965
0
    s = pclxl_stream(xdev);
1966
0
    {
1967
0
        static const byte mi_[] = {
1968
0
            DUB(e1Bit), DA(pxaColorDepth),
1969
0
            DUB(eIndexedPixel), DA(pxaColorMapping)
1970
0
        };
1971
1972
0
        PX_PUT_LIT(s, mi_);
1973
0
    }
1974
0
    pclxl_write_begin_image(xdev, w, h, w, h);
1975
0
    pclxl_write_image_data(xdev, data, data_x, raster, w, 0, h, false);
1976
0
    pclxl_write_end_image(xdev);
1977
0
    return 0;
1978
0
}
1979
1980
/* Do a RasterOp. */
1981
static int
1982
pclxl_strip_copy_rop2(gx_device * dev, const byte * sdata, int sourcex,
1983
                     uint sraster, gx_bitmap_id id,
1984
                     const gx_color_index * scolors,
1985
                     const gx_strip_bitmap * textures,
1986
                     const gx_color_index * tcolors,
1987
                     int x, int y, int width, int height,
1988
                     int phase_x, int phase_y, gs_logical_operation_t lop,
1989
                     uint plane_height)
1990
0
{
1991
0
    lop = lop_sanitize(lop);
1992
    /* Improvements possible here using PXL ROP3
1993
       for some combinations of args; use gx_default for now */
1994
0
    if (!rop3_uses_D(lop))  /* gx_default() cannot cope with D ops */
1995
0
        return gx_default_strip_copy_rop2(dev, sdata, sourcex,
1996
0
                                          sraster, id,
1997
0
                                          scolors,
1998
0
                                          textures,
1999
0
                                          tcolors,
2000
0
                                          x, y, width, height,
2001
0
                                          phase_x, phase_y, lop,
2002
0
                                          plane_height);
2003
0
    return 0;
2004
0
}
2005
2006
/* ------ High-level images ------ */
2007
2008
168k
#define MAX_ROW_DATA 500000     /* arbitrary */
2009
typedef struct pclxl_image_enum_s {
2010
    gdev_vector_image_enum_common;
2011
    gs_matrix mat;
2012
    struct ir_ {
2013
        byte *data;
2014
        int num_rows;           /* # of allocated rows */
2015
        int first_y;
2016
        uint raster;
2017
    } rows;
2018
    bool flipped;
2019
    gsicc_link_t *icclink;
2020
} pclxl_image_enum_t;
2021
2022
gs_private_st_suffix_add2(st_pclxl_image_enum, pclxl_image_enum_t,
2023
                          "pclxl_image_enum_t", pclxl_image_enum_enum_ptrs,
2024
                          pclxl_image_enum_reloc_ptrs, st_vector_image_enum,
2025
                          rows.data, icclink);
2026
2027
/* Start processing an image. */
2028
static int
2029
pclxl_begin_typed_image(gx_device * dev,
2030
                  const gs_gstate * pgs,
2031
                  const gs_matrix *pmat,
2032
                  const gs_image_common_t * pic,
2033
                  const gs_int_rect * prect,
2034
                  const gx_drawing_color * pdcolor,
2035
                  const gx_clip_path * pcpath, gs_memory_t * mem,
2036
                  gx_image_enum_common_t ** pinfo)
2037
234k
{
2038
234k
    gx_device_vector *const vdev = (gx_device_vector *) dev;
2039
234k
    gx_device_pclxl *const xdev = (gx_device_pclxl *) dev;
2040
234k
    const gs_image_t *pim = (const gs_image_t *)pic;
2041
234k
    const gs_color_space *pcs;
2042
234k
    pclxl_image_enum_t *pie;
2043
234k
    byte *row_data;
2044
234k
    int num_rows;
2045
234k
    uint row_raster;
2046
234k
    int bits_per_pixel;
2047
234k
    gs_matrix mat;
2048
234k
    int code;
2049
2050
    /* We only cope with image type 1 here. */
2051
234k
    if (pic->type->index != 1)
2052
97
        goto use_default;
2053
2054
234k
    pcs = pim->ColorSpace;
2055
    /*
2056
     * Following should divide by num_planes, but we only handle chunky
2057
     * images, i.e., num_planes = 1.
2058
     */
2059
234k
    bits_per_pixel =
2060
234k
        (pim->ImageMask ? 1 :
2061
234k
         pim->BitsPerComponent * gs_color_space_num_components(pcs));
2062
2063
    /*
2064
     * Check whether we can handle this image.  PCL XL 1.0 and 2.0 only
2065
     * handle orthogonal transformations.
2066
     */
2067
234k
    code = gs_matrix_invert(&pim->ImageMatrix, &mat);
2068
234k
    if (code < 0)
2069
4
        goto use_default;
2070
234k
    if (pmat == NULL)
2071
234k
        pmat = &ctm_only(pgs);
2072
234k
    gs_matrix_multiply(&mat, pmat, &mat);
2073
2074
234k
    if (pclxl_nontrivial_transfer(pgs))
2075
11
        goto use_default;
2076
2077
234k
    if (pim->Width == 0 || pim->Height == 0)
2078
16
        goto use_default;
2079
2080
234k
    if (bits_per_pixel == 32) {
2081
        /*
2082
           32-bit cmyk depends on transformable to 24-bit rgb.
2083
           Some 32-bit aren't cmyk's. (e.g. DeviceN)
2084
         */
2085
216
        if (!pclxl_can_icctransform(pim))
2086
0
            goto use_default;
2087
2088
        /*
2089
           Strictly speaking, regardless of bits_per_pixel,
2090
           we cannot handle non-Identity Decode array.
2091
           Historically we did (wrongly), so this is inside
2092
           a 32-bit conditional to avoid regression, but shouldn't.
2093
         */
2094
216
        if (pim->Decode[0] != 0 || pim->Decode[1] != 1 ||
2095
216
            pim->Decode[2] != 0 || pim->Decode[3] != 1 ||
2096
216
            pim->Decode[4] != 0 || pim->Decode[5] != 1)
2097
0
            goto use_default;
2098
216
    }
2099
2100
    /*
2101
     * NOTE: this predicate should be fixed to be readable and easily
2102
     * debugged.  Each condition should be separate.  See the large
2103
     * similar conditional in clist_begin_typed_image which has
2104
     * already been reworked.  We can handle rotations of 90 degs +
2105
     * scaling + reflections.  * These have one of the diagonals being
2106
     * zeros * (and the other diagonals having non-zeros).
2107
     */
2108
234k
    if ((!((mat.xx * mat.yy != 0) && (mat.xy == 0) && (mat.yx == 0)) &&
2109
85.8k
         !((mat.xx == 0) && (mat.yy == 0) && (mat.xy * mat.yx != 0))) ||
2110
169k
        (pim->ImageMask ?
2111
164k
         (!gx_dc_is_pure(pdcolor) || pim->CombineWithColor) :
2112
169k
         ((!pclxl_can_handle_color_space(pcs) ||
2113
4.64k
           (bits_per_pixel != 1 && bits_per_pixel != 4 &&
2114
4.38k
            bits_per_pixel != 8 && bits_per_pixel != 24 &&
2115
255
            bits_per_pixel != 32))
2116
1.31k
          && !(pclxl_can_icctransform(pim) && xdev->iccTransform))) ||
2117
168k
        pim->format != gs_image_format_chunky || pim->Interpolate || prect)
2118
66.2k
        goto use_default;
2119
168k
    row_raster = (bits_per_pixel * pim->Width + 7) >> 3;
2120
168k
    num_rows = MAX_ROW_DATA / row_raster;
2121
168k
    if (num_rows > pim->Height)
2122
166k
        num_rows = pim->Height;
2123
168k
    if (num_rows <= 0)
2124
2
        num_rows = 1;
2125
168k
    pie = gs_alloc_struct(mem, pclxl_image_enum_t, &st_pclxl_image_enum,
2126
168k
                          "pclxl_begin_image");
2127
168k
    row_data = gs_alloc_bytes(mem, (size_t)num_rows * row_raster,
2128
168k
                              "pclxl_begin_image(rows)");
2129
168k
    if (pie == 0 || row_data == 0) {
2130
1
        code = gs_note_error(gs_error_VMerror);
2131
1
        goto fail;
2132
1
    }
2133
168k
    code = gdev_vector_begin_image(vdev, pgs, pim, pim->format, prect,
2134
168k
                                   pdcolor, pcpath, mem,
2135
168k
                                   &pclxl_image_enum_procs,
2136
168k
                                   (gdev_vector_image_enum_t *) pie);
2137
168k
    if (code < 0)
2138
0
        goto fail;
2139
2140
    /* emit a PXL XL rotation and adjust mat correspondingly */
2141
168k
    pie->flipped = false;
2142
168k
    if (mat.xx * mat.yy > 0) {
2143
147k
        if (mat.xx < 0) {
2144
6.77k
            stream *s = pclxl_stream(xdev);
2145
2146
6.77k
            mat.xx = -mat.xx;
2147
6.77k
            mat.yy = -mat.yy;
2148
6.77k
            mat.tx = -mat.tx;
2149
6.77k
            mat.ty = -mat.ty;
2150
6.77k
            px_put_ss(s, 180);
2151
6.77k
            xdev->state_rotated = 2;
2152
6.77k
            px_put_ac(s, pxaPageAngle, pxtSetPageRotation);
2153
6.77k
        }
2154
        /* leave the matrix alone if it is portrait */
2155
147k
    } else if (mat.xx * mat.yy < 0) {
2156
277
        pie->flipped = true;
2157
277
        if (mat.xx < 0) {
2158
15
            stream *s = pclxl_stream(xdev);
2159
2160
15
            mat.xx = -mat.xx;
2161
15
            mat.tx = -mat.tx;
2162
15
            px_put_ss(s, +180);
2163
15
            xdev->state_rotated = +2;
2164
15
            px_put_ac(s, pxaPageAngle, pxtSetPageRotation);
2165
262
        } else {
2166
262
            mat.yy = -mat.yy;
2167
262
            mat.ty = -mat.ty;
2168
262
        }
2169
21.1k
    } else if (mat.xy * mat.yx < 0) {
2170
        /* rotate +90 or -90 */
2171
21.0k
        float tmpf;
2172
21.0k
        stream *s = pclxl_stream(xdev);
2173
2174
21.0k
        if (mat.xy > 0) {
2175
33
            mat.xx = mat.xy;
2176
33
            mat.yy = -mat.yx;
2177
33
            tmpf = mat.tx;
2178
33
            mat.tx = mat.ty;
2179
33
            mat.ty = -tmpf;
2180
33
            px_put_ss(s, -90);
2181
33
            xdev->state_rotated = -1;
2182
21.0k
        } else {
2183
21.0k
            mat.xx = -mat.xy;
2184
21.0k
            mat.yy = mat.yx;
2185
21.0k
            tmpf = mat.tx;
2186
21.0k
            mat.tx = -mat.ty;
2187
21.0k
            mat.ty = tmpf;
2188
21.0k
            px_put_ss(s, +90);
2189
21.0k
            xdev->state_rotated = +1;
2190
21.0k
        }
2191
21.0k
        mat.xy = mat.yx = 0;
2192
21.0k
        px_put_ac(s, pxaPageAngle, pxtSetPageRotation);
2193
21.0k
    } else if (mat.xy * mat.yx > 0) {
2194
55
        float tmpf;
2195
55
        stream *s = pclxl_stream(xdev);
2196
2197
55
        pie->flipped = true;
2198
55
        if (mat.xy > 0) {
2199
40
            mat.xx = mat.xy;
2200
40
            mat.yy = mat.yx;
2201
40
            tmpf = mat.tx;
2202
40
            mat.tx = mat.ty;
2203
40
            mat.ty = tmpf;
2204
40
            px_put_ss(s, -90);
2205
40
            xdev->state_rotated = -1;
2206
40
        } else {
2207
15
            mat.xx = -mat.xy;
2208
15
            mat.yy = -mat.yx;
2209
15
            tmpf = mat.tx;
2210
15
            mat.tx = -mat.ty;
2211
15
            mat.ty = -tmpf;
2212
15
            px_put_ss(s, +90);
2213
15
            xdev->state_rotated = +1;
2214
15
        }
2215
55
        mat.xy = mat.yx = 0;
2216
55
        px_put_ac(s, pxaPageAngle, pxtSetPageRotation);
2217
55
    }
2218
2219
168k
    pie->mat = mat;
2220
168k
    pie->rows.data = row_data;
2221
168k
    pie->rows.num_rows = num_rows;
2222
168k
    pie->rows.first_y = 0;
2223
168k
    pie->rows.raster = row_raster;
2224
    /*
2225
       pclxl_can_handle_color_space() is here to avoid regression,
2226
       to avoid icc if traditional code path works (somewhat).
2227
       It is flawed in that it claims to handles device colors with
2228
       icc profiles (and output device colours); so that "can transform"
2229
       won't transform; we need to override it for the case of
2230
       32-bit colour, except when usefastcolor is specified.
2231
2232
       "can_icctransform" should have been accompanied by a
2233
       "will tranform" (xdev->iccTransform) check. However, since
2234
       we want to transform for CMYK regardless, so just as well
2235
       there was not such a check, and we are not adding one now.
2236
2237
       In other words, the bizarre logic in the middle is to keep
2238
       the current workflow as much as possible, but force icc mode
2239
       or confirm fastcolor mode only for 32-bit image, regardless
2240
       of whether xdev->iccTransform is on.
2241
2242
       Whole-sale icc mode is to simply remove entire middle part,
2243
       a risky change.
2244
     */
2245
168k
    if (!pim->ImageMask && (!pclxl_can_handle_color_space(pcs)
2246
4.59k
                            || (bits_per_pixel == 32 && dev->icc_struct
2247
215
                                && !dev->icc_struct->usefastcolor))
2248
215
        && pclxl_can_icctransform(pim) && pcs->cmm_icc_profile_data) {
2249
215
        gsicc_rendering_param_t rendering_params;
2250
2251
215
        rendering_params.black_point_comp = pgs->blackptcomp;
2252
215
        rendering_params.graphics_type_tag = GS_IMAGE_TAG;
2253
215
        rendering_params.rendering_intent = pgs->renderingintent;
2254
215
        pie->icclink = gsicc_get_link(pgs, dev, pcs, NULL /*des */ ,
2255
215
                                      &rendering_params, pgs->memory);
2256
215
    } else
2257
168k
        pie->icclink = NULL;
2258
168k
    *pinfo = (gx_image_enum_common_t *) pie;
2259
168k
    {
2260
168k
        gs_logical_operation_t lop = pgs->log_op;
2261
2262
168k
        if (pim->ImageMask) {
2263
163k
            const byte *palette = (const byte *)
2264
163k
                (pim->Decode[0] ? "\377\000" : "\000\377");
2265
163k
            gx_color_index foreground = gx_dc_pure_color(pdcolor);
2266
2267
163k
            code = gdev_vector_update_fill_color(vdev, NULL,    /* use process color */
2268
163k
                                                 pdcolor);
2269
163k
            if (code < 0)
2270
0
                goto fail;
2271
            /* This is similiar to the copy_mono white-on-mask,
2272
             * except we are drawing white on the black of a black/white mask,
2273
             * so we invert source, compared to copy_mono */
2274
163k
            if (foreground == ((gx_color_index)1 << dev->color_info.depth) - 1) {       /* white */
2275
296
                lop = rop3_not(rop3_S) | (rop3_D & rop3_S);
2276
163k
            } else if (foreground == 0) {       /* black */
2277
160k
                lop = (rop3_S & rop3_D);
2278
160k
            } else
2279
2.85k
                lop |= rop3_S | lop_S_transparent;
2280
2281
163k
            code = gdev_vector_update_log_op(vdev, lop);
2282
163k
            if (code < 0)
2283
0
                goto fail;
2284
163k
            pclxl_set_color_palette(xdev, eGray, palette, 2);
2285
163k
        } else {
2286
4.59k
            if (bits_per_pixel == 24 || bits_per_pixel == 32) {
2287
3.18k
                code = gdev_vector_update_log_op
2288
3.18k
                    (vdev,
2289
3.18k
                     (pim->CombineWithColor ? lop : rop3_know_T_0(lop)));
2290
3.18k
                if (code < 0)
2291
0
                    goto fail;
2292
3.18k
                if (dev->color_info.num_components == 1) {
2293
1.13k
                    pclxl_set_color_space(xdev, eGray);
2294
2.04k
                } else {
2295
2.04k
                    pclxl_set_color_space(xdev, eRGB);
2296
2.04k
                }
2297
3.18k
            } else {
2298
1.41k
                int bpc = pim->BitsPerComponent;
2299
1.41k
                int num_components =
2300
1.41k
                    pie->plane_depths[0] * pie->num_planes / bpc;
2301
1.41k
                int sample_max = (1 << bpc) - 1;
2302
1.41k
                byte palette[256 * 3];
2303
1.41k
                int i;
2304
2305
1.41k
                code = gdev_vector_update_log_op
2306
1.41k
                    (vdev,
2307
1.41k
                     (pim->CombineWithColor ? lop : rop3_know_T_0(lop)));
2308
1.41k
                if (code < 0)
2309
0
                    goto fail;
2310
298k
                for (i = 0; i < 1 << bits_per_pixel; ++i) {
2311
297k
                    gs_client_color cc;
2312
297k
                    gx_device_color devc;
2313
297k
                    int cv = i, j;
2314
297k
                    gx_color_index ci;
2315
2316
594k
                    for (j = num_components - 1; j >= 0; cv >>= bpc, --j)
2317
297k
                        cc.paint.values[j] = pim->Decode[j * 2] +
2318
297k
                            (cv & sample_max) *
2319
297k
                            (pim->Decode[j * 2 + 1] - pim->Decode[j * 2]) /
2320
297k
                            sample_max;
2321
297k
                    (*pcs->type->remap_color)
2322
297k
                        (&cc, pcs, &devc, pgs, dev, gs_color_select_source);
2323
297k
                    if (!gx_dc_is_pure(&devc))
2324
0
                        return_error(gs_error_Fatal);
2325
297k
                    ci = gx_dc_pure_color(&devc);
2326
297k
                    if (dev->color_info.num_components == 1) {
2327
273k
                        palette[i] = (byte) ci;
2328
273k
                    } else {
2329
24.0k
                        byte *ppal = &palette[i * 3];
2330
2331
24.0k
                        ppal[0] = (byte) (ci >> 16);
2332
24.0k
                        ppal[1] = (byte) (ci >> 8);
2333
24.0k
                        ppal[2] = (byte) ci;
2334
24.0k
                    }
2335
297k
                }
2336
1.41k
                if (dev->color_info.num_components == 1)
2337
1.17k
                    pclxl_set_color_palette(xdev, eGray, palette,
2338
1.17k
                                            1 << bits_per_pixel);
2339
237
                else
2340
237
                    pclxl_set_color_palette(xdev, eRGB, palette,
2341
237
                                            3 << bits_per_pixel);
2342
1.41k
            }
2343
4.59k
        }
2344
168k
    }
2345
168k
    return 0;
2346
1
  fail:
2347
1
    gs_free_object(mem, row_data, "pclxl_begin_image(rows)");
2348
1
    gs_free_object(mem, pie, "pclxl_begin_image");
2349
66.3k
  use_default:
2350
66.3k
    if (dev->color_info.num_components == 1)
2351
42.2k
        pclxl_set_color_space(xdev, eGray);
2352
24.0k
    else
2353
24.0k
        pclxl_set_color_space(xdev, eRGB);
2354
66.3k
    return gx_default_begin_typed_image(dev, pgs, pmat, pic, prect,
2355
66.3k
                                        pdcolor, pcpath, mem, pinfo);
2356
1
}
2357
2358
/* Write one strip of an image, from pie->rows.first_y to pie->y. */
2359
static int
2360
image_transform_x(const pclxl_image_enum_t * pie, int sx)
2361
387k
{
2362
387k
    return (int)((pie->mat.tx + sx * pie->mat.xx + 0.5) /
2363
387k
                 ((const gx_device_pclxl *)pie->dev)->scale.x);
2364
387k
}
2365
static int
2366
image_transform_y(const pclxl_image_enum_t * pie, int sy)
2367
387k
{
2368
387k
    return (int)((pie->mat.ty + sy * pie->mat.yy + 0.5) /
2369
387k
                 ((const gx_device_pclxl *)pie->dev)->scale.y);
2370
387k
}
2371
2372
static int
2373
pclxl_image_write_rows(pclxl_image_enum_t * pie)
2374
193k
{
2375
193k
    gx_device_pclxl *const xdev = (gx_device_pclxl *) pie->dev;
2376
193k
    stream *s = pclxl_stream(xdev);
2377
193k
    int y = pie->rows.first_y;
2378
193k
    int h = pie->y - y;
2379
193k
    int xo = image_transform_x(pie, 0);
2380
193k
    int yo = image_transform_y(pie, y);
2381
193k
    int dw = image_transform_x(pie, pie->width) - xo;
2382
193k
    int dh = image_transform_y(pie, y + h) - yo;
2383
193k
    int rows_raster = pie->rows.raster;
2384
193k
    int offset_lastflippedstrip = 0;
2385
2386
193k
    if (pie->flipped) {
2387
327
        yo = -yo - dh;
2388
327
        if (!pie->icclink)
2389
327
            offset_lastflippedstrip =
2390
327
                pie->rows.raster * (pie->rows.num_rows - h);
2391
0
        else
2392
0
            offset_lastflippedstrip =
2393
0
                (pie->rows.raster / (pie->bits_per_pixel >> 3))
2394
0
                * xdev->color_info.num_components * (pie->rows.num_rows - h);
2395
327
    };
2396
2397
193k
    if (dw <= 0 || dh <= 0)
2398
71.1k
        return 0;
2399
122k
    pclxl_set_cursor(xdev, xo, yo);
2400
122k
    if (pie->bits_per_pixel == 24) {
2401
21.2k
        static const byte ci_[] = {
2402
21.2k
            DA(pxaColorDepth),
2403
21.2k
            DUB(eDirectPixel), DA(pxaColorMapping)
2404
21.2k
        };
2405
2406
21.2k
        px_put_ub(s, eBit_values[8]);
2407
21.2k
        PX_PUT_LIT(s, ci_);
2408
21.2k
        if (xdev->color_info.depth == 8) {
2409
1.82k
            rows_raster /= 3;
2410
1.82k
            if (!pie->icclink) {
2411
1.82k
                byte *in = pie->rows.data + offset_lastflippedstrip;
2412
1.82k
                byte *out = pie->rows.data + offset_lastflippedstrip;
2413
1.82k
                int i;
2414
1.82k
                int j;
2415
2416
153k
                for (j = 0; j < h; j++) {
2417
205M
                    for (i = 0; i < rows_raster; i++) {
2418
205M
                        *out =
2419
205M
                            (byte) (((*(in + 0) * (ulong) lum_red_weight) +
2420
205M
                                     (*(in + 1) * (ulong) lum_green_weight) +
2421
205M
                                     (*(in + 2) * (ulong) lum_blue_weight) +
2422
205M
                                     (lum_all_weights / 2)) /
2423
205M
                                    lum_all_weights);
2424
205M
                        in += 3;
2425
205M
                        out++;
2426
205M
                    }
2427
151k
                }
2428
1.82k
            }
2429
1.82k
        }
2430
101k
    } else if (pie->bits_per_pixel == 32) {
2431
297
        static const byte ci_[] = {
2432
297
            DA(pxaColorDepth),
2433
297
            DUB(eDirectPixel), DA(pxaColorMapping)
2434
297
        };
2435
2436
297
        px_put_ub(s, eBit_values[8]);
2437
297
        PX_PUT_LIT(s, ci_);
2438
297
        if (xdev->color_info.depth == 8) {
2439
            /* CMYK to Gray */
2440
156
            rows_raster /= 4;
2441
156
            if (!pie->icclink) {
2442
0
                byte *in = pie->rows.data + offset_lastflippedstrip;
2443
0
                byte *out = pie->rows.data + offset_lastflippedstrip;
2444
0
                int i;
2445
0
                int j;
2446
2447
0
                for (j = 0; j < h; j++) {
2448
0
                    for (i = 0; i < rows_raster; i++) {
2449
                        /* DeviceCMYK to DeviceGray */
2450
0
                        int v =
2451
0
                            (255 - (*(in + 3))) * (int)lum_all_weights
2452
0
                            + (lum_all_weights / 2)
2453
0
                            - (*(in + 0) * (int)lum_red_weight)
2454
0
                            - (*(in + 1) * (int)lum_green_weight)
2455
0
                            - (*(in + 2) * (int)lum_blue_weight);
2456
2457
0
                        *out = max(v, 0) / lum_all_weights;
2458
0
                        in += 4;
2459
0
                        out++;
2460
0
                    }
2461
0
                }
2462
0
            }
2463
156
        } else {
2464
            /* CMYK to RGB */
2465
141
            rows_raster /= 4;
2466
141
            if (!pie->icclink) {
2467
0
                byte *in = pie->rows.data + offset_lastflippedstrip;
2468
0
                byte *out = pie->rows.data + offset_lastflippedstrip;
2469
0
                int i;
2470
0
                int j;
2471
2472
0
                for (j = 0; j < h; j++) {
2473
0
                    for (i = 0; i < rows_raster; i++) {
2474
                        /* DeviceCMYK to DeviceRGB */
2475
0
                        int r = (int)255 - (*(in + 0)) - (*(in + 3));
2476
0
                        int g = (int)255 - (*(in + 1)) - (*(in + 3));
2477
0
                        int b = (int)255 - (*(in + 2)) - (*(in + 3));
2478
2479
                        /* avoid trashing the first 12->9 conversion */
2480
0
                        *out = max(r, 0);
2481
0
                        out++;
2482
0
                        *out = max(g, 0);
2483
0
                        out++;
2484
0
                        *out = max(b, 0);
2485
0
                        out++;
2486
0
                        in += 4;
2487
0
                    }
2488
0
                }
2489
0
            }
2490
141
            rows_raster *= 3;
2491
141
        }
2492
100k
    } else {
2493
100k
        static const byte ii_[] = {
2494
100k
            DA(pxaColorDepth),
2495
100k
            DUB(eIndexedPixel), DA(pxaColorMapping)
2496
100k
        };
2497
100k
        px_put_ub(s, eBit_values[pie->bits_per_pixel]);
2498
100k
        PX_PUT_LIT(s, ii_);
2499
100k
    }
2500
122k
    pclxl_write_begin_image(xdev, pie->width, h, dw, dh);
2501
    /* 8-bit gray image may compress with jpeg, but we
2502
       cannot tell if it is 8-bit gray or 8-bit indexed */
2503
122k
    pclxl_write_image_data(xdev, pie->rows.data + offset_lastflippedstrip, 0,
2504
122k
                           rows_raster, rows_raster << 3, 0, h,
2505
122k
                           ((pie->bits_per_pixel == 24
2506
101k
                             || pie->bits_per_pixel == 32) ? true : false));
2507
122k
    pclxl_write_end_image(xdev);
2508
122k
    return 0;
2509
193k
}
2510
2511
/* Process the next piece of an image. */
2512
static int
2513
pclxl_image_plane_data(gx_image_enum_common_t * info,
2514
                       const gx_image_plane_t * planes, int height,
2515
                       int *rows_used)
2516
7.41M
{
2517
7.41M
    pclxl_image_enum_t *pie = (pclxl_image_enum_t *) info;
2518
7.41M
    int data_bit = planes[0].data_x * info->plane_depths[0];
2519
7.41M
    int width_bits = pie->width * info->plane_depths[0];
2520
7.41M
    int i;
2521
2522
    /****** SHOULD HANDLE NON-BYTE-ALIGNED DATA ******/
2523
7.41M
    if (width_bits != pie->bits_per_row || (data_bit & 7) != 0)
2524
48
        return_error(gs_error_rangecheck);
2525
7.41M
    if (height > pie->height - pie->y)
2526
90.6k
        height = pie->height - pie->y;
2527
17.9M
    for (i = 0; i < height; pie->y++, ++i) {
2528
10.5M
        int flipped_strip_offset;
2529
2530
10.5M
        if (pie->y - pie->rows.first_y == pie->rows.num_rows) {
2531
26.0k
            int code = pclxl_image_write_rows(pie);
2532
2533
26.0k
            if (code < 0)
2534
0
                return code;
2535
26.0k
            pie->rows.first_y = pie->y;
2536
26.0k
        }
2537
10.5M
        flipped_strip_offset = (pie->flipped ?
2538
12.6k
                                (pie->rows.num_rows -
2539
12.6k
                                 (pie->y - pie->rows.first_y) -
2540
10.5M
                                 1) : (pie->y - pie->rows.first_y));
2541
10.5M
        if (!pie->icclink)
2542
10.5M
            memcpy(pie->rows.data +
2543
10.5M
                   pie->rows.raster * flipped_strip_offset,
2544
10.5M
                   planes[0].data + planes[0].raster * i + (data_bit >> 3),
2545
10.5M
                   pie->rows.raster);
2546
46.3k
        else {
2547
46.3k
            gsicc_bufferdesc_t input_buff_desc;
2548
46.3k
            gsicc_bufferdesc_t output_buff_desc;
2549
46.3k
            int pixels_per_row =
2550
46.3k
                pie->rows.raster / (pie->bits_per_pixel >> 3);
2551
46.3k
            int out_raster_stride =
2552
46.3k
                pixels_per_row * info->dev->color_info.num_components;
2553
46.3k
            gsicc_init_buffer(&input_buff_desc,
2554
46.3k
                              (pie->bits_per_pixel >> 3) /*num_chan */ ,
2555
46.3k
                              1 /*bytes_per_chan */ ,
2556
46.3k
                              false /*has_alpha */ , false /*alpha_first */ ,
2557
46.3k
                              false /*is_planar */ ,
2558
46.3k
                              0 /*plane_stride */ ,
2559
46.3k
                              pie->rows.raster /*row_stride */ ,
2560
46.3k
                              1 /*num_rows */ ,
2561
46.3k
                              pixels_per_row /*pixels_per_row */ );
2562
46.3k
            gsicc_init_buffer(&output_buff_desc,
2563
46.3k
                              info->dev->color_info.num_components, 1, false,
2564
46.3k
                              false, false, 0, out_raster_stride, 1,
2565
46.3k
                              pixels_per_row);
2566
46.3k
            gscms_transform_color_buffer(info->dev, pie->icclink,
2567
46.3k
                                         &input_buff_desc, &output_buff_desc,
2568
46.3k
                                         (void *)(planes[0].data +
2569
46.3k
                                                  planes[0].raster * i +
2570
46.3k
                                                  (data_bit >> 3)) /*src */ ,
2571
46.3k
                                         pie->rows.data + out_raster_stride * flipped_strip_offset      /*des */
2572
46.3k
                                         );
2573
46.3k
        }
2574
10.5M
    }
2575
7.41M
    *rows_used = height;
2576
7.41M
    return pie->y >= pie->height;
2577
7.41M
}
2578
2579
/* Clean up by releasing the buffers. */
2580
static int
2581
pclxl_image_end_image(gx_image_enum_common_t * info, bool draw_last)
2582
168k
{
2583
168k
    pclxl_image_enum_t *pie = (pclxl_image_enum_t *) info;
2584
168k
    int code = 0;
2585
2586
    /* Write the final strip, if any. */
2587
168k
    if (pie->y > pie->rows.first_y && draw_last)
2588
167k
        code = pclxl_image_write_rows(pie);
2589
168k
    if (draw_last) {
2590
168k
        gx_device_pclxl *xdev = (gx_device_pclxl *) info->dev;
2591
168k
        stream *s = pclxl_stream(xdev);
2592
2593
168k
        switch (xdev->state_rotated) {
2594
21.0k
            case 1:
2595
21.0k
                xdev->state_rotated = 0;
2596
21.0k
                px_put_ss(s, -90);
2597
21.0k
                px_put_ac(s, pxaPageAngle, pxtSetPageRotation);
2598
21.0k
                break;
2599
73
            case -1:
2600
73
                xdev->state_rotated = 0;
2601
73
                px_put_ss(s, +90);
2602
73
                px_put_ac(s, pxaPageAngle, pxtSetPageRotation);
2603
73
                break;
2604
6.79k
            case 2:
2605
6.79k
                xdev->state_rotated = 0;
2606
6.79k
                px_put_ss(s, -180);
2607
6.79k
                px_put_ac(s, pxaPageAngle, pxtSetPageRotation);
2608
6.79k
                break;
2609
140k
            case 0:
2610
140k
            default:
2611
                /* do nothing */
2612
140k
                break;
2613
168k
        }
2614
168k
    }
2615
168k
    if (pie->icclink)
2616
215
        gsicc_release_link(pie->icclink);
2617
168k
    gs_free_object(pie->memory, pie->rows.data, "pclxl_end_image(rows)");
2618
168k
    gx_image_free_enum(&info);
2619
168k
    return code;
2620
168k
}
2621
2622
/*
2623
 * 'pclxl_get_params()' - Get pagedevice parameters.
2624
 */
2625
2626
static int                      /* O - Error status */
2627
pclxl_get_params(gx_device * dev,       /* I - Device info */
2628
                 gs_param_list * plist)
2629
373k
{                               /* I - Parameter list */
2630
373k
    gx_device_pclxl *xdev;      /* PCL XL device */
2631
373k
    int code;                   /* Return code */
2632
373k
    gs_param_string s;          /* Temporary string value */
2633
2634
    /*
2635
     * First process the "standard" page device parameters...
2636
     */
2637
2638
373k
    if ((code = gdev_vector_get_params(dev, plist)) < 0)
2639
0
        return (code);
2640
2641
    /*
2642
     * Then write the PCL-XL parameters...
2643
     */
2644
2645
373k
    xdev = (gx_device_pclxl *) dev;
2646
2647
373k
    if ((code = param_write_bool(plist, "Duplex", &(xdev->Duplex))) < 0)
2648
0
        return (code);
2649
2650
373k
    if ((code = param_write_bool(plist, "ManualFeed",
2651
373k
         &(xdev->ManualFeed))) < 0)
2652
0
        return (code);
2653
2654
373k
    if ((code = param_write_int(plist, "MediaPosition",
2655
373k
        &(xdev->MediaPosition))) < 0)
2656
0
        return (code);
2657
2658
373k
    param_string_from_string(s, xdev->MediaType);
2659
2660
373k
    if ((code = param_write_string(plist, "MediaType", &s)) < 0)
2661
0
        return (code);
2662
2663
373k
    if ((code = param_write_bool(plist, "Staple", &(xdev->Staple))) < 0)
2664
0
        return (code);
2665
2666
373k
    if ((code = param_write_bool(plist, "Tumble", &(xdev->Tumble))) < 0)
2667
0
        return (code);
2668
2669
373k
    if ((code = param_write_int(plist, "CompressMode",
2670
373k
                                &(xdev->CompressMode))) < 0)
2671
0
        return (code);
2672
2673
373k
    if ((code =
2674
373k
         param_write_bool(plist, "iccTransform", &(xdev->iccTransform))) < 0)
2675
0
        return (code);
2676
2677
373k
    return (0);
2678
373k
}
2679
2680
/*
2681
 * 'pclxl_put_params()' - Set pagedevice parameters.
2682
 */
2683
2684
static int                      /* O - Error status */
2685
pclxl_put_params(gx_device * dev,       /* I - Device info */
2686
                 gs_param_list * plist)
2687
181k
{                               /* I - Parameter list */
2688
181k
    gx_device_pclxl *xdev;      /* PCL XL device */
2689
181k
    int code;                   /* Error code */
2690
181k
    int intval;                 /* Integer value */
2691
181k
    bool boolval;               /* Boolean value */
2692
181k
    gs_param_string stringval;  /* String value */
2693
181k
    bool ManualFeed;
2694
181k
    bool ManualFeed_set = false;
2695
181k
    int MediaPosition;
2696
181k
    bool MediaPosition_set = false;
2697
2698
    /*
2699
     * Process PCL-XL driver parameters...
2700
     */
2701
2702
181k
    xdev = (gx_device_pclxl *) dev;
2703
2704
181k
#define intoption(name, sname, type) \
2705
181k
  if ((code = param_read_int(plist, sname, &intval)) < 0) \
2706
181k
  { \
2707
0
    if_debug1('|', "Error setting %s\n", sname); \
2708
0
    param_signal_error(plist, sname, code); \
2709
0
    return (code); \
2710
0
  } \
2711
181k
  else if (code == 0) \
2712
181k
  { \
2713
16.0k
    if_debug2('|', "setting %s to %d\n", sname, intval); \
2714
16.0k
    xdev->name = (type)intval; \
2715
16.0k
  }
2716
2717
181k
#define booloption(name, sname) \
2718
726k
  if ((code = param_read_bool(plist, sname, &boolval)) < 0) \
2719
726k
  { \
2720
0
    if_debug1('|', "Error setting bool %s\n", sname);    \
2721
0
    if ((code = param_read_null(plist, sname)) < 0) \
2722
0
    { \
2723
0
      if_debug1('|', "Error setting bool %s null\n", sname);     \
2724
0
      param_signal_error(plist, sname, code); \
2725
0
      return (code); \
2726
0
    } \
2727
0
    if (code == 0) \
2728
0
      xdev->name = false; \
2729
0
  } \
2730
726k
  else if (code == 0) {                                   \
2731
64.1k
    if_debug2('|', "setting %s to %d\n", sname, boolval); \
2732
64.1k
    xdev->name = (bool)boolval; \
2733
64.1k
  }
2734
2735
181k
#define stringoption(name, sname)                                \
2736
181k
  if ((code = param_read_string(plist, sname, &stringval)) < 0)  \
2737
181k
    {                                                            \
2738
0
      if_debug1('|', "Error setting %s string\n", sname);        \
2739
0
      if ((code = param_read_null(plist, sname)) < 0)            \
2740
0
        {                                                        \
2741
0
          if_debug1('|', "Error setting %s null\n", sname);      \
2742
0
          param_signal_error(plist, sname, code);                \
2743
0
          return (code);                                         \
2744
0
        }                                                        \
2745
0
      if (code == 0) {                                           \
2746
0
        if_debug1('|', "setting %s to empty\n", sname);          \
2747
0
        xdev->name[0] = '\0';                                    \
2748
0
      }                                                          \
2749
0
    }                                                            \
2750
181k
  else if (code == 0) {                                          \
2751
16.0k
    strncpy(xdev->name, (const char *)(stringval.data),          \
2752
16.0k
            stringval.size);                                     \
2753
16.0k
    xdev->name[stringval.size] = '\0';                           \
2754
16.0k
    if_debug2('|', "setting %s to %s\n", sname, xdev->name);     \
2755
16.0k
  }
2756
2757
    /* We need to have *_set to distinguish defaults from explicitly sets */
2758
181k
    booloption(Duplex, "Duplex");
2759
181k
    if (code == 0)
2760
16.0k
        if (xdev->Duplex) {
2761
0
            if_debug0('|', "round up page count\n");
2762
0
            xdev->page = (xdev->page + 1) & ~1;
2763
0
        }
2764
181k
    code = param_read_bool(plist, "ManualFeed", &ManualFeed);
2765
181k
    if (code == 0)
2766
16.0k
        ManualFeed_set = true;
2767
181k
    if (code >= 0) {
2768
181k
        code = param_read_int(plist, "MediaPosition", &MediaPosition);
2769
181k
        if (code == 0)
2770
41.2k
            MediaPosition_set = true;
2771
140k
        else if (code < 0) {
2772
0
            if (param_read_null(plist, "MediaPosition") == 0) {
2773
0
                code = 0;
2774
0
            }
2775
0
        }
2776
181k
    }
2777
181k
    stringoption(MediaType, "MediaType");
2778
181k
    if (code == 0) {
2779
16.0k
        xdev->MediaType_set = true;
2780
        /* round up for duplex */
2781
16.0k
        if (strcmp(xdev->MediaType_old, xdev->MediaType)) {
2782
0
            if_debug0('|', "round up page count\n");
2783
0
            xdev->page = (xdev->page + 1) & ~1;
2784
0
            strcpy(xdev->MediaType_old, xdev->MediaType);
2785
0
        }
2786
16.0k
    }
2787
181k
    booloption(Staple, "Staple");
2788
181k
    booloption(Tumble, "Tumble");
2789
181k
    intoption(CompressMode, "CompressMode", int);
2790
2791
181k
    booloption(iccTransform, "iccTransform");
2792
2793
    /*
2794
     * Then process standard page device parameters...
2795
     */
2796
2797
181k
    if (code >= 0)
2798
181k
        if ((code = gdev_vector_put_params(dev, plist)) < 0)
2799
419
            return (code);
2800
2801
181k
    if (code >= 0) {
2802
181k
        if (ManualFeed_set) {
2803
16.0k
            xdev->ManualFeed = ManualFeed;
2804
16.0k
            xdev->ManualFeed_set = true;
2805
16.0k
        }
2806
181k
        if (MediaPosition_set) {
2807
41.2k
            xdev->MediaPosition = MediaPosition;
2808
41.2k
            xdev->MediaPosition_set = true;
2809
41.2k
            if (xdev->MediaPosition_old != xdev->MediaPosition) {
2810
25.2k
                if_debug0('|', "round up page count\n");
2811
25.2k
                xdev->page = (xdev->page+1) & ~1 ;
2812
25.2k
                xdev->MediaPosition_old = xdev->MediaPosition;
2813
25.2k
            }
2814
41.2k
        }
2815
181k
    }
2816
2817
181k
    return (0);
2818
181k
}
2819
2820
static int pclxl_text_begin(gx_device * dev, gs_gstate * pgs,
2821
                    const gs_text_params_t *text, gs_font * font,
2822
                    const gx_clip_path * pcpath,
2823
                    gs_text_enum_t ** ppte)
2824
1.36M
{
2825
1.36M
    font->dir->ccache.upper = 0;
2826
1.36M
    return gx_default_text_begin(dev, pgs, text, font, pcpath, ppte);
2827
1.36M
}