Coverage Report

Created: 2026-02-14 07:09

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ghostpdl/pdf/pdf_check.c
Line
Count
Source
1
/* Copyright (C) 2019-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
/* Checks for transparency and spots for the PDF interpreter */
17
18
#include "pdf_int.h"
19
#include "pdf_stack.h"
20
#include "pdf_deref.h"
21
#include "pdf_page.h"
22
#include "pdf_file.h"
23
#include "pdf_dict.h"
24
#include "pdf_array.h"
25
#include "pdf_loop_detect.h"
26
#include "pdf_colour.h"
27
#include "pdf_trans.h"
28
#include "pdf_font_types.h"
29
#include "pdf_gstate.h"
30
#include "pdf_misc.h"
31
#include "pdf_check.h"
32
#include "pdf_device.h"
33
#include "gsdevice.h"       /* For gs_setdevice_no_erase */
34
#include "gspaint.h"        /* For gs_erasepage() */
35
36
/* For performance and resource reasons we do not want to install the transparency blending
37
 * compositor unless we need it. Similarly, if a device handles spot colours it can minimise
38
 * memory usage if it knows ahead of time how many spot colours there will be.
39
 *
40
 * The PDF interpreter written in PostScript performed these as two separate tasks, on opening
41
 * a PDF file it would count spot colour usage and then for each page it would chek if the page
42
 * used any transparency. The code below is used to check for both transparency and spot colours.
43
 * If the int pointer to num_spots is NULL then we aren't interested in spot colours (not supported
44
 * by the device), if the int pointer to transparent is NULL then we aren't interested in transparency
45
 * (-dNOTRANSPARENCY is set).
46
 *
47
 * Currently the code is run when we open the PDF file, the existence of transparency on any given
48
 * page is recorded in a bit array for later use. If it turns out that checking every page when we
49
 * open the file is a performance burden then we could do it on a per-page basis instead. NB if
50
 * the device supports spot colours then we do need to scan the file before starting any page.
51
 *
52
 * The technique is fairly straight-forward, we start with each page, and open its Resources
53
 * dictionary, we then check by type each possible resource. Some resources (eg Pattern, XObject)
54
 * can themselves contain Resources, in which case we recursively check that dictionary. Note; not
55
 * all Resource types need to be checked for both transparency and spot colours, some types can
56
 * only contain one or the other.
57
 *
58
 * Routines with the name pdfi_check_xxx_dict are intended to check a Resource dictionary entry, this
59
 * will be a dictioanry of names and values, where the values are objects of the given Resource type.
60
 *
61
 */
62
63
/* Structure for keeping track of resources as they are being checked */
64
/* CheckedResources is a bit of a hack, for the file Bug697655.pdf. That file has many (unused)
65
 * XObjects which use Resources and the Resources use other XObjects and so on nested
66
 * very deeply. This takes *hours* to check. Ghostscript gets around this because it
67
 * stores all the objects (which we don't want to do because its wasteful) and checking
68
 * to see if its already tested a given resource for spots/transparency.
69
 * This is a temporary allocation, big enough to hold all the objects in the file (1 per bit)
70
 * each time we have fully checked a resource we add it here, when checking a resource we
71
 * first check this list to see if its already been checked, in which case we can skip
72
 * it. When done we release the memory.
73
 */
74
typedef struct {
75
    bool transparent;
76
    bool BM_Not_Normal;
77
    bool has_overprint; /* Does it have OP or op in an ExtGState? */
78
    pdf_dict *spot_dict;
79
    pdf_array *font_array;
80
    uint32_t size;
81
    byte *CheckedResources;
82
} pdfi_check_tracker_t;
83
84
static int pdfi_check_Resources(pdf_context *ctx, pdf_dict *Resources_dict, pdf_dict *page_dict, pdfi_check_tracker_t *tracker);
85
86
static inline bool resource_is_checked(pdfi_check_tracker_t *tracker, pdf_obj *o)
87
4.96M
{
88
4.96M
    uint32_t byte_offset;
89
4.96M
    byte bit_offset;
90
4.96M
    int object_num;
91
92
4.96M
    if(tracker->CheckedResources == NULL)
93
59.1k
        return 0;
94
95
    /* objects with object number 0 are directly defined, we can't
96
     * store those so just return immediately
97
     */
98
4.90M
    object_num = pdf_object_num(o);
99
4.90M
    if (object_num > 0 && (object_num >> 3) < tracker->size) {
100
        /* CheckedResources is a byte array, each byte represents
101
         * 8 objects. So the object number / 8 is the byte offset
102
         * into the array, and then object number % 8 is the bit
103
         * within that byte that we want.
104
         */
105
2.03M
        bit_offset = 0x01 << (object_num % 8);
106
2.03M
        byte_offset = object_num >> 3;
107
108
        /* If its already set, then return that. */
109
2.03M
        if (tracker->CheckedResources[byte_offset] & bit_offset)
110
237k
            return true;
111
1.80M
        else
112
            /* Otherwise set it for futre reference */
113
1.80M
            tracker->CheckedResources[byte_offset] |= bit_offset;
114
2.03M
    }
115
4.67M
    return false;
116
4.90M
}
117
118
119
static int
120
pdfi_check_free_tracker(pdf_context *ctx, pdfi_check_tracker_t *tracker)
121
219k
{
122
219k
    gs_free_object(ctx->memory, tracker->CheckedResources, "pdfi_check_free_tracker(flags)");
123
219k
    pdfi_countdown(tracker->spot_dict);
124
219k
    pdfi_countdown(tracker->font_array);
125
219k
    memset(tracker, 0, sizeof(*tracker));
126
219k
    return 0;
127
219k
}
128
129
static int
130
pdfi_check_init_tracker(pdf_context *ctx, pdfi_check_tracker_t *tracker, pdf_array **fonts_array, pdf_array **spot_array)
131
219k
{
132
219k
    int code = 0;
133
134
219k
    memset(tracker, 0, sizeof(*tracker));
135
136
219k
    tracker->size = (ctx->xref_table->xref_size + 7) / 8;
137
219k
    tracker->CheckedResources = gs_alloc_bytes(ctx->memory, tracker->size,
138
219k
                                            "pdfi_check_init_tracker(flags)");
139
219k
    if (tracker->CheckedResources == NULL)
140
0
        return_error(gs_error_VMerror);
141
142
219k
    memset(tracker->CheckedResources, 0x00, tracker->size);
143
144
219k
    if (ctx->device_state.spot_capable ||
145
194k
        (ctx->pgs->device->icc_struct->overprint_control) == gs_overprint_control_simulate ||
146
194k
        spot_array != NULL)
147
24.9k
    {
148
24.9k
        code = pdfi_dict_alloc(ctx, 32, &tracker->spot_dict);
149
24.9k
        if (code < 0)
150
0
            goto cleanup;
151
24.9k
        pdfi_countup(tracker->spot_dict);
152
24.9k
    }
153
154
219k
    if (fonts_array != NULL) {
155
0
        code = pdfi_array_alloc(ctx, 0, &tracker->font_array);
156
0
        if (code < 0)
157
0
            goto cleanup;
158
0
        pdfi_countup(tracker->font_array);
159
0
    }
160
161
219k
    return 0;
162
163
0
cleanup:
164
0
    pdfi_check_free_tracker(ctx, tracker);
165
0
    return code;
166
219k
}
167
168
/*
169
 * Check the Resources dictionary ColorSpace entry. pdfi_check_ColorSpace_for_spots is defined
170
 * in pdf_colour.c
171
 */
172
static int pdfi_check_ColorSpace_dict(pdf_context *ctx, pdf_dict *cspace_dict,
173
                                      pdf_dict *page_dict, pdfi_check_tracker_t *tracker)
174
57.2k
{
175
57.2k
    int code;
176
57.2k
    uint64_t i, index;
177
57.2k
    pdf_obj *Key = NULL, *Value = NULL;
178
179
57.2k
    if (resource_is_checked(tracker, (pdf_obj *)cspace_dict))
180
24
        return 0;
181
182
57.2k
    if (pdfi_type_of(cspace_dict) != PDF_DICT)
183
49.7k
        return_error(gs_error_typecheck);
184
185
7.44k
    if (pdfi_dict_entries(cspace_dict) > 0) {
186
7.42k
        code = pdfi_loop_detector_mark(ctx); /* Mark the start of the ColorSpace dictionary loop */
187
7.42k
        if (code < 0)
188
0
            return code;
189
190
7.42k
        code = pdfi_dict_first(ctx, cspace_dict, &Key, &Value, &index);
191
7.42k
        if (code < 0)
192
1.22k
            goto error1;
193
194
6.20k
        i = 1;
195
8.37k
        do {
196
8.37k
            code = pdfi_check_ColorSpace_for_spots(ctx, Value, cspace_dict, page_dict, tracker->spot_dict);
197
8.37k
            if (code < 0)
198
128
                goto error2;
199
200
8.24k
            pdfi_countdown(Key);
201
8.24k
            Key = NULL;
202
8.24k
            pdfi_countdown(Value);
203
8.24k
            Value = NULL;
204
205
8.24k
            (void)pdfi_loop_detector_cleartomark(ctx); /* Clear to the mark for the Shading dictionary loop */
206
207
8.24k
            code = pdfi_loop_detector_mark(ctx); /* Mark the new start of the Shading dictionary loop */
208
8.24k
            if (code < 0)
209
0
                goto error1;
210
211
8.40k
            do {
212
8.40k
                if (i++ >= pdfi_dict_entries(cspace_dict)) {
213
6.07k
                    code = 0;
214
6.07k
                    goto transparency_exit;
215
6.07k
                }
216
217
2.32k
                code = pdfi_dict_next(ctx, cspace_dict, &Key, &Value, &index);
218
2.32k
                if (code == 0 && pdfi_type_of(Value) == PDF_ARRAY)
219
2.17k
                    break;
220
156
                pdfi_countdown(Key);
221
156
                Key = NULL;
222
156
                pdfi_countdown(Value);
223
156
                Value = NULL;
224
156
            } while(1);
225
8.24k
        }while (1);
226
6.20k
    }
227
14
    return 0;
228
229
6.07k
transparency_exit:
230
6.20k
error2:
231
6.20k
    pdfi_countdown(Key);
232
6.20k
    pdfi_countdown(Value);
233
234
7.42k
error1:
235
7.42k
    (void)pdfi_loop_detector_cleartomark(ctx); /* Clear to the mark for the current resource loop */
236
7.42k
    return code;
237
6.20k
}
238
239
/*
240
 * Process an individual Shading dictionary to see if it contains a ColorSpace with a spot colour
241
 */
