Coverage Report

Created: 2026-04-09 07:06

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ghostpdl/base/gscspace.c
Line
Count
Source
1
/* Copyright (C) 2001-2026 Artifex Software, Inc.
2
   All Rights Reserved.
3
4
   This software is provided AS-IS with no warranty, either express or
5
   implied.
6
7
   This software is distributed under license and may not be copied,
8
   modified or distributed except as expressly authorized under the terms
9
   of the license contained in the file LICENSE in this distribution.
10
11
   Refer to licensing information at http://www.artifex.com or contact
12
   Artifex Software, Inc.,  39 Mesa Street, Suite 108A, San Francisco,
13
   CA 94129, USA, for further information.
14
*/
15
16
17
/* Color space operators and support */
18
#include "memory_.h"
19
#include "gx.h"
20
#include "gserrors.h"
21
#include "gsstruct.h"
22
#include "gsccolor.h"
23
#include "gsutil.h"   /* for gs_next_ids */
24
#include "gxcmap.h"
25
#include "gxcspace.h"
26
#include "gxgstate.h"
27
#include "gsovrc.h"
28
#include "gsstate.h"
29
#include "gsdevice.h"
30
#include "gxdevcli.h"
31
#include "gzstate.h"
32
#include "stream.h"
33
#include "gsnamecl.h"  /* Custom color call back define */
34
#include "gsicc.h"
35
#include "gsicc_manage.h"
36
#include "string_.h"
37
#include "strmio.h"         /* needed for sfclose */
38
#include "gsicc_cache.h"    /* Needed for gsicc_get_icc_buff_hash */
39
#include "gxdevsop.h"
40
41
static cs_proc_install_cspace(gx_install_DeviceGray);
42
static cs_proc_install_cspace(gx_install_DeviceRGB);
43
static cs_proc_install_cspace(gx_install_DeviceCMYK);
44
/*
45
 * Define the standard color space types.  We include DeviceCMYK in the base
46
 * build because it's too awkward to omit it, but we don't provide any of
47
 * the PostScript operator procedures (setcmykcolor, etc.) for dealing with
48
 * it.
49
 */
50
static const gs_color_space_type gs_color_space_type_DeviceGray = {
51
    gs_color_space_index_DeviceGray, true, true,
52
    &st_base_color_space, gx_num_components_1,
53
    gx_init_paint_1, gx_restrict01_paint_1,
54
    gx_same_concrete_space,
55
    gx_concretize_DeviceGray, gx_remap_concrete_DGray,
56
    gx_remap_DeviceGray, gx_install_DeviceGray,
57
    gx_spot_colors_set_overprint,
58
    NULL, gx_no_adjust_color_count,
59
    gx_serialize_cspace_type,
60
    gx_cspace_is_linear_default, gx_polarity_additive
61
};
62
static const gs_color_space_type gs_color_space_type_DeviceRGB = {
63
    gs_color_space_index_DeviceRGB, true, true,
64
    &st_base_color_space, gx_num_components_3,
65
    gx_init_paint_3, gx_restrict01_paint_3,
66
    gx_same_concrete_space,
67
    gx_concretize_DeviceRGB, gx_remap_concrete_DRGB,
68
    gx_remap_DeviceRGB, gx_install_DeviceRGB,
69
    gx_spot_colors_set_overprint,
70
    NULL, gx_no_adjust_color_count,
71
    gx_serialize_cspace_type,
72
    gx_cspace_is_linear_default, gx_polarity_additive
73
};
74
75
static cs_proc_set_overprint(gx_set_overprint_DeviceCMYK);
76
77
static const gs_color_space_type gs_color_space_type_DeviceCMYK = {
78
    gs_color_space_index_DeviceCMYK, true, true,
79
    &st_base_color_space, gx_num_components_4,
80
    gx_init_paint_4, gx_restrict01_paint_4,
81
    gx_same_concrete_space,
82
    gx_concretize_DeviceCMYK, gx_remap_concrete_DCMYK,
83
    gx_remap_DeviceCMYK, gx_install_DeviceCMYK,
84
    gx_set_overprint_DeviceCMYK,
85
    NULL, gx_no_adjust_color_count,
86
    gx_serialize_cspace_type,
87
    gx_cspace_is_linear_default, gx_polarity_subtractive
88
};
89
90
/* Structure descriptors */
91
public_st_color_space();
92
public_st_base_color_space();
93
94
/* ------ Create/copy/destroy ------ */
95
96
/* Ghostscript object finalizers can be called many times and hence
97
 * must be idempotent. */
98
static void
99
gs_cspace_final(const gs_memory_t *cmem, void *vptr)
100
29.4M
{
101
29.4M
    gs_color_space *pcs = (gs_color_space *)vptr;
102
29.4M
    (void)cmem; /* unused */
103
104
29.4M
    if (pcs->interpreter_free_cspace_proc != NULL) {
105
1.38M
        (*pcs->interpreter_free_cspace_proc) ((gs_memory_t *)cmem, pcs);
106
1.38M
        pcs->interpreter_free_cspace_proc = NULL;
107
1.38M
    }
108
29.4M
    if (pcs->type->final)
109
16.7M
        pcs->type->final(pcs);
110
29.4M
    if_debug2m('c', cmem, "[c]cspace final "PRI_INTPTR" %d\n", (intptr_t)pcs, (int)pcs->id);
111
29.4M
    rc_decrement_only_cs(pcs->base_space, "gs_cspace_final");
112
29.4M
    pcs->base_space = NULL;
113
29.4M
    if (gs_color_space_get_index(pcs) == gs_color_space_index_DeviceN) {
114
10.4k
        if (pcs->params.device_n.devn_process_space != NULL) {
115
0
            rc_decrement_only_cs(pcs->params.device_n.devn_process_space, "gs_cspace_final");
116
0
            pcs->params.device_n.devn_process_space = NULL;
117
0
        }
118
10.4k
    }
119
    /* No need to decrement the ICC profile data.  It is handled
120
       by the finalize of the ICC space which is called above using
121
       pcs->type->final(pcs);  */
122
29.4M
}
123
124
static gs_color_space *
125
gs_cspace_alloc_with_id(gs_memory_t *mem, ulong id,
126
                   const gs_color_space_type *pcstype)
127
29.4M
{
128
29.4M
    gs_color_space *pcs;
129
130
29.4M
    rc_alloc_struct_1(pcs, gs_color_space, &st_color_space, mem, return NULL,
131
29.4M
                      "gs_cspace_alloc_with_id");
132
29.4M
    if_debug3m('c', mem, "[c]cspace alloc "PRI_INTPTR" %s %d\n",
133
29.4M
               (intptr_t)pcs, pcstype->stype->sname, pcstype->index);
134
29.4M
    pcs->type = pcstype;
135
29.4M
    pcs->id = id;
136
29.4M
    pcs->base_space = NULL;
137
29.4M
    pcs->pclient_color_space_data = NULL;
138
29.4M
    pcs->interpreter_data = NULL;
139
29.4M
    pcs->interpreter_free_cspace_proc = NULL;
140
29.4M
    pcs->cmm_icc_profile_data = NULL;
141
29.4M
    pcs->ICC_Alternate_space = gs_ICC_Alternate_None;
142
29.4M
    pcs->icc_equivalent = NULL;
143
29.4M
    pcs->params.device_n.devn_process_space = NULL;
144
29.4M
    pcs->params.device_n.all_none = false;
145
29.4M
    return pcs;
146
29.4M
}
147
148
static cs_proc_install_cspace(gx_install_DeviceGray);
149
static cs_proc_install_cspace(gx_install_DeviceRGB);
150
static cs_proc_install_cspace(gx_install_DeviceCMYK);
151
152
/*
153
 * Generic allocation function for colorspace implementations. Return
154
 * NULL on allocation failure.
155
 */
