Coverage Report

Created: 2025-08-28 07:06

/src/ghostpdl/devices/vector/gdevpdti.c
Line
Count
Source (jump to first uncovered line)
1
/* Copyright (C) 2001-2025 Artifex Software, Inc.
2
   All Rights Reserved.
3
4
   This software is provided AS-IS with no warranty, either express or
5
   implied.
6
7
   This software is distributed under license and may not be copied,
8
   modified or distributed except as expressly authorized under the terms
9
   of the license contained in the file LICENSE in this distribution.
10
11
   Refer to licensing information at http://www.artifex.com or contact
12
   Artifex Software, Inc.,  39 Mesa Street, Suite 108A, San Francisco,
13
   CA 94129, USA, for further information.
14
*/
15
16
17
/* Bitmap font implementation for pdfwrite */
18
#include "memory_.h"
19
#include "gx.h"
20
#include "gxpath.h"
21
#include "gserrors.h"
22
#include "gsutil.h"
23
#include "gdevpdfx.h"
24
#include "gdevpdfg.h"
25
#include "gdevpdtf.h"
26
#include "gdevpdti.h"
27
#include "gdevpdts.h"
28
#include "gdevpdtw.h"
29
#include "gdevpdtt.h"
30
#include "gdevpdfo.h"
31
#include "gxchar.h"        /* For gs_show_enum */
32
33
/* ---------------- Private ---------------- */
34
35
/* Define the structure for a CharProc pseudo-resource. */
36
/*typedef struct pdf_char_proc_s pdf_char_proc_t;*/  /* gdevpdfx.h */
37
struct pdf_char_proc_s {
38
    pdf_resource_common(pdf_char_proc_t);
39
    pdf_char_proc_ownership_t *owner_fonts; /* fonts using this charproc. */
40
    int y_offset;                /* of character (0,0) */
41
    int x_offset;                /* of character (0,0) */
42
    gs_point real_width;        /* Not used with synthesised bitmap fonts. */
43
    gs_point v;                        /* Not used with synthesised bitmap fonts. */
44
};
45
46
/* The descriptor is public for pdf_resource_type_structs. */
47
gs_public_st_suffix_add1(st_pdf_char_proc, pdf_char_proc_t,
48
  "pdf_char_proc_t", pdf_char_proc_enum_ptrs, pdf_char_proc_reloc_ptrs,
49
  st_pdf_resource, owner_fonts);
50
51
struct pdf_char_proc_ownership_s {
52
    pdf_char_proc_t *char_proc;
53
    pdf_char_proc_ownership_t *font_next;        /* next char_proc for same font */
54
    pdf_char_proc_ownership_t *char_next;        /* next char_proc for same charproc */
55
    pdf_font_resource_t *font;
56
    gs_char char_code;                /* Character code in PDF font. */
57
    gs_glyph glyph;                /* Glyph id in Postscript font. */
58
    gs_string char_name;
59
    bool duplicate_char_name;
60
};
61
gs_private_st_strings1_ptrs4(st_pdf_char_proc_ownership, pdf_char_proc_ownership_t,
62
  "pdf_char_proc_ownership_t", pdf_char_proc_ownership_enum_ptrs,
63
  pdf_char_proc_ownership_reloc_ptrs, char_name, char_proc, char_next, font_next, font);
64
65
gs_private_st_ptrs1(st_pdf_bitmap_fonts, pdf_bitmap_fonts_t,
66
  "pdf_bitmap_fonts_t", pdf_bitmap_fonts_enum_ptrs,
67
  pdf_bitmap_fonts_reloc_ptrs, open_font);
68
69
static inline int64_t
70
pdf_char_proc_id(const pdf_char_proc_t *pcp)
71
296k
{
72
296k
    return pdf_resource_id((const pdf_resource_t *)pcp);
73
296k
}
74
75
/* Assign a code for a char_proc. */
76
static int
77
assign_char_code(gx_device_pdf * pdev, gs_text_enum_t *pte)
78
126k
{
79
126k
    pdf_bitmap_fonts_t *pbfs = pdev->text->bitmap_fonts;
80
126k
    pdf_font_resource_t *pdfont = pbfs->open_font; /* Type 3 */
81
126k
    int i, c = 0, code;
82
126k
    uint operation = pte->text.operation;
83
84
126k
    if (pbfs->bitmap_encoding_id == 0)
85
1.75k
        pbfs->bitmap_encoding_id = pdf_obj_ref(pdev);
86
126k
    if (pdfont == 0 || pdfont->u.simple.LastChar == 255 ||
87
126k
        !pbfs->use_open_font
88
126k
        ) {
89
        /* Start a new synthesized font. */
90
2.29k
        char *pc;
91
92
2.29k
        code = pdf_font_type3_alloc(pdev, &pdfont, pdf_write_contents_bitmap);
93
2.29k
        if (code < 0)
94
0
            return code;
95
2.29k
        pdfont->u.simple.s.type3.bitmap_font = true;
96
2.29k
        if (pbfs->open_font == 0)
97
1.75k
            pdfont->rname[0] = 0;
98
539
        else
99
539
            strcpy(pdfont->rname, pbfs->open_font->rname);
100
2.29k
        pdfont->u.simple.s.type3.FontBBox.p.x = 0;
101
2.29k
        pdfont->u.simple.s.type3.FontBBox.p.y = 0;
102
2.29k
        pdfont->u.simple.s.type3.FontBBox.q.x = 0;
103
2.29k
        pdfont->u.simple.s.type3.FontBBox.q.y = 0;
104
2.29k
        pdfont->mark_glyph = NULL;
105
2.29k
        gs_make_identity(&pdfont->u.simple.s.type3.FontMatrix);
106
        /*
107
         * We "increment" the font name as a radix-26 "number".
108
         * This cannot possibly overflow.
109
         */
110
2.29k
        for (pc = pdfont->rname; *pc == 'Z'; ++pc)
111
0
            *pc = '@';
112
2.29k
        if ((*pc)++ == 0)
113
1.75k
            *pc = 'A', pc[1] = 0;
114
2.29k
        pbfs->open_font = pdfont;
115
2.29k
        pbfs->use_open_font = true;
116
2.29k
        pdfont->u.simple.FirstChar = 255;
117
2.29k
    }
118
126k
    if ((operation & (TEXT_FROM_STRING | TEXT_FROM_BYTES)) ||
119
126k
        (operation & (TEXT_FROM_CHARS | TEXT_FROM_SINGLE_CHAR))) {
120
126k
        unsigned char p = *pte->text.data.bytes;
121
126k
        unsigned char index = p / 8, bit = 0x01 << (p % 8);
122
123
126k
        if (pdfont->used[index] & bit) {
124
9.95M
            for (i = 0;i < 256;i++) {
125
9.95M
                index = i / 8;
126
9.95M
                bit = 0x01 << (i % 8);
127
9.95M
                if (!(pdfont->used[index] & bit)) {
128
114k
                    c = i;
129
114k
                    break;
130
114k
                }
131
9.95M
            }
132
114k
        } else
133
11.8k
            c = p;
134
126k
        pdfont->used[index] |= bit;
135
126k
        if (c > pdfont->u.simple.LastChar)
136
80.2k
            pdfont->u.simple.LastChar = c;
137
138
126k
    } else {
139
0
        unsigned char index, bit;
140
0
        c = ++(pdfont->u.simple.LastChar);
141
0
        index = c / 8;
142
0
        bit = 0x01 << (c % 8);
143
0
        pdfont->used[index] |= bit;
144
0
    }
145
126k
    if (c < pdfont->u.simple.FirstChar)
146
3.69k
        pdfont->u.simple.FirstChar = c;
147
148
126k
    pdfont->Widths[c] = psdf_round(pdev->char_width.x, 100, 10); /* See
149
                        pdf_write_Widths about rounding. We need to provide
150
                        a compatible data for Tj. */
151
126k
    if (c > pbfs->max_embedded_code)
152
57.9k
        pbfs->max_embedded_code = c;
153
154
126k
    return c;
155
126k
}
156
157
/* Write the contents of a Type 3 bitmap or vector font resource. */
158
int
159
pdf_write_contents_bitmap(gx_device_pdf *pdev, pdf_font_resource_t *pdfont)
160
3.63k
{
161
3.63k
    stream *s = pdev->strm;
162
3.63k
    const pdf_char_proc_ownership_t *pcpo;
163
3.63k
    int64_t diff_id = 0;
164
3.63k
    int code;
165
166
3.63k
    if (pdfont->u.simple.s.type3.bitmap_font)
167
2.29k
        diff_id = pdev->text->bitmap_fonts->bitmap_encoding_id;
168
1.34k
    else {
169
        /* See comment in pdf_write_encoding. */
170
1.34k
        diff_id = pdf_obj_ref(pdev);
171
1.34k
    }
172
3.63k
    code = pdf_write_encoding_ref(pdev, pdfont, diff_id);
173
3.63k
    if (code < 0)
174
0
        return code;
175
3.63k
    stream_puts(s, "/CharProcs <<");
176
    /* Write real characters. */
177
151k
    for (pcpo = pdfont->u.simple.s.type3.char_procs; pcpo;
178
148k
         pcpo = pcpo->char_next
179
148k
         ) {
180
148k
        if (pdfont->u.simple.s.type3.bitmap_font)
181
126k
            pprinti64d2(s, "/a%"PRId64" %"PRId64" 0 R\n", pcpo->char_code,
182
126k
                      pdf_char_proc_id(pcpo->char_proc));
183
21.9k
        else if (!pcpo-> duplicate_char_name) {
184
21.8k
            pdf_put_name(pdev, pcpo->char_name.data, pcpo->char_name.size);
185
21.8k
            pprinti64d1(s, " %"PRId64" 0 R\n", pdf_char_proc_id(pcpo->char_proc));
186
21.8k
        }
187
148k
        pdf_record_usage_by_parent(pdev, pdf_char_proc_id(pcpo->char_proc), pdfont->object->id);
188
148k
    }
189
3.63k
    stream_puts(s, ">>");
190
3.63k
    pprintg6(s, "/FontMatrix[%g %g %g %g %g %g]",
191
3.63k
            (float)pdfont->u.simple.s.type3.FontMatrix.xx,
192
3.63k
            (float)pdfont->u.simple.s.type3.FontMatrix.xy,
193
3.63k
            (float)pdfont->u.simple.s.type3.FontMatrix.yx,
194
3.63k
            (float)pdfont->u.simple.s.type3.FontMatrix.yy,
195
3.63k
            (float)pdfont->u.simple.s.type3.FontMatrix.tx,
196
3.63k
            (float)pdfont->u.simple.s.type3.FontMatrix.ty);
197
3.63k
    code = pdf_finish_write_contents_type3(pdev, pdfont);
198
3.63k
    if (code < 0)
199
0
        return code;
200
3.63k
    if (!pdfont->u.simple.s.type3.bitmap_font && diff_id > 0) {
201
1.34k
        code = pdf_write_encoding(pdev, pdfont, diff_id, 0);
202
1.34k
        if (code < 0)
203
0
            return code;
204
1.34k
    }
205
3.63k
    return 0;
206
3.63k
}
207
208
/* ---------------- Public ---------------- */
209
210
/*
211
 * Allocate and initialize bookkeeping for bitmap fonts.
212
 */