242
static int pdfi_check_Shading(pdf_context *ctx, pdf_obj *shading,
243
                              pdf_dict *page_dict, pdfi_check_tracker_t *tracker)
244
6.18k
{
245
6.18k
    int code;
246
6.18k
    pdf_obj *o = NULL;
247
6.18k
    pdf_dict *shading_dict = NULL;
248
249
6.18k
    if (resource_is_checked(tracker, shading))
250
418
        return 0;
251
252
5.76k
    code = pdfi_dict_from_obj(ctx, shading, &shading_dict);
253
5.76k
    if (code < 0)
254
6
        return code;
255
256
5.76k
    if (pdfi_type_of(shading_dict) != PDF_DICT)
257
0
        return_error(gs_error_typecheck);
258
259
5.76k
    code = pdfi_dict_knownget(ctx, shading_dict, "ColorSpace", (pdf_obj **)&o);
260
5.76k
    if (code > 0) {
261
5.43k
        code = pdfi_check_ColorSpace_for_spots(ctx, o, shading_dict, page_dict, tracker->spot_dict);
262
5.43k
        pdfi_countdown(o);
263
5.43k
        return code;
264
5.43k
    }
265
332
    return 0;
266
5.76k
}
267
268
/*
269
 * Check the Resources dictionary Shading entry.
270
 */
271
static int pdfi_check_Shading_dict(pdf_context *ctx, pdf_dict *shading_dict,
272
                                   pdf_dict *page_dict, pdfi_check_tracker_t *tracker)
273
57.2k
{
274
57.2k
    int code;
275
57.2k
    uint64_t i, index;
276
57.2k
    pdf_obj *Key = NULL, *Value = NULL;
277
278
57.2k
    if (resource_is_checked(tracker, (pdf_obj *)shading_dict))
279
0
        return 0;
280
281
57.2k
    if (pdfi_type_of(shading_dict) != PDF_DICT)
282
55.4k
        return_error(gs_error_typecheck);
283
284
1.82k
    if (pdfi_dict_entries(shading_dict) > 0) {
285
1.82k
        code = pdfi_loop_detector_mark(ctx); /* Mark the start of the Shading dictionary loop */
286
1.82k
        if (code < 0)
287
0
            return code;
288
289
1.82k
        code = pdfi_dict_first(ctx, shading_dict, &Key, &Value, &index);
290
1.82k
        if (code < 0)
291
986
            goto error2;
292
293
834
        i = 1;
294
5.42k
        do {
295
5.42k
            if ((pdfi_type_of(Value) == PDF_DICT || pdfi_type_of(Value) == PDF_STREAM)) {
296
5.41k
                code = pdfi_check_Shading(ctx, Value, page_dict, tracker);
297
5.41k
                if (code < 0)
298
22
                    goto error2;
299
5.41k
            }
300
301
5.40k
            pdfi_countdown(Key);
302
5.40k
            Key = NULL;
303
5.40k
            pdfi_countdown(Value);
304
5.40k
            Value = NULL;
305
306
5.40k
            (void)pdfi_loop_detector_cleartomark(ctx); /* Clear to the mark for the Shading dictionary loop */
307
308
5.40k
            code = pdfi_loop_detector_mark(ctx); /* Mark the new start of the Shading dictionary loop */
309
5.40k
            if (code < 0)
310
0
                goto error1;
311
312
10.9k
            do {
313
10.9k
                if (i++ >= pdfi_dict_entries(shading_dict)) {
314
812
                    code = 0;
315
812
                    goto transparency_exit;
316
812
                }
317
318
10.1k
                code = pdfi_dict_next(ctx, shading_dict, &Key, &Value, &index);
319
10.1k
                if (code == 0 && pdfi_type_of(Value) == PDF_DICT)
320
4.58k
                    break;
321
5.57k
                pdfi_countdown(Key);
322
5.57k
                Key = NULL;
323
5.57k
                pdfi_countdown(Value);
324
5.57k
                Value = NULL;
325
5.57k
            } while(1);
326
5.40k
        }while (1);
327
834
    }
328
0
    return 0;
329
330
812
transparency_exit:
331
1.82k
error2:
332
1.82k
    pdfi_countdown(Key);
333
1.82k
    pdfi_countdown(Value);
334
335
1.82k
error1:
336
1.82k
    (void)pdfi_loop_detector_cleartomark(ctx); /* Clear to the mark for the current resource loop */
337
1.82k
    return code;
338
1.82k
}
339
340
/*
341
 * This routine checks an XObject to see if it contains any spot
342
 * colour definitions, or transparency usage.
343
 */
344
static int pdfi_check_XObject(pdf_context *ctx, pdf_dict *xobject, pdf_dict *page_dict,
345
                              pdfi_check_tracker_t *tracker)
346
343k
{
347
343k
    int code = 0;
348
343k
    pdf_name *n = NULL;
349
343k
    bool known = false;
350
343k
    double f;
351
352
343k
    if (resource_is_checked(tracker, (pdf_obj *)xobject))
353
123k
        return 0;
354
355
220k
    if (pdfi_type_of(xobject) != PDF_DICT)
356
250
        return_error(gs_error_typecheck);
357
358
219k
    code = pdfi_dict_get_type(ctx, xobject, "Subtype", PDF_NAME, (pdf_obj **)&n);
359
219k
    if (code >= 0) {
360
218k
        if (pdfi_name_is((const pdf_name *)n, "Image")) {
361
114k
            pdf_obj *CS = NULL;
362
363
114k
            pdfi_countdown(n);
364
114k
            n = NULL;
365
114k
            code = pdfi_dict_known(ctx, xobject, "SMask", &known);
366
114k
            if (code >= 0) {
367
114k
                if (known == true) {
368
27.2k
                    tracker->transparent = true;
369
27.2k
                    if (tracker->spot_dict == NULL)
370
23.7k
                        goto transparency_exit;
371
27.2k
                }
372
90.6k
                code = pdfi_dict_knownget_number(ctx, xobject, "SMaskInData", &f);
373
90.6k
                if (code > 0) {
374
0
                    code = 0;
375
0
                    if (f != 0.0)
376
0
                        tracker->transparent = true;
377
0
                    if (tracker->spot_dict == NULL)
378
0
                        goto transparency_exit;
379
0
                }
380
                /* Check the image dictionary for a ColorSpace entry, if we are checking spot names */
381
90.6k
                if (tracker->spot_dict) {
382
13.4k
                    code = pdfi_dict_knownget(ctx, xobject, "ColorSpace", (pdf_obj **)&CS);
383
13.4k
                    if (code > 0) {
384
                        /* We don't care if there's an error here, it'll be picked up if we use the ColorSpace later */
385
10.6k
                        (void)pdfi_check_ColorSpace_for_spots(ctx, CS, xobject, page_dict, tracker->spot_dict);
386
10.6k
                        pdfi_countdown(CS);
387
10.6k
                    }
388
13.4k
                }
389
90.6k
            }
390
114k
        } else {
391
104k
            if (pdfi_name_is((const pdf_name *)n, "Form")) {
392
103k
                pdf_dict *group_dict = NULL, *resource_dict = NULL;
393
103k
                pdf_obj *CS = NULL;
394
395
103k
                pdfi_countdown(n);
396
103k
                code = pdfi_dict_knownget_type(ctx, xobject, "Group", PDF_DICT, (pdf_obj **)&group_dict);
397
103k
                if (code > 0) {
398
12.7k
                    tracker->transparent = true;
399
12.7k
                    if (tracker->spot_dict != NULL) {
400
                        /* Start a new loop detector group to avoid this being detected in the Resources check below */
401
1.67k
                        code = pdfi_loop_detector_mark(ctx); /* Mark the start of the XObject dictionary loop */
402
1.67k
                        if (code == 0) {
403
1.67k
                            code = pdfi_dict_knownget(ctx, group_dict, "CS", &CS);
404
1.67k
                            if (code > 0)
405
                                /* We don't care if there's an error here, it'll be picked up if we use the ColorSpace later */
406
1.16k
                                (void)pdfi_check_ColorSpace_for_spots(ctx, CS, group_dict, page_dict, tracker->spot_dict);
407
1.67k
                            (void)pdfi_loop_detector_cleartomark(ctx); /* Clear to the mark for the XObject dictionary loop */
408
1.67k
                        }
409
1.67k
                        pdfi_countdown(group_dict);
410
1.67k
                        pdfi_countdown(CS);
411
1.67k
                    }
412
11.0k
                    else if (tracker->BM_Not_Normal)
413
72
                    {
414
                        /* We already know it has a non-normal Blend Mode. No point in keeping searching. */
415
72
                        pdfi_countdown(group_dict);
416
72
                        goto transparency_exit;
417
72
                    }
418
10.9k
                    else
419
10.9k
                    {
420
10.9k
                        pdfi_countdown(group_dict);
421
10.9k
                    }
422
                    /* We need to keep checking Resources in case there are non-Normal blend mode things still to be found. */
423
12.7k
                }
424
425
103k
                code = pdfi_dict_knownget_type(ctx, xobject, "Resources", PDF_DICT, (pdf_obj **)&resource_dict);
426
103k
                if (code > 0) {
427
100k
                    if (ctx->loop_detection && pdf_object_num((pdf_obj *)resource_dict) != 0) {
428
66.3k
                        code = pdfi_loop_detector_add_object(ctx, resource_dict->object_num);
429
66.3k
                        if (code < 0) {
430
0
                            pdfi_countdown(resource_dict);
431
0
                            goto transparency_exit;
432
0
                        }
433
66.3k
                    }
434
100k
                    code = pdfi_check_Resources(ctx, resource_dict, page_dict, tracker);
435
100k
                    pdfi_countdown(resource_dict);
436
100k
                    if (code < 0)
437
0
                        goto transparency_exit;
438
100k
                }
439
103k
            } else
440
518
                pdfi_countdown(n);
441
104k
        }
442
218k
    }
443
444
196k
    return 0;
445
446
23.7k
transparency_exit:
447
23.7k
    return code;
448
219k
}
449
450
/*
451
 * Check the Resources dictionary XObject entry.
452
 */