156
gs_color_space *
157
gs_cspace_alloc(gs_memory_t *mem, const gs_color_space_type *pcstype)
158
3.66M
{
159
3.66M
    return gs_cspace_alloc_with_id(mem, gs_next_ids(mem, 1), pcstype);
160
3.66M
}
161
162
/* Constructors for simple device color spaces. */
163
164
gs_color_space *
165
gs_cspace_new_DeviceGray(gs_memory_t *mem)
166
17.3M
{
167
17.3M
    return gs_cspace_alloc_with_id(mem, cs_DeviceGray_id,
168
17.3M
                                   &gs_color_space_type_DeviceGray);
169
17.3M
}
170
171
gs_color_space *
172
gs_cspace_new_DeviceRGB(gs_memory_t *mem)
173
5.69M
{
174
5.69M
    return gs_cspace_alloc_with_id(mem, cs_DeviceRGB_id,
175
5.69M
                                   &gs_color_space_type_DeviceRGB);
176
5.69M
}
177
gs_color_space *
178
gs_cspace_new_DeviceCMYK(gs_memory_t *mem)
179
2.70M
{
180
2.70M
    return gs_cspace_alloc_with_id(mem, cs_DeviceCMYK_id,
181
2.70M
                                   &gs_color_space_type_DeviceCMYK);
182
2.70M
}
183
184
/* For use in initializing ICC color spaces for XPS */
185
gs_color_space *
186
gs_cspace_new_scrgb(gs_memory_t *pmem, gs_gstate * pgs)
187
8.97k
{
188
8.97k
    gs_color_space *pcspace = gs_cspace_alloc(pmem, &gs_color_space_type_ICC);
189
8.97k
    cmm_profile_t *profile;
190
8.97k
    stream *str;
191
8.97k
    int code;
192
193
8.97k
    if (pcspace == NULL)
194
0
        return pcspace;
195
196
8.97k
    code = gsicc_open_search(SCRGB, strlen(SCRGB), pmem, pmem->gs_lib_ctx->profiledir,
197
8.97k
        pmem->gs_lib_ctx->profiledir_len, &str);
198
199
8.97k
    if (code < 0 || str == NULL) {
200
0
        rc_decrement(pcspace, "gs_cspace_new_scrgb");
201
0
        return NULL;
202
0
    }
203
204
8.97k
    pcspace->cmm_icc_profile_data = gsicc_profile_new(str, pmem, SCRGB, strlen(SCRGB));
205
8.97k
    code = sfclose(str);
206
8.97k
    if (pcspace->cmm_icc_profile_data == NULL) {
207
0
        rc_decrement(pcspace, "gs_cspace_new_scrgb");
208
0
        return NULL;
209
0
    }
210
211
    /* Get the profile handle */
212
8.97k
    pcspace->cmm_icc_profile_data->profile_handle =
213
8.97k
        gsicc_get_profile_handle_buffer(pcspace->cmm_icc_profile_data->buffer,
214
8.97k
            pcspace->cmm_icc_profile_data->buffer_size, pmem);
215
8.97k
    if (!pcspace->cmm_icc_profile_data->profile_handle) {
216
0
        rc_decrement(pcspace, "gs_cspace_new_scrgb");
217
0
        return NULL;
218
0
    }
219
8.97k
    profile = pcspace->cmm_icc_profile_data;
220
221
    /* Compute the hash code of the profile. Everything in the
222
    ICC manager will have it's hash code precomputed */
223
8.97k
    gsicc_get_icc_buff_hash(profile->buffer, &(profile->hashcode),
224
8.97k
        profile->buffer_size);
225
8.97k
    profile->hash_is_valid = true;
226
8.97k
    profile->num_comps =
227
8.97k
        gscms_get_input_channel_count(profile->profile_handle, profile->memory);
228
8.97k
    if (profile->num_comps > ICC_MAX_CHANNELS) {
229
0
        rc_decrement(pcspace, "gs_cspace_new_scrgb");
230
0
        return NULL;
231
0
    }
232
8.97k
    profile->num_comps_out =
233
8.97k
        gscms_get_output_channel_count(profile->profile_handle, profile->memory);
234
8.97k
    profile->data_cs =
235
8.97k
        gscms_get_profile_data_space(profile->profile_handle, profile->memory);
236
8.97k
    gsicc_set_icc_range(&profile);
237
8.97k
    return pcspace;
238
8.97k
}
239
240
gs_color_space *
241
gs_cspace_new_ICC(gs_memory_t *pmem, gs_gstate * pgs, int components)
242
2.38M
{
243
2.38M
    gsicc_manager_t *icc_manage = pgs->icc_manager;
244
2.38M
    int code = 0;
245
2.38M
    gs_color_space *pcspace = gs_cspace_alloc(pmem, &gs_color_space_type_ICC);
246
247
2.38M
    if (pcspace == NULL)
248
0
        return pcspace;
249
250
2.38M
    switch (components) {
251
8.97k
        case -1: /* alpha case */
252
8.97k
            if (icc_manage->smask_profiles == NULL) {
253
8.97k
                code = gsicc_initialize_iccsmask(icc_manage);
254
8.97k
            }
255
8.97k
            if (code == 0) {
256
8.97k
                pcspace->cmm_icc_profile_data =
257
8.97k
                    icc_manage->smask_profiles->smask_gray;
258
8.97k
            } else {
259
0
                pcspace->cmm_icc_profile_data = icc_manage->default_gray;
260
0
            }
261
8.97k
            break;
262
0
        case -3: /* alpha case.  needs linear RGB */
263
0
            if (icc_manage->smask_profiles == NULL) {
264
0
                code = gsicc_initialize_iccsmask(icc_manage);
265
0
            }
266
0
            if (code == 0) {
267
0
                pcspace->cmm_icc_profile_data =
268
0
                    icc_manage->smask_profiles->smask_rgb;
269
0
            } else {
270
0
                pcspace->cmm_icc_profile_data = icc_manage->default_rgb;
271
0
            }
272
0
            break;
273
2.25M
        case 1: pcspace->cmm_icc_profile_data = icc_manage->default_gray; break;
274
62.8k
        case 3: pcspace->cmm_icc_profile_data = icc_manage->default_rgb; break;
275
53.8k
        case 4: pcspace->cmm_icc_profile_data = icc_manage->default_cmyk; break;
276
0
        default: rc_decrement(pcspace,"gs_cspace_new_ICC"); return NULL;
277
2.38M
    }
278
2.38M
    gsicc_adjust_profile_rc(pcspace->cmm_icc_profile_data, 1, "gs_cspace_new_ICC");
279
2.38M
    return pcspace;
280
2.38M
}
281
282
/* ------ Accessors ------ */
283
284
/* Get the index of a color space. */
285
gs_color_space_index
286
gs_color_space_get_index(const gs_color_space * pcs)
287
79.9M
{
288
79.9M
    return pcs->type->index;
289
79.9M
}
290
291
/* See if the space is CIE based */
292
bool gs_color_space_is_CIE(const gs_color_space * pcs)
293
895k
{
294
895k
    switch(gs_color_space_get_index(pcs)){
295
0
        case gs_color_space_index_CIEDEFG:
296
0
        case gs_color_space_index_CIEDEF:
297
0
        case gs_color_space_index_CIEABC:
298
0
        case gs_color_space_index_CIEA:
299
895k
        case gs_color_space_index_ICC:
300
895k
            return true;
301
0
        break;
302
567
        default:
303
567
            return false;
304
895k
    }
305
895k
}
306
307
/* See if the space is Postscript CIE based */
308
bool gs_color_space_is_PSCIE(const gs_color_space * pcs)
309
33.2M
{
310
33.2M
    switch(gs_color_space_get_index(pcs)){
311
0
        case gs_color_space_index_CIEDEFG:
312
0
        case gs_color_space_index_CIEDEF:
313
0
        case gs_color_space_index_CIEABC:
314
0
        case gs_color_space_index_CIEA:
315
0
            return true;
316
0
        break;
317
33.2M
        default:
318
33.2M
            return false;
319
33.2M
}
320
33.2M
}
321
322
/* See if the space is ICC based */
323
bool gs_color_space_is_ICC(const gs_color_space * pcs)
324
2.15M
{
325
2.15M
    return(gs_color_space_get_index(pcs) == gs_color_space_index_ICC);
326
2.15M
}
327
328
/* Get the number of components in a color space. */
329
int
330
gs_color_space_num_components(const gs_color_space * pcs)
331
11.3M
{
332
11.3M
    return cs_num_components(pcs);
333
11.3M
}
334
335
/* Restrict a color to its legal range. */
336
void
337
gs_color_space_restrict_color(gs_client_color *pcc, const gs_color_space *pcs)
338
0
{
339
0
    cs_restrict_color(pcc, pcs);
340
0
}
341
342
/* Install a DeviceGray color space. */
343
static int
344
gx_install_DeviceGray(gs_color_space * pcs, gs_gstate * pgs)
345
11.7M
{
346
    /* If we already have profile data installed, nothing to do here. */
347
11.7M
    if (pcs->cmm_icc_profile_data != NULL)
348
0
        return 0;
349
350
    /* If we haven't initialised the iccmanager, do it now. */
351
11.7M
    if (pgs->icc_manager->default_gray == NULL) {
352
2.10M
        int code = gsicc_init_iccmanager(pgs);
353
2.10M
        if (code < 0)
354
0
            return code;
355
2.10M
    }
356
357
    /* pcs takes a reference to the default_gray profile data */
358
11.7M
    pcs->cmm_icc_profile_data = pgs->icc_manager->default_gray;
359
11.7M
    gsicc_adjust_profile_rc(pgs->icc_manager->default_gray, 1, "gx_install_DeviceGray");
360
11.7M
    pcs->type = &gs_color_space_type_ICC;
361
11.7M
    return 0;
362
11.7M
}
363
364
int
365
gx_num_components_1(const gs_color_space * pcs)
366
18.0M
{
367
18.0M
    return 1;
368
18.0M
}
369
int
370
gx_num_components_3(const gs_color_space * pcs)
371
23.3k
{
372
23.3k
    return 3;
373
23.3k
}
374
int
375
gx_num_components_4(const gs_color_space * pcs)
376
0
{
377
0
    return 4;
378
0
}
379
380
gx_color_polarity_t
381
gx_polarity_subtractive(const gs_color_space * pcs)
382
0
{
383
0
    return GX_CINFO_POLARITY_SUBTRACTIVE;
384
0
}
385
386
gx_color_polarity_t
387
gx_polarity_additive(const gs_color_space * pcs)
388
0
{
389
0
    return GX_CINFO_POLARITY_ADDITIVE;
390
0
}
391
392
gx_color_polarity_t
393
gx_polarity_unknown(const gs_color_space * pcs)
394
0
{
395
0
    return GX_CINFO_POLARITY_UNKNOWN;
396
0
}
397
398
/*
399
 * For color spaces that have a base or alternative color space, return that
400
 * color space. Otherwise return null.
401
 */
