Coverage Report

Created: 2025-08-28 07:06

/src/ghostpdl/devices/vector/gdevpsft.c
Line
Count
Source (jump to first uncovered line)
1
/* Copyright (C) 2001-2024 Artifex Software, Inc.
2
   All Rights Reserved.
3
4
   This software is provided AS-IS with no warranty, either express or
5
   implied.
6
7
   This software is distributed under license and may not be copied,
8
   modified or distributed except as expressly authorized under the terms
9
   of the license contained in the file LICENSE in this distribution.
10
11
   Refer to licensing information at http://www.artifex.com or contact
12
   Artifex Software, Inc.,  39 Mesa Street, Suite 108A, San Francisco,
13
   CA 94129, USA, for further information.
14
*/
15
16
17
/* Write an embedded TrueType font */
18
#include "memory_.h"
19
#include <stdlib.h>   /* for qsort */
20
#include <math.h>   /* for floor */
21
#include "gx.h"
22
#include "gscencs.h"
23
#include "gserrors.h"
24
#include "gsmatrix.h"
25
#include "gsutil.h"
26
#include "gxfcid.h"
27
#include "gxfont.h"
28
#include "gxfont42.h"
29
#include "gxttf.h"
30
#include "stream.h"
31
#include "spprint.h"
32
#include "gdevpsf.h"
33
34
/* Internally used options */
35
144k
#define WRITE_TRUETYPE_STRIPPED 0x1000  /* internal */
36
163k
#define WRITE_TRUETYPE_CID 0x2000 /* internal */
37
38
#define MAX_COMPOSITE_PIECES 3  /* adhoc */
39
40
/*
41
 * The following are only for debugging.  They force various format choices
42
 * in the output.  The normal (non-debugging) values for all of these are
43
 * as indicated in the comments.
44
 *
45
 * Note that these options interact.  Here is the complete list of settings
46
 * that make sense.
47
        0 -1,0,1  N/A 0,1 0,1
48
        0xf000  -1  N/A 1 0,1
49
        0xf000  0,1 0,1 1 0,1
50
 */
51
/* Define whether to use the 0xf000 character bias for generated tables. */
52
24.5k
#define TT_BIAS 0xf000    /* 0xf000 */
53
/* Define whether to use cmap format 6 never(-1), sometimes(0), always(1). */
54
#define TT_FORCE_CMAP_6 0 /* 0 */
55
/* Define whether to use the bias for the cmap format 6 "first code". */
56
#define TT_BIAS_CMAP_6 0  /* 0 */
57
/* Define whether to generate an OS/2 table if none is supplied. */
58
#define TT_GENERATE_OS_2 1  /* 1 */
59
/* Define whether to adjust the OS/2 range bits. */
60
#define TT_ADJUST_OS_2 1  /* 1 */
61
/*
62
 * End of options.
63
 */
64
65
/* ---------------- Utilities ---------------- */
66
67
/* Pad to a multiple of 4 bytes. */
68
static void
69
put_pad(stream *s, uint length)
70
706k
{
71
706k
    static const byte pad_to_4[3] = {0, 0, 0};
72
73
706k
    stream_write(s, pad_to_4, (uint)(-(int)length & 3));
74
706k
}
75
76
/* Put short and long values on a stream. */
77
static void
78
put_ushort(stream *s, uint v)
79
55.2M
{
80
55.2M
    stream_putc(s, (byte)(v >> 8));
81
55.2M
    stream_putc(s, (byte)v);
82
55.2M
}
83
static void
84
put_short(stream *s, short v)
85
64.4M
{
86
64.4M
    stream_putc(s, (byte)(v >> 8));
87
64.4M
    stream_putc(s, (byte)v);
88
64.4M
}
89
static void
90
put_ulong(stream *s, ulong v)
91
0
{
92
0
    put_ushort(s, (uint)(v >> 16));
93
0
    put_ushort(s, (uint)v);
94
0
}
95
static void
96
put_loca(stream *s, ulong offset, int indexToLocFormat)
97
54.8M
{
98
54.8M
    if (indexToLocFormat)
99
0
        put_ulong(s, offset);
100
54.8M
    else
101
54.8M
        put_ushort(s, (uint)(offset >> 1));
102
54.8M
}
103
104
/* Get or put 2- or 4-byte quantities from/into a table. */
105
#define U8(p) ((uint)((p)[0]))
106
#define S8(p) (int)((U8(p) ^ 0x80) - 0x80)
107
3.19M
#define U16(p) (((uint)((p)[0]) << 8) + (p)[1])
108
#define S16(p) (int)((U16(p) ^ 0x8000) - 0x8000)
109
10.2M
#define u32(p) get_u32_msb(p)
110
static void
111
put_u16(byte *p, uint v)
112
6.48M
{
113
6.48M
    p[0] = (byte)(v >> 8);
114
6.48M
    p[1] = (byte)v;
115
6.48M
}
116
static void
117
put_u32(byte *p, ulong v)
118
1.10M
{
119
1.10M
    put_u16(p, (ushort)(v >> 16));
120
1.10M
    put_u16(p + 2, (ushort)v);
121
1.10M
}
122
static ulong
123
put_table(byte tab[16], const char *tname, ulong checksum, ulong offset,
124
          uint length)
125
123k
{
126
123k
    memcpy(tab, (const byte *)tname, 4);
127
123k
    put_u32(tab + 4, checksum);
128
123k
    put_u32(tab + 8, offset + 0x40000000);
129
123k
    put_u32(tab + 12, (ulong)length);
130
123k
    return offset + round_up(length, 4);
131
123k
}
132
133
/* Write one range of a TrueType font. */
134
static int
135
write_range(stream *s, gs_font_type42 *pfont, ulong start, uint length)
136
653k
{
137
653k
    ulong base = start, size = length;
138
139
653k
    if_debug3m('l', s->memory, "[l]write_range pos = %"PRId64", start = %lu, length = %u\n",
140
653k
               stell(s), start, length);
141
1.29M
    while (size > 0) {
142
653k
        const byte *ptr;
143
653k
        int code;
144
145
653k
        code = pfont->data.string_proc(pfont, base, size, &ptr);
146
653k
        if (code < 0)
147
15.8k
            return code;
148
637k
        if (!code)
149
637k
            code = size;
150
637k
        stream_write(s, ptr, code);
151
637k
        base += code;
152
637k
        size -= code;
153
637k
    }
154
637k
    return 0;
155
653k
}
156
157
/*
158
 * Determine the Macintosh glyph number for a given character, if any.
159
 * If no glyph can be found, return -1 and store the name in *pstr.
160
 */