453
static int pdfi_check_XObject_dict(pdf_context *ctx, pdf_dict *xobject_dict, pdf_dict *page_dict,
454
                                   pdfi_check_tracker_t *tracker)
455
593k
{
456
593k
    int code;
457
593k
    uint64_t i, index;
458
593k
    pdf_obj *Key = NULL, *Value = NULL;
459
593k
    pdf_dict *Value_dict = NULL;
460
461
593k
    if (resource_is_checked(tracker, (pdf_obj *)xobject_dict))
462
0
        return 0;
463
464
593k
    if (pdfi_type_of(xobject_dict) != PDF_DICT)
465
370k
        return_error(gs_error_typecheck);
466
467
222k
    if (pdfi_dict_entries(xobject_dict) > 0) {
468
221k
        code = pdfi_loop_detector_mark(ctx); /* Mark the start of the XObject dictionary loop */
469
221k
        if (code < 0)
470
0
            return code;
471
472
221k
        code = pdfi_dict_first(ctx, xobject_dict, &Key, &Value, &index);
473
221k
        if (code < 0)
474
41.9k
            goto error_exit;
475
476
179k
        i = 1;
477
344k
        do {
478
344k
            if (pdfi_type_of(Value) == PDF_STREAM) {
479
342k
                code = pdfi_dict_from_obj(ctx, Value, &Value_dict);
480
342k
                if (code < 0)
481
0
                    goto error_exit;
482
483
342k
                code = pdfi_check_XObject(ctx, Value_dict, page_dict, tracker);
484
342k
                if (code < 0)
485
0
                    goto error_exit;
486
342k
            }
487
488
344k
            (void)pdfi_loop_detector_cleartomark(ctx); /* Clear to the mark for the XObject dictionary loop */
489
490
344k
            code = pdfi_loop_detector_mark(ctx); /* Mark the new start of the XObject dictionary loop */
491
344k
            if (code < 0)
492
0
                goto error_exit;
493
494
344k
            pdfi_countdown(Key);
495
344k
            Key = NULL;
496
344k
            pdfi_countdown(Value);
497
344k
            Value = NULL;
498
344k
            Value_dict = NULL;
499
500
395k
            do {
501
395k
                if (i++ >= pdfi_dict_entries(xobject_dict)) {
502
179k
                    code = 0;
503
179k
                    goto transparency_exit;
504
179k
                }
505
506
215k
                code = pdfi_dict_next(ctx, xobject_dict, &Key, &Value, &index);
507
215k
                if (code == 0 && pdfi_type_of(Value) == PDF_STREAM)
508
164k
                    break;
509
50.8k
                pdfi_countdown(Key);
510
50.8k
                Key = NULL;
511
50.8k
                pdfi_countdown(Value);
512
50.8k
                Value = NULL;
513
50.8k
            } while(1);
514
344k
        }while(1);
515
179k
    }
516
1.67k
    return 0;
517
518
179k
transparency_exit:
519
221k
error_exit:
520
221k
    pdfi_countdown(Key);
521
221k
    pdfi_countdown(Value);
522
221k
    (void)pdfi_loop_detector_cleartomark(ctx); /* Clear to the mark for the current resource loop */
523
221k
    return code;
524
179k
}
525
526
/*
527
 * This routine checks an ExtGState dictionary to see if it contains transparency or OP
528
 */
529
static int pdfi_check_ExtGState(pdf_context *ctx, pdf_dict *extgstate_dict, pdf_dict *page_dict,
530
                                pdfi_check_tracker_t *tracker)
531
187k
{
532
187k
    int code;
533
187k
    pdf_obj *o = NULL;
534
187k
    double f;
535
187k
    bool overprint;
536
537
187k
    if (resource_is_checked(tracker, (pdf_obj *)extgstate_dict))
538
59.3k
        return 0;
539
540
128k
    if (pdfi_type_of(extgstate_dict) != PDF_DICT)
541
1.36k
        return_error(gs_error_typecheck);
542
543
126k
    if (pdfi_dict_entries(extgstate_dict) > 0) {
544
        /* See if /OP or /op is true */
545
126k
        code = pdfi_dict_get_bool(ctx, extgstate_dict, "OP", &overprint);
546
126k
        if (code == 0 && overprint)
547
5.57k
            tracker->has_overprint = true;
548
126k
        code = pdfi_dict_get_bool(ctx, extgstate_dict, "op", &overprint);
549
126k
        if (code == 0 && overprint)
550
6.42k
            tracker->has_overprint = true;
551
552
        /* Check Blend Mode *first* because we short-circuit transparency detection
553
         * when we find transparency, and we want to be able to record the use of
554
         * blend modes other than Compatible/Normal so that Patterns using these
555
         * always use the clist and not tiles. The tiles implementation can't work
556
         * if we change to a blend mode other than Normal or Compatible during
557
         * the course of a Pattern.
558
         */
559
126k
        code = pdfi_dict_knownget_type(ctx, extgstate_dict, "BM", PDF_NAME, &o);
560
126k
        if (code > 0) {
561
51.9k
            if (!pdfi_name_is((pdf_name *)o, "Normal")) {
562
7.48k
                if (!pdfi_name_is((pdf_name *)o, "Compatible")) {
563
7.43k
                    pdfi_countdown(o);
564
7.43k
                    tracker->transparent = true;
565
7.43k
                    tracker->BM_Not_Normal = true;
566
7.43k
                    return 0;
567
7.43k
                }
568
7.48k
            }
569
51.9k
        }
570
119k
        pdfi_countdown(o);
571
119k
        o = NULL;
572
573
        /* Check SMask */
574
119k
        code = pdfi_dict_knownget(ctx, extgstate_dict, "SMask", &o);
575
119k
        if (code > 0) {
576
18.1k
            switch (pdfi_type_of(o)) {
577
15.0k
                case PDF_NAME:
578
15.0k
                    if (!pdfi_name_is((pdf_name *)o, "None")) {
579
52
                        pdfi_countdown(o);
580
52
                        tracker->transparent = true;
581
52
                        return 0;
582
52
                    }
583
14.9k
                    break;
584
14.9k
                case PDF_DICT:
585
3.13k
                {
586
3.13k
                    pdf_obj *G = NULL;
587
588
3.13k
                    tracker->transparent = true;
589
590
3.13k
                    if (tracker->spot_dict != NULL) {
591
                        /* Check if the SMask has a /G (Group) */
592
387
                        code = pdfi_dict_knownget(ctx, (pdf_dict *)o, "G", &G);
593
387
                        if (code > 0) {
594
250
                            code = pdfi_check_XObject(ctx, (pdf_dict *)G, page_dict,
595
250
                                                      tracker);
596
250
                            pdfi_countdown(G);
597
250
                        }
598
387
                    }
599
3.13k
                    pdfi_countdown(o);
600
3.13k
                    return code;
601
15.0k
                }
602
0
                default:
603
0
                    break;
604
18.1k
            }
605
18.1k
        }
606
116k
        pdfi_countdown(o);
607
116k
        o = NULL;
608
609
116k
        code = pdfi_dict_knownget_number(ctx, extgstate_dict, "CA", &f);
610
116k
        if (code > 0) {
611
43.9k
            if (f != 1.0) {
612
8.00k
                tracker->transparent = true;
613
8.00k
                return 0;
614
8.00k
            }
615
43.9k
        }
616
617
108k
        code = pdfi_dict_knownget_number(ctx, extgstate_dict, "ca", &f);
618
108k
        if (code > 0) {
619
38.7k
            if (f != 1.0) {
620
1.24k
                tracker->transparent = true;
621
1.24k
                return 0;
622
1.24k
            }
623
38.7k
        }
624
625
108k
    }
626
106k
    return 0;
627
126k
}
628
629
/*
630
 * Check the Resources dictionary ExtGState entry.
631
 */
632
static int pdfi_check_ExtGState_dict(pdf_context *ctx, pdf_dict *extgstate_dict, pdf_dict *page_dict,
633
                                     pdfi_check_tracker_t *tracker)
634
593k
{
635
593k
    int code;
636
593k
    uint64_t i, index;
637
593k
    pdf_obj *Key = NULL, *Value = NULL;
638
639
593k
    if (resource_is_checked(tracker, (pdf_obj *)extgstate_dict))
640
132
        return 0;
641
642
593k
    if (pdfi_type_of(extgstate_dict) != PDF_DICT)
643
436k
        return_error(gs_error_typecheck);
644
645
157k
    if (pdfi_dict_entries(extgstate_dict) > 0) {
646
155k
        code = pdfi_loop_detector_mark(ctx); /* Mark the start of the ColorSpace dictionary loop */
647
155k
        if (code < 0)
648
0
            return code;
649
650
155k
        code = pdfi_dict_first(ctx, extgstate_dict, &Key, &Value, &index);
651
155k
        if (code < 0)
652
17.2k
            goto error1;
653
654
138k
        i = 1;
655
187k
        do {
656
657
187k
            (void)pdfi_check_ExtGState(ctx, (pdf_dict *)Value, page_dict, tracker);
658
187k
            if (tracker->transparent == true && tracker->spot_dict == NULL)
659
18.6k
                goto transparency_exit;
660
661
168k
            pdfi_countdown(Key);
662
168k
            Key = NULL;
663
168k
            pdfi_countdown(Value);
664
168k
            Value = NULL;
665
666
168k
            (void)pdfi_loop_detector_cleartomark(ctx); /* Clear to the mark for the ExtGState dictionary loop */
667
668
168k
            code = pdfi_loop_detector_mark(ctx); /* Mark the new start of the ExtGState dictionary loop */
669
168k
            if (code < 0)
670
0
                goto error1;
671
672
180k
            do {
673
180k
                if (i++ >= pdfi_dict_entries(extgstate_dict)) {
674
119k
                    code = 0;
675
119k
                    goto transparency_exit;
676
119k
                }
677
678
60.1k
                code = pdfi_dict_next(ctx, extgstate_dict, &Key, &Value, &index);
679
60.1k
                if (code == 0 && pdfi_type_of(Value) == PDF_DICT)
680
48.9k
                    break;
681
11.1k
                pdfi_countdown(Key);
682
11.1k
                Key = NULL;
683
11.1k
                pdfi_countdown(Value);
684
11.1k
                Value = NULL;
685
11.1k
            } while(1);
686
168k
        }while (1);
687
138k
    }
688
1.31k
    return 0;
689
690
138k
transparency_exit:
691
138k
    pdfi_countdown(Key);
692
138k
    pdfi_countdown(Value);
693
694
155k
error1:
695
155k
    (void)pdfi_loop_detector_cleartomark(ctx); /* Clear to the mark for the current resource loop */
696
155k
    return code;
697
138k
}
698
699
/*
700
 * This routine checks a Pattern dictionary to see if it contains any spot
701
 * colour definitions, or transparency usage (or blend modes other than Normal/Compatible).
702
 */
