Coverage Report

Created: 2025-08-28 07:06

/src/ghostpdl/pdf/pdf_mark.c
Line
Count
Source (jump to first uncovered line)
1
/* Copyright (C) 2001-2025 Artifex Software, Inc.
2
   All Rights Reserved.
3
4
   This software is provided AS-IS with no warranty, either express or
5
   implied.
6
7
   This software is distributed under license and may not be copied,
8
   modified or distributed except as expressly authorized under the terms
9
   of the license contained in the file LICENSE in this distribution.
10
11
   Refer to licensing information at http://www.artifex.com or contact
12
   Artifex Software, Inc.,  39 Mesa Street, Suite 108A, San Francisco,
13
   CA 94129, USA, for further information.
14
*/
15
16
/* 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
976k
{
36
976k
    int code = 0;
37
976k
    byte *data = NULL;
38
976k
    int size = 0;
39
40
976k
    code = pdfi_obj_to_string(ctx, obj, &data, &size);
41
976k
    if (code < 0)
42
267
        return code;
43
976k
    entry->data = data;
44
976k
    entry->size = size;
45
976k
    entry->persistent = false;
46
976k
    return 0;
47
976k
}
48
49
static int pdfi_pdfmark_setparam_pair(pdf_context *ctx, pdf_name *Key, pdf_obj *Value,
50
                                   gs_param_string *entry)
51
432k
{
52
432k
    int code = 0;
53
54
    /* Handle the Key */
55
432k
    if (pdfi_type_of(Key) != PDF_NAME) {
56
0
        code = gs_note_error(gs_error_typecheck);
57
0
        goto exit;
58
0
    }
59
60
432k
    code = pdfi_pdfmark_setparam_obj(ctx, (pdf_obj *)Key, entry);
61
432k
    if (code < 0)
62
0
        goto exit;
63
64
432k
    code = pdfi_pdfmark_setparam_obj(ctx, Value, entry+1);
65
432k
    if (code < 0)
66
185
        goto exit;
67
68
432k
 exit:
69
432k
    return code;