213
pdf_bitmap_fonts_t *
214
pdf_bitmap_fonts_alloc(gs_memory_t *mem)
215
50.0k
{
216
50.0k
    pdf_bitmap_fonts_t *pbfs =
217
50.0k
        gs_alloc_struct(mem, pdf_bitmap_fonts_t, &st_pdf_bitmap_fonts,
218
50.0k
                        "pdf_bitmap_fonts_alloc");
219
220
50.0k
    if (pbfs == 0)
221
0
        return 0;
222
50.0k
    memset(pbfs, 0, sizeof(*pbfs));
223
50.0k
    pbfs->max_embedded_code = -1;
224
50.0k
    return pbfs;
225
50.0k
}
226
227
/*
228
 * Update text state at the end of a page.
229
 */
230
void
231
pdf_close_text_page(gx_device_pdf *pdev)
232
66.0k
{
233
    /*
234
     * When Acrobat Reader 3 prints a file containing a Type 3 font with a
235
     * non-standard Encoding, it apparently only emits the subset of the
236
     * font actually used on the page.  Thus, if the "Download Fonts Once"
237
     * option is selected, characters not used on the page where the font
238
     * first appears will not be defined, and hence will print as blank if
239
     * used on subsequent pages.  Thus, we can't allow a Type 3 font to
240
     * add additional characters on subsequent pages.
241
     */
242
66.0k
    if (pdev->CompatibilityLevel <= 1.2)
243
35.4k
        pdev->text->bitmap_fonts->use_open_font = false;
244
66.0k
}
245
246
int
247
pdf_charproc_y_offset(pdf_char_proc_t *pcp)
248
1.56M
{
249
1.56M
    return pcp->y_offset;
250
1.56M
}
251
252
int
253
pdf_charproc_x_offset(pdf_char_proc_t *pcp)
254
1.56M
{
255
1.56M
    return pcp->x_offset;
256
1.56M
}
257
258
/* Attach a CharProc to a font. */
259
static int
260
pdf_attach_charproc(gx_device_pdf * pdev, pdf_font_resource_t *pdfont, pdf_char_proc_t *pcp,
261
                    gs_glyph glyph, gs_char char_code, const gs_const_string *gnstr)
262
299k
{
263
299k
    pdf_char_proc_ownership_t *pcpo;
264
299k
    bool duplicate_char_name = false;
265
266
15.8M
    for (pcpo = pdfont->u.simple.s.type3.char_procs; pcpo != NULL; pcpo = pcpo->char_next) {
267
15.7M
        if (pcpo->glyph == glyph && pcpo->char_code == char_code)
268
145k
            return 0;
269
15.7M
    }
270
153k
    if (!pdfont->u.simple.s.type3.bitmap_font) {
271
992k
        for (pcpo = pdfont->u.simple.s.type3.char_procs; pcpo != NULL; pcpo = pcpo->char_next) {
272
964k
            if (!bytes_compare(pcpo->char_name.data, pcpo->char_name.size, gnstr->data, gnstr->size)) {
273
57
                duplicate_char_name = true;
274
57
                break;
275
57
            }
276
964k
        }
277
27.7k
    }
278
153k
    pcpo = gs_alloc_struct(pdev->pdf_memory,
279
153k
            pdf_char_proc_ownership_t, &st_pdf_char_proc_ownership, "pdf_attach_charproc");
280
281
153k
    if (pcpo == NULL)
282
0
        return_error(gs_error_VMerror);
283
153k
    pcpo->font = pdfont;
284
153k
    pcpo->char_next = pdfont->u.simple.s.type3.char_procs;
285
153k
    pdfont->u.simple.s.type3.char_procs = pcpo;
286
153k
    pcpo->char_proc = pcp;
287
153k
    pcpo->font_next = pcp->owner_fonts;
288
153k
    pcp->owner_fonts = pcpo;
289
153k
    pcpo->char_code = char_code;
290
153k
    pcpo->glyph = glyph;
291
153k
    if (gnstr == NULL) {
292
126k
        pcpo->char_name.data = 0;
293
126k
        pcpo->char_name.size = 0;
294
126k
    } else {
295
27.7k
        if (gnstr->size > 0) {
296
27.7k
            pcpo->char_name.data = gs_alloc_bytes(pdev->pdf_memory->non_gc_memory, gnstr->size, "storage for charproc name");
297
27.7k
            if (pcpo->char_name.data == NULL)
298
0
                return_error(gs_error_VMerror);
299
27.7k
            memcpy(pcpo->char_name.data, gnstr->data, gnstr->size);
300
27.7k
        }
301
27.7k
        pcpo->char_name.size = gnstr->size;
302
27.7k
    }
303
153k
    pcpo->duplicate_char_name = duplicate_char_name;
304
153k
    return 0;
305
153k
}
306
307
int
308
pdf_free_charproc_ownership(gx_device_pdf * pdev, pdf_resource_t *pres)
309
3.72k
{
310
3.72k
    pdf_char_proc_ownership_t *next, *pcpo = (pdf_char_proc_ownership_t *)pres;
311
312
157k
    while (pcpo) {
313
153k
        next = pcpo->char_next;
314
153k
        if(pcpo->char_name.size != 0 && pcpo->char_name.data) {
315
27.7k
            gs_free_object(pdev->pdf_memory->non_gc_memory, pcpo->char_name.data, "free storage for charproc naem");
316
            /* This causes PCL some trouble, don't know why yet FIXME-MEMORY
317
            gs_free_string(pdev->pdf_memory, (byte *)pcpo->char_name.data, pcpo->char_name.size, "Free CharProc name");*/
318
27.7k
            pcpo->char_name.data = (byte *)0L;
319
27.7k
            pcpo->char_name.size = 0;
320
27.7k
        }
321
153k
        gs_free_object(pdev->pdf_memory, pcpo, "Free CharProc");
322
153k
        pcpo = next;
323
153k
    }
324
3.72k
    return 0;
325
3.72k
}
326
327
/* Begin a CharProc for a synthesized (bitmap) font. */
328
int
329
pdf_begin_char_proc(gx_device_pdf * pdev, int w, int h, int x_width,
330
                    int y_offset, int x_offset, gs_id id, pdf_char_proc_t ** ppcp,
331
                    pdf_stream_position_t * ppos)