703
static int pdfi_check_Pattern(pdf_context *ctx, pdf_dict *pattern, pdf_dict *page_dict,
704
                              pdfi_check_tracker_t *tracker)
705
79.7k
{
706
79.7k
    int code = 0;
707
79.7k
    pdf_obj *o = NULL;
708
709
79.7k
    if (resource_is_checked(tracker, (pdf_obj *)pattern))
710
1.06k
        return 0;
711
712
78.6k
    if (pdfi_type_of(pattern) != PDF_DICT)
713
0
        return_error(gs_error_typecheck);
714
715
78.6k
    if (tracker->spot_dict != NULL) {
716
10.9k
        code = pdfi_dict_knownget(ctx, pattern, "Shading", &o);
717
10.9k
        if (code > 0)
718
774
            (void)pdfi_check_Shading(ctx, o, page_dict, tracker);
719
10.9k
        pdfi_countdown(o);
720
10.9k
        o = NULL;
721
10.9k
    }
722
723
78.6k
    code = pdfi_dict_knownget_type(ctx, pattern, "Resources", PDF_DICT, &o);
724
78.6k
    if (code > 0)
725
69.9k
        (void)pdfi_check_Resources(ctx, (pdf_dict *)o, page_dict, tracker);
726
78.6k
    pdfi_countdown(o);
727
78.6k
    o = NULL;
728
78.6k
    if (tracker->transparent == true && tracker->spot_dict == NULL)
729
20.9k
        goto transparency_exit;
730
731
57.6k
    code = pdfi_dict_knownget_type(ctx, pattern, "ExtGState", PDF_DICT, &o);
732
57.6k
    if (code > 0)
733
0
        (void)pdfi_check_ExtGState(ctx, (pdf_dict *)o, page_dict, tracker);
734
57.6k
    pdfi_countdown(o);
735
57.6k
    o = NULL;
736
737
78.6k
transparency_exit:
738
78.6k
    return 0;
739
57.6k
}
740
741
/*
742
 * This routine checks a Pattern dictionary for transparency.
743
 */
744
int pdfi_check_Pattern_transparency(pdf_context *ctx, pdf_dict *pattern, pdf_dict *page_dict,
745
                                    bool *transparent, bool *BM_Not_Normal)
746
8.71k
{
747
8.71k
    int code;
748
8.71k
    pdfi_check_tracker_t tracker = {0, 0, 0, NULL, NULL, 0, NULL};
749
750
    /* NOTE: We use a "null" tracker that won't do any optimization to prevent
751
     * checking the same resource twice.
752
     * This means we don't get the optimization, but we also don't have the overhead
753
     * of setting it up.
754
     * If we do want the optimization, call pdfi_check_init_tracker() and
755
     * pdfi_check_free_tracker() as in pdfi_check_page(), below.
756
     */
757
8.71k
    code = pdfi_check_Pattern(ctx, pattern, page_dict, &tracker);
758
8.71k
    if (code == 0) {
759
8.71k
        *transparent = tracker.transparent;
760
8.71k
        *BM_Not_Normal = tracker.BM_Not_Normal;
761
8.71k
    }
762
0
    else
763
0
        *transparent = false;
764
8.71k
    return code;
765
8.71k
}
766
767
/*
768
 * Check the Resources dictionary Pattern entry.
769
 */
770
static int pdfi_check_Pattern_dict(pdf_context *ctx, pdf_dict *pattern_dict, pdf_dict *page_dict,
771
                                   pdfi_check_tracker_t *tracker)
772
593k
{
773
593k
    int code;
774
593k
    uint64_t i, index;
775
593k
    pdf_obj *Key = NULL, *Value = NULL;
776
593k
    pdf_dict *instance_dict = NULL;
777
778
593k
    if (resource_is_checked(tracker, (pdf_obj *)pattern_dict))
779
0
        return 0;
780
781
593k
    if (pdfi_type_of(pattern_dict) != PDF_DICT)
782
579k
        return_error(gs_error_typecheck);
783
784
13.9k
    if (pdfi_dict_entries(pattern_dict) > 0) {
785
12.6k
        code = pdfi_loop_detector_mark(ctx); /* Mark the start of the Pattern dictionary loop */
786
12.6k
        if (code < 0)
787
0
            return code;
788
789
12.6k
        code = pdfi_dict_first(ctx, pattern_dict, &Key, &Value, &index);
790
12.6k
        if (code < 0)
791
2.40k
            goto error1;
792
793
10.2k
        i = 1;
794
71.1k
        do {
795
71.1k
            if (pdfi_type_of(Value) == PDF_DICT || pdfi_type_of(Value) == PDF_STREAM) {
796
71.0k
                code = pdfi_dict_from_obj(ctx, Value, &instance_dict);
797
71.0k
                if (code < 0)
798
0
                    goto transparency_exit;
799
800
71.0k
                code = pdfi_check_Pattern(ctx, instance_dict, page_dict, tracker);
801
71.0k
                if (code < 0)
802
0
                    goto transparency_exit;
803
71.0k
            }
804
805
71.1k
            pdfi_countdown(Key);
806
71.1k
            Key = NULL;
807
71.1k
            pdfi_countdown(Value);
808
71.1k
            instance_dict = NULL;
809
71.1k
            Value = NULL;
810
71.1k
            (void)pdfi_loop_detector_cleartomark(ctx); /* Clear to the mark for the Shading dictionary loop */
811
812
71.1k
            code = pdfi_loop_detector_mark(ctx); /* Mark the new start of the Shading dictionary loop */
813
71.1k
            if (code < 0)
814
0
                goto error1;
815
816
187k
            do {
817
187k
                if (i++ >= pdfi_dict_entries(pattern_dict)) {
818
10.2k
                    code = 0;
819
10.2k
                    goto transparency_exit;
820
10.2k
                }
821
822
177k
                code = pdfi_dict_next(ctx, pattern_dict, &Key, &Value, &index);
823
177k
                if (code == 0 && (pdfi_type_of(Value) == PDF_DICT || pdfi_type_of(Value) == PDF_STREAM))
824
60.8k
                    break;
825
116k
                pdfi_countdown(Key);
826
116k
                Key = NULL;
827
116k
                pdfi_countdown(Value);
828
116k
                Value = NULL;
829
116k
            } while(1);
830
71.1k
        }while (1);
831
10.2k
    }
832
1.28k
    return 0;
833
834
10.2k
transparency_exit:
835
10.2k
    pdfi_countdown(Key);
836
10.2k
    pdfi_countdown(Value);
837
10.2k
    pdfi_countdown(instance_dict);
838
839
12.6k
error1:
840
12.6k
    (void)pdfi_loop_detector_cleartomark(ctx); /* Clear to the mark for the current resource loop */
841
12.6k
    return code;
842
10.2k
}
843
844
/*
845
 * This routine checks a Font dictionary to see if it contains any spot
846
 * colour definitions, or transparency usage. While we are here, if the tracker's font_array
847
 * is not NULL, pick up the font information and store it in the array.
848
 */
849
static int pdfi_check_Font(pdf_context *ctx, pdf_dict *font, pdf_dict *page_dict,
850
                           pdfi_check_tracker_t *tracker)