70
432k
}
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
182k
{
76
182k
    int size = 100;
77
182k
    char *buf;
78
79
182k
    buf = (char *)gs_alloc_bytes(ctx->memory, size, "pdfi_pdfmark_ctm_str(data)");
80
182k
    if (buf == NULL)
81
0
        return_error(gs_error_VMerror);
82
182k
    snprintf(buf, size, "[%.4f %.4f %.4f %.4f %.4f %.4f]",
83
182k
             ctm->xx, ctm->xy, ctm->yx, ctm->yy, ctm->tx, ctm->ty);
84
182k
    *data = (byte *)buf;
85
182k
    *len = strlen(buf);
86
182k
    return 0;
87
182k
}
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
182k
{
92
182k
    gs_c_param_list list;
93
182k
    int code;
94
95
    /* Set the list to writeable, and initialise it */
96
182k
    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
182k
    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
182k
    gs_c_param_list_write_more(&list);
104
105
    /* Add the param string array to the list */
106
182k
    code = param_write_string_array((gs_param_list *)&list, command, array_list);
107
182k
    if (code < 0)
108
0
        return code;
109
110
    /* Set the param list back to readable, so putceviceparams can readit (mad...) */
111
182k
    gs_c_param_list_read(&list);
112
113
    /* and set the actual device parameters */
114
182k
    code = gs_putdeviceparams(ctx->pgs->device, (gs_param_list *)&list);
115
116
182k
    gs_c_param_list_release(&list);
117
118
182k
    return code;
119
182k
}
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
476
{
124
476
    gs_c_param_list list;
125
476
    int code;
126
127
    /* Set the list to writeable, and initialise it */
128
476
    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
476
    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
476
    gs_c_param_list_write_more(&list);
136
137
    /* Add the param string array to the list */
138
476
    code = param_write_string((gs_param_list *)&list, command, param_string);
139
476
    if (code < 0)
140
0
        return code;
141
142
    /* Set the param list back to readable, so putceviceparams can readit (mad...) */
143
476
    gs_c_param_list_read(&list);
144
145
    /* and set the actual device parameters */
146
476
    code = gs_putdeviceparams(ctx->pgs->device, (gs_param_list *)&list);
147
148
476
    gs_c_param_list_release(&list);
149
150
476
    return code;
151
476
}
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
79.4k
{
170
79.4k
    int code = 0;
171
79.4k
    int size;
172
79.4k
    uint64_t dictsize;
173
79.4k
    uint64_t index;
174
79.4k
    uint64_t keynum = 0;
175
79.4k
    int i;
176
79.4k
    pdf_name *Key = NULL;
177
79.4k
    pdf_obj *Value = NULL;
178
79.4k
    pdf_obj *tempobj = NULL;
179
79.4k
    gs_param_string *parray = NULL;
180
79.4k
    gs_param_string_array array_list;
181
79.4k
    byte *ctm_data = NULL;
182
79.4k
    int ctm_len;
183
79.4k
    gs_matrix ctm_placeholder;
184
79.4k
    int offset = 0;
185
186
    /* If ctm not provided, make a placeholder */
187
79.4k
    if (!ctm) {
188
27.9k
        gs_currentmatrix(ctx->pgs, &ctm_placeholder);
189
27.9k
        ctm = &ctm_placeholder;
190
27.9k
    }
191
192
79.4k
    dictsize = pdfi_dict_entries(dict);
193
79.4k
    size = dictsize*2 + 2; /* pairs + CTM + type */
194
79.4k
    if (label)
195
4.86k
        size += 1;
196
197
79.4k
    parray = (gs_param_string *)gs_alloc_bytes(ctx->memory, (size_t)size*sizeof(gs_param_string),
198
79.4k
                                               "pdfi_pdfmark_from_dict(parray)");
199
79.4k
    if (parray == NULL) {
200
0
        code = gs_note_error(gs_error_VMerror);
201
0
        goto exit;
202
0
    }
203
79.4k
    memset(parray, 0, size*sizeof(gs_param_string));
204
205
79.4k
    if (label) {
206
4.86k
        code = pdfi_pdfmark_setparam_obj(ctx, (pdf_obj *)label, parray+0);
207
4.86k
        offset += 1;
208
4.86k
    }
209
210
    /* Get each (key,val) pair from dict and setup param for it */
211
79.4k
    if (dictsize > 0) {
212
69.2k
        code = pdfi_dict_key_first(ctx, dict, (pdf_obj **)&Key, &index);
213
432k
        while (code >= 0) {
214
432k
            code = pdfi_dict_get_no_deref(ctx, dict, Key, &Value);
215
432k
            if (code < 0) goto exit;
216
217
432k
            code = pdfi_pdfmark_setparam_pair(ctx, Key, Value, parray+offset+(keynum*2));
218
432k
            if (code < 0) goto exit;
219
220
432k
            pdfi_countdown(Key);
221
432k
            Key = NULL;
222
432k
            pdfi_countdown(Value);
223
432k
            Value = NULL;
224
432k
            pdfi_countdown(tempobj);
225
432k
            tempobj = NULL;
226
227
432k
            code = pdfi_dict_key_next(ctx, dict, (pdf_obj **)&Key, &index);
228
432k
            if (code == gs_error_undefined) {
229
69.1k
                code = 0;
230
69.1k
                break;
231
69.1k
            }
232
363k
            keynum ++;
233
363k
        }
234
69.2k
    }
235
79.2k
    if (code < 0) goto exit;
236
237
    /* CTM */
238
79.2k
    code = pdfi_pdfmark_ctm_str(ctx, ctm, &ctm_data, &ctm_len);
239
79.2k
    if (code < 0) goto exit;
240
79.2k
    parray[size-2].data = ctm_data;
241
79.2k
    parray[size-2].size = ctm_len;
242
243
    /* Type (e.g. ANN, DOCINFO) */
244
79.2k
    parray[size-1].data = (const byte *)type;
245
79.2k
    parray[size-1].size = strlen(type);
246
247
79.2k
    array_list.data = parray;
248
79.2k
    array_list.persistent = false;
249
79.2k
    array_list.size = size;
250
251
79.2k
    code = pdfi_pdfmark_write_array(ctx, &array_list, "pdfmark");
252
253
79.4k
 exit:
254
79.4k
    pdfi_countdown(Key);
255
79.4k
    pdfi_countdown(Value);
256
79.4k
    pdfi_countdown(tempobj);
257
79.4k
    if (parray != NULL) {
258
        /* Free the param data except the last two which are handled separately */
259
950k
        for (i=0; i<size-2; i++) {
260
870k
            if (parray[i].data)
261
870k
                gs_free_object(ctx->memory, (byte *)parray[i].data, "pdfi_pdfmark_from_dict(parray)");
262
870k
        }
263
79.4k
    }
264
79.4k
    if (ctm_data)
265
79.2k
        gs_free_object(ctx->memory, ctm_data, "pdfi_pdfmark_from_dict(ctm_data)");
266
79.4k
    gs_free_object(ctx->memory, parray, "pdfi_pdfmark_from_dict(parray)");
267
79.4k
    return code;
268
79.2k
}
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
74.5k
{
273
74.5k
    return pdfi_pdfmark_from_dict_withlabel(ctx, NULL, dict, ctm, type);
274
74.5k
}
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
103k
{
282
103k
    int code = 0;
283
103k
    int size;
284
103k
    gs_param_string *parray = NULL;
285
103k
    gs_param_string_array array_list;
286
103k
    byte *ctm_data = NULL;
287
103k
    int ctm_len;
288
103k
    gs_matrix ctm_placeholder;
289
103k
    int i;
290
291
    /* If ctm not provided, make a placeholder */
292
103k
    if (!ctm) {
293
103k
        gs_currentmatrix(ctx->pgs, &ctm_placeholder);
294
103k
        ctm = &ctm_placeholder;
295
103k
    }
296
297
103k
    size = len + 2; /* data + CTM + type */
298
299
103k
    parray = (gs_param_string *)gs_alloc_bytes(ctx->memory, (size_t)size*sizeof(gs_param_string),
300
103k
                                               "pdfi_pdfmark_from_objarray(parray)");
301
103k
    if (parray == NULL) {
302
0
        code = gs_note_error(gs_error_VMerror);
303
0
        goto exit;
304
0
    }
305
103k
    memset(parray, 0, size *sizeof(gs_param_string));
306
307
209k
    for (i=0; i<len; i++) {
308
106k
        code = pdfi_pdfmark_setparam_obj(ctx, objarray[i], parray+i);
309
106k
        if (code < 0) goto exit;
310
106k
    }
311
312
    /* CTM */
313
103k
    code = pdfi_pdfmark_ctm_str(ctx, ctm, &ctm_data, &ctm_len);
314
103k
    if (code < 0) goto exit;
315
103k
    parray[len].data = ctm_data;
316
103k
    parray[len].size = ctm_len;
317
318
    /* Type (e.g. ANN, DOCINFO) */
319
103k
    parray[len+1].data = (const byte *)type;
320
103k
    parray[len+1].size = strlen(type);
321
322
103k
    array_list.data = parray;
323
103k
    array_list.persistent = false;
324
103k
    array_list.size = size;
325
326
103k
    code = pdfi_pdfmark_write_array(ctx, &array_list, "pdfmark");
327
328
103k
 exit:
329
103k
    if (parray != NULL) {
330
209k
        for (i=0; i<len; i++) {
331
106k
            gs_free_object(ctx->memory, (byte *)parray[i].data, "pdfi_pdfmark_from_objarray(parray)");
332
106k
        }
333
103k
    }
334
103k
    if (ctm_data)
335
103k
        gs_free_object(ctx->memory, ctm_data, "pdfi_pdfmark_from_objarray(ctm_data)");
336
103k
    gs_free_object(ctx->memory, parray, "pdfi_pdfmark_from_objarray(parray)");
337
103k
    return code;
338
103k
}
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
511
{
345
511
    gs_param_string param_string;
346
511
    int code = 0;
347
348
511
    param_string.data = NULL;
349
350
511
    code = pdfi_loop_detector_mark(ctx);
351
511
    if (code < 0)
352
0
        goto exit;
353
354
511
    code = pdfi_resolve_indirect_loop_detect(ctx, NULL, object, true);
355
511
    (void)pdfi_loop_detector_cleartomark(ctx);
356
511
    if (code < 0)
357
35
        goto exit;
358
359
476
    code = pdfi_pdfmark_setparam_obj(ctx, object, &param_string);
360
476
    if (code < 0)
361
0
        goto exit;
362
363
476
    code = pdfi_pdfmark_write_string(ctx, &param_string, cmd);
364
365
511
exit:
366
511
    if (param_string.data != NULL)
367
476
        gs_free_object(ctx->memory, (byte *)param_string.data, "free data transferred to param_string in pdfi_pdfmark_object\n");
368
511
    return code;
369
476
}
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
8.04k
{
381
8.04k
    int code = 0;
382
8.04k
    int i;
383
8.04k
    uint64_t page_num;
384
8.04k
    pdf_dict *page_dict = NULL;
385
8.04k
    pdf_array *view_array = NULL;
386
8.04k
    uint64_t array_size;
387
8.04k
    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
8.04k
    code = pdfi_array_get_no_store_R(ctx, dest_array, 0, (pdf_obj **)&page_dict);
394
8.04k
    if (code < 0) goto exit;
395
396
7.45k
    if(pdfi_type_of(page_dict) == PDF_INT) {
397
14
        page_num = ((pdf_num *)page_dict)->value.i;
398
7.43k
    } else {
399
7.43k
        if (pdfi_type_of(page_dict) != PDF_DICT) {
400
15
            if (pdfi_type_of(page_dict) != PDF_NULL) {
401
2
                code = gs_note_error(gs_error_typecheck);
402
2
                goto exit;
403
2
            }
404
13
            page_num = 0;
405
7.42k
        } else {
406
            /* Find out which page number this is */
407
7.42k
            code = pdfi_page_get_number(ctx, page_dict, &page_num);
408
7.42k
            if (code < 0) goto exit;
409
7.42k
        }
410
7.43k
    }
411
412
7.40k
    page_num += ctx->Pdfmark_InitialPage;
413
414
7.40k
    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
7.39k
        code = pdfi_dict_put_int(ctx, link_dict, "Page", page_num+1);
419
13
    else
420
13
        code = pdfi_dict_put_int(ctx, link_dict, "Page", 0);
421
422
7.40k
    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
7.40k
    code = pdfi_array_get_no_store_R(ctx, dest_array, 1, &temp_obj);
430
7.40k
    if (code < 0) goto exit;
431
432
7.40k
    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
7.40k
    pdfi_countdown(temp_obj);
439
7.40k
    temp_obj = NULL;
440
441
    /* Build an array for /View, out of the remainder of the Dest entry */
442
7.40k
    array_size = pdfi_array_size(dest_array) - 1;
443
7.40k
    code = pdfi_array_alloc(ctx, array_size, &view_array);
444
7.40k
    if (code < 0) goto exit;
445
7.40k
    pdfi_countup(view_array);
446
23.5k
    for (i=0; i<array_size; i++) {
447
16.1k
        code = pdfi_array_get(ctx, dest_array, i+1, &temp_obj);
448
16.1k
        if (code < 0) goto exit;
449
16.1k
        code = pdfi_array_put(ctx, view_array, i, temp_obj);
450
16.1k
        if (code < 0) goto exit;
451
452
16.1k
        pdfi_countdown(temp_obj);
453
16.1k
        temp_obj = NULL;
454
16.1k
    }
455
    /* Add /View key to the link_dict */
456
7.40k
    code = pdfi_dict_put(ctx, link_dict, "View", (pdf_obj *)view_array);
457
7.40k
    if (code < 0) goto exit;
458
459
8.04k
 exit:
460
8.04k
    pdfi_countdown(temp_obj);
461
8.04k
    pdfi_countdown(page_dict);
462
8.04k
    pdfi_countdown(view_array);
463
8.04k
    return code;
464
7.40k
}
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
882
{
549
882
    int code = 0, code1 = 0;
550
882
    pdf_dict *Dests = NULL;
551
882
    pdf_obj *Dest = NULL;
552
882
    bool delete_Dest = true;
553
882
    pdf_array *dest_array = NULL;
554
882
    pdf_array *Names = NULL;
555
882
    pdf_dict *Names_dict = NULL;
556
557
882
    code = pdfi_dict_get(ctx, link_dict, "Dest", (pdf_obj **)&Dest);
558
882
    if (code < 0) goto exit;
559
560
882
    code = pdfi_dict_knownget_type(ctx, ctx->Root, "Dests", PDF_DICT, (pdf_obj **)&Dests);
561
882
    if (code < 0) goto exit;
562
563
879
    code = pdfi_dict_knownget_type(ctx, ctx->Root, "Names", PDF_DICT, (pdf_obj **)&Names_dict);
564
879
    if (code < 0) goto exit;
565
566
842
    switch (pdfi_type_of(Dest)) {
567
809
    case PDF_ARRAY:
568
809
        code = pdfi_pdfmark_add_Page_View(ctx, link_dict, (pdf_array *)Dest);
569
809
        if (code < 0) goto exit;
570
724
        break;
571
724
    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
842
    }
636
637
882
 exit:
638
882
    if (delete_Dest) {
639
        /* Delete the Dest key */
640
882
        code1 = pdfi_dict_delete(ctx, link_dict, "Dest");
641
882
        if (code1 < 0 && code >= 0)
642
0
            code = code1;
643
882
    }
644
882
    pdfi_countdown(Dest);
645
882
    pdfi_countdown(Dests);
646
882
    pdfi_countdown(Names);
647
882
    pdfi_countdown(Names_dict);
648
882
    pdfi_countdown(dest_array);
649
882
    return code;
650
842
}
651
652
static int pdfi_check_limits(pdf_context *ctx, pdf_dict *node, char *str, int len)
653
175
{
654
175
    int code = 0, min, i, len2 = 0;
655
175
    pdf_array *Limits = NULL;
656
175
    pdf_string *Str = NULL;
657
175
    char *str2 = NULL;
658
659
175
    code = pdfi_dict_get_type(ctx, node, "Limits", PDF_ARRAY, (pdf_obj **)&Limits);
660
175
    if (code < 0)
661
0
        goto error;
662
663
175
    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
1
        pdfi_set_warning(ctx, 0, NULL, W_PDF_BAD_TREE_LIMITS, "pdfi_check_limits", 0);
668
1
        goto error;
669
1
    }
