Coverage Report

Created: 2026-04-01 07:17

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ghostpdl/base/gsfont.c
Line
Count
Source
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
/* Font operators for Ghostscript library */
18
#include "gx.h"
19
#include "memory_.h"
20
#include "gserrors.h"
21
#include "gsstruct.h"
22
#include "gsutil.h"
23
#include "gxfixed.h"
24
#include "gxmatrix.h"
25
#include "gzstate.h"    /* must precede gxdevice */
26
#include "gxdevice.h"   /* must precede gxfont */
27
#include "gxfont.h"
28
#include "gxfcache.h"
29
#include "gzpath.h"   /* for default implementation */
30
31
/* Define the sizes of the various aspects of the font/character cache. */
32
/*** Big memory machines ***/
33
258k
#define smax_LARGE 50    /* smax - # of scaled fonts */
34
258k
#define bmax_LARGE 1000000  /* bmax - space for cached chars */
35
258k
#define mmax_LARGE 200    /* mmax - # of cached font/matrix pairs */
36
258k
#define cmax_LARGE 5000    /* cmax - # of cached chars */
37
258k
#define blimit_LARGE 32000  /* blimit/upper - max size of a single cached char */
38
/*** Small memory machines ***/
39
0
#define smax_SMALL 20    /* smax - # of scaled fonts */
40
0
#define bmax_SMALL 25000  /* bmax - space for cached chars */
41
0
#define mmax_SMALL 40    /* mmax - # of cached font/matrix pairs */
42
0
#define cmax_SMALL 500    /* cmax - # of cached chars */
43
0
#define blimit_SMALL 100  /* blimit/upper - max size of a single cached char */
44
45
/* Define a default procedure vector for fonts. */
46
const gs_font_procs gs_font_procs_default = {
47
    gs_no_define_font,    /* (actually a default) */
48
    gs_no_make_font,    /* (actually a default) */
49
    gs_default_font_info,
50
    gs_default_same_font,
51
    gs_no_encode_char,
52
    gs_no_decode_glyph,
53
    gs_no_enumerate_glyph,
54
    gs_default_glyph_info,
55
    gs_no_glyph_outline,
56
    gs_no_glyph_name,
57
    gs_default_init_fstack,
58
    gs_default_next_char_glyph,
59
    gs_no_build_char
60
};
61
62
private_st_font_dir();
63
static struct_proc_enum_ptrs(font_enum_ptrs);
64
static struct_proc_reloc_ptrs(font_reloc_ptrs);
65
66
public_st_gs_font_info();
67
public_st_gs_font();
68
public_st_gs_font_base();
69
private_st_gs_font_ptr();
70
public_st_gs_font_ptr_element();
71
72
/*
73
 * Garbage collection of fonts poses some special problems.  On the one
74
 * hand, we need to keep track of all existing base (not scaled) fonts,
75
 * using the next/prev list whose head is the orig_fonts member of the font
76
 * directory; on the other hand, we want these to be "weak" pointers that
77
 * don't keep fonts in existence if the fonts aren't referenced from
78
 * anywhere else.  We accomplish this as follows:
79
 *
80
 *     We don't trace through gs_font_dir.orig_fonts or gs_font.{next,prev}
81
 * during the mark phase of the GC.
82
 *
83
 *     When we finalize a base gs_font, we unlink it from the list.  (A
84
 * gs_font is a base font iff its base member points to itself.)
85
 *
86
 *     We *do* relocate the orig_fonts and next/prev pointers during the
87
 * relocation phase of the GC.  */
88
89
/* Font directory GC procedures */
90
static
91
8.16M
ENUM_PTRS_WITH(font_dir_enum_ptrs, gs_font_dir *dir)
92
1.39M
{
93
    /* Enumerate pointers from cached characters to f/m pairs, */
94
    /* and mark the cached character glyphs. */
95
    /* See gxfcache.h for why we do this here. */
96
1.39M
    uint cci = index - st_font_dir_max_ptrs;
97
1.39M
    uint offset, count;
98
1.39M
    uint tmask = dir->ccache.table_mask;
99
100
1.39M
    if (cci == 0)
101
846k
        offset = 0, count = 1;
102
549k
    else if (cci == dir->enum_index + 1)
103
549k
        offset = dir->enum_offset + 1, count = 1;
104
0
    else
105
0
        offset = 0, count = cci;
106
13.8G
    for (; offset <= tmask; ++offset) {
107
13.8G
        cached_char *cc = dir->ccache.table[offset];
108
109
13.8G
        if (cc != 0 && !--count) {
110
549k
            (*dir->ccache.mark_glyph)
111
549k
                (mem, cc->code, dir->ccache.mark_glyph_data);
112
            /****** HACK: break const.  We'll fix this someday. ******/
113
549k
            ((gs_font_dir *)dir)->enum_index = cci;
114
549k
            ((gs_font_dir *)dir)->enum_offset = offset;
115
549k
            ENUM_RETURN(cc_pair(cc) - cc->pair_index);
116
549k
        }
117
13.8G
    }
118
1.39M
}
119
846k
return 0;
120
6.77M
#define e1(i,elt) ENUM_PTR(i,gs_font_dir,elt);
121
8.16M
font_dir_do_ptrs(e1)
122
8.16M
#undef e1
123
8.16M
ENUM_PTRS_END
124
468k
static RELOC_PTRS_WITH(font_dir_reloc_ptrs, gs_font_dir *dir);
125
    /* Relocate the pointers from cached characters to f/m pairs. */
126
    /* See gxfcache.h for why we do this here. */
127
468k
{
128
468k
    int chi;
129
130
7.67G
    for (chi = dir->ccache.table_mask; chi >= 0; --chi) {
131
7.67G
        cached_char *cc = dir->ccache.table[chi];
132
133
7.67G
        if (cc != 0)
134
274k
            cc_set_pair_only(cc,
135
7.67G
                             (cached_fm_pair *)
136
7.67G
                             RELOC_OBJ(cc_pair(cc) - cc->pair_index) +
137
7.67G
                             cc->pair_index);
138
7.67G
    }
139
468k
}
140
    /* We have to relocate the cached characters before we */
141
    /* relocate dir->ccache.table! */
142
468k
RELOC_PTR(gs_font_dir, orig_fonts);
143
3.74M
#define r1(i,elt) RELOC_PTR(gs_font_dir, elt);
144
3.74M
font_dir_do_ptrs(r1)
145
468k
#undef r1
146
468k
RELOC_PTRS_END
147
148
/* GC procedures for fonts */
149
/*
150
 * When we finalize a base font, we unlink it from the orig_fonts list;
151
 * when we finalize a scaled font, we unlink it from scaled_fonts.
152
 * See above for more information.
153
 */
