Coverage Report

Created: 2025-06-10 07:27

/src/ghostpdl/pdf/pdf_mark.c
Line
Count
Source (jump to first uncovered line)
1
/* Copyright (C) 2020-2024 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
/* pdfmark handling for the PDF interpreter */
17
18
#include "pdf_int.h"
19
#include "pdf_stack.h"
20
#include "pdf_file.h"
21
#include "pdf_dict.h"
22
#include "pdf_array.h"
23
#include "pdf_loop_detect.h"
24
#include "pdf_mark.h"
25
#include "pdf_obj.h"
26
#include "pdf_misc.h"
27
#include "pdf_page.h"
28
#include "pdf_deref.h"
29
30
#include "gscoord.h"         /* For gs_currentmatrix */
31
32
static int pdfi_get_named_dest(pdf_context *ctx, pdf_obj *Named, pdf_obj **Dest);
33
34
static int pdfi_pdfmark_setparam_obj(pdf_context *ctx, pdf_obj *obj, gs_param_string *entry)
35
709k
{
36
709k
    int code = 0;
37
709k
    byte *data = NULL;
38
709k
    int size = 0;
39
40
709k
    code = pdfi_obj_to_string(ctx, obj, &data, &size);
41
709k
    if (code < 0)
42
240
        return code;
43
709k
    entry->data = data;
44
709k
    entry->size = size;
45
709k
    entry->persistent = false;
46
709k
    return 0;
47
709k
}
48
49
static int pdfi_pdfmark_setparam_pair(pdf_context *ctx, pdf_name *Key, pdf_obj *Value,
50
                                   gs_param_string *entry)
51
308k
{
52
308k
    int code = 0;
53
54
    /* Handle the Key */
55
308k
    if (pdfi_type_of(Key) != PDF_NAME) {
56
0
        code = gs_note_error(gs_error_typecheck);
57
0
        goto exit;
58
0
    }
59
60
308k
    code = pdfi_pdfmark_setparam_obj(ctx, (pdf_obj *)Key, entry);
61
308k
    if (code < 0)
62
0
        goto exit;
63
64
308k
    code = pdfi_pdfmark_setparam_obj(ctx, Value, entry+1);
65
308k
    if (code < 0)
66
161
        goto exit;
67
68
308k
 exit:
69
308k
    return code;
70
308k
}
71
72
73
/* Note: this isn't part of the obj_to_string() stuff */
74
static int pdfi_pdfmark_ctm_str(pdf_context *ctx, gs_matrix *ctm, byte **data, int *len)
75
148k
{
76
148k
    int size = 100;
77
148k
    char *buf;
78
79
148k
    buf = (char *)gs_alloc_bytes(ctx->memory, size, "pdfi_pdfmark_ctm_str(data)");
80
148k
    if (buf == NULL)
81
0
        return_error(gs_error_VMerror);
82
148k
    snprintf(buf, size, "[%.4f %.4f %.4f %.4f %.4f %.4f]",
83
148k
             ctm->xx, ctm->xy, ctm->yx, ctm->yy, ctm->tx, ctm->ty);
84
148k
    *data = (byte *)buf;
85
148k
    *len = strlen(buf);
86
148k
    return 0;
87
148k
}
88
89
/* Write an string array command to the device (e.g. pdfmark) */
90
static int pdfi_pdfmark_write_array(pdf_context *ctx, gs_param_string_array *array_list, const char *command)
91
148k
{
92
148k
    gs_c_param_list list;
93
148k
    int code;
94
95
    /* Set the list to writeable, and initialise it */
96
148k
    gs_c_param_list_write(&list, ctx->memory);
97
    /* We don't want keys to be persistent, as we are going to throw
98
     * away our array, force them to be copied
99
     */
100
148k
    gs_param_list_set_persistent_keys((gs_param_list *) &list, false);
101
102
    /* Make really sure the list is writable, but don't initialise it */
103
148k
    gs_c_param_list_write_more(&list);
104
105
    /* Add the param string array to the list */
106
148k
    code = param_write_string_array((gs_param_list *)&list, command, array_list);
107
148k
    if (code < 0)
108
0
        return code;
109
110
    /* Set the param list back to readable, so putceviceparams can readit (mad...) */
111
148k
    gs_c_param_list_read(&list);
112
113
    /* and set the actual device parameters */
114
148k
    code = gs_putdeviceparams(ctx->pgs->device, (gs_param_list *)&list);
115
116
148k
    gs_c_param_list_release(&list);
117
118
148k
    return code;
119
148k
}
120
121
/* Write an string array command to the device (e.g. pdfmark) */
122
static int pdfi_pdfmark_write_string(pdf_context *ctx, gs_param_string *param_string, const char *command)
123
443
{
124
443
    gs_c_param_list list;
125
443
    int code;
126
127
    /* Set the list to writeable, and initialise it */
128
443
    gs_c_param_list_write(&list, ctx->memory);
129
    /* We don't want keys to be persistent, as we are going to throw
130
     * away our array, force them to be copied
131
     */
132
443
    gs_param_list_set_persistent_keys((gs_param_list *) &list, false);
133
134
    /* Make really sure the list is writable, but don't initialise it */
135
443
    gs_c_param_list_write_more(&list);
136
137
    /* Add the param string array to the list */
138
443
    code = param_write_string((gs_param_list *)&list, command, param_string);
139
443
    if (code < 0)
140
0
        return code;
141
142
    /* Set the param list back to readable, so putceviceparams can readit (mad...) */
143
443
    gs_c_param_list_read(&list);
144
145
    /* and set the actual device parameters */
146
443
    code = gs_putdeviceparams(ctx->pgs->device, (gs_param_list *)&list);
147
148
443
    gs_c_param_list_release(&list);
149
150
443
    return code;
151
443
}
152
153
/* Apparently the strings to pass to the device are:
154
   key1 val1 .... keyN valN CTM "ANN"
155
   CTM is (for example) "[1.0 0 0 1.0 0 0]"
156
157
   This takes an optional 'label' argument which can be NULL.
158
159
   the /pdfmark command has this array of strings as a parameter, i.e.
160
   [ 'label' key1 val1 .... keyN valN CTM "ANN" ] /pdfmark
161
162
   Apparently the "type" doesn't have a leading "/" but the other names in the
163
   keys do need the "/"
164
165
   See plparams.c/process_pdfmark()
166
*/
167
static int pdfi_pdfmark_from_dict_withlabel(pdf_context *ctx, pdf_indirect_ref *label,
168
                                         pdf_dict *dict, gs_matrix *ctm, const char *type)
