Coverage Report

Created: 2026-04-01 07:17

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
31.2M
{
101
31.2M
    gs_color_space *pcs = (gs_color_space *)vptr;
102
31.2M
    (void)cmem; /* unused */
103
104
31.2M
    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
31.2M
    if (pcs->type->final)
109
17.2M
        pcs->type->final(pcs);
110
31.2M
    if_debug2m('c', cmem, "[c]cspace final "PRI_INTPTR" %d\n", (intptr_t)pcs, (int)pcs->id);
111
31.2M
    rc_decrement_only_cs(pcs->base_space, "gs_cspace_final");
112
31.2M
    pcs->base_space = NULL;
113
31.2M
    if (gs_color_space_get_index(pcs) == gs_color_space_index_DeviceN) {
114
3.10k
        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
3.10k
    }
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
31.2M
}
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
31.2M
{
128
31.2M
    gs_color_space *pcs;
129
130
31.2M
    rc_alloc_struct_1(pcs, gs_color_space, &st_color_space, mem, return NULL,
131
31.2M
                      "gs_cspace_alloc_with_id");
132
31.2M
    if_debug3m('c', mem, "[c]cspace alloc "PRI_INTPTR" %s %d\n",
133
31.2M
               (intptr_t)pcs, pcstype->stype->sname, pcstype->index);
134
31.2M
    pcs->type = pcstype;
135
31.2M
    pcs->id = id;
136
31.2M
    pcs->base_space = NULL;
137
31.2M
    pcs->pclient_color_space_data = NULL;
138
31.2M
    pcs->interpreter_data = NULL;
139
31.2M
    pcs->interpreter_free_cspace_proc = NULL;
140
31.2M
    pcs->cmm_icc_profile_data = NULL;
141
31.2M
    pcs->ICC_Alternate_space = gs_ICC_Alternate_None;
142
31.2M
    pcs->icc_equivalent = NULL;
143
31.2M
    pcs->params.device_n.devn_process_space = NULL;
144
31.2M
    pcs->params.device_n.all_none = false;
145
31.2M
    return pcs;
146
31.2M
}
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.95M
{
159
3.95M
    return gs_cspace_alloc_with_id(mem, gs_next_ids(mem, 1), pcstype);
160
3.95M
}
161
162
/* Constructors for simple device color spaces. */
163
164
gs_color_space *
165
gs_cspace_new_DeviceGray(gs_memory_t *mem)
166
18.3M
{
167
18.3M
    return gs_cspace_alloc_with_id(mem, cs_DeviceGray_id,
168
18.3M
                                   &gs_color_space_type_DeviceGray);
169
18.3M
}
170
171
gs_color_space *
172
gs_cspace_new_DeviceRGB(gs_memory_t *mem)
173
6.13M
{
174
6.13M
    return gs_cspace_alloc_with_id(mem, cs_DeviceRGB_id,
175
6.13M
                                   &gs_color_space_type_DeviceRGB);
176
6.13M
}
177
gs_color_space *
178
gs_cspace_new_DeviceCMYK(gs_memory_t *mem)
179
2.79M
{
180
2.79M
    return gs_cspace_alloc_with_id(mem, cs_DeviceCMYK_id,
181
2.79M
                                   &gs_color_space_type_DeviceCMYK);
182
2.79M
}
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.09k
{
188
8.09k
    gs_color_space *pcspace = gs_cspace_alloc(pmem, &gs_color_space_type_ICC);
189
8.09k
    cmm_profile_t *profile;
190
8.09k
    stream *str;
191
8.09k
    int code;
192
193
8.09k
    if (pcspace == NULL)
194
0
        return pcspace;
195
196
8.09k
    code = gsicc_open_search(SCRGB, strlen(SCRGB), pmem, pmem->gs_lib_ctx->profiledir,
197
8.09k
        pmem->gs_lib_ctx->profiledir_len, &str);
198
199
8.09k
    if (code < 0 || str == NULL) {
200
0
        rc_decrement(pcspace, "gs_cspace_new_scrgb");
201
0
        return NULL;
202
0
    }
203
204
8.09k
    pcspace->cmm_icc_profile_data = gsicc_profile_new(str, pmem, SCRGB, strlen(SCRGB));
205
8.09k
    code = sfclose(str);
206
8.09k
    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.09k
    pcspace->cmm_icc_profile_data->profile_handle =
213
8.09k
        gsicc_get_profile_handle_buffer(pcspace->cmm_icc_profile_data->buffer,
214
8.09k
            pcspace->cmm_icc_profile_data->buffer_size, pmem);
215
8.09k
    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.09k
    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.09k
    gsicc_get_icc_buff_hash(profile->buffer, &(profile->hashcode),
224
8.09k
        profile->buffer_size);
225
8.09k
    profile->hash_is_valid = true;
226
8.09k
    profile->num_comps =
227
8.09k
        gscms_get_input_channel_count(profile->profile_handle, profile->memory);
228
8.09k
    if (profile->num_comps > ICC_MAX_CHANNELS) {
229
0
        rc_decrement(pcspace, "gs_cspace_new_scrgb");
230
0
        return NULL;
231
0
    }
232
8.09k
    profile->num_comps_out =
233
8.09k
        gscms_get_output_channel_count(profile->profile_handle, profile->memory);
234
8.09k
    profile->data_cs =
235
8.09k
        gscms_get_profile_data_space(profile->profile_handle, profile->memory);
236
8.09k
    gsicc_set_icc_range(&profile);
237
8.09k
    return pcspace;
238
8.09k
}
239
240
gs_color_space *
241
gs_cspace_new_ICC(gs_memory_t *pmem, gs_gstate * pgs, int components)
242
2.68M
{
243
2.68M
    gsicc_manager_t *icc_manage = pgs->icc_manager;
244
2.68M
    int code = 0;
245
2.68M
    gs_color_space *pcspace = gs_cspace_alloc(pmem, &gs_color_space_type_ICC);
246
247
2.68M
    if (pcspace == NULL)
248
0
        return pcspace;
249
250
2.68M
    switch (components) {
251
8.09k
        case -1: /* alpha case */
252
8.09k
            if (icc_manage->smask_profiles == NULL) {
253
8.09k
                code = gsicc_initialize_iccsmask(icc_manage);
254
8.09k
            }
255
8.09k
            if (code == 0) {
256
8.09k
                pcspace->cmm_icc_profile_data =
257
8.09k
                    icc_manage->smask_profiles->smask_gray;
258
8.09k
            } else {
259
0
                pcspace->cmm_icc_profile_data = icc_manage->default_gray;
260
0
            }
261
8.09k
            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.56M
        case 1: pcspace->cmm_icc_profile_data = icc_manage->default_gray; break;
274
56.6k
        case 3: pcspace->cmm_icc_profile_data = icc_manage->default_rgb; break;
275
48.5k
        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.68M
    }
278
2.68M
    gsicc_adjust_profile_rc(pcspace->cmm_icc_profile_data, 1, "gs_cspace_new_ICC");
279
2.68M
    return pcspace;
280
2.68M
}
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
82.3M
{
288
82.3M
    return pcs->type->index;
289
82.3M
}
290
291
/* See if the space is CIE based */
292
bool gs_color_space_is_CIE(const gs_color_space * pcs)
293
1.05M
{
294
1.05M
    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
1.05M
        case gs_color_space_index_ICC:
300
1.05M
            return true;
301
0
        break;
302
614
        default:
303
614
            return false;
304
1.05M
    }