332
126k
{
333
126k
    gs_char char_code = 0;
334
126k
    pdf_bitmap_fonts_t *const pbfs = pdev->text->bitmap_fonts;
335
126k
    pdf_font_resource_t *font;
336
126k
    pdf_resource_t *pres;
337
126k
    pdf_char_proc_t *pcp;
338
126k
    int code;
339
    /* This code added to store PCL bitmap glyphs in type 3 fonts where possible */
340
126k
    gs_glyph glyph = GS_NO_GLYPH;
341
126k
    gs_const_string str2, *str = NULL;
342
126k
    gs_show_enum *show_enum = (gs_show_enum *)pdev->pte;
343
126k
    pdf_encoding_element_t *pet = 0;
344
    /* Since this is for text searching, its only useful if the character code
345
     * lies in an ASCII range, so we only handle some kinds of text layout.
346
     */
347
126k
    int allowed_op = (show_enum->text.operation &
348
126k
        (TEXT_FROM_STRING | TEXT_FROM_BYTES | TEXT_FROM_CHARS | TEXT_FROM_SINGLE_CHAR));
349
350
    /* Check to see the current font is a type 3. We can get here if pdfwrite decides
351
     * it can't handle a font type, and renders to a bitmap instead. If that's the
352
     * case then we can't add the bitmap to the existing font (its not a type 3 font)
353
     * and must fall back to holding it in our fallback type 3 font 'collection'.
354
     */
355
    /* Because the bitmaps are stored directly in the cache they already have any
356
     * effects caused by non-identity FontMatrix entries applied. So if the type 3
357
     * font we created has a non-identity FontMatrix we can't use it and must
358
     * go back to collecting the bitmap into our fallback font.
359
     */
360
126k
    if ((show_enum->current_font->FontType == ft_user_defined ||
361
126k
        show_enum->current_font->FontType == ft_PDF_user_defined ||
362
126k
        show_enum->current_font->FontType == ft_PCL_user_defined ||
363
126k
        show_enum->current_font->FontType == ft_MicroType ||
364
126k
        show_enum->current_font->FontType == ft_GL2_stick_user_defined ||
365
126k
        show_enum->current_font->FontType == ft_GL2_531) && allowed_op &&
366
126k
        show_enum->current_font->FontMatrix.xx == 1 && show_enum->current_font->FontMatrix.xy == 0 &&
367
126k
        show_enum->current_font->FontMatrix.yx == 0 && show_enum->current_font->FontMatrix.yy == 1) {
368
0
        pdf_char_proc_ownership_t *pcpo;
369
370
0
        gs_font_base *base = (gs_font_base *)show_enum->current_font;
371
0
        code = pdf_attached_font_resource(pdev, show_enum->current_font, &font, NULL, NULL, NULL, NULL);
372
0
        if (code < 0)
373
0
            return code;
374
0
        if (font == NULL)
375
0
            return_error(gs_error_invalidfont);
376
377
        /* The text processing will have run past the glyph, so we need to 'back up'
378
         * by one and get it again in order to get the character code and glyph, and update
379
         * the pointer correctly.
380
         */
381
0
        show_enum->index--;
382
0
        code = gs_default_next_char_glyph((gs_text_enum_t *)show_enum, (gs_char *)&char_code, &glyph);
383
0
        if (code < 0)
384
0
            return code;
385
386
        /* If the returned character code is outside the possible Encoding for
387
         * a type 3 font, then set pet to NULL, this means we will fall back to
388
         * the 'collection' font, as pet is checked below.
389
         */
390
0
        if ((int)char_code >= 0 && (int)char_code <= 255) {
391
0
            pet = &font->u.simple.Encoding[char_code];
392
0
            if (pet) {
393
                /* Check to see if we *already* have this glyph in this font. If
394
                 * we do then we can't add it to this font. Setting pet to 0
395
                 * falls back to the collection method.
396
                 */
397
0
                for (pcpo = font->u.simple.s.type3.char_procs; pcpo != NULL; pcpo = pcpo->char_next) {
398
0
                    if (pcpo->glyph == pet->glyph && pcpo->char_code == char_code) {
399
0
                        pet = 0x00;
400
0
                        break;
401
0
                    }
402
0
                }
403
0
            }
404
0
        }
405
0
        else
406
0
            pet = 0x00;
407
408
        /* We need a glyph name for the type 3 font's Encoding, if we haven't got one
409
         * then we need to give up, something about the font or text is not acceptable
410
         * (see various comments above).
411
         */
412
0
        if (pet && pet->glyph != GS_NO_GLYPH && !(pet->size == 7 &&
413
0
            !strncmp((const char *)pet->data, ".notdef", 7))) {
414
0
            if (char_code < font->u.simple.FirstChar)
415
0
                font->u.simple.FirstChar = char_code;
416
0
            if ((int)char_code > font->u.simple.LastChar)
417
0
                font->u.simple.LastChar = char_code;
418
0
            base->FontBBox.q.x = max(base->FontBBox.q.x, w);
419
0
            base->FontBBox.q.y = max(base->FontBBox.q.y, y_offset + h);
420
0
            str2.data = pet->data;
421
0
            str2.size = pet->size;
422
0
            str = &str2;
423
0
            glyph = pet->glyph;
424
            /* This is to work around a weird Acrobat bug. If the Encoding of a type 3
425
             * (possibly other types) is simply a standard encoding (eg WinAnsiEncoding)
426
             * then Acrobat 4 & 8 just ignore the glyphs altogether. This forces us to write
427
             * all the used glyphs as /Differencess, and that makes it work <sigh>
428
             */
429
0
            pet->is_difference = 1;
430
0
            font->Widths[char_code] = psdf_round(pdev->char_width.x, 100, 10); /* See
431
                        pdf_write_Widths about rounding. We need to provide
432
                        a compatible data for Tj. */
433
0
        } else {
434
0
            char_code = assign_char_code(pdev, pdev->pte);
435
0
            font = pbfs->open_font; /* Type 3 */
436
0
        }
437
126k
    } else {
438
126k
        char_code = assign_char_code(pdev, pdev->pte);
439
126k
        font = pbfs->open_font; /* Type 3 */
440
126k
    }
441
442
126k
    code = pdf_begin_resource(pdev, resourceCharProc, id, &pres);
443
126k
    if (code < 0)
444
0
        return code;
445
126k
    pcp = (pdf_char_proc_t *) pres;
446
126k
    code = pdf_attach_charproc(pdev, font, pcp, glyph, char_code, str);
447
126k
    if (code < 0)
448
0
        return code;
449
126k
    pres->object->written = true;
450
126k
    {
451
126k
        stream *s = pdev->strm;
452
453
        /*
454
         * The resource file is positionable, so rather than use an
455
         * object reference for the length, we'll go back and fill it in
456
         * at the end of the definition.  Take 1M as the longest
457
         * definition we can handle.  (This used to be 10K, but there was
458
         * a real file that exceeded this limit.)
459
         */
460
126k
        stream_puts(s, "<</Length       >>stream\n");
461
126k
        ppos->start_pos = stell(s);
462
126k
    }
463
126k
    code = pdf_begin_encrypt(pdev, &pdev->strm, pres->object->id);
464
126k
    if (code < 0)
465
0
        return code;
466
126k
    pcp->y_offset = y_offset;
467
126k
    pcp->x_offset = x_offset;
468
126k
    font->u.simple.s.type3.FontBBox.q.x =
469
126k
        max(font->u.simple.s.type3.FontBBox.q.x, w);
470
126k
    font->u.simple.s.type3.FontBBox.q.y =
471
126k
        max(font->u.simple.s.type3.FontBBox.q.y, y_offset + h);
472
126k
    font->u.simple.s.type3.max_y_offset =
473
126k
        max(font->u.simple.s.type3.max_y_offset, h + (h >> 2));
474
126k
    pcp->real_width.x = w;
475
126k
    pcp->real_width.y = y_offset + h;
476
126k
    *ppcp = pcp;
477
126k
    return 0;
478
126k
}
479
480
/* End a CharProc. */
481
int
482
pdf_end_char_proc(gx_device_pdf * pdev, pdf_stream_position_t * ppos)
483
126k
{
484
126k
    stream *s;
485
126k
    gs_offset_t start_pos, end_pos, length;
486
487
126k
    if (pdf_end_encrypt(pdev))
488
0
        s_close_filters(&pdev->strm, pdev->strm->strm);
489
490
126k
    s = pdev->strm;
491
126k
    start_pos = ppos->start_pos;
492
126k
    end_pos = stell(s);
493
126k
    length = end_pos - start_pos;
494
126k
    if (length > 999999)
495
0
        return_error(gs_error_limitcheck);
496
126k
    sseek(s, start_pos - 15);
497
126k
    pprintd1(s, "%d", length);
498
126k
    sseek(s, end_pos);
499
126k
    if (pdev->PDFA != 0)
500
0
        stream_puts(s, "\n");
501
126k
    stream_puts(s, "endstream\n");
502
126k
    pdf_end_separate(pdev, resourceCharProc);
503
126k
    return 0;
504
126k
}
505
506
/* Mark glyph names for garbager. */
507
void
508
pdf_mark_glyph_names(const pdf_font_resource_t *pdfont, const gs_memory_t *memory)
509
50.4k
{
510
50.4k
    if (pdfont->mark_glyph == NULL) {
511
        /* Synthesised bitmap fonts pass here. */
512
47.3k
        return;
513
47.3k
    }
514
3.05k
    if (pdfont->u.simple.Encoding != NULL) {
515
3.05k
         int i;
516
517
785k
         for (i = 0; i < 256; i++)
518
782k
             if (pdfont->u.simple.Encoding[i].glyph != GS_NO_GLYPH)
519
115k
                pdfont->mark_glyph(memory, pdfont->u.simple.Encoding[i].glyph, pdfont->mark_glyph_data);
520
3.05k
     }
521
3.05k
    if (pdfont->FontType == ft_user_defined ||
522
3.05k
        pdfont->FontType == ft_PDF_user_defined ||
523
3.05k
        pdfont->FontType == ft_PCL_user_defined ||
524
3.05k
        pdfont->FontType == ft_MicroType ||
525
3.05k
        pdfont->FontType == ft_GL2_stick_user_defined ||
526
3.05k
        pdfont->FontType == ft_GL2_531) {
527
599
        const pdf_char_proc_ownership_t *pcpo = pdfont->u.simple.s.type3.char_procs;
528
529
14.6k
        for (; pcpo != NULL; pcpo = pcpo->font_next)
530
14.0k
            pdfont->mark_glyph(memory, pcpo->glyph, pdfont->mark_glyph_data);
531
599
    }
532
3.05k
}
533
534
/* Put out a reference to an image as a character in a synthesized font. */
535
int
536
pdf_do_char_image(gx_device_pdf * pdev, const pdf_char_proc_t * pcp,
537
                  const gs_matrix * pimat)