169
59.2k
{
170
59.2k
    int code = 0;
171
59.2k
    int size;
172
59.2k
    uint64_t dictsize;
173
59.2k
    uint64_t index;
174
59.2k
    uint64_t keynum = 0;
175
59.2k
    int i;
176
59.2k
    pdf_name *Key = NULL;
177
59.2k
    pdf_obj *Value = NULL;
178
59.2k
    pdf_obj *tempobj = NULL;
179
59.2k
    gs_param_string *parray = NULL;
180
59.2k
    gs_param_string_array array_list;
181
59.2k
    byte *ctm_data = NULL;
182
59.2k
    int ctm_len;
183
59.2k
    gs_matrix ctm_placeholder;
184
59.2k
    int offset = 0;
185
186
    /* If ctm not provided, make a placeholder */
187
59.2k
    if (!ctm) {
188
23.0k
        gs_currentmatrix(ctx->pgs, &ctm_placeholder);
189
23.0k
        ctm = &ctm_placeholder;
190
23.0k
    }
191
192
59.2k
    dictsize = pdfi_dict_entries(dict);
193
59.2k
    size = dictsize*2 + 2; /* pairs + CTM + type */
194
59.2k
    if (label)
195
4.26k
        size += 1;
196
197
59.2k
    parray = (gs_param_string *)gs_alloc_bytes(ctx->memory, size*sizeof(gs_param_string),
198
59.2k
                                               "pdfi_pdfmark_from_dict(parray)");
199
59.2k
    if (parray == NULL) {
200
0
        code = gs_note_error(gs_error_VMerror);
201
0
        goto exit;
202
0
    }
203
59.2k
    memset(parray, 0, size*sizeof(gs_param_string));
204
205
59.2k
    if (label) {
206
4.26k
        code = pdfi_pdfmark_setparam_obj(ctx, (pdf_obj *)label, parray+0);
207
4.26k
        offset += 1;
208
4.26k
    }
209
210
    /* Get each (key,val) pair from dict and setup param for it */
211
59.2k
    if (dictsize > 0) {
212
50.9k
        code = pdfi_dict_key_first(ctx, dict, (pdf_obj **)&Key, &index);
213
308k
        while (code >= 0) {
214
308k
            code = pdfi_dict_get_no_deref(ctx, dict, Key, &Value);
215
308k
            if (code < 0) goto exit;
216
217
308k
            code = pdfi_pdfmark_setparam_pair(ctx, Key, Value, parray+offset+(keynum*2));
218
308k
            if (code < 0) goto exit;
219
220
308k
            pdfi_countdown(Key);
221
308k
            Key = NULL;
222
308k
            pdfi_countdown(Value);
223
308k
            Value = NULL;
224
308k
            pdfi_countdown(tempobj);
225
308k
            tempobj = NULL;
226
227
308k
            code = pdfi_dict_key_next(ctx, dict, (pdf_obj **)&Key, &index);
228
308k
            if (code == gs_error_undefined) {
229
50.7k
                code = 0;
230
50.7k
                break;
231
50.7k
            }
232
257k
            keynum ++;
233
257k
        }
234
50.9k
    }
235
59.1k
    if (code < 0) goto exit;
236
237
    /* CTM */
238
59.1k
    code = pdfi_pdfmark_ctm_str(ctx, ctm, &ctm_data, &ctm_len);
239
59.1k
    if (code < 0) goto exit;
240
59.1k
    parray[size-2].data = ctm_data;
241
59.1k
    parray[size-2].size = ctm_len;
242
243
    /* Type (e.g. ANN, DOCINFO) */
244
59.1k
    parray[size-1].data = (const byte *)type;
245
59.1k
    parray[size-1].size = strlen(type);
246
247
59.1k
    array_list.data = parray;
248
59.1k
    array_list.persistent = false;
249
59.1k
    array_list.size = size;
250
251
59.1k
    code = pdfi_pdfmark_write_array(ctx, &array_list, "pdfmark");
252
253
59.2k
 exit:
254
59.2k
    pdfi_countdown(Key);
255
59.2k
    pdfi_countdown(Value);
256
59.2k
    pdfi_countdown(tempobj);
257
59.2k
    if (parray != NULL) {
258
        /* Free the param data except the last two which are handled separately */
259
680k
        for (i=0; i<size-2; i++) {
260
621k
            if (parray[i].data)
261
620k
                gs_free_object(ctx->memory, (byte *)parray[i].data, "pdfi_pdfmark_from_dict(parray)");
262
621k
        }
263
59.2k
    }
264
59.2k
    if (ctm_data)
265
59.1k
        gs_free_object(ctx->memory, ctm_data, "pdfi_pdfmark_from_dict(ctm_data)");
266
59.2k
    gs_free_object(ctx->memory, parray, "pdfi_pdfmark_from_dict(parray)");
267
59.2k
    return code;
268
59.1k
}
269
270
/* Do a pdfmark from a dictionary */
271
int pdfi_pdfmark_from_dict(pdf_context *ctx, pdf_dict *dict, gs_matrix *ctm, const char *type)
272
55.0k
{
273
55.0k
    return pdfi_pdfmark_from_dict_withlabel(ctx, NULL, dict, ctm, type);
274
55.0k
}
275
276
/* Does a pdfmark, from a c-array of pdf_obj's
277
 * This will put in a dummy ctm if none provided
278
 */
279
int pdfi_pdfmark_from_objarray(pdf_context *ctx, pdf_obj **objarray, int len,
280
                            gs_matrix *ctm, const char *type)
281
89.3k
{
282
89.3k
    int code = 0;
283
89.3k
    int size;
284
89.3k
    gs_param_string *parray = NULL;
285
89.3k
    gs_param_string_array array_list;
286
89.3k
    byte *ctm_data = NULL;
287
89.3k
    int ctm_len;
288
89.3k
    gs_matrix ctm_placeholder;
289
89.3k
    int i;
290
291
    /* If ctm not provided, make a placeholder */
292
89.3k
    if (!ctm) {
293
89.3k
        gs_currentmatrix(ctx->pgs, &ctm_placeholder);
294
89.3k
        ctm = &ctm_placeholder;
295
89.3k
    }
296
297
89.3k
    size = len + 2; /* data + CTM + type */
298
299
89.3k
    parray = (gs_param_string *)gs_alloc_bytes(ctx->memory, size*sizeof(gs_param_string),
300
89.3k
                                               "pdfi_pdfmark_from_objarray(parray)");
301
89.3k
    if (parray == NULL) {
302
0
        code = gs_note_error(gs_error_VMerror);
303
0
        goto exit;
304
0
    }
305
89.3k
    memset(parray, 0, size *sizeof(gs_param_string));
306
307
177k
    for (i=0; i<len; i++) {
308
88.0k
        code = pdfi_pdfmark_setparam_obj(ctx, objarray[i], parray+i);
309
88.0k
        if (code < 0) goto exit;
310
88.0k
    }
311
312
    /* CTM */
313
89.3k
    code = pdfi_pdfmark_ctm_str(ctx, ctm, &ctm_data, &ctm_len);
314
89.3k
    if (code < 0) goto exit;
315
89.3k
    parray[len].data = ctm_data;
316
89.3k
    parray[len].size = ctm_len;
317
318
    /* Type (e.g. ANN, DOCINFO) */
319
89.3k
    parray[len+1].data = (const byte *)type;
320
89.3k
    parray[len+1].size = strlen(type);
321
322
89.3k
    array_list.data = parray;
323
89.3k
    array_list.persistent = false;
324
89.3k
    array_list.size = size;
325
326
89.3k
    code = pdfi_pdfmark_write_array(ctx, &array_list, "pdfmark");
327
328
89.3k
 exit:
329
89.3k
    if (parray != NULL) {
330
177k
        for (i=0; i<len; i++) {
331
88.0k
            gs_free_object(ctx->memory, (byte *)parray[i].data, "pdfi_pdfmark_from_objarray(parray)");
332
88.0k
        }
333
89.3k
    }
334
89.3k
    if (ctm_data)
335
89.3k
        gs_free_object(ctx->memory, ctm_data, "pdfi_pdfmark_from_objarray(ctm_data)");
336
89.3k
    gs_free_object(ctx->memory, parray, "pdfi_pdfmark_from_objarray(parray)");
337
89.3k
    return code;
338
89.3k
}
339
340
/* Send an arbitrary object as a string, with command 'cmd'
341
 * This is not a pdfmark, has no ctm.
342
 */
343
int pdfi_pdfmark_object(pdf_context *ctx, pdf_obj *object, const char *cmd)
344
467
{
345
467
    gs_param_string param_string;
346
467
    int code = 0;
347
348
467
    param_string.data = NULL;
349
350
467
    code = pdfi_loop_detector_mark(ctx);
351
467
    if (code < 0)
352
0
        goto exit;
353
354
467
    code = pdfi_resolve_indirect_loop_detect(ctx, NULL, object, true);
355
467
    (void)pdfi_loop_detector_cleartomark(ctx);
356
467
    if (code < 0)
357
24
        goto exit;
358
359
443
    code = pdfi_pdfmark_setparam_obj(ctx, object, &param_string);
360
443
    if (code < 0)
361
0
        goto exit;
362
363
443
    code = pdfi_pdfmark_write_string(ctx, &param_string, cmd);
364
365
467
exit:
366
467
    if (param_string.data != NULL)
367
443
        gs_free_object(ctx->memory, (byte *)param_string.data, "free data transferred to param_string in pdfi_pdfmark_object\n");
368
467
    return code;
369
443
}
370
371
/* Convert a Dest array to the hacky Page and View keys that pdfwrite expects
372
 * dest_array: [<page_ref> <view_info>]
373
 * page_ref: indirect ref to a page dict
374
 * view_info: see table 8.2 in PDF 1.7, for example "/Fit"
375
 *
376
 * Removes /Dest and inserts two key pairs: /Page N and /View <view_info>
377
 * N is the page number, which starts at 1, not 0.
378
 */