161
static int
162
mac_glyph_index(gs_font *font, int ch, gs_const_string *pstr, int *index)
163
0
{
164
0
    gs_glyph glyph = font->procs.encode_char(font, (gs_char)ch,
165
0
                                             GLYPH_SPACE_NAME);
166
0
    int code;
167
168
0
    if (glyph == GS_NO_GLYPH) {
169
0
        *index = 0;
170
0
        return 0;   /* .notdef */
171
0
    }
172
0
    code = font->procs.glyph_name(font, glyph, pstr);
173
0
    if (code < 0)
174
0
        return code;
175
0
    if (glyph < GS_MIN_CID_GLYPH) {
176
0
        gs_char mac_char;
177
0
        gs_glyph mac_glyph;
178
0
        gs_const_string mstr;
179
180
        /* Look (not very hard) for a match in the Mac glyph space. */
181
0
        if (ch >= 32 && ch <= 126)
182
0
            mac_char = ch - 29;
183
0
        else if (ch >= 128 && ch <= 255)
184
0
            mac_char = ch - 30;
185
0
        else {
186
0
            *index = -1;
187
0
            return 0;
188
0
        }
189
0
        mac_glyph = gs_c_known_encode(mac_char, ENCODING_INDEX_MACGLYPH);
190
0
        if (mac_glyph == GS_NO_GLYPH) {
191
0
            *index = -1;
192
0
            return 0;
193
0
        }
194
0
        code = gs_c_glyph_name(mac_glyph, &mstr);
195
0
        if (code < 0)
196
0
            return code;
197
0
        if (!bytes_compare(pstr->data, pstr->size, mstr.data, mstr.size)) {
198
0
            *index = (int)mac_char;
199
0
            return 0;
200
0
        }
201
0
    }
202
0
    *index = -1;
203
0
    return 0;
204
0
}
205
206
/* ---------------- Individual tables ---------------- */
207
208
/* ------ cmap ------ */
209
210
/* Write a generated cmap table. */
211
static const byte cmap_initial_0[] = {
212
    0, 0,   /* table version # = 0 */
213
    0, 2,   /* # of encoding tables = 2 */
214
215
        /* First table, Macintosh */
216
    0, 1,   /* platform ID = Macintosh */
217
    0, 0,   /* platform encoding ID = ??? */
218
    0, 0, 0, 4+8+8, /* offset to table start */
219
        /* Second table, Windows */
220
    0, 3,   /* platform ID = Microsoft */
221
    0, 0,   /* platform encoding ID = unknown */
222
    0, 0, 1, 4+8+8+6, /* offset to table start */
223
224
        /* Start of Macintosh format 0 table */
225
    0, 0,   /* format = 0, byte encoding table */
226
    1, 6,   /* length */
227
    0, 0    /* version number */
228
};
229
static const byte cmap_initial_6[] = {
230
    0, 0,   /* table version # = 0 */
231
    0, 2,   /* # of encoding tables = 2 */
232
233
        /* First table, Macintosh */
234
    0, 1,   /* platform ID = Macintosh */
235
    0, 0,   /* platform encoding ID = ??? */
236
    0, 0, 0, 4+8+8, /* offset to table start */
237
        /* Second table, Windows */
238
    0, 3,   /* platform ID = Microsoft */
239
    0, 0,   /* platform encoding ID = unknown */
240
    0, 0, 0, 4+8+8+10,  /* offset to table start */
241
                        /* *VARIABLE*, add 2 x # of entries */
242
243
        /* Start of Macintosh format 6 table */
244
    0, 6,   /* format = 6, trimmed table mapping */
245
    0, 10,    /* length *VARIABLE*, add 2 x # of entries */
246
    0, 0,   /* version number */
247
    0, 0,   /* first character code */
248
    0, 0    /* # of entries *VARIABLE* */
249
};
250
static const byte cmap_unicode_initial_6[] = {
251
    0, 0,   /* table version # = 0 */
252
    0, 2,   /* # of encoding tables = 2 */
253
254
        /* First table, Macintosh */
255
    0, 1,   /* platform ID = Macintosh */
256
    0, 0,   /* platform encoding ID = ??? */
257
    0, 0, 0, 4+8+8, /* offset to table start */
258
        /* Second table, Windows */
259
    0, 3,   /* platform ID = Microsoft */
260
    0, 1,   /* platform encoding ID = Unicode */
261
    0, 0, 0, 4+8+8+10,  /* offset to table start */
262
                        /* *VARIABLE*, add 2 x # of entries */
263
264
        /* Start of Macintosh format 6 table */
265
    0, 6,   /* format = 6, trimmed table mapping */
266
    0, 10,    /* length *VARIABLE*, add 2 x # of entries */
267
    0, 0,   /* version number */
268
    0, 0,   /* first character code */
269
    0, 0    /* # of entries *VARIABLE* */
270
};
271
static const byte cmap_initial_4[] = {
272
    0, 0,   /* table version # = 0 */
273
    0, 1,   /* # of encoding tables = 2 */
274
275
        /* Single table, Windows */
276
    0, 3,   /* platform ID = Microsoft */
277
    0, 0,   /* platform encoding ID = unknown */
278
    0, 0, 0, 4+8  /* offset to table start */
279
};
280
static const byte cmap_sub_initial[] = {
281
    0, 4,   /* format = 4, segment mapping */
282
    0, 32,    /* length ** VARIABLE, add 2 x # of glyphs ** */
283
    0, 0,   /* version # */
284
    0, 4,   /* 2 x segCount */
285
    0, 4,   /* searchRange = 2 x 2 ^ floor(log2(segCount)) */
286
    0, 1,   /* floor(log2(segCount)) */
287
    0, 0,   /* 2 x segCount - searchRange */
288
289
    0, 0,   /* endCount[0] **VARIABLE** */
290
    255, 255,   /* endCount[1] */
291
    0, 0,   /* reservedPad */
292
    0, 0,   /* startCount[0] **VARIABLE** */
293
    255, 255,   /* startCount[1] */
294
    0, 0,   /* idDelta[0] */
295
    0, 1,   /* idDelta[1] */
296
    0, 4,   /* idRangeOffset[0] */
297
    0, 0    /* idRangeOffset[1] */
298
};
299
/*
300
 * The following nonsense is required because C defines sizeof()
301
 * inconsistently.
302
 */
303
14.6k
#define CMAP_ENTRIES_SIZE (256 * 2)
304
static void
305
write_cmap_0(stream *s, byte* entries /*[CMAP_ENTRIES_SIZE]*/, uint num_glyphs)
306
14.6k
{
307
14.6k
    int i;
308
309
14.6k
    if (CMAP_ENTRIES_SIZE - 2 * num_glyphs>0)
310
0
      memset(entries + 2 * num_glyphs, 0, CMAP_ENTRIES_SIZE - 2 * num_glyphs);
311
14.6k
    stream_write(s, cmap_initial_0, sizeof(cmap_initial_0));
312
3.75M
    for (i = 0; i <= 0xff; ++i)
313
3.74M
        sputc(s, (byte)entries[2 * i + 1]);
314
14.6k
}
315
static void
316
write_cmap_6(stream *s, byte *entries /*[CMAP_ENTRIES_SIZE]*/, uint first_code,
317
             uint first_entry, uint num_entries)
318
1.32k
{
319
1.32k
    byte cmap_data[sizeof(cmap_initial_6)];
320
321
1.32k
    memcpy(cmap_data, cmap_initial_6, sizeof(cmap_initial_6));
322
1.32k
    put_u16(cmap_data + 18,
323
1.32k
            U16(cmap_data + 18) + num_entries * 2);  /* offset */
324
1.32k
    put_u16(cmap_data + 22,
325
1.32k
            U16(cmap_data + 22) + num_entries * 2);  /* length */
326
1.32k
    put_u16(cmap_data + 26,
327
#if TT_BIAS_CMAP_6
328
            first_code +
329
#endif
330
1.32k
            first_entry);
331
1.32k
    put_u16(cmap_data + 28, num_entries);
332
1.32k
    stream_write(s, cmap_data, sizeof(cmap_data));
333
1.32k
    stream_write(s, entries + first_entry * 2, num_entries * 2);
334
1.32k
}
335
static void write_unicode_cmap_6(stream *s, byte *entries, uint first_code,
336
             uint first_entry, uint num_entries)
337
0
{
338
0
    byte cmap_data[sizeof(cmap_unicode_initial_6)];
339
340
0
    memcpy(cmap_data, cmap_unicode_initial_6, sizeof(cmap_unicode_initial_6));
341
0
    put_u16(cmap_data + 18,
342
0
            U16(cmap_data + 18) + num_entries * 2);  /* offset */
343
0
    put_u16(cmap_data + 22,
344
0
            U16(cmap_data + 22) + num_entries * 2);  /* length */
345
0
    put_u16(cmap_data + 26, first_entry);
346
0
    put_u16(cmap_data + 28, num_entries);
347
0
    stream_write(s, cmap_data, sizeof(cmap_data));
348
0
    stream_write(s, entries + first_entry * 2, num_entries * 2);
349
0
}
350
static void
351
write_cmap(stream *s, gs_font *font, uint first_code, int num_glyphs,
352
           gs_glyph max_glyph, int options, uint cmap_length)