538
1.69M
{
539
    /* We need to choose a font, which use the charproc.
540
       In most cases it is the last font, which the charproc is attached to.
541
       If the charproc is substituted, it causes a font change. */
542
1.69M
    const pdf_char_proc_ownership_t * pcpo = pcp->owner_fonts;
543
1.69M
    pdf_font_resource_t *pdfont = pcpo->font;
544
1.69M
    byte ch = pcpo->char_code;
545
1.69M
    pdf_text_state_values_t values;
546
547
1.69M
    values.character_spacing = 0;
548
1.69M
    values.pdfont = pdfont;
549
1.69M
    values.size = 1;
550
1.69M
    values.matrix = *pimat;
551
1.69M
    values.render_mode = pdev->pte->pgs->text_rendering_mode;
552
1.69M
    values.word_spacing = 0;
553
1.69M
    pdf_set_text_state_values(pdev, &values);
554
1.69M
    pdf_bitmap_char_update_bbox(pdev, pcp->x_offset, pcp->y_offset, pcp->real_width.x, pcp->real_width.y);
555
1.69M
    pdf_append_chars(pdev, &ch, 1, pdfont->Widths[ch] * pimat->xx, 0.0, false);
556
1.69M
    return 0;
557
1.69M
}
558
559
/*
560
 * Write the Encoding for bitmap fonts, if needed.
561
 */
562
int
563
pdf_write_bitmap_fonts_Encoding(gx_device_pdf *pdev)
564
50.0k
{
565
50.0k
    pdf_bitmap_fonts_t *pbfs;
566
567
50.0k
    if (pdev->text == NULL || pdev->text->bitmap_fonts == NULL)
568
0
        return 0;
569
570
50.0k
    pbfs = pdev->text->bitmap_fonts;
571
572
50.0k
    if (pbfs->bitmap_encoding_id) {
573
1.75k
        stream *s;
574
1.75k
        int i;
575
576
1.75k
        pdf_open_separate(pdev, pbfs->bitmap_encoding_id, resourceEncoding);
577
1.75k
        s = pdev->strm;
578
        /*
579
         * Even though the PDF reference documentation says that a
580
         * BaseEncoding key is required unless the encoding is
581
         * "based on the base font's encoding" (and there is no base
582
         * font in this case), Acrobat 2.1 gives an error if the
583
         * BaseEncoding key is present.
584
         */
585
1.75k
        stream_puts(s, "<</Type/Encoding/Differences[0");
586
133k
        for (i = 0; i <= pbfs->max_embedded_code; ++i) {
587
131k
            if (!(i & 15))
588
9.09k
                stream_puts(s, "\n");
589
131k
            pprintd1(s, "/a%d", i);
590
131k
        }
591
1.75k
        stream_puts(s, "\n] >>\n");
592
1.75k
        pdf_end_separate(pdev, resourceEncoding);
593
1.75k
        pbfs->bitmap_encoding_id = 0;
594
1.75k
    }
595
50.0k
    return 0;
596
50.0k
}
597
598
/*
599
 * Start charproc accumulation for a Type 3 font.
600
 */
601
int
602
pdf_start_charproc_accum(gx_device_pdf *pdev)
603
173k
{
604
173k
    pdf_char_proc_t *pcp;
605
173k
    pdf_resource_t *pres;
606
173k
    int id = gs_next_ids(pdev->memory, 1);
607
173k
    int code = pdf_enter_substream(pdev, resourceCharProc, id,
608
173k
                                   &pres, false, pdev->CompressFonts);
609
610
173k
    if (code < 0)
611
1
       return code;
612
173k
    pres->rid = id;
613
173k
    pcp = (pdf_char_proc_t *)pres;
614
173k
    pcp->owner_fonts = NULL;
615
173k
    return 0;
616
173k
}
617
618
/*
619
 * Install charproc accumulator for a Type 3 font.
620
 */
621
int
622
pdf_set_charproc_attrs(gx_device_pdf *pdev, gs_font *font, double *pw, int narg,
623
                gs_text_cache_control_t control, gs_char ch, bool scale_100)