379
static int pdfi_pdfmark_add_Page_View(pdf_context *ctx, pdf_dict *link_dict, pdf_array *dest_array)
380
6.02k
{
381
6.02k
    int code = 0;
382
6.02k
    int i;
383
6.02k
    uint64_t page_num;
384
6.02k
    pdf_dict *page_dict = NULL;
385
6.02k
    pdf_array *view_array = NULL;
386
6.02k
    uint64_t array_size;
387
6.02k
    pdf_obj *temp_obj = NULL;
388
389
    /* Get the page_dict, without storing the deref in the array.
390
     * (This is needed because otherwise there could be a circular ref that will
391
     * lead to a memory leak)
392
     */
393
6.02k
    code = pdfi_array_get_no_store_R(ctx, dest_array, 0, (pdf_obj **)&page_dict);
394
6.02k
    if (code < 0) goto exit;
395
396
5.54k
    if(pdfi_type_of(page_dict) == PDF_INT) {
397
11
        page_num = ((pdf_num *)page_dict)->value.i;
398
5.53k
    } else {
399
5.53k
        if (pdfi_type_of(page_dict) != PDF_DICT) {
400
27
            if (pdfi_type_of(page_dict) != PDF_NULL) {
401
1
                code = gs_note_error(gs_error_typecheck);
402
1
                goto exit;
403
1
            }
404
26
            page_num = 0;
405
5.50k
        } else {
406
            /* Find out which page number this is */
407
5.50k
            code = pdfi_page_get_number(ctx, page_dict, &page_num);
408
5.50k
            if (code < 0) goto exit;
409
5.50k
        }
410
5.53k
    }
411
412
5.51k
    page_num += ctx->Pdfmark_InitialPage;
413
414
5.51k
    if (pdfi_type_of(page_dict) != PDF_NULL)
415
        /* Add /Page key to the link_dict
416
         * Of course pdfwrite is numbering its pages starting at 1, because... of course :(
417
         */
418
5.48k
        code = pdfi_dict_put_int(ctx, link_dict, "Page", page_num+1);
419
26
    else
420
26
        code = pdfi_dict_put_int(ctx, link_dict, "Page", 0);
421
422
5.51k
    if (code < 0)
423
0
        goto exit;
424
425
    /* Get and check the destination type, again without storing the deref in the array.
426
     * In this case a memory leak  can only happen if the destination
427
     * is an invalid type, but lets take no chances.
428
     */
429
5.51k
    code = pdfi_array_get_no_store_R(ctx, dest_array, 1, &temp_obj);
430
5.51k
    if (code < 0) goto exit;
431
432
5.51k
    if (pdfi_type_of(temp_obj) != PDF_NAME) {
433
1
        pdfi_countdown(temp_obj);
434
1
        temp_obj = NULL;
435
1
        code = gs_note_error(gs_error_typecheck);
436
1
        goto exit;
437
1
    }
438
5.51k
    pdfi_countdown(temp_obj);
439
5.51k
    temp_obj = NULL;
440
441
    /* Build an array for /View, out of the remainder of the Dest entry */
442
5.51k
    array_size = pdfi_array_size(dest_array) - 1;
443
5.51k
    code = pdfi_array_alloc(ctx, array_size, &view_array);
444
5.51k
    if (code < 0) goto exit;
445
5.51k
    pdfi_countup(view_array);
446
17.7k
    for (i=0; i<array_size; i++) {
447
12.2k
        code = pdfi_array_get(ctx, dest_array, i+1, &temp_obj);
448
12.2k
        if (code < 0) goto exit;
449
12.2k
        code = pdfi_array_put(ctx, view_array, i, temp_obj);
450
12.2k
        if (code < 0) goto exit;
451
452
12.2k
        pdfi_countdown(temp_obj);
453
12.2k
        temp_obj = NULL;
454
12.2k
    }
455
    /* Add /View key to the link_dict */
456
5.51k
    code = pdfi_dict_put(ctx, link_dict, "View", (pdf_obj *)view_array);
457
5.51k
    if (code < 0) goto exit;
458
459
6.02k
 exit:
460
6.02k
    pdfi_countdown(temp_obj);
461
6.02k
    pdfi_countdown(page_dict);
462
6.02k
    pdfi_countdown(view_array);
463
6.02k
    return code;
464
5.51k
}
465
466
/* Lookup a Dest string(or name) in the Names array and try to resolve it */
467
static int pdfi_pdfmark_handle_dest_names(pdf_context *ctx, pdf_dict *link_dict,
468
                                         pdf_obj *dest, pdf_array *Names)
469
0
{
470
0
    int code = 0;
471
0
    int i;
472
0
    uint64_t array_size;
473
0
    pdf_obj *name = NULL;
474
0
    pdf_dict *D_dict = NULL;
475
0
    bool found = false;
476
0
    pdf_array *dest_array = NULL;
477
478
0
    array_size = pdfi_array_size(Names);
479
480
    /* Needs to be an even number - pairs of [name, obj] */
481
0
    if (array_size % 2 != 0) {
482
        /* TODO: flag an error? */
483
        /* Let's just ignore the last unpaired item for now */
484
0
        array_size -= 1;
485
0
    }
486
487
0
    for (i=0; i<array_size; i+=2) {
488
0
        code = pdfi_array_get(ctx, Names, i, (pdf_obj **)&name);
489
0
        if (code < 0) goto exit;
490
        /* Note: in current implementation, PDF_STRING and PDF_NAME have all the same
491
         * fields, but just in case that changes I treat them separately here.
492
         */
493
0
        if (pdfi_type_of(name) == PDF_STRING && pdfi_type_of(dest) == PDF_STRING) {
494
0
            if (!pdfi_string_cmp((pdf_string *)name, (pdf_string *)dest)) {
495
0
                found = true;
496
0
                break;
497
0
            }
498
0
        } else if (pdfi_type_of(name) == PDF_NAME && pdfi_type_of(dest) == PDF_NAME) {
499
0
            if (!pdfi_name_cmp((pdf_name *)name, (pdf_name *)dest)) {
500
0
                found = true;
501
0
                break;
502
0
            }
503
0
        }
504
0
        pdfi_countdown(name);
505
0
        name = NULL;
506
0
    }
507
508
0
    if (!found) {
509
        /* TODO: flag a warning? */
510
0
        code = 0;
511
0
        goto exit;
512
0
    }
513
514
    /* Next entry is either a dictionary (with a /D key) or an array */
515
0
    code = pdfi_array_get(ctx, Names, i+1, (pdf_obj **)&D_dict);
516
0
    if (code < 0) goto exit;
517
518
0
    switch (pdfi_type_of(D_dict)) {
519
0
        case PDF_DICT:
520
            /* Dict is supposed to contain key "D" with Dest array */
521
0
            code = pdfi_dict_knownget_type(ctx, D_dict, "D", PDF_ARRAY, (pdf_obj **)&dest_array);
522
0
            if (code <= 0) goto exit;
523
0
            break;
524
0
        case PDF_ARRAY:
525
0
            dest_array = (pdf_array *)D_dict;
526
0
            D_dict = NULL;
527
0
            break;
528
0
        default:
529
0
            code = gs_note_error(gs_error_typecheck);
530
0
            goto exit;
531
0
    }
532
533
    /* Process the dest_array to replace with /Page /View */
534
0
    code = pdfi_pdfmark_add_Page_View(ctx, link_dict, dest_array);
535
0
    if (code < 0) goto exit;
536
537
0
 exit:
538
0
    pdfi_countdown(name);
539
0
    pdfi_countdown(D_dict);
540
0
    pdfi_countdown(dest_array);
541
0
    return code;
542
0
}
543
544
/* Special handling for "Dest" in Links
545
 * Will replace /Dest with /Page /View in link_dict (for pdfwrite)
546
 */