353
16.3k
{
354
16.3k
    byte cmap_sub[sizeof(cmap_sub_initial)];
355
16.3k
    byte entries[CMAP_ENTRIES_SIZE];
356
16.3k
    int first_entry = 0, end_entry = num_glyphs;
357
16.3k
    bool can_use_trimmed = !(options & WRITE_TRUETYPE_NO_TRIMMED_TABLE);
358
16.3k
    uint merge = 0;
359
16.3k
    uint num_entries;
360
16.3k
    int i;
361
362
    /* Collect the table entries. */
363
364
4.19M
    for (i = 0; i < num_glyphs; ++i) {
365
4.18M
        gs_glyph glyph =
366
4.18M
            font->procs.encode_char(font, (gs_char)i, GLYPH_SPACE_INDEX);
367
4.18M
        uint glyph_index;
368
369
4.18M
        if (glyph == GS_NO_GLYPH || glyph < GS_MIN_GLYPH_INDEX ||
370
4.18M
            glyph > max_glyph
371
4.18M
            )
372
3.80M
            glyph = GS_MIN_GLYPH_INDEX;
373
4.18M
        glyph_index = (uint)(glyph - GS_MIN_GLYPH_INDEX);
374
4.18M
        merge |= glyph_index;
375
4.18M
        put_u16(entries + 2 * i, glyph_index);
376
4.18M
    }
377
2.39M
    while (end_entry > first_entry && !U16(entries + 2 * end_entry - 2))
378
2.37M
        --end_entry;
379
629k
    while (first_entry < end_entry && !U16(entries + 2 * first_entry))
380
613k
        ++first_entry;
381
16.3k
    num_entries = end_entry - first_entry;
382
383
    /* Write the table header and Macintosh sub-table (if any). */
384
385
16.3k
    if (options & WRITE_TRUETYPE_UNICODE_CMAP) {
386
0
        write_unicode_cmap_6(s, entries, first_code, first_entry, num_entries);
387
388
        /* Write the Windows sub-table. */
389
0
        memcpy(cmap_sub, cmap_sub_initial, sizeof(cmap_sub_initial));
390
0
        put_u16(cmap_sub + 2, U16(cmap_sub + 2) + num_entries * 2); /* length */
391
0
        put_u16(cmap_sub + 14, end_entry - 1); /* endCount[0] */
392
0
        put_u16(cmap_sub + 20, first_entry); /* startCount[0] */
393
0
        stream_write(s, cmap_sub, sizeof(cmap_sub));
394
0
        stream_write(s, entries + first_entry * 2, num_entries * 2);
395
0
        put_pad(s, cmap_length);
396
0
        return;
397
0
    }
398
#if TT_FORCE_CMAP_6 > 0
399
    /* Always use format 6. */
400
    write_cmap_6(s, entries, first_code, first_entry, num_entries);
401
#else
402
# if TT_FORCE_CMAP_6 < 0
403
    /* Never use format 6.  Use format 0 if possible. */
404
    if (merge == (byte)merge)
405
        write_cmap_0(s, entries, num_glyphs);
406
    else
407
# else /* TT_FORCE_CMAP == 0 */
408
    /*
409
     * Use format 0 if possible and (economical or format 6 disallowed),
410
     * otherwise format 6 if allowed.
411
     */
412
16.3k
    if (merge == (byte)merge && (num_entries <= 127 || !can_use_trimmed))
413
14.6k
        write_cmap_0(s, entries, num_glyphs);
414
1.71k
    else if (can_use_trimmed)
415
1.32k
        write_cmap_6(s, entries, first_code, first_entry, num_entries);
416
396
    else
417
396
# endif
418
396
    {
419
        /*
420
         * Punt.  Acrobat Reader 3 can't handle any other Mac table format.
421
         * (AR3 for Linux doesn't seem to be able to handle Windows format,
422
         * either, but maybe AR3 for Windows can.)
423
         */
424
396
        stream_write(s, cmap_initial_4, sizeof(cmap_initial_4));
425
396
    }
426
16.3k
#endif
427
428
    /* Write the Windows sub-table. */
429
430
16.3k
    memcpy(cmap_sub, cmap_sub_initial, sizeof(cmap_sub_initial));
431
16.3k
    put_u16(cmap_sub + 2, U16(cmap_sub + 2) + num_entries * 2); /* length */
432
16.3k
    put_u16(cmap_sub + 14, first_code + end_entry - 1); /* endCount[0] */
433
16.3k
    put_u16(cmap_sub + 20, first_code + first_entry); /* startCount[0] */
434
16.3k
    stream_write(s, cmap_sub, sizeof(cmap_sub));
435
16.3k
    stream_write(s, entries + first_entry * 2, num_entries * 2);
436
16.3k
    put_pad(s, cmap_length);
437
16.3k
}
438
static uint
439
size_cmap(gs_font *font, uint first_code, int num_glyphs, gs_glyph max_glyph,
440
          int options)
441
8.17k
{
442
8.17k
    stream poss;
443
444
8.17k
    s_init(&poss, NULL);
445
8.17k
    swrite_position_only(&poss);
446
8.17k
    write_cmap(&poss, font, first_code, num_glyphs, max_glyph, options, 0);
447
8.17k
    return stell(&poss);
448
8.17k
}
449
450
/* ------ hmtx/vmtx ------ */
451
452
/* We must include mtx data for every single glyph, even if we don't
453
 * actually write the glyph data. If we don't, the font will be invalid (cf bug 697376)
454
 * We also must not simply copy the table from the original font, because its possible
455
 * that we are combining two different fonts, and the metrics in each font may not
456
 * be the same (or at least, not in the same position) cf bug 700099.
457
 * Originally we only wrote the metrics for the glyphs we copy, then we disabled
458
 * generat_mtx so that we copied the table. This solution; generating all the metrics
459
 * we need and dummies for the glyphs we don't actually embed, works best so far.
460
 */
461
static void
462
write_mtx(stream *s, gs_font_type42 *pfont, const gs_type42_mtx_t *pmtx,
463
          int wmode)
464
9.53k
{
465
9.53k
    uint num_metrics = pmtx->numMetrics;
466
9.53k
    uint len = num_metrics * 4;
467
9.53k
    double factor = (double)pfont->data.unitsPerEm * (wmode ? -1 : 1);
468
9.53k
    float sbw[4];
469
9.53k
    uint i;
470
471
9.53k
    sbw[0] = sbw[1] = sbw[2] = sbw[3] = 0; /* in case of failures */
472
5.23M
    for (i = 0; i < pmtx->numMetrics; ++i) {
473
5.22M
        float f;
474
5.22M
        DISCARD(pfont->data.get_metrics(pfont, i, wmode, sbw));
475
        /* the temporary assignment to a float is necessary for AIX else the result is always 0 if sbw[] < 0
476
           this happens even with gcc and I'm not sure why it happens at all nor why only on AIX */
477
5.22M
        f = (float) (sbw[wmode + 2] * factor); /* width */
478
5.22M
        put_short(s, (short) floor(f + 0.5));
479
5.22M
        f = (float) (sbw[wmode] * factor); /* lsb, may be <0 */
480
5.22M
        put_short(s, (short) floor(f + 0.5));
481
5.22M
    }
482
54.0M
    for (; len < pmtx->length; ++i, len += 2) {
483
53.9M
        float f;
484
53.9M
        DISCARD(pfont->data.get_metrics(pfont, i, wmode, sbw));
485
53.9M
        f = (float) (sbw[wmode] * factor); /* lsb, may be <0 */
486
53.9M
        put_short(s, (short) floor(f + 0.5));
487
53.9M
    }
488
9.53k
}
489
490
/* Compute the metrics from the glyph_info. */
491
static uint
492
size_mtx(gs_font_type42 *pfont, gs_type42_mtx_t *pmtx, uint max_glyph, uint numGlyphs,
493
         int wmode)
