Coverage Report

Created: 2026-04-01 07:17

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ghostpdl/pdf/pdf_page.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
/* Page-level operations for the PDF interpreter */
17
18
#include "pdf_int.h"
19
#include "pdf_stack.h"
20
#include "pdf_doc.h"
21
#include "pdf_deref.h"
22
#include "pdf_page.h"
23
#include "pdf_file.h"
24
#include "pdf_dict.h"
25
#include "pdf_array.h"
26
#include "pdf_loop_detect.h"
27
#include "pdf_colour.h"
28
#include "pdf_trans.h"
29
#include "pdf_font_types.h"
30
#include "pdf_gstate.h"
31
#include "pdf_misc.h"
32
#include "pdf_optcontent.h"
33
#include "pdf_device.h"
34
#include "pdf_annot.h"
35
#include "pdf_check.h"
36
#include "pdf_mark.h"
37
#include "pdf_font.h"
38
39
#include "gscoord.h"        /* for gs_concat() and others */
40
#include "gspaint.h"        /* For gs_erasepage() */
41
#include "gsstate.h"        /* For gs_initgraphics() */
42
#include "gspath2.h"        /* For gs_rectclip() */
43
#include "gxdevsop.h"               /* For special ops */
44
45
static int pdfi_process_page_contents(pdf_context *ctx, pdf_dict *page_dict)
46
104k
{
47
104k
    int i, code = 0;
48
104k
    pdf_obj *o, *o1;
49
50
#if PURGE_CACHE_PER_PAGE
51
    pdfi_purge_obj_cache(ctx);
52
#endif
53
54
104k
    code = pdfi_dict_get(ctx, page_dict, "Contents", &o);
55
104k
    if (code == gs_error_undefined)
56
        /* Don't throw an error if there are no contents, just render nothing.... */
57
8.62k
        return 0;
58
95.9k
    if (code < 0)
59
1.56k
        return code;
60
61
94.3k
    if (pdfi_type_of(o) == PDF_INDIRECT) {
62
0
        if (((pdf_indirect_ref *)o)->ref_object_num == page_dict->object_num)
63
0
            return_error(gs_error_circular_reference);
64
65
0
        code = pdfi_dereference(ctx, ((pdf_indirect_ref *)o)->ref_object_num, ((pdf_indirect_ref *)o)->ref_generation_num, &o1);
66
0
        pdfi_countdown(o);
67
0
        if (code < 0) {
68
0
            if (code == gs_error_VMerror)
69
0
                return code;
70
0
            return 0;
71
0
        }
72
0
        o = o1;
73
0
    }
74
75
94.3k
    code = pdfi_gsave(ctx);
76
94.3k
    if (code < 0) {
77
0
        pdfi_countdown(o);
78
0
        return code;
79
0
    }
80
    /* Increment the saved gsave_level by 1 to allow for the gsave we've just
81
     * done. Otherwise excess 'Q' operators in the stream can cause us to pop
82
     * one higher than we should. Bug 707753. */
83
94.3k
    ctx->current_stream_save.gsave_level++;
84
85
94.3k
    ctx->encryption.decrypt_strings = false;
86
94.3k
    if (pdfi_type_of(o) == PDF_ARRAY) {
87
9.02k
        pdf_array *a = (pdf_array *)o;
88
89
35.9k
        for (i=0;i < pdfi_array_size(a); i++) {
90
28.6k
            pdf_indirect_ref *r;
91
28.6k
            code = pdfi_array_get_no_deref(ctx, a, i, (pdf_obj **)&r);
92
28.6k
            if (code < 0)
93
0
                goto page_error;
94
28.6k
            if (pdfi_type_of (r) == PDF_STREAM) {
95
61
                code = pdfi_interpret_content_stream(ctx, NULL, (pdf_stream *)r, page_dict);
96
61
                pdfi_countdown(r);
97
61
                if (code < 0)
98
1
                    goto page_error;
99
28.5k
            } else {
100
28.5k
                if (pdfi_type_of(r) != PDF_INDIRECT) {
101
24
                        pdfi_countdown(r);
102
24
                        code = gs_note_error(gs_error_typecheck);
103
24
                        goto page_error;
104
28.5k
                } else {
105
28.5k
                    if (r->ref_object_num == page_dict->object_num) {
106
3
                        pdfi_countdown(r);
107
3
                        code = gs_note_error(gs_error_circular_reference);
108
3
                        goto page_error;
109
3
                    }
110
28.5k
                    code = pdfi_dereference(ctx, r->ref_object_num, r->ref_generation_num, &o1);
111
28.5k
                    pdfi_countdown(r);
112
28.5k
                    if (code < 0) {
113
1.46k
                        if ((code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_NOERROR, "pdfi_process_page_contents", NULL)) < 0) {
114
0
                            goto page_error;
115
0
                        }
116
1.46k
                        code = 0;
117
1.46k
                        goto page_error;
118
1.46k
                    }
119
27.0k
                    if (pdfi_type_of(o1) != PDF_STREAM) {
120
202
                        pdfi_countdown(o1);
121
202
                        code = gs_note_error(gs_error_typecheck);
122
202
                        goto page_error;
123
202
                    }
124
26.8k
                    code = pdfi_interpret_content_stream(ctx, NULL, (pdf_stream *)o1, page_dict);
125
26.8k
                    pdfi_countdown(o1);
126
26.8k
                    if (code < 0) {
127
1.17k
                        if ((code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_NOERROR, "pdfi_process_page_contents", NULL)) < 0) {
128
8
                            goto page_error;
129
8
                        }
130
1.17k
                    }
131
26.8k
                }
132
28.5k
            }
133
28.6k
        }
134
85.3k
    } else {
135
85.3k
        if (pdfi_type_of(o) == PDF_STREAM) {
136
84.9k
            code = pdfi_interpret_content_stream(ctx, NULL, (pdf_stream *)o, page_dict);
137
84.9k
        } else
138
426
            code = gs_note_error(gs_error_typecheck);
139
85.3k
    }
140
94.3k
page_error:
141
    /* Decrement the stream level to counterbalance the increment above. */
142
94.3k
    ctx->current_stream_save.gsave_level--;
143
94.3k
    ctx->encryption.decrypt_strings = true;
144
94.3k
    pdfi_clearstack(ctx);
145
94.3k
    pdfi_grestore(ctx);
146
94.3k
    pdfi_countdown(o);
147
94.3k
    return code;