670
671
174
    code = pdfi_array_get_type(ctx, Limits, 0, PDF_STRING, (pdf_obj **)&Str);
672
174
    if (code < 0)
673
0
        goto error;
674
675
174
    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
174
    } else {
680
174
        len2 = ((pdf_string *)Str)->length;
681
174
        str2 = (char *)gs_alloc_bytes(ctx->memory, (size_t)len2 + 1, "pdfi_check_limits");
682
174
        if (str2 == NULL) {
683
0
            code = gs_note_error(gs_error_VMerror);
684
0
            goto error;
685
0
        }
686
174
         memcpy(str2, ((pdf_string *)Str)->data, len2);
687
174
         str2[len2] = 0;
688
174
    }
689
690
174
    pdfi_countdown(Str);
691
174
    Str = NULL;
692
693
174
    min = len;
694
174
    if (len2 < min)
695
134
        min = len2;
696
697
519
    for (i=0;i< min;i++) {
698
504
        if (str[i] < str2[i]) {
699
2
            code = gs_note_error(gs_error_undefined);
700
2
            goto error;
701
2
        }
702
502
        if (str[i] != str2[i])
703
157
            break;
704
502
    }
705
172
    if (i > min && len2 < Str->length) {
706
0
        code = gs_note_error(gs_error_undefined);
707
0
        goto error;
708
0
    }