494
9.53k
{
495
9.53k
    int prev_width = min_int;
496
9.53k
    uint last_width = 0; /* pacify compilers */
497
9.53k
    double factor = pfont->data.unitsPerEm * (wmode ? -1 : 1);
498
9.53k
    uint i;
499
500
9.57M
    for (i = 0; i <= max_glyph; ++i) {
501
9.56M
        float sbw[4];
502
9.56M
        int code = pfont->data.get_metrics(pfont, i, wmode, sbw);
503
9.56M
        int width;
504
505
9.56M
        if (code < 0)
506
9.33M
            continue;
507
226k
        width = (int)(sbw[wmode + 2] * factor + 0.5);
508
226k
        if (width != prev_width)
509
170k
            prev_width = width, last_width = i;
510
226k
    }
511
9.53k
    pmtx->numMetrics = last_width + 1;
512
9.53k
    pmtx->length = pmtx->numMetrics * 4 + (max_glyph - last_width) * 2;
513
9.53k
    pmtx->length += (numGlyphs - pmtx->numMetrics) * 2;
514
9.53k
    return pmtx->length;
515
9.53k
}
516
517
/* ------ name ------ */
518
519
/* Write a generated name table. */
520
static const byte name_initial[] = {
521
    0, 0,     /* format */
522
    0, 1,     /* # of records = 1 */
523
    0, 18,      /* start of string storage */
524
525
    0, 2,     /* platform ID = ISO */
526
    0, 2,     /* encoding ID = ISO 8859-1 */
527
    0, 0,     /* language ID (none) */
528
    0, 6,     /* name ID = PostScript name */
529
    0, 0,     /* length *VARIABLE* */
530
    0, 0      /* start of string within string storage */
531
};
532
static uint
533
size_name(const gs_const_string *font_name)
534
352
{
535
352
    return sizeof(name_initial) + font_name->size;
536
352
}
537
static void
538
write_name(stream *s, const gs_const_string *font_name)
539
176
{
540
176
    byte name_bytes[sizeof(name_initial)];
541
542
176
    memcpy(name_bytes, name_initial, sizeof(name_initial));
543
176
    put_u16(name_bytes + 14, font_name->size);
544
176
    stream_write(s, name_bytes, sizeof(name_bytes));
545
176
    stream_write(s, font_name->data, font_name->size);
546
176
    put_pad(s, size_name(font_name));
547
176
}
548
549
/* ------ OS/2 ------ */
550
551
/* Write a generated OS/2 table. */
552
76.8k
#define OS_2_LENGTH1 offset_of(ttf_OS_2_t, sxHeight[0]) /* OS/2 version 1. */
553
#define OS_2_LENGTH2 offset_of(ttf_OS_2_t, usLowerOpticalPointSize[0]) /* OS/2 version 2. */
554
51.9k
#define OS_2_LENGTH5 sizeof(ttf_OS_2_t) /* OS/2 version 5 (OpenType 1.7) */
555
556
static void
557
update_OS_2(ttf_OS_2_t *pos2, uint first_glyph, int num_glyphs)
558
8.17k
{
559
8.17k
    put_u16(pos2->usFirstCharIndex, first_glyph);
560
8.17k
    put_u16(pos2->usLastCharIndex, first_glyph + num_glyphs - 1);
561
8.17k
#if TT_ADJUST_OS_2
562
8.17k
    if (first_glyph >= 0xf000) {
563
        /* This font is being treated as a symbolic font. */
564
8.17k
        memset(pos2->ulUnicodeRanges, 0, sizeof(pos2->ulUnicodeRanges));
565
8.17k
        pos2->ulUnicodeRanges[7] = 8; /* bit 60, private use range */
566
8.17k
        memset(pos2->ulCodePageRanges, 0, sizeof(pos2->ulCodePageRanges));
567
8.17k
        pos2->ulCodePageRanges[3] = 1; /* bit 31, symbolic */
568
8.17k
    }
569
8.17k
#endif
570
8.17k
}
571
static void
572
write_OS_2(stream *s, gs_font *font, uint first_glyph, int num_glyphs)
573
8.17k
{
574
8.17k
    ttf_OS_2_t os2;
575
8.17k
    gs_font_info_t info;
576
8.17k
    int code;
577
578
    /*
579
     * We don't bother to set most of the fields.  The really important
580
     * ones, which affect character mapping, are usFirst/LastCharIndex.
581
     * We also need to set usWeightClass and usWidthClass to avoid
582
     * crashing ttfdump. Version 1 86-byte structure has all the fields
583
     * we need.
584
     */
585
8.17k
    memset(&os2, 0, sizeof(os2));
586
8.17k
    put_u16(os2.version, 1);
587
8.17k
    put_u16(os2.usWeightClass, 400); /* Normal */
588
8.17k
    put_u16(os2.usWidthClass, 5); /* Normal */
589
8.17k
    update_OS_2(&os2, first_glyph, num_glyphs);
590
591
    /*
592
     * We should also preserve the licensed embedding rights, to prevent
593
     * 'laundering' a TrueType font. These can be non-zero even when embedding is permitted.
594
     */
595
8.17k
    memset(&info, 0x00, sizeof(gs_font_info_t));
596
8.17k
    code = font->procs.font_info(font, NULL, FONT_INFO_EMBEDDING_RIGHTS, &info);
597
8.17k
    if (code == 0 && (info.members & FONT_INFO_EMBEDDING_RIGHTS)) {
598
5.39k
        put_u16(os2.fsType, info.EmbeddingRights);
599
5.39k
    }
600
601
8.17k
    stream_write(s, &os2, offset_of(ttf_OS_2_t, sxHeight[0]));
602
8.17k
    put_pad(s, offset_of(ttf_OS_2_t, sxHeight[0]));
603
8.17k
}
604
605
/* ------ post ------ */
606
607
/* Construct and then write the post table. */
608
typedef struct post_glyph_s {
609
    byte char_index;
610
    byte size;
611
    ushort glyph_index;
612
} post_glyph_t;
613
static int
614
compare_post_glyphs(const void *pg1, const void *pg2)
615
0
{
616
0
    gs_glyph g1 = ((const post_glyph_t *)pg1)->glyph_index,
617
0
        g2 = ((const post_glyph_t *)pg2)->glyph_index;
618
619
0
    return (g1 < g2 ? -1 : g1 > g2 ? 1 : 0);
620
0
}
621
typedef struct post_s {
622
    post_glyph_t glyphs[256 + 1];
623
    int count, glyph_count;
624
    uint length;
625
} post_t;
626
627
/*
628
 * If necessary, compute the length of the post table.  Note that we
629
 * only generate post entries for characters in the Encoding.
630
 */
631
static int
632
compute_post(gs_font *font, post_t *post)
633
0
{
634
0
    int i;
635
636
0
    for (i = 0, post->length = 32 + 2; i <= 255; ++i) {
637
0
        gs_const_string str;
638
0
        gs_glyph glyph = font->procs.encode_char(font, (gs_char)i,
639
0
                                                 GLYPH_SPACE_INDEX);
640
0
        int mac_index;
641
642
0
        int code = mac_glyph_index(font, i, &str, &mac_index);
643
0
        if (code < 0)
644
0
            return code;
645
0
        if (mac_index != 0) {
646
0
            post->glyphs[post->count].char_index = i;
647
0
            post->glyphs[post->count].size =
648
0
                (mac_index < 0 ? str.size + 1 : 0);
649
0
            post->glyphs[post->count].glyph_index = glyph - GS_MIN_GLYPH_INDEX;
650
0
            post->count++;
651
0
        }
652
0
    }
653
0
    if (post->count) {
654
0
        int j;
655
656
0
        qsort(post->glyphs, post->count, sizeof(post->glyphs[0]),
657
0
              compare_post_glyphs);
658
        /* Eliminate duplicate references to the same glyph. */
659
0
        for (i = j = 0; i < post->count; ++i) {
660
0
            if (i == 0 ||
661
0
                post->glyphs[i].glyph_index !=
662
0
                post->glyphs[i - 1].glyph_index
663
0
                ) {
664
0
                post->length += post->glyphs[i].size;
665
0
                post->glyphs[j++] = post->glyphs[i];
666
0
            }
667
0
        }
668
0
        post->count = j;
669
0
        post->glyph_count = post->glyphs[post->count - 1].glyph_index + 1;
670
0
    }
671
0
    post->length += post->glyph_count * 2;
672
0
    return 0;
673
0
}
674
675
/* Write the post table */
676
static int
677
write_post(stream *s, gs_font *font, post_t *post)
678
0
{
679
0
    byte post_initial[32 + 2];
680
0
    uint name_index;
681
0
    uint glyph_index;
682
0
    int i;
683
684
0
    memset(post_initial, 0, 32);
685
0
    put_u32(post_initial, 0x00020000);
686
0
    put_u16(post_initial + 32, post->glyph_count);
687
0
    stream_write(s, post_initial, sizeof(post_initial));
688
689
    /* Write the name index table. */
690
691
0
    for (i = 0, name_index = 258, glyph_index = 0; i < post->count; ++i) {
692
0
        gs_const_string str;
693
0
        int ch = post->glyphs[i].char_index;
694
0
        int mac_index;
695
0
        int code = mac_glyph_index(font, ch, &str, &mac_index);
696
697
0
        if (code < 0)
698
0
            return code;
699
0
        for (; glyph_index < post->glyphs[i].glyph_index; ++glyph_index)
700
0
            put_ushort(s, 0);
701
0
        glyph_index++;
702
0
        if (mac_index >= 0)
703
0
            put_ushort(s, mac_index);
704
0
        else {
705
0
            put_ushort(s, name_index);
706
0
            name_index++;
707
0
        }
708
0
    }
709
710
    /* Write the string names of the glyphs. */
711
712
0
    for (i = 0; i < post->count; ++i) {
713
0
        gs_const_string str;
714
0
        int ch = post->glyphs[i].char_index;
715
0
        int mac_index;
716
0
        int code = mac_glyph_index(font, ch, &str, &mac_index);
717
718
0
        if (code < 0)
719
0
            return code;
720
0
        if (mac_index < 0) {
721
0
            spputc(s, (byte)str.size);
722
0
            stream_write(s, str.data, str.size);
723
0
        }
724
0
    }
725
0
    put_pad(s, post->length);
726
0
    return 0;
727
0
}
728
729
static inline bool check_position(const gs_memory_t *mem, gs_offset_t pos1, gs_offset_t pos2)
730
123k
{
731
123k
    if (pos1 == pos2)
732
108k
        return false;
733
14.4k
    emprintf2(mem,
734
14.4k
              "Actual TT subtable offset %"PRId64" differs from one in the TT header %"PRId64".\n",
735
14.4k
              pos1,
736
14.4k
              pos2);
737
14.4k
    return true;
738
123k
}
739
740
static void remove_table(byte *tables, char *tag, uint *numTables)
741
132
{
742
    /* Not a high performance implementation because it is called seldom. */
743
132
    int i;
744
745
1.05k
    for (i = 0; i < *numTables;) {
746
924
        byte *tab = tables + i * 16;
747
748
924
        if (!memcmp(tab, tag, 4)) {
749
0
            memmove(tab, tab + 16, 16 * (*numTables - i - 1));
750
0
            --*numTables;
751
0
        } else
752
924
            ++i;
753
924
    }
754
132
}
755
756
/* ---------------- Main program ---------------- */
757
758
/* Write the definition of a TrueType font. */
759
static int
760
compare_table_tags(const void *pt1, const void *pt2)
761
1.48M
{
762
1.48M
    ulong t1 = u32(pt1), t2 = u32(pt2);
763
764
1.48M
    return (t1 < t2 ? -1 : t1 > t2 ? 1 : 0);
765
1.48M
}
766
static int
767
psf_write_truetype_data(stream *s, gs_font_type42 *pfont, int options,
768
                        psf_glyph_enum_t *penum, bool is_subset,
769
                        const gs_const_string *alt_font_name)