154
void
155
gs_font_finalize(const gs_memory_t *cmem, void *vptr)
156
2.15M
{
157
2.15M
    gs_font *const pfont = vptr;
158
2.15M
    gs_font **ppfirst;
159
2.15M
    gs_font *next = pfont->next;
160
2.15M
    gs_font *prev = pfont->prev;
161
2.15M
    (void)cmem; /* unused */
162
163
2.15M
    if_debug4m('u', cmem, "[u]unlinking font "PRI_INTPTR", base="PRI_INTPTR", prev="PRI_INTPTR", next="PRI_INTPTR"\n",
164
2.15M
               (intptr_t)pfont, (intptr_t)pfont->base, (intptr_t)prev, (intptr_t)next);
165
    /* Notify clients that the font is being freed. */
166
2.15M
    gs_notify_all(&pfont->notify_list, NULL);
167
2.15M
    gs_purge_font_from_char_caches(pfont);
168
2.15M
    if (pfont->dir == 0)
169
0
        ppfirst = 0;
170
2.15M
    else if (pfont->base == pfont)
171
2.12M
        ppfirst = &pfont->dir->orig_fonts;
172
24.7k
    else {
173
        /*
174
         * Track the number of cached scaled fonts.  Only decrement the
175
         * count if we didn't do this already in gs_makefont.
176
         */
177
24.7k
        if (next || prev || pfont->dir->scaled_fonts == pfont)
178
22.0k
            pfont->dir->ssize--;
179
24.7k
        ppfirst = &pfont->dir->scaled_fonts;
180
24.7k
    }
181
    /*
182
     * gs_purge_font may have unlinked this font already:
183
     * don't unlink it twice.
184
     */
185
2.15M
    if (next != 0 && next->prev == pfont)
186
639k
        next->prev = prev;
187
2.15M
    if (prev != 0) {
188
406k
        if (prev->next == pfont)
189
406k
            prev->next = next;
190
1.74M
    } else if (ppfirst != 0 && *ppfirst == pfont)
191
322k
        *ppfirst = next;
192
193
2.15M
    if (pfont->FontType != ft_composite) {
194
2.13M
        gs_font_base *pbfont = (gs_font_base *)pfont;
195
2.13M
        if (uid_is_XUID(&pbfont->UID)) {
196
68.6k
            uid_free(&pbfont->UID, pbfont->memory, "gs_font_finalize");
197
68.6k
        }
198
2.13M
    }
199
200
2.15M
    gs_notify_release(&pfont->notify_list);
201
2.15M
}
202
static
203
4.02M
ENUM_PTRS_WITH(font_enum_ptrs, gs_font *pfont) return ENUM_USING(st_gs_notify_list, &pfont->notify_list, sizeof(gs_notify_list_t), index - 5);
204
        /* We don't enumerate next or prev of base fonts. */
205
        /* See above for details. */
206
575k
case 0: ENUM_RETURN((pfont->base == pfont ? 0 : pfont->next));
207
575k
case 1: ENUM_RETURN((pfont->base == pfont ? 0 : pfont->prev));
208
4.02M
ENUM_PTR3(2, gs_font, dir, base, client_data);
209
4.02M
ENUM_PTRS_END
210
574k
static RELOC_PTRS_WITH(font_reloc_ptrs, gs_font *pfont);
211
574k
RELOC_USING(st_gs_notify_list, &pfont->notify_list, sizeof(gs_notify_list_t));
212
        /* We *do* always relocate next and prev. */
213
        /* Again, see above for details. */
214
574k
RELOC_PTR(gs_font, next);
215
574k
RELOC_PTR(gs_font, prev);
216
574k
RELOC_PTR3(gs_font, dir, base, client_data);
217
574k
RELOC_PTRS_END
218
219
/* Allocate a font directory */
220
static bool
221
cc_no_mark_glyph(const gs_memory_t *mem, gs_glyph glyph, void *ignore_data)
222
0
{
223
0
    return false;
224
0
}
225
gs_font_dir *
226
gs_font_dir_alloc2(gs_memory_t * struct_mem, gs_memory_t * bits_mem)
227
258k
{
228
258k
    gs_font_dir *pdir = 0;
229
230
258k
#if !ARCH_SMALL_MEMORY
231
#  ifdef DEBUG
232
    if (!gs_debug_c('.'))
233
#  endif
234
258k
    {       /* Try allocating a very large cache. */
235
        /* If this fails, allocate a small one. */
236
258k
        pdir = gs_font_dir_alloc2_limits(struct_mem, bits_mem,
237
258k
                                         smax_LARGE, bmax_LARGE, mmax_LARGE,
238
258k
                                         cmax_LARGE, blimit_LARGE);
239
258k
    }
240
258k
    if (pdir == 0)
241
0
#endif
242
0
        pdir = gs_font_dir_alloc2_limits(struct_mem, bits_mem,
243
0
                                         smax_SMALL, bmax_SMALL, mmax_SMALL,
244
0
                                         cmax_SMALL, blimit_SMALL);
245
258k
    if (pdir == NULL)
246
0
        return NULL;
247
258k
    pdir->ccache.mark_glyph = cc_no_mark_glyph;
248
258k
    pdir->ccache.mark_glyph_data = 0;
249
258k
    return pdir;
250
258k
}
251
gs_font_dir *
252
gs_font_dir_alloc2_limits(gs_memory_t * struct_mem, gs_memory_t * bits_mem,
253
                     uint smax, uint bmax, uint mmax, uint cmax, uint upper)