624
155k
{
625
155k
    pdf_font_resource_t *pdfont;
626
155k
    pdf_resource_t *pres = pdev->accumulating_substream_resource;
627
155k
    pdf_char_proc_t *pcp;
628
155k
    int code;
629
630
155k
    if (pres == NULL)
631
0
        return_error(gs_error_undefined);
632
155k
    code = pdf_attached_font_resource(pdev, font, &pdfont, NULL, NULL, NULL, NULL);
633
155k
    if (code < 0)
634
0
        return code;
635
155k
    pcp = (pdf_char_proc_t *)pres;
636
155k
    pcp->owner_fonts = NULL;
637
155k
    pcp->real_width.x = pw[font->WMode && narg > 6 ? 6 : 0];
638
155k
    pcp->real_width.y = pw[font->WMode && narg > 6 ? 7 : 1];
639
155k
    pcp->v.x = (narg > 8 ? pw[8] : 0);
640
155k
    pcp->v.y = (narg > 8 ? pw[9] : 0);
641
155k
    if (control == TEXT_SET_CHAR_WIDTH) {
642
        /* PLRM 5.7.1 "BuildGlyph" reads : "Normally, it is unnecessary and
643
        undesirable to initialize the current color parameter, because show
644
        is defined to paint glyphs with the current color."
645
        However comparefiles/Bug687044.ps doesn't follow that. */
646
144k
        pdev->skip_colors = false;
647
144k
        pprintg1(pdev->strm, "%g 0 d0\n", (float)pw[0]);
648
        /* The colour change described above can't affect PCL fonts and we need
649
         * all glyphs to be noted as cached in order for the bitmap font cache
650
         * probing to work properly.
651
         */
652
144k
        if (font->FontType == ft_PCL_user_defined || font->FontType == ft_GL2_stick_user_defined
653
144k
            || font->FontType == ft_GL2_531 || font->FontType == ft_MicroType)
654
0
            pdfont->u.simple.s.type3.cached[ch >> 3] |= 0x80 >> (ch & 7);
655
144k
    } else {
656
11.0k
        double d;
657
11.0k
        pdev->skip_colors = true;
658
11.0k
        if (pw[4] < pw[2]) {
659
24
            d = pw[2];
660
24
            pw[2] = pw[4];
661
24
            pw[4] = d;
662
24
        }
663
11.0k
        if (pw[5] < pw[3]) {
664
39
            d = pw[5];
665
39
            pw[5] = pw[3];
666
39
            pw[3] = d;
667
39
        }
668
11.0k
        pprintg6(pdev->strm, "%g %g %g %g %g %g d1\n",
669
11.0k
            (float)pw[0], (float)0.0, (float)pw[2],
670
11.0k
            (float)pw[3], (float)pw[4], (float)pw[5]);
671
11.0k
        pdfont->u.simple.s.type3.cached[ch >> 3] |= 0x80 >> (ch & 7);
672
11.0k
    }
673
    /* See comments in pdf_text_process regarding type 3 CharProc accumulation
674
     * Initially this matrix was emitted there, at the start of the accumulator
675
     * but if we do that then GS incorrectly applied the matrix to the 'd1'
676
     * operator. We write the scale matrix here because this is *after* the
677
     * 'd1' has been emitted above, and so does not affect it.
678
     */
679
155k
    if (scale_100) {
680
155k
    code = stream_puts(pdev->strm, "0.01 0 0 0.01 0 0 cm\n");
681
155k
    if (code < 0)
682
0
        return code;
683
155k
    }
684
155k
    return 0;
685
155k
}
686
687
/*
688
 * Open a stream object in the temporary file.
689
 */
690
691
int
692
pdf_open_aside(gx_device_pdf *pdev, pdf_resource_type_t rtype,
693
        gs_id id, pdf_resource_t **ppres, bool reserve_object_id, int options)
694
320k
{
695
320k
    int code;
696
320k
    pdf_resource_t *pres;
697
320k
    stream *s, *save_strm = pdev->strm;
698
320k
    pdf_data_writer_t writer;
699
320k
    static const pdf_filter_names_t fnames = {
700
320k
        PDF_FILTER_NAMES
701
320k
    };
702
703
320k
    pdev->streams.save_strm = pdev->strm;
704
705
320k
    if (rtype >= NUM_RESOURCE_TYPES)
706
85.9k
        rtype = resourceOther;
707
320k
    code = pdf_alloc_aside(pdev, PDF_RESOURCE_CHAIN(pdev, rtype, id),
708
320k
                pdf_resource_type_structs[rtype], &pres, reserve_object_id ? 0 : -1);
709
320k
    if (code < 0)
710
0
        return code;
711
320k
    cos_become(pres->object, cos_type_stream);
712
320k
    s = cos_write_stream_alloc((cos_stream_t *)pres->object, pdev, "pdf_enter_substream");
713
320k
    if (s == 0)
714
0
        return_error(gs_error_VMerror);
715
320k
    pdev->strm = s;
716
320k
    code = pdf_append_data_stream_filters(pdev, &writer,
717
320k
                             options | DATA_STREAM_NOLENGTH, pres->object->id);
718
320k
    if (code < 0) {
719
0
        pdev->strm = save_strm;
720
0
        return code;
721
0
    }
722
320k
    code = pdf_put_filters((cos_dict_t *)pres->object, pdev, writer.binary.strm, &fnames);
723
320k
    if (code < 0) {
724
0
        pdev->strm = save_strm;
725
0
        return code;
726
0
    }
727
320k
    pdev->strm = writer.binary.strm;
728
320k
    *ppres = pres;
729
320k
    return 0;
730
320k
}
731
732
/*
733
 * Close a stream object in the temporary file.
734
 */
735
int
736
pdf_close_aside(gx_device_pdf *pdev)
737
320k
{
738
    /* We should call pdf_end_data here, but we don't want to put pdf_data_writer_t
739
       into pdf_substream_save stack to simplify garbager descriptors.
740
       Use a lower level functions instead that. */
741
320k
    stream *s = pdev->strm;
742
320k
    cos_stream_t *pcs = cos_stream_from_pipeline(s);
743
320k
    int status = s_close_filters(&s, NULL);
744
745
320k
    pdev->strm = pdev->streams.save_strm;
746
320k
    if (status < 0)
747
0
         return(gs_note_error(gs_error_ioerror));
748
749
320k
    if (!pcs)
750
0
        return gs_note_error(gs_error_ioerror);
751
752
320k
    pcs->is_open = false;
753
320k
    return 0;
754
320k
}
755
756
/*
757
 * Enter the substream accumulation mode.
758
 */
759
int
760
pdf_enter_substream(gx_device_pdf *pdev, pdf_resource_type_t rtype,
761
        gs_id id, pdf_resource_t **ppres, bool reserve_object_id, bool compress)
762
235k
{
763
235k
    int sbstack_ptr = pdev->sbstack_depth;
764
235k
    pdf_resource_t *pres;
765
235k
    stream *save_strm = pdev->strm;
766
235k
    int code;
767
768
235k
    if (pdev->sbstack_depth >= pdev->sbstack_size)
769
1
        return_error(gs_error_unregistered); /* Must not happen. */
770
235k
    if (pdev->sbstack[sbstack_ptr].text_state == 0) {
771
235k
        pdev->sbstack[sbstack_ptr].text_state = pdf_text_state_alloc(pdev->pdf_memory);
772
235k
        if (pdev->sbstack[sbstack_ptr].text_state == 0)
773
0
            return_error(gs_error_VMerror);
774
235k
    }
775
235k
    code = pdf_open_aside(pdev, rtype, id, &pres, reserve_object_id,
776
235k
                    (compress ? DATA_STREAM_COMPRESS : 0));
777
235k
    if (code < 0)
778
0
        return code;
779
235k
    code = pdf_save_viewer_state(pdev, NULL);
780
235k
    if (code < 0) {
781
0
        pdev->strm = save_strm;
782
0
        return code;
783
0
    }
784
235k
    pdev->sbstack[sbstack_ptr].context = pdev->context;
785
235k
    pdf_text_state_copy(pdev->sbstack[sbstack_ptr].text_state, pdev->text->text_state);
786
235k
    pdf_set_text_state_default(pdev->text->text_state);
787
235k
    pdev->sbstack[sbstack_ptr].clip_path = pdev->clip_path;
788
235k
    pdev->clip_path = 0;
789
235k
    pdev->sbstack[sbstack_ptr].clip_path_id = pdev->clip_path_id;
790
235k
    pdev->clip_path_id = pdev->no_clip_path_id;
791
235k
    pdev->sbstack[sbstack_ptr].vgstack_bottom = pdev->vgstack_bottom;
792
235k
    pdev->vgstack_bottom = pdev->vgstack_depth;
793
235k
    pdev->sbstack[sbstack_ptr].strm = save_strm;
794
235k
    pdev->sbstack[sbstack_ptr].procsets = pdev->procsets;
795
235k
    pdev->sbstack[sbstack_ptr].substream_Resources = pdev->substream_Resources;
796
235k
    pdev->sbstack[sbstack_ptr].skip_colors = pdev->skip_colors;
797
235k
    pdev->sbstack[sbstack_ptr].font3 = pdev->font3;
798
235k
    pdev->sbstack[sbstack_ptr].accumulating_substream_resource = pdev->accumulating_substream_resource;
799
235k
    pdev->sbstack[sbstack_ptr].charproc_just_accumulated = pdev->charproc_just_accumulated;
800
235k
    pdev->sbstack[sbstack_ptr].accumulating_a_global_object = pdev->accumulating_a_global_object;
801
235k
    pdev->sbstack[sbstack_ptr].pres_soft_mask_dict = pdev->pres_soft_mask_dict;
802
235k
    pdev->sbstack[sbstack_ptr].objname = pdev->objname;
803
235k
    pdev->sbstack[sbstack_ptr].last_charpath_op = pdev->last_charpath_op;
804
235k
    pdev->skip_colors = false;
805
235k
    pdev->charproc_just_accumulated = false;
806
235k
    pdev->pres_soft_mask_dict = NULL;
807
235k
    pdev->objname.data = NULL;
808
235k
    pdev->objname.size = 0;
809
    /* Do not reset pdev->accumulating_a_global_object - it inherits. */
810
235k
    pdev->sbstack_depth++;
811
235k
    pdev->procsets = 0;
812
235k
    pdev->font3 = 0;
813
235k
    pdev->context = PDF_IN_STREAM;
814
235k
    pdev->accumulating_substream_resource = pres;
815
235k
    pdev->last_charpath_op = 0;
816
    /* Do not alter type3charpath, inherit the current value. We need to know if */
817
    /* we are inside a charpath operation, and only reset this when the charpath */
818
    /* is complete */
819
235k
    if (rtype != resourceXObject)
820
214k
        pdf_reset_graphics(pdev);
821
20.2k
    else {
822
20.2k
        if (pdev->vg_initial_set)
823
20.2k
            pdev->state.blend_mode = pdev->vg_initial.blend_mode;
824
20.2k
    }
825
235k
    *ppres = pres;
826
235k
    return 0;
827
235k
}
828
829
/*
830
 * Exit the substream accumulation mode.
831
 */
