Coverage Report

Created: 2025-12-31 07:31

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ghostpdl/pdf/pdf_trans.c
Line
Count
Source
1
/* Copyright (C) 2019-2025 Artifex Software, Inc.
2
   All Rights Reserved.
3
4
   This software is provided AS-IS with no warranty, either express or
5
   implied.
6
7
   This software is distributed under license and may not be copied,
8
   modified or distributed except as expressly authorized under the terms
9
   of the license contained in the file LICENSE in this distribution.
10
11
   Refer to licensing information at http://www.artifex.com or contact
12
   Artifex Software, Inc.,  39 Mesa Street, Suite 108A, San Francisco,
13
   CA 94129, USA, for further information.
14
*/
15
16
/* Transparency support */
17
18
#include "pdf_int.h"
19
#include "pdf_stack.h"
20
#include "pdf_trans.h"
21
#include "pdf_dict.h"
22
#include "pdf_colour.h"
23
#include "pdf_font_types.h"
24
#include "pdf_gstate.h"
25
#include "pdf_array.h"
26
#include "pdf_image.h"
27
#include "pdf_device.h"
28
#include "pdf_misc.h"
29
#include "pdf_func.h"
30
31
#include "gstparam.h"
32
#include "gsicc_manage.h"       /* For gs_currentoverrideicc() */
33
#include "gscoord.h"            /* For gs_setmatrix()*/
34
#include "gsstate.h"            /* For gs_currentstrokeoverprint() and others */
35
#include "gspath.h"             /* For gs_clippath() */
36
#include "gsicc_cache.h"        /* For gsicc_profiles_equal() */
37
38
/* Implement the TransferFunction using a Function. */
39
static int
40
pdfi_tf_using_function(double in_val, float *out, void *proc_data)
41
104k
{
42
104k
    float in = in_val;
43
104k
    gs_function_t *const pfn = proc_data;
44
45
104k
    return gs_function_evaluate(pfn, &in, out);
46
104k
}
47
48
static void
49
pdfi_set_GrayBackground(gs_transparency_mask_params_t *params)
50
2.37k
{
51
2.37k
    float num;
52
2.37k
    int i;
53
54
8.42k
    for (i = 0; i < params->Background_components; i++)
55
6.05k
    {
56
6.05k
        if (params->Background[i] < 0)
57
0
            params->Background[i] = 0;
58
6.05k
        else if (params->Background[i] > 1)
59
0
            params->Background[i] = 1;
60
6.05k
    }
61
62
    /* This uses definition from PLRM2 6.2.1 and 6.2.2 */
63
    /* TODO: We are assuming 3 components is RGB and 4 components is CMYK,
64
       which might not strictly be true?  But it provides a better
65
       estimated value than doing nothing.
66
    */
67
2.37k
    switch(params->Background_components) {
68
25
    case 3:
69
        /* RGB: currentgray = 0.3*R + 0.59*G + 0.11*B */
70
25
        params->GrayBackground = (0.3 * params->Background[0] +
71
25
                                  0.59 * params->Background[1] +
72
25
                                  0.11 * params->Background[2]);
73
25
        break;
74
1.21k
    case 4:
75
        /* CMYK: currentgray = 1.0 – min (1.0, .3*C + .59*M + .11*Y + K)
76
        */
77
1.21k
        num = 0.3*params->Background[0] + 0.59*params->Background[1] +
78
1.21k
            0.11*params->Background[2] + params->Background[3];
79
1.21k
        params->GrayBackground = 1 - num;
80
1.21k
        break;
81
1.13k
    case 1:
82
1.13k
        params->GrayBackground = params->Background[0];
83
1.13k
        break;
84
0
    default:
85
        /* No clue... */
86
0
        params->GrayBackground = 0;
87
2.37k
    }
88
2.37k
    if (params->GrayBackground < 0)
89
1.14k
        params->GrayBackground = 0;
90
1.22k
    else if (params->GrayBackground > 1)
91
0
        params->GrayBackground = 1;
92
2.37k
}
93
94
/* (see pdf_draw.ps/execmaskgroup) */
95
static int pdfi_trans_set_mask(pdf_context *ctx, pdfi_int_gstate *igs, int colorindex)
96
14.4k
{
97
14.4k
    int code = 0, code1 = 0;
98
14.4k
    pdf_dict *SMask = igs->SMask;
99
14.4k
    gs_color_space *pcs = NULL;
100
14.4k
    gs_rect bbox;
101
14.4k
    gs_transparency_mask_params_t params;
102
14.4k
    pdf_array *BBox = NULL;
103
14.4k
    pdf_array *Matrix = NULL;
104
14.4k
    pdf_array *a = NULL;
105
14.4k
    pdf_array *BC = NULL;
106
14.4k
    pdf_stream *G_stream = NULL;
107
14.4k
    pdf_dict *G_stream_dict = NULL;
108
14.4k
    pdf_dict *Group = NULL;
109
14.4k
    pdf_obj *TR = NULL;
110
14.4k
    gs_function_t *gsfunc = NULL;
111
14.4k
    pdf_name *n = NULL;
112
14.4k
    pdf_name *S = NULL;
113
14.4k
    pdf_obj *CS = NULL;
114
14.4k
    gs_matrix save_matrix, GroupMat, group_Matrix;
115
14.4k
    gs_transparency_mask_subtype_t subtype = TRANSPARENCY_MASK_Luminosity;
116
14.4k
    bool Processed, ProcessedKnown = 0;
117
14.4k
    bool save_OverrideICC = gs_currentoverrideicc(ctx->pgs);
118
14.4k
    gs_gstate *saved_gs = NULL;
119
14.4k
    gs_rect clip_bbox;
120
14.4k
    int empty = 0;
121
122
#if DEBUG_TRANSPARENCY
123
    dbgmprintf(ctx->memory, "pdfi_trans_set_mask (.execmaskgroup) BEGIN\n");
124
#endif
125
14.4k
    memset(&params, 0, sizeof(params));
126
127
    /* Following the logic of the ps code, cram a /Processed key in the SMask dict to
128
     * track whether it's already been processed.
129
     */
130
14.4k
    code = pdfi_dict_knownget_bool(ctx, SMask, "Processed", &Processed);
131
14.4k
    if (code > 0) {
132
8.25k
        if (Processed) {
133
#if DEBUG_TRANSPARENCY
134
          dbgmprintf(ctx->memory, "SMask already built, skipping\n");
135
#endif
136
2.04k
          code = 0;
137
2.04k
          goto exit;
138
2.04k
        }
139
6.21k
        ProcessedKnown = 1;
140
6.21k
    }
141
142
12.4k
    gs_setoverrideicc(ctx->pgs, true);
143
144
    /* If /Processed not in the dict, put it there */
145
12.4k
    if (code == 0) {
146
6.21k
        code = pdfi_dict_put_bool(ctx, SMask, "Processed", false);
147
6.21k
        if (code < 0)
148
0
            goto exit;
149
6.21k
        ProcessedKnown = 1;
150
6.21k
    }
151
152
    /* See pdf1.7 pg 553 (pain in the butt to find this!) */
153
12.4k
    code = pdfi_dict_knownget_type(ctx, SMask, "Type", PDF_NAME, (pdf_obj **)&n);
154
12.4k
    if (code == 0 || (code > 0 && pdfi_name_is(n, "Mask"))) {
155
        /* G is transparency group XObject (required) */
156
12.3k
        code = pdfi_dict_knownget_type(ctx, SMask, "G", PDF_STREAM, (pdf_obj **)&G_stream);
157
12.3k
        if (code <= 0) {
158
5.40k
            pdfi_trans_end_smask_notify(ctx);
159
5.40k
            code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_SMASK_MISSING_G, "pdfi_trans_set_mask", "");
160
5.40k
            goto exit;
161
5.40k
        }
162
163
6.92k
        code = pdfi_dict_from_obj(ctx, (pdf_obj *)G_stream, &G_stream_dict);
164
6.92k
        if (code < 0)
165
0
            goto exit;
166
167
        /* S is a subtype name (required) */
168
6.92k
        code = pdfi_dict_knownget_type(ctx, SMask, "S", PDF_NAME, (pdf_obj **)&S);
169
6.92k
        if (code <= 0) {
170
217
            subtype = TRANSPARENCY_MASK_Luminosity;
171
217
            pdfi_set_warning(ctx, 0, NULL, W_PDF_SMASK_MISSING_S, "pdfi_trans_set_mask", "");
172
217
        }
173
6.70k
        else if (pdfi_name_is(S, "Luminosity")) {
174
6.18k
            subtype = TRANSPARENCY_MASK_Luminosity;
175
6.18k
        } else if (pdfi_name_is(S, "Alpha")) {
176
516
            subtype = TRANSPARENCY_MASK_Alpha;
177
516
        } else {
178
13
            subtype = TRANSPARENCY_MASK_Luminosity;
179
13
            pdfi_set_warning(ctx, 0, NULL, W_PDF_SMASK_UNKNOWN_S, "pdfi_trans_set_mask", "");
180
13
        }
181
182
        /* TR is transfer function (Optional) */
183
6.92k
        code = pdfi_dict_knownget(ctx, SMask, "TR", (pdf_obj **)&TR);
184
6.92k
        if (code > 0) {
185
462
            switch (pdfi_type_of(TR)) {
186
6
                case PDF_DICT:
187
462
                case PDF_STREAM:
188
462
                    code = pdfi_build_function(ctx, &gsfunc, NULL, 1, TR, NULL);
189
462
                    if (code < 0)
190
52
                        goto exit;
191
410
                    if (gsfunc->params.m != 1 || gsfunc->params.n != 1) {
192
0
                        pdfi_free_function(ctx, gsfunc);
193
0
                        gsfunc = NULL;
194
0
                        pdfi_set_warning(ctx, 0, NULL, W_PDF_SMASK_INVALID_TR, "pdfi_trans_set_mask", "");
195
0
                    }
196
410
                    break;
197
0
                case PDF_NAME:
198
0
                    if (!pdfi_name_is((pdf_name *)TR, "Identity")) {
199
0
                        pdfi_set_warning(ctx, 0, NULL, W_PDF_SMASK_UNKNOWN_TR, "pdfi_trans_set_mask", "");
200
0
                    }
201
0
                    break;
202
0
                default:
203
0
                    pdfi_set_warning(ctx, 0, NULL, W_PDF_SMASK_UNKNOWN_TR_TYPE, "pdfi_trans_set_mask", "");
204
462
            }
205
462
        }
206
207
        /* BC is Background Color array (Optional) */
208
6.87k
        if (subtype == TRANSPARENCY_MASK_Luminosity)
209
6.40k
        {
210
6.40k
            code = pdfi_dict_knownget_type(ctx, SMask, "BC", PDF_ARRAY, (pdf_obj **)&BC);
211
6.40k
            if (code < 0)
212
120
                goto exit;
213
6.40k
        }
214
215
6.75k
        code = pdfi_dict_knownget_type(ctx, G_stream_dict, "BBox", PDF_ARRAY, (pdf_obj **)&BBox);
216
6.75k
        if (code < 0)
217
9
            goto exit;
218
6.74k
        code = pdfi_array_to_gs_rect(ctx, BBox, &bbox);
219
6.74k
        if (code < 0)
220
16
            goto exit;
221
222
6.72k
        gs_trans_mask_params_init(&params, subtype);
223
6.72k
        params.replacing = true;
224
6.72k
        if (gsfunc) {
225
410
            params.TransferFunction = pdfi_tf_using_function;
226
410
            params.TransferFunction_data = gsfunc;
227
410
        }
228
229
        /* Need to set just the ctm (GroupMat) from the saved GroupGState, to
230
           have gs_begin_transparency_mask work correctly.  Or at least that's
231
           what the PS code comments claim (see pdf_draw.ps/.execmaskgroup)
232
        */
233
6.72k
        gs_currentmatrix(ctx->pgs, &save_matrix);
234
6.72k
        gs_currentmatrix(igs->GroupGState, &GroupMat);
235
6.72k
        gs_setmatrix(ctx->pgs, &GroupMat);
236
237
6.72k
        code = pdfi_dict_knownget_type(ctx, G_stream_dict, "Matrix", PDF_ARRAY, (pdf_obj **)&Matrix);
238
6.72k
        if (code < 0)
239
0
            goto exit;
240
6.72k
        code = pdfi_array_to_gs_matrix(ctx, Matrix, &group_Matrix);
241
6.72k
        if (code < 0)
242
10
            goto exit;
243
244
        /* Transform the BBox by the Matrix */
245
6.71k
        pdfi_bbox_transform(ctx, &bbox, &group_Matrix);
246
247
6.71k
        if (gs_clip_bounds_in_user_space(ctx->pgs, &clip_bbox) >= 0)
248
6.66k
        {
249
6.66k
            rect_intersect(bbox, clip_bbox);
250
6.66k
            if (bbox.p.x >= bbox.q.x || bbox.p.y >= bbox.q.y)
251
1.15k
            {
252
                /* If the bbox is illegal, we still need to set up an empty mask.
253
                 * We can't just skip forwards as everything will now be unmasked!
254
                 * We can skip doing the actual work though. */
255
1.15k
                empty = 1;
256
1.15k
            }
257
6.66k
        }
258
259
        /* CS is in the dict "Group" inside the dict "G" */
260
        /* TODO: Not sure if this is a required thing or just one possibility */
261
6.71k
        code = pdfi_dict_knownget_type(ctx, G_stream_dict, "Group", PDF_DICT, (pdf_obj **)&Group);
262
6.71k
        if (code < 0)
263
2.59k
            goto exit;
264
4.12k
        if (code > 0) {
265
            /* TODO: Stuff with colorspace, see .execmaskgroup */
266
3.87k
            code = pdfi_dict_knownget(ctx, Group, "CS", &CS);
267
3.87k
            if (code < 0) {
268
37
                code = pdfi_dict_knownget(ctx, Group, "ColorSpace", &CS);
269
37
                if (code < 0) {
270
0
                    pdfi_set_error(ctx, 0, NULL, E_PDF_GROUP_NO_CS, "pdfi_trans_set_mask", (char *)"*** Defaulting to currrent colour space");
271
0
                    goto exit;
272
0
                }
273
37
                pdfi_set_warning(ctx, 0, NULL, W_PDF_GROUP_HAS_COLORSPACE, "pdfi_trans_set_mask", NULL);
274
37
            }
275
3.87k
            if (code > 0) {
276
3.79k
                code = pdfi_create_colorspace(ctx, CS, (pdf_dict *)ctx->main_stream,
277
3.79k
                                              ctx->page.CurrentPageDict, &pcs, false);
278
3.79k
                params.ColorSpace = pcs;
279
3.79k
                if (code < 0)
280
109
                    goto exit;
281
3.79k
            }
282
79
            else
283
79
                   params.ColorSpace = NULL;
284
3.87k
        } else {
285
            /* GS and Adobe will ignore the whole mask in this case, so we do the same.
286
            */
287
246
            pdfi_set_error(ctx, 0, NULL, E_PDF_INVALID_TRANS_XOBJECT, "pdfi_trans_set_mask", (char *)"*** Error: Ignoring a transparency group XObject without /Group attribute");
288
246
            goto exit;
289
246
        }
290
291
        /* If there's a BC, put it in the params */
292
3.76k
        if (BC) {
293
2.40k
            if (params.ColorSpace == NULL) {
294
37
                if ((code = pdfi_set_error_stop(ctx, gs_note_error(gs_error_undefined), NULL, E_PDF_GROUP_BAD_BC_NO_CS, "pdfi_trans_set_mask", NULL)) < 0)
295
0
                    goto exit;
296
2.37k
            } else {
297
2.37k
                int i, components = pdfi_array_size(BC);
298
2.37k
                double num;
299
2.37k
                int cspace_cmps;
300
301
2.37k
                if (components > GS_CLIENT_COLOR_MAX_COMPONENTS) {
302
0
                    if ((code = pdfi_set_error_stop(ctx, gs_note_error(gs_error_limitcheck), NULL, E_PDF_GROUP_BAD_BC_TOO_BIG, "pdfi_trans_set_mask", NULL)) < 0)
303
0
                        goto exit;
304
2.37k
                } else {
305
2.37k
                    if ((cspace_cmps = gs_color_space_num_components(params.ColorSpace)) < 0) {
306
0
                        if ((code = pdfi_set_error_stop(ctx, gs_note_error(gs_error_undefinedresult), NULL, E_PDF_GROUP_BAD_CS, "pdfi_trans_set_mask", NULL)) < 0)
307
0
                            goto exit;
308
0
                    }
309
2.37k
                    else {
310
2.37k
                        if (gs_color_space_num_components(params.ColorSpace) != components) {
311
0
                            pdfi_set_warning(ctx, 0, NULL, W_PDF_GROUP_BAD_BC, "pdfi_trans_set_mask", NULL);
312
0
                            components = min(components, gs_color_space_num_components(params.ColorSpace));
313
0
                        }
314
315
8.42k
                        for (i=0; i < components; i++) {
316
6.05k
                            code = pdfi_array_get_number(ctx, BC, i, &num);
317
6.05k
                            if (code < 0)
318
0
                                break;
319
6.05k
                            params.Background[i] = (float)num;
320
6.05k
                        }
321
2.37k
                        params.Background_components = components;
322
323
                        /* TODO: Not sure how to handle this...  recheck PS code (pdf_draw.ps/gssmask) */
324
                        /* This should be "currentgray" for the color that we put in params.ColorSpace,
325
                         * It looks super-convoluted to actually get this value.  Really?
326
                         * (see zcurrentgray())
327
                         * For now, use simple definition from PLRM2 and assume it is RGB or CMYK
328
                         */
329
2.37k
                        pdfi_set_GrayBackground(&params);
330
2.37k
                    }
331
2.37k
                }
332
2.37k
            }
333
2.40k
        }
334
335
        /* When we call gs_begin_transparency_mask, the pdf14 compisitor will perform
336
         * a horrid pass through the graphics state, replacing "default" colorspaces
337
         * with "ps" ones. This is required, apparently, to give us proper rendering
338
         * of SMasks. Bug 705993 shows a case where the current colorspace differs
339
         * from the colorspace that will be used once the igs->GroupGState is swapped
340
         * in. Simply proceeding to call gs_begin_transparency_mask here, and then
341
         * setting the igs->GroupGState, leaves us with a colorspace where the
342
         * "default" -> "ps" change has not taken place, and we get incorrect rendering.
343
         *
344
         * As the igs->GroupGState can only be applied AFTER the
345
         * gs_begin_transparancy_mask call (that is, I couldn't make it work applying
346
         * it beforehand!), this means we need to manually swap in the colorspaces
347
         * and colors, and be careful to restore them afterwards.
348
         *
349
         * NOTE: Attempts to gsave/grestore around the gs_begin_transparency_mask/
350
         * gs_end_transparency_mask calls fail, showing diffs. My suspicion is that
351
         * this actually disables the softmask that we've just rendered!
352
         */
353
3.76k
        if (code >= 0) {
354
            /* Override the colorspace if specified */
355
3.76k
            saved_gs = gs_gstate_copy(ctx->pgs, ctx->memory);
356
3.76k
            if (saved_gs == NULL)
357
0
                code = gs_note_error(gs_error_VMerror);
358
3.76k
            if (code >= 0)
359
3.76k
                code = pdfi_gs_setcolorspace(ctx, igs->GroupGState->color[0].color_space);
360
3.76k
            if (code >= 0)
361
3.76k
                code = gs_setcolor(ctx->pgs, igs->GroupGState->color[0].ccolor);
362
3.76k
            gs_swapcolors_quick(ctx->pgs);
363
3.76k
            if (code >= 0)
364
3.76k
                code = pdfi_gs_setcolorspace(ctx, igs->GroupGState->color[1].color_space);
365
3.76k
            if (code >= 0)
366
3.76k
                code = gs_setcolor(ctx->pgs, igs->GroupGState->color[1].ccolor);
367
3.76k
            gs_swapcolors_quick(ctx->pgs);
368
3.76k
        }
369
370
3.76k
        if (code >= 0)
371
3.76k
            code = gs_begin_transparency_mask(ctx->pgs, &params, &bbox, false);
372
373
3.76k
        if (code >= 0) {
374
3.76k
            if (!empty)
375
3.60k
                code = pdfi_form_execgroup(ctx, ctx->page.CurrentPageDict, G_stream,
376
3.60k
                                           igs->GroupGState, NULL, NULL, &group_Matrix);
377
3.76k
            code1 = gs_end_transparency_mask(ctx->pgs, colorindex);
378
3.76k
            if (code == 0)
379
3.76k
                code = code1;
380
3.76k
        }
381
382
3.76k
        if (saved_gs) {
383
3.76k
            code1 = pdfi_gs_setcolorspace(ctx, saved_gs->color[0].color_space);
384
3.76k
            if (code >= 0) code = code1;
385
3.76k
            code1 = gs_setcolor(ctx->pgs, saved_gs->color[0].ccolor);
386
3.76k
            if (code >= 0) code = code1;
387
3.76k
            gs_swapcolors_quick(ctx->pgs);
388
3.76k
            code1 = pdfi_gs_setcolorspace(ctx, saved_gs->color[1].color_space);
389
3.76k
            if (code >= 0) code = code1;
390
3.76k
            code1 = gs_setcolor(ctx->pgs, saved_gs->color[1].ccolor);
391
3.76k
            if (code >= 0) code = code1;
392
3.76k
            gs_swapcolors_quick(ctx->pgs);
393
3.76k
        }
394
395
        /* Put back the matrix (we couldn't just rely on gsave/grestore for whatever reason,
396
         * according to PS code anyway...
397
         */
398
3.76k
        gs_setmatrix(ctx->pgs, &save_matrix);
399
400
        /* Set Processed flag */
401
3.76k
        if (code == 0 && ProcessedKnown)
402
3.76k
        {
403
3.76k
            code = pdfi_dict_put_bool(ctx, SMask, "Processed", true);
404
3.76k
            if (code < 0)
405
0
                goto exit;
406
3.76k
        }
407
3.76k
    } else {
408
        /* take action on a non-/Mask entry. What does this mean ? What do we need to do */
409
101
        pdfi_set_warning(ctx, 0, NULL, W_PDF_SMASK_UNKNOWN_TYPE, "pdfi_trans_set_mask", "");
410
101
    }