709
172
    gs_free_object(ctx->memory, str2, "pdfi_check_limits");
710
172
    str2 = NULL;
711
712
172
    code = pdfi_array_get_type(ctx, Limits, 1, PDF_STRING, (pdf_obj **)&Str);
713
172
    if (code < 0)
714
0
        goto error;
715
716
172
    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
172
    } else {
721
172
        len2 = ((pdf_string *)Str)->length;
722
172
        str2 = (char *)gs_alloc_bytes(ctx->memory, (size_t)len2 + 1, "pdfi_check_limits");
723
172
        if (str2 == NULL) {
724
0
            code = gs_note_error(gs_error_VMerror);
725
0
            goto error;
726
0
        }
727
172
         memcpy(str2, ((pdf_string *)Str)->data, len2);
728
172
         str2[len2] = 0;
729
172
    }
730
731
172
    pdfi_countdown(Str);
732
172
    Str = NULL;
733
734
172
    min = len;
735
172
    if (len2 < min)
736
122
        min = len2;
737
738
673
    for (i=0;i< min;i++) {
739
661
        if (str[i] > str2[i]) {
740
130
            code = gs_note_error(gs_error_undefined);
741
130
            goto error;
742
130
        }
743
531
        if (str[i] != str2[i])
744
30
            break;
745
531
    }
746
747
42
    if (i > min && len > len2)
748
0
        code = gs_note_error(gs_error_undefined);
749
750
175
error:
751
175
    gs_free_object(ctx->memory, str2, "pdfi_check_limits");
752
175
    pdfi_countdown(Str);
753
175
    pdfi_countdown(Limits);
754
175
    return code;
755
42
}
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
233
{
759
233
    int i = 0, code = 0;
760
233
    pdf_string *StrKey = NULL;
761
233
    pdf_array *NamesArray = NULL;
762
233
    pdf_dict *Kid = NULL;
763
233
    bool known;
764
765
233
    code = pdfi_loop_detector_mark(ctx);
766
233
    if (code < 0)
767
0
        return code;
768
769
233
    code = pdfi_dict_known(ctx, node, "Names", &known);
770
233
    if (code < 0)
771
0
        goto error;
772
773
233
    if (known) {
774
175
        code = pdfi_dict_known(ctx, node, "Limits", &known);
775
175
        if (code < 0)
776
0
            goto error;
777
778
175
        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
175
        } else {
785
175
            code = pdfi_check_limits(ctx, node, str, len);
786
175
            if (code < 0)
787
132
                goto error;
788
175
        }
789
790
43
        code = pdfi_dict_get_type(ctx, node, "Names", PDF_ARRAY, (pdf_obj **)&NamesArray);
791
43
        if (code < 0)
792
5
            goto error;
793
794
38
        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
121
        for (i = 0;i < pdfi_array_size(NamesArray) / 2; i++) {
798
119
            code = pdfi_array_get_type(ctx, NamesArray, i * 2, PDF_STRING, (pdf_obj **)&StrKey);
799
119
            if (code < 0)
800
0
                goto error;
801
802
119
            if (StrKey->length == len && memcmp((const char *)StrKey->data, str, len) == 0) {
803
36
                code = pdfi_array_get(ctx, NamesArray, (i * 2) + 1, (pdf_obj **)Name);
804
36
                goto error;
805
36
            }
806
83
            pdfi_countdown(StrKey);
807
83
            StrKey = NULL;
808
83
        }
809
2
        pdfi_countdown(NamesArray);
810
2
        NamesArray = NULL;
811
2
    }