305
1.05M
}
306
307
/* See if the space is Postscript CIE based */
308
bool gs_color_space_is_PSCIE(const gs_color_space * pcs)
309
32.7M
{
310
32.7M
    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
32.7M
        default:
318
32.7M
            return false;
319
32.7M
}
320
32.7M
}
321
322
/* See if the space is ICC based */
323
bool gs_color_space_is_ICC(const gs_color_space * pcs)
324
2.13M
{
325
2.13M
    return(gs_color_space_get_index(pcs) == gs_color_space_index_ICC);
326
2.13M
}
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
13.2M
{
332
13.2M
    return cs_num_components(pcs);
333
13.2M
}
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.21M
        int code = gsicc_init_iccmanager(pgs);
353
2.21M
        if (code < 0)
354
0
            return code;
355
2.21M
    }
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
17.8M
{
367
17.8M
    return 1;
368
17.8M
}
369
int
370
gx_num_components_3(const gs_color_space * pcs)
371
21.8k
{
372
21.8k
    return 3;
373
21.8k
}
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
189k
{
405
189k
    return pcspace->base_space;
406
189k
}
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
17.2M
{
420
17.2M
    rc_increment(pcs);
421
17.2M
}
422
423
5.90M
void rc_decrement_cs(gs_color_space *pcs, const char *cname) {
424
425
5.90M
    if (pcs) {
426
5.90M
        rc_decrement(pcs, cname);
427
5.90M
    }
428
5.90M
}
429
430
void rc_decrement_only_cs(gs_color_space *pcs, const char *cname)
431
61.6M
{
432
61.6M
    if (pcs) {
433
30.5M
        rc_decrement_only(pcs, cname);
434
30.5M
    }
435
61.6M
}
436
437
void cs_adjust_counts_icc(gs_gstate *pgs, int delta)
438
154M
{
439
154M
    gs_color_space *pcs = gs_currentcolorspace_inline(pgs);
440
441
154M
    if (pcs) {
442
102M
        cs_adjust_color_count(pgs, delta);
443
102M
        rc_adjust_const(pcs, delta, "cs_adjust_counts_icc");
444
102M
    }
445
154M
}
446
447
void cs_adjust_swappedcounts_icc(gs_gstate *pgs, int delta)
448
154M
{
449
154M
    gs_color_space *pcs = gs_swappedcolorspace_inline(pgs);
450
451
154M
    if (pcs) {
452
102M
        cs_adjust_swappedcolor_count(pgs, delta);
453
102M
        rc_adjust_const(pcs, delta, "cs_adjust_swappedcounts_icc");
454
102M
    }
455
154M
}
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.22M
{
470
    /* If we already have profile_data, nothing to do here. */
471
1.22M
    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.22M
    if (pgs->icc_manager->default_rgb == NULL)
476
0
        gsicc_init_iccmanager(pgs);
477
478
    /* pcs takes a reference to default_rgb */
479
1.22M
    pcs->cmm_icc_profile_data = pgs->icc_manager->default_rgb;
480
1.22M
    gsicc_adjust_profile_rc(pcs->cmm_icc_profile_data, 1, "gx_install_DeviceRGB");
481
1.22M
    pcs->type = &gs_color_space_type_ICC;
482
1.22M
    return 0;
483
1.22M
}
484
485
/* Install a DeviceCMYK color space. */
486
static int
487
gx_install_DeviceCMYK(gs_color_space * pcs, gs_gstate * pgs)
488
92.0k
{
489
    /* If we already have profile data, nothing to do here. */
490
92.0k
    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
92.0k
    if (pgs->icc_manager->default_cmyk == NULL)
495
0
        gsicc_init_iccmanager(pgs);
496
497
    /* pcs takes a reference to default_cmyk */
498
92.0k
    pcs->cmm_icc_profile_data = pgs->icc_manager->default_cmyk;
499
92.0k
    gsicc_adjust_profile_rc(pcs->cmm_icc_profile_data, 1, "gx_install_DeviceCMYK");
500
92.0k
    pcs->type = &gs_color_space_type_ICC;
501
92.0k
    return 0;
502
92.0k
}
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
519k
{
513
519k
    gs_overprint_params_t   params = { 0 };
514
515
519k
    params.retain_any_comps = false;
516
519k
    params.op_state = OP_STATE_NONE;
517
519k
    params.is_fill_color = pgs->is_fill_color;
518
519k
    params.effective_opm = pgs->color[0].effective_opm = 0;
519
520
519k
    return gs_gstate_update_overprint(pgs, &params);
521
519k
}
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.50k
{
530
2.50k
    gs_overprint_params_t   params = { 0 };
531
2.50k
    gx_device* dev = pgs->device;
532
2.50k
    gx_color_index drawn_comps = dev == NULL ? 0 : gx_get_process_comps(dev);
533
534
2.50k
    params.retain_any_comps = true;
535
2.50k
    params.op_state = OP_STATE_NONE;
536
2.50k
    params.is_fill_color = pgs->is_fill_color;
537
2.50k
    params.effective_opm = pgs->color[0].effective_opm = 0;
538
2.50k
    params.drawn_comps = drawn_comps;
539
540
2.50k
    return gs_gstate_update_overprint(pgs, &params);
541
2.50k
}
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
792
{
572
792
    int     i;
573
574
4.01k
    for (i = 0; i < ncomps; i++) {
575
3.22k
        if ( (i != comp && pval[i] != frac_0)  ||
576
3.22k
             (i == comp && pval[i] != targ_val)  )
577
0
            return false;
578
3.22k
    }
579
792
    return true;
580
792
}
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
1.12k
{
602
1.12k
    gx_device_color_info *          pcinfo = &dev->color_info;
603
1.12k
    uchar                           ncomps = pcinfo->num_components;
604
1.12k
    int                             cyan_c, magenta_c, yellow_c, black_c;
605
1.12k
    frac                            frac_14 = frac_1 / 4;
606
1.12k
    frac                            out[GX_DEVICE_COLOR_MAX_COMPONENTS];
607
1.12k
    gx_color_index                  process_comps;
608
1.12k
    const gx_cm_color_map_procs    *cmprocs;
609
1.12k
    const gx_device                *cmdev;
610
611
612
1.12k
    if (pcinfo->num_components < 4                     ||
613
208
        pcinfo->polarity == GX_CINFO_POLARITY_ADDITIVE ||
614
931
        pcinfo->gray_index == GX_CINFO_COMP_NO_INDEX) {
615
931
        pcinfo->opmsupported = GX_CINFO_OPMSUPPORTED_NOT;
616
931
        return 0;
617
931
    }
618
619
    /* check for the appropriate components */
620
198
    if ( ncomps < 4                                       ||
621
198
         (cyan_c = dev_proc(dev, get_color_comp_index)(
622
198
                       dev,
623
198
                       "Cyan",
624
198
                       sizeof("Cyan") - 1,
625
198
                       NO_COMP_NAME_TYPE_OP)) < 0           ||
626
198
         cyan_c == GX_DEVICE_COLOR_MAX_COMPONENTS         ||
627
198
         (magenta_c = dev_proc(dev, get_color_comp_index)(
628
198
                          dev,
629
198
                          "Magenta",
630
198
                          sizeof("Magenta") - 1,
631
198
                          NO_COMP_NAME_TYPE_OP)) < 0        ||
632
198
         magenta_c == GX_DEVICE_COLOR_MAX_COMPONENTS      ||
633
198
         (yellow_c = dev_proc(dev, get_color_comp_index)(
634
198
                        dev,
635
198
                        "Yellow",
636
198
                        sizeof("Yellow") - 1,
637
198
                        NO_COMP_NAME_TYPE_OP)) < 0               ||
638
198
         yellow_c == GX_DEVICE_COLOR_MAX_COMPONENTS       ||
639
198
         (black_c = dev_proc(dev, get_color_comp_index)(
640
198
                        dev,
641
198
                        "Black",
642
198
                        sizeof("Black") - 1,
643
198
                        NO_COMP_NAME_TYPE_OP)) < 0                         ||
644
198
         black_c == GX_DEVICE_COLOR_MAX_COMPONENTS          )
645
0
        return 0;
646
647
    /* check the mapping */
648
198
    cmprocs = dev_proc(dev, get_color_mapping_procs)(dev, &cmdev);
649
650
198
    ncomps -= device_encodes_tags(dev);
651
198
    cmprocs->map_cmyk(cmdev, frac_14, frac_0, frac_0, frac_0, out);
652
198
    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
198
    cmprocs->map_cmyk(cmdev, frac_0, frac_14, frac_0, frac_0, out);
657
198
    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
198
    cmprocs->map_cmyk(cmdev, frac_0, frac_0, frac_14, frac_0, out);
662
198
    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
198
    cmprocs->map_cmyk(cmdev, frac_0, frac_0, frac_0, frac_14, out);
667
198
    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
198
    process_comps =  ((gx_color_index)1 << cyan_c)
673
198
                   | ((gx_color_index)1 << magenta_c)
674
198
                   | ((gx_color_index)1 << yellow_c)
675
198
                   | ((gx_color_index)1 << black_c);
676
198
    pcinfo->opmsupported = GX_CINFO_OPMSUPPORTED;
677
198
    pcinfo->process_comps = process_comps;
678
198
    pcinfo->black_component = black_c;
679
198
    return process_comps;
680
198
}
681
682
static void
683
check_rgb_color_model_comps(gx_device * dev)
684
4.97k
{
685
4.97k
    gx_device_color_info *          pcinfo = &dev->color_info;
686
4.97k
    uchar                           ncomps = pcinfo->num_components;
687
4.97k
    int                             red_c, green_c, blue_c;
688
4.97k
    frac                            frac_14 = frac_1 / 4;
689
4.97k
    frac                            out[GX_DEVICE_COLOR_MAX_COMPONENTS];
690
4.97k
    gx_color_index                  process_comps;
691
4.97k
    const gx_cm_color_map_procs    *cmprocs;
692
4.97k
    const gx_device                *cmdev;
693
694
695
4.97k
    if (pcinfo->num_components < 3                     ||
696
3.62k
        pcinfo->polarity != GX_CINFO_POLARITY_ADDITIVE ||
697
3.62k
        pcinfo->gray_index == GX_CINFO_COMP_NO_INDEX   ||
698
1.62k
        dev_proc(dev, dev_spec_op)(dev, gxdso_is_sep_supporting_additive_device, NULL, 0) <= 0
699
4.97k
        ) {
700
4.97k
        pcinfo->opmsupported = GX_CINFO_OPMSUPPORTED_NOT;
701
4.97k
        return;
702
4.97k
    }
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.09k
{
759
6.09k
    if (dev->color_info.polarity == GX_CINFO_POLARITY_ADDITIVE)
760
4.97k
        check_rgb_color_model_comps(dev);
761
1.12k
    else
762
1.12k
        (void)check_cmyk_color_model_comps(dev);
763
6.09k
}
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.43k
{
805
1.43k
    gx_device *             dev = pgs->device;
806
1.43k
    gx_color_index          drawn_comps = 0;
807
1.43k
    gs_overprint_params_t   params = { 0 };
808
1.43k
    gx_device_color        *pdc;
809
1.43k
    cmm_dev_profile_t      *dev_profile;
810
1.43k
    cmm_profile_t          *output_profile = 0;
811
1.43k
    int                     code;
812
1.43k
    bool                    profile_ok = false;
813
1.43k
    gsicc_rendering_param_t        render_cond;
814
1.43k
    bool                    eop;
815
816
1.43k
    if_debug0m(gs_debug_flag_overprint, pgs->memory,
817
1.43k
        "[overprint] gx_set_overprint_cmyk\n");
818
819
1.43k
    if (dev) {
820
1.43k
        code = dev_proc(dev, get_profile)(dev, &dev_profile);
821
1.43k
        if (code < 0)
822
0
            return code;
823
824
1.43k
        gsicc_extract_profile(dev->graphics_type_tag, dev_profile, &(output_profile),
825
1.43k
                              &render_cond);
826
827
1.43k
        drawn_comps = gx_get_process_comps(dev);
828
1.43k
    }
829
830
1.43k
    if_debug1m(gs_debug_flag_overprint, pgs->memory,
831
1.43k
        "[overprint] gx_set_overprint_cmyk. drawn_comps = 0x%x\n", (uint)drawn_comps);
832
833
1.43k
    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.43k
    if (pcs->cmm_icc_profile_data != NULL && output_profile != NULL) {
839
1.43k
        if (gsicc_profiles_equal(output_profile, pcs->cmm_icc_profile_data)) {
840
1.25k
            profile_ok = true;
841
1.25k
        }
842
1.43k
    }
843
844
1.43k
    eop = gs_currentcolor_eopm(pgs);
845
846
1.43k
    if_debug3m(gs_debug_flag_overprint, pgs->memory,
847
1.43k
        "[overprint] gx_set_overprint_cmyk. is_fill_color = %d, pgs->color[0].effective_opm = %d pgs->color[1].effective_opm = %d\n",
848
1.43k
        pgs->is_fill_color, pgs->color[0].effective_opm, pgs->color[1].effective_opm);
849
850
1.43k
    if (profile_ok && eop) {
851
1.19k
        gx_color_index  nz_comps, one, temp;
852
1.19k
        int             code;
853
1.19k
        int             num_colorant[4], k;
854
1.19k
        bool            colorant_ok;
855
1.19k
        dev_color_proc_get_nonzero_comps((*procp));
856
857
1.19k
        if_debug0m(gs_debug_flag_overprint, pgs->memory,
858
1.19k
            "[overprint] gx_set_overprint_cmyk. color_is_set, profile_ok and eop\n");
859
860
1.19k
        code = gx_set_dev_color(pgs);
861
1.19k
        if (code < 0)
862
0
            return code;
863
1.19k
        pdc = gs_currentdevicecolor_inline(pgs);
864
1.19k
        procp = pdc->type->get_nonzero_comps;
865
1.19k
        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.19k
            num_colorant[0] = (dev_proc(dev, get_color_comp_index))\
873
1.19k
                             (dev, "Cyan", strlen("Cyan"), NO_COMP_NAME_TYPE_OP);
874
1.19k
            num_colorant[1] = (dev_proc(dev, get_color_comp_index))\
875
1.19k
                             (dev, "Magenta", strlen("Magenta"), NO_COMP_NAME_TYPE_OP);
876
1.19k
            num_colorant[2] = (dev_proc(dev, get_color_comp_index))\
877
1.19k
                             (dev, "Yellow", strlen("Yellow"), NO_COMP_NAME_TYPE_OP);
878
1.19k
            num_colorant[3] = (dev_proc(dev, get_color_comp_index))\
879
1.19k
                             (dev, "Black", strlen("Black"), NO_COMP_NAME_TYPE_OP);
880
1.19k
            nz_comps = 0;
881
1.19k
            one = 1;
882
1.19k
            colorant_ok = true;
883
5.97k
            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.78k
                if (pdc->ccolor.paint.values[k] > (0.5 / 255.0)) {
887
1.20k
                    if (num_colorant[k] == -1) {
888
0
                        colorant_ok = false;
889
1.20k
                    } else {
890
1.20k
                        temp = one << num_colorant[k];
891
1.20k
                        nz_comps = nz_comps | temp;
892
1.20k
                    }
893
1.20k
                }
894
4.78k
            }
895
            /* For some reason we don't have one of the standard colorants */
896
1.19k
            if (!colorant_ok) {
897
0
                if ((code = procp(pdc, dev, &nz_comps)) < 0)
898
0
                    return code;
899
0
            }
900
1.19k
        } else {
901
0
            if ((code = procp(pdc, dev, &nz_comps)) < 0)
902
0
                return code;
903
0
        }
904
1.19k
        drawn_comps &= nz_comps;
905
1.19k
    }