254
357k
{
255
357k
    gs_font_dir *pdir =
256
357k
        gs_alloc_struct(struct_mem, gs_font_dir, &st_font_dir,
257
357k
                        "font_dir_alloc(dir)");
258
357k
    int code;
259
260
357k
    if (pdir == NULL)
261
0
        return NULL;
262
357k
    memset(pdir, 0, sizeof(*pdir));
263
357k
    pdir->memory = struct_mem;
264
357k
    code = gx_char_cache_alloc(struct_mem, bits_mem, pdir,
265
357k
                               bmax, mmax, cmax, upper);
266
357k
    if (code < 0) {
267
0
        gs_free_object(struct_mem, pdir, "font_dir_alloc(dir)");
268
0
        return NULL;
269
0
    }
270
357k
    pdir->orig_fonts = 0;
271
357k
    pdir->scaled_fonts = 0;
272
357k
    pdir->ssize = 0;
273
357k
    pdir->smax = smax;
274
357k
    pdir->align_to_pixels = false;
275
357k
    pdir->glyph_to_unicode_table = NULL;
276
357k
    pdir->grid_fit_tt = 1;
277
357k
    pdir->tti = 0;
278
357k
    pdir->ttm = 0;
279
357k
    pdir->san = 0;
280
357k
    pdir->global_glyph_code = NULL;
281
357k
    pdir->text_enum_id = 0;
282
357k
    pdir->hash = 42;  /* initialize the hash to a randomly picked number */
283
357k
    return pdir;
284
357k
}
285
static void
286
gs_font_dir_finalize(const gs_memory_t *cmem, void *vptr)
287
357k
{
288
357k
    gs_font_dir *pdir = vptr;
289
357k
    gx_bits_cache_chunk *chunk = pdir->ccache.chunks;
290
357k
    gx_bits_cache_chunk *start_chunk = chunk;
291
357k
    gx_bits_cache_chunk *prev_chunk;
292
357k
    int i;
293
294
357k
    if (pdir == cmem->gs_lib_ctx->font_dir) {
295
182k
        cmem->gs_lib_ctx->font_dir = NULL;
296
182k
    }
297
298
71.8M
    for (i = 0; i < pdir->fmcache.mmax; i++) {
299
71.5M
        if (uid_is_XUID(&pdir->fmcache.mdata[i].UID)) {
300
63.9k
            gs_free_object(pdir->memory->stable_memory, pdir->fmcache.mdata[i].UID.xvalues, "gs_font_dir_finalize");
301
63.9k
        }
302
71.5M
    }
303
304
    /* free character cache machinery */
305
357k
    gs_free_object(pdir->memory, pdir->fmcache.mdata, "gs_font_dir_finalize");
306
357k
    gs_free_object(pdir->memory, pdir->ccache.table, "gs_font_dir_finalize");
307
308
    /* free the circular list of memory chunks */
309
399k
    while (chunk) {
310
399k
        if (start_chunk == chunk->next) {
311
357k
            gs_free_object(pdir->ccache.bits_memory, chunk->data, "gs_font_dir_finalize");
312
357k
            gs_free_object(pdir->ccache.bits_memory, chunk, "gs_font_dir_finalize");
313
357k
            break;
314
357k
        }
315
316
41.7k
        prev_chunk = chunk;
317
41.7k
        chunk = chunk->next;
318
41.7k
        gs_free_object(pdir->ccache.bits_memory, prev_chunk->data, "gs_font_dir_finalize");
319
41.7k
        gs_free_object(pdir->ccache.bits_memory, prev_chunk, "gs_font_dir_finalize");
320
41.7k
    }
321
357k
    pdir->ccache.chunks = NULL;
322
357k
}
323
void
324
gs_font_dir_free(gs_font_dir *dir)
325
8.09k
{
326
8.09k
    if (dir == NULL)
327
0
        return;
328
8.09k
    gs_free_object(dir->memory, dir, "gs_font_dir_free");
329
8.09k
}
330
331
/* Allocate and minimally initialize a font. */
332
gs_font *
333
gs_font_alloc(gs_memory_t *mem, gs_memory_type_ptr_t pstype,
334
              const gs_font_procs *procs, gs_font_dir *dir,
335
              client_name_t cname)
336
287k
{
337
287k
    gs_font *pfont = gs_alloc_struct(mem, gs_font, pstype, cname);
338
339
287k
    if (pfont == 0)
340
0
        return 0;
341
287k
#if 1 /* Clear entire structure to avoid unitialized pointers
342
         when the initialization exits prematurely by error. */
343
287k
    memset(pfont, 0, pstype->ssize);
344
287k
    pfont->memory = mem;
345
287k
    pfont->dir = dir;
346
287k
    gs_font_notify_init(pfont);
347
287k
    pfont->id = gs_next_ids(mem, 1);
348
287k
    pfont->base = pfont;
349
287k
    pfont->ExactSize = pfont->InBetweenSize = pfont->TransformedChar =
350
287k
        fbit_use_outlines;
351
287k
    pfont->procs = *procs;
352
#else
353
    /* For clarity we leave old initializations here
354
       to know which fields needs to be initialized. */
355
    pfont->next = pfont->prev = 0;
356
    pfont->memory = mem;
357
    pfont->dir = dir;
358
    pfont->is_resource = false;
359
    gs_font_notify_init(pfont);
360
    pfont->id = gs_next_ids(mem, 1);
361
    pfont->base = pfont;
362
    pfont->client_data = 0;
363
    /* not FontMatrix, FontType */
364
    pfont->BitmapWidths = false;
365
    pfont->ExactSize = pfont->InBetweenSize = pfont->TransformedChar =
366
        fbit_use_outlines;
367
    pfont->WMode = 0;
368
    pfont->PaintType = 0;
369
    pfont->StrokeWidth = 0;
370
    pfont->is_cached = false;
371
    pfont->procs = *procs;
372
    memset(&pfont->orig_FontMatrix, 0, sizeof(pfont->orig_FontMatrix));
373
#endif
374
    /* not key_name, font_name */
375
287k
    return pfont;
376
287k
}
377
/* Allocate and minimally initialize a base font. */
378
gs_font_base *
379
gs_font_base_alloc(gs_memory_t *mem, gs_memory_type_ptr_t pstype,
380
                   const gs_font_procs *procs, gs_font_dir *dir,
381
                   client_name_t cname)
382
0
{
383
0
    gs_font_base *pfont =
384
0
        (gs_font_base *)gs_font_alloc(mem, pstype, procs, dir, cname);
385
386
0
    if (pfont == 0)
387
0
        return 0;
388
0
    pfont->FontBBox.p.x = pfont->FontBBox.p.y =
389
0
        pfont->FontBBox.q.x = pfont->FontBBox.q.y = 0;
390
0
    uid_set_invalid(&pfont->UID);
391
0
    pfont->encoding_index = pfont->nearest_encoding_index = -1;
392
0
    return pfont;
393
0
}
394
395
/* Initialize the notification list for a font. */
396
void
397
gs_font_notify_init(gs_font *font)
398
311k
{
399
    /*
400
     * The notification list for a font must be allocated in the font's
401
     * stable memory, because of the following possible sequence of events:
402
     *
403
     *   - Allocate font X in local VM.
404
     *   - Client A registers for notification when X is freed.
405
     *   - 'save'
406
     *   - Client B registers for notification when X is freed.
407
     *   - 'restore'
408
     *
409
     * If the notification list element for client B is allocated in
410
     * restorable local VM (i.e., the same VM as the font), then when the
411
     * 'restore' occurs, either the list element will be deleted (not what
412
     * client B wants, because font X hasn't been freed yet), or there will
413
     * be a dangling pointer.
414
     */
415
311k
    gs_notify_init(&font->notify_list, gs_memory_stable(font->memory));
416
311k
}
417
418
/*
419
 * Register/unregister a client for notification by a font.  Currently
420
 * the clients are only notified when a font is freed.  Note that any
421
 * such client must unregister itself when *it* is freed.
422
 */