770
76.8k
{
771
76.8k
    gs_font *const font = (gs_font *)pfont;
772
76.8k
    gs_type42_data *d0 = &pfont->data;
773
76.8k
    gs_const_string font_name;
774
76.8k
    byte OffsetTable[12];
775
76.8k
    uint numTables_stored, numTables, numTables_out;
776
76.8k
    byte tables[MAX_NUM_TT_TABLES * 16];
777
76.8k
    uint i;
778
76.8k
    ulong offset;
779
76.8k
    gs_glyph glyph, glyph_prev;
780
76.8k
    ulong max_glyph;
781
76.8k
    uint glyf_length, loca_length;
782
76.8k
    ulong glyf_checksum = 0L; /****** NO CHECKSUM ******/
783
76.8k
    ulong loca_checksum[2] = {0L,0L};
784
76.8k
    ulong glyf_alignment = 0;
785
76.8k
    uint numGlyphs = 0;   /* original value from maxp */
786
76.8k
    byte head[56];    /* 0 mod 4 */
787
76.8k
    gs_type42_mtx_t mtx[2];
788
76.8k
    post_t post;
789
76.8k
    ulong head_checksum, file_checksum = 0;
790
76.8k
    int indexToLocFormat = 0;
791
76.8k
    bool
792
76.8k
        writing_cid = (options & WRITE_TRUETYPE_CID) != 0,
793
76.8k
        writing_stripped = (options & WRITE_TRUETYPE_STRIPPED) != 0,
794
76.8k
        generate_mtx = (options & WRITE_TRUETYPE_HVMTX) != 0,
795
76.8k
        no_generate = writing_cid | writing_stripped,
796
76.8k
        have_cmap = no_generate,
797
76.8k
        have_name = !(options & WRITE_TRUETYPE_NAME),
798
76.8k
        have_OS_2 = no_generate,
799
76.8k
        have_post = no_generate;
800
76.8k
    int have_hvhea[2];
801
76.8k
    uint cmap_length = 0;
802
76.8k
    ulong OS_2_start = 0;
803
76.8k
    uint OS_2_length = OS_2_LENGTH1;
804
76.8k
    ulong maxp_start = 0;
805
76.8k
    struct { int glyf, loca, cmap, name, os_2, mtx[2], post, head;
806
76.8k
           } subtable_positions;
807
76.8k
    gs_offset_t start_position = stell(s);
808
76.8k
    int enlarged_numGlyphs = 0;
809
76.8k
    int code;
810
76.8k
    int TTCFontOffset = 0;
811
812
76.8k
    memset(&subtable_positions, 0, sizeof(subtable_positions));
813
76.8k
    have_hvhea[0] = have_hvhea[1] = 0;
814
76.8k
    if (alt_font_name)
815
9.53k
        font_name = *alt_font_name;
816
67.2k
    else
817
67.2k
        font_name.data = font->font_name.chars,
818
67.2k
            font_name.size = font->font_name.size;
819
820
    /*
821
     * Count the number of tables, including the eventual glyf and loca
822
     * (which may not actually be present in the font), and copy the
823
     * table directory.
824
     */
825
826
76.8k
#define W(a,b,c,d)\
827
2.09M
  ( ((a) << 24) + ((b) << 16) + ((c) << 8) + (d))
828
829
76.8k
    TTCFontOffset = d0->subfontOffset;
830
76.8k
    READ_SFNTS(pfont, TTCFontOffset, 12, OffsetTable);
831
76.8k
    numTables_stored = U16(OffsetTable + 4);
832
1.03M
    for (i = numTables = 0; i < numTables_stored; ++i) {
833
956k
        byte tab[16];
834
956k
        byte data[54];
835
956k
        ulong start;
836
956k
        uint length;
837
838
956k
        if (numTables == MAX_NUM_TT_TABLES)
839
0
            return_error(gs_error_limitcheck);
840
956k
        READ_SFNTS(pfont, TTCFontOffset + 12 + i * 16, 16, tab);
841
956k
        start = u32(tab + 8);
842
956k
        length = u32(tab + 12);
843
        /* Copy the table data now (a rudiment of old code). */
844
956k
        memcpy(&tables[numTables * 16], tab, 16);
845
956k
        switch (u32(tab)) {
846
76.8k
        case W('h','e','a','d'):
847
76.8k
            if (length < 54)
848
0
                return_error(gs_error_invalidfont);
849
76.8k
            length = 54; /* bug 688409 fig2.eps has length=56. */
850
76.8k
            READ_SFNTS(pfont, start, length, data);
851
76.8k
            memcpy(head, data, length);
852
76.8k
            continue;
853
67.2k
        case W('g','l','y','f'): /* synthesized */
854
67.2k
        case W('g','l','y','x'): /* Adobe bogus */
855
134k
        case W('l','o','c','a'): /* synthesized */
856
134k
        case W('l','o','c','x'): /* Adobe bogus */
857
134k
        case W('g','d','i','r'): /* Adobe marker */
858
134k
            continue;
859
71.9k
        case W('c','m','a','p'):
860
71.9k
            if (options & (WRITE_TRUETYPE_CMAP | WRITE_TRUETYPE_CID))
861
11.2k
                continue;
862
60.6k
            have_cmap = true;
863
60.6k
            break;
864
76.2k
        case W('m','a','x','p'):
865
76.2k
            READ_SFNTS(pfont, start, length, data);
866
76.2k
            numGlyphs = U16(data + 4);
867
76.2k
            maxp_start = start;
868
76.2k
            break;
869
71.8k
        case W('n','a','m','e'):
870
71.8k
            if (writing_cid)
871
3.88k
                continue;
872
67.9k
            have_name = true;
873
67.9k
            break;
874
55.6k
        case W('O','S','/','2'):
875
55.6k
            if (writing_cid)
876
3.67k
                continue;
877
51.9k
            have_OS_2 = true;
878
51.9k
            if (length > OS_2_LENGTH5)
879
0
                return_error(gs_error_invalidfont);
880
51.9k
            OS_2_start = start;
881
51.9k
            OS_2_length = length;
882
51.9k
            continue;
883
70.2k
        case W('p','o','s','t'):
884
70.2k
            have_post = true;
885
70.2k
            break;
886
76.8k
        case W('h','h','e','a'):
887
76.8k
            have_hvhea[0] = 1;
888
76.8k
            break;
889
0
        case W('v','h','e','a'):
890
0
            have_hvhea[1] = 1;
891
0
            break;
892
76.4k
        case W('h','m','t','x'):
893
76.4k
        case W('v','m','t','x'):
894
76.4k
            if (generate_mtx)
895
9.51k
                continue;
896
            /* falls through */
897
142k
        case W('c','v','t',' '):
898
218k
        case W('f','p','g','m'):
899
293k
        case W('p','r','e','p'):
900
293k
            break;    /* always copy these if present */
901
676
        case W('D','S','I','G'):
902
756
        case W('E','B','D','T'):
903
836
        case W('E','B','L','C'):
904
836
        case W('E','B','S','C'):
905
1.51k
        case W('G','D','E','F'):
906
1.91k
        case W('G','P','O','S'):
907
5.70k
        case W('g','a','s','p'):
908
7.76k
        case W('k','e','r','n'):
909
7.76k
            continue;
910
11.8k
        default:
911
11.8k
            if (writing_cid)
912
2.11k
                continue;
913
9.75k
            break;
914
956k
        }
915
655k
        numTables++;
916
655k
    }
917
918
    /*
919
     * Enumerate the glyphs to get the size of glyf and loca,
920
     * and to compute the checksums for these tables.
921
     */
922
923
    /****** NO CHECKSUMS YET ******/
924
76.7k
    for (max_glyph = 0, glyf_length = 0;
925
303k
         (code = psf_enumerate_glyphs_next(penum, &glyph)) != 1;
926
226k
         ) {
927
226k
        uint glyph_index;
928
226k
        gs_glyph_data_t glyph_data;
929
930
226k
        if (glyph < GS_MIN_CID_GLYPH)
931
0
            return_error(gs_error_invalidfont);
932
226k
        glyph_index = glyph  & ~GS_GLYPH_TAG;
933
226k
        if_debug1m('L', s->memory, "[L]glyph_index %u\n", glyph_index);
934
226k
        glyph_data.memory = pfont->memory;
935
226k
        if ((code = pfont->data.get_outline(pfont, glyph_index, &glyph_data)) >= 0) {
936
            /* Since indexToLocFormat==0 assumes even glyph lengths,
937
               round it up here. If later we choose indexToLocFormat==1,
938
               subtract the glyf_alignment to compensate it. */
939
226k
            uint l = (glyph_data.bits.size + 1) & ~1;
940
941
226k
            max_glyph = max(max_glyph, glyph_index);
942
226k
            glyf_length += l;
943
226k
            if (l != glyph_data.bits.size)
944
175
                glyf_alignment++;
945
226k
            if_debug1m('L', s->memory, "[L]  size %u\n", glyph_data.bits.size);
946
226k
            gs_glyph_data_free(&glyph_data, "psf_write_truetype_data");
947
226k
        }
948
226k
    }
949
76.7k
    if (writing_stripped) {
950
67.2k
        glyf_length = 0;
951
67.2k
        loca_length = 0;
952
67.2k
    } else {
953
9.53k
        if (max_glyph + 1 > numGlyphs) {
954
            /* Either original font is wrong,
955
               or we added glyphs to it due to font merge.
956
               Need to adjust maxp, hmtx, vmtx, vdmx, hdmx,
957
               assuming that the merge doesn't change hhea
958
               and other tables.
959
               Since changing hdmx, vdmx is too difficult,
960
               and since they're not required for PDF,
961
               we'll simply skip them.
962
             */
963
66
            enlarged_numGlyphs = max_glyph + 1;
964
66
            if (enlarged_numGlyphs > 0xFFFF) {
965
0
                emprintf1(pfont->memory,
966
0
                          "The number of glyphs %d exceeds capability of True Type format.\n",
967
0
                          enlarged_numGlyphs);
968
0
                return_error(gs_error_unregistered);
969
0
            }
970
66
            loca_length = (enlarged_numGlyphs + 1) << 2;
971
66
            remove_table(tables, (char *)"hdmx", &numTables);
972
66
            remove_table(tables, (char *)"vdmx", &numTables);
973
66
        } else
974
9.46k
            loca_length = (numGlyphs + 1) << 2;
975
9.53k
        indexToLocFormat = (glyf_length > 0x1fffc);
976
9.53k
        if (!indexToLocFormat)
977
9.53k
            loca_length >>= 1;
978
0
        else
979
0
            glyf_length -= glyf_alignment;
980
        /* Acrobat Reader won't accept fonts with empty glyfs. */
981
9.53k
        if (glyf_length == 0)
982
981
            glyf_length = 1;
983
9.53k
    }
984
76.7k
    if_debug2m('l', s->memory, "[l]max_glyph = %lu, glyf_length = %lu\n",
985
76.7k
               (ulong)max_glyph, (ulong)glyf_length);
986
987
    /*
988
     * If necessary, compute the length of the post table.  Note that we
989
     * only generate post entries for characters in the Encoding.  */
990
991
76.7k
    if (!have_post) {
992
1.11k
        memset(&post, 0, sizeof(post));
993
1.11k
        if (options & WRITE_TRUETYPE_POST) {
994
0
            code = compute_post(font, &post);
995
0
            if (code < 0)
996
0
                return code;
997
0
        } else
998
1.11k
            post.length = 32; /* dummy table */
999
1.11k
    }
1000
1001
    /* Fix up the head table. */
1002
1003
76.7k
    memset(head + 8, 0, 4);
1004
76.7k
    head[50] = 0x00;
1005
76.7k
    head[51] = (byte)indexToLocFormat;
1006
76.7k
    memset(head + 54, 0, 2);
1007
1.15M
    for (head_checksum = 0, i = 0; i < 56; i += 4)
1008
1.07M
        head_checksum += u32(&head[i]);
1009
1010
    /*
1011
     * Construct the table directory, except for glyf, loca, head, OS/2,
1012
     * and, if necessary, generated cmap, name, and post tables.
1013
     * Note that the existing directory is already sorted by tag.
1014
     */
1015
1016
76.7k
    numTables_out = numTables + 1 /* head */
1017
76.7k
        + !writing_stripped * 2 /* glyf, loca */
1018
76.7k
        + generate_mtx * (have_hvhea[0] + have_hvhea[1]) /* hmtx, vmtx */
1019
76.7k
        + !have_OS_2    /* OS/2 */
1020
76.7k
        + !have_cmap + !have_name + !have_post;
1021
76.7k
    if (numTables_out >= MAX_NUM_TT_TABLES)
1022
0
        return_error(gs_error_limitcheck);
1023
76.7k
    offset = 12 + numTables_out * 16;
1024
732k
    for (i = 0; i < numTables; ++i) {
1025
655k
        byte *tab = &tables[i * 16];
1026
655k
        ulong length = u32(tab + 12);
1027
1028
655k
        offset += round_up(length, 4);
1029
655k
    }
1030
1031
    /* Make the table directory entries for generated tables. */
1032
1033
76.7k
    {
1034
76.7k
        byte *tab = &tables[numTables * 16];
1035
1036
76.7k
        if (!writing_stripped) {
1037
9.53k
            subtable_positions.glyf = offset;
1038
9.53k
            offset = put_table(tab, "glyf", glyf_checksum,
1039
9.53k
                               offset, glyf_length);
1040
9.53k
            tab += 16;
1041
9.53k
            subtable_positions.loca = offset;
1042
9.53k
            offset = put_table(tab, "loca", loca_checksum[indexToLocFormat],
1043
9.53k
                               offset, loca_length);
1044
9.53k
            tab += 16;
1045
9.53k
        }
1046
1047
76.7k
        if (!have_cmap) {
1048
8.17k
            cmap_length = size_cmap(font, TT_BIAS, 256,
1049
8.17k
                                    GS_MIN_GLYPH_INDEX + max_glyph, options);
1050
8.17k
            subtable_positions.cmap = offset;
1051
8.17k
            offset = put_table(tab, "cmap", 0L /****** NO CHECKSUM ******/,
1052
8.17k
                               offset, cmap_length);
1053
8.17k
            tab += 16;
1054
8.17k
        }
1055
1056
76.7k
        if (!have_name) {
1057
176
            subtable_positions.name = offset;
1058
176
            offset = put_table(tab, "name", 0L /****** NO CHECKSUM ******/,
1059
176
                               offset, size_name(&font_name));
1060
176
            tab += 16;
1061
176
        }
1062
1063
76.7k
        if (!no_generate) {
1064
8.17k
            subtable_positions.os_2 = offset;
1065
8.17k
            offset = put_table(tab, "OS/2", 0L /****** NO CHECKSUM ******/,
1066
8.17k
                               offset, OS_2_length);
1067
8.17k
            tab += 16;
1068
8.17k
        }
1069
1070
76.7k
        if (generate_mtx)
1071
28.6k
            for (i = 0; i < 2; ++i)
1072
19.0k
                if (have_hvhea[i]) {
1073
9.53k
                    subtable_positions.mtx[i] = offset;
1074
9.53k
                    offset = put_table(tab, (i ? "vmtx" : "hmtx"),
1075
9.53k
                                       0L /****** NO CHECKSUM ******/,
1076
9.53k
                                       offset,
1077
9.53k
                                       size_mtx(pfont, &mtx[i], max_glyph, numGlyphs, i));
1078
9.53k
                    tab += 16;
1079
9.53k
                }
1080
1081
76.7k
        if (!have_post) {
1082
1.11k
            subtable_positions.post = offset;
1083
1.11k
            offset = put_table(tab, "post", 0L /****** NO CHECKSUM ******/,
1084
1.11k
                               offset, post.length);
1085
1.11k
            tab += 16;
1086
1.11k
        }
1087
1088
        /*
1089
         * Note that the 'head' table must have length 54, even though
1090
         * it occupies 56 bytes on the file.
1091
         */
1092
76.7k
        subtable_positions.head = offset;
1093
76.7k
        (void)put_table(tab, "head", head_checksum, offset, 54);
1094
76.7k
    }
1095
76.7k
    numTables = numTables_out;
1096
1097
    /* Write the font header. */
1098
1099
76.7k
    {
1100
76.7k
        static const byte version[4] = {0, 1, 0, 0};
1101
1102
76.7k
        stream_write(s, version, 4);
1103
76.7k
    }
1104
76.7k
    put_ushort(s, numTables);
1105
382k
    for (i = 0; 1 << i <= numTables; ++i)
1106
305k
        DO_NOTHING;
1107
76.7k
    --i;
1108
76.7k
    put_ushort(s, 16 << i); /* searchRange */
1109
76.7k
    put_ushort(s, i);   /* entrySelectors */
1110
76.7k
    put_ushort(s, numTables * 16 - (16 << i)); /* rangeShift */
1111
1112
    /* Write the table directory. */
1113
1114
76.7k
    qsort(tables, numTables, 16, compare_table_tags);
1115
76.7k
    offset = 12 + numTables * 16;
1116
855k
    for (i = 0; i < numTables; ++i) {
1117
778k
        const byte *tab = &tables[i * 16];
1118
778k
        byte entry[16];
1119
1120
778k
        memcpy(entry, tab, 16);
1121
778k
        if (entry[8] < 0x40) {
1122
            /* Not a generated table. */
1123
653k
            uint length = u32(tab + 12);
1124
1125
653k
            put_u32(entry + 8, offset);
1126
653k
            offset += round_up(length, 4);
1127
653k
        } else {
1128
124k
            entry[8] -= 0x40;
1129
124k
        }
1130
778k
        stream_write(s, entry, 16);
1131
778k
    }
1132
1133
    /* Write tables other than the ones we generate here. */
1134
1135
855k
    for (i = 0; i < numTables; ++i) {
1136
778k
        const byte *tab = &tables[i * 16];
1137
1138
778k
        if (tab[8] < 0x40) {
1139
653k
            ulong start = u32(tab + 8);
1140
653k
            uint length = u32(tab + 12);
1141
1142
653k
            switch (u32(tab)) {
1143
0
            case W('O','S','/','2'):
1144
0
                if (!have_cmap) {
1145
                    /*
1146
                     * Adjust the first and last character indices in the OS/2
1147
                     * table to reflect the values in the generated cmap.
1148
                     */
1149
0
                    byte pos2[OS_2_LENGTH5];
1150
0
                    ttf_OS_2_t os2;
1151
1152
0
                    READ_SFNTS(pfont, OS_2_start, OS_2_length, pos2);
1153
0
                    memcpy(&os2, pos2, min(OS_2_length, sizeof(os2)));
1154
0
                    update_OS_2(&os2, TT_BIAS, 256);
1155
0
                    stream_write(s, &os2, OS_2_length);
1156
0
                    put_pad(s, OS_2_length);
1157
0
                } else {
1158
                    /* Just copy the existing OS/2 table. */
1159
0
                    write_range(s, pfont, OS_2_start, OS_2_length);
1160
0
                    put_pad(s, OS_2_length);
1161
0
                }
1162
0
            break;
1163
76.2k
            case W('m','a','x','p'):
1164
76.2k
                if (enlarged_numGlyphs) {
1165
                    /* Must keep the table size. */
1166
66
                    byte buf[6];
1167
1168
66
                    READ_SFNTS(pfont, maxp_start, sizeof(buf), buf);
1169
66
                    put_u16(buf + 4, enlarged_numGlyphs);
1170
66
                    stream_write(s, buf, min(length, sizeof(buf)));
1171
66
                    if (length > sizeof(buf)) /* Paranoid Safety */
1172
0
                        write_range(s, pfont, start + sizeof(buf), length - sizeof(buf));
1173
66
                } else
1174
76.1k
                    write_range(s, pfont, start, length);
1175
76.2k
                break;
1176
76.7k
            case W('h','h','e','a'):
1177
76.7k
            case W('v','h','e','a'):
1178
76.7k
                if (generate_mtx) {
1179
9.53k
                    write_range(s, pfont, start, length - 2); /* 34 */
1180
9.53k
                    put_ushort(s, mtx[tab[0] == 'v'].numMetrics);
1181
9.53k
                    break;
1182
9.53k
                }
1183
                /* falls through */
1184
567k
            default:
1185
567k
                write_range(s, pfont, start, length);
1186
653k
            }
1187
653k
            put_pad(s, length);
1188
653k
        }
1189
778k
    }
1190
1191
76.7k
    if (!writing_stripped) {
1192
9.53k
        int n = max(numGlyphs, enlarged_numGlyphs) + 1;
1193
1194
        /* Write glyf. */
1195
1196
9.53k
        if (check_position(pfont->memory,
1197
9.53k
                           subtable_positions.glyf + start_position,
1198
9.53k
                           stell(s)))
1199
0
            return_error(gs_error_unregistered);
1200
9.53k
        psf_enumerate_glyphs_reset(penum);
1201
236k
        for (offset = 0; psf_enumerate_glyphs_next(penum, &glyph) != 1; ) {
1202
226k
            gs_glyph_data_t glyph_data;
1203
1204
226k
            glyph_data.memory = pfont->memory;
1205
226k
            if ((code = pfont->data.get_outline(pfont,
1206
226k
                                                glyph & ~GS_GLYPH_TAG,
1207
226k
                                                &glyph_data)) >= 0
1208
226k
                ) {
1209
226k
                uint l = glyph_data.bits.size, zero = 0;
1210
1211
226k
                if (!indexToLocFormat)
1212
226k
                    l = (l + 1) & ~1;
1213
226k
                stream_write(s, glyph_data.bits.data, glyph_data.bits.size);
1214
226k
                if (glyph_data.bits.size < l)
1215
175
                    stream_write(s, &zero, 1);
1216
226k
                offset += l;
1217
226k
                if_debug2m('L', s->memory, "[L]glyf index = %u, size = %u\n",
1218
226k
                          i, glyph_data.bits.size);
1219
226k
                gs_glyph_data_free(&glyph_data, "psf_write_truetype_data");
1220
226k
            }
1221
226k
        }
1222
9.53k
        if_debug1m('l', s->memory, "[l]glyf final offset = %lu\n", offset);
1223
        /* Add a dummy byte if necessary to make glyf non-empty. */
1224
10.5k
        while (offset < glyf_length)
1225
981
            stream_putc(s, 0), ++offset;
1226
9.53k
        put_pad(s, (uint)offset);
1227
1228
        /* Write loca. */
1229
1230
9.53k
        if (check_position(pfont->memory,
1231
9.53k
                           subtable_positions.loca + start_position,
1232
9.53k
                           stell(s)))
1233
0
            return_error(gs_error_unregistered);
1234
9.53k
        psf_enumerate_glyphs_reset(penum);
1235
9.53k
        glyph_prev = 0;
1236
236k
        for (offset = 0; psf_enumerate_glyphs_next(penum, &glyph) != 1; ) {
1237
226k
            gs_glyph_data_t glyph_data;
1238
226k
            uint glyph_index = glyph & ~GS_GLYPH_TAG;
1239
1240
9.78M
            for (; glyph_prev <= glyph_index; ++glyph_prev)
1241
9.56M
                put_loca(s, offset, indexToLocFormat);
1242
226k
            glyph_data.memory = pfont->memory;
1243
226k
            if ((code = pfont->data.get_outline(pfont, glyph_index,
1244
226k
                                                &glyph_data)) >= 0
1245
226k
                ) {
1246
226k
                uint l = glyph_data.bits.size;
1247
1248
226k
                if (!indexToLocFormat)
1249
226k
                    l = (l + 1) & ~1;
1250
226k
                offset += l;
1251
226k
                gs_glyph_data_free(&glyph_data, "psf_write_truetype_data");
1252
226k
            }
1253
1254
226k
        }
1255
        /* Pad to numGlyphs + 1 entries (including the trailing entry). */
1256
45.3M
        for (; glyph_prev < n; ++glyph_prev)
1257
45.3M
            put_loca(s, offset, indexToLocFormat);
1258
9.53k
        put_pad(s, loca_length);
1259
1260
        /* If necessary, write cmap, name, and OS/2. */
1261
1262
9.53k
        if (!have_cmap) {
1263
8.17k
            if (check_position(pfont->memory,
1264
8.17k
                               subtable_positions.cmap + start_position,
1265
8.17k
                               stell(s)))
1266
0
                return_error(gs_error_unregistered);
1267
8.17k
            write_cmap(s, font, TT_BIAS, 256, GS_MIN_GLYPH_INDEX + max_glyph,
1268
8.17k
                       options, cmap_length);
1269
8.17k
        }
1270
9.53k
        if (!have_name) {
1271
176
            if (check_position(pfont->memory,
1272
176
                               subtable_positions.name + start_position,
1273
176
                               stell(s)))
1274
0
                return_error(gs_error_unregistered);
1275
176
            write_name(s, &font_name);
1276
176
        }
1277
9.53k
        if (!have_OS_2) {
1278
8.17k
            if (check_position(pfont->memory,
1279
8.17k
                               subtable_positions.os_2 + start_position,
1280
8.17k
                               stell(s)))
1281
0
                return_error(gs_error_unregistered);
1282
8.17k
            write_OS_2(s, font, TT_BIAS, 256);
1283
8.17k
        }
1284
1285
        /* If necessary, write [hv]mtx. */
1286
1287
9.53k
        if (generate_mtx)
1288
28.6k
            for (i = 0; i < 2; ++i)
1289
19.0k
                if (have_hvhea[i]) {
1290
9.53k
                    if (check_position(pfont->memory,
1291
9.53k
                                       subtable_positions.mtx[i] + start_position,
1292
9.53k
                                       stell(s)))
1293
0
                        return_error(gs_error_unregistered);
1294
9.53k
                    write_mtx(s, pfont, &mtx[i], i);
1295
9.53k
                    put_pad(s, mtx[i].length);
1296
9.53k
                }
1297
1298
        /* If necessary, write post. */
1299
1300
9.53k
        if (!have_post) {
1301
1.11k
            if (check_position(pfont->memory,
1302
1.11k
                               subtable_positions.post + start_position,
1303
1.11k
                               stell(s)))
1304
0
                return_error(gs_error_unregistered);
1305
1.11k
            if (options & WRITE_TRUETYPE_POST) {
1306
0
                code = write_post(s, font, &post);
1307
0
                if (code < 0)
1308
0
                    return code;
1309
1.11k
            } else {
1310
1.11k
                byte post_initial[32 + 2];
1311
1312
1.11k
                memset(post_initial, 0, 32);
1313
1.11k
                put_u32(post_initial, 0x00030000);
1314
1.11k
                stream_write(s, post_initial, 32);
1315
1.11k
            }
1316
1.11k
        }
1317
9.53k
    }
1318
1319
    /* Write head. */
1320
1321
    /****** CHECKSUM WAS NEVER COMPUTED ******/
1322
    /*
1323
     * The following nonsense is to avoid warnings about the constant
1324
     * 0xb1b0afbaL being "unsigned in ANSI C, signed with -traditional".
1325
     */
1326
76.7k
#if ARCH_SIZEOF_LONG > ARCH_SIZEOF_INT
1327
76.7k
#  define HEAD_MAGIC 0xb1b0afbaL
1328
#else
1329
#  define HEAD_MAGIC ((ulong)~0x4e4f5045)
1330
#endif
1331
76.7k
    put_u32(head + 8, HEAD_MAGIC - file_checksum); /* per spec */
1332
76.7k
#undef HEAD_MAGIC
1333
76.7k
    if (check_position(pfont->memory,
1334
76.7k
                       subtable_positions.head + start_position,
1335
76.7k
                       stell(s)))
1336
14.4k
        return_error(gs_error_unregistered);
1337
62.3k
    stream_write(s, head, 56);
1338
1339
62.3k
    return 0;
1340
76.7k
}
1341
1342
/* Write a TrueType font. */
1343
int
1344
psf_write_truetype_font(stream *s, gs_font_type42 *pfont, int options,
1345
                        gs_glyph *orig_subset_glyphs, uint orig_subset_size,
1346
                        const gs_const_string *alt_font_name)