906
1.43k
    params.is_fill_color = pgs->is_fill_color;
907
1.43k
    params.retain_any_comps = true;
908
1.43k
    params.drawn_comps = drawn_comps;
909
1.43k
    params.op_state = OP_STATE_NONE;
910
911
1.43k
    if_debug2m(gs_debug_flag_overprint, pgs->memory,
912
1.43k
        "[overprint] gx_set_overprint_cmyk. retain_any_comps = %d, drawn_comps = 0x%x\n",
913
1.43k
        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.43k
    params.effective_opm = pgs->color[0].effective_opm =
919
1.43k
        pgs->overprint_mode && gs_currentcolor_eopm(pgs);
920
1.43k
    return gs_gstate_update_overprint(pgs, &params);
921
1.43k
}
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
343k
{
1055
343k
    return cs->type->remap_color(cc, cs, dc, pgs, dev, gs_color_select_texture);
1056
343k
}
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
184k
{
1062
184k
    int i;
1063
1064
399k
    for (i = 0; i < n; i++)
1065
215k
        c->paint.values[i] = c0->paint.values[i] * t + c1->paint.values[i] * (1 - t);
1066
184k
}
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
184k
{
1073
184k
    uchar i;
1074
1075
184k
    if (c0->type == &gx_dc_type_data_pure) {
1076
183k
        gx_color_index pure0 = c0->colors.pure;
1077
183k
        gx_color_index pure1 = c1->colors.pure;
1078
183k
        gx_color_index pure = c->colors.pure;
1079
1080
881k
        for (i = 0; i < n; i++) {
1081
699k
            int shift = dev->color_info.comp_shift[i];
1082
699k
            int mask = (1 << dev->color_info.comp_bits[i]) - 1;
1083
699k
            int max_color = (i == dev->color_info.gray_index ? dev->color_info.max_gray
1084
699k
                                                             : dev->color_info.max_color);
1085
699k
            float max_diff = max(1, max_color * smoothness);
1086
699k
            int b0 = (pure0 >> shift) & mask, b1 = (pure1 >> shift) & mask;
1087
699k
            int b = (pure >> shift) & mask;
1088
699k
            double bb = b0 * t + b1 * (1 - t);
1089
1090
699k
            if (any_abs(b - bb) > max_diff)
1091
875
                return false;
1092
699k
        }
1093
182k
        return true;
1094
183k
    } else if (c0->type == &gx_dc_type_data_devn) {
1095
3.97k
        for (i = 0; i < n; i++) {
1096
3.18k
            int max_color = (i == dev->color_info.gray_index ? dev->color_info.max_gray
1097
3.18k
                : dev->color_info.max_color);
1098
3.18k
            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
3.18k
            double ratio = (double)max_color / (double)gx_max_color_value;
1103
3.18k
            double b0 = (c0->colors.devn.values[i]) * ratio;
1104
3.18k
            double b1 = (c1->colors.devn.values[i]) * ratio;
1105
3.18k
            double b = (c->colors.devn.values[i]) * ratio;
1106
3.18k
            double bb = b0 * t + b1 * (1 - t);
1107
3.18k
            if (any_abs(b - bb) > max_diff)
1108
22
                return false;
1109
3.18k
        }
1110
786
        return true;
1111
808
    } else {
1112
        /* Halftones must not paint with fill_linear_color_*. */
1113
0
        return false;
1114
0
    }
1115
184k
}
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
40.6k
{
1124
40.6k
    gs_client_color c01a, c01b;
1125
40.6k
    gx_device_color d[2], d01a, d01b;
1126
40.6k
    int n = cs->type->num_components(cs);
1127
40.6k
    uchar ndev = dev->color_info.num_components;
1128
40.6k
    int code;
1129
1130
40.6k
    code = cc2dc(cs, pgs, dev, &d[0], c0);
1131
40.6k
    if (code < 0)
1132
0
        return code;
1133
40.6k
    code = cc2dc(cs, pgs, dev, &d[1], c1);
1134
40.6k
    if (code < 0)
1135
0
        return code;
1136
40.6k
    interpolate_cc(&c01a, c0, c1, 0.3, n);
1137
40.6k
    code = cc2dc(cs, pgs, dev, &d01a, &c01a);
1138
40.6k
    if (code < 0)
1139
0
        return code;
1140
40.6k
    if (!is_dc_nearly_linear(dev, &d01a, &d[0], &d[1], 0.3, ndev, smoothness))
1141
666
        return 0;
1142
39.9k
    interpolate_cc(&c01b, c0, c1, 0.7, n);
1143
39.9k
    code = cc2dc(cs, pgs, dev, &d01b, &c01b);
1144
39.9k
    if (code < 0)
1145
0
        return code;
1146
39.9k
    if (!is_dc_nearly_linear(dev, &d01b, &d[0], &d[1], 0.7, ndev, smoothness))
1147
214
        return 0;
1148
39.7k
    return 1;
1149
39.9k
}
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
25.9k
{
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
25.9k
    gs_client_color c01, c12, c20, c012;
1162
25.9k
    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
25.9k
    int n = cs->type->num_components(cs);
1168
25.9k
    uchar ndev = dev->color_info.num_components;
1169
1170
25.9k
    int code;
1171
1172
25.9k
    code = cc2dc(cs, pgs, dev, &d[0], c0);
1173
25.9k
    if (code < 0)
1174
0
        return code;
1175
25.9k
    code = cc2dc(cs, pgs, dev, &d[1], c1);
1176
25.9k
    if (code < 0)
1177
0
        return code;
1178
25.9k
    code = cc2dc(cs, pgs, dev, &d[2], c2);
1179
25.9k
    if (code < 0)
1180
0
        return code;
1181
1182
25.9k
    interpolate_cc(&c01, c0, c1, 0.5, n);
1183
25.9k
    code = cc2dc(cs, pgs, dev, &d01, &c01);
1184
25.9k
    if (code < 0)
1185
0
        return code;
1186
25.9k
    if (!is_dc_nearly_linear(dev, &d01, &d[0], &d[1], 0.5, ndev, smoothness))
1187
10
        return 0;
1188
1189
25.9k
    interpolate_cc(&c012, c2, &c01, 2.0 / 3, n);
1190
25.9k
    code = cc2dc(cs, pgs, dev, &d012, &c012);
1191
25.9k
    if (code < 0)
1192
0
        return code;
1193
25.9k
    if (!is_dc_nearly_linear(dev, &d012, &d[2], &d01, 2.0 / 3, ndev, smoothness))
1194
7
        return 0;
1195
1196
25.9k
    interpolate_cc(&c12, c1, c2, 0.5, n);
1197
25.9k
    code = cc2dc(cs, pgs, dev, &d12, &c12);
1198
25.9k
    if (code < 0)
1199
0
        return code;
1200
25.9k
    if (!is_dc_nearly_linear(dev, &d12, &d[1], &d[2], 0.5, ndev, smoothness))
1201
0
        return 0;
1202
1203
25.9k
    interpolate_cc(&c20, c2, c0, 0.5, n);
1204
25.9k
    code = cc2dc(cs, pgs, dev, &d20, &c20);
1205
25.9k
    if (code < 0)
1206
0
        return code;
1207
25.9k
    if (!is_dc_nearly_linear(dev, &d20, &d[2], &d[0], 0.5, ndev, smoothness))
1208
0
        return 0;
1209
25.9k
    return 1;
1210
25.9k
}
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
66.5k
{
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
66.5k
    int code;
1223
1224
66.5k
    if (!colors_are_separable_and_linear(&dev->color_info))
1225
0
        return_error(gs_error_rangecheck);
1226
66.5k
    if (c2 == NULL)
1227
40.6k
        return gx_cspace_is_linear_in_line(cs, pgs, dev, c0, c1, smoothness);
1228
25.9k
    code = gx_cspace_is_linear_in_triangle(cs, pgs, dev, c0, c1, c2, smoothness);
1229
25.9k
    if (code <= 0)
1230
17
        return code;
1231
25.9k
    if (c3 == NULL)
1232
25.9k
        return 1;
1233
0
    return gx_cspace_is_linear_in_triangle(cs, pgs, dev, c1, c2, c3, smoothness);
1234
25.9k
}
1235
1236
/* Serialization. */
1237
int
1238
gx_serialize_cspace_type(const gs_color_space * pcs, stream * s)
1239
70.6k
{
1240
70.6k
    const gs_color_space_type * type = pcs->type;
1241
70.6k
    uint n;
1242
70.6k
    return sputs(s, (const byte *)&type->index, sizeof(type->index), &n);
1243
70.6k
}
1244
1245
/* GC procedures */
1246
1247
static
1248
ENUM_PTRS_BEGIN_PROC(color_space_enum_ptrs)
1249
10.2M
{
1250
10.2M
    EV_CONST gs_color_space *pcs = vptr;
1251
1252
10.2M
    if (index == 0)
1253
2.05M
        return ENUM_OBJ(pcs->base_space);
1254
8.23M
    if (index == 1)
1255
2.05M
        return ENUM_OBJ(pcs->pclient_color_space_data);
1256
6.17M
    if (index == 2)
1257
2.05M
        return ENUM_OBJ(pcs->icc_equivalent);
1258
4.12M
    if (index == 3) {
1259
2.05M
        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
2.05M
        else
1262
2.05M
            return ENUM_OBJ(NULL);
1263
2.05M
    }
1264
1265
2.06M
    return ENUM_USING(*pcs->type->stype, vptr, size, index - 4);
1266
4.12M
    ENUM_PTRS_END_PROC
1267
4.12M
}
1268
static
1269
2.05M
RELOC_PTRS_WITH(color_space_reloc_ptrs, gs_color_space *pcs)
1270
2.05M
{
1271
2.05M
    RELOC_VAR(pcs->base_space);
1272
2.05M
    RELOC_VAR(pcs->pclient_color_space_data);
1273
2.05M
    RELOC_VAR(pcs->icc_equivalent);
1274
2.05M
    if (gs_color_space_get_index(pcs) == gs_color_space_index_DeviceN)
1275
2.05M
        RELOC_VAR(pcs->params.device_n.devn_process_space);
1276
2.05M
    RELOC_USING(*pcs->type->stype, vptr, size);
1277
2.05M
}
1278
2.05M
RELOC_PTRS_END