423
int
424
gs_font_notify_register(gs_font *font, gs_notify_proc_t proc, void *proc_data)
425
1.12M
{
426
1.12M
    return gs_notify_register(&font->notify_list, proc, proc_data);
427
1.12M
}
428
int
429
gs_font_notify_unregister(gs_font *font, gs_notify_proc_t proc, void *proc_data)
430
1.12M
{
431
1.12M
    return gs_notify_unregister(&font->notify_list, proc, proc_data);
432
1.12M
}
433
434
/* Link an element at the head of a chain. */
435
static void
436
font_link_first(gs_font **pfirst, gs_font *elt)
437
2.02M
{
438
2.02M
    gs_font *first = elt->next = *pfirst;
439
440
2.02M
    if (first)
441
1.76M
        first->prev = elt;
442
2.02M
    elt->prev = 0;
443
2.02M
    *pfirst = elt;
444
2.02M
}
445
446
/* definefont */
447
/* Use this only for original (unscaled) fonts! */
448
/* Note that it expects pfont->procs.define_font to be set already. */
449
int
450
gs_definefont(gs_font_dir * pdir, gs_font * pfont)
451
1.99M
{
452
1.99M
    int code;
453
454
1.99M
    pfont->dir = pdir;
455
1.99M
    pfont->base = pfont;
456
1.99M
    code = (*pfont->procs.define_font) (pdir, pfont);
457
1.99M
    if (code < 0) {   /* Make sure we don't try to finalize this font. */
458
0
        pfont->base = 0;
459
0
        return code;
460
0
    }
461
1.99M
    font_link_first(&pdir->orig_fonts, pfont);
462
1.99M
    if_debug2m('m', pfont->memory, "[m]defining font "PRI_INTPTR", next="PRI_INTPTR"\n",
463
1.99M
               (intptr_t)pfont, (intptr_t)pfont->next);
464
1.99M
    return 0;
465
1.99M
}
466
467
/* Find a sililar registered font of same type. */
468
int
469
gs_font_find_similar(const gs_font_dir * pdir, const gs_font **ppfont,
470
                       int (*similar)(const gs_font *, const gs_font *))
471
157
{
472
157
    const gs_font *pfont0 = *ppfont;
473
157
    const gs_font *pfont1 = pdir->orig_fonts;
474
475
469
    for (; pfont1 != NULL; pfont1 = pfont1->next) {
476
312
        if (pfont1 != pfont0 && pfont1->FontType == pfont0->FontType) {
477
155
            int code = similar(pfont0, pfont1);
478
155
            if (code != 0) {
479
0
                *ppfont = pfont1;
480
0
                return code;
481
0
            }
482
155
        }
483
312
    }
484
157
    return 0;
485
157
}
486
487
/* scalefont */
488
int
489
gs_scalefont(gs_font_dir * pdir, const gs_font * pfont, double scale,
490
             gs_font ** ppfont)
491
0
{
492
0
    gs_matrix mat;
493
494
0
    gs_make_scaling(scale, scale, &mat);
495
0
    return gs_makefont(pdir, pfont, &mat, ppfont);
496
0
}
497
498
/* makefont */
499
int
500
gs_makefont(gs_font_dir * pdir, const gs_font * pfont,
501
            const gs_matrix * pmat, gs_font ** ppfont)