812
813
    /* Either no Names array (initial node) or not in array */
814
60
    code = pdfi_dict_get_type(ctx, node, "Kids", PDF_ARRAY, (pdf_obj **)&NamesArray);
815
60
    if (code < 0)
816
4
        goto error;
817
818
190
    for (i = 0;i < pdfi_array_size(NamesArray); i++) {
819
189
        code = pdfi_array_get_type(ctx, NamesArray, i, PDF_DICT, (pdf_obj **)&Kid);
820
189
        if (code < 0)
821
14
            goto error;
822
823
175
        code = pdfi_get_name_from_node(ctx, Kid, str, len, Name, false);
824
175
        pdfi_countdown(Kid);
825
175
        Kid = NULL;
826
175
        if (code == 0)
827
36
            break;
828
829
139
        if (code < 0) {
830
139
            if (code == gs_error_undefined)
831
134
                continue;
832
5
            goto error;
833
139
        }
834
139
    }
835
836
233
error:
837
233
    pdfi_countdown(Kid);
838
233
    pdfi_countdown(StrKey);
839
233
    pdfi_countdown(NamesArray);
840
233
    (void)pdfi_loop_detector_cleartomark(ctx);
841
233
    return code;
842
56
}
843
844
static int pdfi_get_named_dest(pdf_context *ctx, pdf_obj *Named, pdf_obj **Dest)
845
162
{
846
162
    int code = 0, len = 0;
847
162
    pdf_dict *Names = NULL, *Dests = NULL;
848
162
    char *str = NULL;
849
850
162
    code = pdfi_loop_detector_mark(ctx);
851
162
    if (code < 0)
852
0
        return code;
853
854
162
    code = pdfi_dict_get_type(ctx, ctx->Root, "Names", PDF_DICT, (pdf_obj **)&Names);
855
162
    if (code < 0)
856
103
        goto error;
857
858
59
    code = pdfi_dict_get_type(ctx, Names, "Dests", PDF_DICT, (pdf_obj **)&Dests);
859
59
    if (code < 0)
860
1
        goto error;
861
862
58
    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
58
    } else {
867
58
        len = ((pdf_string *)Named)->length;
868
58
        str = (char *)gs_alloc_bytes(ctx->memory, (size_t)len + 1, "pdfi_get_named_dest");
869
58
        if (str == NULL) {
870
0
            code = gs_note_error(gs_error_VMerror);
871
0
            goto error;
872
0
        }
873
58
        memcpy(str, ((pdf_string *)Named)->data, len);
874
58
        str[len] = 0;
875
58
    }
876
877
58
    code = pdfi_get_name_from_node(ctx, Dests, str, len, Dest, true);
878
879
162
error:
880
162
    if (pdfi_type_of(Named) == PDF_NAME)
881
0
        (void)pdfi_free_string_from_name(ctx, str);
882
162
    else
883
162
        gs_free_object(ctx->memory, str, "pdfi_get_named_dest");
884
162
    pdfi_countdown(Names);
885
162
    pdfi_countdown(Dests);
886
162
    pdfi_loop_detector_cleartomark(ctx);
887
162
    return code;
888
58
}
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
9.24k
{
895
9.24k
    int code = 0;
896
9.24k
    pdf_dict *A_dict = NULL;
897
9.24k
    bool known;
898
9.24k
    pdf_name *S_name = NULL;
899
9.24k
    pdf_array *D_array = NULL;
900
9.24k
    bool delete_A = false;
901
9.24k
    bool deref_A = true;
902
903
9.24k
    code = pdfi_dict_get_no_store_R(ctx, dict, "A", (pdf_obj **)&A_dict);
904
9.24k
    if (code < 0) goto exit;
905
906
9.08k
    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
479
        delete_A = true;
911
479
        goto exit;
912
479
    }
913
914
    /* Handle URI */
915
8.60k
    code = pdfi_dict_known(ctx, A_dict, "URI", &known);
916
8.60k
    if (code < 0) goto exit;
917
8.60k
    if (known) {
918
1.08k
        code = pdfi_resolve_indirect_loop_detect(ctx, (pdf_obj *)NULL, (pdf_obj *)A_dict, true);
919
1.08k
        goto exit;
920
1.08k
    }
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
7.52k
    code = pdfi_dict_knownget_type(ctx, A_dict, "S", PDF_NAME, (pdf_obj **)&S_name);
927
7.52k
    if (code <= 0) goto exit;
928
    /* We only handle GoTo for now */