832
int
833
pdf_exit_substream(gx_device_pdf *pdev)
834
235k
{
835
235k
    int code, code1;
836
235k
    int sbstack_ptr;
837
838
235k
    if (pdev->sbstack_depth <= 0)
839
2
        return_error(gs_error_unregistered); /* Must not happen. */
840
235k
    code = pdf_open_contents(pdev, PDF_IN_STREAM);
841
235k
    sbstack_ptr = pdev->sbstack_depth - 1;
842
254k
    while (pdev->vgstack_depth > pdev->vgstack_bottom) {
843
19.2k
        code1 = pdf_restore_viewer_state(pdev, pdev->strm);
844
19.2k
        if (code >= 0)
845
19.2k
            code = code1;
846
19.2k
    }
847
235k
    if (pdev->clip_path != 0)
848
19.2k
        gx_path_free(pdev->clip_path, "pdf_end_charproc_accum");
849
235k
    code1 = pdf_close_aside(pdev);
850
235k
    if (code1 < 0 && code >= 0)
851
0
        code = code1;
852
235k
    pdev->context = pdev->sbstack[sbstack_ptr].context;
853
235k
    pdf_text_state_copy(pdev->text->text_state, pdev->sbstack[sbstack_ptr].text_state);
854
235k
    gs_free_object(pdev->pdf_memory, pdev->sbstack[sbstack_ptr].text_state, "free text state for stream");
855
235k
    pdev->sbstack[sbstack_ptr].text_state = 0;
856
235k
    pdev->clip_path = pdev->sbstack[sbstack_ptr].clip_path;
857
235k
    pdev->sbstack[sbstack_ptr].clip_path = 0;
858
235k
    pdev->clip_path_id = pdev->sbstack[sbstack_ptr].clip_path_id;
859
235k
    pdev->vgstack_bottom = pdev->sbstack[sbstack_ptr].vgstack_bottom;
860
235k
    pdev->strm = pdev->sbstack[sbstack_ptr].strm;
861
235k
    pdev->sbstack[sbstack_ptr].strm = 0;
862
235k
    pdev->procsets = pdev->sbstack[sbstack_ptr].procsets;
863
235k
    pdev->substream_Resources = pdev->sbstack[sbstack_ptr].substream_Resources;
864
235k
    pdev->sbstack[sbstack_ptr].substream_Resources = 0;
865
235k
    pdev->skip_colors = pdev->sbstack[sbstack_ptr].skip_colors;
866
235k
    pdev->font3 = pdev->sbstack[sbstack_ptr].font3;
867
235k
    pdev->sbstack[sbstack_ptr].font3 = 0;
868
235k
    pdev->accumulating_substream_resource = pdev->sbstack[sbstack_ptr].accumulating_substream_resource;
869
235k
    pdev->sbstack[sbstack_ptr].accumulating_substream_resource = 0;
870
235k
    pdev->charproc_just_accumulated = pdev->sbstack[sbstack_ptr].charproc_just_accumulated;
871
235k
    pdev->accumulating_a_global_object = pdev->sbstack[sbstack_ptr].accumulating_a_global_object;
872
235k
    pdev->pres_soft_mask_dict = pdev->sbstack[sbstack_ptr].pres_soft_mask_dict;
873
235k
    pdev->objname = pdev->sbstack[sbstack_ptr].objname;
874
235k
    pdev->last_charpath_op = pdev->sbstack[sbstack_ptr].last_charpath_op;
875
235k
    pdev->sbstack_depth = sbstack_ptr;
876
235k
    code1 = pdf_restore_viewer_state(pdev, NULL);
877
235k
    if (code1 < 0 && code >= 0)
878
0
        code = code1;
879
235k
    return code;
880
235k
}
881
882
static bool
883
pdf_is_same_charproc_attrs1(gx_device_pdf *pdev, pdf_char_proc_t *pcp0, pdf_char_proc_t *pcp1)
884
366k
{
885
366k
    if (pcp0->real_width.x != pcp1->real_width.x)
886
182k
        return false;
887
183k
    if (pcp0->real_width.y != pcp1->real_width.y)
888
0
        return false;
889
183k
    if (pcp0->v.x != pcp1->v.x)
890
0
        return false;
891
183k
    if (pcp0->v.y != pcp1->v.y)
892
0
        return false;
893
183k
    return true;
894
183k
}
895
896
typedef struct charproc_compatibility_data_s {
897
    const pdf_char_glyph_pairs_t *cgp;
898
    pdf_font_resource_t *pdfont;
899
    gs_char char_code;
900
    gs_glyph glyph;
901
    gs_font *font;
902
} charproc_compatibility_data_t;
903
904
static bool
905
is_char_code_used(pdf_font_resource_t *pdfont, gs_char char_code)
906
212
{
907
212
    pdf_char_proc_ownership_t *pcpo;
908
909
1.54k
    for (pcpo = pdfont->u.simple.s.type3.char_procs; pcpo != NULL; pcpo = pcpo->char_next) {
910
1.33k
        if (pcpo->char_code == char_code) {
911
3
            return true;
912
3
        }
913
1.33k
    }
914
209
    return false;
915
212
}
916
917
static int
918
pdf_is_charproc_compatible(gx_device_pdf * pdev, pdf_resource_t *pres0, pdf_resource_t *pres1)
919
212
{
920
212
    charproc_compatibility_data_t *data = (charproc_compatibility_data_t *)pdev->find_resource_param;
921
212
    pdf_char_proc_t *pcp0 = (pdf_char_proc_t *)pres0;
922
212
    pdf_char_proc_t *pcp1 = (pdf_char_proc_t *)pres1;
923
212
    pdf_font_resource_t *pdfont = data->pdfont;
924
212
    pdf_char_proc_ownership_t *pcpo;
925
212
    pdf_font_cache_elem_t **e;
926
212
    bool can_add_to_current_font = false, computed_can_add_to_current_font = false;
927
928
    /* Does it have same attributes ? */
929
212
    if (!pdf_is_same_charproc_attrs1(pdev, pcp0, pcp1))
930
0
        return 0;
931
    /* Is it from same font ? */
932
2.42k
    for (pcpo = pcp1->owner_fonts; pcpo != NULL; pcpo = pcpo->char_next) {
933
2.21k
        if (pdfont == pcpo->font) {
934
            /* Check for encoding conflict. */
935
0
            if (pcpo->char_code == data->char_code && pcpo->glyph == data->glyph)
936
0
                return 1; /* Same char code. */
937
0
            if (!computed_can_add_to_current_font) {
938
0
                can_add_to_current_font = !is_char_code_used(pdfont, data->char_code);
939
0
                computed_can_add_to_current_font = true;
940
0
            }
941
0
            if (can_add_to_current_font)
942
0
                return 1; /* No conflict. */
943
0
        }
944
2.21k
    }
945
    /* Look for another font with same encoding,
946
       because we want to reduce the number of new fonts.
947
       We also restrict with ones attached to same PS font,
948
       otherwise it creates too mixed fonts and disturbs word breaks.
949
     */
950
212
    e = pdf_locate_font_cache_elem(pdev, data->font);
951
212
    if (e != NULL) {
952
2.42k
        for (pcpo = pcp1->owner_fonts; pcpo != NULL; pcpo = pcpo->char_next) {
953
2.21k
            if (pcpo->char_code != data->char_code || pcpo->glyph != data->glyph)
954
2.11k
                continue; /* Need same Encoding to generate a proper ToUnicode. */
955
107
            if (pdfont->u.simple.s.type3.bitmap_font != pcpo->font->u.simple.s.type3.bitmap_font)
956
0
                continue;
957
107
            if (gs_matrix_compare(&pdfont->u.simple.s.type3.FontMatrix, &pcpo->font->u.simple.s.type3.FontMatrix))
958
24
                continue;
959
83
            if (data->cgp != NULL) {
960
80
                if (!pdf_check_encoding_compatibility(pcpo->font, data->cgp->s, data->cgp->num_all_chars))
961
0
                    continue;
962
80
            }
963
83
            if ((*e)->pdfont != pcpo->font)
964
83
                continue;
965
0
            data->pdfont = pcpo->font; /* Switch to the other font. */
966
0
            return 1;
967
83
        }
968
212
    }
969
    /* Check whether it can be added into the current font. */
970
212
    if (!computed_can_add_to_current_font)
971
212
        can_add_to_current_font = !is_char_code_used(pdfont, data->char_code);
972
212
    if (!can_add_to_current_font) {
973
        /* Can't substitute due to encoding conflict. */
974
3
        return 0;
975
3
    }
976
    /* The current font will share it with another font. */
977
209
    return 1;
978
212
}
979
980
static int
981
pdf_find_same_charproc_aux(gx_device_pdf *pdev,
982
            pdf_font_resource_t **ppdfont, pdf_char_proc_t **ppcp)