402
const gs_color_space *
403
gs_cspace_base_space(const gs_color_space * pcspace)
404
158k
{
405
158k
    return pcspace->base_space;
406
158k
}
407
408
const gs_color_space *
409
gs_cspace_devn_process_space(const gs_color_space * pcspace)
410
0
{
411
0
    return pcspace->params.device_n.devn_process_space;
412
0
}
413
414
/* Abstract the reference counting for color spaces
415
   so that we can also increment the ICC profile
416
   if there is one associated with the color space */
417
418
void rc_increment_cs(gs_color_space *pcs)
419
16.7M
{
420
16.7M
    rc_increment(pcs);
421
16.7M
}
422
423
5.50M
void rc_decrement_cs(gs_color_space *pcs, const char *cname) {
424
425
5.50M
    if (pcs) {
426
5.50M
        rc_decrement(pcs, cname);
427
5.50M
    }
428
5.50M
}
429
430
void rc_decrement_only_cs(gs_color_space *pcs, const char *cname)
431
58.5M
{
432
58.5M
    if (pcs) {
433
29.2M
        rc_decrement_only(pcs, cname);
434
29.2M
    }
435
58.5M
}
436
437
void cs_adjust_counts_icc(gs_gstate *pgs, int delta)
438
122M
{
439
122M
    gs_color_space *pcs = gs_currentcolorspace_inline(pgs);
440
441
122M
    if (pcs) {
442
81.8M
        cs_adjust_color_count(pgs, delta);
443
81.8M
        rc_adjust_const(pcs, delta, "cs_adjust_counts_icc");
444
81.8M
    }
445
122M
}
446
447
void cs_adjust_swappedcounts_icc(gs_gstate *pgs, int delta)
448
122M
{
449
122M
    gs_color_space *pcs = gs_swappedcolorspace_inline(pgs);
450
451
122M
    if (pcs) {
452
81.8M
        cs_adjust_swappedcolor_count(pgs, delta);
453
81.8M
        rc_adjust_const(pcs, delta, "cs_adjust_swappedcounts_icc");
454
81.8M
    }
455
122M
}
456
457
/* ------ Other implementation procedures ------ */
458
459
/* Null color space installation procedure. */
460
int
461
gx_no_install_cspace(gs_color_space * pcs, gs_gstate * pgs)
462
0
{
463
0
    return 0;
464
0
}
465
466
/* Install a DeviceRGB color space. */
467
static int
468
gx_install_DeviceRGB(gs_color_space * pcs, gs_gstate * pgs)
469
1.09M
{
470
    /* If we already have profile_data, nothing to do here. */
471
1.09M
    if (pcs->cmm_icc_profile_data != NULL)
472
0
        return 0;
473
474
    /* If the icc manager hasn't been set up yet, then set it up. */
475
1.09M
    if (pgs->icc_manager->default_rgb == NULL)
476
0
        gsicc_init_iccmanager(pgs);
477
478
    /* pcs takes a reference to default_rgb */
479
1.09M
    pcs->cmm_icc_profile_data = pgs->icc_manager->default_rgb;
480
1.09M
    gsicc_adjust_profile_rc(pcs->cmm_icc_profile_data, 1, "gx_install_DeviceRGB");
481
1.09M
    pcs->type = &gs_color_space_type_ICC;
482
1.09M
    return 0;
483
1.09M
}
484
485
/* Install a DeviceCMYK color space. */
486
static int
487
gx_install_DeviceCMYK(gs_color_space * pcs, gs_gstate * pgs)
488
123k
{
489
    /* If we already have profile data, nothing to do here. */
490
123k
    if (pcs->cmm_icc_profile_data != NULL)
491
0
        return 0;
492
493
    /* If the icc manager hasn't been set up yet, then set it up. */
494
123k
    if (pgs->icc_manager->default_cmyk == NULL)
495
0
        gsicc_init_iccmanager(pgs);
496
497
    /* pcs takes a reference to default_cmyk */
498
123k
    pcs->cmm_icc_profile_data = pgs->icc_manager->default_cmyk;
499
123k
    gsicc_adjust_profile_rc(pcs->cmm_icc_profile_data, 1, "gx_install_DeviceCMYK");
500
123k
    pcs->type = &gs_color_space_type_ICC;
501
123k
    return 0;
502
123k
}
503
504
/*
505
 * Communicate to the overprint compositor that this particular
506
 * state overprint is not enabled.  This could be due to a
507
 * mismatched color space, or that overprint is false or the
508
 * device does not support it.
509
 */
510
int
511
gx_set_no_overprint(gs_gstate* pgs)
512
457k
{
513
457k
    gs_overprint_params_t   params = { 0 };
514
515
457k
    params.retain_any_comps = false;
516
457k
    params.op_state = OP_STATE_NONE;
517
457k
    params.is_fill_color = pgs->is_fill_color;
518
457k
    params.effective_opm = pgs->color[0].effective_opm = 0;
519
520
457k
    return gs_gstate_update_overprint(pgs, &params);
521
457k
}
522
523
/* Retain all the spot colorants and not the process
524
   colorants.  This occurs if we have a process color
525
   mismatch between the source and the destination but
526
   the output device supports spot colors */
