Coverage Report

Created: 2026-04-09 07:06

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ghostpdl/devices/vector/gdevpdti.c
Line
Count
Source
1
/* Copyright (C) 2001-2025 Artifex Software, Inc.
2
   All Rights Reserved.
3
4
   This software is provided AS-IS with no warranty, either express or
5
   implied.
6
7
   This software is distributed under license and may not be copied,
8
   modified or distributed except as expressly authorized under the terms
9
   of the license contained in the file LICENSE in this distribution.
10
11
   Refer to licensing information at http://www.artifex.com or contact
12
   Artifex Software, Inc.,  39 Mesa Street, Suite 108A, San Francisco,
13
   CA 94129, USA, for further information.
14
*/
15
16
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
177k
{
72
177k
    return pdf_resource_id((const pdf_resource_t *)pcp);
73
177k
}
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
74.2k
{
79
74.2k
    pdf_bitmap_fonts_t *pbfs = pdev->text->bitmap_fonts;
80
74.2k
    pdf_font_resource_t *pdfont = pbfs->open_font; /* Type 3 */
81
74.2k
    int i, c = 0, code;
82
74.2k
    uint operation = pte->text.operation;
83
84
74.2k
    if (pbfs->bitmap_encoding_id == 0)
85
1.50k
        pbfs->bitmap_encoding_id = pdf_obj_ref(pdev);
86
74.2k
    if (pdfont == 0 || pdfont->u.simple.LastChar == 255 ||
87
72.7k
        !pbfs->use_open_font
88
74.2k
        ) {
89
        /* Start a new synthesized font. */
90
1.72k
        char *pc;
91
92
1.72k
        code = pdf_font_type3_alloc(pdev, &pdfont, pdf_write_contents_bitmap);
93
1.72k
        if (code < 0)
94
0
            return code;
95
1.72k
        pdfont->u.simple.s.type3.bitmap_font = true;
96
1.72k
        if (pbfs->open_font == 0)
97
1.50k
            pdfont->rname[0] = 0;
98
226
        else
99
226
            strcpy(pdfont->rname, pbfs->open_font->rname);
100
1.72k
        pdfont->u.simple.s.type3.FontBBox.p.x = 0;
101
1.72k
        pdfont->u.simple.s.type3.FontBBox.p.y = 0;
102
1.72k
        pdfont->u.simple.s.type3.FontBBox.q.x = 0;
103
1.72k
        pdfont->u.simple.s.type3.FontBBox.q.y = 0;
104
1.72k
        pdfont->mark_glyph = NULL;
105
1.72k
        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
1.72k
        for (pc = pdfont->rname; *pc == 'Z'; ++pc)
111
0
            *pc = '@';
112
1.72k
        if ((*pc)++ == 0)
113
1.50k
            *pc = 'A', pc[1] = 0;
114
1.72k
        pbfs->open_font = pdfont;
115
1.72k
        pbfs->use_open_font = true;
116
1.72k
        pdfont->u.simple.FirstChar = 255;
117
1.72k
    }
118
74.2k
    if ((operation & (TEXT_FROM_STRING | TEXT_FROM_BYTES)) ||
119
74.2k
        (operation & (TEXT_FROM_CHARS | TEXT_FROM_SINGLE_CHAR))) {
120
74.2k
        unsigned char p = *pte->text.data.bytes;
121
74.2k
        unsigned char index = p / 8, bit = 0x01 << (p % 8);
122
123
74.2k
        if (pdfont->used[index] & bit) {
124
4.93M
            for (i = 0;i < 256;i++) {
125
4.93M
                index = i / 8;
126
4.93M
                bit = 0x01 << (i % 8);
127
4.93M
                if (!(pdfont->used[index] & bit)) {
128
66.7k
                    c = i;
129
66.7k
                    break;
130
66.7k
                }
131
4.93M
            }
132
66.7k
        } else
133
7.53k
            c = p;
134
74.2k
        pdfont->used[index] |= bit;
135
74.2k
        if (c > pdfont->u.simple.LastChar)
136
49.9k
            pdfont->u.simple.LastChar = c;
137
138
74.2k
    } 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
74.2k
    if (c < pdfont->u.simple.FirstChar)
146
2.68k
        pdfont->u.simple.FirstChar = c;
147
148
74.2k
    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
74.2k
    if (c > pbfs->max_embedded_code)
152
41.7k
        pbfs->max_embedded_code = c;
153
154
74.2k
    return c;
155
74.2k
}
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
2.69k
{
161
2.69k
    stream *s = pdev->strm;
162
2.69k
    const pdf_char_proc_ownership_t *pcpo;
163
2.69k
    int64_t diff_id = 0;
164
2.69k
    int code;
165
166
2.69k
    if (pdfont->u.simple.s.type3.bitmap_font)
167
1.72k
        diff_id = pdev->text->bitmap_fonts->bitmap_encoding_id;
168
965
    else {
169
        /* See comment in pdf_write_encoding. */
170
965
        diff_id = pdf_obj_ref(pdev);
171
965
    }
172
2.69k
    code = pdf_write_encoding_ref(pdev, pdfont, diff_id);
173
2.69k
    if (code < 0)
174
0
        return code;
175
2.69k
    stream_puts(s, "/CharProcs <<");
176
    /* Write real characters. */
177
91.5k
    for (pcpo = pdfont->u.simple.s.type3.char_procs; pcpo;
178
88.8k
         pcpo = pcpo->char_next
179
88.8k
         ) {
180
88.8k
        if (pdfont->u.simple.s.type3.bitmap_font)
181
74.2k
            pprinti64d2(s, "/a%"PRId64" %"PRId64" 0 R\n", pcpo->char_code,
182
74.2k
                      pdf_char_proc_id(pcpo->char_proc));
183
14.5k
        else if (!pcpo-> duplicate_char_name) {
184
14.4k
            pdf_put_name(pdev, pcpo->char_name.data, pcpo->char_name.size);
185
14.4k
            pprinti64d1(s, " %"PRId64" 0 R\n", pdf_char_proc_id(pcpo->char_proc));
186
14.4k
        }
187
88.8k
        pdf_record_usage_by_parent(pdev, pdf_char_proc_id(pcpo->char_proc), pdfont->object->id);
188
88.8k
    }
189
2.69k
    stream_puts(s, ">>");
190
2.69k
    pprintg6(s, "/FontMatrix[%g %g %g %g %g %g]",
191
2.69k
            (float)pdfont->u.simple.s.type3.FontMatrix.xx,
192
2.69k
            (float)pdfont->u.simple.s.type3.FontMatrix.xy,
193
2.69k
            (float)pdfont->u.simple.s.type3.FontMatrix.yx,
194
2.69k
            (float)pdfont->u.simple.s.type3.FontMatrix.yy,
195
2.69k
            (float)pdfont->u.simple.s.type3.FontMatrix.tx,
196
2.69k
            (float)pdfont->u.simple.s.type3.FontMatrix.ty);
197
2.69k
    code = pdf_finish_write_contents_type3(pdev, pdfont);
198
2.69k
    if (code < 0)
199
0
        return code;
200
2.69k
    if (!pdfont->u.simple.s.type3.bitmap_font && diff_id > 0) {
201
965
        code = pdf_write_encoding(pdev, pdfont, diff_id, 0);
202
965
        if (code < 0)
203
0
            return code;
204
965
    }
205
2.69k
    return 0;
206
2.69k
}
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
32.9k
{
216
32.9k
    pdf_bitmap_fonts_t *pbfs =
217
32.9k
        gs_alloc_struct(mem, pdf_bitmap_fonts_t, &st_pdf_bitmap_fonts,
218
32.9k
                        "pdf_bitmap_fonts_alloc");
219
220
32.9k
    if (pbfs == 0)
221
0
        return 0;
222
32.9k
    memset(pbfs, 0, sizeof(*pbfs));
223
32.9k
    pbfs->max_embedded_code = -1;
224
32.9k
    return pbfs;
225
32.9k
}
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
42.6k
{
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
42.6k
    if (pdev->CompatibilityLevel <= 1.2)
243
30.2k
        pdev->text->bitmap_fonts->use_open_font = false;
244
42.6k
}
245
246
int
247
pdf_charproc_y_offset(pdf_char_proc_t *pcp)
248
722k
{
249
722k
    return pcp->y_offset;
250
722k
}
251
252
int
253
pdf_charproc_x_offset(pdf_char_proc_t *pcp)
254
722k
{
255
722k
    return pcp->x_offset;
256
722k
}
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
159k
{
263
159k
    pdf_char_proc_ownership_t *pcpo;
264
159k
    bool duplicate_char_name = false;
265
266
7.53M
    for (pcpo = pdfont->u.simple.s.type3.char_procs; pcpo != NULL; pcpo = pcpo->char_next) {
267
7.44M
        if (pcpo->glyph == glyph && pcpo->char_code == char_code)
268
67.2k
            return 0;
269
7.44M
    }
270
92.0k
    if (!pdfont->u.simple.s.type3.bitmap_font) {
271
568k
        for (pcpo = pdfont->u.simple.s.type3.char_procs; pcpo != NULL; pcpo = pcpo->char_next) {
272
551k
            if (!bytes_compare(pcpo->char_name.data, pcpo->char_name.size, gnstr->data, gnstr->size)) {
273
38
                duplicate_char_name = true;
274
38
                break;
275
38
            }
276
551k
        }
277
17.7k
    }
278
92.0k
    pcpo = gs_alloc_struct(pdev->pdf_memory,
279
92.0k
            pdf_char_proc_ownership_t, &st_pdf_char_proc_ownership, "pdf_attach_charproc");
280
281
92.0k
    if (pcpo == NULL)
282
0
        return_error(gs_error_VMerror);
283
92.0k
    pcpo->font = pdfont;
284
92.0k
    pcpo->char_next = pdfont->u.simple.s.type3.char_procs;
285
92.0k
    pdfont->u.simple.s.type3.char_procs = pcpo;
286
92.0k
    pcpo->char_proc = pcp;
287
92.0k
    pcpo->font_next = pcp->owner_fonts;
288
92.0k
    pcp->owner_fonts = pcpo;
289
92.0k
    pcpo->char_code = char_code;
290
92.0k
    pcpo->glyph = glyph;
291
92.0k
    if (gnstr == NULL) {
292
74.2k
        pcpo->char_name.data = 0;
293
74.2k
        pcpo->char_name.size = 0;
294
74.2k
    } else {
295
17.7k
        if (gnstr->size > 0) {
296
17.7k
            pcpo->char_name.data = gs_alloc_bytes(pdev->pdf_memory->non_gc_memory, gnstr->size, "storage for charproc name");
297
17.7k
            if (pcpo->char_name.data == NULL)
298
0
                return_error(gs_error_VMerror);
299
17.7k
            memcpy(pcpo->char_name.data, gnstr->data, gnstr->size);
300
17.7k
        }
301
17.7k
        pcpo->char_name.size = gnstr->size;
302
17.7k
    }
303
92.0k
    pcpo->duplicate_char_name = duplicate_char_name;
304
92.0k
    return 0;
305
92.0k
}
306
307
int
308
pdf_free_charproc_ownership(gx_device_pdf * pdev, pdf_resource_t *pres)
309
2.73k
{
310
2.73k
    pdf_char_proc_ownership_t *next, *pcpo = (pdf_char_proc_ownership_t *)pres;
311
312
94.8k
    while (pcpo) {
313
92.0k
        next = pcpo->char_next;
314
92.0k
        if(pcpo->char_name.size != 0 && pcpo->char_name.data) {
315
17.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
17.7k
            pcpo->char_name.data = (byte *)0L;
319
17.7k
            pcpo->char_name.size = 0;
320
17.7k
        }
321
92.0k
        gs_free_object(pdev->pdf_memory, pcpo, "Free CharProc");
322
92.0k
        pcpo = next;
323
92.0k
    }
324
2.73k
    return 0;
325
2.73k
}
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
74.2k
{
333
74.2k
    gs_char char_code = 0;
334
74.2k
    pdf_bitmap_fonts_t *const pbfs = pdev->text->bitmap_fonts;
335
74.2k
    pdf_font_resource_t *font;
336
74.2k
    pdf_resource_t *pres;
337
74.2k
    pdf_char_proc_t *pcp;
338
74.2k
    int code;
339
    /* This code added to store PCL bitmap glyphs in type 3 fonts where possible */
340
74.2k
    gs_glyph glyph = GS_NO_GLYPH;
341
74.2k
    gs_const_string str2, *str = NULL;
342
74.2k
    gs_show_enum *show_enum = (gs_show_enum *)pdev->pte;
343
74.2k
    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
74.2k
    int allowed_op = (show_enum->text.operation &
348
74.2k
        (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
74.2k
    if ((show_enum->current_font->FontType == ft_user_defined ||
361
74.2k
        show_enum->current_font->FontType == ft_PDF_user_defined ||
362
74.2k
        show_enum->current_font->FontType == ft_PCL_user_defined ||
363
74.2k
        show_enum->current_font->FontType == ft_MicroType ||
364
74.2k
        show_enum->current_font->FontType == ft_GL2_stick_user_defined ||
365
74.2k
        show_enum->current_font->FontType == ft_GL2_531) && allowed_op &&
366
0
        show_enum->current_font->FontMatrix.xx == 1 && show_enum->current_font->FontMatrix.xy == 0 &&
367
0
        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
74.2k
    } else {
438
74.2k
        char_code = assign_char_code(pdev, pdev->pte);
439
74.2k
        font = pbfs->open_font; /* Type 3 */
440
74.2k
    }
441
442
74.2k
    code = pdf_begin_resource(pdev, resourceCharProc, id, &pres);
443
74.2k
    if (code < 0)
444
0
        return code;
445
74.2k
    pcp = (pdf_char_proc_t *) pres;
446
74.2k
    code = pdf_attach_charproc(pdev, font, pcp, glyph, char_code, str);
447
74.2k
    if (code < 0)
448
0
        return code;
449
74.2k
    pres->object->written = true;
450
74.2k
    {
451
74.2k
        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
74.2k
        stream_puts(s, "<</Length       >>stream\n");
461
74.2k
        ppos->start_pos = stell(s);
462
74.2k
    }
463
74.2k
    code = pdf_begin_encrypt(pdev, &pdev->strm, pres->object->id);
464
74.2k
    if (code < 0)
465
0
        return code;
466
74.2k
    pcp->y_offset = y_offset;
467
74.2k
    pcp->x_offset = x_offset;
468
74.2k
    font->u.simple.s.type3.FontBBox.q.x =
469
74.2k
        max(font->u.simple.s.type3.FontBBox.q.x, w);
470
74.2k
    font->u.simple.s.type3.FontBBox.q.y =
471
74.2k
        max(font->u.simple.s.type3.FontBBox.q.y, y_offset + h);
472
74.2k
    font->u.simple.s.type3.max_y_offset =
473
74.2k
        max(font->u.simple.s.type3.max_y_offset, h + (h >> 2));
474
74.2k
    pcp->real_width.x = w;
475
74.2k
    pcp->real_width.y = y_offset + h;
476
74.2k
    *ppcp = pcp;
477
74.2k
    return 0;
478
74.2k
}
479
480
/* End a CharProc. */
481
int
482
pdf_end_char_proc(gx_device_pdf * pdev, pdf_stream_position_t * ppos)
483
74.2k
{
484
74.2k
    stream *s;
485
74.2k
    gs_offset_t start_pos, end_pos, length;
486
487
74.2k
    if (pdf_end_encrypt(pdev))
488
0
        s_close_filters(&pdev->strm, pdev->strm->strm);
489
490
74.2k
    s = pdev->strm;
491
74.2k
    start_pos = ppos->start_pos;
492
74.2k
    end_pos = stell(s);
493
74.2k
    length = end_pos - start_pos;
494
74.2k
    if (length > 999999)
495
0
        return_error(gs_error_limitcheck);
496
74.2k
    sseek(s, start_pos - 15);
497
74.2k
    pprintd1(s, "%d", length);
498
74.2k
    sseek(s, end_pos);
499
74.2k
    if (pdev->PDFA != 0)
500
0
        stream_puts(s, "\n");
501
74.2k
    stream_puts(s, "endstream\n");
502
74.2k
    pdf_end_separate(pdev, resourceCharProc);
503
74.2k
    return 0;
504
74.2k
}
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
25.8k
{
510
25.8k
    if (pdfont->mark_glyph == NULL) {
511
        /* Synthesised bitmap fonts pass here. */
512
23.8k
        return;
513
23.8k
    }
514
2.07k
    if (pdfont->u.simple.Encoding != NULL) {
515
2.07k
         int i;
516
517
532k
         for (i = 0; i < 256; i++)
518
530k
             if (pdfont->u.simple.Encoding[i].glyph != GS_NO_GLYPH)
519
76.0k
                pdfont->mark_glyph(memory, pdfont->u.simple.Encoding[i].glyph, pdfont->mark_glyph_data);
520
2.07k
     }
521
2.07k
    if (pdfont->FontType == ft_user_defined ||
522
1.66k
        pdfont->FontType == ft_PDF_user_defined ||
523
1.66k
        pdfont->FontType == ft_PCL_user_defined ||
524
1.66k
        pdfont->FontType == ft_MicroType ||
525
1.66k
        pdfont->FontType == ft_GL2_stick_user_defined ||
526
1.66k
        pdfont->FontType == ft_GL2_531) {
527
402
        const pdf_char_proc_ownership_t *pcpo = pdfont->u.simple.s.type3.char_procs;
528
529
7.64k
        for (; pcpo != NULL; pcpo = pcpo->font_next)
530
7.24k
            pdfont->mark_glyph(memory, pcpo->glyph, pdfont->mark_glyph_data);
531
402
    }
532
2.07k
}
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
796k
{
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
796k
    const pdf_char_proc_ownership_t * pcpo = pcp->owner_fonts;
543
796k
    pdf_font_resource_t *pdfont = pcpo->font;
544
796k
    byte ch = pcpo->char_code;
545
796k
    pdf_text_state_values_t values;
546
547
796k
    values.character_spacing = 0;
548
796k
    values.pdfont = pdfont;
549
796k
    values.size = 1;
550
796k
    values.matrix = *pimat;
551
796k
    values.render_mode = pdev->pte->pgs->text_rendering_mode;
552
796k
    values.word_spacing = 0;
553
796k
    pdf_set_text_state_values(pdev, &values);
554
796k
    pdf_bitmap_char_update_bbox(pdev, pcp->x_offset, pcp->y_offset, pcp->real_width.x, pcp->real_width.y);
555
796k
    pdf_append_chars(pdev, &ch, 1, pdfont->Widths[ch] * pimat->xx, 0.0, false);
556
796k
    return 0;
557
796k
}
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
32.9k
{
565
32.9k
    pdf_bitmap_fonts_t *pbfs;
566
567
32.9k
    if (pdev->text == NULL || pdev->text->bitmap_fonts == NULL)
568
0
        return 0;
569
570
32.9k
    pbfs = pdev->text->bitmap_fonts;
571
572
32.9k
    if (pbfs->bitmap_encoding_id) {
573
1.50k
        stream *s;
574
1.50k
        int i;
575
576
1.50k
        pdf_open_separate(pdev, pbfs->bitmap_encoding_id, resourceEncoding);
577
1.50k
        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.50k
        stream_puts(s, "<</Type/Encoding/Differences[0");
586
102k
        for (i = 0; i <= pbfs->max_embedded_code; ++i) {
587
100k
            if (!(i & 15))
588
7.07k
                stream_puts(s, "\n");
589
100k
            pprintd1(s, "/a%d", i);
590
100k
        }
591
1.50k
        stream_puts(s, "\n] >>\n");
592
1.50k
        pdf_end_separate(pdev, resourceEncoding);
593
1.50k
        pbfs->bitmap_encoding_id = 0;
594
1.50k
    }
595
32.9k
    return 0;
596
32.9k
}
597
598
/*
599
 * Start charproc accumulation for a Type 3 font.
600
 */
601
int
602
pdf_start_charproc_accum(gx_device_pdf *pdev)
603
85.0k
{
604
85.0k
    pdf_char_proc_t *pcp;
605
85.0k
    pdf_resource_t *pres;
606
85.0k
    int id = gs_next_ids(pdev->memory, 1);
607
85.0k
    int code = pdf_enter_substream(pdev, resourceCharProc, id,
608
85.0k
                                   &pres, false, pdev->CompressFonts);
609
610
85.0k
    if (code < 0)
611
0
       return code;
612
85.0k
    pres->rid = id;
613
85.0k
    pcp = (pdf_char_proc_t *)pres;
614
85.0k
    pcp->owner_fonts = NULL;
615
85.0k
    return 0;
616
85.0k
}
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
69.8k
{
625
69.8k
    pdf_font_resource_t *pdfont;
626
69.8k
    pdf_resource_t *pres = pdev->accumulating_substream_resource;
627
69.8k
    pdf_char_proc_t *pcp;
628
69.8k
    int code;
629
630
69.8k
    if (pres == NULL)
631
0
        return_error(gs_error_undefined);
632
69.8k
    code = pdf_attached_font_resource(pdev, font, &pdfont, NULL, NULL, NULL, NULL);
633
69.8k
    if (code < 0)
634
0
        return code;
635
69.8k
    pcp = (pdf_char_proc_t *)pres;
636
69.8k
    pcp->owner_fonts = NULL;
637
69.8k
    pcp->real_width.x = pw[font->WMode && narg > 6 ? 6 : 0];
638
69.8k
    pcp->real_width.y = pw[font->WMode && narg > 6 ? 7 : 1];
639
69.8k
    pcp->v.x = (narg > 8 ? pw[8] : 0);
640
69.8k
    pcp->v.y = (narg > 8 ? pw[9] : 0);
641
69.8k
    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
61.5k
        pdev->skip_colors = false;
647
61.5k
        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
61.5k
        if (font->FontType == ft_PCL_user_defined || font->FontType == ft_GL2_stick_user_defined
653
61.5k
            || font->FontType == ft_GL2_531 || font->FontType == ft_MicroType)
654
0
            pdfont->u.simple.s.type3.cached[ch >> 3] |= 0x80 >> (ch & 7);
655
61.5k
    } else {
656
8.31k
        double d;
657
8.31k
        pdev->skip_colors = true;
658
8.31k
        if (pw[4] < pw[2]) {
659
17
            d = pw[2];
660
17
            pw[2] = pw[4];
661
17
            pw[4] = d;
662
17
        }
663
8.31k
        if (pw[5] < pw[3]) {
664
28
            d = pw[5];
665
28
            pw[5] = pw[3];
666
28
            pw[3] = d;
667
28
        }
668
8.31k
        pprintg6(pdev->strm, "%g %g %g %g %g %g d1\n",
669
8.31k
            (float)pw[0], (float)0.0, (float)pw[2],
670
8.31k
            (float)pw[3], (float)pw[4], (float)pw[5]);
671
8.31k
        pdfont->u.simple.s.type3.cached[ch >> 3] |= 0x80 >> (ch & 7);
672
8.31k
    }
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
69.8k
    if (scale_100) {
680
69.8k
    code = stream_puts(pdev->strm, "0.01 0 0 0.01 0 0 cm\n");
681
69.8k
    if (code < 0)
682
0
        return code;
683
69.8k
    }
684
69.8k
    return 0;
685
69.8k
}
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
158k
{
695
158k
    int code;
696
158k
    pdf_resource_t *pres;
697
158k
    stream *s, *save_strm = pdev->strm;
698
158k
    pdf_data_writer_t writer;
699
158k
    static const pdf_filter_names_t fnames = {
700
158k
        PDF_FILTER_NAMES
701
158k
    };
702
703
158k
    pdev->streams.save_strm = pdev->strm;
704
705
158k
    if (rtype >= NUM_RESOURCE_TYPES)
706
36.7k
        rtype = resourceOther;
707
158k
    code = pdf_alloc_aside(pdev, PDF_RESOURCE_CHAIN(pdev, rtype, id),
708
158k
                pdf_resource_type_structs[rtype], &pres, reserve_object_id ? 0 : -1);
709
158k
    if (code < 0)
710
0
        return code;
711
158k
    cos_become(pres->object, cos_type_stream);
712
158k
    s = cos_write_stream_alloc((cos_stream_t *)pres->object, pdev, "pdf_enter_substream");
713
158k
    if (s == 0)
714
0
        return_error(gs_error_VMerror);
715
158k
    pdev->strm = s;
716
158k
    code = pdf_append_data_stream_filters(pdev, &writer,
717
158k
                             options | DATA_STREAM_NOLENGTH, pres->object->id);
718
158k
    if (code < 0) {
719
0
        pdev->strm = save_strm;
720
0
        return code;
721
0
    }
722
158k
    code = pdf_put_filters((cos_dict_t *)pres->object, pdev, writer.binary.strm, &fnames);
723
158k
    if (code < 0) {
724
0
        pdev->strm = save_strm;
725
0
        return code;
726
0
    }
727
158k
    pdev->strm = writer.binary.strm;
728
158k
    *ppres = pres;
729
158k
    return 0;
730
158k
}
731
732
/*
733
 * Close a stream object in the temporary file.
734
 */
735
int
736
pdf_close_aside(gx_device_pdf *pdev)
737
158k
{
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
158k
    stream *s = pdev->strm;
742
158k
    cos_stream_t *pcs = cos_stream_from_pipeline(s);
743
158k
    int status = s_close_filters(&s, NULL);
744
745
158k
    pdev->strm = pdev->streams.save_strm;
746
158k
    if (status < 0)
747
0
         return(gs_note_error(gs_error_ioerror));
748
749
158k
    if (!pcs)
750
0
        return gs_note_error(gs_error_ioerror);
751
752
158k
    pcs->is_open = false;
753
158k
    return 0;
754
158k
}
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
121k
{
763
121k
    int sbstack_ptr = pdev->sbstack_depth;
764
121k
    pdf_resource_t *pres;
765
121k
    stream *save_strm = pdev->strm;
766
121k
    int code;
767
768
121k
    if (pdev->sbstack_depth >= pdev->sbstack_size)
769
0
        return_error(gs_error_unregistered); /* Must not happen. */
770
121k
    if (pdev->sbstack[sbstack_ptr].text_state == 0) {
771
121k
        pdev->sbstack[sbstack_ptr].text_state = pdf_text_state_alloc(pdev->pdf_memory);
772
121k
        if (pdev->sbstack[sbstack_ptr].text_state == 0)
773
0
            return_error(gs_error_VMerror);
774
121k
    }
775
121k
    code = pdf_open_aside(pdev, rtype, id, &pres, reserve_object_id,
776
121k
                    (compress ? DATA_STREAM_COMPRESS : 0));
777
121k
    if (code < 0)
778
0
        return code;
779
121k
    code = pdf_save_viewer_state(pdev, NULL);
780
121k
    if (code < 0) {
781
0
        pdev->strm = save_strm;
782
0
        return code;
783
0
    }
784
121k
    pdev->sbstack[sbstack_ptr].context = pdev->context;
785
121k
    pdf_text_state_copy(pdev->sbstack[sbstack_ptr].text_state, pdev->text->text_state);
786
121k
    pdf_set_text_state_default(pdev->text->text_state);
787
121k
    pdev->sbstack[sbstack_ptr].clip_path = pdev->clip_path;
788
121k
    pdev->clip_path = 0;
789
121k
    pdev->sbstack[sbstack_ptr].clip_path_id = pdev->clip_path_id;
790
121k
    pdev->clip_path_id = pdev->no_clip_path_id;
791
121k
    pdev->sbstack[sbstack_ptr].vgstack_bottom = pdev->vgstack_bottom;
792
121k
    pdev->vgstack_bottom = pdev->vgstack_depth;
793
121k
    pdev->sbstack[sbstack_ptr].strm = save_strm;
794
121k
    pdev->sbstack[sbstack_ptr].procsets = pdev->procsets;
795
121k
    pdev->sbstack[sbstack_ptr].substream_Resources = pdev->substream_Resources;
796
121k
    pdev->sbstack[sbstack_ptr].skip_colors = pdev->skip_colors;
797
121k
    pdev->sbstack[sbstack_ptr].font3 = pdev->font3;
798
121k
    pdev->sbstack[sbstack_ptr].accumulating_substream_resource = pdev->accumulating_substream_resource;
799
121k
    pdev->sbstack[sbstack_ptr].charproc_just_accumulated = pdev->charproc_just_accumulated;
800
121k
    pdev->sbstack[sbstack_ptr].accumulating_a_global_object = pdev->accumulating_a_global_object;
801
121k
    pdev->sbstack[sbstack_ptr].pres_soft_mask_dict = pdev->pres_soft_mask_dict;
802
121k
    pdev->sbstack[sbstack_ptr].objname = pdev->objname;
803
121k
    pdev->sbstack[sbstack_ptr].last_charpath_op = pdev->last_charpath_op;
804
121k
    pdev->skip_colors = false;
805
121k
    pdev->charproc_just_accumulated = false;
806
121k
    pdev->pres_soft_mask_dict = NULL;
807
121k
    pdev->objname.data = NULL;
808
121k
    pdev->objname.size = 0;
809
    /* Do not reset pdev->accumulating_a_global_object - it inherits. */
810
121k
    pdev->sbstack_depth++;
811
121k
    pdev->procsets = 0;
812
121k
    pdev->font3 = 0;
813
121k
    pdev->context = PDF_IN_STREAM;
814
121k
    pdev->accumulating_substream_resource = pres;
815
121k
    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
121k
    if (rtype != resourceXObject)
820
117k
        pdf_reset_graphics(pdev);
821
3.66k
    else {
822
3.66k
        if (pdev->vg_initial_set)
823
3.66k
            pdev->state.blend_mode = pdev->vg_initial.blend_mode;
824
3.66k
    }
825
121k
    *ppres = pres;
826
121k
    return 0;
827
121k
}
828
829
/*
830
 * Exit the substream accumulation mode.
831
 */
832
int
833
pdf_exit_substream(gx_device_pdf *pdev)
834
121k
{
835
121k
    int code, code1;
836
121k
    int sbstack_ptr;
837
838
121k
    if (pdev->sbstack_depth <= 0)
839
0
        return_error(gs_error_unregistered); /* Must not happen. */
840
121k
    code = pdf_open_contents(pdev, PDF_IN_STREAM);
841
121k
    sbstack_ptr = pdev->sbstack_depth - 1;
842
125k
    while (pdev->vgstack_depth > pdev->vgstack_bottom) {
843
3.99k
        code1 = pdf_restore_viewer_state(pdev, pdev->strm);
844
3.99k
        if (code >= 0)
845
3.99k
            code = code1;
846
3.99k
    }
847
121k
    if (pdev->clip_path != 0)
848
0
        gx_path_free(pdev->clip_path, "pdf_end_charproc_accum");
849
121k
    code1 = pdf_close_aside(pdev);
850
121k
    if (code1 < 0 && code >= 0)
851
0
        code = code1;
852
121k
    pdev->context = pdev->sbstack[sbstack_ptr].context;
853
121k
    pdf_text_state_copy(pdev->text->text_state, pdev->sbstack[sbstack_ptr].text_state);
854
121k
    gs_free_object(pdev->pdf_memory, pdev->sbstack[sbstack_ptr].text_state, "free text state for stream");
855
121k
    pdev->sbstack[sbstack_ptr].text_state = 0;
856
121k
    pdev->clip_path = pdev->sbstack[sbstack_ptr].clip_path;
857
121k
    pdev->sbstack[sbstack_ptr].clip_path = 0;
858
121k
    pdev->clip_path_id = pdev->sbstack[sbstack_ptr].clip_path_id;
859
121k
    pdev->vgstack_bottom = pdev->sbstack[sbstack_ptr].vgstack_bottom;
860
121k
    pdev->strm = pdev->sbstack[sbstack_ptr].strm;
861
121k
    pdev->sbstack[sbstack_ptr].strm = 0;
862
121k
    pdev->procsets = pdev->sbstack[sbstack_ptr].procsets;
863
121k
    pdev->substream_Resources = pdev->sbstack[sbstack_ptr].substream_Resources;
864
121k
    pdev->sbstack[sbstack_ptr].substream_Resources = 0;
865
121k
    pdev->skip_colors = pdev->sbstack[sbstack_ptr].skip_colors;
866
121k
    pdev->font3 = pdev->sbstack[sbstack_ptr].font3;
867
121k
    pdev->sbstack[sbstack_ptr].font3 = 0;
868
121k
    pdev->accumulating_substream_resource = pdev->sbstack[sbstack_ptr].accumulating_substream_resource;
869
121k
    pdev->sbstack[sbstack_ptr].accumulating_substream_resource = 0;
870
121k
    pdev->charproc_just_accumulated = pdev->sbstack[sbstack_ptr].charproc_just_accumulated;
871
121k
    pdev->accumulating_a_global_object = pdev->sbstack[sbstack_ptr].accumulating_a_global_object;
872
121k
    pdev->pres_soft_mask_dict = pdev->sbstack[sbstack_ptr].pres_soft_mask_dict;
873
121k
    pdev->objname = pdev->sbstack[sbstack_ptr].objname;
874
121k
    pdev->last_charpath_op = pdev->sbstack[sbstack_ptr].last_charpath_op;
875
121k
    pdev->sbstack_depth = sbstack_ptr;
876
121k
    code1 = pdf_restore_viewer_state(pdev, NULL);
877
121k
    if (code1 < 0 && code >= 0)
878
0
        code = code1;
879
121k
    return code;
880
121k
}
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
245k
{
885
245k
    if (pcp0->real_width.x != pcp1->real_width.x)
886
153k
        return false;
887
91.4k
    if (pcp0->real_width.y != pcp1->real_width.y)
888
0
        return false;
889
91.4k
    if (pcp0->v.x != pcp1->v.x)
890
0
        return false;
891
91.4k
    if (pcp0->v.y != pcp1->v.y)
892
0
        return false;
893
91.4k
    return true;
894
91.4k
}
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
142
{
907
142
    pdf_char_proc_ownership_t *pcpo;
908
909
1.13k
    for (pcpo = pdfont->u.simple.s.type3.char_procs; pcpo != NULL; pcpo = pcpo->char_next) {
910
996
        if (pcpo->char_code == char_code) {
911
2
            return true;
912
2
        }
913
996
    }
914
140
    return false;
915
142
}
916
917
static int
918
pdf_is_charproc_compatible(gx_device_pdf * pdev, pdf_resource_t *pres0, pdf_resource_t *pres1)
919
142
{
920
142
    charproc_compatibility_data_t *data = (charproc_compatibility_data_t *)pdev->find_resource_param;
921
142
    pdf_char_proc_t *pcp0 = (pdf_char_proc_t *)pres0;
922
142
    pdf_char_proc_t *pcp1 = (pdf_char_proc_t *)pres1;
923
142
    pdf_font_resource_t *pdfont = data->pdfont;
924
142
    pdf_char_proc_ownership_t *pcpo;
925
142
    pdf_font_cache_elem_t **e;
926
142
    bool can_add_to_current_font = false, computed_can_add_to_current_font = false;
927
928
    /* Does it have same attributes ? */
929
142
    if (!pdf_is_same_charproc_attrs1(pdev, pcp0, pcp1))
930
0
        return 0;
931
    /* Is it from same font ? */
932
1.81k
    for (pcpo = pcp1->owner_fonts; pcpo != NULL; pcpo = pcpo->char_next) {
933
1.66k
        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
1.66k
    }
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
142
    e = pdf_locate_font_cache_elem(pdev, data->font);
951
142
    if (e != NULL) {
952
1.81k
        for (pcpo = pcp1->owner_fonts; pcpo != NULL; pcpo = pcpo->char_next) {
953
1.66k
            if (pcpo->char_code != data->char_code || pcpo->glyph != data->glyph)
954
1.60k
                continue; /* Need same Encoding to generate a proper ToUnicode. */
955
66
            if (pdfont->u.simple.s.type3.bitmap_font != pcpo->font->u.simple.s.type3.bitmap_font)
956
0
                continue;
957
66
            if (gs_matrix_compare(&pdfont->u.simple.s.type3.FontMatrix, &pcpo->font->u.simple.s.type3.FontMatrix))
958
20
                continue;
959
46
            if (data->cgp != NULL) {
960
44
                if (!pdf_check_encoding_compatibility(pcpo->font, data->cgp->s, data->cgp->num_all_chars))
961
0
                    continue;
962
44
            }
963
46
            if ((*e)->pdfont != pcpo->font)
964
46
                continue;
965
0
            data->pdfont = pcpo->font; /* Switch to the other font. */
966
0
            return 1;
967
46
        }
968
142
    }
969
    /* Check whether it can be added into the current font. */
970
142
    if (!computed_can_add_to_current_font)
971
142
        can_add_to_current_font = !is_char_code_used(pdfont, data->char_code);
972
142
    if (!can_add_to_current_font) {
973
        /* Can't substitute due to encoding conflict. */
974
2
        return 0;
975
2
    }
976
    /* The current font will share it with another font. */
977
140
    return 1;
978
142
}
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
85.0k
{
984
85.0k
    pdf_char_proc_ownership_t *pcpo;
985
85.0k
    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
253k
    for (pcpo = (*ppdfont)->u.simple.s.type3.char_procs; pcpo != NULL; pcpo = pcpo->char_next) {
992
244k
        pdf_char_proc_t *pcp = pcpo->char_proc;
993
994
244k
        if (*ppcp != pcp && pdf_is_same_charproc_attrs1(pdev, *ppcp, pcp)) {
995
91.3k
            cos_object_t *pco0 = pcp->object;
996
91.3k
            cos_object_t *pco1 = (*ppcp)->object;
997
998
91.3k
            code = pco0->cos_procs->equal(pco0, pco1, pdev);
999
91.3k
            if (code < 0) {
1000
0
                return code;
1001
0
            }
1002
91.3k
            if (code) {
1003
76.3k
                *ppcp = pcp;
1004
76.3k
                return 1;
1005
76.3k
            }
1006
91.3k
        }
1007
244k
    }
1008
8.73k
    return pdf_find_same_resource(pdev, resourceCharProc, (pdf_resource_t **)ppcp, pdf_is_charproc_compatible);
1009
85.0k
}
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
85.0k
{
1016
85.0k
    charproc_compatibility_data_t data;
1017
85.0k
    int code;
1018
1019
85.0k
    data.cgp = cgp;
1020
85.0k
    data.pdfont = *ppdfont;
1021
85.0k
    data.char_code = char_code;
1022
85.0k
    data.glyph = glyph;
1023
85.0k
    data.font = font;
1024
85.0k
    pdev->find_resource_param = &data;
1025
85.0k
    code = pdf_find_same_charproc_aux(pdev, ppdfont, ppcp);
1026
85.0k
    pdev->find_resource_param = NULL;
1027
85.0k
    *ppdfont = data.pdfont;
1028
85.0k
    return code;
1029
85.0k
}
1030
1031
static bool
1032
pdf_is_charproc_defined(gx_device_pdf *pdev, pdf_font_resource_t *pdfont, gs_char ch)
1033
8.59k
{
1034
8.59k
    pdf_char_proc_ownership_t *pcpo;
1035
1036
120k
    for (pcpo = pdfont->u.simple.s.type3.char_procs; pcpo != NULL; pcpo = pcpo->char_next) {
1037
111k
        if (pcpo->char_code == ch)
1038
18
            return true;
1039
111k
    }
1040
8.57k
    return false;
1041
8.59k
}
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
85.0k
{
1048
85.0k
    pdf_font_resource_t *pdfont;
1049
85.0k
    double *real_widths;
1050
85.0k
    byte *glyph_usage;
1051
85.0k
    int char_cache_size, width_cache_size;
1052
85.0k
    pdf_encoding_element_t *pet;
1053
85.0k
    int code;
1054
1055
85.0k
    code = pdf_attached_font_resource(pdev, font, &pdfont,
1056
85.0k
                &glyph_usage, &real_widths, &char_cache_size, &width_cache_size);
1057
85.0k
    if (code < 0)
1058
0
        return code;
1059
85.0k
    code = pdf_attach_charproc(pdev, pdfont, pcp, glyph, ch, gnstr);
1060
85.0k
    if (code < 0)
1061
0
        return code;
1062
85.0k
    if (ch >= char_cache_size || ch >= width_cache_size)
1063
0
        return_error(gs_error_unregistered); /* Must not happen. */
1064
85.0k
    pet = &pdfont->u.simple.Encoding[ch];
1065
85.0k
    pdfont->Widths[ch] = pcp->real_width.x;
1066
85.0k
    real_widths[ch * 2    ] = pcp->real_width.x;
1067
85.0k
    real_widths[ch * 2 + 1] = pcp->real_width.y;
1068
85.0k
    glyph_usage[ch / 8] |= 0x80 >> (ch & 7);
1069
85.0k
    pdfont->used[ch >> 3] |= 0x80 >> (ch & 7);
1070
85.0k
    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
85.0k
    pet->glyph = glyph;
1075
85.0k
    pet->is_difference = true;
1076
85.0k
    if (pdfont->u.simple.LastChar < (int)ch)
1077
3.39k
        pdfont->u.simple.LastChar = (int)ch;
1078
85.0k
    if (pdfont->u.simple.FirstChar > (int)ch)
1079
2.19k
        pdfont->u.simple.FirstChar = (int)ch;
1080
85.0k
    return pdf_copy_string_to_encoding(pdev, (gs_const_string *)gnstr, pet);
1081
85.0k
}
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
85.0k
{
1115
85.0k
    int code;
1116
85.0k
    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
85.0k
    pdf_char_proc_t *pcp = (pdf_char_proc_t *)pres;
1121
85.0k
    pdf_font_resource_t *pdfont;
1122
85.0k
    gs_char ch = output_char_code;
1123
85.0k
    bool checking_glyph_variation = false;
1124
1125
85.0k
    if (ch == GS_NO_CHAR)
1126
0
        return_error(gs_error_unregistered); /* Must not happen. */
1127
85.0k
    if (ch >= 256)
1128
0
        return_error(gs_error_unregistered); /* Must not happen. */
1129
85.0k
    if (pres == NULL)
1130
0
        return_error(gs_error_unregistered); /* Must not happen. */
1131
85.0k
    code = pdf_attached_font_resource(pdev, font, &pdfont, NULL, NULL, NULL, NULL);
1132
85.0k
    if (code < 0)
1133
0
        return code;
1134
85.0k
    if (pdfont != (pdf_font_resource_t *)pdev->font3)
1135
9
        return_error(gs_error_unregistered); /* Must not happen. */
1136
85.0k
    code = pdf_exit_substream(pdev);
1137
85.0k
    if (code < 0)
1138
0
        return code;
1139
85.0k
    if (!(pdfont->used[ch >> 3] & (0x80 >> (ch & 7))) ||
1140
85.0k
        !(pdfont->u.simple.s.type3.cached[ch >> 3] & (0x80 >> (ch & 7)))) {
1141
        /* First appearence or not cached - check for duplicates. */
1142
85.0k
        pdf_font_resource_t *pdfont1 = pdfont;
1143
1144
85.0k
        checking_glyph_variation = true;
1145
        /* CAUTION : a possible font change. */
1146
85.0k
        code = pdf_find_same_charproc(pdev, &pdfont, cgp, &pcp, glyph, ch, font);
1147
85.0k
        if (code < 0)
1148
0
            return code;
1149
85.0k
        if (code != 0) {
1150
76.4k
            code = pdf_cancel_resource(pdev, pres, resourceCharProc);
1151
76.4k
            if (code < 0)
1152
0
                return code;
1153
76.4k
            pdf_forget_resource(pdev, pres, resourceCharProc);
1154
76.4k
            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
76.4k
            pdev->charproc_just_accumulated = true;
1163
76.4k
            return complete_adding_char(pdev, font, glyph, ch, pcp, gnstr);
1164
76.4k
        }
1165
8.59k
        if (pdf_is_charproc_defined(pdev, pdfont, ch)) {
1166
            /* Encoding conflict after a font change. */
1167
18
            gs_font *base_font = font, *below;
1168
1169
18
            while ((below = base_font->base) != base_font &&
1170
0
                    base_font->procs.same_font(base_font, below, FONT_SAME_OUTLINES))
1171
0
                base_font = below;
1172
18
            code = pdf_make_font3_resource(pdev, base_font, &pdfont);
1173
18
            if (code < 0)
1174
0
                return code;
1175
18
            code = pdf_attach_font_resource(pdev, font, pdfont);
1176
18
            if (code < 0)
1177
0
                return code;
1178
18
        }
1179
8.59k
    }
1180
8.60k
    pdf_reserve_object_id(pdev, pres, 0);
1181
8.60k
    if (checking_glyph_variation)
1182
8.59k
        pdev->charproc_just_accumulated = true;
1183
8.60k
    return complete_adding_char(pdev, font, glyph, ch, pcp, gnstr);
1184
85.0k
}
1185
1186
/* Add procsets to substream Resources. */
1187
int
1188
pdf_add_procsets(cos_dict_t *pcd, pdf_procset_t procsets)
1189
5.05k
{
1190
5.05k
    char str[5 + 7 + 7 + 7 + 5 + 2];
1191
5.05k
    cos_value_t v;
1192
1193
5.05k
    strcpy(str, "[/PDF");
1194
5.05k
    if (procsets & ImageB)
1195
0
        strcat(str, "/ImageB");
1196
5.05k
    if (procsets & ImageC)
1197
968
        strcat(str, "/ImageC");
1198
5.05k
    if (procsets & ImageI)
1199
5
        strcat(str, "/ImageI");
1200
5.05k
    if (procsets & Text)
1201
80
        strcat(str, "/Text");
1202
5.05k
    strcat(str, "]");
1203
5.05k
    cos_string_value(&v, (byte *)str, strlen(str));
1204
5.05k
    return cos_dict_put_c_key(pcd, "/ProcSet", &v);
1205
5.05k
}
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
779k
{
1211
779k
    if (pcd != 0) {
1212
12.7k
        const cos_value_t *v = cos_dict_find(pcd, (const byte *)key, strlen(key));
1213
12.7k
        cos_dict_t *list;
1214
12.7k
        int code;
1215
12.7k
        char buf[10 + (sizeof(long) * 8 / 3 + 1)], buf1[sizeof(pres->rname) + 1];
1216
1217
12.7k
        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
12.7k
        gs_snprintf(buf, sizeof(buf), "%"PRId64" 0 R\n", pres->object->id);
1224
12.7k
        if (v != NULL) {
1225
7.42k
            if (v->value_type != COS_VALUE_OBJECT &&
1226
0
                v->value_type != COS_VALUE_RESOURCE)
1227
0
                return_error(gs_error_unregistered); /* Must not happen. */
1228
7.42k
            list = (cos_dict_t *)v->contents.object;
1229
7.42k
            if (list->cos_procs != &cos_dict_procs)
1230
0
                return_error(gs_error_unregistered); /* Must not happen. */
1231
7.42k
        } else {
1232
5.36k
            list = cos_dict_alloc(pdev, "pdf_add_resource");
1233
5.36k
            if (list == NULL)
1234
0
                return_error(gs_error_VMerror);
1235
5.36k
            code = cos_dict_put_c_key_object((cos_dict_t *)pcd, key, (cos_object_t *)list);
1236
5.36k
            if (code < 0)
1237
0
                return code;
1238
5.36k
        }
1239
12.7k
        buf1[0] = '/';
1240
12.7k
        strcpy(buf1 + 1, pres->rname);
1241
12.7k
        return cos_dict_put_string(list, (const byte *)buf1, strlen(buf1),
1242
12.7k
                        (const byte *)buf, strlen(buf));
1243
12.7k
    }
1244
766k
    return 0;
1245
779k
}