/* * Copyright © 2016 Igalia S.L. * * This is part of HarfBuzz, a text shaping library. * * Permission is hereby granted, without written agreement and without * license or royalty fees, to use, copy, modify, and distribute this * software and its documentation for any purpose, provided that the * above copyright notice and the following two paragraphs appear in * all copies of this software. * * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH * DAMAGE. * * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING, * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND * FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS. * * Igalia Author(s): Frédéric Wang */ #ifndef HB_OT_MATH_TABLE_HH #define HB_OT_MATH_TABLE_HH #include "hb-open-type.hh" #include "hb-ot-layout-common.hh" #include "hb-ot-math.h" namespace OT { struct MathValueRecord { hb_position_t get_x_value (hb_font_t *font, const void *base) const { return font->em_scale_x (value) + (base+deviceTable).get_x_delta (font); } hb_position_t get_y_value (hb_font_t *font, const void *base) const { return font->em_scale_y (value) + (base+deviceTable).get_y_delta (font); } MathValueRecord* copy (hb_serialize_context_t *c, const void *base) const { TRACE_SERIALIZE (this); auto *out = c->embed (this); if (unlikely (!out)) return_trace (nullptr); out->deviceTable.serialize_copy (c, deviceTable, base, 0, hb_serialize_context_t::Head); return_trace (out); } bool sanitize (hb_sanitize_context_t *c, const void *base) const { TRACE_SANITIZE (this); return_trace (c->check_struct (this) && deviceTable.sanitize (c, base)); } protected: HBINT16 value; /* The X or Y value in design units */ Offset16To<Device> deviceTable; /* Offset to the device table - from the * beginning of parent table. May be NULL. * Suggested format for device table is 1. */ public: DEFINE_SIZE_STATIC (4); }; struct MathConstants { MathConstants* copy (hb_serialize_context_t *c) const { TRACE_SERIALIZE (this); auto *out = c->start_embed (this); if (unlikely (!out)) return_trace (nullptr); HBINT16 *p = c->allocate_size<HBINT16> (HBINT16::static_size * 2); if (unlikely (!p)) return_trace (nullptr); memcpy (p, percentScaleDown, HBINT16::static_size * 2); HBUINT16 *m = c->allocate_size<HBUINT16> (HBUINT16::static_size * 2); if (unlikely (!m)) return_trace (nullptr); memcpy (m, minHeight, HBUINT16::static_size * 2); unsigned count = ARRAY_LENGTH (mathValueRecords); for (unsigned i = 0; i < count; i++) if (!c->copy (mathValueRecords[i], this)) return_trace (nullptr); if (!c->embed (radicalDegreeBottomRaisePercent)) return_trace (nullptr); return_trace (out); } bool sanitize_math_value_records (hb_sanitize_context_t *c) const { TRACE_SANITIZE (this); unsigned int count = ARRAY_LENGTH (mathValueRecords); for (unsigned int i = 0; i < count; i++) if (!mathValueRecords[i].sanitize (c, this)) return_trace (false); return_trace (true); } bool sanitize (hb_sanitize_context_t *c) const { TRACE_SANITIZE (this); return_trace (c->check_struct (this) && sanitize_math_value_records (c)); } hb_position_t get_value (hb_ot_math_constant_t constant, hb_font_t *font) const { switch (constant) { case HB_OT_MATH_CONSTANT_SCRIPT_PERCENT_SCALE_DOWN: case HB_OT_MATH_CONSTANT_SCRIPT_SCRIPT_PERCENT_SCALE_DOWN: return percentScaleDown[constant - HB_OT_MATH_CONSTANT_SCRIPT_PERCENT_SCALE_DOWN]; case HB_OT_MATH_CONSTANT_DELIMITED_SUB_FORMULA_MIN_HEIGHT: case HB_OT_MATH_CONSTANT_DISPLAY_OPERATOR_MIN_HEIGHT: return font->em_scale_y (minHeight[constant - HB_OT_MATH_CONSTANT_DELIMITED_SUB_FORMULA_MIN_HEIGHT]); case HB_OT_MATH_CONSTANT_RADICAL_KERN_AFTER_DEGREE: case HB_OT_MATH_CONSTANT_RADICAL_KERN_BEFORE_DEGREE: case HB_OT_MATH_CONSTANT_SKEWED_FRACTION_HORIZONTAL_GAP: case HB_OT_MATH_CONSTANT_SPACE_AFTER_SCRIPT: return mathValueRecords[constant - HB_OT_MATH_CONSTANT_MATH_LEADING].get_x_value (font, this); case HB_OT_MATH_CONSTANT_ACCENT_BASE_HEIGHT: case HB_OT_MATH_CONSTANT_AXIS_HEIGHT: case HB_OT_MATH_CONSTANT_FLATTENED_ACCENT_BASE_HEIGHT: case HB_OT_MATH_CONSTANT_FRACTION_DENOMINATOR_DISPLAY_STYLE_SHIFT_DOWN: case HB_OT_MATH_CONSTANT_FRACTION_DENOMINATOR_GAP_MIN: case HB_OT_MATH_CONSTANT_FRACTION_DENOMINATOR_SHIFT_DOWN: case HB_OT_MATH_CONSTANT_FRACTION_DENOM_DISPLAY_STYLE_GAP_MIN: case HB_OT_MATH_CONSTANT_FRACTION_NUMERATOR_DISPLAY_STYLE_SHIFT_UP: case HB_OT_MATH_CONSTANT_FRACTION_NUMERATOR_GAP_MIN: case HB_OT_MATH_CONSTANT_FRACTION_NUMERATOR_SHIFT_UP: case HB_OT_MATH_CONSTANT_FRACTION_NUM_DISPLAY_STYLE_GAP_MIN: case HB_OT_MATH_CONSTANT_FRACTION_RULE_THICKNESS: case HB_OT_MATH_CONSTANT_LOWER_LIMIT_BASELINE_DROP_MIN: case HB_OT_MATH_CONSTANT_LOWER_LIMIT_GAP_MIN: case HB_OT_MATH_CONSTANT_MATH_LEADING: case HB_OT_MATH_CONSTANT_OVERBAR_EXTRA_ASCENDER: case HB_OT_MATH_CONSTANT_OVERBAR_RULE_THICKNESS: case HB_OT_MATH_CONSTANT_OVERBAR_VERTICAL_GAP: case HB_OT_MATH_CONSTANT_RADICAL_DISPLAY_STYLE_VERTICAL_GAP: case HB_OT_MATH_CONSTANT_RADICAL_EXTRA_ASCENDER: case HB_OT_MATH_CONSTANT_RADICAL_RULE_THICKNESS: case HB_OT_MATH_CONSTANT_RADICAL_VERTICAL_GAP: case HB_OT_MATH_CONSTANT_SKEWED_FRACTION_VERTICAL_GAP: case HB_OT_MATH_CONSTANT_STACK_BOTTOM_DISPLAY_STYLE_SHIFT_DOWN: case HB_OT_MATH_CONSTANT_STACK_BOTTOM_SHIFT_DOWN: case HB_OT_MATH_CONSTANT_STACK_DISPLAY_STYLE_GAP_MIN: case HB_OT_MATH_CONSTANT_STACK_GAP_MIN: case HB_OT_MATH_CONSTANT_STACK_TOP_DISPLAY_STYLE_SHIFT_UP: case HB_OT_MATH_CONSTANT_STACK_TOP_SHIFT_UP: case HB_OT_MATH_CONSTANT_STRETCH_STACK_BOTTOM_SHIFT_DOWN: case HB_OT_MATH_CONSTANT_STRETCH_STACK_GAP_ABOVE_MIN: case HB_OT_MATH_CONSTANT_STRETCH_STACK_GAP_BELOW_MIN: case HB_OT_MATH_CONSTANT_STRETCH_STACK_TOP_SHIFT_UP: case HB_OT_MATH_CONSTANT_SUBSCRIPT_BASELINE_DROP_MIN: case HB_OT_MATH_CONSTANT_SUBSCRIPT_SHIFT_DOWN: case HB_OT_MATH_CONSTANT_SUBSCRIPT_TOP_MAX: case HB_OT_MATH_CONSTANT_SUB_SUPERSCRIPT_GAP_MIN: case HB_OT_MATH_CONSTANT_SUPERSCRIPT_BASELINE_DROP_MAX: case HB_OT_MATH_CONSTANT_SUPERSCRIPT_BOTTOM_MAX_WITH_SUBSCRIPT: case HB_OT_MATH_CONSTANT_SUPERSCRIPT_BOTTOM_MIN: case HB_OT_MATH_CONSTANT_SUPERSCRIPT_SHIFT_UP: case HB_OT_MATH_CONSTANT_SUPERSCRIPT_SHIFT_UP_CRAMPED: case HB_OT_MATH_CONSTANT_UNDERBAR_EXTRA_DESCENDER: case HB_OT_MATH_CONSTANT_UNDERBAR_RULE_THICKNESS: case HB_OT_MATH_CONSTANT_UNDERBAR_VERTICAL_GAP: case HB_OT_MATH_CONSTANT_UPPER_LIMIT_BASELINE_RISE_MIN: case HB_OT_MATH_CONSTANT_UPPER_LIMIT_GAP_MIN: return mathValueRecords[constant - HB_OT_MATH_CONSTANT_MATH_LEADING].get_y_value (font, this); case HB_OT_MATH_CONSTANT_RADICAL_DEGREE_BOTTOM_RAISE_PERCENT: return radicalDegreeBottomRaisePercent; default: return 0; } } protected: HBINT16 percentScaleDown[2]; HBUINT16 minHeight[2]; MathValueRecord mathValueRecords[51]; HBINT16 radicalDegreeBottomRaisePercent; public: DEFINE_SIZE_STATIC (214); }; struct MathItalicsCorrectionInfo { bool subset (hb_subset_context_t *c) const { TRACE_SUBSET (this); const hb_set_t &glyphset = *c->plan->_glyphset_mathed; const hb_map_t &glyph_map = *c->plan->glyph_map; auto *out = c->serializer->start_embed (*this); if (unlikely (!c->serializer->extend_min (out))) return_trace (false); hb_sorted_vector_t<hb_codepoint_t> new_coverage; + hb_zip (this+coverage, italicsCorrection) | hb_filter (glyphset, hb_first) | hb_filter (serialize_math_record_array (c->serializer, out->italicsCorrection, this), hb_second) | hb_map (hb_first) | hb_map (glyph_map) | hb_sink (new_coverage) ; out->coverage.serialize_serialize (c->serializer, new_coverage.iter ()); return_trace (true); } bool sanitize (hb_sanitize_context_t *c) const { TRACE_SANITIZE (this); return_trace (c->check_struct (this) && coverage.sanitize (c, this) && italicsCorrection.sanitize (c, this)); } hb_position_t get_value (hb_codepoint_t glyph, hb_font_t *font) const { unsigned int index = (this+coverage).get_coverage (glyph); return italicsCorrection[index].get_x_value (font, this); } protected: Offset16To<Coverage> coverage; /* Offset to Coverage table - * from the beginning of * MathItalicsCorrectionInfo * table. */ Array16Of<MathValueRecord> italicsCorrection; /* Array of MathValueRecords * defining italics correction * values for each * covered glyph. */ public: DEFINE_SIZE_ARRAY (4, italicsCorrection); }; struct MathTopAccentAttachment { bool subset (hb_subset_context_t *c) const { TRACE_SUBSET (this); const hb_set_t &glyphset = *c->plan->_glyphset_mathed; const hb_map_t &glyph_map = *c->plan->glyph_map; auto *out = c->serializer->start_embed (*this); if (unlikely (!c->serializer->extend_min (out))) return_trace (false); hb_sorted_vector_t<hb_codepoint_t> new_coverage; + hb_zip (this+topAccentCoverage, topAccentAttachment) | hb_filter (glyphset, hb_first) | hb_filter (serialize_math_record_array (c->serializer, out->topAccentAttachment, this), hb_second) | hb_map (hb_first) | hb_map (glyph_map) | hb_sink (new_coverage) ; out->topAccentCoverage.serialize_serialize (c->serializer, new_coverage.iter ()); return_trace (true); } bool sanitize (hb_sanitize_context_t *c) const { TRACE_SANITIZE (this); return_trace (c->check_struct (this) && topAccentCoverage.sanitize (c, this) && topAccentAttachment.sanitize (c, this)); } hb_position_t get_value (hb_codepoint_t glyph, hb_font_t *font) const { unsigned int index = (this+topAccentCoverage).get_coverage (glyph); if (index == NOT_COVERED) return font->get_glyph_h_advance (glyph) / 2; return topAccentAttachment[index].get_x_value (font, this); } protected: Offset16To<Coverage> topAccentCoverage; /* Offset to Coverage table - * from the beginning of * MathTopAccentAttachment * table. */ Array16Of<MathValueRecord> topAccentAttachment; /* Array of MathValueRecords * defining top accent * attachment points for each * covered glyph. */ public: DEFINE_SIZE_ARRAY (2 + 2, topAccentAttachment); }; struct MathKern { MathKern* copy (hb_serialize_context_t *c) const { TRACE_SERIALIZE (this); auto *out = c->start_embed (this); if (unlikely (!out)) return_trace (nullptr); if (unlikely (!c->embed (heightCount))) return_trace (nullptr); unsigned count = 2 * heightCount + 1; for (unsigned i = 0; i < count; i++) if (!c->copy (mathValueRecordsZ.arrayZ[i], this)) return_trace (nullptr); return_trace (out); } bool sanitize_math_value_records (hb_sanitize_context_t *c) const { TRACE_SANITIZE (this); unsigned int count = 2 * heightCount + 1; for (unsigned int i = 0; i < count; i++) if (!mathValueRecordsZ.arrayZ[i].sanitize (c, this)) return_trace (false); return_trace (true); } bool sanitize (hb_sanitize_context_t *c) const { TRACE_SANITIZE (this); return_trace (c->check_struct (this) && c->check_array (mathValueRecordsZ.arrayZ, 2 * heightCount + 1) && sanitize_math_value_records (c)); } hb_position_t get_value (hb_position_t correction_height, hb_font_t *font) const { const MathValueRecord* correctionHeight = mathValueRecordsZ.arrayZ; const MathValueRecord* kernValue = mathValueRecordsZ.arrayZ + heightCount; int sign = font->y_scale < 0 ? -1 : +1; /* The description of the MathKern table is a ambiguous, but interpreting * "between the two heights found at those indexes" for 0 < i < len as * * correctionHeight[i-1] < correction_height <= correctionHeight[i] * * makes the result consistent with the limit cases and we can just use the * binary search algorithm of std::upper_bound: */ unsigned int i = 0; unsigned int count = heightCount; while (count > 0) { unsigned int half = count / 2; hb_position_t height = correctionHeight[i + half].get_y_value (font, this); if (sign * height < sign * correction_height) { i += half + 1; count -= half + 1; } else count = half; } return kernValue[i].get_x_value (font, this); } protected: HBUINT16 heightCount; UnsizedArrayOf<MathValueRecord> mathValueRecordsZ; /* Array of correction heights at * which the kern value changes. * Sorted by the height value in * design units (heightCount entries), * Followed by: * Array of kern values corresponding * to heights. (heightCount+1 entries). */ public: DEFINE_SIZE_ARRAY (2, mathValueRecordsZ); }; struct MathKernInfoRecord { MathKernInfoRecord* copy (hb_serialize_context_t *c, const void *base) const { TRACE_SERIALIZE (this); auto *out = c->embed (this); if (unlikely (!out)) return_trace (nullptr); unsigned count = ARRAY_LENGTH (mathKern); for (unsigned i = 0; i < count; i++) out->mathKern[i].serialize_copy (c, mathKern[i], base, 0, hb_serialize_context_t::Head); return_trace (out); } bool sanitize (hb_sanitize_context_t *c, const void *base) const { TRACE_SANITIZE (this); unsigned int count = ARRAY_LENGTH (mathKern); for (unsigned int i = 0; i < count; i++) if (unlikely (!mathKern[i].sanitize (c, base))) return_trace (false); return_trace (true); } hb_position_t get_kerning (hb_ot_math_kern_t kern, hb_position_t correction_height, hb_font_t *font, const void *base) const { unsigned int idx = kern; if (unlikely (idx >= ARRAY_LENGTH (mathKern))) return 0; return (base+mathKern[idx]).get_value (correction_height, font); } protected: /* Offset to MathKern table for each corner - * from the beginning of MathKernInfo table. May be NULL. */ Offset16To<MathKern> mathKern[4]; public: DEFINE_SIZE_STATIC (8); }; struct MathKernInfo { bool subset (hb_subset_context_t *c) const { TRACE_SUBSET (this); const hb_set_t &glyphset = *c->plan->_glyphset_mathed; const hb_map_t &glyph_map = *c->plan->glyph_map; auto *out = c->serializer->start_embed (*this); if (unlikely (!c->serializer->extend_min (out))) return_trace (false); hb_sorted_vector_t<hb_codepoint_t> new_coverage; + hb_zip (this+mathKernCoverage, mathKernInfoRecords) | hb_filter (glyphset, hb_first) | hb_filter (serialize_math_record_array (c->serializer, out->mathKernInfoRecords, this), hb_second) | hb_map (hb_first) | hb_map (glyph_map) | hb_sink (new_coverage) ; out->mathKernCoverage.serialize_serialize (c->serializer, new_coverage.iter ()); return_trace (true); } bool sanitize (hb_sanitize_context_t *c) const { TRACE_SANITIZE (this); return_trace (c->check_struct (this) && mathKernCoverage.sanitize (c, this) && mathKernInfoRecords.sanitize (c, this)); } hb_position_t get_kerning (hb_codepoint_t glyph, hb_ot_math_kern_t kern, hb_position_t correction_height, hb_font_t *font) const { unsigned int index = (this+mathKernCoverage).get_coverage (glyph); return mathKernInfoRecords[index].get_kerning (kern, correction_height, font, this); } protected: Offset16To<Coverage> mathKernCoverage; /* Offset to Coverage table - * from the beginning of the * MathKernInfo table. */ Array16Of<MathKernInfoRecord> mathKernInfoRecords; /* Array of MathKernInfoRecords, * per-glyph information for * mathematical positioning * of subscripts and * superscripts. */ public: DEFINE_SIZE_ARRAY (4, mathKernInfoRecords); }; struct MathGlyphInfo { bool subset (hb_subset_context_t *c) const { TRACE_SUBSET (this); auto *out = c->serializer->embed (*this); if (unlikely (!out)) return_trace (false); out->mathItalicsCorrectionInfo.serialize_subset (c, mathItalicsCorrectionInfo, this); out->mathTopAccentAttachment.serialize_subset (c, mathTopAccentAttachment, this); const hb_set_t &glyphset = *c->plan->_glyphset_mathed; const hb_map_t &glyph_map = *c->plan->glyph_map; auto it = + hb_iter (this+extendedShapeCoverage) | hb_filter (glyphset) | hb_map_retains_sorting (glyph_map) ; out->extendedShapeCoverage.serialize_serialize (c->serializer, it); out->mathKernInfo.serialize_subset (c, mathKernInfo, this); return_trace (true); } bool sanitize (hb_sanitize_context_t *c) const { TRACE_SANITIZE (this); return_trace (c->check_struct (this) && mathItalicsCorrectionInfo.sanitize (c, this) && mathTopAccentAttachment.sanitize (c, this) && extendedShapeCoverage.sanitize (c, this) && mathKernInfo.sanitize (c, this)); } hb_position_t get_italics_correction (hb_codepoint_t glyph, hb_font_t *font) const { return (this+mathItalicsCorrectionInfo).get_value (glyph, font); } hb_position_t get_top_accent_attachment (hb_codepoint_t glyph, hb_font_t *font) const { return (this+mathTopAccentAttachment).get_value (glyph, font); } bool is_extended_shape (hb_codepoint_t glyph) const { return (this+extendedShapeCoverage).get_coverage (glyph) != NOT_COVERED; } hb_position_t get_kerning (hb_codepoint_t glyph, hb_ot_math_kern_t kern, hb_position_t correction_height, hb_font_t *font) const { return (this+mathKernInfo).get_kerning (glyph, kern, correction_height, font); } protected: /* Offset to MathItalicsCorrectionInfo table - * from the beginning of MathGlyphInfo table. */ Offset16To<MathItalicsCorrectionInfo> mathItalicsCorrectionInfo; /* Offset to MathTopAccentAttachment table - * from the beginning of MathGlyphInfo table. */ Offset16To<MathTopAccentAttachment> mathTopAccentAttachment; /* Offset to coverage table for Extended Shape glyphs - * from the beginning of MathGlyphInfo table. When the left or right glyph of * a box is an extended shape variant, the (ink) box (and not the default * position defined by values in MathConstants table) should be used for * vertical positioning purposes. May be NULL.. */ Offset16To<Coverage> extendedShapeCoverage; /* Offset to MathKernInfo table - * from the beginning of MathGlyphInfo table. */ Offset16To<MathKernInfo> mathKernInfo; public: DEFINE_SIZE_STATIC (8); }; struct MathGlyphVariantRecord { friend struct MathGlyphConstruction; bool subset (hb_subset_context_t *c) const { TRACE_SUBSET (this); auto *out = c->serializer->embed (this); if (unlikely (!out)) return_trace (false); const hb_map_t& glyph_map = *c->plan->glyph_map; return_trace (c->serializer->check_assign (out->variantGlyph, glyph_map.get (variantGlyph), HB_SERIALIZE_ERROR_INT_OVERFLOW)); } bool sanitize (hb_sanitize_context_t *c) const { TRACE_SANITIZE (this); return_trace (c->check_struct (this)); } void closure_glyphs (hb_set_t *variant_glyphs) const { variant_glyphs->add (variantGlyph); } protected: HBGlyphID16 variantGlyph; /* Glyph ID for the variant. */ HBUINT16 advanceMeasurement; /* Advance width/height, in design units, of the * variant, in the direction of requested * glyph extension. */ public: DEFINE_SIZE_STATIC (4); }; struct PartFlags : HBUINT16 { enum Flags { Extender = 0x0001u, /* If set, the part can be skipped or repeated. */ Defined = 0x0001u, /* All defined flags. */ }; public: DEFINE_SIZE_STATIC (2); }; struct MathGlyphPartRecord { bool subset (hb_subset_context_t *c) const { TRACE_SUBSET (this); auto *out = c->serializer->embed (this); if (unlikely (!out)) return_trace (false); const hb_map_t& glyph_map = *c->plan->glyph_map; return_trace (c->serializer->check_assign (out->glyph, glyph_map.get (glyph), HB_SERIALIZE_ERROR_INT_OVERFLOW)); } bool sanitize (hb_sanitize_context_t *c) const { TRACE_SANITIZE (this); return_trace (c->check_struct (this)); } void extract (hb_ot_math_glyph_part_t &out, int64_t mult, hb_font_t *font) const { out.glyph = glyph; out.start_connector_length = font->em_mult (startConnectorLength, mult); out.end_connector_length = font->em_mult (endConnectorLength, mult); out.full_advance = font->em_mult (fullAdvance, mult); static_assert ((unsigned int) HB_OT_MATH_GLYPH_PART_FLAG_EXTENDER == (unsigned int) PartFlags::Extender, ""); out.flags = (hb_ot_math_glyph_part_flags_t) (unsigned int) (partFlags & PartFlags::Defined); } void closure_glyphs (hb_set_t *variant_glyphs) const { variant_glyphs->add (glyph); } protected: HBGlyphID16 glyph; /* Glyph ID for the part. */ HBUINT16 startConnectorLength; /* Advance width/ height of the straight bar * connector material, in design units, is at * the beginning of the glyph, in the * direction of the extension. */ HBUINT16 endConnectorLength; /* Advance width/ height of the straight bar * connector material, in design units, is at * the end of the glyph, in the direction of * the extension. */ HBUINT16 fullAdvance; /* Full advance width/height for this part, * in the direction of the extension. * In design units. */ PartFlags partFlags; /* Part qualifiers. */ public: DEFINE_SIZE_STATIC (10); }; struct MathGlyphAssembly { bool subset (hb_subset_context_t *c) const { TRACE_SUBSET (this); auto *out = c->serializer->start_embed (*this); if (unlikely (!out)) return_trace (false); if (!c->serializer->copy (italicsCorrection, this)) return_trace (false); if (!c->serializer->copy<HBUINT16> (partRecords.len)) return_trace (false); for (const auto& record : partRecords.iter ()) if (!record.subset (c)) return_trace (false); return_trace (true); } bool sanitize (hb_sanitize_context_t *c) const { TRACE_SANITIZE (this); return_trace (c->check_struct (this) && italicsCorrection.sanitize (c, this) && partRecords.sanitize (c)); } unsigned int get_parts (hb_direction_t direction, hb_font_t *font, unsigned int start_offset, unsigned int *parts_count, /* IN/OUT */ hb_ot_math_glyph_part_t *parts /* OUT */, hb_position_t *italics_correction /* OUT */) const { if (parts_count) { int64_t mult = font->dir_mult (direction); for (auto _ : hb_zip (partRecords.sub_array (start_offset, parts_count), hb_array (parts, *parts_count))) _.first.extract (_.second, mult, font); } if (italics_correction) *italics_correction = italicsCorrection.get_x_value (font, this); return partRecords.len; } void closure_glyphs (hb_set_t *variant_glyphs) const { for (const auto& _ : partRecords.iter ()) _.closure_glyphs (variant_glyphs); } protected: MathValueRecord italicsCorrection; /* Italics correction of this * MathGlyphAssembly. Should not * depend on the assembly size. */ Array16Of<MathGlyphPartRecord> partRecords; /* Array of part records, from * left to right and bottom to * top. */ public: DEFINE_SIZE_ARRAY (6, partRecords); }; struct MathGlyphConstruction { bool subset (hb_subset_context_t *c) const { TRACE_SUBSET (this); auto *out = c->serializer->start_embed (*this); if (unlikely (!c->serializer->extend_min (out))) return_trace (false); out->glyphAssembly.serialize_subset (c, glyphAssembly, this); if (!c->serializer->check_assign (out->mathGlyphVariantRecord.len, mathGlyphVariantRecord.len, HB_SERIALIZE_ERROR_INT_OVERFLOW)) return_trace (false); for (const auto& record : mathGlyphVariantRecord.iter ()) if (!record.subset (c)) return_trace (false); return_trace (true); } bool sanitize (hb_sanitize_context_t *c) const { TRACE_SANITIZE (this); return_trace (c->check_struct (this) && glyphAssembly.sanitize (c, this) && mathGlyphVariantRecord.sanitize (c)); } const MathGlyphAssembly &get_assembly () const { return this+glyphAssembly; } unsigned int get_variants (hb_direction_t direction, hb_font_t *font, unsigned int start_offset, unsigned int *variants_count, /* IN/OUT */ hb_ot_math_glyph_variant_t *variants /* OUT */) const { if (variants_count) { int64_t mult = font->dir_mult (direction); for (auto _ : hb_zip (mathGlyphVariantRecord.sub_array (start_offset, variants_count), hb_array (variants, *variants_count))) _.second = {_.first.variantGlyph, font->em_mult (_.first.advanceMeasurement, mult)}; } return mathGlyphVariantRecord.len; } void closure_glyphs (hb_set_t *variant_glyphs) const { (this+glyphAssembly).closure_glyphs (variant_glyphs); for (const auto& _ : mathGlyphVariantRecord.iter ()) _.closure_glyphs (variant_glyphs); } protected: /* Offset to MathGlyphAssembly table for this shape - from the beginning of MathGlyphConstruction table. May be NULL. */ Offset16To<MathGlyphAssembly> glyphAssembly; /* MathGlyphVariantRecords for alternative variants of the glyphs. */ Array16Of<MathGlyphVariantRecord> mathGlyphVariantRecord; public: DEFINE_SIZE_ARRAY (4, mathGlyphVariantRecord); }; struct MathVariants { void closure_glyphs (const hb_set_t *glyph_set, hb_set_t *variant_glyphs) const { const hb_array_t<const Offset16To<MathGlyphConstruction>> glyph_construction_offsets = glyphConstruction.as_array (vertGlyphCount + horizGlyphCount); if (vertGlyphCoverage) { const auto vert_offsets = glyph_construction_offsets.sub_array (0, vertGlyphCount); + hb_zip (this+vertGlyphCoverage, vert_offsets) | hb_filter (glyph_set, hb_first) | hb_map (hb_second) | hb_map (hb_add (this)) | hb_apply ([=] (const MathGlyphConstruction &_) { _.closure_glyphs (variant_glyphs); }) ; } if (horizGlyphCoverage) { const auto hori_offsets = glyph_construction_offsets.sub_array (vertGlyphCount, horizGlyphCount); + hb_zip (this+horizGlyphCoverage, hori_offsets) | hb_filter (glyph_set, hb_first) | hb_map (hb_second) | hb_map (hb_add (this)) | hb_apply ([=] (const MathGlyphConstruction &_) { _.closure_glyphs (variant_glyphs); }) ; } } void collect_coverage_and_indices (hb_sorted_vector_t<hb_codepoint_t>& new_coverage, const Offset16To<Coverage>& coverage, unsigned i, unsigned end_index, hb_set_t& indices, const hb_set_t& glyphset, const hb_map_t& glyph_map) const { if (!coverage) return; for (const auto _ : (this+coverage).iter ()) { if (i >= end_index) return; if (glyphset.has (_)) { unsigned new_gid = glyph_map.get (_); new_coverage.push (new_gid); indices.add (i); } i++; } } bool subset (hb_subset_context_t *c) const { TRACE_SUBSET (this); const hb_set_t &glyphset = *c->plan->_glyphset_mathed; const hb_map_t &glyph_map = *c->plan->glyph_map; auto *out = c->serializer->start_embed (*this); if (unlikely (!c->serializer->extend_min (out))) return_trace (false); if (!c->serializer->check_assign (out->minConnectorOverlap, minConnectorOverlap, HB_SERIALIZE_ERROR_INT_OVERFLOW)) return_trace (false); hb_sorted_vector_t<hb_codepoint_t> new_vert_coverage; hb_sorted_vector_t<hb_codepoint_t> new_hori_coverage; hb_set_t indices; collect_coverage_and_indices (new_vert_coverage, vertGlyphCoverage, 0, vertGlyphCount, indices, glyphset, glyph_map); collect_coverage_and_indices (new_hori_coverage, horizGlyphCoverage, vertGlyphCount, vertGlyphCount + horizGlyphCount, indices, glyphset, glyph_map); if (!c->serializer->check_assign (out->vertGlyphCount, new_vert_coverage.length, HB_SERIALIZE_ERROR_INT_OVERFLOW)) return_trace (false); if (!c->serializer->check_assign (out->horizGlyphCount, new_hori_coverage.length, HB_SERIALIZE_ERROR_INT_OVERFLOW)) return_trace (false); for (unsigned i : indices.iter ()) { auto *o = c->serializer->embed (glyphConstruction[i]); if (!o) return_trace (false); o->serialize_subset (c, glyphConstruction[i], this); } out->vertGlyphCoverage.serialize_serialize (c->serializer, new_vert_coverage.iter ()); out->horizGlyphCoverage.serialize_serialize (c->serializer, new_hori_coverage.iter ()); return_trace (true); } bool sanitize_offsets (hb_sanitize_context_t *c) const { TRACE_SANITIZE (this); unsigned int count = vertGlyphCount + horizGlyphCount; for (unsigned int i = 0; i < count; i++) if (!glyphConstruction.arrayZ[i].sanitize (c, this)) return_trace (false); return_trace (true); } bool sanitize (hb_sanitize_context_t *c) const { TRACE_SANITIZE (this); return_trace (c->check_struct (this) && vertGlyphCoverage.sanitize (c, this) && horizGlyphCoverage.sanitize (c, this) && c->check_array (glyphConstruction.arrayZ, vertGlyphCount + horizGlyphCount) && sanitize_offsets (c)); } hb_position_t get_min_connector_overlap (hb_direction_t direction, hb_font_t *font) const { return font->em_scale_dir (minConnectorOverlap, direction); } unsigned int get_glyph_variants (hb_codepoint_t glyph, hb_direction_t direction, hb_font_t *font, unsigned int start_offset, unsigned int *variants_count, /* IN/OUT */ hb_ot_math_glyph_variant_t *variants /* OUT */) const { return get_glyph_construction (glyph, direction, font) .get_variants (direction, font, start_offset, variants_count, variants); } unsigned int get_glyph_parts (hb_codepoint_t glyph, hb_direction_t direction, hb_font_t *font, unsigned int start_offset, unsigned int *parts_count, /* IN/OUT */ hb_ot_math_glyph_part_t *parts /* OUT */, hb_position_t *italics_correction /* OUT */) const { return get_glyph_construction (glyph, direction, font) .get_assembly () .get_parts (direction, font, start_offset, parts_count, parts, italics_correction); } private: const MathGlyphConstruction & get_glyph_construction (hb_codepoint_t glyph, hb_direction_t direction, hb_font_t *font HB_UNUSED) const { bool vertical = HB_DIRECTION_IS_VERTICAL (direction); unsigned int count = vertical ? vertGlyphCount : horizGlyphCount; const Offset16To<Coverage> &coverage = vertical ? vertGlyphCoverage : horizGlyphCoverage; unsigned int index = (this+coverage).get_coverage (glyph); if (unlikely (index >= count)) return Null (MathGlyphConstruction); if (!vertical) index += vertGlyphCount; return this+glyphConstruction[index]; } protected: HBUINT16 minConnectorOverlap; /* Minimum overlap of connecting * glyphs during glyph construction, * in design units. */ Offset16To<Coverage> vertGlyphCoverage; /* Offset to Coverage table - * from the beginning of MathVariants * table. */ Offset16To<Coverage> horizGlyphCoverage; /* Offset to Coverage table - * from the beginning of MathVariants * table. */ HBUINT16 vertGlyphCount; /* Number of glyphs for which * information is provided for * vertically growing variants. */ HBUINT16 horizGlyphCount;/* Number of glyphs for which * information is provided for * horizontally growing variants. */ /* Array of offsets to MathGlyphConstruction tables - from the beginning of the MathVariants table, for shapes growing in vertical/horizontal direction. */ UnsizedArrayOf<Offset16To<MathGlyphConstruction>> glyphConstruction; public: DEFINE_SIZE_ARRAY (10, glyphConstruction); }; /* * MATH -- Mathematical typesetting * https://docs.microsoft.com/en-us/typography/opentype/spec/math */ struct MATH { static constexpr hb_tag_t tableTag = HB_OT_TAG_MATH; bool has_data () const { return version.to_int (); } void closure_glyphs (hb_set_t *glyph_set) const { if (mathVariants) { hb_set_t variant_glyphs; (this+mathVariants).closure_glyphs (glyph_set, &variant_glyphs); hb_set_union (glyph_set, &variant_glyphs); } } bool subset (hb_subset_context_t *c) const { TRACE_SUBSET (this); auto *out = c->serializer->embed (*this); if (unlikely (!out)) return_trace (false); out->mathConstants.serialize_copy (c->serializer, mathConstants, this, 0, hb_serialize_context_t::Head); out->mathGlyphInfo.serialize_subset (c, mathGlyphInfo, this); out->mathVariants.serialize_subset (c, mathVariants, this); return_trace (true); } bool sanitize (hb_sanitize_context_t *c) const { TRACE_SANITIZE (this); return_trace (version.sanitize (c) && likely (version.major == 1) && mathConstants.sanitize (c, this) && mathGlyphInfo.sanitize (c, this) && mathVariants.sanitize (c, this)); } hb_position_t get_constant (hb_ot_math_constant_t constant, hb_font_t *font) const { return (this+mathConstants).get_value (constant, font); } const MathGlyphInfo &get_glyph_info () const { return this+mathGlyphInfo; } const MathVariants &get_variants () const { return this+mathVariants; } protected: FixedVersion<>version; /* Version of the MATH table * initially set to 0x00010000u */ Offset16To<MathConstants> mathConstants; /* MathConstants table */ Offset16To<MathGlyphInfo> mathGlyphInfo; /* MathGlyphInfo table */ Offset16To<MathVariants> mathVariants; /* MathVariants table */ public: DEFINE_SIZE_STATIC (10); }; } /* namespace OT */ #endif /* HB_OT_MATH_TABLE_HH */