411
412
14.4k
 exit:
413
14.4k
    gs_gstate_free(saved_gs);
414
14.4k
    gs_setoverrideicc(ctx->pgs, save_OverrideICC);
415
14.4k
    if (gsfunc)
416
410
        pdfi_free_function(ctx, gsfunc);
417
14.4k
    if (pcs)
418
3.68k
        rc_decrement_cs(pcs, "pdfi_trans_set_mask");
419
14.4k
    pdfi_countdown(n);
420
14.4k
    pdfi_countdown(S);
421
14.4k
    pdfi_countdown(Group);
422
14.4k
    pdfi_countdown(G_stream);
423
14.4k
    pdfi_countdown(a);
424
14.4k
    pdfi_countdown(BC);
425
14.4k
    pdfi_countdown(TR);
426
14.4k
    pdfi_countdown(BBox);
427
14.4k
    pdfi_countdown(Matrix);
428
14.4k
    pdfi_countdown(CS);
429
#if DEBUG_TRANSPARENCY
430
    dbgmprintf(ctx->memory, "pdfi_trans_set_mask (.execmaskgroup) END\n");
431
#endif
432
14.4k
    return code;
433
12.4k
}
434
435
/* Wrapper around gs call to setup the transparency params correctly */
436
static int pdfi_gs_begin_transparency_group(gs_gstate * pgs,
437
                                       gs_transparency_group_params_t *params,
438
                                       const gs_rect *pbbox, pdf14_compositor_operations group_type)