851
359k
{
852
359k
    int code = 0;
853
359k
    pdf_obj *o = NULL;
854
855
359k
    if (resource_is_checked(tracker, (pdf_obj *)font))
856
51.8k
        return 0;
857
858
307k
    if (pdfi_type_of(font) != PDF_DICT)
859
0
        return_error(gs_error_typecheck);
860
861
307k
    if (tracker->font_array != NULL) {
862
        /* If we get to here this is a font we have not seen before. We need
863
         * to make a new font array big enough to hold the existing entries +1
864
         * copy the existing entries to the new array and free the old array.
865
         * Finally create a dictionary with all the font information we want
866
         * and add it to the array.
867
         */
868
0
        pdf_array *new_fonts = NULL;
869
0
        int index = 0;
870
0
        pdf_obj *array_obj = NULL;
871
0
        pdf_dict *font_info_dict = NULL;
872
873
        /* Let's start by gathering the information we need and storing it in a dictionary */
874
0
        code = pdfi_dict_alloc(ctx, 4, &font_info_dict);
875
0
        if (code < 0)
876
0
            return code;
877
0
        pdfi_countup(font_info_dict);
878
879
0
        if (font->object_num != 0) {
880
0
            pdf_num *int_obj = NULL;
881
882
0
            code = pdfi_object_alloc(ctx, PDF_INT, 0, (pdf_obj **)&int_obj);
883
0
            if (code >= 0) {
884
0
                pdfi_countup(int_obj);
885
0
                int_obj->value.i = font->object_num;
886
0
                code = pdfi_dict_put(ctx, font_info_dict, "ObjectNum", (pdf_obj *)int_obj);
887
0
                pdfi_countdown(int_obj);
888
0
            }
889
0
            if (code < 0) {
890
0
                pdfi_countdown(font_info_dict);
891
0
                return code;
892
0
            }
893
0
        }
894
895
        /* Try to get font name with fallback:
896
         * 1. Try /BaseFont (standard for most fonts)
897
         * 2. Try /FontDescriptor/FontName (common for Type 3 fonts)
898
         * 3. Try /Name as last resort
899
         */
900
0
        {
901
0
            pdf_obj *font_name = NULL;
902
0
            int lookup_code;
903
0
            bool found = false;
904
905
0
            lookup_code = pdfi_dict_get(ctx, font, "BaseFont", &font_name);
906
0
            if (lookup_code >= 0 && pdfi_type_of(font_name) == PDF_NAME) {
907
0
                code = pdfi_dict_put(ctx, font_info_dict, "BaseFont", font_name);
908
0
                pdfi_countdown(font_name);
909
0
                if (code < 0) {
910
0
                    pdfi_countdown(font_info_dict);
911
0
                    return code;
912
0
                }
913
0
                found = true;
914
0
            } else {
915
0
                pdfi_countdown(font_name);
916
0
                font_name = NULL;
917
0
            }
918
919
0
            if (!found) {
920
0
                lookup_code = pdfi_dict_get(ctx, font, "FontDescriptor", &font_name);
921
0
                if (lookup_code >= 0 && pdfi_type_of(font_name) == PDF_DICT) {
922
0
                    pdf_dict *font_desc = (pdf_dict *)font_name;
923
0
                    pdf_obj *desc_font_name = NULL;
924
925
0
                    lookup_code = pdfi_dict_get(ctx, font_desc, "FontName", &desc_font_name);
926
0
                    pdfi_countdown(font_name);
927
0
                    font_name = NULL;
928
929
0
                    if (lookup_code >= 0 && pdfi_type_of(desc_font_name) == PDF_NAME) {
930
0
                        code = pdfi_dict_put(ctx, font_info_dict, "BaseFont", desc_font_name);
931
0
                        pdfi_countdown(desc_font_name);
932
0
                        if (code < 0) {
933
0
                            pdfi_countdown(font_info_dict);
934
0
                            return code;
935
0
                        }
936
0
                        found = true;
937
0
                    } else {
938
0
                        pdfi_countdown(desc_font_name);
939
0
                    }
940
0
                } else {
941
0
                    pdfi_countdown(font_name);
942
0
                    font_name = NULL;
943
0
                }
944
0
            }
945
946
0
            if (!found) {
947
0
                lookup_code = pdfi_dict_get(ctx, font, "Name", &font_name);
948
0
                if (lookup_code >= 0 && pdfi_type_of(font_name) == PDF_NAME) {
949
0
                    code = pdfi_dict_put(ctx, font_info_dict, "BaseFont", font_name);
950
0
                    pdfi_countdown(font_name);
951
0
                    if (code < 0) {
952
0
                        pdfi_countdown(font_info_dict);
953
0
                        return code;
954
0
                    }
955
0
                } else {
956
0
                    pdfi_countdown(font_name);
957
0
                }
958
0
            }
959
960
0
            array_obj = NULL;
961
0
        }
962
963
0
        code = pdfi_dict_get(ctx, font, "ToUnicode", &array_obj);
964
0
        if (code >= 0)
965
0
            code = pdfi_dict_put(ctx, font_info_dict, "ToUnicode", PDF_TRUE_OBJ);
966
0
        else
967
0
            code = pdfi_dict_put(ctx, font_info_dict, "ToUnicode", PDF_FALSE_OBJ);
968
0
        pdfi_countdown(array_obj);
969
0
        array_obj = NULL;
970
0
        if (code < 0)
971
0
            return code;
972
973
0
        code = pdfi_dict_get(ctx, font, "FontDescriptor", &array_obj);
974
0
        if (code >= 0) {
975
0
            bool known = false;
976
977
0
            (void)pdfi_dict_known(ctx, (pdf_dict *)array_obj, "FontFile", &known);
978
0
            if (!known) {
979
0
                (void)pdfi_dict_known(ctx, (pdf_dict *)array_obj, "FontFile2", &known);
980
0
                if (!known) {
981
0
                    (void)pdfi_dict_known(ctx, (pdf_dict *)array_obj, "FontFile3", &known);
982
0
                }
983
0
            }
984
985
0
            if (known > 0)
986
0
                code = pdfi_dict_put(ctx, font_info_dict, "Embedded", PDF_TRUE_OBJ);
987
0
            else
988
0
                code = pdfi_dict_put(ctx, font_info_dict, "Embedded", PDF_FALSE_OBJ);
989
0
        } else
990
0
            code = pdfi_dict_put(ctx, font_info_dict, "Embedded", PDF_FALSE_OBJ);
991
992
0
        pdfi_countdown(array_obj);
993
0
        array_obj = NULL;
994
995
0
        if (code < 0)
996
0
            return code;
997
998
999
0
        code = pdfi_dict_knownget_type(ctx, font, "Subtype", PDF_NAME, &array_obj);
1000
0
        if (code > 0) {
1001
0
            code = pdfi_dict_put(ctx, font_info_dict, "Subtype", array_obj);
1002
0
            if (code < 0) {
1003
0
                pdfi_countdown(array_obj);
1004
0
                pdfi_countdown(font_info_dict);
1005
0
                return code;
1006
0
            }
1007
1008
0
            if (pdfi_name_is((pdf_name *)array_obj, "Type3")) {
1009
0
                pdfi_countdown(o);
1010
0
                o = NULL;
1011
1012
0
                code = pdfi_dict_knownget_type(ctx, font, "Resources", PDF_DICT, &o);
1013
0
                if (code > 0)
1014
0
                    (void)pdfi_check_Resources(ctx, (pdf_dict *)o, page_dict, tracker);
1015
0
            }
1016
1017
0
            if (pdfi_name_is((const pdf_name *)array_obj, "Type0")){
1018
0
                pdf_array *descendants = NULL;
1019
0
                pdf_dict *desc_font = NULL;
1020
1021
0
                code = pdfi_dict_get(ctx, font, "DescendantFonts", (pdf_obj **)&descendants);
1022
0
                if (code >= 0) {
1023
0
                    code = pdfi_array_get(ctx, descendants, 0, (pdf_obj **)&desc_font);
1024
0
                    if (code >= 0){
1025
0
                        pdf_array *desc_array = NULL;
1026
1027
0
                        code = pdfi_array_alloc(ctx, 0, &desc_array);
1028
0
                        if (code >= 0) {
1029
0
                            pdf_array *saved = tracker->font_array;
1030
1031
0
                            pdfi_countup(desc_array);
1032
0
                            tracker->font_array = desc_array;
1033
0
                            (void)pdfi_check_Font(ctx, desc_font, page_dict, tracker);
1034
0
                            (void)pdfi_dict_put(ctx, font_info_dict, "Descendants", (pdf_obj *)tracker->font_array);
1035
0
                            pdfi_countdown((pdf_obj *)tracker->font_array);
1036
0
                            tracker->font_array = saved;
1037
0
                        }
1038
0
                        pdfi_countdown(descendants);
1039
0
                        pdfi_countdown(desc_font);
1040
0
                    }
1041
0
                }
1042
0
            }
1043
0
        }
1044
0
        pdfi_countdown(array_obj);
1045
0
        array_obj = NULL;
1046
1047
0
        code = pdfi_array_alloc(ctx, pdfi_array_size(tracker->font_array) + 1, &new_fonts);
1048
0
        if (code < 0) {
1049
0
            pdfi_countdown(font_info_dict);
1050
0
            return code;
1051
0
        }
1052
0
        pdfi_countup(new_fonts);
1053
1054
0
        for (index = 0; index < pdfi_array_size(tracker->font_array); index++) {
1055
0
            code = pdfi_array_get(ctx, tracker->font_array, index, &array_obj);
1056
0
            if (code < 0) {
1057
0
                pdfi_countdown(font_info_dict);
1058
0
                pdfi_countdown(new_fonts);
1059
0
                return code;
1060
0
            }
1061
0
            code = pdfi_array_put(ctx, new_fonts, index, array_obj);
1062
0
            pdfi_countdown(array_obj);
1063
0
            if (code < 0) {
1064
0
                pdfi_countdown(font_info_dict);
1065
0
                pdfi_countdown(new_fonts);
1066
0
                return code;
1067
0
            }
1068
0
        }
1069
0
        code = pdfi_array_put(ctx, new_fonts, index, (pdf_obj *)font_info_dict);
1070
0
        if (code < 0) {
1071
0
            pdfi_countdown(font_info_dict);
1072
0
            pdfi_countdown(new_fonts);
1073
0
            return code;
1074
0
        }
1075
0
        pdfi_countdown(font_info_dict);
1076
0
        pdfi_countdown(tracker->font_array);
1077
0
        tracker->font_array = new_fonts;
1078
307k
    } else {
1079
307k
        code = pdfi_dict_knownget_type(ctx, font, "Subtype", PDF_NAME, &o);
1080
307k
        if (code > 0) {
1081
304k
            if (pdfi_name_is((pdf_name *)o, "Type3")) {
1082
9.26k
                pdfi_countdown(o);
1083
9.26k
                o = NULL;
1084
1085
9.26k
                code = pdfi_dict_knownget_type(ctx, font, "Resources", PDF_DICT, &o);
1086
9.26k
                if (code > 0)
1087
2.86k
                    (void)pdfi_check_Resources(ctx, (pdf_dict *)o, page_dict, tracker);
1088
9.26k
            }
1089
304k
        }
1090
1091
307k
        pdfi_countdown(o);
1092
307k
        o = NULL;
1093
307k
    }
1094
1095
307k
    return 0;
1096
307k
}
1097
1098
/*
1099
 * Check the Resources dictionary Font entry.
1100
 */
1101
static int pdfi_check_Font_dict(pdf_context *ctx, pdf_dict *font_dict, pdf_dict *page_dict,
1102
                                pdfi_check_tracker_t *tracker)