1347
8.17k
{
1348
8.17k
    gs_font *const font = (gs_font *)pfont;
1349
8.17k
    psf_glyph_enum_t genum;
1350
8.17k
    gs_glyph subset_data[256 * MAX_COMPOSITE_PIECES];
1351
8.17k
    gs_glyph *subset_glyphs = orig_subset_glyphs;
1352
8.17k
    uint subset_size = orig_subset_size;
1353
1354
    /* Sort the subset glyphs, if any. */
1355
1356
8.17k
    if (subset_glyphs) {
1357
        /* Add the component glyphs for composites. */
1358
0
        int code;
1359
1360
0
        memcpy(subset_data, orig_subset_glyphs,
1361
0
               sizeof(gs_glyph) * subset_size);
1362
0
        subset_glyphs = subset_data;
1363
0
        code = psf_add_subset_pieces(subset_glyphs, &subset_size,
1364
0
                                     countof(subset_data),
1365
0
                                     countof(subset_data),
1366
0
                                     font);
1367
0
        if (code < 0)
1368
0
            return code;
1369
0
        subset_size = psf_sort_glyphs(subset_glyphs, subset_size);
1370
0
    }
1371
8.17k
    psf_enumerate_glyphs_begin(&genum, font, subset_glyphs,
1372
8.17k
                               (subset_glyphs ? subset_size : 0),
1373
8.17k
                               GLYPH_SPACE_INDEX);
1374
8.17k
    return psf_write_truetype_data(s, pfont, options & ~WRITE_TRUETYPE_CID,
1375
8.17k
                                   &genum, subset_glyphs != 0, alt_font_name);
1376
8.17k
}
1377
/* Write a stripped TrueType font. */
1378
int
1379
psf_write_truetype_stripped(stream *s, gs_font_type42 *pfont)
1380
62.2k
{
1381
62.2k
    psf_glyph_enum_t genum;
1382
62.2k
    byte no_subset = 0;
1383
1384
62.2k
    psf_enumerate_bits_begin(&genum, (gs_font *)pfont, &no_subset, 0,
1385
62.2k
                             GLYPH_SPACE_INDEX);
1386
62.2k
    return psf_write_truetype_data(s, pfont, WRITE_TRUETYPE_STRIPPED,
1387
62.2k
                                   &genum, true, NULL);
1388
62.2k
}
1389
1390
/* Write a CIDFontType 2 font. */
1391
int
1392
psf_write_cid2_font(stream *s, gs_font_cid2 *pfont, int options,
1393
                    const byte *subset_bits, uint subset_size,
1394
                    const gs_const_string *alt_font_name)