439
107k
{
440
107k
    if (gs_getalphaisshape(pgs)) {
441
4.78k
        params->group_shape = gs_getfillconstantalpha(pgs);
442
4.78k
        params->group_opacity = 1.0;
443
103k
    } else {
444
103k
        params->group_opacity = gs_getfillconstantalpha(pgs);
445
103k
        params->group_shape = 1.0;
446
103k
    }
447
448
107k
    return gs_begin_transparency_group(pgs, params, pbbox, group_type);
449
107k
}
450
451
static int pdfi_transparency_group_common(pdf_context *ctx, pdf_dict *page_dict,
452
                                          pdf_dict *group_dict,
453
                                          gs_rect *bbox, pdf14_compositor_operations group_type)
454
28.8k
{
455
28.8k
    gs_transparency_group_params_t params;
456
28.8k
    pdf_obj *CS = NULL;
457
28.8k
    bool b;
458
28.8k
    int code;
459
460
28.8k
    gs_trans_group_params_init(&params, 1.0);
461
    /*    gs_setopacityalpha(ctx->pgs, ctx->pgs->fillconstantalpha); */
462
463
    /* It seems the flag for Isolated is /I */
464
28.8k
    code = pdfi_dict_get_bool(ctx, group_dict, "I", &b);
465
28.8k
    if (code < 0 && code != gs_error_undefined)
466
0
        return_error(code);
467
28.8k
    if (code == gs_error_undefined)
468
8.74k
        params.Isolated = false;
469
20.0k
    else
470
20.0k
        params.Isolated = b;
471
472
    /* It seems the flag for Knockout is /K */
473
28.8k
    code = pdfi_dict_get_bool(ctx, group_dict, "K", &b);
474
28.8k
    if (code < 0 && code != gs_error_undefined)
475
0
        goto exit;
476
28.8k
    if (code == gs_error_undefined)
477
8.95k
        params.Knockout = false;
478
19.8k
    else
479
19.8k
        params.Knockout = b;
480
481
28.8k
    params.image_with_SMask = false;
482
28.8k
    params.ColorSpace = NULL;
483
484
28.8k
    code = pdfi_dict_knownget(ctx, group_dict, "CS", &CS);
485
28.8k
    if (code == 0) {
486
        /* Didn't find a /CS key, try again using /ColorSpace */
487
3.76k
        code = pdfi_dict_knownget(ctx, group_dict, "ColorSpace", &CS);
488
3.76k
    }
489
28.8k
    if (code > 0 && pdfi_type_of(CS) != PDF_NULL) {
490
24.5k
        code = pdfi_setcolorspace(ctx, CS, group_dict, page_dict);
491
24.5k
        if (code < 0)
492
188
            goto exit;
493
24.3k
        params.ColorSpace = gs_currentcolorspace(ctx->pgs);
494
24.3k
    } else {
495
4.22k
        params.ColorSpace = NULL;
496
4.22k
    }
497
498
28.8k
 exit:
499
28.8k
    pdfi_countdown(CS);
500
28.8k
    if (code < 0)
501
355
        return_error(code);
502
503
28.4k
    return pdfi_gs_begin_transparency_group(ctx->pgs, &params, (const gs_rect *)bbox, group_type);
504
28.8k
}
505
506
static bool pdfi_outputprofile_matches_oiprofile(pdf_context *ctx)
507
0
{
508
0
    cmm_dev_profile_t *profile_struct;
509
0
    int code;
510
0
    int k;
511
512
0
    code = dev_proc(ctx->pgs->device, get_profile)(ctx->pgs->device,  &profile_struct);
513
0
    if (code < 0)
514
0
        return true;  /* Assume they match by default and in error condition */
515
516
0
    if (profile_struct->oi_profile == NULL)
517
0
        return true; /* no OI profile so no special case to worry about */
518
0
    else {
519
        /* Check the device profile(s). If any of them do not match, then
520
           we assume there is not a match and it may be necessary to
521
           use the pdf14 device to prerender to the OI profile */
522
0
        for (k = 0; k < NUM_DEVICE_PROFILES; k++) {
523
0
            if (profile_struct->device_profile[k] != NULL) {
524
0
                if (!gsicc_profiles_equal(profile_struct->oi_profile, profile_struct->device_profile[k]))
525
0
                    return false;
526
0
            }
527
0
        }
528
0
        return true;
529
0
    }
530
0
}
531
532
/* Begin a simple group
533
 * pathbbox -- bbox to use, but can be NULL
534
 */