929
7.51k
    if (pdfi_name_is(S_name, "GoTo")) {
930
7.36k
        code = pdfi_dict_knownget(ctx, A_dict, "D", (pdf_obj **)&D_array);
931
7.36k
        if (code <= 0)
932
2
            goto exit;
933
7.35k
        if (pdfi_type_of(D_array) == PDF_STRING || pdfi_type_of(D_array) == PDF_NAME)
934
160
        {
935
160
            pdf_obj *Dest = NULL;
936
937
160
            code = pdfi_get_named_dest(ctx, (pdf_obj *)D_array, &Dest);
938
160
            if (code < 0)
939
126
                goto exit;
940
34
            pdfi_countdown(D_array);
941
34
            D_array = NULL;
942
34
            if (pdfi_type_of(Dest) != PDF_ARRAY) {
943
34
                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
34
                code = pdfi_dict_knownget(ctx, (pdf_dict *)Dest, "D", (pdf_obj **)&D_array);
949
34
                pdfi_countdown(Dest);
950
34
                if (code <= 0)
951
0
                    goto exit;
952
34
            } else
953
0
                D_array = (pdf_array *)Dest;
954
34
        }
955
7.23k
        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
7.23k
        code = pdfi_pdfmark_add_Page_View(ctx, dict, D_array);
962
7.23k
        if (code < 0) goto exit;
963
6.67k
        delete_A = true;
964
6.67k
    } 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
151
    } 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
7
        delete_A = false;
985
7
        code = 0;
986
144
    } else if (pdfi_name_is(S_name, "GoToE")) {
987
        /* TODO: ??
988
         * File: fts_33_3303.pdf
989
         */
990
86
    } 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
86
    } else if (pdfi_name_is(S_name, "Sound")) {
997
        /* TODO: ??
998
         * File: fts_33_3307.pdf
999
         */
1000
0
        deref_A = false;
1001
86
    } else if (pdfi_name_is(S_name, "Movie")) {
1002
        /* TODO: ??
1003
         * File: fts_33_3308.pdf
1004
         */
1005
0
        deref_A = false;
1006
86
    } else if (pdfi_name_is(S_name, "GoTo3DView")) {
1007
        /* TODO: ??
1008
         * File: fts_33_3318.pdf
1009
         */
1010
86
    } else if (pdfi_name_is(S_name, "RichMediaExecute")) {
1011
        /* TODO: ??
1012
         * File: fts_33_3319.pdf
1013
         */
1014
86
    } 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
86
    } else {
1019
        /* TODO: flag warning? */
1020
86
    }
1021
1022
9.24k
 exit:
1023
9.24k
    if (delete_A) {
1024
7.15k
        code = pdfi_dict_delete(ctx, dict, "A");
1025
7.15k
    } else if (deref_A) {
1026
2.09k
        pdfi_countdown(A_dict);
1027
2.09k
        A_dict = NULL;
1028
2.09k
    }
1029
9.24k
    pdfi_countdown(A_dict);
1030
9.24k
    pdfi_countdown(S_name);
1031
9.24k
    pdfi_countdown(D_array);
1032
9.24k
    return code;