527
int
528
gx_set_spot_only_overprint(gs_gstate* pgs)
529
2.49k
{
530
2.49k
    gs_overprint_params_t   params = { 0 };
531
2.49k
    gx_device* dev = pgs->device;
532
2.49k
    gx_color_index drawn_comps = dev == NULL ? 0 : gx_get_process_comps(dev);
533
534
2.49k
    params.retain_any_comps = true;
535
2.49k
    params.op_state = OP_STATE_NONE;
536
2.49k
    params.is_fill_color = pgs->is_fill_color;
537
2.49k
    params.effective_opm = pgs->color[0].effective_opm = 0;
538
2.49k
    params.drawn_comps = drawn_comps;
539
540
2.49k
    return gs_gstate_update_overprint(pgs, &params);
541
2.49k
}
542
543
/*
544
 * Push an overprint compositor onto the current device indicating that,
545
 * at most, the spot color parameters are to be preserved.
546
 *
547
 * This routine should be used for all Device, CIEBased, and ICCBased
548
 * color spaces, except for DeviceCMKY.
549
 */
550
int
551
gx_spot_colors_set_overprint(const gs_color_space * pcs, gs_gstate * pgs)
552
0
{
553
0
    gs_overprint_params_t   params = {0};
554
0
    bool op = pgs->is_fill_color ? pgs->overprint : pgs->stroke_overprint;
555
556
0
    if (!op)
557
0
        params.retain_any_comps = false;
558
0
    else
559
0
        params.retain_any_comps = true;
560
561
0
    params.is_fill_color = pgs->is_fill_color;
562
0
    params.op_state = OP_STATE_NONE;
563
564
    /* Only DeviceCMYK case can have overprint mode set to true */
565
0
    params.effective_opm = pgs->color[0].effective_opm = 0;
566
0
    return gs_gstate_update_overprint(pgs, &params);
567
0
}
568
569
static bool
570
check_single_comp(int comp, frac targ_val, int ncomps, const frac * pval)
571
728
{
572
728
    int     i;
573
574
3.69k
    for (i = 0; i < ncomps; i++) {
575
2.96k
        if ( (i != comp && pval[i] != frac_0)  ||
576
2.96k
             (i == comp && pval[i] != targ_val)  )
577
0
            return false;
578
2.96k
    }
579
728
    return true;
580
728
}
581
582
/*
583
 * Determine if the current color model is a "DeviceCMYK" color model, and
584
 * if so what are its process color components. This information is required
585
 * when PLRM defines special rules for CMYK devices. This includes:
586
 * 1. DeviceGray to CMYK color conversion
587
 * 2. when overprint is true and overprint mode is set to 1.
588
 *
589
 * A color model is considered a "DeviceCMYK" color model if it supports the
590
 * cyan, magenta, yellow, and black color components, and maps the DeviceCMYK
591
 * color model components directly to these color components. Note that this
592
 * does not require any particular component order, allows for additional
593
 * spot color components, and does admit DeviceN color spaces if they have
594
 * the requisite behavior.
595
 *
596
 * If the color model is a "DeviceCMYK" color model, return the set of
597
 * process color components; otherwise return 0.
598
 */
599
gx_color_index
600
check_cmyk_color_model_comps(gx_device * dev)
601
943
{
602
943
    gx_device_color_info *          pcinfo = &dev->color_info;
603
943
    uchar                           ncomps = pcinfo->num_components;
604
943
    int                             cyan_c, magenta_c, yellow_c, black_c;
605
943
    frac                            frac_14 = frac_1 / 4;
606
943
    frac                            out[GX_DEVICE_COLOR_MAX_COMPONENTS];
607
943
    gx_color_index                  process_comps;
608
943
    const gx_cm_color_map_procs    *cmprocs;
609
943
    const gx_device                *cmdev;
610
611
612
943
    if (pcinfo->num_components < 4                     ||
613
210
        pcinfo->polarity == GX_CINFO_POLARITY_ADDITIVE ||
614
761
        pcinfo->gray_index == GX_CINFO_COMP_NO_INDEX) {
615
761
        pcinfo->opmsupported = GX_CINFO_OPMSUPPORTED_NOT;
616
761
        return 0;
617
761
    }
618
619
    /* check for the appropriate components */
620
182
    if ( ncomps < 4                                       ||
621
182
         (cyan_c = dev_proc(dev, get_color_comp_index)(
622
182
                       dev,
623
182
                       "Cyan",
624
182
                       sizeof("Cyan") - 1,
625
182
                       NO_COMP_NAME_TYPE_OP)) < 0           ||
626
182
         cyan_c == GX_DEVICE_COLOR_MAX_COMPONENTS         ||
627
182
         (magenta_c = dev_proc(dev, get_color_comp_index)(
628
182
                          dev,
629
182
                          "Magenta",
630
182
                          sizeof("Magenta") - 1,
631
182
                          NO_COMP_NAME_TYPE_OP)) < 0        ||
632
182
         magenta_c == GX_DEVICE_COLOR_MAX_COMPONENTS      ||
633
182
         (yellow_c = dev_proc(dev, get_color_comp_index)(
634
182
                        dev,
635
182
                        "Yellow",
636
182
                        sizeof("Yellow") - 1,
637
182
                        NO_COMP_NAME_TYPE_OP)) < 0               ||
638
182
         yellow_c == GX_DEVICE_COLOR_MAX_COMPONENTS       ||
639
182
         (black_c = dev_proc(dev, get_color_comp_index)(
640
182
                        dev,
641
182
                        "Black",
642
182
                        sizeof("Black") - 1,
643
182
                        NO_COMP_NAME_TYPE_OP)) < 0                         ||
644
182
         black_c == GX_DEVICE_COLOR_MAX_COMPONENTS          )
645
0
        return 0;
646
647
    /* check the mapping */
648
182
    cmprocs = dev_proc(dev, get_color_mapping_procs)(dev, &cmdev);
649
650
182
    ncomps -= device_encodes_tags(dev);
651
182
    cmprocs->map_cmyk(cmdev, frac_14, frac_0, frac_0, frac_0, out);
652
182
    if (!check_single_comp(cyan_c, frac_14, ncomps, out)) {
653
0
        pcinfo->opmsupported = GX_CINFO_OPMSUPPORTED_NOT;
654
0
        return 0;
655
0
    }
656
182
    cmprocs->map_cmyk(cmdev, frac_0, frac_14, frac_0, frac_0, out);
657
182
    if (!check_single_comp(magenta_c, frac_14, ncomps, out)) {
658
0
        pcinfo->opmsupported = GX_CINFO_OPMSUPPORTED_NOT;
659
0
        return 0;
660
0
    }
661
182
    cmprocs->map_cmyk(cmdev, frac_0, frac_0, frac_14, frac_0, out);
662
182
    if (!check_single_comp(yellow_c, frac_14, ncomps, out)) {
663
0
        pcinfo->opmsupported = GX_CINFO_OPMSUPPORTED_NOT;
664
0
        return 0;
665
0
    }
666
182
    cmprocs->map_cmyk(cmdev, frac_0, frac_0, frac_0, frac_14, out);
667
182
    if (!check_single_comp(black_c, frac_14, ncomps, out)) {
668
0
        pcinfo->opmsupported = GX_CINFO_OPMSUPPORTED_NOT;
669
0
        return 0;
670
0
    }
671
672
182
    process_comps =  ((gx_color_index)1 << cyan_c)
673
182
                   | ((gx_color_index)1 << magenta_c)
674
182
                   | ((gx_color_index)1 << yellow_c)
675
182
                   | ((gx_color_index)1 << black_c);
676
182
    pcinfo->opmsupported = GX_CINFO_OPMSUPPORTED;
677
182
    pcinfo->process_comps = process_comps;
678
182
    pcinfo->black_component = black_c;
679
182
    return process_comps;
680
182
}
681
682
static void
683
check_rgb_color_model_comps(gx_device * dev)
684
5.27k
{
685
5.27k
    gx_device_color_info *          pcinfo = &dev->color_info;
686
5.27k
    uchar                           ncomps = pcinfo->num_components;
687
5.27k
    int                             red_c, green_c, blue_c;
688
5.27k
    frac                            frac_14 = frac_1 / 4;
689
5.27k
    frac                            out[GX_DEVICE_COLOR_MAX_COMPONENTS];
690
5.27k
    gx_color_index                  process_comps;
691
5.27k
    const gx_cm_color_map_procs    *cmprocs;
692
5.27k
    const gx_device                *cmdev;
693
694
695
5.27k
    if (pcinfo->num_components < 3                     ||
696
3.38k
        pcinfo->polarity != GX_CINFO_POLARITY_ADDITIVE ||
697
3.38k
        pcinfo->gray_index == GX_CINFO_COMP_NO_INDEX   ||
698
1.32k
        dev_proc(dev, dev_spec_op)(dev, gxdso_is_sep_supporting_additive_device, NULL, 0) <= 0
699
5.27k
        ) {
700
5.27k
        pcinfo->opmsupported = GX_CINFO_OPMSUPPORTED_NOT;
701
5.27k
        return;
702
5.27k
    }
703
704
    /* check for the appropriate components */
705
0
    if ( ncomps < 3                                      ||
706
0
         (red_c = dev_proc(dev, get_color_comp_index)(
707
0
                       dev,
708
0
                       "Red",
709
0
                       sizeof("Red") - 1,
710
0
                       NO_COMP_NAME_TYPE_OP)) < 0        ||
711
0
         red_c == GX_DEVICE_COLOR_MAX_COMPONENTS         ||
712
0
         (green_c = dev_proc(dev, get_color_comp_index)(
713
0
                          dev,
714
0
                          "Green",
715
0
                          sizeof("Green") - 1,
716
0
                          NO_COMP_NAME_TYPE_OP)) < 0     ||
717
0
         green_c == GX_DEVICE_COLOR_MAX_COMPONENTS       ||
718
0
         (blue_c = dev_proc(dev, get_color_comp_index)(
719
0
                        dev,
720
0
                        "Blue",
721
0
                        sizeof("Blue") - 1,
722
0
                        NO_COMP_NAME_TYPE_OP)) < 0       ||
723
0
         blue_c == GX_DEVICE_COLOR_MAX_COMPONENTS)
724
0
    {
725
0
        pcinfo->opmsupported = GX_CINFO_OPMSUPPORTED_NOT;
726
0
        return;
727
0
    }
728
729
    /* check the mapping */
730
0
    cmprocs = dev_proc(dev, get_color_mapping_procs)(dev, &cmdev);
731
732
0
    ncomps -= device_encodes_tags(dev);
733
0
    cmprocs->map_rgb(cmdev, NULL, frac_14, frac_0, frac_0, out);
734
0
    if (!check_single_comp(red_c, frac_14, ncomps, out)) {
735
0
        pcinfo->opmsupported = GX_CINFO_OPMSUPPORTED_NOT;
736
0
        return;
737
0
    }
738
0
    cmprocs->map_rgb(cmdev, NULL, frac_0, frac_14, frac_0, out);
739
0
    if (!check_single_comp(green_c, frac_14, ncomps, out)) {
740
0
        pcinfo->opmsupported = GX_CINFO_OPMSUPPORTED_NOT;
741
0
        return;
742
0
    }
743
0
    cmprocs->map_rgb(cmdev, NULL, frac_0, frac_0, frac_14, out);
744
0
    if (!check_single_comp(blue_c, frac_14, ncomps, out)) {
745
0
        pcinfo->opmsupported = GX_CINFO_OPMSUPPORTED_NOT;
746
0
        return;
747
0
    }
748
749
0
    process_comps =  ((gx_color_index)1 << red_c)
750
0
                   | ((gx_color_index)1 << green_c)
751
0
                   | ((gx_color_index)1 << blue_c);
752
0
    pcinfo->opmsupported = GX_CINFO_OPMSUPPORTED;
753
0
    pcinfo->process_comps = process_comps;
754
0
    pcinfo->black_component = 0; /* ? */
755
0
}
756
757
void check_opmsupported(gx_device * dev)
758
6.21k
{
759
6.21k
    if (dev->color_info.polarity == GX_CINFO_POLARITY_ADDITIVE)
760
5.27k
        check_rgb_color_model_comps(dev);
761
943
    else
762
943
        (void)check_cmyk_color_model_comps(dev);
763
6.21k
}
764
765
/*
766
 * This set_overprint method is unique. If overprint is true, overprint
767
 * mode is set to 1, the process color model has DeviceCMYK behavior (see
768
 * the comment ahead of gx_is_cmyk_color_model above), and the device
769
 * color is set, the device color needs to be considered in setting up
770
 * the set of drawn components.
771
 */