148
94.3k
}
149
150
/* Render one page (including annotations) (see pdf_main.ps/showpagecontents) */
151
static int pdfi_process_one_page(pdf_context *ctx, pdf_dict *page_dict)
152
104k
{
153
104k
    stream_save local_entry_save;
154
104k
    int code, code1;
155
156
    /* Save the current stream state, for later cleanup, in a local variable */
157
104k
    local_save_stream_state(ctx, &local_entry_save);
158
104k
    initialise_stream_save(ctx);
159
160
104k
    code = pdfi_process_page_contents(ctx, page_dict);
161
162
    /* Put our state back the way it was before we ran the contents
163
     * and check if the stream had problems
164
     */
165
#if PROBE_STREAMS
166
    if (ctx->pgs->level > ctx->current_stream_save.gsave_level ||
167
        pdfi_count_stack(ctx) > ctx->current_stream_save.stack_count)
168
        code = ((pdf_context *)0)->first_page;
169
#endif
170
171
104k
    cleanup_context_interpretation(ctx, &local_entry_save);
172
104k
    local_restore_stream_state(ctx, &local_entry_save);
173
174
104k
    local_save_stream_state(ctx, &local_entry_save);
175
104k
    initialise_stream_save(ctx);
176
177
104k
    code1 = pdfi_do_annotations(ctx, page_dict);
178
104k
    if (code >= 0) code = code1;
179
180
104k
    cleanup_context_interpretation(ctx, &local_entry_save);
181
104k
    local_restore_stream_state(ctx, &local_entry_save);
182
183
104k
    local_save_stream_state(ctx, &local_entry_save);
184
104k
    initialise_stream_save(ctx);
185
186
104k
    code1 = pdfi_do_acroform(ctx, page_dict);
187
104k
    if (code >= 0) code = code1;
188
189
104k
    cleanup_context_interpretation(ctx, &local_entry_save);
190
104k
    local_restore_stream_state(ctx, &local_entry_save);
191
192
104k
    if (ctx->text.BlockDepth != 0) {
193
8.01k
        ctx->text.BlockDepth = 0;
194
8.01k
        if (ctx->text.TextClip) {
195
47
            gx_device *dev = gs_currentdevice_inline(ctx->pgs);
196
197
47
            ctx->text.TextClip = false;
198
47
            (void)dev_proc(dev, dev_spec_op)(dev, gxdso_hilevel_text_clip, (void *)0, 1);
199
47
        }
200
8.01k
        code = pdfi_set_warning_stop(ctx, gs_note_error(gs_error_syntaxerror), NULL, W_PDF_UNBLANACED_BT, "pdfi_process_one_page", NULL);
201
8.01k
        if (code < 0)
202
0
            return code;
203
8.01k
    }
204
205
104k
    return code;
206
104k
}
207
208
/* See pdf_PDF2PS_matrix and .pdfshowpage_Install */
209
static void pdfi_set_ctm(pdf_context *ctx)
210
0
{
211
0
    gs_matrix mat;
212
213
    /* Adjust for page.UserUnits */
214
0
    mat.xx = ctx->page.UserUnit;
215
0
    mat.xy = 0;
216
0
    mat.yx = 0;
217
0
    mat.yy = ctx->page.UserUnit;
218
0
    mat.tx = 0;
219
0
    mat.ty = 0;
220
0
    gs_concat(ctx->pgs, &mat);
221
222
    /* We need to make sure the default matrix is properly set.
223
     * If we do gs_initgraphics() later (such as for annotations)
224
     * then it uses this default matrix if it is set.
225
     * Needed for page rotations to work correctly with Annotations.
226
     */
227
0
    gs_setdefaultmatrix(ctx->pgs, &ctm_only(ctx->pgs));
228
229
0
}
230
231
/* Get .MediaSize from the device to setup page.Size in context */
232
static int pdfi_get_media_size(pdf_context *ctx, pdf_dict *page_dict)
233
104k
{
234
104k
    pdf_array *a = NULL, *default_media = NULL;
235
104k
    double d[4];
236
104k
    int code = 0;
237
104k
    uint64_t i;
238
104k
    double userunit = 1.0;
239
240
104k
    code = pdfi_dict_get_type(ctx, page_dict, "MediaBox", PDF_ARRAY, (pdf_obj **)&default_media);
241
104k
    if (code < 0) {
242
1.45k
        pdfi_set_warning(ctx, code, NULL, W_PDF_BAD_MEDIABOX, "pdfi_get_media_size", NULL);
243
1.45k
        code = gs_erasepage(ctx->pgs);
244
1.45k
        return 0;
245
1.45k
    }
246
247
103k
    if (ctx->args.usecropbox) {
248
0
        if (a != NULL)
249
0
            pdfi_countdown(a);
250
0
        (void)pdfi_dict_get_type(ctx, page_dict, "CropBox", PDF_ARRAY, (pdf_obj **)&a);
251
0
    }
252
103k
    if (ctx->args.useartbox) {
253
0
        if (a != NULL)
254
0
            pdfi_countdown(a);
255
0
        (void)pdfi_dict_get_type(ctx, page_dict, "ArtBox", PDF_ARRAY, (pdf_obj **)&a);
256
0
    }
257
103k
    if (ctx->args.usebleedbox) {
258
0
        if (a != NULL)
259
0
            pdfi_countdown(a);
260
0
        (void)pdfi_dict_get_type(ctx, page_dict, "BleedBox", PDF_ARRAY, (pdf_obj **)&a);
261
0
    }
262
103k
    if (ctx->args.usetrimbox) {
263
0
        if (a != NULL)
264
0
            pdfi_countdown(a);
265
0
        (void)pdfi_dict_get_type(ctx, page_dict, "TrimBox", PDF_ARRAY, (pdf_obj **)&a);
266
0
    }
267
103k
    if (a == NULL) {
268
103k
        a = default_media;
269
103k
        pdfi_countup(a);
270
103k
    }
271
272
103k
    if (!ctx->args.nouserunit && !ctx->device_state.PassUserUnit) {
273
91.7k
        (void)pdfi_dict_knownget_number(ctx, page_dict, "UserUnit", &userunit);
274
91.7k
    }
275
103k
    ctx->page.UserUnit = userunit;
276
277
515k
    for (i=0;i<4;i++) {
278
412k
        code = pdfi_array_get_number(ctx, a, i, &d[i]);
279
412k
        d[i] *= userunit;
280
412k
    }
281
103k
    pdfi_countdown(a);
282
103k
    pdfi_countdown(default_media);
283
284
103k
    normalize_rectangle(d);
285
103k
    memcpy(ctx->page.Size, d, 4 * sizeof(double));
286
287
103k
    return code;
288
104k
}
289
290
static int pdfi_set_media_size(pdf_context *ctx, pdf_dict *page_dict)
291
0
{
292
0
    gs_c_param_list list;
293
0
    gs_param_float_array fa;
294
0
    pdf_array *a = NULL, *default_media = NULL;
295
0
    float fv[2];
296
0
    double d[4], d_crop[4];
297
0
    gs_rect bbox;
298
0
    int code, do_crop = false;
299
0
    uint64_t i;
300
0
    int64_t rotate = 0;
301
0
    double userunit = 1.0;
302
303
0
    code = pdfi_dict_get_type(ctx, page_dict, "MediaBox", PDF_ARRAY, (pdf_obj **)&default_media);
304
0
    if (code < 0) {
305
0
        pdfi_set_warning(ctx, code, NULL, W_PDF_BAD_MEDIABOX, "pdfi_get_media_size", NULL);
306
0
        code = gs_erasepage(ctx->pgs);
307
0
        return 0;
308
0
    }
309
310
0
    if (ctx->args.usecropbox) {
311
0
        if (a != NULL)
312
0
            pdfi_countdown(a);
313
0
        (void)pdfi_dict_get_type(ctx, page_dict, "CropBox", PDF_ARRAY, (pdf_obj **)&a);
314
0
    }
315
0
    if (ctx->args.useartbox) {
316
0
        if (a != NULL)
317
0
            pdfi_countdown(a);
318
0
        (void)pdfi_dict_get_type(ctx, page_dict, "ArtBox", PDF_ARRAY, (pdf_obj **)&a);
319
0
    }
320
0
    if (ctx->args.usebleedbox) {
321
0
        if (a != NULL)
322
0
            pdfi_countdown(a);
323
0
        (void)pdfi_dict_get_type(ctx, page_dict, "BleedBox", PDF_ARRAY, (pdf_obj **)&a);
324
0
    }
325
0
    if (ctx->args.usetrimbox) {
326
0
        if (a != NULL)
327
0
            pdfi_countdown(a);
328
0
        (void)pdfi_dict_get_type(ctx, page_dict, "TrimBox", PDF_ARRAY, (pdf_obj **)&a);
329
0
    }
330
0
    if (a == NULL) {
331
0
        code = pdfi_dict_get_type(ctx, page_dict, "CropBox", PDF_ARRAY, (pdf_obj **)&a);
332
0
        if (code >= 0 && pdfi_array_size(a) >= 4) {
333
0
            pdf_obj *box_obj = NULL;
334
335
0
            for (i=0;i<4;i++) {
336
0
                code = pdfi_array_get_no_store_R(ctx, a, i, &box_obj);
337
0
                if (code >= 0) {
338
0
                    code = pdfi_obj_to_real(ctx, box_obj, &d_crop[i]);
339
0
                    pdfi_countdown(box_obj);
340
0
                }
341
0
                if (code < 0)
342
0
                    break;
343
0
            }
344
0
            pdfi_countdown(a);
345
0
            if (code >= 0) {
346
0
                normalize_rectangle(d_crop);
347
0
                memcpy(ctx->page.Crop, d_crop, 4 * sizeof(double));
348
0
                do_crop = true;
349
0
            }
350
0
        }
351
0
        a = default_media;
352
0
    }
353
354
0
    if (!ctx->args.nouserunit) {
355
0
        if (ctx->device_state.PassUserUnit) {
356
0
            double unit = 1.0;
357
0
            (void)pdfi_dict_knownget_number(ctx, page_dict, "UserUnit", &unit);
358
0
            (void)pdfi_device_set_param_float(ctx->pgs->device, "UserUnit", unit);
359
0
        } else {
360
0
            (void)pdfi_dict_knownget_number(ctx, page_dict, "UserUnit", &userunit);
361
0
        }
362
0
    }
363
0
    ctx->page.UserUnit = userunit;
364
365
0
    for (i=0;i<4;i++) {
366
0
        pdf_obj *box_obj = NULL;
367
368
0
        code = pdfi_array_get_no_store_R(ctx, a, i, &box_obj);
369
0
        if (code >= 0) {
370
0
            code = pdfi_obj_to_real(ctx, box_obj, &d[i]);
371
0
            pdfi_countdown(box_obj);
372
0
        }
373
374
0
        if (code < 0) {
375
0
            pdfi_countdown(a);
376
0
            pdfi_set_warning(ctx, code, NULL, W_PDF_BAD_MEDIABOX, "pdfi_get_media_size", NULL);
377
0
            code = gs_erasepage(ctx->pgs);
378
0
            return 0;
379
0
        }
380
0
        d[i] *= userunit;
381
0
    }
382
0
    pdfi_countdown(a);
383
384
0
    normalize_rectangle(d);
385
0
    memcpy(ctx->page.Size, d, 4 * sizeof(double));
386
387
0
    code = pdfi_dict_get_int(ctx, page_dict, "Rotate", &rotate);
388
389
0
    rotate = rotate % 360;
390
391
0
    switch(rotate) {
392
0
        default:
393
0
        case 0:
394
0
        case 360:
395
0
        case -180:
396
0
        case 180:
397
0
            fv[0] = (float)(d[2] - d[0]);
398
0
            fv[1] = (float)(d[3] - d[1]);
399
0
            break;
400
0
        case -90:
401
0
        case 90:
402
0
        case -270:
403
0
        case 270:
404
0
            fv[1] = (float)(d[2] - d[0]);
405
0
            fv[0] = (float)(d[3] - d[1]);
406
0
            break;
407
0
    }
408
409
0
    fa.persistent = false;
410
0
    fa.data = fv;
411
0
    fa.size = 2;
412
413
    /* ----- setup specific device parameters ----- */
414
0
    gs_c_param_list_write(&list, ctx->memory);
415
416
0
    code = param_write_float_array((gs_param_list *)&list, ".MediaSize", &fa);
417
0
    if (code >= 0)
418
0
    {
419
0
        gx_device *dev = gs_currentdevice(ctx->pgs);
420
421
0
        gs_c_param_list_read(&list);
422
0
        code = gs_putdeviceparams(dev, (gs_param_list *)&list);
423
0
        if (code < 0) {
424
0
            gs_c_param_list_release(&list);
425
0
            return code;
426
0
        }
427
0
    }
428
0
    gs_c_param_list_release(&list);
429
    /* ----- end setup specific device parameters ----- */
430
431
    /* Resets the default matrix to NULL before doing initgraphics, because
432
     * otherwise initgraphics would keep old matrix.
433
     * (see pdfi_set_ctm())
434
     */
435
0
    gs_setdefaultmatrix(ctx->pgs, NULL);
436
0
    gs_initgraphics(ctx->pgs);
437
438
0
    switch(rotate) {
439
0
        case 0:
440
0
            break;
441
0
        case -270:
442
0
        case 90:
443
0
            gs_translate(ctx->pgs, 0, fv[1]);
444
0
            gs_rotate(ctx->pgs, -90);
445
0
            break;
446
0
        case -180:
447
0
        case 180:
448
0
            gs_translate(ctx->pgs, fv[0], fv[1]);
449
0
            gs_rotate(ctx->pgs, 180);
450
0
            break;
451
0
        case -90:
452
0
        case 270:
453
0
            gs_translate(ctx->pgs, fv[0], 0);
454
0
            gs_rotate(ctx->pgs, 90);
455
0
            break;
456
0
        default:
457
0
            break;
458
0
    }
459
460
0
    if (do_crop) {
461
0
        bbox.p.x = d_crop[0] - d[0];
462
0
        bbox.p.y = d_crop[1] - d[1];
463
0
        bbox.q.x = d_crop[2] - d[0];
464
0
        bbox.q.y = d_crop[3] - d[1];
465
466
0
        code = gs_rectclip(ctx->pgs, &bbox, 1);
467
0
        if (code < 0)
468
0
            return code;
469
0
    }
470
471
0
    gs_translate(ctx->pgs, d[0] * -1, d[1] * -1);
472
473
0
    code = gs_erasepage(ctx->pgs);
474
0
    return 0;
475
0
}
476
477
/* Setup default transfer functions */
478
static void pdfi_setup_transfers(pdf_context *ctx)
479
104k
{
480
104k
    if (ctx->pgs->set_transfer.red == 0x00) {
481
104k
        ctx->page.DefaultTransfers[0].proc = gs_identity_transfer;
482
104k
        memset(ctx->page.DefaultTransfers[0].values, 0x00, transfer_map_size * sizeof(frac));
483
104k
    } else {
484
0
        ctx->page.DefaultTransfers[0].proc = ctx->pgs->set_transfer.red->proc;
485
0
        memcpy(ctx->page.DefaultTransfers[0].values, ctx->pgs->set_transfer.red->values, transfer_map_size * sizeof(frac));
486
0
    }
487
104k
    if (ctx->pgs->set_transfer.green == 0x00) {
488
104k
        ctx->page.DefaultTransfers[1].proc = gs_identity_transfer;
489
104k
        memset(ctx->page.DefaultTransfers[1].values, 0x00, transfer_map_size * sizeof(frac));
490
104k
    } else {
491
0
        ctx->page.DefaultTransfers[1].proc = ctx->pgs->set_transfer.green->proc;
492
0
        memcpy(ctx->page.DefaultTransfers[1].values, ctx->pgs->set_transfer.green->values, transfer_map_size * sizeof(frac));
493
0
    }
494
104k
    if (ctx->pgs->set_transfer.blue == 0x00) {
495
104k
        ctx->page.DefaultTransfers[2].proc = gs_identity_transfer;
496
104k
        memset(ctx->page.DefaultTransfers[2].values, 0x00, transfer_map_size * sizeof(frac));
497
104k
    } else {
498
0
        ctx->page.DefaultTransfers[2].proc = ctx->pgs->set_transfer.blue->proc;
499
0
        memcpy(ctx->page.DefaultTransfers[2].values, ctx->pgs->set_transfer.blue->values, transfer_map_size * sizeof(frac));
500
0
    }
501
104k
    if (ctx->pgs->set_transfer.gray == 0x00) {
502
0
        ctx->page.DefaultTransfers[3].proc = gs_identity_transfer;
503
0
        memset(ctx->page.DefaultTransfers[3].values, 0x00, transfer_map_size * sizeof(frac));
504
104k
    } else {
505
104k
        ctx->page.DefaultTransfers[3].proc = ctx->pgs->set_transfer.gray->proc;
506
104k
        memcpy(ctx->page.DefaultTransfers[3].values, ctx->pgs->set_transfer.gray->values, transfer_map_size * sizeof(frac));
507
104k
    }
508
104k
    if (ctx->pgs->black_generation == 0x00) {
509
0
        ctx->page.DefaultBG.proc = gs_identity_transfer;
510
0
        memset(ctx->page.DefaultBG.values, 0x00, transfer_map_size * sizeof(frac));
511
104k
    } else {
512
104k
        ctx->page.DefaultBG.proc = ctx->pgs->black_generation->proc;
513
104k
        memcpy(ctx->page.DefaultBG.values, ctx->pgs->black_generation->values, transfer_map_size * sizeof(frac));
514
104k
    }
515
104k
    if (ctx->pgs->undercolor_removal == 0x00) {
516
0
        ctx->page.DefaultUCR.proc = gs_identity_transfer;
517
0
        memset(ctx->page.DefaultUCR.values, 0x00, transfer_map_size * sizeof(frac));
518
104k
    } else {
519
104k
        ctx->page.DefaultUCR.proc = ctx->pgs->undercolor_removal->proc;
520
104k
        memcpy(ctx->page.DefaultUCR.values, ctx->pgs->undercolor_removal->values, transfer_map_size * sizeof(frac));
521
104k
    }
522
104k
}
523
524
/* Return a dictionary containing information about the page. Basic information is that
525
 * required to render the page; if extended is true then additionally contains an
526
 * array of spot ink names and an array of dictionaries each of which contains
527
 * information about a font used on the page. THis is normally only used for tools
528
 * like pdf_info.ps
529
 */