1395
1.36k
{
1396
1.36k
    gs_font *const font = (gs_font *)pfont;
1397
1.36k
    psf_glyph_enum_t genum;
1398
1399
1.36k
    psf_enumerate_bits_begin(&genum, font, subset_bits,
1400
1.36k
                             (subset_bits ? subset_size : 0),
1401
1.36k
                             GLYPH_SPACE_INDEX);
1402
1.36k
    return psf_write_truetype_data(s, (gs_font_type42 *)font,
1403
1.36k
                                   options | WRITE_TRUETYPE_CID, &genum,
1404
1.36k
                                   subset_bits != 0, alt_font_name);
1405
1.36k
}
1406
1407
/* Write a stripped CIDFontType 2 font. */
1408
int
1409
psf_write_cid2_stripped(stream *s, gs_font_cid2 *pfont)
1410
5.02k
{
1411
5.02k
    gs_font *const font = (gs_font *)pfont;
1412
5.02k
    psf_glyph_enum_t genum;
1413
5.02k
    byte no_subset = 0;
1414
1415
5.02k
    psf_enumerate_bits_begin(&genum, font, &no_subset, 0,
1416
5.02k
                             GLYPH_SPACE_INDEX);
1417
5.02k
    return psf_write_truetype_data(s, (gs_font_type42 *)font,
1418
5.02k
                                   WRITE_TRUETYPE_STRIPPED |
1419
5.02k
                                     WRITE_TRUETYPE_CID,
1420
5.02k
                                   &genum, true, NULL);
1421
5.02k
}