1103
593k
{
1104
593k
    int code = 0;
1105
593k
    uint64_t i, index;
1106
593k
    pdf_obj *Key = NULL, *Value = NULL;
1107
1108
593k
    if (resource_is_checked(tracker, (pdf_obj *)font_dict))
1109
166
        return 0;
1110
1111
593k
    if (pdfi_type_of(font_dict) != PDF_DICT)
1112
388k
        return_error(gs_error_typecheck);
1113
1114
205k
    if (pdfi_dict_entries(font_dict) > 0) {
1115
204k
        code = pdfi_loop_detector_mark(ctx); /* Mark the start of the Font dictionary loop */
1116
204k
        if (code < 0)
1117
0
            return code;
1118
1119
        /* We don't want to store the dereferenced Font objects in the Resources dictionary
1120
         * Because we later create a substitute object, if we use the object here we will not
1121
         * find the correct font if we reference it through the Resources dictionary.
1122
         */
1123
204k
        code = pdfi_dict_first_no_store_R(ctx, font_dict, &Key, &Value, &index);
1124
204k
        if (code < 0) {
1125
45.2k
            (void)pdfi_loop_detector_cleartomark(ctx); /* Clear to the mark for the current resource loop */
1126
45.2k
            goto error1;
1127
45.2k
        }
1128
1129
159k
        i = 1;
1130
367k
        do {
1131
367k
            if (pdfi_type_of(Value) == PDF_DICT)
1132
286k
                code = pdfi_check_Font(ctx, (pdf_dict *)Value, page_dict, tracker);
1133
80.4k
            else if (pdfi_type_of(Value) == PDF_FONT) {
1134
72.2k
                pdf_dict *d = ((pdf_font *)Value)->PDF_font;
1135
1136
72.2k
                code = pdfi_check_Font(ctx, d, page_dict, tracker);
1137
72.2k
            } else {
1138
8.19k
                pdfi_set_warning(ctx, 0, NULL, W_PDF_FONTRESOURCE_TYPE, "pdfi_check_Font_dict", "");
1139
8.19k
            }
1140
367k
            if (code < 0)
1141
0
                break;
1142
1143
367k
            pdfi_countdown(Key);
1144
367k
            Key = NULL;
1145
367k
            pdfi_countdown(Value);
1146
367k
            Value = NULL;
1147
1148
367k
            (void)pdfi_loop_detector_cleartomark(ctx); /* Clear to the mark for the Font dictionary loop */
1149
1150
367k
            code = pdfi_loop_detector_mark(ctx); /* Mark the new start of the Font dictionary loop */
1151
367k
            if (code < 0)
1152
0
                goto error1;
1153
1154
367k
            if (i++ >= pdfi_dict_entries(font_dict)) {
1155
141k
                code = 0;
1156
141k
                break;
1157
141k
            }
1158
1159
225k
            code = pdfi_dict_next_no_store_R(ctx, font_dict, &Key, &Value, &index);
1160
225k
            if (code < 0)
1161
17.6k
                break;
1162
225k
        }while (1);
1163
1164
159k
        (void)pdfi_loop_detector_cleartomark(ctx); /* Clear to the mark for the current resource loop */
1165
159k
    }
1166
1167
160k
    pdfi_countdown(Key);
1168
160k
    pdfi_countdown(Value);
1169
1170
205k
error1:
1171
205k
    return code;
1172
160k
}
1173
1174
static int pdfi_check_Resources(pdf_context *ctx, pdf_dict *Resources_dict,
1175
                                pdf_dict *page_dict, pdfi_check_tracker_t *tracker)
1176
595k
{
1177
595k
    int code;
1178
595k
    pdf_obj *d = NULL;
1179
1180
595k
    if (resource_is_checked(tracker, (pdf_obj *)Resources_dict))
1181
1.32k
        return 0;
1182
1183
593k
    if (pdfi_type_of(Resources_dict) != PDF_DICT)
1184
0
        return_error(gs_error_typecheck);
1185
1186
    /* First up, check any colour spaces, for new spot colours.
1187
     * We only do this if asked because its expensive. spot_dict being NULL
1188
     * means we aren't interested in spot colours (not a DeviceN or Separation device)
1189
     */
1190
593k
    if (tracker->spot_dict != NULL) {
1191
57.2k
        code = pdfi_dict_knownget_type(ctx, Resources_dict, "ColorSpace", PDF_DICT, &d);
1192
57.2k
        if (code < 0 && code != gs_error_undefined) {
1193
18
            if ((code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_BAD_RESOURCE, "pdfi_check_Resources", "ColorSpace")) < 0)
1194
0
                return code;
1195
18
        }
1196
57.2k
        (void)pdfi_check_ColorSpace_dict(ctx, (pdf_dict *)d, page_dict, tracker);
1197
1198
57.2k
        pdfi_countdown(d);
1199
57.2k
        d = NULL;
1200
1201
        /* Put the Resources dictionary back in cache (or promote it) in case checking ColorSpace dict
1202
         * created so many cache entires it flushed Resources from cache
1203
         */
1204
57.2k
        code = pdfi_cache_object(ctx, (pdf_obj *)Resources_dict);
1205
57.2k
        if (code < 0) {
1206
0
            if ((code = pdfi_set_warning_stop(ctx, code, NULL, W_PDF_CACHE_FAIL, "pdfi_check_Resources", "")) < 0)
1207
0
                return code;
1208
0
        }
1209
1210
57.2k
        code = pdfi_dict_knownget_type(ctx, Resources_dict, "Shading", PDF_DICT, &d);
1211
57.2k
        if (code < 0 && code != gs_error_undefined) {
1212
11
            if ((code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_BAD_RESOURCE, "pdfi_check_Resources", "Shading")) < 0)
1213
0
                return code;
1214
11
        }
1215
57.2k
        (void)pdfi_check_Shading_dict(ctx, (pdf_dict *)d, page_dict, tracker);
1216
57.2k
        pdfi_countdown(d);
1217
57.2k
        d = NULL;
1218
1219
        /* Put the Resources dictionary back in cache (or promote it) in case checking Shading dict
1220
         * created so many cache entires it flushed Resources from cache
1221
         */
1222
57.2k
        code = pdfi_cache_object(ctx, (pdf_obj *)Resources_dict);
1223
57.2k
        if (code < 0) {
1224
0
            if ((code = pdfi_set_warning_stop(ctx, code, NULL, W_PDF_CACHE_FAIL, "pdfi_check_Resources", "")) < 0)
1225
0
                return code;
1226
0
        }
1227
57.2k
    }
1228
1229
593k
    code = pdfi_dict_knownget_type(ctx, Resources_dict, "XObject", PDF_DICT, &d);
1230
593k
    if (code < 0 && code != gs_error_undefined) {
1231
196
        if ((code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_BAD_RESOURCE, "pdfi_check_Resources", "XObject")) < 0)
1232
0
            return code;
1233
196
    }
1234
593k
    code = pdfi_check_XObject_dict(ctx, (pdf_dict *)d, page_dict, tracker);
1235
593k
    if (code == gs_error_Fatal) {
1236
0
        if ((code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_BAD_RESOURCE, "pdfi_check_Resources", "XObject")) < 0)
1237
0
            return code;
1238
0
    }
1239
593k
    pdfi_countdown(d);
1240
593k
    d = NULL;
1241
1242
    /* Put the Resources dictionary back in cache (or promote it) in case checking XObject dict
1243
     * created so many cache entires it flushed Resources from cache
1244
     */
1245
593k
    code = pdfi_cache_object(ctx, (pdf_obj *)Resources_dict);
1246
593k
    if (code < 0) {
1247
0
        if ((code = pdfi_set_warning_stop(ctx, code, NULL, W_PDF_CACHE_FAIL, "pdfi_check_Resources", "")) < 0)
1248
0
            return code;
1249
0
    }
1250
1251
593k
    code = pdfi_dict_knownget_type(ctx, Resources_dict, "Pattern", PDF_DICT, &d);
1252
593k
    if (code < 0 && code != gs_error_undefined) {
1253
187
        if ((code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_BAD_RESOURCE, "pdfi_check_Resources", "Pattern")) < 0) {
1254
0
            return code;
1255
0
        }
1256
187
    }
1257
593k
    (void)pdfi_check_Pattern_dict(ctx, (pdf_dict *)d, page_dict, tracker);
1258
593k
    pdfi_countdown(d);
1259
593k
    d = NULL;
1260
1261
    /* Put the Resources dictionary back in cache (or promote it) in case checking Pattern dict
1262
     * created so many cache entires it flushed Resources from cache
1263
     */
1264
593k
    code = pdfi_cache_object(ctx, (pdf_obj *)Resources_dict);
1265
593k
    if (code < 0) {
1266
0
        if ((code = pdfi_set_warning_stop(ctx, code, NULL, W_PDF_CACHE_FAIL, "pdfi_check_Resources", "")) < 0)
1267
0
            return code;
1268
0
    }
1269
1270
593k
    code = pdfi_dict_knownget_type(ctx, Resources_dict, "Font", PDF_DICT, &d);
1271
593k
    if (code < 0) {
1272
7.17k
        if ((code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_BAD_RESOURCE, "pdfi_check_Resources", "Font")) < 0)
1273
0
            return code;
1274
7.17k
    }
1275
593k
    (void)pdfi_check_Font_dict(ctx, (pdf_dict *)d, page_dict, tracker);
1276
    /* From this point onwards, if we detect transparency (or have already detected it) we
1277
     * can exit, we have already counted up any spot colours.
1278
     */
1279
593k
    pdfi_countdown(d);
1280
593k
    d = NULL;
1281
1282
    /* Put the Resources dictionary back in cache (or promote it) in case checking Font dict
1283
     * created so many cache entires it flushed Resources from cache
1284
     */
1285
593k
    code = pdfi_cache_object(ctx, (pdf_obj *)Resources_dict);
1286
593k
    if (code < 0) {
1287
0
        if ((code = pdfi_set_warning_stop(ctx, code, NULL, W_PDF_CACHE_FAIL, "pdfi_check_Resources", "")) < 0)
1288
0
            return code;
1289
0
    }
1290
1291
593k
    code = pdfi_dict_knownget_type(ctx, Resources_dict, "ExtGState", PDF_DICT, &d);
1292
593k
    if (code < 0 && code != gs_error_undefined) {
1293
116
        if ((code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_BAD_RESOURCE, "pdfi_check_Resources", "ExtGState")) < 0)
1294
0
            return code;
1295
116
    }
1296
593k
    (void)pdfi_check_ExtGState_dict(ctx, (pdf_dict *)d, page_dict, tracker);
1297
593k
    pdfi_countdown(d);
1298
593k
    d = NULL;
1299
1300
    /* Put the Resources dictionary back in cache (or promote it) in case checking ExtGState dict
1301
     * created so many cache entires it flushed Resources from cache
1302
     */
1303
593k
    code = pdfi_cache_object(ctx, (pdf_obj *)Resources_dict);
1304
593k
    if (code < 0) {
1305
0
        if ((code = pdfi_set_warning_stop(ctx, code, NULL, W_PDF_CACHE_FAIL, "pdfi_check_Resources", "")) < 0)
1306
0
            return code;
1307
0
    }
1308
1309
593k
    return 0;
