Coverage Report

Created: 2026-02-14 07:09

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