547
int pdfi_pdfmark_modDest(pdf_context *ctx, pdf_dict *link_dict)
548
773
{
549
773
    int code = 0, code1 = 0;
550
773
    pdf_dict *Dests = NULL;
551
773
    pdf_obj *Dest = NULL;
552
773
    bool delete_Dest = true;
553
773
    pdf_array *dest_array = NULL;
554
773
    pdf_array *Names = NULL;
555
773
    pdf_dict *Names_dict = NULL;
556
557
773
    code = pdfi_dict_get(ctx, link_dict, "Dest", (pdf_obj **)&Dest);
558
773
    if (code < 0) goto exit;
559
560
773
    code = pdfi_dict_knownget_type(ctx, ctx->Root, "Dests", PDF_DICT, (pdf_obj **)&Dests);
561
773
    if (code < 0) goto exit;
562
563
772
    code = pdfi_dict_knownget_type(ctx, ctx->Root, "Names", PDF_DICT, (pdf_obj **)&Names_dict);
564
772
    if (code < 0) goto exit;
565
566
736
    switch (pdfi_type_of(Dest)) {
567
703
    case PDF_ARRAY:
568
703
        code = pdfi_pdfmark_add_Page_View(ctx, link_dict, (pdf_array *)Dest);
569
703
        if (code < 0) goto exit;
570
636
        break;
571
636
    case PDF_NAME:
572
31
        if (Dests != NULL) {
573
            /* Case where it's a name to look up in Contents(Root) /Dests */
574
31
            code = pdfi_dict_get_by_key(ctx, Dests, (const pdf_name *)Dest, (pdf_obj **)&dest_array);
575
31
            if (code == gs_error_undefined) {
576
                /* TODO: Not found, should flag a warning */
577
27
                code = 0;
578
27
                goto exit;
579
27
            }
580
4
            if (code < 0) goto exit;
581
4
            if (pdfi_type_of(dest_array) != PDF_ARRAY) {
582
0
                code = gs_note_error(gs_error_typecheck);
583
0
                goto exit;
584
0
            }
585
4
            code = pdfi_pdfmark_add_Page_View(ctx, link_dict, dest_array);
586
4
            if (code < 0) goto exit;
587
4
            break;
588
4
        }
589
        /* fallthrough */
590
2
    case PDF_STRING:
591
2
        if (Names_dict != NULL) {
592
            /* Looking in Catalog(Root) for /Names<</Dests<</Names [name dict array]>>>> */
593
2
            code = pdfi_dict_knownget_type(ctx, Names_dict, "Dests", PDF_DICT, (pdf_obj **)&Dests);
594
2
            if (code < 0) goto exit;
595
2
            if (code == 0) {
596
                /* TODO: Not found -- not sure if there is another case here or not */
597
0
                code = gs_note_error(gs_error_undefined);
598
0
                goto exit;
599
0
            }
600
601
2
            code = pdfi_dict_knownget_type(ctx, Dests, "Names", PDF_ARRAY, (pdf_obj **)&Names);
602
2
            if (code < 0) goto exit;
603
2
            if (code == 0) {
604
2
                code = pdfi_get_named_dest(ctx, Dest, (pdf_obj **)&dest_array);
605
2
                if (code < 0)
606
0
                    goto exit;
607
2
                if (pdfi_type_of(dest_array) == PDF_DICT) {
608
0
                    pdf_obj *D = NULL;
609
                    /* Dict is supposed to contain key "D" with Dest array */
610
0
                    code = pdfi_dict_knownget_type(ctx, (pdf_dict *)dest_array, "D", PDF_ARRAY, &D);
611
0
                    if (code <= 0) goto exit;
612
613
0
                    pdfi_countdown(dest_array);
614
0
                    dest_array = (pdf_array *)D;
615
0
                }
616
2
                if (pdfi_type_of(dest_array) == PDF_ARRAY) {
617
2
                    code = pdfi_pdfmark_add_Page_View(ctx, link_dict, dest_array);
618
2
                    if (code < 0)
619
0
                        goto exit;
620
2
                } else {
621
0
                    if ((code = pdfi_set_error_stop(ctx, gs_note_error(gs_error_undefined), NULL, E_PDF_BAD_NAMED_DEST, "pdfi_pdfmark_modDest", NULL)) < 0)
622
0
                        goto exit;
623
0
                }
624
2
                break;
625
2
            } else
626
0
                code = pdfi_pdfmark_handle_dest_names(ctx, link_dict, Dest, Names);
627
0
            if (code < 0) goto exit;
628
0
        } else {
629
            /* TODO: Ignore it -- flag a warning? */
630
0
        }
631
0
        break;
632
0
    default:
633
        /* TODO: Ignore it -- flag a warning? */
634
0
        break;
635
736
    }
636
637
773
 exit:
638
773
    if (delete_Dest) {
639
        /* Delete the Dest key */
640
773
        code1 = pdfi_dict_delete(ctx, link_dict, "Dest");
641
773
        if (code1 < 0 && code >= 0)
642
0
            code = code1;
643
773
    }
644
773
    pdfi_countdown(Dest);
645
773
    pdfi_countdown(Dests);
646
773
    pdfi_countdown(Names);
647
773
    pdfi_countdown(Names_dict);
648
773
    pdfi_countdown(dest_array);
649
773
    return code;
650
736
}
651
652
static int pdfi_check_limits(pdf_context *ctx, pdf_dict *node, char *str, int len)
653
123
{
654
123
    int code = 0, min, i, len2 = 0;
655
123
    pdf_array *Limits = NULL;
656
123
    pdf_string *Str = NULL;
657
123
    char *str2 = NULL;
658
659
123
    code = pdfi_dict_get_type(ctx, node, "Limits", PDF_ARRAY, (pdf_obj **)&Limits);
660
123
    if (code < 0)
661
0
        goto error;
662
663
123
    if (pdfi_array_size(Limits) != 2) {
664
        /* Limits are not valid, just ignore them. The calling code will then check
665
         * the Names array.
666
         */
667
0
        pdfi_set_warning(ctx, 0, NULL, W_PDF_BAD_TREE_LIMITS, "pdfi_check_limits", 0);
668
0
        goto error;
669
0
    }
670
671
123
    code = pdfi_array_get_type(ctx, Limits, 0, PDF_STRING, (pdf_obj **)&Str);
672
123
    if (code < 0)
673
0
        goto error;
674
675
123
    if (pdfi_type_of(Str) == PDF_NAME) {
676
0
        code = pdfi_string_from_name(ctx, (pdf_name *)Str, &str2, &len2);
677
0
        if (code < 0)
678
0
            return code;
679
123
    } else {
680
123
        len2 = ((pdf_string *)Str)->length;
681
123
        str2 = (char *)gs_alloc_bytes(ctx->memory, len2 + 1, "pdfi_check_limits");
682
123
        if (str2 == NULL) {
683
0
            code = gs_note_error(gs_error_VMerror);
684
0
            goto error;
685
0
        }
686
123
         memcpy(str2, ((pdf_string *)Str)->data, len2);
687
123
         str2[len2] = 0;
688
123
    }
689
690
123
    pdfi_countdown(Str);
691
123
    Str = NULL;
692
693
123
    min = len;
694
123
    if (len2 < min)
695
90
        min = len2;
696
697
427
    for (i=0;i< min;i++) {
698
416
        if (str[i] < str2[i]) {
699
0
            code = gs_note_error(gs_error_undefined);
700
0
            goto error;
701
0
        }
702
416
        if (str[i] != str2[i])
703
112
            break;
704
416
    }
705
123
    if (i > min && len2 < Str->length) {
706
0
        code = gs_note_error(gs_error_undefined);
707
0
        goto error;
708
0
    }
709
123
    gs_free_object(ctx->memory, str2, "pdfi_check_limits");
710
123
    str2 = NULL;
711
712
123
    code = pdfi_array_get_type(ctx, Limits, 1, PDF_STRING, (pdf_obj **)&Str);
713
123
    if (code < 0)
714
0
        goto error;
715
716
123
    if (pdfi_type_of(Str) == PDF_NAME) {
717
0
        code = pdfi_string_from_name(ctx, (pdf_name *)Str, &str2, &len2);
718
0
        if (code < 0)
719
0
            return code;
720
123
    } else {
721
123
        len2 = ((pdf_string *)Str)->length;
722
123
        str2 = (char *)gs_alloc_bytes(ctx->memory, len2 + 1, "pdfi_check_limits");
723
123
        if (str2 == NULL) {
724
0
            code = gs_note_error(gs_error_VMerror);
725
0
            goto error;
726
0
        }
727
123
         memcpy(str2, ((pdf_string *)Str)->data, len2);
728
123
         str2[len2] = 0;
729
123
    }
730
731
123
    pdfi_countdown(Str);
732
123
    Str = NULL;
733
734
123
    min = len;
735
123
    if (len2 < min)
736
76
        min = len2;
737
738
592
    for (i=0;i< min;i++) {
739
582
        if (str[i] > str2[i]) {
740
87
            code = gs_note_error(gs_error_undefined);
741
87
            goto error;
742
87
        }
743
495
        if (str[i] != str2[i])
744
26
            break;
745
495
    }
746
747
36
    if (i > min && len > len2)
748
0
        code = gs_note_error(gs_error_undefined);
749
750
123
error:
751
123
    gs_free_object(ctx->memory, str2, "pdfi_check_limits");
752
123
    pdfi_countdown(Str);
753
123
    pdfi_countdown(Limits);
754
123
    return code;
755
36
}
756
757
static int pdfi_get_name_from_node(pdf_context *ctx, pdf_dict *node, char *str, int len, pdf_obj **Name, bool is_root)
758
161
{
759
161
    int i = 0, code = 0;
760
161
    pdf_string *StrKey = NULL;
761
161
    pdf_array *NamesArray = NULL;
762
161
    pdf_dict *Kid = NULL;
763
161
    bool known;
764
765
161
    code = pdfi_loop_detector_mark(ctx);
766
161
    if (code < 0)
767
0
        return code;
768
769
161
    code = pdfi_dict_known(ctx, node, "Names", &known);
770
161
    if (code < 0)
771
0
        goto error;
772
773
161
    if (known) {
774
123
        code = pdfi_dict_known(ctx, node, "Limits", &known);
775
123
        if (code < 0)
776
0
            goto error;
777
778
123
        if (!known) {
779
0
            if (!is_root)
780
                /* No Limits array (a required entry, except in the Root), so just assume that the
781
                 * string is in this node and check all the Names anyway
782
                 */
783
0
                pdfi_set_warning(ctx, 0, NULL, W_PDF_NO_TREE_LIMITS, "pdfi_get_name_from_node", 0);
784
123
        } else {
785
123
            code = pdfi_check_limits(ctx, node, str, len);
786
123
            if (code < 0)
787
87
                goto error;
788
123
        }
789
790
36
        code = pdfi_dict_get_type(ctx, node, "Names", PDF_ARRAY, (pdf_obj **)&NamesArray);
791
36
        if (code < 0)
792
5
            goto error;
793
794
31
        if (pdfi_array_size(NamesArray) & 1)
795
0
            pdfi_set_warning(ctx, 0, NULL, W_PDF_NAMES_ARRAY_SIZE, "pdfi_get_name_from_node", 0);
796
797
93
        for (i = 0;i < pdfi_array_size(NamesArray) / 2; i++) {
798
93
            code = pdfi_array_get_type(ctx, NamesArray, i * 2, PDF_STRING, (pdf_obj **)&StrKey);
799
93
            if (code < 0)
800
0
                goto error;
801
802
93
            if (StrKey->length == len && memcmp((const char *)StrKey->data, str, len) == 0) {
803
31
                code = pdfi_array_get(ctx, NamesArray, (i * 2) + 1, (pdf_obj **)Name);
804
31
                goto error;
805
31
            }
806
62
            pdfi_countdown(StrKey);
807
62
            StrKey = NULL;
808
62
        }
809
0
        pdfi_countdown(NamesArray);
810
0
        NamesArray = NULL;
811
0
    }
812
813
    /* Either no Names array (initial node) or not in array */
814
38
    code = pdfi_dict_get_type(ctx, node, "Kids", PDF_ARRAY, (pdf_obj **)&NamesArray);
815
38
    if (code < 0)
816
2
        goto error;
817
818
123
    for (i = 0;i < pdfi_array_size(NamesArray); i++) {
819
123
        code = pdfi_array_get_type(ctx, NamesArray, i, PDF_DICT, (pdf_obj **)&Kid);
820
123
        if (code < 0)
821
0
            goto error;
822
823
123
        code = pdfi_get_name_from_node(ctx, Kid, str, len, Name, false);
824
123
        pdfi_countdown(Kid);
825
123
        Kid = NULL;
826
123
        if (code == 0)
827
31
            break;
828
829
92
        if (code < 0) {
830
92
            if (code == gs_error_undefined)
831
87
                continue;
832
5
            goto error;
833
92
        }
834
92
    }
835
836
161
error:
837
161
    pdfi_countdown(Kid);
838
161
    pdfi_countdown(StrKey);
839
161
    pdfi_countdown(NamesArray);
840
161
    (void)pdfi_loop_detector_cleartomark(ctx);
841
161
    return code;
842
36
}
843
844
static int pdfi_get_named_dest(pdf_context *ctx, pdf_obj *Named, pdf_obj **Dest)
845
185
{
846
185
    int code = 0, len = 0;
847
185
    pdf_dict *Names = NULL, *Dests = NULL;
848
185
    char *str = NULL;
849
850
185
    code = pdfi_loop_detector_mark(ctx);
851
185
    if (code < 0)
852
0
        return code;
853
854
185
    code = pdfi_dict_get_type(ctx, ctx->Root, "Names", PDF_DICT, (pdf_obj **)&Names);
855
185
    if (code < 0)
856
147
        goto error;
857
858
38
    code = pdfi_dict_get_type(ctx, Names, "Dests", PDF_DICT, (pdf_obj **)&Dests);
859
38
    if (code < 0)
860
0
        goto error;
861
862
38
    if (pdfi_type_of(Named) == PDF_NAME) {
863
0
        code = pdfi_string_from_name(ctx, (pdf_name *)Named, &str, &len);
864
0
        if (code < 0)
865
0
            return code;
866
38
    } else {
867
38
        len = ((pdf_string *)Named)->length;
868
38
        str = (char *)gs_alloc_bytes(ctx->memory, len + 1, "pdfi_get_named_dest");
869
38
        if (str == NULL) {
870
0
            code = gs_note_error(gs_error_VMerror);
871
0
            goto error;
872
0
        }
873
38
        memcpy(str, ((pdf_string *)Named)->data, len);
874
38
        str[len] = 0;
875
38
    }
876
877
38
    code = pdfi_get_name_from_node(ctx, Dests, str, len, Dest, true);
878
879
185
error:
880
185
    if (pdfi_type_of(Named) == PDF_NAME)
881
0
        (void)pdfi_free_string_from_name(ctx, str);
882
185
    else
883
185
        gs_free_object(ctx->memory, str, "pdfi_get_named_dest");
884
185
    pdfi_countdown(Names);
885
185
    pdfi_countdown(Dests);
886
185
    pdfi_loop_detector_cleartomark(ctx);
887
185
    return code;
888
38
}
889
890
/* Special handling for "A" in Link annotations and Outlines
891
 * Will delete A if handled and if A_key is provided.
892
 */