772
static int
773
gx_set_overprint_DeviceCMYK(const gs_color_space * pcs, gs_gstate * pgs)
774
0
{
775
0
    gx_device *             dev = pgs->device;
776
0
    gx_device_color_info *  pcinfo = (dev == 0 ? 0 : &dev->color_info);
777
778
    /* check if we require special handling */
779
0
    if ( !pgs->overprint                      ||
780
0
         pgs->overprint_mode != 1             ||
781
0
         pcinfo == 0                          ||
782
0
         pcinfo->opmsupported == GX_CINFO_OPMSUPPORTED_NOT)
783
0
        return gx_spot_colors_set_overprint(pcs, pgs);
784
    /* Share code with CMYK ICC case */
785
0
    return gx_set_overprint_cmyk(pcs, pgs);
786
0
}
787
788
/* A few comments about ICC profiles and overprint simulation.  In order
789
   to do proper overprint simulation, the source ICC profile and the
790
   destination ICC profile must be the same.  If they are not, then
791
   we end up mapping the source CMYK data to a different CMYK value.  In
792
   this case, the non-zero components, which with overprint mode = 1 specify
793
   which are to be overprinted will not be correct to produce the proper
794
   overprint simulation.  This is seen with AR when doing output preview,
795
   overprint simulation enabled of the file overprint_icc.pdf (see our
796
   test files) which has SWOP ICC based CMYK fills.  In AR, if we use a
797
   simulation ICC profile that is different than the source profile,
798
   overprinting is no longer previewed. We follow the same logic here.
799
   If the source and destination ICC profiles do not match, then there is
800
   effectively no overprinting enabled.  This is bug 692433.  However,
801
   even with the mismatch, if the device supports spot colorants, those
802
   colors should be maintained. This is bug 702725. */