530
int pdfi_page_info(pdf_context *ctx, uint64_t page_num, pdf_dict **info, bool extended)
531
144k
{
532
144k
    int code = 0, i=0;
533
144k
    pdf_dict *page_dict = NULL, *info_dict = NULL;
534
144k
    pdf_array *fonts_array = NULL, *spots_array = NULL;
535
144k
    pdf_array *a = NULL;
536
144k
    pdf_obj *o = NULL;
537
144k
    bool known = false;
538
144k
    double dummy;
539
540
144k
    code = pdfi_page_get_dict(ctx, page_num, &page_dict);
541
144k
    if (code < 0)
542
39.5k
        return code;
543
544
105k
    if (code > 0) {
545
0
        code = gs_note_error(gs_error_unknownerror);
546
0
        goto done;
547
0
    }
548
549
105k
    code = pdfi_dict_alloc(ctx, 6, &info_dict);
550
105k
    if (code < 0)
551
0
        goto done;
552
553
105k
    pdfi_countup(info_dict);
554
555
105k
    if (extended)
556
0
        code = pdfi_check_page(ctx, page_dict, &fonts_array, &spots_array, false);
557
105k
    else
558
105k
        code = pdfi_check_page(ctx, page_dict, NULL, NULL, false);
559
105k
    if (code < 0)
560
0
        goto done;
561
562
105k
    if (spots_array != NULL) {
563
0
        code = pdfi_dict_put(ctx, info_dict, "Spots", (pdf_obj *)spots_array);
564
0
        if (code < 0)
565
0
            goto done;
566
0
        pdfi_countdown(spots_array);
567
0
    }
568
569
105k
    if (fonts_array != NULL) {
570
0
        code = pdfi_dict_put(ctx, info_dict, "Fonts", (pdf_obj *)fonts_array);
571
0
        if (code < 0)
572
0
            goto done;
573
0
        pdfi_countdown(fonts_array);
574
0
    }
575
576
105k
    code = pdfi_dict_get_type(ctx, page_dict, "MediaBox", PDF_ARRAY, (pdf_obj **)&a);
577
105k
    if (code < 0)
578
1.45k
        pdfi_set_warning(ctx, code, NULL, W_PDF_BAD_MEDIABOX, "pdfi_page_info", NULL);
579
580
105k
    if (code >= 0) {
581
103k
        pdf_obj *box_obj = NULL;
582
583
518k
        for (i = 0;i < pdfi_array_size(a); i++) {
584
414k
            code = pdfi_array_get_no_store_R(ctx, a, i, &box_obj);
585
414k
            if (code >= 0) {
586
414k
                code = pdfi_obj_to_real(ctx, box_obj, &dummy);
587
414k
                pdfi_countdown(box_obj);
588
414k
            }
589
414k
            if (code < 0) {
590
163
                pdfi_set_warning(ctx, code, NULL, W_PDF_BAD_MEDIABOX, "pdfi_page_info", NULL);
591
163
                goto done;
592
163
            }
593
414k
        }
594
595
103k
        code = pdfi_dict_put(ctx, info_dict, "MediaBox", (pdf_obj *)a);
596
103k
        if (code < 0)
597
0
            goto done;
598
103k
        pdfi_countdown(a);
599
103k
        a = NULL;
600
103k
    }
601
602
104k
    code = pdfi_dict_get_type(ctx, page_dict, "ArtBox", PDF_ARRAY, (pdf_obj **)&a);
603
104k
    if (code >= 0) {
604
8.43k
        pdf_obj *box_obj = NULL;
605
606
42.1k
        for (i = 0;i < pdfi_array_size(a); i++) {
607
33.7k
            code = pdfi_array_get_no_store_R(ctx, a, i, &box_obj);
608
33.7k
            if (code >= 0) {
609
33.7k
                code = pdfi_obj_to_real(ctx, box_obj, &dummy);
610
33.7k
                pdfi_countdown(box_obj);
611
33.7k
            }
612
33.7k
            if (code < 0)
613
9
                break;
614
33.7k
        }
615
8.43k
        if (code >= 0) {
616
8.42k
            code = pdfi_dict_put(ctx, info_dict, "ArtBox", (pdf_obj *)a);
617
8.42k
            if (code < 0)
618
0
                goto done;
619
8.42k
        }
620
8.43k
        pdfi_countdown(a);
621
8.43k
        a = NULL;
622
8.43k
    }
623
624
104k
    code = pdfi_dict_get_type(ctx, page_dict, "CropBox", PDF_ARRAY, (pdf_obj **)&a);
625
104k
    if (code >= 0) {
626
36.7k
        pdf_obj *box_obj = NULL;
627
628
183k
        for (i = 0;i < pdfi_array_size(a); i++) {
629
146k
            code = pdfi_array_get_no_store_R(ctx, a, i, &box_obj);
630
146k
            if (code >= 0) {
631
146k
                code = pdfi_obj_to_real(ctx, box_obj, &dummy);
632
146k
                pdfi_countdown(box_obj);
633
146k
            }
634
146k
            if (code < 0)
635
68
                break;
636
146k
        }
637
36.7k
        if (code >= 0) {
638
36.6k
            code = pdfi_dict_put(ctx, info_dict, "CropBox", (pdf_obj *)a);
639
36.6k
            if (code < 0)
640
0
                goto done;
641
36.6k
        }
642
36.7k
        pdfi_countdown(a);
643
36.7k
        a = NULL;
644
36.7k
    }
645
646
104k
    code = pdfi_dict_get_type(ctx, page_dict, "TrimBox", PDF_ARRAY, (pdf_obj **)&a);
647
104k
    if (code >= 0) {
648
8.92k
        pdf_obj *box_obj = NULL;
649
650
44.5k
        for (i = 0;i < pdfi_array_size(a); i++) {
651
35.6k
            code = pdfi_array_get_no_store_R(ctx, a, i, &box_obj);
652
35.6k
            if (code >= 0) {
653
35.6k
                code = pdfi_obj_to_real(ctx, box_obj, &dummy);
654
35.6k
                pdfi_countdown(box_obj);
655
35.6k
            }
656
35.6k
            if (code < 0)
657
11
                break;
658
35.6k
        }
659
8.92k
        if (code >= 0) {
660
8.91k
            code = pdfi_dict_put(ctx, info_dict, "TrimBox", (pdf_obj *)a);
661
8.91k
            if (code < 0)
662
0
                goto done;
663
8.91k
        }
664
8.92k
        pdfi_countdown(a);
665
8.92k
        a = NULL;
666
8.92k
    }
667
668
104k
    code = pdfi_dict_get_type(ctx, page_dict, "BleedBox", PDF_ARRAY, (pdf_obj **)&a);
669
104k
    if (code >= 0) {
670
8.80k
        pdf_obj *box_obj = NULL;
671
672
44.0k
        for (i = 0;i < pdfi_array_size(a); i++) {
673
35.2k
            code = pdfi_array_get_no_store_R(ctx, a, i, &box_obj);
674
35.2k
            if (code >= 0) {
675
35.2k
                code = pdfi_obj_to_real(ctx, box_obj, &dummy);
676
35.2k
                pdfi_countdown(box_obj);
677
35.2k
            }
678
35.2k
            if (code < 0)
679
23
                break;
680
35.2k
        }
681
8.80k
        if (code >= 0) {
682
8.78k
            code = pdfi_dict_put(ctx, info_dict, "BleedBox", (pdf_obj *)a);
683
8.78k
            if (code < 0)
684
0
                goto done;
685
8.78k
        }
686
8.80k
        pdfi_countdown(a);
687
8.80k
        a = NULL;
688
8.80k
    }
689
104k
    code = 0;
690
691
104k
    code = pdfi_dict_get(ctx, page_dict, "Rotate", &o);
692
104k
    if (code >= 0) {
693
40.9k
        if (pdfi_type_of(o) == PDF_INT || pdfi_type_of(o) == PDF_REAL) {
694
40.9k
            code = pdfi_dict_put(ctx, info_dict, "Rotate", o);
695
40.9k
            if (code < 0)
696
0
                goto done;
697
40.9k
        }
698
40.9k
        pdfi_countdown(o);
699
40.9k
    }
700
701
104k
    code = pdfi_dict_get(ctx, page_dict, "UserUnit", &o);
702
104k
    if (code >= 0) {
703
611
        if (pdfi_type_of(o) == PDF_INT || pdfi_type_of(o) == PDF_REAL) {
704
611
            code = pdfi_dict_put(ctx, info_dict, "UserUnit", o);
705
611
            if (code < 0)
706
0
                goto done;
707
611
        }
708
611
        pdfi_countdown(o);
709
611
    }
710
711
104k
    if (ctx->page.has_transparency)
712
11.1k
        code = pdfi_dict_put(ctx, info_dict, "UsesTransparency", PDF_TRUE_OBJ);
713
93.8k
    else
714
93.8k
        code = pdfi_dict_put(ctx, info_dict, "UsesTransparency", PDF_FALSE_OBJ);
715
104k
    if (code < 0)
716
0
        goto done;
717
718
104k
    code = pdfi_dict_known(ctx, page_dict, "Annots", &known);
719
104k
    if (code >= 0 && known)
720
26.9k
        code = pdfi_dict_put(ctx, info_dict, "Annots", PDF_TRUE_OBJ);
721
78.0k
    else
722
78.0k
        code = pdfi_dict_put(ctx, info_dict, "Annots", PDF_FALSE_OBJ);
723
104k
    if (code < 0)
724
0
        goto done;
725
726
104k
    code = pdfi_object_alloc(ctx, PDF_INT, 0, &o);
727
104k
    if (code >= 0) {
728
104k
        pdfi_countup(o);
729
104k
        ((pdf_num *)o)->value.i = ctx->page.num_spots;
730
104k
        code = pdfi_dict_put(ctx, info_dict, "NumSpots", o);
731
104k
        pdfi_countdown(o);
732
104k
        o = NULL;
733
104k
        if (code < 0)
734
0
            goto done;
735
104k
    }
736
737
105k
done:
738
105k
    if (code < 0) {
739
163
        pdfi_countdown(info_dict);
740
163
        info_dict = NULL;
741
163
        *info = NULL;
742
163
    } else
743
104k
        *info = info_dict;
744
745
105k
    pdfi_countdown(a);
746
105k
    pdfi_countdown(page_dict);
747
105k
    return code;
748
104k
}
749
750
int pdfi_page_get_dict(pdf_context *ctx, uint64_t page_num, pdf_dict **dict)
751
250k
{
752
250k
    int code = 0;
753
250k
    uint64_t page_offset = 0;
754
755
250k
    code = pdfi_loop_detector_mark(ctx);
756
250k
    if (code < 0)
757
0
        return code;
758
759
250k
    if (ctx->PagesTree == NULL) {
760
1.15k
        pdf_obj *o = NULL;
761
1.15k
        pdf_name *n = NULL;
762
        /* The only way this should be true is if the Pages entry in the Root dictionary
763
         * points to a single instance of a Page dictionary, instead of to a Pages dictionary.
764
         * in which case, simply retrieve that dictionary and return.
765
         */
766
1.15k
        code = pdfi_dict_get(ctx, ctx->Root, "Pages", &o);
767
1.15k
        if (code < 0)
768
0
            goto page_error;
769
1.15k
        if (pdfi_type_of(o) != PDF_DICT) {
770
0
            code = gs_note_error(gs_error_typecheck);
771
0
            goto page_error;
772
0
        }
773
1.15k
        code = pdfi_dict_get_type(ctx, (pdf_dict *)o, "Type", PDF_NAME, (pdf_obj **)&n);
774
1.15k
        if (code == 0) {
775
1.15k
            if(pdfi_name_is(n, "Page")) {
776
1.15k
                *dict = (pdf_dict *)o;
777
1.15k
                pdfi_countup(*dict);
778
1.15k
            } else
779
0
                code = gs_note_error(gs_error_undefined);
780
1.15k
        }
781
1.15k
page_error:
782
1.15k
        pdfi_loop_detector_cleartomark(ctx);
783
1.15k
        pdfi_countdown(o);
784
1.15k
        pdfi_countdown(n);
785
1.15k
        return code;
786
1.15k
    }
787
788
249k
    code = pdfi_loop_detector_add_object(ctx, ctx->PagesTree->object_num);
789
249k
    if (code < 0)
790
0
        goto exit;
791
792
249k
    code = pdfi_get_page_dict(ctx, ctx->PagesTree, page_num, &page_offset, dict, NULL);
793
249k
    if (code > 0)
794
34
        code = gs_error_unknownerror;
795
796
    /* Cache the page_dict number in page_array */
797
249k
    if (*dict)
798
209k
        ctx->page_array[page_num] = (*dict)->object_num;
799
800
249k
 exit:
801
249k
    pdfi_loop_detector_cleartomark(ctx);
802
249k
    return code;
803
249k
}
804
805
/* Find the page number that corresponds to a page dictionary
806
 * Uses page_array cache to minimize the number of times a page_dict needs to
807
 * be fetched, because this is expensive.
808
 */