535
int pdfi_trans_begin_simple_group(pdf_context *ctx, gs_rect *pathbbox,
536
                                  bool stroked_bbox, bool isolated, bool knockout)
537
21.1k
{
538
21.1k
    gs_transparency_group_params_t params;
539
21.1k
    gs_rect bbox;
540
21.1k
    int code;
541
542
21.1k
    gs_trans_group_params_init(&params, 1.0);
543
21.1k
    params.Isolated = isolated;
544
21.1k
    params.Knockout = knockout;
545
546
21.1k
    if (!pathbbox) {
547
17.5k
        code = pdfi_get_current_bbox(ctx, &bbox, stroked_bbox);
548
17.5k
        if (code < 0)
549
1.84k
            return code;
550
15.7k
        pathbbox = &bbox;
551
15.7k
    }
552
553
19.3k
    code = pdfi_gs_begin_transparency_group(ctx->pgs, &params, pathbbox, PDF14_BEGIN_TRANS_GROUP);
554
19.3k
    if (code >=  0)
555
19.3k
        ctx->current_stream_save.group_depth++;
556
19.3k
    return code;
557
21.1k
}
558
559
int pdfi_trans_begin_page_group(pdf_context *ctx, pdf_dict *page_dict, pdf_dict *group_dict)
560
7.97k
{
561
7.97k
    gs_rect bbox;
562
7.97k
    int code;
563
564
7.97k
    if (group_dict == NULL)
565
0
        return_error(gs_error_undefined);
566
567
7.97k
    code = pdfi_gsave(ctx);
568
7.97k
    bbox.p.x = ctx->page.Size[0];
569
7.97k
    bbox.p.y = ctx->page.Size[1];
570
7.97k
    bbox.q.x = ctx->page.Size[2];
571
7.97k
    bbox.q.y = ctx->page.Size[3];
572
573
7.97k
    code = pdfi_transparency_group_common(ctx, page_dict, group_dict, &bbox, PDF14_BEGIN_TRANS_PAGE_GROUP);
574
7.97k
    if (code < 0)
575
145
        pdfi_grestore(ctx);
576
7.83k
    else
577
7.83k
        ctx->current_stream_save.group_depth++;
578
579
7.97k
    return code;
580
7.97k
}
581
582
int pdfi_trans_begin_form_group(pdf_context *ctx, pdf_dict *page_dict, pdf_dict *form_dict)
583
21.9k
{
584
21.9k
    pdf_obj *group_obj = NULL;
585
21.9k
    gs_rect bbox;
586
21.9k
    pdf_array *BBox = NULL;
587
21.9k
    int code;
588
21.9k
    pdf_dict *group_dict = NULL;
589
590
    /* TODO: Maybe sometimes this is actually a stream?
591
     * Otherwise should just fetch it as a dict.
592
     * Anyway this will work for either dict or stream
593
     */
594
21.9k
    code = pdfi_dict_get(ctx, form_dict, "Group", &group_obj);
595
21.9k
    if (code < 0)
596
501
        return_error(code);
597
598
21.4k
    code = pdfi_dict_from_obj(ctx, (pdf_obj *)group_obj, &group_dict);
599
21.4k
    if (code < 0)
600
619
        goto exit;
601
602
20.8k
    code = pdfi_gsave(ctx);
603
20.8k
    code = pdfi_dict_knownget_type(ctx, form_dict, "BBox", PDF_ARRAY, (pdf_obj **)&BBox);
604
20.8k
    if (code < 0)
605
0
        goto exit;
606
20.8k
    if (code > 0) {
607
20.8k
        code = pdfi_array_to_gs_rect(ctx, BBox, &bbox);
608
20.8k
        if (code < 0)
609
0
            goto exit;
610
20.8k
    } else {
611
2
        bbox.p.x = 0;
612
2
        bbox.p.y = 0;
613
2
        bbox.q.x = 0;
614
2
        bbox.q.y = 0;
615
2
    }
616
617
20.8k
    code = pdfi_transparency_group_common(ctx, page_dict, group_dict, &bbox, PDF14_BEGIN_TRANS_GROUP);
618
20.8k
    if (code < 0)
619
210
        pdfi_grestore(ctx);
620
20.6k
    else
621
20.6k
        ctx->current_stream_save.group_depth++;
622
623
21.4k
 exit:
624
21.4k
    pdfi_countdown(BBox);
625
21.4k
    pdfi_countdown(group_obj);
626
21.4k
    return code;
627
20.8k
}
628
629
630
int pdfi_trans_end_group(pdf_context *ctx)
631
28.4k
{
632
28.4k
    int code;
633
634
28.4k
    code = gs_end_transparency_group(ctx->pgs);
635
28.4k
    if (code < 0)
636
0
        pdfi_grestore(ctx);
637
28.4k
    else
638
28.4k
        code = pdfi_grestore(ctx);
639
640
28.4k
    ctx->current_stream_save.group_depth--;
641
642
28.4k
    return code;
643
28.4k
}
644
645
/* Ends group with no grestore */
646
int pdfi_trans_end_simple_group(pdf_context *ctx)
647
19.2k
{
648
19.2k
    int code;
649
650
19.2k
    code = gs_end_transparency_group(ctx->pgs);
651
19.2k
    ctx->current_stream_save.group_depth--;
652
653
19.2k
    return code;
654
19.2k
}
655
656
657
int pdfi_trans_begin_isolated_group(pdf_context *ctx, bool image_with_SMask, gs_color_space *pcs)
658
60.1k
{
659
60.1k
    gs_transparency_group_params_t params;
660
60.1k
    gs_rect bbox;
661
60.1k
    gs_rect clip_box;
662
663
60.1k
    gs_trans_group_params_init(&params, 1.0);
664
665
60.1k
    params.ColorSpace = pcs;
666
60.1k
    params.Isolated = true;
667
60.1k
    params.Knockout = false;
668
60.1k
    params.image_with_SMask = image_with_SMask;
669
60.1k
    bbox.p.x = 0;
670
60.1k
    bbox.p.y = 0;
671
60.1k
    bbox.q.x = 1;
672
60.1k
    bbox.q.y = 1;
673
60.1k
    if (gs_clip_bounds_in_user_space(ctx->pgs, &clip_box) >= 0)
674
60.1k
        rect_intersect(bbox, clip_box);
675
676
60.1k
    return pdfi_gs_begin_transparency_group(ctx->pgs, &params, &bbox, PDF14_BEGIN_TRANS_GROUP);
677
60.1k
}
678
679
int pdfi_trans_end_isolated_group(pdf_context *ctx)
680
60.1k
{
681
60.1k
    return gs_end_transparency_group(ctx->pgs);
682
60.1k
}
683
684
685
/* This notifies the compositor that we're done with an smask.  Seems hacky.
686
 * See pdf_draw.ps/doimagesmask.
687
 */