502
33.7k
{
503
33.7k
    int code;
504
33.7k
    gs_font *prev = 0;
505
33.7k
    gs_font *pf_out = pdir->scaled_fonts;
506
33.7k
    gs_memory_t *mem = pfont->memory;
507
33.7k
    gs_matrix newmat;
508
33.7k
    bool can_cache;
509
510
33.7k
    if ((code = gs_matrix_multiply(&pfont->FontMatrix, pmat, &newmat)) < 0)
511
0
        return code;
512
    /*
513
     * Check for the font already being in the scaled font cache.
514
     * Until version 5.97, we only cached scaled fonts if the base
515
     * (unscaled) font had a valid UniqueID or XUID; now, we will cache
516
     * scaled versions of any non-composite font.
517
     */
518
#ifdef DEBUG
519
    if (gs_debug_c('m')) {
520
        const gs_font_base *const pbfont = (const gs_font_base *)pfont;
521
522
        if (pfont->FontType == ft_composite)
523
            dmlprintf(mem, "[m]composite");
524
        else if (uid_is_UniqueID(&pbfont->UID))
525
            dmlprintf1(mem, "[m]UniqueID=%ld", pbfont->UID.id);
526
        else if (uid_is_XUID(&pbfont->UID))
527
            dmlprintf1(mem, "[m]XUID(%u)", (uint) (-pbfont->UID.id));
528
        else
529
            dmlprintf(mem, "[m]no UID");
530
        dmprintf8(mem, ", FontType=%d, base="PRI_INTPTR",\n[m]  new FontMatrix=[%g %g %g %g %g %g]\n",
531
                 pfont->FontType, (intptr_t)pfont->base,
532
                 pmat->xx, pmat->xy, pmat->yx, pmat->yy,
533
                 pmat->tx, pmat->ty);
534
    }
535
#endif
536
    /*
537
     * Don't try to cache scaled composite fonts, because of the side
538
     * effects on FDepVector and descendant fonts that occur in makefont.
539
     */
540
33.7k
    if (pfont->FontType != ft_composite) {
541
361k
        for (; pf_out != 0; prev = pf_out, pf_out = pf_out->next)
542
336k
            if (pf_out->FontType == pfont->FontType &&
543
336k
                pf_out->base == pfont->base &&
544
226k
                pf_out->FontMatrix.xx == newmat.xx &&
545
39.4k
                pf_out->FontMatrix.xy == newmat.xy &&
546
39.4k
                pf_out->FontMatrix.yx == newmat.yx &&
547
39.4k
                pf_out->FontMatrix.yy == newmat.yy &&
548
39.4k
                pf_out->FontMatrix.tx == newmat.tx &&
549
9.06k
                pf_out->FontMatrix.ty == newmat.ty
550
336k
                ) {
551
9.06k
                *ppfont = pf_out;
552
9.06k
                if_debug1m('m', pfont->memory, "[m]found font="PRI_INTPTR"\n", (intptr_t)pf_out);
553
9.06k
                return 0;
554
9.06k
            }
555
24.7k
        can_cache = true;
556
24.7k
    } else
557
0
        can_cache = false;
558
24.7k
    pf_out = gs_alloc_struct(mem, gs_font, gs_object_type(mem, pfont),
559
24.7k
                             "gs_makefont");
560
24.7k
    if (!pf_out)
561
0
        return_error(gs_error_VMerror);
562
24.7k
    memcpy(pf_out, pfont, gs_object_size(mem, pfont));
563
24.7k
    gs_font_notify_init(pf_out);
564
24.7k
    pf_out->FontMatrix = newmat;
565
24.7k
    pf_out->client_data = 0;
566
24.7k
    pf_out->dir = pdir;
567
24.7k
    pf_out->base = pfont->base;
568
24.7k
    *ppfont = pf_out;
569
24.7k
    code = (*pf_out->procs.make_font) (pdir, pfont, pmat, ppfont);
570
24.7k
    if (code < 0)
571
0
        return code;
572
24.7k
    if (can_cache) {
573
24.7k
        if (pdir->ssize >= pdir->smax && prev != 0) {
574
            /*
575
             * We must discard a cached scaled font.
576
             * prev points to the last (oldest) font.
577
             * (We can't free it, because there might be
578
             * other references to it.)
579
             */
580
2.63k
            if_debug1m('m', pfont->memory, "[m]discarding font "PRI_INTPTR"\n",
581
2.63k
                      (intptr_t)prev);
582
2.63k
            if (prev->prev != 0)
583
2.63k
                prev->prev->next = 0;
584
0
            else
585
0
                pdir->scaled_fonts = 0;
586
2.63k
            pdir->ssize--;
587
2.63k
            prev->prev = 0;
588
            /* This comment is a relatively new reconstruction of old assumptions,
589
               which were done 5+ years ago (see gsfont.c revision 1.1).
590
               Here the font is only removed from the pdir->scaled_fonts list
591
               to prevent the latter to grow huge. Thus the list is used only to
592
               merge scaled font duplicates by the 'for' loop in the beginning
593
               of this function. We do not discard related character rasters
594
               from character cache due to 3 reasons :
595
               1. At this point a cached_char instance may be referred
596
                  by one or more gs_show_enum instances, which may exist on the
597
                  PS estack while execution of a Type 3 BuildChar or BuildGlyph.
598
                  Such event really happens while rendering a re-distilled tpc2.ps .
599
                  We must not remove those isntances, but currently there is no
600
                  mechanizm for distinguishing them from othgers.
601
               2. If the font has an UID, another scaled font may use same fm_pair
602
                  instance due to different CTMs. Therefore same character rasters
603
                  may be useful for another scaled font.
604
               3. We don't know whether the font will be used again in nearest
605
                  future. Maybe it will be used again in the next 'show' operation.
606
                  Therefore we delay the decision about discarding character
607
                  rasters untill we need to release memory from them.
608
               4. Also note that the last created font, rather than the last used one,
609
                  is being discarded. An useful improvement would be
610
                  to move a font t the beginning of the list whenever it
611
                  appears in a show-like operation.
612
             */
613
#if 0     /* We disabled this code portion due to Bug 688392.
614
               The problem was dangling pointers, which appear in fm_pair instances
615
               after uid_free is applied to a font's UID,
616
               because they share same xvalues array. We're unable to guess
617
               for which reason uid_free was applied to the font's UID here
618
               5+ years ago (see gsfont.c revision 1.1).
619
               We do not remove this code portion until we get
620
               a complete understanding.
621
             */
622
            if (prev->FontType != ft_composite) {
623
                if_debug1m('m', pfont->memory, "[m]discarding UID 0x%lx\n",
624
                           (ulong) ((gs_font_base *) prev)->
625
                           UID.xvalues);
626
                uid_free(&((gs_font_base *) prev)->UID,
627
                         prev->memory,
628
                         "gs_makefont(discarding)");
629
                uid_set_invalid(&((gs_font_base *) prev)->UID);
630
            }
631
#endif
632
2.63k
        }
633
24.7k
        pdir->ssize++;
634
24.7k
        font_link_first(&pdir->scaled_fonts, pf_out);
635
24.7k
    } else {     /* Prevent garbage pointers. */
636
0
        pf_out->next = pf_out->prev = 0;
637
0
    }
638
24.7k
    if_debug2m('m', pfont->memory, "[m]new font="PRI_INTPTR" can_cache=%s\n",
639
24.7k
               (intptr_t)*ppfont, (can_cache ? "true" : "false"));
640
24.7k
    return 1;
641
24.7k
}
642
643
/* Set the current font.  This is provided only for the benefit of cshow, */
644
/* which must reset the current font without disturbing the root font. */
645
void
646
gs_set_currentfont(gs_gstate * pgs, gs_font * pfont)
647
1.43M
{
648
1.43M
    pgs->font = pfont;
649
1.43M
    pgs->char_tm_valid = false;
650
1.43M
}
651
652
/* setfont */
653
int
654
gs_setfont(gs_gstate * pgs, gs_font * pfont)
655
14.1M
{
656
14.1M
    pgs->font = pgs->root_font = pfont;
657
14.1M
    pgs->char_tm_valid = false;
658
14.1M
    return 0;
659
14.1M
}
660
661
/* currentfont */
662
gs_font *
663
gs_currentfont(const gs_gstate * pgs)
664
2.35M
{
665
2.35M
    return pgs->font;
666
2.35M
}
667
668
/* rootfont */
669
gs_font *
670
gs_rootfont(const gs_gstate * pgs)
671
29.9M
{
672
29.9M
    return pgs->root_font;
673
29.9M
}
674
675
/* cachestatus */
676
void
677
gs_cachestatus(register const gs_font_dir * pdir, register uint pstat[7])
678
1.09M
{
679
1.09M
    pstat[0] = pdir->ccache.bsize;
680
1.09M
    pstat[1] = pdir->ccache.bmax;
681
1.09M
    pstat[2] = pdir->fmcache.msize;
682
1.09M
    pstat[3] = pdir->fmcache.mmax;
683
1.09M
    pstat[4] = pdir->ccache.csize;
684
1.09M
    pstat[5] = pdir->ccache.cmax;
685
1.09M
    pstat[6] = pdir->ccache.upper;
686
1.09M
}
687
688
/* setcacheparams */
689
int
690
gs_setcachesize(gs_gstate * pgs, gs_font_dir * pdir, uint size)
691
0
{
692
0
    gs_memory_t *stable_mem = pdir->memory->stable_memory;
693
0
    if (size < 100000)             /* limits derived from CPSI emulation (CET 27-07) */
694
0
        size = 100000;
695
0
    else if (size > 100000000)
696
0
        size = 100000000;
697
698
    /* Changing the cache size precipitates rebuilding the cache data
699
       structures.  Start with freeing cached chars and fm pairs. */
700
0
    {
701
0
        gs_font *pfont;
702
0
        int code;
703
0
        for (pfont = pdir->scaled_fonts; pfont != 0; pfont = pfont->next) {
704
0
            code = gs_purge_font_from_char_caches_completely(pfont);
705
0
            if (code != 0)
706
0
                gs_rethrow_code(code);
707
708
0
        }
709
0
    }
710
711
    /* now free the cache structures and rebuild everything with the
712
       new cache size */
713
0
    gs_free_object(stable_mem, pdir->fmcache.mdata, "gs_setcachesize(mdata)");
714
0
    gs_free_object(stable_mem, pdir->ccache.table, "gs_setcachesize(table)");
715
0
    pdir->ccache.bmax = size;
716
0
    return gx_char_cache_alloc(stable_mem, stable_mem->non_gc_memory, pdir,
717
0
                               pdir->ccache.bmax, pdir->fmcache.mmax,
718
0
                               pdir->ccache.cmax, pdir->ccache.upper);
719
0
}
720
721
int
722
gs_setcachelower(gs_font_dir * pdir, uint size)
723
1.14M
{
724
1.14M
    pdir->ccache.lower = ((int)size < 0) ? 0 : size; /* ?: for CET 27-07 */
725
1.14M
    return 0;
726
1.14M
}
727
int
728
gs_setcacheupper(gs_font_dir * pdir, uint size)
729
1.14M
{
730
1.14M
    pdir->ccache.upper = ((int)size < 0) ? 0 : size; /* ?: for CET 27-06 */
731
1.14M
    return 0;
732
1.14M
}
733
int
734
gs_setaligntopixels(gs_font_dir * pdir, uint v)
735
607k
{
736
607k
    pdir->align_to_pixels = v;
737
607k
    return 0;
738
607k
}
739
int
740
gs_setgridfittt(gs_font_dir * pdir, uint v)
741
607k
{
742
607k
    pdir->grid_fit_tt = v;
743
607k
    return 0;
744
607k
}
745
746
/* currentcacheparams */
747
uint
748
gs_currentcachesize(const gs_font_dir * pdir)
749
1.09M
{
750
1.09M
    return pdir->ccache.bmax;
751
1.09M
}
752
uint
753
gs_currentcachelower(const gs_font_dir * pdir)
754
1.09M
{
755
1.09M
    return pdir->ccache.lower;
756
1.09M
}
757
uint
758
gs_currentcacheupper(const gs_font_dir * pdir)
759
1.09M
{
760
1.09M
    return pdir->ccache.upper;
761
1.09M
}
762
uint
763
gs_currentaligntopixels(const gs_font_dir * pdir)
764
79.4M
{
765
79.4M
    return pdir->align_to_pixels;
766
79.4M
}
767
uint
768
gs_currentgridfittt(const gs_font_dir * pdir)
769
797k
{
770
797k
    return pdir->grid_fit_tt;
771
797k
}
772
773
/* Purge a font from all font- and character-related tables. */
774
/* This is only used by restore (and, someday, the GC). */
775
int
776
gs_purge_font(gs_font * pfont)
777
1.28M
{
778
1.28M
    gs_font_dir *pdir = pfont->dir;
779
1.28M
    gs_font *pf;
780
781
    /* Remove the font from its list (orig_fonts or scaled_fonts). */
782
1.28M
    gs_font *prev = pfont->prev;
783
1.28M
    gs_font *next = pfont->next;
784
785
1.28M
    if (next != 0)
786
1.09M
        next->prev = prev, pfont->next = 0;
787
1.28M
    if (prev != 0)
788
0
        prev->next = next, pfont->prev = 0;
789
1.28M
    else if (pdir->orig_fonts == pfont)
790
1.28M
        pdir->orig_fonts = next;
791
3
    else if (pdir->scaled_fonts == pfont)
792
0
        pdir->scaled_fonts = next;
793
3
    else {     /* Shouldn't happen! */
794
3
        if_debug1m('u', pfont->memory, "purged font "PRI_INTPTR" not found\n", (intptr_t)pfont);
795
3
    }
796
797
    /* Purge the font from the scaled font cache. */
798
1.28M
    for (pf = pdir->scaled_fonts; pf != 0;) {
799
0
        if (pf->base == pfont) {
800
0
            int code = gs_purge_font(pf);
801
802
0
            if (code < 0)
803
0
                return code;
804
0
            pf = pdir->scaled_fonts;  /* start over */
805
0
        } else
806
0
            pf = pf->next;
807
0
    }
808
809
    /* Purge the font from the font/matrix pair cache, */
810
    /* including all cached characters rendered with that font. */
811
1.28M
    return gs_purge_font_from_char_caches(pfont);
812
1.28M
}
813
814
/* Locate a gs_font by gs_id. */
815
gs_font *
816
gs_find_font_by_id(gs_font_dir *pdir, gs_id id, gs_matrix *FontMatrix)
817
0
 {
818
0
    gs_font *pfont = pdir->orig_fonts;
819
820
0
    for(; pfont != NULL; pfont = pfont->next)
821
0
        if(pfont->id == id &&
822
0
            !gs_matrix_compare(&pfont->FontMatrix, FontMatrix))
823
0
            return pfont;
824
0
    return NULL;
825
0
 }