1310
593k
}
1311
1312
static int pdfi_check_annot_for_transparency(pdf_context *ctx, pdf_dict *annot, pdf_dict *page_dict,
1313
                                             pdfi_check_tracker_t *tracker)
1314
854k
{
1315
854k
    int code;
1316
854k
    pdf_name *n;
1317
854k
    pdf_obj *N = NULL;
1318
854k
    pdf_dict *ap = NULL;
1319
854k
    pdf_dict *Resources = NULL;
1320
854k
    double f;
1321
1322
854k
    if (resource_is_checked(tracker, (pdf_obj *)annot))
1323
108
        return 0;
1324
1325
854k
    if (pdfi_type_of(annot) != PDF_DICT)
1326
0
        return_error(gs_error_typecheck);
1327
1328
    /* Check #1 Does the (Normal) Appearnce stream use any Resources which include transparency.
1329
     * We check this first, because this also checks for spot colour spaces. Once we've done that we
1330
     * can exit the checks as soon as we detect transparency.
1331
     */
1332
854k
    code = pdfi_dict_knownget_type(ctx, annot, "AP", PDF_DICT, (pdf_obj **)&ap);
1333
854k
    if (code > 0)
1334
407k
    {
1335
        /* Fetch without resolving indirect ref because pdfmark wants it that way later */
1336
407k
        code = pdfi_dict_get_no_store_R(ctx, ap, "N", (pdf_obj **)&N);
1337
407k
        if (code >= 0) {
1338
306k
            pdf_dict *dict = NULL;
1339
1340
306k
            code = pdfi_dict_from_obj(ctx, N, &dict);
1341
306k
            if (code == 0)
1342
306k
                code = pdfi_dict_knownget_type(ctx, dict, "Resources", PDF_DICT, (pdf_obj **)&Resources);
1343
306k
            if (code > 0)
1344
213k
                code = pdfi_check_Resources(ctx, (pdf_dict *)Resources, page_dict, tracker);
1345
306k
        }
1346
407k
        if (code == gs_error_undefined)
1347
94.8k
            code = 0;
1348
407k
    }
1349
854k
    pdfi_countdown(ap);
1350
854k
    pdfi_countdown(N);
1351
854k
    pdfi_countdown(Resources);
1352
1353
854k
    if (code < 0)
1354
22.7k
        return code;
1355
    /* We've checked the Resources, and nothing else in an annotation can define spot colours, so
1356
     * if we detected transparency in the Resources we need not check further.
1357
     */
1358
831k
    if (tracker->transparent == true)
1359
12.7k
        return 0;
1360
1361
818k
    code = pdfi_dict_get_type(ctx, annot, "Subtype", PDF_NAME, (pdf_obj **)&n);
1362
818k
    if (code < 0) {
1363
1.48k
        if ((code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_BAD_TYPE, "pdfi_check_annot_for_transparency", "")) < 0) {
1364
0
            return code;
1365
0
        }
1366
817k
    } else {
1367
        /* Check #2, Highlight annotations are always preformed with transparency */
1368
817k
        if (pdfi_name_is((const pdf_name *)n, "Highlight")) {
1369
2.89k
            pdfi_countdown(n);
1370
2.89k
            tracker->transparent = true;
1371
2.89k
            return 0;
1372
2.89k
        }
1373
814k
        pdfi_countdown(n);
1374
814k
        n = NULL;
1375
1376
        /* Check #3 Blend Mode (BM) not being 'Normal' or 'Compatible' */
1377
814k
        code = pdfi_dict_knownget_type(ctx, annot, "BM", PDF_NAME, (pdf_obj **)&n);
1378
814k
        if (code > 0) {
1379
0
            if (!pdfi_name_is((const pdf_name *)n, "Normal")) {
1380
0
                if (!pdfi_name_is((const pdf_name *)n, "Compatible")) {
1381
0
                    pdfi_countdown(n);
1382
0
                    tracker->transparent = true;
1383
0
                    return 0;
1384
0
                }
1385
0
            }
1386
0
            code = 0;
1387
0
        }
1388
814k
        pdfi_countdown(n);
1389
814k
        if (code < 0)
1390
0
            return code;
1391
1392
        /* Check #4 stroke constant alpha (CA) is not 1 (100% opaque) */
1393
814k
        code = pdfi_dict_knownget_number(ctx, annot, "CA", &f);
1394
814k
        if (code > 0) {
1395
2.47k
            if (f != 1.0) {
1396
262
                tracker->transparent = true;
1397
262
                return 0;
1398
262
            }
1399
2.47k
        }
1400
813k
        if (code < 0)
1401
0
            return code;
1402
1403
        /* Check #5 non-stroke constant alpha (ca) is not 1 (100% opaque) */
1404
813k
        code = pdfi_dict_knownget_number(ctx, annot, "ca", &f);
1405
813k
        if (code > 0) {
1406
0
            if (f != 1.0) {
1407
0
                tracker->transparent = true;
1408
0
                return 0;
1409
0
            }
1410
0
        }
1411
813k
        if (code < 0)
1412
0
            return code;
1413
813k
    }
1414
1415
815k
    return 0;
1416
818k
}
1417
1418
static int pdfi_check_Annots_for_transparency(pdf_context *ctx, pdf_array *annots_array,
1419
                                              pdf_dict *page_dict, pdfi_check_tracker_t *tracker)
1420
52.3k
{
1421
52.3k
    int i, code = 0;
1422
52.3k
    pdf_dict *annot = NULL;
1423
1424
52.3k
    if (resource_is_checked(tracker, (pdf_obj *)annots_array))
1425
0
        return 0;
1426
1427
52.3k
    if (pdfi_type_of(annots_array) != PDF_ARRAY)
1428
0
        return_error(gs_error_typecheck);
1429
1430
1.03M
    for (i=0; i < pdfi_array_size(annots_array); i++) {
1431
998k
        code = pdfi_array_get_type(ctx, annots_array, (uint64_t)i, PDF_DICT, (pdf_obj **)&annot);
1432
998k
        if (code >= 0) {
1433
854k
            code = pdfi_check_annot_for_transparency(ctx, annot, page_dict, tracker);
1434
854k
            if (code < 0 && (code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_INVALID_TRANS_XOBJECT, "pdfi_check_Annots_for_transparency", "")) < 0) {
1435
0
                goto exit;
1436
0
            }
1437
1438
            /* If we've found transparency, and don't need to continue checkign for spot colours
1439
             * just exit as fast as possible.
1440
             */
1441
854k
            if (tracker->transparent == true && tracker->spot_dict == NULL)
1442
10.7k
                goto exit;
1443
1444
843k
            pdfi_countdown(annot);
1445
843k
            annot = NULL;
1446
843k
        }
1447
144k
        else if ((code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_INVALID_TRANS_XOBJECT, "pdfi_check_Annots_for_transparency", "")) < 0) {
1448
0
            goto exit;
1449
0
        }
1450
998k
    }
1451
52.3k
exit:
1452
52.3k
    pdfi_countdown(annot);
1453
52.3k
    return code;
1454
52.3k
}
1455
1456
/* Check for transparency and spots on page.
1457
 *
1458
 * Sets ctx->spot_capable_device
1459
 * Builds a dictionary of the unique spot names in spot_dict
1460
 * Set 'transparent' to true if there is transparency on the page
1461
 *
1462
 * From the original PDF interpreter written in PostScript:
1463
 * Note: we deliberately don't check to see whether a Group is defined,
1464
 * because Adobe Illustrator 10 (and possibly other applications) define
1465
 * a page-level group whether transparency is actually used or not.
1466
 * Ignoring the presence of Group is justified because, in the absence
1467
 * of any other transparency features, they have no effect.
1468
 */
1469
static int pdfi_check_page_inner(pdf_context *ctx, pdf_dict *page_dict,
1470
                                 pdfi_check_tracker_t *tracker)
1471
219k
{
1472
219k
    int code;
1473
219k
    pdf_dict *Resources = NULL;
1474
219k
    pdf_array *Annots = NULL;
1475
219k
    pdf_dict *Group = NULL;
1476
219k
    pdf_obj *CS = NULL;
1477
1478
219k
    tracker->transparent = false;
1479
1480
219k
    if (pdfi_type_of(page_dict) != PDF_DICT)
1481
5
        return_error(gs_error_typecheck);
1482
1483
1484
    /* Check if the page dictionary has a page Group entry (for spots).
1485
     * Page group should mean the page has transparency but we ignore it for the purposes
1486
     * of transparency detection. See above.
1487
     */
1488
219k
    if (tracker->spot_dict) {
1489
24.9k
        code = pdfi_dict_knownget_type(ctx, page_dict, "Group", PDF_DICT, (pdf_obj **)&Group);
1490
24.9k
        if (code > 0) {
1491
            /* If Group has a ColorSpace (CS), then check it for spot colours */
1492
3.28k
            code = pdfi_dict_knownget(ctx, Group, "CS", &CS);
1493
3.28k
            if (code > 0)
1494
3.20k
                code = pdfi_check_ColorSpace_for_spots(ctx, CS, Group, page_dict, tracker->spot_dict);
1495
1496
3.28k
            if (code < 0) {
1497
58
                if ((code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_GS_LIB_ERROR, "pdfi_check_page_inner", "")) < 0) {
1498
0
                    goto exit;
1499
0
                }
1500
58
            }
1501
3.28k
        }
1502
24.9k
    }
1503
1504
    /* Now check any Resources dictionary in the Page dictionary */
1505
219k
    code = pdfi_dict_knownget_type(ctx, page_dict, "Resources", PDF_DICT, (pdf_obj **)&Resources);
1506
219k
    if (code > 0)
1507
207k
        code = pdfi_check_Resources(ctx, Resources, page_dict, tracker);
1508
1509
219k
    if (code == gs_error_pdf_stackoverflow || (code < 0 &&
1510
8.27k
       (code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_GS_LIB_ERROR, "pdfi_check_page_inner", "")) < 0)) {
1511
0
        goto exit;
1512
0
    }
1513
1514
    /* If we are drawing Annotations, check to see if the page uses any Annots */