803
int gx_set_overprint_cmyk(const gs_color_space * pcs, gs_gstate * pgs)
804
1.29k
{
805
1.29k
    gx_device *             dev = pgs->device;
806
1.29k
    gx_color_index          drawn_comps = 0;
807
1.29k
    gs_overprint_params_t   params = { 0 };
808
1.29k
    gx_device_color        *pdc;
809
1.29k
    cmm_dev_profile_t      *dev_profile;
810
1.29k
    cmm_profile_t          *output_profile = 0;
811
1.29k
    int                     code;
812
1.29k
    bool                    profile_ok = false;
813
1.29k
    gsicc_rendering_param_t        render_cond;
814
1.29k
    bool                    eop;
815
816
1.29k
    if_debug0m(gs_debug_flag_overprint, pgs->memory,
817
1.29k
        "[overprint] gx_set_overprint_cmyk\n");
818
819
1.29k
    if (dev) {
820
1.29k
        code = dev_proc(dev, get_profile)(dev, &dev_profile);
821
1.29k
        if (code < 0)
822
0
            return code;
823
824
1.29k
        gsicc_extract_profile(dev->graphics_type_tag, dev_profile, &(output_profile),
825
1.29k
                              &render_cond);
826
827
1.29k
        drawn_comps = gx_get_process_comps(dev);
828
1.29k
    }
829
830
1.29k
    if_debug1m(gs_debug_flag_overprint, pgs->memory,
831
1.29k
        "[overprint] gx_set_overprint_cmyk. drawn_comps = 0x%x\n", (uint)drawn_comps);
832
833
1.29k
    if (drawn_comps == 0)
834
0
        return gx_spot_colors_set_overprint(pcs, pgs);
835
836
    /* correct for any zero'ed color components.  But only if profiles
837
       match AND pgs->overprint_mode is true */
838
1.29k
    if (pcs->cmm_icc_profile_data != NULL && output_profile != NULL) {
839
1.29k
        if (gsicc_profiles_equal(output_profile, pcs->cmm_icc_profile_data)) {
840
1.12k
            profile_ok = true;
841
1.12k
        }
842
1.29k
    }
843
844
1.29k
    eop = gs_currentcolor_eopm(pgs);
845
846
1.29k
    if_debug3m(gs_debug_flag_overprint, pgs->memory,
847
1.29k
        "[overprint] gx_set_overprint_cmyk. is_fill_color = %d, pgs->color[0].effective_opm = %d pgs->color[1].effective_opm = %d\n",
848
1.29k
        pgs->is_fill_color, pgs->color[0].effective_opm, pgs->color[1].effective_opm);
849
850
1.29k
    if (profile_ok && eop) {
851
1.07k
        gx_color_index  nz_comps, one, temp;
852
1.07k
        int             code;
853
1.07k
        int             num_colorant[4], k;
854
1.07k
        bool            colorant_ok;
855
1.07k
        dev_color_proc_get_nonzero_comps((*procp));
856
857
1.07k
        if_debug0m(gs_debug_flag_overprint, pgs->memory,
858
1.07k
            "[overprint] gx_set_overprint_cmyk. color_is_set, profile_ok and eop\n");
859
860
1.07k
        code = gx_set_dev_color(pgs);
861
1.07k
        if (code < 0)
862
0
            return code;
863
1.07k
        pdc = gs_currentdevicecolor_inline(pgs);
864
1.07k
        procp = pdc->type->get_nonzero_comps;
865
1.07k
        if (pdc->ccolor_valid) {
866
            /* If we have the source colors, then use those in making the
867
               decision as to which ones are non-zero.  Then we avoid
868
               accidently looking at small values that get quantized to zero
869
               Note that to get here in the code, the source color data color
870
               space has to be CMYK. Trick is that we do need to worry about
871
               the colorant order on the target device */
872
1.07k
            num_colorant[0] = (dev_proc(dev, get_color_comp_index))\
873
1.07k
                             (dev, "Cyan", strlen("Cyan"), NO_COMP_NAME_TYPE_OP);
874
1.07k
            num_colorant[1] = (dev_proc(dev, get_color_comp_index))\
875
1.07k
                             (dev, "Magenta", strlen("Magenta"), NO_COMP_NAME_TYPE_OP);
876
1.07k
            num_colorant[2] = (dev_proc(dev, get_color_comp_index))\
877
1.07k
                             (dev, "Yellow", strlen("Yellow"), NO_COMP_NAME_TYPE_OP);
878
1.07k
            num_colorant[3] = (dev_proc(dev, get_color_comp_index))\
879
1.07k
                             (dev, "Black", strlen("Black"), NO_COMP_NAME_TYPE_OP);
880
1.07k
            nz_comps = 0;
881
1.07k
            one = 1;
882
1.07k
            colorant_ok = true;
883
5.37k
            for (k = 0; k < 4; k++) {
884
                /* Note: AR assumes the value is zero if it
885
                   is less than 0.5 out of 255 */
886
4.29k
                if (pdc->ccolor.paint.values[k] > (0.5 / 255.0)) {
887
1.08k
                    if (num_colorant[k] == -1) {
888
0
                        colorant_ok = false;
889
1.08k
                    } else {
890
1.08k
                        temp = one << num_colorant[k];
891
1.08k
                        nz_comps = nz_comps | temp;
892
1.08k
                    }
893
1.08k
                }
894
4.29k
            }
895
            /* For some reason we don't have one of the standard colorants */
896
1.07k
            if (!colorant_ok) {
897
0
                if ((code = procp(pdc, dev, &nz_comps)) < 0)
898
0
                    return code;
899
0
            }
900
1.07k
        } else {
901
0
            if ((code = procp(pdc, dev, &nz_comps)) < 0)
902
0
                return code;
903
0
        }
904
1.07k
        drawn_comps &= nz_comps;
905
1.07k
    }
906
1.29k
    params.is_fill_color = pgs->is_fill_color;
907
1.29k
    params.retain_any_comps = true;
908
1.29k
    params.drawn_comps = drawn_comps;
909
1.29k
    params.op_state = OP_STATE_NONE;
910
911
1.29k
    if_debug2m(gs_debug_flag_overprint, pgs->memory,
912
1.29k
        "[overprint] gx_set_overprint_cmyk. retain_any_comps = %d, drawn_comps = 0x%x\n",
913
1.29k
        params.retain_any_comps, (uint)(params.drawn_comps));
914
915
    /* We are in CMYK, the profiles match and overprint is true.  Set effective
916
       overprint mode to overprint mode but only if effective has not already
917
       been set to 0 */
918
1.29k
    params.effective_opm = pgs->color[0].effective_opm =
919
1.29k
        pgs->overprint_mode && gs_currentcolor_eopm(pgs);
920
1.29k
    return gs_gstate_update_overprint(pgs, &params);
921
1.29k
}
922
923
int gx_set_overprint_rgb(const gs_color_space * pcs, gs_gstate * pgs)
924
0
{
925
0
    gx_device *             dev = pgs->device;
926
0
    gx_color_index          drawn_comps = 0;
927
0
    gs_overprint_params_t   params = { 0 };
928
0
    gx_device_color        *pdc;
929
0
    cmm_dev_profile_t      *dev_profile;
930
0
    cmm_profile_t          *output_profile = 0;
931
0
    int                     code;
932
0
    bool                    profile_ok = false;
933
0
    gsicc_rendering_param_t        render_cond;
934
0
    bool                    eop;
935
936
0
    if_debug0m(gs_debug_flag_overprint, pgs->memory,
937
0
        "[overprint] gx_set_overprint_rgb\n");
938
939
0
    if (dev) {
940
0
        code = dev_proc(dev, get_profile)(dev, &dev_profile);
941
0
        if (code < 0)
942
0
            return code;
943
944
0
        gsicc_extract_profile(dev->graphics_type_tag, dev_profile, &(output_profile),
945
0
                              &render_cond);
946
947
0
        drawn_comps = gx_get_process_comps(dev);
948
0
    }
949
950
0
    if_debug1m(gs_debug_flag_overprint, pgs->memory,
951
0
        "[overprint] gx_set_overprint_rgb. drawn_comps = 0x%x\n", (uint)drawn_comps);
952
953
0
    if (drawn_comps == 0)
954
0
        return gx_spot_colors_set_overprint(pcs, pgs);
955
956
    /* correct for any zero'ed color components.  But only if profiles
957
       match AND pgs->overprint_mode is true */
958
0
    if (pcs->cmm_icc_profile_data != NULL && output_profile != NULL) {
959
0
        if (gsicc_profiles_equal(output_profile, pcs->cmm_icc_profile_data)) {
960
0
            profile_ok = true;
961
0
        }
962
0
    }
963
964
0
    eop = gs_currentcolor_eopm(pgs);
965
966
0
    if_debug3m(gs_debug_flag_overprint, pgs->memory,
967
0
        "[overprint] gx_set_overprint_rgb. is_fill_color = %d, pgs->color[0].effective_opm = %d pgs->color[1].effective_opm = %d\n",
968
0
        pgs->is_fill_color, pgs->color[0].effective_opm, pgs->color[1].effective_opm);
969
970
0
    if (profile_ok && eop) {
971
0
        gx_color_index  nz_comps, one, temp;
972
0
        int             code;
973
0
        int             num_colorant[3], k;
974
0
        bool            colorant_ok;
975
0
        dev_color_proc_get_nonzero_comps((*procp));
976
977
0
        if_debug0m(gs_debug_flag_overprint, pgs->memory,
978
0
            "[overprint] gx_set_overprint_cmyk. color_is_set, profile_ok and eop\n");
979
980
0
        code = gx_set_dev_color(pgs);
981
0
        if (code < 0)
982
0
            return code;
983
0
        pdc = gs_currentdevicecolor_inline(pgs);
984
0
        procp = pdc->type->get_nonzero_comps;
985
0
        if (pdc->ccolor_valid) {
986
            /* If we have the source colors, then use those in making the
987
               decision as to which ones are non-zero.  Then we avoid
988
               accidently looking at small values that get quantized to zero
989
               Note that to get here in the code, the source color data color
990
               space has to be CMYK. Trick is that we do need to worry about
991
               the colorant order on the target device */
992
0
            num_colorant[0] = (dev_proc(dev, get_color_comp_index))\
993
0
                             (dev, "Red", strlen("Red"), NO_COMP_NAME_TYPE_OP);
994
0
            num_colorant[1] = (dev_proc(dev, get_color_comp_index))\
995
0
                             (dev, "Green", strlen("Green"), NO_COMP_NAME_TYPE_OP);
996
0
            num_colorant[2] = (dev_proc(dev, get_color_comp_index))\
997
0
                             (dev, "Blue", strlen("Blue"), NO_COMP_NAME_TYPE_OP);
998
0
            nz_comps = 0;
999
0
            one = 1;
1000
0
            colorant_ok = true;
1001
0
            for (k = 0; k < 3; k++) {
1002
                /* Note: AR assumes the value is zero if it
1003
                   is less than 0.5 out of 255 */
1004
0
                if (pdc->ccolor.paint.values[k] > (0.5 / 255.0)) {
1005
0
                    if (num_colorant[k] == -1) {
1006
0
                        colorant_ok = false;
1007
0
                    } else {
1008
0
                        temp = one << num_colorant[k];
1009
0
                        nz_comps = nz_comps | temp;
1010
0
                    }
1011
0
                }
1012
0
            }
1013
            /* For some reason we don't have one of the standard colorants */
1014
0
            if (!colorant_ok) {
1015
0
                if ((code = procp(pdc, dev, &nz_comps)) < 0)
1016
0
                    return code;
1017
0
            }
1018
0
        } else {
1019
0
            if ((code = procp(pdc, dev, &nz_comps)) < 0)
1020
0
                return code;
1021
0
        }
1022
0
        drawn_comps &= nz_comps;
1023
0
    }
1024
0
    params.is_fill_color = pgs->is_fill_color;
1025
0
    params.retain_any_comps = true;
1026
0
    params.drawn_comps = drawn_comps;
1027
0
    params.op_state = OP_STATE_NONE;
1028
1029
0
    if_debug2m(gs_debug_flag_overprint, pgs->memory,
1030
0
        "[overprint] gx_set_overprint_rgb. retain_any_comps = %d, drawn_comps = 0x%x\n",
1031
0
        params.retain_any_comps, (uint)(params.drawn_comps));
1032
1033
    /* We are in RGB, the profiles match and overprint is true.  Set effective
1034
       overprint mode to overprint mode but only if effective has not already
1035
       been set to 0 */
1036
0
    params.effective_opm = pgs->color[0].effective_opm =
1037
0
        pgs->overprint_mode && gs_currentcolor_eopm(pgs);
1038
0
    return gs_gstate_update_overprint(pgs, &params);
1039
0
}
1040
/* A stub for a color mapping linearity check, when it is inapplicable. */
1041
int
1042
gx_cspace_no_linear(const gs_color_space *cs, const gs_gstate * pgs,
1043
                gx_device * dev,
1044
                const gs_client_color *c0, const gs_client_color *c1,
1045
                const gs_client_color *c2, const gs_client_color *c3,
1046
                float smoothness, gsicc_link_t *icclink)