826
827
/* ---------------- Default font procedures ---------------- */
828
829
/* ------ Font-level procedures ------ */
830
831
/* Default (vacuous) definefont handler. */
832
int
833
gs_no_define_font(gs_font_dir * pdir, gs_font * pfont)
834
1.99M
{
835
1.99M
    return 0;
836
1.99M
}
837
838
/* Default (vacuous) makefont handler. */
839
int
840
gs_no_make_font(gs_font_dir * pdir, const gs_font * pfont,
841
                const gs_matrix * pmat, gs_font ** ppfont)
842
0
{
843
0
    return 0;
844
0
}
845
/* Makefont handler for base fonts, which must copy the XUID. */
846
int
847
gs_base_make_font(gs_font_dir * pdir, const gs_font * pfont,
848
                  const gs_matrix * pmat, gs_font ** ppfont)
849
24.7k
{
850
24.7k
    return uid_copy(&((gs_font_base *)*ppfont)->UID, (*ppfont)->memory,
851
24.7k
                    "gs_base_make_font(XUID)");
852
24.7k
}
853
854
/* Default font info procedure */
855
int
856
gs_default_font_info(gs_font *font, const gs_point *pscale, int members,
857
                     gs_font_info_t *info)
858
143k
{
859
143k
    int wmode = font->WMode;
860
143k
    gs_font_base *bfont = (gs_font_base *)font;
861
143k
    gs_point scale;
862
143k
    gs_matrix smat;
863
143k
    const gs_matrix *pmat;
864
865
143k
    if (pscale == 0) {
866
143k
        scale.x = scale.y = 0;
867
143k
        pmat = 0;
868
143k
    } else {
869
0
        scale = *pscale;
870
0
        gs_make_scaling(scale.x, scale.y, &smat);
871
0
        pmat = &smat;
872
0
    }
873
143k
    info->members = 0;
874
143k
    if (members & FONT_INFO_FLAGS)
875
84.8k
        info->Flags_returned = 0;
876
143k
    if (font->FontType == ft_composite)
877
0
        return 0;   /* nothing available */
878
143k
    if (members & FONT_INFO_BBOX) {
879
84.8k
        info->BBox.p.x = (int)bfont->FontBBox.p.x;
880
84.8k
        info->BBox.p.y = (int)bfont->FontBBox.p.y;
881
84.8k
        info->BBox.q.x = (int)bfont->FontBBox.q.x;
882
84.8k
        info->BBox.q.y = (int)bfont->FontBBox.q.y;
883
84.8k
        info->Flags_returned |= FONT_INFO_BBOX;
884
84.8k
    }
885
143k
    if ((members & FONT_INFO_FLAGS) &&
886
84.8k
        (info->Flags_requested & FONT_IS_FIXED_WIDTH)
887
143k
        ) {
888
        /*
889
         * Scan the glyph space to compute the fixed width if any.
890
         */
891
84.8k
        gs_glyph notdef = GS_NO_GLYPH;
892
84.8k
        gs_glyph glyph;
893
84.8k
        int fixed_width = 0;
894
84.8k
        int index;
895
84.8k
        int code = 0; /* Quiet compiler. */
896
84.8k
        int ecode = 0;
897
84.8k
        bool has_glyphs = false;
898
899
84.8k
        for (index = 0;
900
5.34M
             fixed_width >= 0 &&
901
5.27M
             (code = font->procs.enumerate_glyph(font, &index, GLYPH_SPACE_NAME, &glyph)) >= 0 &&
902
5.27M
                 index != 0;
903
5.25M
             ) {
904
5.25M
            gs_glyph_info_t glyph_info;
905
906
5.25M
            memset(&glyph_info, 0x00, sizeof(gs_glyph_info_t));
907
5.25M
            code = font->procs.glyph_info(font, glyph, pmat,
908
5.25M
                                          (GLYPH_INFO_WIDTH0 << wmode),
909
5.25M
                                          &glyph_info);
910
5.25M
            if (code < 0) {
911
337k
                ecode = code;
912
337k
                continue;
913
337k
            }
914
4.92M
            if (notdef == GS_NO_GLYPH && gs_font_glyph_is_notdef(bfont, glyph)) {
915
31.7k
                notdef = glyph;
916
31.7k
                info->MissingWidth = (int)glyph_info.width[wmode].x;
917
31.7k
                info->members |= FONT_INFO_MISSING_WIDTH;
918
31.7k
            }
919
4.92M
            if (glyph_info.width[wmode].y != 0)
920
0
                fixed_width = min_int;
921
4.92M
            else if (fixed_width == 0)
922
2.95M
                fixed_width = (int)glyph_info.width[wmode].x;
923
1.96M
            else if (glyph_info.width[wmode].x != fixed_width)
924
68.3k
                fixed_width = min_int;
925
4.92M
            has_glyphs = true;
926
4.92M
        }
927
84.8k
        if (ecode < 0 && !has_glyphs)
928
2
            return ecode;
929
84.8k
        if (fixed_width > 0) {
930
2.18k
            info->Flags |= FONT_IS_FIXED_WIDTH;
931
2.18k
            info->members |= FONT_INFO_AVG_WIDTH | FONT_INFO_MAX_WIDTH |
932
2.18k
                FONT_INFO_MISSING_WIDTH;
933
2.18k
            info->AvgWidth = info->MaxWidth = info->MissingWidth = fixed_width;
934
2.18k
        }
935
84.8k
        info->Flags_returned |= FONT_IS_FIXED_WIDTH;
936
84.8k
    } else if (members & FONT_INFO_MISSING_WIDTH) {
937
0
        gs_glyph glyph;
938
0
        int index;
939
940
0
        for (index = 0;
941
0
             font->procs.enumerate_glyph(font, &index, GLYPH_SPACE_NAME, &glyph) >= 0 &&
942
0
                 index != 0;
943
0
             ) {
944
            /*
945
             * If this is a CIDFont or TrueType font that uses integers as
946
             * glyph names, check for glyph 0; otherwise, check for .notdef.
947
             */
948
0
            if (!gs_font_glyph_is_notdef(bfont, glyph))
949
0
                continue;
950
0
            {
951
0
                gs_glyph_info_t glyph_info;
952
0
                int code = font->procs.glyph_info(font, glyph, pmat,
953
0
                                                  (GLYPH_INFO_WIDTH0 << wmode),
954
0
                                                  &glyph_info);
955
956
0
                if (code < 0)
957
0
                    return code;
958
0
                info->MissingWidth = (int)glyph_info.width[wmode].x;
959
0
                info->members |= FONT_INFO_MISSING_WIDTH;
960
0
                break;
961
0
            }
962
0
        }
963
0
    }
964
143k
    return 0;
965
143k
}
966
967
/* Default font similarity testing procedure */
968
int
969
gs_default_same_font(const gs_font *font, const gs_font *ofont, int mask)
970
240
{
971
360
    while (font->base != font)
972
120
        font = font->base;
973
360
    while (ofont->base != ofont)
974
120
        ofont = ofont->base;
975
240
    if (ofont == font)
976
240
        return mask;
977
    /* In general, we can't determine similarity. */
978
0
    return 0;
979
240
}
980
int
981
gs_base_same_font(const gs_font *font, const gs_font *ofont, int mask)
982
0
{
983
0
    int same = gs_default_same_font(font, ofont, mask);
984
985
0
    if (!same) {
986
0
        const gs_font_base *const bfont = (const gs_font_base *)font;
987
0
        const gs_font_base *const obfont = (const gs_font_base *)ofont;
988
989
0
        if (mask & FONT_SAME_ENCODING) {
990
0
            if (bfont->encoding_index != ENCODING_INDEX_UNKNOWN ||
991
0
                obfont->encoding_index != ENCODING_INDEX_UNKNOWN
992
0
                ) {
993
0
                if (bfont->encoding_index == obfont->encoding_index)
994
0
                    same |= FONT_SAME_ENCODING;
995
0
            }
996
0
        }
997
0
    }
998
0
    return same;
999
0
}
1000
1001
/* ------ Glyph-level procedures ------ */
1002
1003
/*
1004
 * Test whether a glyph is the notdef glyph for a base font.
1005
 * The test is somewhat adhoc: perhaps this should be a virtual procedure.
1006
 */