809
int pdfi_page_get_number(pdf_context *ctx, pdf_dict *target_dict, uint64_t *page_num)
810
6.25k
{
811
6.25k
    uint64_t i;
812
6.25k
    int code = 0;
813
6.25k
    pdf_dict *page_dict = NULL;
814
6.25k
    uint32_t object_num;
815
816
6.97k
    for (i=0; i<ctx->num_pages; i++) {
817
        /* If the page has been processed before, then its object_num should already
818
         * be cached in the page_array, so check that first
819
         */
820
6.94k
        object_num = ctx->page_array[i];
821
6.94k
        if (object_num == 0) {
822
            /* It wasn't cached, so this will cache it */
823
1.05k
            code = pdfi_page_get_dict(ctx, i, &page_dict);
824
1.05k
            if (code < 0)
825
203
                continue;
826
848
            object_num = ctx->page_array[i];
827
848
        }
828
6.73k
        if (target_dict->object_num == object_num) {
829
6.22k
            *page_num = i;
830
6.22k
            goto exit;
831
6.22k
        }
832
833
513
        pdfi_countdown(page_dict);
834
513
        page_dict = NULL;
835
513
    }
836
837
34
    code = gs_note_error(gs_error_undefined);
838
839
6.25k
 exit:
840
6.25k
    pdfi_countdown(page_dict);
841
6.25k
    return code;
842
34
}
843
844
static void release_page_DefaultSpaces(pdf_context *ctx)
845
209k
{
846
209k
    if (ctx->page.DefaultGray_cs != NULL) {
847
0
        if (ctx->page.DefaultGray_cs->interpreter_data != NULL) {
848
0
            pdf_obj *o = (pdf_obj *)(ctx->page.DefaultGray_cs->interpreter_data);
849
0
            if (o != NULL && pdfi_type_of(o) == PDF_NAME) {
850
0
                pdfi_countdown(o);
851
0
                ctx->page.DefaultGray_cs->interpreter_data = NULL;
852
0
            }
853
0
        }
854
0
        rc_decrement(ctx->page.DefaultGray_cs, "pdfi_page_render");
855
0
        ctx->page.DefaultGray_cs = NULL;
856
0
    }
857
209k
    if (ctx->page.DefaultRGB_cs != NULL) {
858
513
        if (ctx->page.DefaultRGB_cs->interpreter_data != NULL) {
859
513
            pdf_obj *o = (pdf_obj *)(ctx->page.DefaultRGB_cs->interpreter_data);
860
513
            if (o != NULL && pdfi_type_of(o) == PDF_NAME) {
861
78
                pdfi_countdown(o);
862
78
                ctx->page.DefaultRGB_cs->interpreter_data = NULL;
863
78
            }
864
513
        }
865
513
        rc_decrement(ctx->page.DefaultRGB_cs, "pdfi_page_render");
866
513
        ctx->page.DefaultRGB_cs = NULL;
867
513
    }
868
209k
    if (ctx->page.DefaultCMYK_cs != NULL) {
869
27
        if (ctx->page.DefaultCMYK_cs->interpreter_data != NULL) {
870
27
            pdf_obj *o = (pdf_obj *)(ctx->page.DefaultCMYK_cs->interpreter_data);
871
27
            if (o != NULL && pdfi_type_of(o) == PDF_NAME) {
872
0
                pdfi_countdown(o);
873
0
                ctx->page.DefaultCMYK_cs->interpreter_data = NULL;
874
0
            }
875
27
        }
876
27
        rc_decrement(ctx->page.DefaultCMYK_cs, "pdfi_page_render");
877
27
        ctx->page.DefaultCMYK_cs = NULL;
878
27
    }
879
209k
}
880
881
static int setup_page_DefaultSpaces(pdf_context *ctx, pdf_dict *page_dict)
882
104k
{
883
    /* First off, discard any dangling Default* colour spaces, just in case. */
884
104k
    release_page_DefaultSpaces(ctx);
885
886
104k
    return(pdfi_setup_DefaultSpaces(ctx, page_dict));
887
104k
}
888
889
int pdfi_page_render(pdf_context *ctx, uint64_t page_num, bool init_graphics)
890
104k
{
891
104k
    int code, code1=0;
892
104k
    pdf_dict *page_dict = NULL;
893
104k
    bool page_group_known = false;
894
104k
    pdf_dict *group_dict = NULL;
895
104k
    bool page_dict_error = false;
896
104k
    bool need_pdf14 = false; /* true if the device is needed and was successfully pushed */
897
104k
    int trans_depth = 0; /* -1 means special mode for transparency simulation */
898
899
104k
    if (page_num > ctx->num_pages)
900
0
        return_error(gs_error_rangecheck);
901
902
104k
    if (ctx->args.pdfdebug)
903
0
        outprintf(ctx->memory, "%% Processing Page %"PRIi64" content stream\n", page_num + 1);
904
905
104k
    code = pdfi_page_get_dict(ctx, page_num, &page_dict);
906
104k
    if (code < 0) {
907
3
        char extra_info[256];
908
909
3
        page_dict_error = true;
910
3
        gs_snprintf(extra_info, sizeof(extra_info), "*** ERROR: Page %ld has invalid Page dict, skipping\n", page_num+1);
911
3
        if (code == gs_error_VMerror ||
912
3
        ((code = pdfi_set_error_stop(ctx, code, NULL, E_PDF_PAGEDICTERROR, "pdfi_page_render", extra_info)) < 0)) {
913
0
            goto exit3;
914
0
        }
915
3
    }
916
917
104k
    code = pdfi_check_page(ctx, page_dict, NULL, NULL, init_graphics);
918
104k
    if (code < 0)
919
3
        goto exit3;
920
921
104k
    if (ctx->args.pdfdebug) {
922
0
        dbgmprintf2(ctx->memory, "Current page %ld transparency setting is %d", page_num+1,
923
0
                ctx->page.has_transparency);
924
925
0
        if (ctx->device_state.spot_capable)
926
0
            dbgmprintf1(ctx->memory, ", spots=%d\n", ctx->page.num_spots);
927
0
        else
928
0
            dbgmprintf(ctx->memory, "\n");
929
0
    }
930
931
104k
    code = pdfi_dict_knownget_type(ctx, page_dict, "Group", PDF_DICT, (pdf_obj **)&group_dict);
932
    /* Ignore errors retrieving the Group dictionary, we will just ignore it. This allows us
933
     * to handle files such as Bug #705206 where the Group dictionary is a free object in a
934
     * compressed object stream.
935
     */
936
104k
    if (code < 0 && (code = pdfi_set_error_stop(ctx, code, NULL, E_BAD_GROUP_DICT, "pdfi_page_render", NULL)) < 0)
937
0
        goto exit2;
938
939
104k
    if (group_dict != NULL)
940
12.7k
        page_group_known = true;
941
942
104k
    pdfi_countdown(ctx->page.CurrentPageDict);
943
104k
    ctx->page.CurrentPageDict = page_dict;
944
104k
    pdfi_countup(ctx->page.CurrentPageDict);
945
946
    /* In case we don't call pdfi_set_media_size, which sets this up.
947
     * We shouldn't ever use it in that case, but best to be safe.
948
     */
949
104k
    ctx->page.UserUnit = 1.0f;
950
    /* If we are being called from the PDF interpreter then
951
     * we need to set up the page  and the default graphics state
952
     * but if we are being called from PostScript we do *not*
953
     * want to alter any of the graphics state or the media size.
954
     */
955
    /* TODO: I think this is a mix of things we might need to
956
     * still be setting up.
957
     * (for example, I noticed the blendmode and moved it outside the if)
958
     */
959
104k
    if (init_graphics) {
960
0
        code = pdfi_set_media_size(ctx, page_dict);
961
0
        if (code < 0)
962
0
            goto exit2;
963
964
0
        pdfi_set_ctm(ctx);
965
966
104k
    } else {
967
        /* Gets ctx->page.Size setup correctly
968
         * TODO: Probably not right if the page is rotated?
969
         * page.Size is needed by the transparency code,
970
         * not sure where else it might be used, if anywhere.
971
         */
972
104k
        pdfi_get_media_size(ctx, page_dict);
973
104k
    }
974
975
    /* Write the various CropBox, TrimBox etc to the device */
976
104k
    pdfi_pdfmark_write_boxes(ctx, page_dict);
977
978
104k
    code = setup_page_DefaultSpaces(ctx, page_dict);
979
104k
    if (code < 0)
980
0
        goto exit2;
981
982
104k
    pdfi_setup_transfers(ctx);
983
984
    /* Set whether device needs OP support
985
     * This needs to be before transparency device is pushed, if applicable
986
     */
987
104k
    pdfi_trans_set_needs_OP(ctx);
988
104k
    pdfi_oc_init(ctx);
989
990
104k
    code = pdfi_gsave(ctx);
991
104k
    if (code < 0)
992
0
        goto exit2;
993
994
    /* Figure out if pdf14 device is needed.
995
     * This can be either for normal transparency deviceN, or because we are using
996
     * Overprint=/simulate for other devices
997
     */
998
104k
    if (ctx->page.has_transparency) {
999
11.1k
        need_pdf14 = true;
1000
11.1k
        if (ctx->page.simulate_op)
1001
0
            trans_depth = -1;
1002
93.4k
    } else {
1003
        /* This is the case where we are simulating overprint without transparency */
1004
93.4k
        if (ctx->page.simulate_op) {
1005
0
            need_pdf14 = true;
1006
0
            trans_depth = -1;
1007
0
        }
1008
93.4k
    }
1009
104k
    if (need_pdf14) {
1010
        /* We don't retain the PDF14 device */
1011
11.1k
        code = gs_push_pdf14trans_device(ctx->pgs, false, false, trans_depth, ctx->page.num_spots);
1012
11.1k
        if (code >= 0) {
1013
11.1k
            if (ctx->page.has_transparency && page_group_known) {
1014
4.49k
                code = pdfi_trans_begin_page_group(ctx, page_dict, group_dict);
1015
                /* If setting the page group failed for some reason, abandon the page group,
1016
                 *  but continue with the page
1017
                 */
1018
4.49k
                if (code < 0)
1019
120
                    page_group_known = false;
1020
4.49k
            }
1021
11.1k
        } else {
1022
            /* Couldn't push the transparency compositor.
1023
             * This is probably fatal, but attempt to recover by abandoning transparency
1024
             */
1025
0
            ctx->page.has_transparency = false;
1026
0
            need_pdf14 = false;
1027
0
        }
1028
11.1k
    }
1029
1030
    /* Init a base_pgs graphics state for Patterns
1031
     * (this has to be after transparency device pushed, if applicable)
1032
     */
1033
104k
    pdfi_set_DefaultQState(ctx, ctx->pgs);
1034
1035
    /* Render one page (including annotations) */
1036
104k
    if (!ctx->args.QUIET)
1037
0
        outprintf(ctx->memory, "Page %"PRId64"\n", page_num + 1);
1038
1039
104k
    code = pdfi_process_one_page(ctx, page_dict);
1040
1041
104k
    if (need_pdf14 && ctx->page.has_transparency && page_group_known) {
1042
4.37k
        code1 = pdfi_trans_end_group(ctx);
1043
4.37k
    }
1044
1045
104k
    if (need_pdf14) {
1046
11.1k
        if (code1 < 0) {
1047
0
            (void)gs_abort_pdf14trans_device(ctx->pgs);
1048
0
            code = code1;
1049
0
            goto exit1;
1050
0
        }
1051
1052
11.1k
        code1 = gs_pop_pdf14trans_device(ctx->pgs, false);
1053
11.1k
        if (code1 < 0) {
1054
7
            code = code1;
1055
7
            goto exit1;
1056
7
        }
1057
11.1k
    }
1058
1059
104k
exit1:
1060
104k
    pdfi_free_DefaultQState(ctx);
1061
104k
    pdfi_grestore(ctx);
1062
1063
104k
exit2:
1064
104k
    pdfi_countdown(ctx->page.CurrentPageDict);
1065
104k
    ctx->page.CurrentPageDict = NULL;
1066
1067
104k
exit3:
1068
104k
    pdfi_countdown(page_dict);
1069
104k
    pdfi_countdown(group_dict);
1070
1071
104k
    release_page_DefaultSpaces(ctx);
1072
1073
    /* Flush any pattern tiles. We don't want to (potentially) return to PostScript
1074
     * with any pattern tiles referencing our objects, in case the garbager runs.
1075
     */
1076
104k
    gx_pattern_cache_flush(gstate_pattern_cache(ctx->pgs));
1077
    /* We could be smarter, but for now.. purge for each page */
1078
104k
    pdfi_purge_cache_resource_font(ctx);
1079
1080
104k
    if (code == 0 || (!ctx->args.pdfstoponerror && code != gs_error_pdf_stackoverflow))
1081
104k
        if (!page_dict_error && ctx->finish_page != NULL)
1082
0
            code = ctx->finish_page(ctx);
1083
104k
    return code;
1084
104k
}