1033
7.51k
}
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.86k
{
1041
4.86k
    int code;
1042
4.86k
    pdf_obj *objarray[4];
1043
4.86k
    int num_objects = 4;
1044
4.86k
    int i;
1045
1046
4.86k
    memset(objarray, 0, sizeof(objarray));
1047
1048
4.86k
    code = pdfi_obj_charstr_to_name(ctx, "_objdef", (pdf_name **)&objarray[0]);
1049
4.86k
    if (code < 0) goto exit;
1050
1051
4.86k
    objarray[1] = (pdf_obj *)label;
1052
4.86k
    pdfi_countup(label);
1053
1054
4.86k
    code = pdfi_obj_charstr_to_name(ctx, "type", (pdf_name **)&objarray[2]);
1055
4.86k
    if (code < 0) goto exit;
1056
1057
4.86k
    code = pdfi_obj_charstr_to_name(ctx, type, (pdf_name **)&objarray[3]);
1058
4.86k
    if (code < 0) goto exit;
1059
1060
4.86k
    code = pdfi_pdfmark_from_objarray(ctx, objarray, num_objects, NULL, "OBJ");
1061
4.86k
    if (code < 0) goto exit;
1062
1063
4.86k
 exit:
1064
24.3k
    for (i=0; i<num_objects; i++)
1065
19.4k
        pdfi_countdown(objarray[i]);
1066
4.86k
    return code;
1067
4.86k
}
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
504
{
1075
504
    int code;
1076
504
    pdf_obj *objarray[1];
1077
504
    int num_objects = 1;
1078
504
    int i;
1079
1080
504
    memset(objarray, 0, sizeof(objarray));
1081
1082
504
    objarray[0] = (pdf_obj *)label;
1083
504
    pdfi_countup(label);
1084
1085
504
    code = pdfi_pdfmark_from_objarray(ctx, objarray, num_objects, NULL, "CLOSE");
1086
504
    if (code < 0) goto exit;
1087
1088
504
 exit:
1089
1.00k
    for (i=0; i<num_objects; i++)
1090
504
        pdfi_countdown(objarray[i]);
1091
504
    return code;
1092
504
}
1093
1094
static int pdfi_pdfmark_stream_contents(pdf_context *ctx, pdf_indirect_ref *label, pdf_stream *stream)
1095
504
{
1096
504
    int code;
1097
504
    pdf_obj *objarray[2];
1098
504
    int num_objects = 2;
1099
504
    int i;
1100
1101
504
    objarray[0] = (pdf_obj *)label;
1102
504
    pdfi_countup(label);
1103
1104
504
    objarray[1] = (pdf_obj *)stream;
1105
504
    pdfi_countup(stream);
1106
504
    stream->is_marking = true;
1107
1108
504
    code = pdfi_pdfmark_from_objarray(ctx, objarray, num_objects, NULL, ".PUTSTREAM");
1109
504
    if (code < 0) goto exit;
1110
1111
504
 exit:
1112
504
    stream->is_marking = false;
1113
1.51k
    for (i=0; i<num_objects; i++)
1114
1.00k
        pdfi_countdown(objarray[i]);
1115
504
    return code;
1116
504
}
1117
1118
/* Mark a stream object */
1119
int pdfi_pdfmark_stream(pdf_context *ctx, pdf_stream *stream)
1120
563
{
1121
563
    int code;
1122
563
    pdf_dict *streamdict = NULL;
1123
563
    pdf_indirect_ref *streamref = NULL;
1124
563
    pdf_dict *tempdict = NULL;
1125
563
    uint64_t dictsize;
1126
563
    uint64_t index;
1127
563
    pdf_name *Key = NULL;
1128
1129
563
    if (stream->stream_written)
1130
55
        return 0;
1131
1132
508
    stream->stream_written = true;
1133
1134
508
    if (!ctx->device_state.writepdfmarks)
1135
0
        return 0;
1136
1137
    /* Create an indirect ref for the stream */
1138
508
    code = pdfi_object_alloc(ctx, PDF_INDIRECT, 0, (pdf_obj **)&streamref);
1139
508
    if (code < 0) goto exit;
1140
508
    pdfi_countup(streamref);
1141
508
    streamref->ref_object_num = stream->object_num;
1142
508
    streamref->ref_generation_num = stream->generation_num;
1143
508
    streamref->is_marking = true;
1144
1145
508
    code = pdfi_dict_from_obj(ctx, (pdf_obj *)stream, &streamdict);
1146
508
    if (code < 0) goto exit;
1147
1148
    /* Make a copy of the dict and remove Filter keyword */
1149
508
    dictsize = pdfi_dict_entries(streamdict);
1150
508
    code = pdfi_dict_alloc(ctx, dictsize, &tempdict);
1151
508
    if (code < 0) goto exit;
1152
508
    pdfi_countup(tempdict);
1153
508
    code = pdfi_dict_copy(ctx, tempdict, streamdict);
1154
508
    if (code < 0) goto exit;
1155
508
    code = pdfi_dict_key_first(ctx, streamdict, (pdf_obj **)&Key, &index);
1156
1.89k
    while (code >= 0) {
1157
1.89k
        if (pdfi_name_is(Key, "Filter") || pdfi_name_is(Key, "Length")) {
1158
736
            code = pdfi_dict_delete_pair(ctx, tempdict, Key);
1159
736
            if (code < 0) goto exit;
1160
736
        }
1161
1.89k
        pdfi_countdown(Key);
1162
1.89k
        Key = NULL;
1163
1164
1.89k
        code = pdfi_dict_key_next(ctx, streamdict, (pdf_obj **)&Key, &index);
1165
1.89k
        if (code == gs_error_undefined) {
1166
507
            code = 0;
1167
507
            break;
1168
507
        }
1169
1.89k
    }
1170
508
    if (code < 0) goto exit;
1171
1172
507
    code = pdfi_pdfmark_objdef_begin(ctx, streamref, "stream");
1173
507
    if (code < 0) goto exit;
1174
1175
507
    code = pdfi_pdfmark_from_dict_withlabel(ctx, streamref, tempdict, NULL, ".PUTDICT");
1176
507
    if (code < 0) goto exit;
1177
1178
504
    code = pdfi_pdfmark_stream_contents(ctx, streamref, stream);
1179
504
    if (code < 0) goto exit;
1180
1181
504
    code = pdfi_pdfmark_objdef_close(ctx, streamref);
1182
504
    if (code < 0) goto exit;
1183
1184
508
 exit:
1185
508
    pdfi_countdown(tempdict);
1186
508
    pdfi_countdown(streamref);
1187
508
    return code;
1188
504
}
1189
1190
/* Mark a dict object */
1191
int pdfi_pdfmark_dict(pdf_context *ctx, pdf_dict *dict)
1192
29.1k
{
1193
29.1k
    int code;
1194
29.1k
    pdf_indirect_ref *dictref = NULL;
1195
1196
29.1k
    if (dict->dict_written)
1197
24.8k
        return 0;
1198
1199
4.35k
    dict->dict_written = true;
1200
1201
4.35k
    if (!ctx->device_state.writepdfmarks)
1202
0
        return 0;
1203
1204
    /* Create an indirect ref for the dict */
1205
4.35k
    code = pdfi_object_alloc(ctx, PDF_INDIRECT, 0, (pdf_obj **)&dictref);
1206
4.35k
    if (code < 0) goto exit;
1207
4.35k
    pdfi_countup(dictref);
1208
4.35k
    dictref->ref_object_num = dict->object_num;
1209
4.35k
    dictref->ref_generation_num = dict->generation_num;
1210
4.35k
    dictref->is_marking = true;
1211
1212
4.35k
    code = pdfi_pdfmark_objdef_begin(ctx, dictref, "dict");
1213
4.35k
    if (code < 0) goto exit;
1214
1215
4.35k
    code = pdfi_pdfmark_from_dict_withlabel(ctx, dictref, dict, NULL, ".PUTDICT");
1216
4.35k
    if (code < 0) goto exit;
1217
1218
4.35k
 exit:
1219
4.35k
    pdfi_countdown(dictref);
1220
4.35k
    return code;
1221
4.35k
}
1222
1223
static int pdfi_pdfmark_filespec(pdf_context *ctx, pdf_string *name, pdf_dict *filespec)
1224
26
{
1225
26
    int code;
1226
26
    pdf_dict *tempdict = NULL;
1227
1228
26
    code = pdfi_dict_alloc(ctx, 40, &tempdict);
1229
26
    if (code < 0) goto exit;
1230
26
    pdfi_countup(tempdict);
1231
1232
26
    code = pdfi_dict_put(ctx, tempdict, "Name", (pdf_obj *)name);
1233
26
    if (code < 0) goto exit;
1234
1235
    /* Flatten the filespec */
1236
26
    code = pdfi_resolve_indirect(ctx, (pdf_obj *)filespec, true);
1237
26
    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
26
 exit:
1246
26
    pdfi_countdown(tempdict);
1247
26
    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
26
{
1253
26
    int code;
1254
1255
26
    code = pdfi_pdfmark_filespec(ctx, name, filespec);
1256
26
    if (code < 0) goto exit;
1257
1258
26
 exit:
1259
26
    return code;
1260
26
}
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
26.8k
{
1268
26.8k
    int i, code = 0;
1269
26.8k
    pdf_dict *Info = NULL;
1270
26.8k
    pdf_obj *o = NULL;
1271
    /* We don't preserve the Producer, we are the Producer */
1272
26.8k
    const char *KeyNames[] = {
1273
26.8k
        "Title", "Author", "Subject", "Keywords", "Creator"
1274
26.8k
    };
1275
1276
26.8k
    if (!ctx->device_state.writepdfmarks)
1277
23.9k
        return;
1278
1279
2.93k
    code = pdfi_dict_alloc(ctx, 5, &Info);
1280
2.93k
    if (code < 0)
1281
0
        goto exit;
1282
2.93k
    pdfi_countup(Info);
1283
1284
17.6k
    for (i=0;i<5;i++)
1285
14.6k
    {
1286
14.6k
        if (pdfi_dict_knownget(ctx, info_dict, KeyNames[i], &o) > 0)
1287
5.69k
        {
1288
5.69k
            (void)pdfi_dict_put(ctx, Info, KeyNames[i], (pdf_obj *)o);
1289
5.69k
            pdfi_countdown(o);
1290
5.69k
        }
1291
14.6k
    }
1292
1293
2.93k
    code = pdfi_pdfmark_from_dict(ctx, Info, NULL, "DOCINFO");
1294
2.93k
exit:
1295
2.93k
    pdfi_countdown(Info);
1296
2.93k
    return;
1297
2.93k
}
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
123k
{
1312
123k
    int i, code = 0;
1313
123k
    pdf_dict *BoxDict = NULL;
1314
123k
    pdf_obj *o = NULL;
1315
123k
    gx_device *device = gs_currentdevice(ctx->pgs);
1316
123k
    gs_matrix scale, m, ctm;
1317
123k
    const char *BoxNames[] = {
1318
123k
        "CropBox", "BleedBox", "TrimBox", "ArtBox"
1319
123k
    };
1320
1321
    /* If the device doesn't support pdfmark, or the device modifies the Page Size (making the other boxes unreliable) exit now */
1322
123k
    if (!ctx->device_state.writepdfmarks || ctx->device_state.ModifiesPageSize)
1323
108k
        return;
1324
1325
    /* If we are using somethign other than the MediaBox, don't send these */
1326
14.8k
    if (ctx->args.usecropbox || ctx->args.usebleedbox ||
1327
14.8k
        ctx->args.usetrimbox || ctx->args.useartbox)
1328
0
        return;
1329
1330
14.8k
    code = pdfi_dict_alloc(ctx, 4, &BoxDict);
1331
14.8k
    if (code < 0)
1332
0
        goto exit;
1333
1334
14.8k
    pdfi_countup(BoxDict);
1335
1336
    /* Undo the resolution scaling from the CTM, we don't want to apply that */
1337
14.8k
    scale.xx = 72.0 / device->HWResolution[0];
1338
14.8k
    scale.xy = 0;
1339
14.8k
    scale.yx = 0;
1340
14.8k
    scale.yy = 72.0 / device->HWResolution[1];
1341
14.8k
    scale.tx = 0;
1342
14.8k
    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
14.8k
    gs_currentmatrix(ctx->pgs, &ctm);
1348
14.8k
    code = gs_matrix_multiply(&ctm, &scale, &m);
1349
14.8k
    if (code < 0) goto exit;
1350
1351
74.3k
    for (i=0;i<4;i++)
1352
59.5k
    {
1353
        /* Check each Bos name in turn */
1354
59.5k
        if (pdfi_dict_knownget(ctx, page_dict, BoxNames[i], &o) > 0){
1355
9.22k
            gs_rect box;
1356
9.22k
            pdf_array *new_array = NULL;
1357
1358
            /* Box is present in page dicitonayr, check it's an array */
1359
9.22k
            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
9.22k
            code = pdfi_array_to_gs_rect(ctx, (pdf_array *)o, &box);
1366
9.22k
            pdfi_countdown(o);
1367
9.22k
            if (code < 0)
1368
34
                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
9.19k
            pdfi_normalize_rect(ctx, &box);
1374
1375
            /* Transform the resulting box by the calculated matrix */
1376
9.19k
            pdfi_bbox_transform(ctx, &box, &m);
1377
1378
            /* Get a new array created from the box values */
1379
9.19k
            code = pdfi_gs_rect_to_array(ctx, &box, &new_array);
1380
9.19k
            if (code < 0)
1381
0
                continue;
1382
1383
            /* And store it in the working dictionary */
1384
9.19k
            (void)pdfi_dict_put(ctx, BoxDict, BoxNames[i], (pdf_obj *)new_array);
1385
9.19k
            pdfi_countdown(new_array);
1386
9.19k
        }
1387
59.5k
    }
1388
1389
    /* Send all the Box entries to the device */
1390
14.8k
    (void)pdfi_pdfmark_from_dict(ctx, BoxDict, NULL, "PAGE");
1391
1392
14.8k
exit:
1393
14.8k
    pdfi_countdown(BoxDict);
1394
14.8k
    return;
1395
14.8k
}