1007
bool
1008
gs_font_glyph_is_notdef(gs_font_base *bfont, gs_glyph glyph)
1009
15.8M
{
1010
15.8M
    gs_const_string gnstr;
1011
1012
15.8M
    if (glyph == GS_NO_GLYPH)
1013
0
        return false;
1014
15.8M
    if (glyph >= GS_MIN_CID_GLYPH)
1015
310k
        return (glyph == GS_MIN_CID_GLYPH);
1016
15.5M
    return (bfont->procs.glyph_name((gs_font *)bfont, glyph, &gnstr) >= 0 &&
1017
15.5M
            gnstr.size == 7 && !memcmp(gnstr.data, ".notdef", 7));
1018
15.8M
}
1019
1020
/* Dummy character encoding procedure */
1021
gs_glyph
1022
gs_no_encode_char(gs_font *pfont, gs_char chr, gs_glyph_space_t glyph_space)
1023
0
{
1024
0
    return GS_NO_GLYPH;
1025
0
}
1026
1027
/* Dummy glyph decoding procedure */
1028
int
1029
gs_no_decode_glyph(gs_font *pfont, gs_glyph glyph, int ch, ushort *unicode_return, unsigned int length)
1030
0
{
1031
0
    return (int)GS_NO_CHAR;
1032
0
}
1033
1034
/* Dummy glyph enumeration procedure */
1035
int
1036
gs_no_enumerate_glyph(gs_font *font, int *pindex, gs_glyph_space_t glyph_space,
1037
                      gs_glyph *pglyph)