893
int pdfi_pdfmark_modA(pdf_context *ctx, pdf_dict *dict)
894
7.01k
{
895
7.01k
    int code = 0;
896
7.01k
    pdf_dict *A_dict = NULL;
897
7.01k
    bool known;
898
7.01k
    pdf_name *S_name = NULL;
899
7.01k
    pdf_array *D_array = NULL;
900
7.01k
    bool delete_A = false;
901
7.01k
    bool deref_A = true;
902
903
7.01k
    code = pdfi_dict_get_no_store_R(ctx, dict, "A", (pdf_obj **)&A_dict);
904
7.01k
    if (code < 0) goto exit;
905
906
6.91k
    if (pdfi_type_of(A_dict) != PDF_DICT) {
907
        /* Invalid AP, just delete it because I dunno what to do...
908
         * TODO: Should flag a warning here
909
         */
910
276
        delete_A = true;
911
276
        goto exit;
912
276
    }
913
914
    /* Handle URI */
915
6.63k
    code = pdfi_dict_known(ctx, A_dict, "URI", &known);
916
6.63k
    if (code < 0) goto exit;
917
6.63k
    if (known) {
918
999
        code = pdfi_resolve_indirect_loop_detect(ctx, (pdf_obj *)NULL, (pdf_obj *)A_dict, true);
919
999
        goto exit;
920
999
    }
921
922
    /* Handle S = GoTo */
923
    /* TODO: this handles <</S /GoTo /D [dest array]>>
924
     * Not sure if there are other cases to handle?
925
     */
926
5.63k
    code = pdfi_dict_knownget_type(ctx, A_dict, "S", PDF_NAME, (pdf_obj **)&S_name);
927
5.63k
    if (code <= 0) goto exit;
928
    /* We only handle GoTo for now */
929
5.63k
    if (pdfi_name_is(S_name, "GoTo")) {
930
5.46k
        code = pdfi_dict_knownget(ctx, A_dict, "D", (pdf_obj **)&D_array);
931
5.46k
        if (code <= 0)
932
2
            goto exit;
933
5.46k
        if (pdfi_type_of(D_array) == PDF_STRING || pdfi_type_of(D_array) == PDF_NAME)
934
183
        {
935
183
            pdf_obj *Dest = NULL;
936
937
183
            code = pdfi_get_named_dest(ctx, (pdf_obj *)D_array, &Dest);
938
183
            if (code < 0)
939
154
                goto exit;
940
29
            pdfi_countdown(D_array);
941
29
            D_array = NULL;
942
29
            if (pdfi_type_of(Dest) != PDF_ARRAY) {
943
29
                if (pdfi_type_of(Dest) != PDF_DICT) {
944
0
                    pdfi_countdown(Dest);
945
0
                    code = gs_note_error(gs_error_typecheck);
946
0
                    goto exit;
947
0
                }
948
29
                code = pdfi_dict_knownget(ctx, (pdf_dict *)Dest, "D", (pdf_obj **)&D_array);
949
29
                pdfi_countdown(Dest);
950
29
                if (code <= 0)
951
0
                    goto exit;
952
29
            } else
953
0
                D_array = (pdf_array *)Dest;
954
29
        }
955
5.31k
        if (pdfi_type_of(D_array) != PDF_ARRAY) {
956
0
            code = gs_note_error(gs_error_typecheck);
957
0
            goto exit;
958
0
        }
959
960
        /* Process the D array to replace with /Page /View */
961
5.31k
        code = pdfi_pdfmark_add_Page_View(ctx, dict, D_array);
962
5.31k
        if (code < 0) goto exit;
963
4.87k
        delete_A = true;
964
4.87k
    } else if (pdfi_name_is(S_name, "GoToR") || pdfi_name_is(S_name, "Launch")) {
965
        /* These point out of the document.
966
         * Flatten out the reference, but otherwise leave it alone.
967
         * gs does some wack stuff here.
968
         *
969
         * Currently this is same behavior as gs, but it is not correct behavior.
970
         * In at least some cases we could do better, for example if the doc
971
         * pointed to happens to be the same file.
972
         * Sample: fts_28_2808.pdf
973
         * Sample: ~/work/samples/tests_private/pdf/sumatra/1874_-_clicking_ToC_link_crashes.pdf
974
         */
975
2
        code = pdfi_resolve_indirect_loop_detect(ctx, (pdf_obj *)dict, (pdf_obj *)A_dict, true);
976
2
        delete_A = false;
977
2
        code = 0;
978
162
    } else if (pdfi_name_is(S_name, "Named")) {
979
        /* We can just pass this through and it will work fine
980
         * This should be a name like "FirstPage" or "LastPage".
981
         * Note: gs implementation translates into page numbers and also has some bugs...
982
         * Sample: fts_33_3310.pdf
983
         */
984
11
        delete_A = false;
985
11
        code = 0;
986
151
    } else if (pdfi_name_is(S_name, "GoToE")) {
987
        /* TODO: ??
988
         * File: fts_33_3303.pdf
989
         */
990
85
    } else if (pdfi_name_is(S_name, "Thread")) {
991
        /* TODO: For basically all of these below, I think just need to preserve
992
         * any references and streams and pass it all through.
993
         * File: fts_33_3305.pdf fts_33_3317.pdf
994
         */
995
0
        deref_A = false;
996
85
    } else if (pdfi_name_is(S_name, "Sound")) {
997
        /* TODO: ??
998
         * File: fts_33_3307.pdf
999
         */
1000
0
        deref_A = false;
1001
85
    } else if (pdfi_name_is(S_name, "Movie")) {
1002
        /* TODO: ??
1003
         * File: fts_33_3308.pdf
1004
         */
1005
2
        deref_A = false;
1006
83
    } else if (pdfi_name_is(S_name, "GoTo3DView")) {
1007
        /* TODO: ??
1008
         * File: fts_33_3318.pdf
1009
         */
1010
83
    } else if (pdfi_name_is(S_name, "RichMediaExecute")) {
1011
        /* TODO: ??
1012
         * File: fts_33_3319.pdf
1013
         */
1014
83
    } else if (pdfi_name_is(S_name, "Rendition")) {
1015
        /* TODO: make sure to pass through accurately?
1016
         * File: fts_07_0709.pdf fts_33_3316.pdf
1017
         */
1018
83
    } else {
1019
        /* TODO: flag warning? */
1020
83
    }
1021
1022
7.01k
 exit:
1023
7.01k
    if (delete_A) {
1024
5.14k
        code = pdfi_dict_delete(ctx, dict, "A");
1025
5.14k
    } else if (deref_A) {
1026
1.86k
        pdfi_countdown(A_dict);
1027
1.86k
        A_dict = NULL;
1028
1.86k
    }
1029
7.01k
    pdfi_countdown(A_dict);
1030
7.01k
    pdfi_countdown(S_name);
1031
7.01k
    pdfi_countdown(D_array);
1032
7.01k
    return code;
1033
5.63k
}
1034
1035
/* Begin defining an object
1036
 * Send an OBJ (_objdef) command
1037
 * (_objdef) (<label>) (/type) (/<type>) OBJ
1038
 */