983
173k
{
984
173k
    pdf_char_proc_ownership_t *pcpo;
985
173k
    int code;
986
987
    /* fixme: this passes parameters to pdf_is_charproc_compatible
988
       through special gx_device_pdf field pdev->find_resource_param
989
       due to prototype limitation of pdf_find_same_resource.
990
       It would be better to change the client data argument type in there to void. */
991
377k
    for (pcpo = (*ppdfont)->u.simple.s.type3.char_procs; pcpo != NULL; pcpo = pcpo->char_next) {
992
366k
        pdf_char_proc_t *pcp = pcpo->char_proc;
993
994
366k
        if (*ppcp != pcp && pdf_is_same_charproc_attrs1(pdev, *ppcp, pcp)) {
995
183k
            cos_object_t *pco0 = pcp->object;
996
183k
            cos_object_t *pco1 = (*ppcp)->object;
997
998
183k
            code = pco0->cos_procs->equal(pco0, pco1, pdev);
999
183k
            if (code < 0) {
1000
0
                return code;
1001
0
            }
1002
183k
            if (code) {
1003
162k
                *ppcp = pcp;
1004
162k
                return 1;
1005
162k
            }
1006
183k
        }
1007
366k
    }
1008
11.6k
    return pdf_find_same_resource(pdev, resourceCharProc, (pdf_resource_t **)ppcp, pdf_is_charproc_compatible);
1009
173k
}
1010
static int
1011
pdf_find_same_charproc(gx_device_pdf *pdev,
1012
            pdf_font_resource_t **ppdfont, const pdf_char_glyph_pairs_t *cgp,
1013
            pdf_char_proc_t **ppcp, gs_glyph glyph, gs_char char_code,
1014
            gs_font *font)
1015
173k
{
1016
173k
    charproc_compatibility_data_t data;
1017
173k
    int code;
1018
1019
173k
    data.cgp = cgp;
1020
173k
    data.pdfont = *ppdfont;
1021
173k
    data.char_code = char_code;
1022
173k
    data.glyph = glyph;
1023
173k
    data.font = font;
1024
173k
    pdev->find_resource_param = &data;
1025
173k
    code = pdf_find_same_charproc_aux(pdev, ppdfont, ppcp);
1026
173k
    pdev->find_resource_param = NULL;
1027
173k
    *ppdfont = data.pdfont;
1028
173k
    return code;
1029
173k
}
1030
1031
static bool
1032
pdf_is_charproc_defined(gx_device_pdf *pdev, pdf_font_resource_t *pdfont, gs_char ch)
1033
11.3k
{
1034
11.3k
    pdf_char_proc_ownership_t *pcpo;
1035
1036
152k
    for (pcpo = pdfont->u.simple.s.type3.char_procs; pcpo != NULL; pcpo = pcpo->char_next) {
1037
141k
        if (pcpo->char_code == ch)
1038
32
            return true;
1039
141k
    }
1040
11.3k
    return false;
1041
11.3k
}
1042
1043
static int
1044
complete_adding_char(gx_device_pdf *pdev, gs_font *font,
1045
                     gs_glyph glyph, gs_char ch, pdf_char_proc_t *pcp,
1046
                     const gs_const_string *gnstr)
1047
173k
{
1048
173k
    pdf_font_resource_t *pdfont;
1049
173k
    double *real_widths;
1050
173k
    byte *glyph_usage;
1051
173k
    int char_cache_size, width_cache_size;
1052
173k
    pdf_encoding_element_t *pet;
1053
173k
    int code;
1054
1055
173k
    code = pdf_attached_font_resource(pdev, font, &pdfont,
1056
173k
                &glyph_usage, &real_widths, &char_cache_size, &width_cache_size);
1057
173k
    if (code < 0)
1058
0
        return code;
1059
173k
    code = pdf_attach_charproc(pdev, pdfont, pcp, glyph, ch, gnstr);
1060
173k
    if (code < 0)
1061
0
        return code;
1062
173k
    if (ch >= char_cache_size || ch >= width_cache_size)
1063
0
        return_error(gs_error_unregistered); /* Must not happen. */
1064
173k
    pet = &pdfont->u.simple.Encoding[ch];
1065
173k
    pdfont->Widths[ch] = pcp->real_width.x;
1066
173k
    real_widths[ch * 2    ] = pcp->real_width.x;
1067
173k
    real_widths[ch * 2 + 1] = pcp->real_width.y;
1068
173k
    glyph_usage[ch / 8] |= 0x80 >> (ch & 7);
1069
173k
    pdfont->used[ch >> 3] |= 0x80 >> (ch & 7);
1070
173k
    if (pdfont->u.simple.v != NULL && font->WMode) {
1071
0
        pdfont->u.simple.v[ch].x = pcp->v.x;
1072
0
        pdfont->u.simple.v[ch].y = pcp->v.x;
1073
0
    }
1074
173k
    pet->glyph = glyph;
1075
173k
    pet->is_difference = true;
1076
173k
    if (pdfont->u.simple.LastChar < (int)ch)
1077
4.81k
        pdfont->u.simple.LastChar = (int)ch;
1078
173k
    if (pdfont->u.simple.FirstChar > (int)ch)
1079
3.29k
        pdfont->u.simple.FirstChar = (int)ch;
1080
173k
    return pdf_copy_string_to_encoding(pdev, (gs_const_string *)gnstr, pet);
1081
173k
}
1082
1083
static int
1084
pdf_char_widths_from_charprocs(gx_device_pdf *pdev, gs_font *font)
1085
0
{
1086
0
    pdf_font_resource_t *pdfont;
1087
0
    double *real_widths;
1088
0
    byte *glyph_usage;
1089
0
    int char_cache_size, width_cache_size;
1090
0
    pdf_char_proc_ownership_t *pcpo;
1091
0
    int code;
1092
1093
0
    code = pdf_attached_font_resource(pdev, font, &pdfont,
1094
0
                &glyph_usage, &real_widths, &char_cache_size, &width_cache_size);
1095
0
    if (code < 0)
1096
0
        return code;
1097
0
    for (pcpo = pdfont->u.simple.s.type3.char_procs; pcpo != NULL; pcpo = pcpo->char_next) {
1098
0
        pdf_char_proc_t *pcp = pcpo->char_proc;
1099
0
        gs_char ch = pcpo->char_code;
1100
1101
0
        real_widths[ch * 2    ] = pcp->real_width.x;
1102
0
        real_widths[ch * 2 + 1] = pcp->real_width.y;
1103
0
        glyph_usage[ch / 8] |= 0x80 >> (ch & 7);
1104
0
    }
1105
0
    return 0;
1106
0
}
1107
1108
/*
1109
 * Complete charproc accumulation for a Type 3 font.
1110
 */