1038
0
{
1039
0
    return_error(gs_error_undefined);
1040
0
}
1041
1042
/* Default glyph info procedure */
1043
int
1044
gs_default_glyph_info(gs_font *font, gs_glyph glyph, const gs_matrix *pmat,
1045
                      int members, gs_glyph_info_t *info)
1046
2.05M
{   /* WMode may be inherited from an upper font. */
1047
2.05M
    gx_path path;
1048
2.05M
    int returned = 0;
1049
2.05M
    int code;
1050
2.05M
    int wmode = ((members & GLYPH_INFO_WIDTH1) != 0);
1051
2.05M
    double sbw[4] = {0, 0, 0, 0};
1052
    /* Currently glyph_outline retrieves sbw only with type 1,2,9 fonts. */
1053
2.05M
    bool charstrings_font = (font->FontType == ft_encrypted ||
1054
790k
                             font->FontType == ft_encrypted2 ||
1055
507k
                             font->FontType == ft_CID_encrypted);
1056
1057
2.05M
    gx_path_init_bbox_accumulator(&path);
1058
2.05M
    code = gx_path_add_point(&path, fixed_0, fixed_0);
1059
2.05M
    if (code < 0)
1060
0
        goto out;
1061
2.05M
    code = font->procs.glyph_outline(font, wmode, glyph, pmat, &path, sbw);
1062
2.05M
    if (code < 0)
1063
432k
        goto out;
1064
1.62M
    if (members & GLYPH_INFO_WIDTHS) {
1065
909k
        int wmode = font->WMode;
1066
909k
        int wmask = GLYPH_INFO_WIDTH0 << wmode;
1067
1068
909k
        if (members & wmask) {
1069
907k
            gs_fixed_point pt;
1070
1071
907k
            code = gx_path_current_point(&path, &pt);
1072
907k
            if (code < 0)
1073
0
                goto out;
1074
907k
            info->width[wmode].x = fixed2float(pt.x);
1075
907k
            info->width[wmode].y = fixed2float(pt.y);
1076
907k
            returned |= wmask;
1077
907k
        }
1078
909k
    }
1079
1.62M
    if (members & GLYPH_INFO_BBOX) {
1080
712k
        gs_fixed_rect bbox;
1081
1082
712k
        code = gx_path_bbox(&path, &bbox);
1083
712k
        if (code < 0)
1084
0
            goto out;
1085
712k
        info->bbox.p.x = fixed2float(bbox.p.x);
1086
712k
        info->bbox.p.y = fixed2float(bbox.p.y);
1087
712k
        info->bbox.q.x = fixed2float(bbox.q.x);
1088
712k
        info->bbox.q.y = fixed2float(bbox.q.y);
1089
712k
        returned |= GLYPH_INFO_BBOX;
1090
712k
    }
1091
1.62M
    if (members & (GLYPH_INFO_WIDTH0 << wmode) && charstrings_font) {
1092
909k
        if (pmat == 0) {
1093
11.2k
            info->width[wmode].x = sbw[2];
1094
11.2k
            info->width[wmode].y = sbw[3];
1095
897k
        } else {
1096
897k
            code = gs_distance_transform(sbw[2], sbw[3], pmat, &info->width[wmode]);
1097
897k
            if (code < 0)
1098
0
                return code;
1099
897k
        }
1100
909k
        returned |= GLYPH_INFO_WIDTH0 << wmode;
1101
909k
    }
1102
1.62M
    if (members & (GLYPH_INFO_VVECTOR0 << wmode) && charstrings_font) {
1103
10.9k
        if (pmat == 0) {
1104
10.9k
            info->v.x = sbw[0];
1105
10.9k
            info->v.y = sbw[1];
1106
10.9k
        } else {
1107
0
            code = gs_distance_transform(sbw[0], sbw[1], pmat, &info->v);
1108
0
            if (code < 0)
1109
0
                return code;
1110
0
        }
1111
10.9k
        returned |= GLYPH_INFO_VVECTOR0 << wmode;
1112
10.9k
    }
1113
1.62M
    if (members & GLYPH_INFO_NUM_PIECES) {
1114
901k
        info->num_pieces = 0;
1115
901k
        returned |= GLYPH_INFO_NUM_PIECES;
1116
901k
    }
1117
1.62M
    returned |= members & GLYPH_INFO_PIECES; /* no pieces stored */
1118
2.05M
 out:
1119
2.05M
    info->members = returned;
1120
2.05M
    return code;
1121
1.62M
}
1122
1123
/* Dummy glyph outline procedure */
1124
int
1125
gs_no_glyph_outline(gs_font *font, int WMode, gs_glyph glyph, const gs_matrix *pmat,
1126
                    gx_path *ppath, double sbw[4])
1127
419k
{
1128
419k
    return_error(gs_error_undefined);
1129
419k
}
1130
1131
/* Dummy glyph name procedure */
1132
int
1133
gs_no_glyph_name(gs_font *font, gs_glyph glyph, gs_const_string *pstr)
1134
0
{
1135
0
    return_error(gs_error_undefined);
1136
0
}
1137
1138
#ifdef DEBUG
1139
/* Reserve a text enumerator instance id. */
1140
ulong gs_next_text_enum_id(const gs_font *font)
1141
{
1142
    return ++font->dir->text_enum_id;
1143
}
1144
#endif