1039
static int pdfi_pdfmark_objdef_begin(pdf_context *ctx, pdf_indirect_ref *label, const char *type)
1040
4.26k
{
1041
4.26k
    int code;
1042
4.26k
    pdf_obj *objarray[4];
1043
4.26k
    int num_objects = 4;
1044
4.26k
    int i;
1045
1046
4.26k
    memset(objarray, 0, sizeof(objarray));
1047
1048
4.26k
    code = pdfi_obj_charstr_to_name(ctx, "_objdef", (pdf_name **)&objarray[0]);
1049
4.26k
    if (code < 0) goto exit;
1050
1051
4.26k
    objarray[1] = (pdf_obj *)label;
1052
4.26k
    pdfi_countup(label);
1053
1054
4.26k
    code = pdfi_obj_charstr_to_name(ctx, "type", (pdf_name **)&objarray[2]);
1055
4.26k
    if (code < 0) goto exit;
1056
1057
4.26k
    code = pdfi_obj_charstr_to_name(ctx, type, (pdf_name **)&objarray[3]);
1058
4.26k
    if (code < 0) goto exit;
1059
1060
4.26k
    code = pdfi_pdfmark_from_objarray(ctx, objarray, num_objects, NULL, "OBJ");
1061
4.26k
    if (code < 0) goto exit;
1062
1063
4.26k
 exit:
1064
21.3k
    for (i=0; i<num_objects; i++)
1065
17.0k
        pdfi_countdown(objarray[i]);
1066
4.26k
    return code;
1067
4.26k
}
1068
1069
/* Close an object
1070
 * Send a CLOSE command
1071
 * (<label>) CLOSE
1072
 */