1047
0
{
1048
0
    return_error(gs_error_rangecheck);
1049
0
}
1050
1051
static inline int
1052
cc2dc(const gs_color_space *cs, const gs_gstate * pgs, gx_device *dev,
1053
            gx_device_color *dc, const gs_client_color *cc)
1054
2.07M
{
1055
2.07M
    return cs->type->remap_color(cc, cs, dc, pgs, dev, gs_color_select_texture);
1056
2.07M
}
1057
1058
static inline void
1059
interpolate_cc(gs_client_color *c,
1060
        const gs_client_color *c0, const gs_client_color *c1, double t, int n)
1061
1.09M
{
1062
1.09M
    int i;
1063
1064
2.41M
    for (i = 0; i < n; i++)
1065
1.32M
        c->paint.values[i] = c0->paint.values[i] * t + c1->paint.values[i] * (1 - t);
1066
1.09M
}
1067
1068
static inline bool
1069
is_dc_nearly_linear(const gx_device *dev, const gx_device_color *c,
1070
        const gx_device_color *c0, const gx_device_color *c1,
1071
        double t, uchar n, float smoothness)
1072
1.09M
{
1073
1.09M
    uchar i;
1074
1075
1.09M
    if (c0->type == &gx_dc_type_data_pure) {
1076
698k
        gx_color_index pure0 = c0->colors.pure;
1077
698k
        gx_color_index pure1 = c1->colors.pure;
1078
698k
        gx_color_index pure = c->colors.pure;
1079
1080
3.34M
        for (i = 0; i < n; i++) {
1081
2.66M
            int shift = dev->color_info.comp_shift[i];
1082
2.66M
            int mask = (1 << dev->color_info.comp_bits[i]) - 1;
1083
2.66M
            int max_color = (i == dev->color_info.gray_index ? dev->color_info.max_gray
1084
2.66M
                                                             : dev->color_info.max_color);
1085
2.66M
            float max_diff = max(1, max_color * smoothness);
1086
2.66M
            int b0 = (pure0 >> shift) & mask, b1 = (pure1 >> shift) & mask;
1087
2.66M
            int b = (pure >> shift) & mask;
1088
2.66M
            double bb = b0 * t + b1 * (1 - t);
1089
1090
2.66M
            if (any_abs(b - bb) > max_diff)
1091
11.4k
                return false;
1092
2.66M
        }
1093
686k
        return true;
1094
698k
    } else if (c0->type == &gx_dc_type_data_devn) {
1095
1.95M
        for (i = 0; i < n; i++) {
1096
1.57M
            int max_color = (i == dev->color_info.gray_index ? dev->color_info.max_gray
1097
1.57M
                : dev->color_info.max_color);
1098
1.57M
            double max_diff = max(1, max_color * smoothness);
1099
            /* Color values are 16 bit.  We are basing the smoothness on the
1100
               device bit depth.  So make sure to adjust the above max diff
1101
               based upon our device bit depth */
1102
1.57M
            double ratio = (double)max_color / (double)gx_max_color_value;
1103
1.57M
            double b0 = (c0->colors.devn.values[i]) * ratio;
1104
1.57M
            double b1 = (c1->colors.devn.values[i]) * ratio;
1105
1.57M
            double b = (c->colors.devn.values[i]) * ratio;
1106
1.57M
            double bb = b0 * t + b1 * (1 - t);
1107
1.57M
            if (any_abs(b - bb) > max_diff)
1108
4.32k
                return false;
1109
1.57M
        }
1110
388k
        return true;
1111
392k
    } else {
1112
        /* Halftones must not paint with fill_linear_color_*. */
1113
0
        return false;
1114
0
    }
1115
1.09M
}
1116
1117
/* Default color mapping linearity check, a 2-points case. */
1118
static int
1119
gx_cspace_is_linear_in_line(const gs_color_space *cs, const gs_gstate * pgs,
1120
                gx_device *dev,
1121
                const gs_client_color *c0, const gs_client_color *c1,
1122
                float smoothness)