688
int pdfi_trans_end_smask_notify(pdf_context *ctx)
689
68.5k
{
690
68.5k
    gs_transparency_mask_params_t params;
691
68.5k
    gs_rect bbox;
692
693
68.5k
    gs_trans_mask_params_init(&params, TRANSPARENCY_MASK_None);
694
68.5k
    params.replacing = true;
695
696
68.5k
    bbox.p.x = 0;
697
68.5k
    bbox.p.y = 0;
698
68.5k
    bbox.q.x = 0;
699
68.5k
    bbox.q.y = 0;
700
701
68.5k
    return gs_begin_transparency_mask(ctx->pgs, &params, &bbox, false);
702
68.5k
}
703
704
/* Setup whether or not we need to support overprint (for device)
705
 * Check for:
706
 *   1) how many components the device has (CMYK or other)
707
 *   2) whether it is a device that has transparency support
708
 *   3) Overprint mode (enable, disable, simulate)
709
 * Based on pdf_main.ps/pdfshowpage_finish
710
 * Test file (run with -dOverprint=/simulate)
711
 *    tests_private/pdf/uploads/op_trans_spot_testing.pdf
712
 */
713
void pdfi_trans_set_needs_OP(pdf_context *ctx)
714
133k
{
715
133k
    bool is_cmyk;
716
133k
    bool device_transparency = false;
717
718
    /* PS code checks for >= 4 components... */
719
133k
    is_cmyk = ctx->pgs->device->color_info.num_components >= 4 && ctx->pgs->device->color_info.polarity == GX_CINFO_POLARITY_SUBTRACTIVE;
720
721
133k
    device_transparency = pdfi_device_check_param_bool(ctx->pgs->device, "HaveTransparency");
722
723
133k
    ctx->page.needs_OP = false;
724
133k
    ctx->page.simulate_op = false;
725
133k
    switch(ctx->pgs->device->icc_struct->overprint_control) {
726
0
    case gs_overprint_control_disable:
727
        /* Use defaults */
728
0
        break;
729
0
    case gs_overprint_control_simulate:
730
0
        if (!device_transparency && ctx->page.has_OP) {
731
0
            if (is_cmyk) {
732
                /* If the page has spots and the device is not spot capable OR
733
                   if the output intent profile is to be used, but we have
734
                   a device output profile that is different, then we will be
735
                   doing simulation with the pdf14 device buffer */
736
0
                if ((ctx->page.num_spots > 0  && !ctx->device_state.spot_capable) ||
737
0
                    !pdfi_outputprofile_matches_oiprofile(ctx)) {
738
0
                    ctx->page.needs_OP = true;
739
0
                    ctx->page.simulate_op = true;
740
0
                }
741
0
            } else {
742
0
                ctx->page.needs_OP = true;
743
0
                ctx->page.simulate_op = true;
744
0
            }
745
0
        }
746
0
        break;
747
133k
    case gs_overprint_control_enable:
748
133k
    default:
749
133k
        if (!is_cmyk || device_transparency)
750
119k
            ctx->page.needs_OP = false;
751
14.8k
        else
752
14.8k
            ctx->page.needs_OP = true;
753
133k
        break;
754
133k
    }
755
756
133k
    if(ctx->args.pdfdebug)
757
133k
        dbgmprintf2(ctx->memory, "Page %s Overprint, Simulate is %s\n",
758
133k
                    ctx->page.needs_OP ? "NEEDS" : "does NOT NEED",
759
133k
                    ctx->page.simulate_op ? "TRUE" : "FALSE");
760
133k
}
761
762
/* Figures out if current colorspace is okay for Overprint (see pdf_ops.ps/okOPcs and setupOPtrans) */
763
bool pdfi_trans_okOPcs(pdf_context *ctx)
764
319k
{
765
319k
    gs_color_space_index csi;
766
767
319k
    csi = pdfi_currentcolorspace(ctx, 0);
768
769
319k
    switch (csi) {
770
52.4k
    case gs_color_space_index_DeviceGray:
771
78.5k
    case gs_color_space_index_DeviceCMYK:
772
78.7k
    case gs_color_space_index_DeviceN:
773
80.1k
    case gs_color_space_index_Separation:
774
        /* These are colorspaces that don't require special handling for overprint.
775
         * (pdf1.7 pg 259,578 may apply)
776
         * According to mvrhel, DeviceGray should also be included (see comment in gx_set_overprint_ICC()).
777
         * Sample: 030_Gray_K_black_OP_x1a.pdf (??)
778
         */
779
#if DEBUG_TRANSPARENCY
780
        dbgmprintf1(ctx->memory, "Colorspace is %d, OKAY for OVERPRINT\n", csi);
781
#endif
782
80.1k
        return true;
783
239k
    default:
784
#if DEBUG_TRANSPARENCY
785
        dbgmprintf1(ctx->memory, "Colorspace is %d, NOT OKAY for OVERPRINT\n", csi);
786
#endif
787
239k
        return false;
788
319k
    }
789
790
0
    return false;
791
319k
}
792
793
int pdfi_trans_setup(pdf_context *ctx, pdfi_trans_state_t *state, gs_rect *bbox,
794
                           pdfi_transparency_caller_t caller)
