/* Copyright (C) 1996, 1997, 1998, 1999 Aladdin Enterprises. All rights reserved. This file is part of Aladdin Ghostscript. Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND. No author or distributor accepts any responsibility for the consequences of using it, or for whether it serves any particular purpose or works at all, unless he or she says so in writing. Refer to the Aladdin Ghostscript Free Public License (the "License") for full details. Every copy of Aladdin Ghostscript must include a copy of the License, normally in a plain ASCII text file named PUBLIC. The License grants you the right to copy, modify and redistribute Aladdin Ghostscript, but only under certain conditions described in the License. Among other things, the License requires that the copyright notice and this notice be preserved on all copies. */ /* Type 42 (TrueType) font library routines */ #include "memory_.h" #include "gx.h" #include "gserrors.h" #include "gsstruct.h" #include "gsccode.h" #include "gsmatrix.h" #include "gxfixed.h" /* for gxpath.h */ #include "gxpath.h" #include "gxfont.h" #include "gxfont42.h" #include "gxistate.h" /* Structure descriptor */ public_st_gs_font_type42(); /* Forward references */ private int append_outline(P4(uint glyph_index, const gs_matrix_fixed * pmat, gx_path * ppath, gs_font_type42 * pfont)); /* Set up a pointer to a substring of the font data. */ /* Free variables: pfont, string_proc. */ #define ACCESS(base, length, vptr)\ BEGIN\ code = (*string_proc)(pfont, (ulong)(base), length, &vptr);\ if ( code < 0 ) return code;\ END /* Get 2- or 4-byte quantities from a table. */ #define U8(p) ((uint)((p)[0])) #define S8(p) (int)((U8(p) ^ 0x80) - 0x80) #define U16(p) (((uint)((p)[0]) << 8) + (p)[1]) #define S16(p) (int)((U16(p) ^ 0x8000) - 0x8000) private ulong u32(const byte *p) { return ((ulong)U16(p) << 16) + U16((p) + 2); } /* Define the bits in the component glyph flags. */ #define cg_argsAreWords 1 #define cg_argsAreXYValues 2 #define cg_haveScale 8 #define cg_moreComponents 32 #define cg_haveXYScale 64 #define cg_have2x2 128 #define cg_useMyMetrics 512 /* * Parse the definition of one component of a composite glyph. We don't * bother to parse the component index, since the caller can do this so * easily. */ private void parse_component(const byte **pdata, uint *pflags, gs_matrix_fixed *psmat, const gs_font_type42 *pfont, const gs_matrix_fixed *pmat) { const byte *glyph = *pdata; uint flags; double factor = 1.0 / pfont->data.unitsPerEm; gs_matrix_fixed mat; gs_matrix scale_mat; flags = U16(glyph); glyph += 4; mat = *pmat; if (flags & cg_argsAreXYValues) { int arg1, arg2; gs_fixed_point pt; if (flags & cg_argsAreWords) arg1 = S16(glyph), arg2 = S16(glyph + 2), glyph += 4; else arg1 = S8(glyph), arg2 = S8(glyph + 1), glyph += 2; gs_point_transform2fixed(pmat, arg1 * factor, arg2 * factor, &pt); /****** HACK: WE KNOW ABOUT FIXED MATRICES ******/ mat.tx = fixed2float(mat.tx_fixed = pt.x); mat.ty = fixed2float(mat.ty_fixed = pt.y); } else { /****** WE DON'T HANDLE POINT MATCHING YET ******/ glyph += (flags & cg_argsAreWords ? 4 : 2); } #define S2_14(p) (S16(p) / 16384.0) if (flags & cg_haveScale) { scale_mat.xx = scale_mat.yy = S2_14(glyph); scale_mat.xy = scale_mat.yx = 0; glyph += 2; } else if (flags & cg_haveXYScale) { scale_mat.xx = S2_14(glyph); scale_mat.yy = S2_14(glyph + 2); scale_mat.xy = scale_mat.yx = 0; glyph += 4; } else if (flags & cg_have2x2) { scale_mat.xx = S2_14(glyph); scale_mat.xy = S2_14(glyph + 2); scale_mat.yx = S2_14(glyph + 4); scale_mat.yy = S2_14(glyph + 6); glyph += 8; } else goto no_scale; #undef S2_14 scale_mat.tx = 0; scale_mat.ty = 0; /* The scale doesn't affect mat.t{x,y}, so we don't */ /* need to update the fixed components. */ gs_matrix_multiply(&scale_mat, (const gs_matrix *)&mat, (gs_matrix *)&mat); no_scale: *pdata = glyph; *pflags = flags; *psmat = mat; } /* Define the default implementation for getting the outline data for */ /* a glyph, using indexToLocFormat and the loca and glyf tables. */ /* Set pglyph->data = 0 if the glyph is empty. */ private int default_get_outline(gs_font_type42 * pfont, uint glyph_index, gs_const_string * pglyph) { int (*string_proc) (P4(gs_font_type42 *, ulong, uint, const byte **)) = pfont->data.string_proc; const byte *ploca; ulong glyph_start; uint glyph_length; int code; /* * We can't assume that consecutive loca entries are stored * contiguously in memory: we have to access each entry * individually. */ if (pfont->data.indexToLocFormat) { ACCESS(pfont->data.loca + glyph_index * 4, 4, ploca); glyph_start = u32(ploca); ACCESS(pfont->data.loca + glyph_index * 4 + 4, 4, ploca); glyph_length = u32(ploca) - glyph_start; } else { ACCESS(pfont->data.loca + glyph_index * 2, 2, ploca); glyph_start = (ulong) U16(ploca) << 1; ACCESS(pfont->data.loca + glyph_index * 2 + 2, 2, ploca); glyph_length = ((ulong) U16(ploca) << 1) - glyph_start; } pglyph->size = glyph_length; if (glyph_length == 0) pglyph->data = 0; else ACCESS(pfont->data.glyf + glyph_start, glyph_length, pglyph->data); return 0; } /* Parse a glyph into pieces, if any. */ private int parse_pieces(gs_font_type42 *pfont, gs_glyph glyph, gs_glyph *pieces, int *pnum_pieces) { uint glyph_index = glyph - gs_min_cid_glyph; gs_const_string glyph_string; int code = pfont->data.get_outline(pfont, glyph_index, &glyph_string); if (code < 0) return code; if (glyph_string.size != 0 && S16(glyph_string.data) == -1) { /* This is a composite glyph. */ int i = 0; uint flags = cg_moreComponents; const byte *glyph = glyph_string.data + 10; gs_matrix_fixed mat; memset(&mat, 0, sizeof(mat)); /* arbitrary */ for (i = 0; flags & cg_moreComponents; ++i) { if (pieces) pieces[i] = U16(glyph + 2) + gs_min_cid_glyph; parse_component(&glyph, &flags, &mat, pfont, &mat); } *pnum_pieces = i; } else *pnum_pieces = 0; return 0; } /* Define the font procedures for a Type 42 font. */ int gs_type42_glyph_outline(gs_font *font, gs_glyph glyph, const gs_matrix *pmat, gx_path *ppath) { gs_font_type42 *const pfont = (gs_font_type42 *)font; uint glyph_index = glyph - gs_min_cid_glyph; gs_fixed_point origin; int code; gs_glyph_info_t info; gs_matrix_fixed fmat; static const gs_matrix imat = { identity_matrix_body }; if (pmat == 0) pmat = &imat; if ((code = gs_matrix_fixed_from_matrix(&fmat, pmat)) < 0 || (code = gx_path_current_point(ppath, &origin)) < 0 || (code = append_outline(glyph_index, &fmat, ppath, pfont)) < 0 || (code = font->procs.glyph_info(font, glyph, pmat, GLYPH_INFO_WIDTH, &info)) < 0 ) return code; if (pmat) { if ((code = gs_point_transform(info.width[0].x, info.width[0].y, pmat, &info.width[0])) < 0) return code; } return gx_path_add_point(ppath, origin.x + float2fixed(info.width[0].x), origin.y + float2fixed(info.width[0].y)); } int gs_type42_glyph_info(gs_font *font, gs_glyph glyph, const gs_matrix *pmat, int members, gs_glyph_info_t *info) { gs_font_type42 *const pfont = (gs_font_type42 *)font; uint glyph_index = glyph - gs_min_cid_glyph; int default_members = members & ~(GLYPH_INFO_WIDTHS | GLYPH_INFO_NUM_PIECES | GLYPH_INFO_PIECES); gs_const_string outline; int code = 0; if (default_members) { code = gs_default_glyph_info(font, glyph, pmat, default_members, info); if (code < 0) return code; } else if ((code = pfont->data.get_outline(pfont, glyph_index, &outline)) < 0) return code; /* non-existent glyph */ else info->members = 0; if (members & GLYPH_INFO_WIDTH) { float sbw[4]; code = gs_type42_get_metrics(pfont, glyph_index, sbw); if (code < 0) return code; if (pmat) code = gs_point_transform(sbw[2], sbw[3], pmat, &info->width[0]); else info->width[0].x = sbw[2], info->width[0].y = sbw[3]; info->members |= GLYPH_INFO_WIDTH; } if (members & (GLYPH_INFO_NUM_PIECES | GLYPH_INFO_PIECES)) { gs_glyph *pieces = (members & GLYPH_INFO_PIECES ? info->pieces : (gs_glyph *)0); int code = parse_pieces(pfont, glyph, pieces, &info->num_pieces); if (code < 0) return code; info->members |= members & (GLYPH_INFO_NUM_PIECES | GLYPH_INFO_PIECES); } return code; } int gs_type42_enumerate_glyph(gs_font *font, int *pindex, gs_glyph_space_t glyph_space, gs_glyph *pglyph) { gs_font_type42 *const pfont = (gs_font_type42 *)font; while (++*pindex <= pfont->data.numGlyphs) { gs_const_string outline; uint glyph_index = *pindex - 1; int code = pfont->data.get_outline(pfont, glyph_index, &outline); if (code < 0) return code; if (outline.data == 0) continue; /* empty (undefined) glyph */ *pglyph = glyph_index + gs_min_cid_glyph; return 0; } /* We are done. */ *pindex = 0; return 0; } /* Initialize the cached values in a Type 42 font. */ /* Note that this initializes get_outline and the font procedures as well. */ int gs_type42_font_init(gs_font_type42 * pfont) { int (*string_proc)(P4(gs_font_type42 *, ulong, uint, const byte **)) = pfont->data.string_proc; const byte *OffsetTable; uint numTables; const byte *TableDirectory; uint i; int code; byte head_box[8]; ulong loca_size = 0; ACCESS(0, 12, OffsetTable); { static const byte version1_0[4] = {0, 1, 0, 0}; static const byte version_true[4] = {'t', 'r', 'u', 'e'}; if (memcmp(OffsetTable, version1_0, 4) && memcmp(OffsetTable, version_true, 4)) return_error(gs_error_invalidfont); } numTables = U16(OffsetTable + 4); ACCESS(12, numTables * 16, TableDirectory); /* Clear optional entries. */ pfont->data.numLongMetrics = 0; for (i = 0; i < numTables; ++i) { const byte *tab = TableDirectory + i * 16; ulong offset = u32(tab + 8); if (!memcmp(tab, "glyf", 4)) pfont->data.glyf = offset; else if (!memcmp(tab, "head", 4)) { const byte *head; ACCESS(offset, 54, head); pfont->data.unitsPerEm = U16(head + 18); memcpy(head_box, head + 36, 8); pfont->data.indexToLocFormat = U16(head + 50); } else if (!memcmp(tab, "hhea", 4)) { const byte *hhea; ACCESS(offset, 36, hhea); pfont->data.numLongMetrics = U16(hhea + 34); } else if (!memcmp(tab, "hmtx", 4)) { pfont->data.hmtx = offset; pfont->data.hmtx_length = (uint)u32(tab + 12); } else if (!memcmp(tab, "loca", 4)) { pfont->data.loca = offset; loca_size = u32(tab + 12); } } loca_size >>= pfont->data.indexToLocFormat + 1; pfont->data.numGlyphs = (loca_size == 0 ? 0 : loca_size - 1); /* * If the font doesn't have a valid FontBBox, compute one from the * 'head' information. Since the Adobe PostScript driver sometimes * outputs garbage FontBBox values, we use a "reasonableness" check * here. */ if (pfont->FontBBox.p.x >= pfont->FontBBox.q.x || pfont->FontBBox.p.y >= pfont->FontBBox.q.y || pfont->FontBBox.p.x < -0.5 || pfont->FontBBox.p.x > 0.5 || pfont->FontBBox.p.y < -0.5 || pfont->FontBBox.p.y > 0.5 ) { float upem = pfont->data.unitsPerEm; pfont->FontBBox.p.x = S16(head_box) / upem; pfont->FontBBox.p.y = S16(head_box + 2) / upem; pfont->FontBBox.q.x = S16(head_box + 4) / upem; pfont->FontBBox.q.y = S16(head_box + 6) / upem; } pfont->data.get_outline = default_get_outline; pfont->procs.glyph_outline = gs_type42_glyph_outline; pfont->procs.glyph_info = gs_type42_glyph_info; pfont->procs.enumerate_glyph = gs_type42_enumerate_glyph; return 0; } /* Get the metrics of a simple glyph. */ private int simple_glyph_metrics(gs_font_type42 * pfont, uint glyph_index, float sbw[4]) { int (*string_proc)(P4(gs_font_type42 *, ulong, uint, const byte **)) = pfont->data.string_proc; double factor = 1.0 / pfont->data.unitsPerEm; uint widthx; int lsbx; int code; { uint num_metrics = pfont->data.numLongMetrics; const byte *hmetrics; if (glyph_index < num_metrics) { ACCESS(pfont->data.hmtx + glyph_index * 4, 4, hmetrics); widthx = U16(hmetrics); lsbx = S16(hmetrics + 2); } else { uint offset = pfont->data.hmtx + num_metrics * 4; uint glyph_offset = (glyph_index - num_metrics) * 2; const byte *lsb; ACCESS(offset - 4, 4, hmetrics); widthx = U16(hmetrics); if (glyph_offset >= pfont->data.hmtx_length) glyph_offset = pfont->data.hmtx_length - 2; ACCESS(offset + glyph_offset, 2, lsb); lsbx = S16(lsb); } } sbw[0] = lsbx * factor; sbw[1] = 0; sbw[2] = widthx * factor; sbw[3] = 0; return 0; } /* Get the metrics of a glyph. */ int gs_type42_get_metrics(gs_font_type42 * pfont, uint glyph_index, float sbw[4]) { gs_const_string glyph_string; int code = pfont->data.get_outline(pfont, glyph_index, &glyph_string); if (code < 0) return code; if (glyph_string.size != 0 && S16(glyph_string.data) == -1) { /* This is a composite glyph. */ uint flags; const byte *glyph = glyph_string.data + 10; gs_matrix_fixed mat; memset(&mat, 0, sizeof(mat)); /* arbitrary */ do { uint comp_index = U16(glyph + 2); parse_component(&glyph, &flags, &mat, pfont, &mat); if (flags & cg_useMyMetrics) { return simple_glyph_metrics(pfont, comp_index, sbw); } } while (flags & cg_moreComponents); } return simple_glyph_metrics(pfont, glyph_index, sbw); } /* Define the bits in the glyph flags. */ #define gf_OnCurve 1 #define gf_xShort 2 #define gf_yShort 4 #define gf_Repeat 8 #define gf_xPos 16 /* xShort */ #define gf_xSame 16 /* !xShort */ #define gf_yPos 32 /* yShort */ #define gf_ySame 32 /* !yShort */ /* Append a TrueType outline to a path. */ /* Note that this does not append the final moveto for the width. */ int gs_type42_append(uint glyph_index, gs_imager_state * pis, gx_path * ppath, const gs_log2_scale_point * pscale, bool charpath_flag, int paint_type, gs_font_type42 * pfont) { return append_outline(glyph_index, &pis->ctm, ppath, pfont); } /* Append a simple glyph outline. */ private int append_simple(const byte *glyph, float sbw[4], const gs_matrix_fixed *pmat, gx_path *ppath, gs_font_type42 * pfont) { int numContours = S16(glyph); const byte *pends = glyph + 10; const byte *pinstr = pends + numContours * 2; const byte *pflags; uint npoints; const byte *pxc, *pyc; int code; if (numContours == 0) return 0; /* * It appears that the only way to find the X and Y coordinate * tables is to parse the flags. If this is true, it is an * incredible piece of bad design. */ { const byte *pf = pflags = pinstr + 2 + U16(pinstr); uint xbytes = npoints = U16(pinstr - 2) + 1; uint np = npoints; while (np > 0) { byte flags = *pf++; uint reps = (flags & gf_Repeat ? *pf++ + 1 : 1); if (!(flags & gf_xShort)) { if (flags & gf_xSame) xbytes -= reps; else xbytes += reps; } np -= reps; } pxc = pf; pyc = pxc + xbytes; } /* Interpret the contours. */ { uint i, np; gs_fixed_point pt; double factor = 1.0 / pfont->data.unitsPerEm; /* * Decode the first flag byte outside the loop, to avoid a * compiler warning about uninitialized variables. */ byte flags = *pflags++; uint reps = (flags & gf_Repeat ? *pflags++ + 1 : 1); /* * The TrueType documentation gives no clue as to how the lsb * should affect placement of the outline. Our best guess is * that the outline should be translated by lsb - xMin. */ gs_point_transform2fixed(pmat, sbw[0] - S16(glyph + 2) * factor, 0.0, &pt); for (i = 0, np = 0; i < numContours; ++i) { bool move = true; uint last_point = U16(pends + i * 2); float dx, dy; int off_curve = 0; gs_fixed_point start; gs_fixed_point cpoints[3]; for (; np <= last_point; --reps, ++np) { gs_fixed_point dpt; if (reps == 0) { flags = *pflags++; reps = (flags & gf_Repeat ? *pflags++ + 1 : 1); } if (flags & gf_xShort) { /* * A bug in the Watcom compiler prevents us from doing * the following with the obvious conditional expression. */ if (flags & gf_xPos) dx = *pxc++ * factor; else dx = -(int)*pxc++ * factor; } else if (!(flags & gf_xSame)) dx = S16(pxc) * factor, pxc += 2; else dx = 0; if (flags & gf_yShort) { /* See above under dx. */ if (flags & gf_yPos) dy = *pyc++ * factor; else dy = -(int)*pyc++ * factor; } else if (!(flags & gf_ySame)) dy = S16(pyc) * factor, pyc += 2; else dy = 0; code = gs_distance_transform2fixed(pmat, dx, dy, &dpt); if (code < 0) return code; pt.x += dpt.x, pt.y += dpt.y; #define control1(xy) cpoints[1].xy #define control2(xy) cpoints[2].xy #define control3off(xy) ((cpoints[1].xy + pt.xy) / 2) #define control4off(xy) ((cpoints[0].xy + 2 * cpoints[1].xy) / 3) #define control5off(xy) ((2 * cpoints[1].xy + cpoints[2].xy) / 3) #define control6off(xy) ((2 * cpoints[1].xy + pt.xy) / 3) #define control7off(xy) ((2 * cpoints[1].xy + start.xy) / 3) if (move) { if_debug2('1', "[1t]start (%g,%g)\n", fixed2float(pt.x), fixed2float(pt.y)); start = pt; code = gx_path_add_point(ppath, pt.x, pt.y); cpoints[0] = pt; move = false; } else if (flags & gf_OnCurve) { if_debug2('1', "[1t]ON (%g,%g)\n", fixed2float(pt.x), fixed2float(pt.y)); if (off_curve) code = gx_path_add_curve(ppath, control4off(x), control4off(y), control6off(x), control6off(y), pt.x, pt.y); else code = gx_path_add_line(ppath, pt.x, pt.y); cpoints[0] = pt; off_curve = 0; } else { if_debug2('1', "[1t]...off (%g,%g)\n", fixed2float(pt.x), fixed2float(pt.y)); switch (off_curve++) { default: /* >= 1 */ control2(x) = control3off(x); control2(y) = control3off(y); code = gx_path_add_curve(ppath, control4off(x), control4off(y), control5off(x), control5off(y), control2(x), control2(y)); cpoints[0] = cpoints[2]; off_curve = 1; /* falls through */ case 0: cpoints[1] = pt; } } if (code < 0) return code; } if (off_curve) code = gx_path_add_curve(ppath, control4off(x), control4off(y), control7off(x), control7off(y), start.x, start.y); code = gx_path_close_subpath(ppath); if (code < 0) return code; } } return 0; } /* Append a glyph outline. */ private int append_outline(uint glyph_index, const gs_matrix_fixed * pmat, gx_path * ppath, gs_font_type42 * pfont) { gs_const_string glyph_string; const byte *glyph; float sbw[4]; int numContours; int code; code = (*pfont->data.get_outline) (pfont, glyph_index, &glyph_string); if (code < 0) return code; glyph = glyph_string.data; if (glyph == 0 || glyph_string.size == 0) /* empty glyph */ return 0; numContours = S16(glyph); if (numContours >= 0) { simple_glyph_metrics(pfont, glyph_index, sbw); return append_simple(glyph, sbw, pmat, ppath, pfont); } if (numContours != -1) return_error(gs_error_rangecheck); /* This is a composite glyph. */ { uint flags; glyph += 10; do { uint comp_index = U16(glyph + 2); gs_matrix_fixed mat; parse_component(&glyph, &flags, &mat, pfont, pmat); code = append_outline(comp_index, &mat, ppath, pfont); if (code < 0) return code; } while (flags & cg_moreComponents); } return 0; }