1123
317k
{
1124
317k
    gs_client_color c01a, c01b;
1125
317k
    gx_device_color d[2], d01a, d01b;
1126
317k
    int n = cs->type->num_components(cs);
1127
317k
    uchar ndev = dev->color_info.num_components;
1128
317k
    int code;
1129
1130
317k
    code = cc2dc(cs, pgs, dev, &d[0], c0);
1131
317k
    if (code < 0)
1132
0
        return code;
1133
317k
    code = cc2dc(cs, pgs, dev, &d[1], c1);
1134
317k
    if (code < 0)
1135
0
        return code;
1136
317k
    interpolate_cc(&c01a, c0, c1, 0.3, n);
1137
317k
    code = cc2dc(cs, pgs, dev, &d01a, &c01a);
1138
317k
    if (code < 0)
1139
0
        return code;
1140
317k
    if (!is_dc_nearly_linear(dev, &d01a, &d[0], &d[1], 0.3, ndev, smoothness))
1141
13.6k
        return 0;
1142
303k
    interpolate_cc(&c01b, c0, c1, 0.7, n);
1143
303k
    code = cc2dc(cs, pgs, dev, &d01b, &c01b);
1144
303k
    if (code < 0)
1145
0
        return code;
1146
303k
    if (!is_dc_nearly_linear(dev, &d01b, &d[0], &d[1], 0.7, ndev, smoothness))
1147
1.64k
        return 0;
1148
301k
    return 1;
1149
303k
}
1150
1151
/* Default color mapping linearity check, a triangle case. */
1152
static int
1153
gx_cspace_is_linear_in_triangle(const gs_color_space *cs, const gs_gstate * pgs,
1154
                gx_device *dev,
1155
                const gs_client_color *c0, const gs_client_color *c1,
1156
                const gs_client_color *c2, float smoothness)
1157
117k
{
1158
    /* We check 4 points - the median center, and middle points of 3 sides.
1159
       Hopely this is enough for reasonable color spaces and color renderings.
1160
       Note it gives 7 points for a quadrangle. */
1161
117k
    gs_client_color c01, c12, c20, c012;
1162
117k
    gx_device_color d[3], d01, d12, d20, d012;
1163
1164
    /* Note that the device and the client color space
1165
       can have a different number of components */
1166
1167
117k
    int n = cs->type->num_components(cs);
1168
117k
    uchar ndev = dev->color_info.num_components;
1169
1170
117k
    int code;
1171
1172
117k
    code = cc2dc(cs, pgs, dev, &d[0], c0);
1173
117k
    if (code < 0)
1174
0
        return code;
1175
117k
    code = cc2dc(cs, pgs, dev, &d[1], c1);
1176
117k
    if (code < 0)
1177
0
        return code;
1178
117k
    code = cc2dc(cs, pgs, dev, &d[2], c2);
1179
117k
    if (code < 0)
1180
0
        return code;
1181
1182
117k
    interpolate_cc(&c01, c0, c1, 0.5, n);
1183
117k
    code = cc2dc(cs, pgs, dev, &d01, &c01);
1184
117k
    if (code < 0)
1185
0
        return code;
1186
117k
    if (!is_dc_nearly_linear(dev, &d01, &d[0], &d[1], 0.5, ndev, smoothness))
1187
270
        return 0;
1188
1189
117k
    interpolate_cc(&c012, c2, &c01, 2.0 / 3, n);
1190
117k
    code = cc2dc(cs, pgs, dev, &d012, &c012);
1191
117k
    if (code < 0)
1192
0
        return code;
1193
117k
    if (!is_dc_nearly_linear(dev, &d012, &d[2], &d01, 2.0 / 3, ndev, smoothness))
1194
172
        return 0;
1195
1196
117k
    interpolate_cc(&c12, c1, c2, 0.5, n);
1197
117k
    code = cc2dc(cs, pgs, dev, &d12, &c12);
1198
117k
    if (code < 0)
1199
0
        return code;
1200
117k
    if (!is_dc_nearly_linear(dev, &d12, &d[1], &d[2], 0.5, ndev, smoothness))
1201
64
        return 0;
1202
1203
117k
    interpolate_cc(&c20, c2, c0, 0.5, n);
1204
117k
    code = cc2dc(cs, pgs, dev, &d20, &c20);
1205
117k
    if (code < 0)
1206
0
        return code;
1207
117k
    if (!is_dc_nearly_linear(dev, &d20, &d[2], &d[0], 0.5, ndev, smoothness))
1208
0
        return 0;
1209
117k
    return 1;
1210
117k
}
1211
1212
/* Default color mapping linearity check. */
1213
int
1214
gx_cspace_is_linear_default(const gs_color_space *cs, const gs_gstate * pgs,
1215
                gx_device *dev,
1216
                const gs_client_color *c0, const gs_client_color *c1,
1217
                const gs_client_color *c2, const gs_client_color *c3,
1218
                float smoothness, gsicc_link_t *icclink)
1219
435k
{
1220
    /* Assuming 2 <= nc <= 4. We don't need other cases. */
1221
    /* With nc == 4 assuming a convex plain quadrangle in the client color space. */
1222
435k
    int code;
1223
1224
435k
    if (!colors_are_separable_and_linear(&dev->color_info))
1225
0
        return_error(gs_error_rangecheck);
1226
435k
    if (c2 == NULL)
1227
317k
        return gx_cspace_is_linear_in_line(cs, pgs, dev, c0, c1, smoothness);
1228
117k
    code = gx_cspace_is_linear_in_triangle(cs, pgs, dev, c0, c1, c2, smoothness);
1229
117k
    if (code <= 0)
1230
506
        return code;
1231
117k
    if (c3 == NULL)
1232
117k
        return 1;
1233
0
    return gx_cspace_is_linear_in_triangle(cs, pgs, dev, c1, c2, c3, smoothness);
1234
117k
}
1235
1236
/* Serialization. */
1237
int
1238
gx_serialize_cspace_type(const gs_color_space * pcs, stream * s)
1239
56.4k
{
1240
56.4k
    const gs_color_space_type * type = pcs->type;
1241
56.4k
    uint n;
1242
56.4k
    return sputs(s, (const byte *)&type->index, sizeof(type->index), &n);
1243
56.4k
}
1244
1245
/* GC procedures */
1246
1247
static
1248
ENUM_PTRS_BEGIN_PROC(color_space_enum_ptrs)
1249
8.53M
{
1250
8.53M
    EV_CONST gs_color_space *pcs = vptr;
1251
1252
8.53M
    if (index == 0)
1253
1.70M
        return ENUM_OBJ(pcs->base_space);
1254
6.82M
    if (index == 1)
1255
1.70M
        return ENUM_OBJ(pcs->pclient_color_space_data);
1256
5.11M
    if (index == 2)
1257
1.70M
        return ENUM_OBJ(pcs->icc_equivalent);
1258
3.41M
    if (index == 3) {
1259
1.70M
        if (gs_color_space_get_index(pcs) == gs_color_space_index_DeviceN)
1260
0
            return ENUM_OBJ(pcs->params.device_n.devn_process_space);
1261
1.70M
        else
1262
1.70M
            return ENUM_OBJ(NULL);
1263
1.70M
    }
1264
1265
1.70M
    return ENUM_USING(*pcs->type->stype, vptr, size, index - 4);
1266
3.41M
    ENUM_PTRS_END_PROC
1267
3.41M
}
1268
static
1269
1.70M
RELOC_PTRS_WITH(color_space_reloc_ptrs, gs_color_space *pcs)
1270
1.70M
{
1271
1.70M
    RELOC_VAR(pcs->base_space);
1272
1.70M
    RELOC_VAR(pcs->pclient_color_space_data);
1273
1.70M
    RELOC_VAR(pcs->icc_equivalent);
1274
1.70M
    if (gs_color_space_get_index(pcs) == gs_color_space_index_DeviceN)
1275
1.70M
        RELOC_VAR(pcs->params.device_n.devn_process_space);
1276
1.70M
    RELOC_USING(*pcs->type->stype, vptr, size);
1277
1.70M
}
1278
1.70M
RELOC_PTRS_END