1073
static int pdfi_pdfmark_objdef_close(pdf_context *ctx, pdf_indirect_ref *label)
1074
446
{
1075
446
    int code;
1076
446
    pdf_obj *objarray[1];
1077
446
    int num_objects = 1;
1078
446
    int i;
1079
1080
446
    memset(objarray, 0, sizeof(objarray));
1081
1082
446
    objarray[0] = (pdf_obj *)label;
1083
446
    pdfi_countup(label);
1084
1085
446
    code = pdfi_pdfmark_from_objarray(ctx, objarray, num_objects, NULL, "CLOSE");
1086
446
    if (code < 0) goto exit;
1087
1088
446
 exit:
1089
892
    for (i=0; i<num_objects; i++)
1090
446
        pdfi_countdown(objarray[i]);
1091
446
    return code;
1092
446
}
1093
1094
static int pdfi_pdfmark_stream_contents(pdf_context *ctx, pdf_indirect_ref *label, pdf_stream *stream)
1095
446
{
1096
446
    int code;
1097
446
    pdf_obj *objarray[2];
1098
446
    int num_objects = 2;
1099
446
    int i;
1100
1101
446
    objarray[0] = (pdf_obj *)label;
1102
446
    pdfi_countup(label);
1103
1104
446
    objarray[1] = (pdf_obj *)stream;
1105
446
    pdfi_countup(stream);
1106
446
    stream->is_marking = true;
1107
1108
446
    code = pdfi_pdfmark_from_objarray(ctx, objarray, num_objects, NULL, ".PUTSTREAM");
1109
446
    if (code < 0) goto exit;
1110
1111
446
 exit:
1112
446
    stream->is_marking = false;
1113
1.33k
    for (i=0; i<num_objects; i++)
1114
892
        pdfi_countdown(objarray[i]);
1115
446
    return code;
1116
446
}
1117
1118
/* Mark a stream object */
1119
int pdfi_pdfmark_stream(pdf_context *ctx, pdf_stream *stream)
1120
478
{
1121
478
    int code;
1122
478
    pdf_dict *streamdict = NULL;
1123
478
    pdf_indirect_ref *streamref = NULL;
1124
478
    pdf_dict *tempdict = NULL;
1125
478
    uint64_t dictsize;
1126
478
    uint64_t index;
1127
478
    pdf_name *Key = NULL;
1128
1129
478
    if (stream->stream_written)
1130
28
        return 0;
1131
1132
450
    stream->stream_written = true;
1133
1134
450
    if (!ctx->device_state.writepdfmarks)
1135
0
        return 0;
1136
1137
    /* Create an indirect ref for the stream */
1138
450
    code = pdfi_object_alloc(ctx, PDF_INDIRECT, 0, (pdf_obj **)&streamref);
1139
450
    if (code < 0) goto exit;
1140
450
    pdfi_countup(streamref);
1141
450
    streamref->ref_object_num = stream->object_num;
1142
450
    streamref->ref_generation_num = stream->generation_num;
1143
450
    streamref->is_marking = true;
1144
1145
450
    code = pdfi_dict_from_obj(ctx, (pdf_obj *)stream, &streamdict);
1146
450
    if (code < 0) goto exit;
1147
1148
    /* Make a copy of the dict and remove Filter keyword */
1149
450
    dictsize = pdfi_dict_entries(streamdict);
1150
450
    code = pdfi_dict_alloc(ctx, dictsize, &tempdict);
1151
450
    if (code < 0) goto exit;
1152
450
    pdfi_countup(tempdict);
1153
450
    code = pdfi_dict_copy(ctx, tempdict, streamdict);
1154
450
    if (code < 0) goto exit;
1155
450
    code = pdfi_dict_key_first(ctx, streamdict, (pdf_obj **)&Key, &index);
1156
1.67k
    while (code >= 0) {
1157
1.67k
        if (pdfi_name_is(Key, "Filter") || pdfi_name_is(Key, "Length")) {
1158
630
            code = pdfi_dict_delete_pair(ctx, tempdict, Key);
1159
630
            if (code < 0) goto exit;
1160
630
        }
1161
1.67k
        pdfi_countdown(Key);
1162
1.67k
        Key = NULL;
1163
1164
1.67k
        code = pdfi_dict_key_next(ctx, streamdict, (pdf_obj **)&Key, &index);
1165
1.67k
        if (code == gs_error_undefined) {
1166
449
            code = 0;
1167
449
            break;
1168
449
        }
1169
1.67k
    }
1170
450
    if (code < 0) goto exit;
1171
1172
449
    code = pdfi_pdfmark_objdef_begin(ctx, streamref, "stream");
1173
449
    if (code < 0) goto exit;
1174
1175
449
    code = pdfi_pdfmark_from_dict_withlabel(ctx, streamref, tempdict, NULL, ".PUTDICT");
1176
449
    if (code < 0) goto exit;
1177
1178
446
    code = pdfi_pdfmark_stream_contents(ctx, streamref, stream);
1179
446
    if (code < 0) goto exit;
1180
1181
446
    code = pdfi_pdfmark_objdef_close(ctx, streamref);
1182
446
    if (code < 0) goto exit;
1183
1184
450
 exit:
1185
450
    pdfi_countdown(tempdict);
1186
450
    pdfi_countdown(streamref);
1187
450
    return code;
1188
446
}
1189
1190
/* Mark a dict object */
1191
int pdfi_pdfmark_dict(pdf_context *ctx, pdf_dict *dict)
1192
23.5k
{
1193
23.5k
    int code;
1194
23.5k
    pdf_indirect_ref *dictref = NULL;
1195
1196
23.5k
    if (dict->dict_written)
1197
19.6k
        return 0;
1198
1199
3.81k
    dict->dict_written = true;
1200
1201
3.81k
    if (!ctx->device_state.writepdfmarks)
1202
0
        return 0;
1203
1204
    /* Create an indirect ref for the dict */
1205
3.81k
    code = pdfi_object_alloc(ctx, PDF_INDIRECT, 0, (pdf_obj **)&dictref);
1206
3.81k
    if (code < 0) goto exit;
1207
3.81k
    pdfi_countup(dictref);
1208
3.81k
    dictref->ref_object_num = dict->object_num;
1209
3.81k
    dictref->ref_generation_num = dict->generation_num;
1210
3.81k
    dictref->is_marking = true;
1211
1212
3.81k
    code = pdfi_pdfmark_objdef_begin(ctx, dictref, "dict");
1213
3.81k
    if (code < 0) goto exit;
1214
1215
3.81k
    code = pdfi_pdfmark_from_dict_withlabel(ctx, dictref, dict, NULL, ".PUTDICT");
1216
3.81k
    if (code < 0) goto exit;
1217
1218
3.81k
 exit:
1219
3.81k
    pdfi_countdown(dictref);
1220
3.81k
    return code;
1221
3.81k
}
1222
1223
static int pdfi_pdfmark_filespec(pdf_context *ctx, pdf_string *name, pdf_dict *filespec)
1224
30
{
1225
30
    int code;
1226
30
    pdf_dict *tempdict = NULL;
1227
1228
30
    code = pdfi_dict_alloc(ctx, 40, &tempdict);
1229
30
    if (code < 0) goto exit;
1230
30
    pdfi_countup(tempdict);
1231
1232
30
    code = pdfi_dict_put(ctx, tempdict, "Name", (pdf_obj *)name);
1233
30
    if (code < 0) goto exit;
1234
1235
    /* Flatten the filespec */
1236
30
    code = pdfi_resolve_indirect(ctx, (pdf_obj *)filespec, true);
1237
30
    if (code < 0) goto exit;
1238
1239
24
    code = pdfi_dict_put(ctx, tempdict, "FS", (pdf_obj *)filespec);
1240
24
    if (code < 0) goto exit;
1241
1242
24
    code = pdfi_pdfmark_from_dict(ctx, tempdict, NULL, "EMBED");
1243
24
    if (code < 0) goto exit;
1244
1245
30
 exit:
1246
30
    pdfi_countdown(tempdict);
1247
30
    return code;
1248
24
}
1249
1250
/* embed a file */
1251
int pdfi_pdfmark_embed_filespec(pdf_context *ctx, pdf_string *name, pdf_dict *filespec)
1252
30
{
1253
30
    int code;
1254
1255
30
    code = pdfi_pdfmark_filespec(ctx, name, filespec);
1256
30
    if (code < 0) goto exit;
1257
1258
30
 exit:
1259
30
    return code;
1260
30
}
1261
1262
/*
1263
 * Create and emit a /DOCINFO pdfmark for any and all of Title,
1264
 * Author, Subject, Keywords and Creator
1265
 */
