Coverage Report

Created: 2025-06-10 07:27

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