795
8.66M
{
796
8.66M
    pdfi_int_gstate *igs = (pdfi_int_gstate *)ctx->pgs->client_data;
797
8.66M
    int code;
798
8.66M
    bool stroked_bbox;
799
8.66M
    bool current_overprint;
800
8.66M
    bool okOPcs = false;
801
8.66M
    bool ChangeBM = false;
802
8.66M
    gs_blend_mode_t  mode = gs_currentblendmode(ctx->pgs);  /* quite warning */
803
8.66M
    bool need_group = false;
804
805
8.66M
    memset(state, 0, sizeof(*state));
806
807
8.66M
    if (!ctx->page.has_transparency)
808
6.82M
        return 0;
809
810
1.83M
    if (ctx->page.needs_OP) {
811
310k
        okOPcs = pdfi_trans_okOPcs(ctx);
812
310k
        if (okOPcs) {
813
80.1k
            if (caller == TRANSPARENCY_Caller_Stroke)
814
33.1k
                current_overprint = gs_currentstrokeoverprint(ctx->pgs);
815
46.9k
            else {
816
46.9k
                current_overprint = gs_currentfilloverprint(ctx->pgs);
817
46.9k
                if (caller == TRANSPARENCY_Caller_FillStroke)
818
1.47k
                    current_overprint |= gs_currentstrokeoverprint(ctx->pgs);
819
46.9k
            }
820
80.1k
            ChangeBM = current_overprint;
821
80.1k
            mode = gs_currentblendmode(ctx->pgs);
822
80.1k
            if (mode != BLEND_MODE_Normal && mode != BLEND_MODE_Compatible)
823
589
                need_group = ChangeBM;
824
79.5k
            else
825
79.5k
                need_group = false;
826
229k
        } else {
827
229k
            need_group = false;
828
229k
        }
829
310k
        need_group = need_group || (igs->SMask != NULL);
830
1.52M
    } else {
831
1.52M
        if (caller == TRANSPARENCY_Caller_Image || igs->SMask == NULL)
832
1.51M
            need_group = false;
833
11.0k
        else
834
11.0k
            need_group = true;
835
1.52M
    }
836
837
1.83M
    code = pdfi_trans_set_params(ctx);
838
1.83M
    if (code != 0)
839
2.79k
        return 0;
840
841
1.83M
    if (!need_group && !ChangeBM)
842
1.82M
        return 0;
843
844
    /* TODO: error handling... */
845
12.8k
    if (need_group) {
846
9.08k
        bool isolated = false;
847
9.08k
        mode = gs_currentblendmode(ctx->pgs);
848
849
9.08k
        stroked_bbox = (caller == TRANSPARENCY_Caller_Stroke || caller == TRANSPARENCY_Caller_FillStroke);
850
851
        /* When changing to compatible overprint bm, the group pushed must be non-isolated. The exception
852
           is if we have a softmask AND the blend mode is not normal and not compatible.
853
           See /setupOPtrans in pdf_ops.ps  */
854
9.08k
        if (igs->SMask != NULL && mode != BLEND_MODE_Normal && mode != BLEND_MODE_Compatible)
855
4.81k
            isolated = true;
856
9.08k
        code = pdfi_trans_begin_simple_group(ctx, bbox, stroked_bbox, isolated, false);
857
858
        /* Group was not pushed if error */
859
9.08k
        if (code >= 0)
860
7.24k
            state->GroupPushed = true;
861
862
9.08k
        state->saveStrokeAlpha = gs_getstrokeconstantalpha(ctx->pgs);
863
9.08k
        state->saveFillAlpha = gs_getfillconstantalpha(ctx->pgs);
864
9.08k
        code = gs_setfillconstantalpha(ctx->pgs, 1.0);
865
9.08k
        code = gs_setstrokeconstantalpha(ctx->pgs, 1.0);
866
9.08k
    }
867
868
    /* If we are in a fill stroke situation, do not change the blend mode.
869
       We can have situations where the fill and the stroke have different
870
       blending needs (one may need compatible overprint and the other may
871
       need the normal mode) This is all handled in the fill stroke logic */
872
12.8k
    if (ChangeBM && caller != TRANSPARENCY_Caller_FillStroke) {
873
4.02k
        state->saveBM = mode;
874
4.02k
        state->ChangeBM = true;
875
4.02k
        code = gs_setblendmode(ctx->pgs, BLEND_MODE_CompatibleOverprint);
876
4.02k
    }
877
12.8k
    return code;
878
1.83M
}
879
880
int pdfi_trans_required(pdf_context *ctx)
881
55.4M
{
882
55.4M
    gs_blend_mode_t mode;
883
884
55.4M
    if (!ctx->page.has_transparency)
885
43.2M
        return 0;
886
887
12.2M
    mode = gs_currentblendmode(ctx->pgs);
888
12.2M
    if ((mode == BLEND_MODE_Normal || mode == BLEND_MODE_Compatible) &&
889
12.2M
        ctx->pgs->fillconstantalpha == 1 &&
890
12.2M
        ctx->pgs->strokeconstantalpha == 1 &&
891
12.2M
        ((pdfi_int_gstate *)ctx->pgs->client_data)->SMask == NULL)
892
12.2M
        return 0;
893
894
61.2k
    return 1;
895
12.2M
}
896
897
int pdfi_trans_setup_text(pdf_context *ctx, pdfi_trans_state_t *state, bool is_show)
898
27.4M
{
899
27.4M
    int Trmode;
900
27.4M
    int code, code1;
901
27.4M
    gs_rect bbox;
902
903
27.4M
    if (!pdfi_trans_required(ctx))
904
27.4M
        return 0;
905
906
23.8k
    Trmode = gs_currenttextrenderingmode(ctx->pgs);
907
23.8k
    code = gs_gsave(ctx->pgs);
908
23.8k
    if (code < 0) goto exit;
909
910
23.8k
    if (is_show) {
911
23.8k
        code = gs_clippath(ctx->pgs);
912
23.8k
    } else {
913
0
        code = gs_strokepath(ctx->pgs);
914
0
    }
915
23.8k
    if (code >= 0)
916
23.8k
        code = gs_upathbbox(ctx->pgs, &bbox, false);
917
23.8k
    if (code < 0) {
918
        /* Just set bbox to [0,0,0,0] */
919
69
        bbox.p.x = bbox.p.y = bbox.q.x = bbox.q.y = 0.0;
920
69
        code = 0;
921
69
    }
922
23.8k
    code1 = gs_grestore(ctx->pgs);
923
23.8k
    if (code == 0) code = code1;
924
23.8k
    if (code < 0) goto exit;
925
926
23.8k
    switch (Trmode) {
927
23.8k
    case 0:
928
23.8k
        code = pdfi_trans_setup(ctx, state, &bbox, TRANSPARENCY_Caller_Fill);
929
23.8k
        break;
930
63
    default:
931
        /* TODO: All the others */
932
63
        code = pdfi_trans_setup(ctx, state, &bbox, TRANSPARENCY_Caller_Fill);
933
63
        break;
934
23.8k
    }
935
936
23.8k
 exit:
937
23.8k
    return code;
938
23.8k
}
939
940
int pdfi_trans_teardown_text(pdf_context *ctx, pdfi_trans_state_t *state)
941
27.4M
{
942
27.4M
    if (!pdfi_trans_required(ctx))
943
27.4M
         return 0;
944
945
23.8k
    return pdfi_trans_teardown(ctx, state);
946
27.4M
}
947
948
int pdfi_trans_teardown(pdf_context *ctx, pdfi_trans_state_t *state)
949
8.66M
{
950
8.66M
    int code = 0;
951
952
8.66M
    if (!ctx->page.has_transparency)
953
6.82M
        return 0;
954
955
1.83M
    if (state->GroupPushed) {
956
7.24k
        code = pdfi_trans_end_simple_group(ctx);
957
7.24k
        code = gs_setstrokeconstantalpha(ctx->pgs, state->saveStrokeAlpha);
958
7.24k
        code = gs_setfillconstantalpha(ctx->pgs, state->saveFillAlpha);
959
7.24k
    }
960
961
1.83M
    if (gs_currentblendmode(ctx->pgs) == BLEND_MODE_CompatibleOverprint)
962
4.02k
        code = gs_setblendmode(ctx->pgs, state->saveBM);
963
964
1.83M
    return code;
965
8.66M
}
966
967
int pdfi_trans_set_params(pdf_context *ctx)
968
2.49M
{
969
2.49M
    int code = 0;
970
2.49M
    pdfi_int_gstate *igs = (pdfi_int_gstate *)ctx->pgs->client_data;
971
2.49M
    gs_transparency_channel_selector_t csel;
972
973
2.49M
    if (ctx->page.has_transparency) {
974
2.18M
        if (gs_getalphaisshape(ctx->pgs))
975
10.4k
            csel = TRANSPARENCY_CHANNEL_Shape;
976
2.17M
        else
977
2.17M
            csel = TRANSPARENCY_CHANNEL_Opacity;
978
2.18M
        if (igs->SMask) {
979
14.4k
            code = pdfi_trans_set_mask(ctx, igs, csel);
980
14.4k
        }
981
2.18M
    }
982
983
2.49M
    return code;
984
2.49M
}