1515
219k
    if (ctx->args.showannots) {
1516
219k
        code = pdfi_dict_knownget_type(ctx, page_dict, "Annots", PDF_ARRAY, (pdf_obj **)&Annots);
1517
219k
        if (code > 0)
1518
52.3k
            code = pdfi_check_Annots_for_transparency(ctx, Annots, page_dict,
1519
52.3k
                                                      tracker);
1520
219k
        if (code < 0) {
1521
2.73k
            if ((code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_GS_LIB_ERROR, "pdfi_check_page_inner", "")) < 0) {
1522
0
               goto exit;
1523
0
            }
1524
2.73k
        }
1525
219k
    }
1526
1527
219k
    code = 0;
1528
219k
 exit:
1529
219k
    pdfi_countdown(Resources);
1530
219k
    pdfi_countdown(Annots);
1531
219k
    pdfi_countdown(CS);
1532
219k
    pdfi_countdown(Group);
1533
219k
    return code;
1534
219k
}
1535
1536
/* Checks page for transparency, and sets up device for spots, if applicable
1537
 * Sets ctx->page.has_transparency and ctx->page.num_spots
1538
 * do_setup -- indicates whether to actually set up the device with the spot count.
1539
 */
1540
int pdfi_check_page(pdf_context *ctx, pdf_dict *page_dict, pdf_array **fonts_array, pdf_array **spots_array, bool do_setup)
1541
219k
{
1542
219k
    int code, code1;
1543
219k
    int spots = 0;
1544
219k
    pdfi_check_tracker_t tracker;
1545
219k
    pdf_dict *Resources = NULL;
1546
1547
219k
    ctx->page.num_spots = 0;
1548
219k
    ctx->page.has_transparency = false;
1549
1550
219k
    code = pdfi_check_init_tracker(ctx, &tracker, fonts_array, spots_array);
1551
219k
    if (code < 0)
1552
0
        goto exit;
1553
1554
    /* Check for spots and transparency in this page */
1555
219k
    code = pdfi_check_page_inner(ctx, page_dict, &tracker);
1556
219k
    if (code < 0)
1557
5
        goto exit;
1558
1559
    /* Count the spots */
1560
219k
    if (tracker.spot_dict)
1561
24.9k
        spots = pdfi_dict_entries(tracker.spot_dict);
1562
1563
    /* If setup requested, tell the device about spots and transparency */
1564
219k
    if (do_setup) {
1565
0
        gs_c_param_list list;
1566
0
        int a = 0;
1567
0
        pdf_name *Key = NULL;
1568
0
        pdf_obj *Value = NULL;
1569
0
        uint64_t index = 0;
1570
1571
0
        gs_c_param_list_write(&list, ctx->memory);
1572
1573
        /* If there are spot colours (and by inference, the device renders spot plates) then
1574
         * send the number of Spots to the device, so it can setup correctly.
1575
         */
1576
0
        if (tracker.spot_dict) {
1577
            /* There is some awkwardness here. If the SeparationColorNames setting
1578
             * fails, we want to ignore it (this can mean that we exceeded the maximum
1579
             * number of colourants and some will be converted to CMYK). But if that happens,
1580
             * any other parameters in the same list which haven't already been prcoessed
1581
             * will be lost. So we need to send two lists, the SeparationColorNames and
1582
             * 'everything else'.
1583
             */
1584
0
            if (spots > 0) {
1585
0
                gs_param_string_array sa;
1586
0
                gs_param_string *table = NULL;
1587
1588
0
                table = (gs_param_string *)gs_alloc_byte_array(ctx->memory, spots, sizeof(gs_param_string), "SeparationNames");
1589
0
                if (table != NULL)
1590
0
                {
1591
0
                    memset(table, 0x00, spots * sizeof(gs_param_string));
1592
1593
0
                    code = pdfi_dict_first(ctx, tracker.spot_dict, (pdf_obj **)&Key, &Value, &index);
1594
0
                    while (code >= 0)
1595
0
                    {
1596
0
                        if (pdfi_type_of(Key) == PDF_NAME) {
1597
0
                            table[a].data = ((pdf_string *)Key)->data;
1598
0
                            table[a].size = ((pdf_string *)Key)->length;
1599
0
                            table[a++].persistent = false;
1600
0
                        }
1601
                        /* Although we count down the returned PDF objects here, the pointers
1602
                         * to the name data remain valid and won't move. Provided we don't
1603
                         * retain the pointers after we free the tracker dictionary this is
1604
                         * safe to do.
1605
                         */
1606
0
                        pdfi_countdown(Key);
1607
0
                        Key = NULL;
1608
0
                        pdfi_countdown(Value);
1609
0
                        Value = NULL;
1610
0
                        code = pdfi_dict_next(ctx, tracker.spot_dict, (pdf_obj **)&Key, &Value, &index);
1611
0
                    }
1612
0
                    sa.data = table;
1613
0
                    sa.size = spots;
1614
0
                    sa.persistent = false;
1615
1616
0
                    (void)param_write_string_array((gs_param_list *)&list, "SeparationColorNames", &sa);
1617
0
                    gs_c_param_list_read(&list);
1618
0
                    code = gs_putdeviceparams(ctx->pgs->device, (gs_param_list *)&list);
1619
0
                    gs_c_param_list_release(&list);
1620
1621
0
                    gs_free_object(ctx->memory, table, "SeparationNames");
1622
0
                    if (code > 0) {
1623
                        /* The device was closed, we need to reopen it */
1624
0
                        code = gs_setdevice_no_erase(ctx->pgs, ctx->pgs->device);
1625
0
                        if (code < 0)
1626
0
                            goto exit;
1627
0
                        gs_erasepage(ctx->pgs);
1628
0
                    }
1629
1630
                    /* Reset the list back to being writeable */
1631
0
                    gs_c_param_list_write(&list, ctx->memory);
1632
0
                }
1633
0
                else {
1634
0
                    code = gs_note_error(gs_error_VMerror);
1635
0
                    goto exit;
1636
0
                }
1637
0
            }
1638
            /* Update the number of spots */
1639
0
            param_write_int((gs_param_list *)&list, "PageSpotColors", &spots);
1640
0
        }
1641
        /* Update the page transparency */
1642
0
        (void)param_write_bool((gs_param_list *)&list, "PageUsesTransparency",
1643
0
                                    &tracker.transparent);
1644
0
        gs_c_param_list_read(&list);
1645
0
        code = gs_putdeviceparams(ctx->pgs->device, (gs_param_list *)&list);
1646
0
        gs_c_param_list_release(&list);
1647
1648
0
        if (code > 0) {
1649
            /* The device was closed, we need to reopen it */
1650
0
            code = gs_setdevice_no_erase(ctx->pgs, ctx->pgs->device);
1651
0
            if (code < 0)
1652
0
                goto exit;
1653
0
            gs_erasepage(ctx->pgs);
1654
0
        }
1655
0
    }
1656
1657
    /* Set our values in the context, for caller */
1658
219k
    if (!ctx->args.notransparency)
1659
219k
        ctx->page.has_transparency = tracker.transparent;
1660
219k
    ctx->page.num_spots = spots;
1661
219k
    ctx->page.has_OP = tracker.has_overprint;
1662
1663
    /* High level devices do not render overprint */
1664
219k
    if (ctx->device_state.HighLevelDevice)
1665
100k
        ctx->page.has_OP = false;
1666
1667
219k
 exit:
1668
219k
    code1 = code;
1669
219k
    if (fonts_array != NULL) {
1670
0
        *fonts_array = tracker.font_array;
1671
0
        pdfi_countup(*fonts_array);
1672
0
    }
1673
1674
219k
    if (spots_array != NULL && tracker.spot_dict != NULL && pdfi_dict_entries(tracker.spot_dict) != 0) {
1675
0
        pdf_array *new_array = NULL;
1676
0
        pdf_name *Key = NULL;
1677
0
        pdf_obj *Value = NULL;
1678
0
        uint64_t index = 0, a_index = 0;
1679
1680
0
        index = pdfi_dict_entries(tracker.spot_dict);
1681
1682
0
        code = pdfi_array_alloc(ctx, index, &new_array);
1683
0
        if (code < 0)
1684
0
            goto error;
1685
0
        pdfi_countup(new_array);
1686
1687
0
        code = pdfi_dict_first(ctx, tracker.spot_dict, (pdf_obj **)&Key, &Value, &index);
1688
0
        while (code >= 0)
1689
0
        {
1690
0
            if (pdfi_type_of(Key) == PDF_NAME) {
1691
0
                code = pdfi_array_put(ctx, new_array, a_index++, (pdf_obj *)Key);
1692
0
                if (code < 0) {
1693
0
                    pdfi_countdown(new_array);
1694
0
                    pdfi_countdown(Key);
1695
0
                    pdfi_countdown(Value);
1696
0
                    goto error;
1697
0
                }
1698
0
            }
1699
1700
0
            pdfi_countdown(Key);
1701
0
            Key = NULL;
1702
0
            pdfi_countdown(Value);
1703
0
            Value = NULL;
1704
0
            code = pdfi_dict_next(ctx, tracker.spot_dict, (pdf_obj **)&Key, &Value, &index);
1705
0
        }
1706
0
        *spots_array = new_array;
1707
0
    }
1708
1709
    /* Put the Resources dictionary back in cache (or promote it) in case checking the page
1710
     * created so many cache entires it flushed Resources from cache
1711
     */
1712
219k
    code = pdfi_dict_knownget_type(ctx, page_dict, "Resources", PDF_DICT, (pdf_obj **)&Resources);
1713
219k
    if (code > 0) {
1714
207k
        code = pdfi_cache_object(ctx, (pdf_obj *)Resources);
1715
207k
        if (code < 0)
1716
0
            code = pdfi_set_warning_stop(ctx, code, NULL, W_PDF_CACHE_FAIL, "pdfi_check_Resources", "");
1717
207k
        pdfi_countdown(Resources);
1718
207k
    }
1719
12.0k
    else {
1720
12.0k
        if (code < 0) {
1721
8.28k
            if (code == gs_error_undefined)
1722
2.78k
                code = 0;
1723
5.49k
            else
1724
5.49k
                code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_BAD_PAGE_RESOURCES, "pdfi_check_page", NULL);
1725
8.28k
        }
1726
12.0k
    }
1727
1728
219k
error:
1729
219k
    (void)pdfi_check_free_tracker(ctx, &tracker);
1730
219k
    if (code1 < 0)
1731
5
        return code1;
1732
219k
    return code;
1733
219k
}