1266
void pdfi_pdfmark_write_docinfo(pdf_context *ctx, pdf_dict *info_dict)
1267
22.2k
{
1268
22.2k
    int i, code = 0;
1269
22.2k
    pdf_dict *Info = NULL;
1270
22.2k
    pdf_obj *o = NULL;
1271
    /* We don't preserve the Producer, we are the Producer */
1272
22.2k
    const char *KeyNames[] = {
1273
22.2k
        "Title", "Author", "Subject", "Keywords", "Creator"
1274
22.2k
    };
1275
1276
22.2k
    if (!ctx->device_state.writepdfmarks)
1277
19.9k
        return;
1278
1279
2.31k
    code = pdfi_dict_alloc(ctx, 5, &Info);
1280
2.31k
    if (code < 0)
1281
0
        goto exit;
1282
2.31k
    pdfi_countup(Info);
1283
1284
13.8k
    for (i=0;i<5;i++)
1285
11.5k
    {
1286
11.5k
        if (pdfi_dict_knownget(ctx, info_dict, KeyNames[i], &o) > 0)
1287
4.23k
        {
1288
4.23k
            (void)pdfi_dict_put(ctx, Info, KeyNames[i], (pdf_obj *)o);
1289
4.23k
            pdfi_countdown(o);
1290
4.23k
        }
1291
11.5k
    }
1292
1293
2.31k
    code = pdfi_pdfmark_from_dict(ctx, Info, NULL, "DOCINFO");
1294
2.31k
exit:
1295
2.31k
    pdfi_countdown(Info);
1296
2.31k
    return;
1297
2.31k
}
1298
1299
/*
1300
 * Create and emit a /PAGE pdfmark for any and all of
1301
 * CropBox, TrimBox, Artbox and BleedBox. If the interpreter
1302
 * has used something other than the MediaBox as the media size, then
1303
 * we don't send these, they are almost certainly incorrect.
1304
 *
1305
 * Because we will have used the MediaBox to create the media size, and
1306
 * will have accounted for any rotation or scaling, we can use the CTM
1307
 * to adjust the various Box entries (note this routine must be called
1308
 * early!).
1309
 */
1310
void pdfi_pdfmark_write_boxes(pdf_context *ctx, pdf_dict *page_dict)
1311
103k
{
1312
103k
    int i, code = 0;
1313
103k
    pdf_dict *BoxDict = NULL;
1314
103k
    pdf_obj *o = NULL;
1315
103k
    gx_device *device = gs_currentdevice(ctx->pgs);
1316
103k
    gs_matrix scale, m, ctm;
1317
103k
    const char *BoxNames[] = {
1318
103k
        "CropBox", "BleedBox", "TrimBox", "ArtBox"
1319
103k
    };
1320
1321
    /* If the device doesn't support pdfmark, or the device modifies the Page Size (making the other boxes unreliable) exit now */
1322
103k
    if (!ctx->device_state.writepdfmarks || ctx->device_state.ModifiesPageSize)
1323
91.3k
        return;
1324
1325
    /* If we are using somethign other than the MediaBox, don't send these */
1326
12.0k
    if (ctx->args.usecropbox || ctx->args.usebleedbox ||
1327
12.0k
        ctx->args.usetrimbox || ctx->args.useartbox)
1328
0
        return;
1329
1330
12.0k
    code = pdfi_dict_alloc(ctx, 4, &BoxDict);
1331
12.0k
    if (code < 0)
1332
0
        goto exit;
1333
1334
12.0k
    pdfi_countup(BoxDict);
1335
1336
    /* Undo the resolution scaling from the CTM, we don't want to apply that */
1337
12.0k
    scale.xx = 72.0 / device->HWResolution[0];
1338
12.0k
    scale.xy = 0;
1339
12.0k
    scale.yx = 0;
1340
12.0k
    scale.yy = 72.0 / device->HWResolution[1];
1341
12.0k
    scale.tx = 0;
1342
12.0k
    scale.ty = 0;
1343
1344
    /* And multiply that by the CTM to get a matrix which represents the
1345
     * scaling/rotation used to set the conetnt to the media.
1346
     */
1347
12.0k
    gs_currentmatrix(ctx->pgs, &ctm);
1348
12.0k
    code = gs_matrix_multiply(&ctm, &scale, &m);
1349
12.0k
    if (code < 0) goto exit;
1350
1351
60.3k
    for (i=0;i<4;i++)
1352
48.2k
    {
1353
        /* Check each Bos name in turn */
1354
48.2k
        if (pdfi_dict_knownget(ctx, page_dict, BoxNames[i], &o) > 0){
1355
7.28k
            gs_rect box;
1356
7.28k
            pdf_array *new_array = NULL;
1357
1358
            /* Box is present in page dicitonayr, check it's an array */
1359
7.28k
            if (pdfi_type_of(o) != PDF_ARRAY) {
1360
0
                pdfi_countdown(o);
1361
0
                continue;
1362
0
            }
1363
1364
            /* Turn the contents into a gs_rect */
1365
7.28k
            code = pdfi_array_to_gs_rect(ctx, (pdf_array *)o, &box);
1366
7.28k
            pdfi_countdown(o);
1367
7.28k
            if (code < 0)
1368
33
                continue;
1369
1370
            /* Rectangles in PDF need not be llx,lly,urx,ury, they can be any
1371
             * two opposite corners. Turn that into the usual format.
1372
             */
1373
7.25k
            pdfi_normalize_rect(ctx, &box);
1374
1375
            /* Transform the resulting box by the calculated matrix */
1376
7.25k
            pdfi_bbox_transform(ctx, &box, &m);
1377
1378
            /* Get a new array created from the box values */
1379
7.25k
            code = pdfi_gs_rect_to_array(ctx, &box, &new_array);
1380
7.25k
            if (code < 0)
1381
0
                continue;
1382
1383
            /* And store it in the working dictionary */
1384
7.25k
            (void)pdfi_dict_put(ctx, BoxDict, BoxNames[i], (pdf_obj *)new_array);
1385
7.25k
            pdfi_countdown(new_array);
1386
7.25k
        }
1387
48.2k
    }
1388
1389
    /* Send all the Box entries to the device */
1390
12.0k
    (void)pdfi_pdfmark_from_dict(ctx, BoxDict, NULL, "PAGE");
1391
1392
12.0k
exit:
1393
12.0k
    pdfi_countdown(BoxDict);
1394
12.0k
    return;
1395
12.0k
}