1111
int
1112
pdf_end_charproc_accum(gx_device_pdf *pdev, gs_font *font, const pdf_char_glyph_pairs_t *cgp,
1113
                       gs_glyph glyph, gs_char output_char_code, const gs_const_string *gnstr)
1114
173k
{
1115
173k
    int code;
1116
173k
    pdf_resource_t *pres = (pdf_resource_t *)pdev->accumulating_substream_resource;
1117
    /* We could use pdfont->u.simple.s.type3.char_procs insted the thing above
1118
       unless the font is defined recursively.
1119
       But we don't want such assumption. */
1120
173k
    pdf_char_proc_t *pcp = (pdf_char_proc_t *)pres;
1121
173k
    pdf_font_resource_t *pdfont;
1122
173k
    gs_char ch = output_char_code;
1123
173k
    bool checking_glyph_variation = false;
1124
1125
173k
    if (ch == GS_NO_CHAR)
1126
0
        return_error(gs_error_unregistered); /* Must not happen. */
1127
173k
    if (ch >= 256)
1128
0
        return_error(gs_error_unregistered); /* Must not happen. */
1129
173k
    if (pres == NULL)
1130
0
        return_error(gs_error_unregistered); /* Must not happen. */
1131
173k
    code = pdf_attached_font_resource(pdev, font, &pdfont, NULL, NULL, NULL, NULL);
1132
173k
    if (code < 0)
1133
0
        return code;
1134
173k
    if (pdfont != (pdf_font_resource_t *)pdev->font3)
1135
15
        return_error(gs_error_unregistered); /* Must not happen. */
1136
173k
    code = pdf_exit_substream(pdev);
1137
173k
    if (code < 0)
1138
0
        return code;
1139
173k
    if (!(pdfont->used[ch >> 3] & (0x80 >> (ch & 7))) ||
1140
173k
        !(pdfont->u.simple.s.type3.cached[ch >> 3] & (0x80 >> (ch & 7)))) {
1141
        /* First appearence or not cached - check for duplicates. */
1142
173k
        pdf_font_resource_t *pdfont1 = pdfont;
1143
1144
173k
        checking_glyph_variation = true;
1145
        /* CAUTION : a possible font change. */
1146
173k
        code = pdf_find_same_charproc(pdev, &pdfont, cgp, &pcp, glyph, ch, font);
1147
173k
        if (code < 0)
1148
0
            return code;
1149
173k
        if (code != 0) {
1150
162k
            code = pdf_cancel_resource(pdev, pres, resourceCharProc);
1151
162k
            if (code < 0)
1152
0
                return code;
1153
162k
            pdf_forget_resource(pdev, pres, resourceCharProc);
1154
162k
            if (pdfont1 != pdfont) {
1155
0
                code = pdf_attach_font_resource(pdev, font, pdfont);
1156
0
                if (code < 0)
1157
0
                    return code;
1158
0
                code = pdf_char_widths_from_charprocs(pdev, font);
1159
0
                if (code < 0)
1160
0
                    return code;
1161
0
            }
1162
162k
            pdev->charproc_just_accumulated = true;
1163
162k
            return complete_adding_char(pdev, font, glyph, ch, pcp, gnstr);
1164
162k
        }
1165
11.3k
        if (pdf_is_charproc_defined(pdev, pdfont, ch)) {
1166
            /* Encoding conflict after a font change. */
1167
32
            gs_font *base_font = font, *below;
1168
1169
32
            while ((below = base_font->base) != base_font &&
1170
32
                    base_font->procs.same_font(base_font, below, FONT_SAME_OUTLINES))
1171
0
                base_font = below;
1172
32
            code = pdf_make_font3_resource(pdev, base_font, &pdfont);
1173
32
            if (code < 0)
1174
0
                return code;
1175
32
            code = pdf_attach_font_resource(pdev, font, pdfont);
1176
32
            if (code < 0)
1177
0
                return code;
1178
32
        }
1179
11.3k
    }
1180
11.4k
    pdf_reserve_object_id(pdev, pres, 0);
1181
11.4k
    if (checking_glyph_variation)
1182
11.3k
        pdev->charproc_just_accumulated = true;
1183
11.4k
    return complete_adding_char(pdev, font, glyph, ch, pcp, gnstr);
1184
173k
}
1185
1186
/* Add procsets to substream Resources. */
1187
int
1188
pdf_add_procsets(cos_dict_t *pcd, pdf_procset_t procsets)
1189
22.5k
{
1190
22.5k
    char str[5 + 7 + 7 + 7 + 5 + 2];
1191
22.5k
    cos_value_t v;
1192
1193
22.5k
    strcpy(str, "[/PDF");
1194
22.5k
    if (procsets & ImageB)
1195
0
        strcat(str, "/ImageB");
1196
22.5k
    if (procsets & ImageC)
1197
3.00k
        strcat(str, "/ImageC");
1198
22.5k
    if (procsets & ImageI)
1199
8
        strcat(str, "/ImageI");
1200
22.5k
    if (procsets & Text)
1201
423
        strcat(str, "/Text");
1202
22.5k
    strcat(str, "]");
1203
22.5k
    cos_string_value(&v, (byte *)str, strlen(str));
1204
22.5k
    return cos_dict_put_c_key(pcd, "/ProcSet", &v);
1205
22.5k
}
1206
1207
/* Add a resource to substream Resources. */
1208
int
1209
pdf_add_resource(gx_device_pdf *pdev, cos_dict_t *pcd, const char *key, pdf_resource_t *pres)
1210
1.91M
{
1211
1.91M
    if (pcd != 0) {
1212
71.9k
        const cos_value_t *v = cos_dict_find(pcd, (const byte *)key, strlen(key));
1213
71.9k
        cos_dict_t *list;
1214
71.9k
        int code;
1215
71.9k
        char buf[10 + (sizeof(long) * 8 / 3 + 1)], buf1[sizeof(pres->rname) + 1];
1216
1217
71.9k
        if (pdev->ForOPDFRead && !pres->global && pdev->accumulating_a_global_object) {
1218
0
            pres->global = true;
1219
0
            code = cos_dict_put_c_key_bool((cos_dict_t *)pres->object, "/.Global", true);
1220
0
            if (code < 0)
1221
0
                return code;
1222
0
        }
1223
71.9k
        gs_snprintf(buf, sizeof(buf), "%"PRId64" 0 R\n", pres->object->id);
1224
71.9k
        if (v != NULL) {
1225
48.0k
            if (v->value_type != COS_VALUE_OBJECT &&
1226
48.0k
                v->value_type != COS_VALUE_RESOURCE)
1227
0
                return_error(gs_error_unregistered); /* Must not happen. */
1228
48.0k
            list = (cos_dict_t *)v->contents.object;
1229
48.0k
            if (list->cos_procs != &cos_dict_procs)
1230
0
                return_error(gs_error_unregistered); /* Must not happen. */
1231
48.0k
        } else {
1232
23.9k
            list = cos_dict_alloc(pdev, "pdf_add_resource");
1233
23.9k
            if (list == NULL)
1234
0
                return_error(gs_error_VMerror);
1235
23.9k
            code = cos_dict_put_c_key_object((cos_dict_t *)pcd, key, (cos_object_t *)list);
1236
23.9k
            if (code < 0)
1237
0
                return code;
1238
23.9k
        }
1239
71.9k
        buf1[0] = '/';
1240
71.9k
        strcpy(buf1 + 1, pres->rname);
1241
71.9k
        return cos_dict_put_string(list, (const byte *)buf1, strlen(buf1),
1242
71.9k
                        (const byte *)buf, strlen(buf));
1243
71.9k
    }
1244
1.83M
    return 0;
1245
1.91M
}