| /* |
| * Copyright © 2018 Google, Inc. |
| * |
| * 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. |
| * |
| * Google Author(s): Garret Rieger, Rod Sheeter, Behdad Esfahbod |
| */ |
| |
| #include "hb-subset-instancer-solver.hh" |
| #include "hb-subset.hh" |
| #include "hb-set.hh" |
| #include "hb-utf.hh" |
| |
| |
| hb_subset_input_t::hb_subset_input_t () |
| { |
| for (auto& set : sets_iter ()) |
| set = hb::shared_ptr<hb_set_t> (hb_set_create ()); |
| |
| if (in_error ()) |
| return; |
| |
| flags = HB_SUBSET_FLAGS_DEFAULT; |
| |
| hb_set_add_range (sets.name_ids, 0, 6); |
| hb_set_add (sets.name_languages, 0x0409); |
| |
| hb_tag_t default_drop_tables[] = { |
| // Layout disabled by default |
| HB_TAG ('m', 'o', 'r', 'x'), |
| HB_TAG ('m', 'o', 'r', 't'), |
| HB_TAG ('k', 'e', 'r', 'x'), |
| HB_TAG ('k', 'e', 'r', 'n'), |
| |
| // Copied from fontTools: |
| HB_TAG ('J', 'S', 'T', 'F'), |
| HB_TAG ('D', 'S', 'I', 'G'), |
| HB_TAG ('E', 'B', 'D', 'T'), |
| HB_TAG ('E', 'B', 'L', 'C'), |
| HB_TAG ('E', 'B', 'S', 'C'), |
| HB_TAG ('S', 'V', 'G', ' '), |
| HB_TAG ('P', 'C', 'L', 'T'), |
| HB_TAG ('L', 'T', 'S', 'H'), |
| // Graphite tables |
| HB_TAG ('F', 'e', 'a', 't'), |
| HB_TAG ('G', 'l', 'a', 't'), |
| HB_TAG ('G', 'l', 'o', 'c'), |
| HB_TAG ('S', 'i', 'l', 'f'), |
| HB_TAG ('S', 'i', 'l', 'l'), |
| }; |
| sets.drop_tables->add_array (default_drop_tables, ARRAY_LENGTH (default_drop_tables)); |
| |
| hb_tag_t default_no_subset_tables[] = { |
| HB_TAG ('g', 'a', 's', 'p'), |
| HB_TAG ('f', 'p', 'g', 'm'), |
| HB_TAG ('p', 'r', 'e', 'p'), |
| HB_TAG ('V', 'D', 'M', 'X'), |
| HB_TAG ('D', 'S', 'I', 'G'), |
| }; |
| sets.no_subset_tables->add_array (default_no_subset_tables, |
| ARRAY_LENGTH (default_no_subset_tables)); |
| |
| //copied from _layout_features_groups in fonttools |
| hb_tag_t default_layout_features[] = { |
| // default shaper |
| // common |
| HB_TAG ('r', 'v', 'r', 'n'), |
| HB_TAG ('c', 'c', 'm', 'p'), |
| HB_TAG ('l', 'i', 'g', 'a'), |
| HB_TAG ('l', 'o', 'c', 'l'), |
| HB_TAG ('m', 'a', 'r', 'k'), |
| HB_TAG ('m', 'k', 'm', 'k'), |
| HB_TAG ('r', 'l', 'i', 'g'), |
| |
| //fractions |
| HB_TAG ('f', 'r', 'a', 'c'), |
| HB_TAG ('n', 'u', 'm', 'r'), |
| HB_TAG ('d', 'n', 'o', 'm'), |
| |
| //horizontal |
| HB_TAG ('c', 'a', 'l', 't'), |
| HB_TAG ('c', 'l', 'i', 'g'), |
| HB_TAG ('c', 'u', 'r', 's'), |
| HB_TAG ('k', 'e', 'r', 'n'), |
| HB_TAG ('r', 'c', 'l', 't'), |
| |
| //vertical |
| HB_TAG ('v', 'a', 'l', 't'), |
| HB_TAG ('v', 'e', 'r', 't'), |
| HB_TAG ('v', 'k', 'r', 'n'), |
| HB_TAG ('v', 'p', 'a', 'l'), |
| HB_TAG ('v', 'r', 't', '2'), |
| |
| //ltr |
| HB_TAG ('l', 't', 'r', 'a'), |
| HB_TAG ('l', 't', 'r', 'm'), |
| |
| //rtl |
| HB_TAG ('r', 't', 'l', 'a'), |
| HB_TAG ('r', 't', 'l', 'm'), |
| |
| //random |
| HB_TAG ('r', 'a', 'n', 'd'), |
| |
| //justify |
| HB_TAG ('j', 'a', 'l', 't'), // HarfBuzz doesn't use; others might |
| |
| //East Asian spacing |
| HB_TAG ('c', 'h', 'w', 's'), |
| HB_TAG ('v', 'c', 'h', 'w'), |
| HB_TAG ('h', 'a', 'l', 't'), |
| HB_TAG ('v', 'h', 'a', 'l'), |
| |
| //private |
| HB_TAG ('H', 'a', 'r', 'f'), |
| HB_TAG ('H', 'A', 'R', 'F'), |
| HB_TAG ('B', 'u', 'z', 'z'), |
| HB_TAG ('B', 'U', 'Z', 'Z'), |
| |
| //shapers |
| |
| //arabic |
| HB_TAG ('i', 'n', 'i', 't'), |
| HB_TAG ('m', 'e', 'd', 'i'), |
| HB_TAG ('f', 'i', 'n', 'a'), |
| HB_TAG ('i', 's', 'o', 'l'), |
| HB_TAG ('m', 'e', 'd', '2'), |
| HB_TAG ('f', 'i', 'n', '2'), |
| HB_TAG ('f', 'i', 'n', '3'), |
| HB_TAG ('c', 's', 'w', 'h'), |
| HB_TAG ('m', 's', 'e', 't'), |
| HB_TAG ('s', 't', 'c', 'h'), |
| |
| //hangul |
| HB_TAG ('l', 'j', 'm', 'o'), |
| HB_TAG ('v', 'j', 'm', 'o'), |
| HB_TAG ('t', 'j', 'm', 'o'), |
| |
| //tibetan |
| HB_TAG ('a', 'b', 'v', 's'), |
| HB_TAG ('b', 'l', 'w', 's'), |
| HB_TAG ('a', 'b', 'v', 'm'), |
| HB_TAG ('b', 'l', 'w', 'm'), |
| |
| //indic |
| HB_TAG ('n', 'u', 'k', 't'), |
| HB_TAG ('a', 'k', 'h', 'n'), |
| HB_TAG ('r', 'p', 'h', 'f'), |
| HB_TAG ('r', 'k', 'r', 'f'), |
| HB_TAG ('p', 'r', 'e', 'f'), |
| HB_TAG ('b', 'l', 'w', 'f'), |
| HB_TAG ('h', 'a', 'l', 'f'), |
| HB_TAG ('a', 'b', 'v', 'f'), |
| HB_TAG ('p', 's', 't', 'f'), |
| HB_TAG ('c', 'f', 'a', 'r'), |
| HB_TAG ('v', 'a', 't', 'u'), |
| HB_TAG ('c', 'j', 'c', 't'), |
| HB_TAG ('i', 'n', 'i', 't'), |
| HB_TAG ('p', 'r', 'e', 's'), |
| HB_TAG ('a', 'b', 'v', 's'), |
| HB_TAG ('b', 'l', 'w', 's'), |
| HB_TAG ('p', 's', 't', 's'), |
| HB_TAG ('h', 'a', 'l', 'n'), |
| HB_TAG ('d', 'i', 's', 't'), |
| HB_TAG ('a', 'b', 'v', 'm'), |
| HB_TAG ('b', 'l', 'w', 'm'), |
| }; |
| |
| sets.layout_features->add_array (default_layout_features, ARRAY_LENGTH (default_layout_features)); |
| |
| sets.layout_scripts->invert (); // Default to all scripts. |
| } |
| |
| /** |
| * hb_subset_input_create_or_fail: |
| * |
| * Creates a new subset input object. |
| * |
| * Return value: (transfer full): New subset input, or `NULL` if failed. Destroy |
| * with hb_subset_input_destroy(). |
| * |
| * Since: 1.8.0 |
| **/ |
| hb_subset_input_t * |
| hb_subset_input_create_or_fail (void) |
| { |
| hb_subset_input_t *input = hb_object_create<hb_subset_input_t>(); |
| |
| if (unlikely (!input)) |
| return nullptr; |
| |
| if (input->in_error ()) |
| { |
| hb_subset_input_destroy (input); |
| return nullptr; |
| } |
| |
| return input; |
| } |
| |
| /** |
| * hb_subset_input_reference: (skip) |
| * @input: a #hb_subset_input_t object. |
| * |
| * Increases the reference count on @input. |
| * |
| * Return value: @input. |
| * |
| * Since: 1.8.0 |
| **/ |
| hb_subset_input_t * |
| hb_subset_input_reference (hb_subset_input_t *input) |
| { |
| return hb_object_reference (input); |
| } |
| |
| /** |
| * hb_subset_input_destroy: |
| * @input: a #hb_subset_input_t object. |
| * |
| * Decreases the reference count on @input, and if it reaches zero, destroys |
| * @input, freeing all memory. |
| * |
| * Since: 1.8.0 |
| **/ |
| void |
| hb_subset_input_destroy (hb_subset_input_t *input) |
| { |
| if (!hb_object_destroy (input)) return; |
| |
| hb_free (input); |
| } |
| |
| /** |
| * hb_subset_input_unicode_set: |
| * @input: a #hb_subset_input_t object. |
| * |
| * Gets the set of Unicode code points to retain, the caller should modify the |
| * set as needed. |
| * |
| * Return value: (transfer none): pointer to the #hb_set_t of Unicode code |
| * points. |
| * |
| * Since: 1.8.0 |
| **/ |
| HB_EXTERN hb_set_t * |
| hb_subset_input_unicode_set (hb_subset_input_t *input) |
| { |
| return input->sets.unicodes; |
| } |
| |
| /** |
| * hb_subset_input_glyph_set: |
| * @input: a #hb_subset_input_t object. |
| * |
| * Gets the set of glyph IDs to retain, the caller should modify the set as |
| * needed. |
| * |
| * Return value: (transfer none): pointer to the #hb_set_t of glyph IDs. |
| * |
| * Since: 1.8.0 |
| **/ |
| HB_EXTERN hb_set_t * |
| hb_subset_input_glyph_set (hb_subset_input_t *input) |
| { |
| return input->sets.glyphs; |
| } |
| |
| /** |
| * hb_subset_input_set: |
| * @input: a #hb_subset_input_t object. |
| * @set_type: a #hb_subset_sets_t set type. |
| * |
| * Gets the set of the specified type. |
| * |
| * Return value: (transfer none): pointer to the #hb_set_t of the specified type. |
| * |
| * Since: 2.9.1 |
| **/ |
| HB_EXTERN hb_set_t * |
| hb_subset_input_set (hb_subset_input_t *input, hb_subset_sets_t set_type) |
| { |
| return input->sets_iter () [set_type]; |
| } |
| |
| /** |
| * hb_subset_input_get_flags: |
| * @input: a #hb_subset_input_t object. |
| * |
| * Gets all of the subsetting flags in the input object. |
| * |
| * Return value: the subsetting flags bit field. |
| * |
| * Since: 2.9.0 |
| **/ |
| HB_EXTERN hb_subset_flags_t |
| hb_subset_input_get_flags (hb_subset_input_t *input) |
| { |
| return (hb_subset_flags_t) input->flags; |
| } |
| |
| /** |
| * hb_subset_input_set_flags: |
| * @input: a #hb_subset_input_t object. |
| * @value: bit field of flags |
| * |
| * Sets all of the flags in the input object to the values specified by the bit |
| * field. |
| * |
| * Since: 2.9.0 |
| **/ |
| HB_EXTERN void |
| hb_subset_input_set_flags (hb_subset_input_t *input, |
| unsigned value) |
| { |
| input->flags = (hb_subset_flags_t) value; |
| } |
| |
| /** |
| * hb_subset_input_set_user_data: (skip) |
| * @input: a #hb_subset_input_t object. |
| * @key: The user-data key to set |
| * @data: A pointer to the user data |
| * @destroy: (nullable): A callback to call when @data is not needed anymore |
| * @replace: Whether to replace an existing data with the same key |
| * |
| * Attaches a user-data key/data pair to the given subset input object. |
| * |
| * Return value: `true` if success, `false` otherwise |
| * |
| * Since: 2.9.0 |
| **/ |
| hb_bool_t |
| hb_subset_input_set_user_data (hb_subset_input_t *input, |
| hb_user_data_key_t *key, |
| void * data, |
| hb_destroy_func_t destroy, |
| hb_bool_t replace) |
| { |
| return hb_object_set_user_data (input, key, data, destroy, replace); |
| } |
| |
| /** |
| * hb_subset_input_get_user_data: (skip) |
| * @input: a #hb_subset_input_t object. |
| * @key: The user-data key to query |
| * |
| * Fetches the user data associated with the specified key, |
| * attached to the specified subset input object. |
| * |
| * Return value: (transfer none): A pointer to the user data |
| * |
| * Since: 2.9.0 |
| **/ |
| void * |
| hb_subset_input_get_user_data (const hb_subset_input_t *input, |
| hb_user_data_key_t *key) |
| { |
| return hb_object_get_user_data (input, key); |
| } |
| |
| /** |
| * hb_subset_input_keep_everything: |
| * @input: a #hb_subset_input_t object |
| * |
| * Configure input object to keep everything in the font face. |
| * That is, all Unicodes, glyphs, names, layout items, |
| * glyph names, etc. |
| * |
| * The input can be tailored afterwards by the caller. |
| * |
| * Since: 7.0.0 |
| */ |
| void |
| hb_subset_input_keep_everything (hb_subset_input_t *input) |
| { |
| const hb_subset_sets_t indices[] = {HB_SUBSET_SETS_UNICODE, |
| HB_SUBSET_SETS_GLYPH_INDEX, |
| HB_SUBSET_SETS_NAME_ID, |
| HB_SUBSET_SETS_NAME_LANG_ID, |
| HB_SUBSET_SETS_LAYOUT_FEATURE_TAG, |
| HB_SUBSET_SETS_LAYOUT_SCRIPT_TAG}; |
| |
| for (auto idx : hb_iter (indices)) |
| { |
| hb_set_t *set = hb_subset_input_set (input, idx); |
| hb_set_clear (set); |
| hb_set_invert (set); |
| } |
| |
| // Don't drop any tables |
| hb_set_clear (hb_subset_input_set (input, HB_SUBSET_SETS_DROP_TABLE_TAG)); |
| |
| hb_subset_input_set_flags (input, |
| HB_SUBSET_FLAGS_NOTDEF_OUTLINE | |
| HB_SUBSET_FLAGS_GLYPH_NAMES | |
| HB_SUBSET_FLAGS_NAME_LEGACY | |
| HB_SUBSET_FLAGS_NO_PRUNE_UNICODE_RANGES | |
| HB_SUBSET_FLAGS_PASSTHROUGH_UNRECOGNIZED); |
| } |
| |
| #ifndef HB_NO_VAR |
| /** |
| * hb_subset_input_pin_all_axes_to_default: (skip) |
| * @input: a #hb_subset_input_t object. |
| * @face: a #hb_face_t object. |
| * |
| * Pin all axes to default locations in the given subset input object. |
| * |
| * All axes in a font must be pinned. Additionally, `CFF2` table, if present, |
| * will be de-subroutinized. |
| * |
| * Return value: `true` if success, `false` otherwise |
| * |
| * Since: 8.3.1 |
| **/ |
| HB_EXTERN hb_bool_t |
| hb_subset_input_pin_all_axes_to_default (hb_subset_input_t *input, |
| hb_face_t *face) |
| { |
| unsigned axis_count = hb_ot_var_get_axis_count (face); |
| if (!axis_count) return false; |
| |
| hb_ot_var_axis_info_t *axis_infos = (hb_ot_var_axis_info_t *) hb_calloc (axis_count, sizeof (hb_ot_var_axis_info_t)); |
| if (unlikely (!axis_infos)) return false; |
| |
| (void) hb_ot_var_get_axis_infos (face, 0, &axis_count, axis_infos); |
| |
| for (unsigned i = 0; i < axis_count; i++) |
| { |
| hb_tag_t axis_tag = axis_infos[i].tag; |
| double default_val = (double) axis_infos[i].default_value; |
| if (!input->axes_location.set (axis_tag, Triple (default_val, default_val, default_val))) |
| { |
| hb_free (axis_infos); |
| return false; |
| } |
| } |
| hb_free (axis_infos); |
| return true; |
| } |
| |
| /** |
| * hb_subset_input_pin_axis_to_default: (skip) |
| * @input: a #hb_subset_input_t object. |
| * @face: a #hb_face_t object. |
| * @axis_tag: Tag of the axis to be pinned |
| * |
| * Pin an axis to its default location in the given subset input object. |
| * |
| * All axes in a font must be pinned. Additionally, `CFF2` table, if present, |
| * will be de-subroutinized. |
| * |
| * Return value: `true` if success, `false` otherwise |
| * |
| * Since: 6.0.0 |
| **/ |
| HB_EXTERN hb_bool_t |
| hb_subset_input_pin_axis_to_default (hb_subset_input_t *input, |
| hb_face_t *face, |
| hb_tag_t axis_tag) |
| { |
| hb_ot_var_axis_info_t axis_info; |
| if (!hb_ot_var_find_axis_info (face, axis_tag, &axis_info)) |
| return false; |
| |
| double default_val = (double) axis_info.default_value; |
| return input->axes_location.set (axis_tag, Triple (default_val, default_val, default_val)); |
| } |
| |
| /** |
| * hb_subset_input_pin_axis_location: (skip) |
| * @input: a #hb_subset_input_t object. |
| * @face: a #hb_face_t object. |
| * @axis_tag: Tag of the axis to be pinned |
| * @axis_value: Location on the axis to be pinned at |
| * |
| * Pin an axis to a fixed location in the given subset input object. |
| * |
| * All axes in a font must be pinned. Additionally, `CFF2` table, if present, |
| * will be de-subroutinized. |
| * |
| * Return value: `true` if success, `false` otherwise |
| * |
| * Since: 6.0.0 |
| **/ |
| HB_EXTERN hb_bool_t |
| hb_subset_input_pin_axis_location (hb_subset_input_t *input, |
| hb_face_t *face, |
| hb_tag_t axis_tag, |
| float axis_value) |
| { |
| hb_ot_var_axis_info_t axis_info; |
| if (!hb_ot_var_find_axis_info (face, axis_tag, &axis_info)) |
| return false; |
| |
| double val = hb_clamp((double) axis_value, (double) axis_info.min_value, (double) axis_info.max_value); |
| return input->axes_location.set (axis_tag, Triple (val, val, val)); |
| } |
| |
| /** |
| * hb_subset_input_set_axis_range: (skip) |
| * @input: a #hb_subset_input_t object. |
| * @face: a #hb_face_t object. |
| * @axis_tag: Tag of the axis |
| * @axis_min_value: Minimum value of the axis variation range to set, if NaN the existing min will be used. |
| * @axis_max_value: Maximum value of the axis variation range to set if NaN the existing max will be used. |
| * @axis_def_value: Default value of the axis variation range to set, if NaN the existing default will be used. |
| * |
| * Restricting the range of variation on an axis in the given subset input object. |
| * New min/default/max values will be clamped if they're not within the fvar axis range. |
| * |
| * If the fvar axis default value is not within the new range, the new default |
| * value will be changed to the new min or max value, whichever is closer to the fvar |
| * axis default. |
| * |
| * Note: input min value can not be bigger than input max value. If the input |
| * default value is not within the new min/max range, it'll be clamped. |
| * Note: currently it supports gvar and cvar tables only. |
| * |
| * Return value: `true` if success, `false` otherwise |
| * |
| * Since: 8.5.0 |
| **/ |
| HB_EXTERN hb_bool_t |
| hb_subset_input_set_axis_range (hb_subset_input_t *input, |
| hb_face_t *face, |
| hb_tag_t axis_tag, |
| float axis_min_value, |
| float axis_max_value, |
| float axis_def_value) |
| { |
| hb_ot_var_axis_info_t axis_info; |
| if (!hb_ot_var_find_axis_info (face, axis_tag, &axis_info)) |
| return false; |
| |
| float min = !std::isnan(axis_min_value) ? axis_min_value : axis_info.min_value; |
| float max = !std::isnan(axis_max_value) ? axis_max_value : axis_info.max_value; |
| float def = !std::isnan(axis_def_value) ? axis_def_value : axis_info.default_value; |
| |
| if (min > max) |
| return false; |
| |
| float new_min_val = hb_clamp(min, axis_info.min_value, axis_info.max_value); |
| float new_max_val = hb_clamp(max, axis_info.min_value, axis_info.max_value); |
| float new_default_val = hb_clamp(def, new_min_val, new_max_val); |
| return input->axes_location.set (axis_tag, Triple ((double) new_min_val, (double) new_default_val, (double) new_max_val)); |
| } |
| |
| /** |
| * hb_subset_input_get_axis_range: (skip) |
| * @input: a #hb_subset_input_t object. |
| * @axis_tag: Tag of the axis |
| * @axis_min_value: Set to the previously configured minimum value of the axis variation range. |
| * @axis_max_value: Set to the previously configured maximum value of the axis variation range. |
| * @axis_def_value: Set to the previously configured default value of the axis variation range. |
| * |
| * Gets the axis range assigned by previous calls to hb_subset_input_set_axis_range. |
| * |
| * Return value: `true` if a range has been set for this axis tag, `false` otherwise. |
| * |
| * Since: 8.5.0 |
| **/ |
| HB_EXTERN hb_bool_t |
| hb_subset_input_get_axis_range (hb_subset_input_t *input, |
| hb_tag_t axis_tag, |
| float *axis_min_value, |
| float *axis_max_value, |
| float *axis_def_value) |
| |
| { |
| Triple* triple; |
| if (!input->axes_location.has(axis_tag, &triple)) { |
| return false; |
| } |
| |
| *axis_min_value = triple->minimum; |
| *axis_def_value = triple->middle; |
| *axis_max_value = triple->maximum; |
| return true; |
| } |
| #endif |
| |
| /** |
| * hb_subset_preprocess: |
| * @source: a #hb_face_t object. |
| * |
| * Preprocesses the face and attaches data that will be needed by the |
| * subsetter. Future subsetting operations can then use the precomputed data |
| * to speed up the subsetting operation. |
| * |
| * See [subset-preprocessing](https://github.com/harfbuzz/harfbuzz/blob/main/docs/subset-preprocessing.md) |
| * for more information. |
| * |
| * Note: the preprocessed face may contain sub-blobs that reference the memory |
| * backing the source #hb_face_t. Therefore in the case that this memory is not |
| * owned by the source face you will need to ensure that memory lives |
| * as long as the returned #hb_face_t. |
| * |
| * Returns: a new #hb_face_t. |
| * |
| * Since: 6.0.0 |
| **/ |
| |
| HB_EXTERN hb_face_t * |
| hb_subset_preprocess (hb_face_t *source) |
| { |
| hb_subset_input_t* input = hb_subset_input_create_or_fail (); |
| if (!input) |
| return hb_face_reference (source); |
| |
| hb_subset_input_keep_everything (input); |
| |
| input->attach_accelerator_data = true; |
| |
| // Always use long loca in the preprocessed version. This allows |
| // us to store the glyph bytes unpadded which allows the future subset |
| // operation to run faster by skipping the trim padding step. |
| input->force_long_loca = true; |
| |
| hb_face_t* new_source = hb_subset_or_fail (source, input); |
| hb_subset_input_destroy (input); |
| |
| if (!new_source) { |
| DEBUG_MSG (SUBSET, nullptr, "Preprocessing failed due to subset failure."); |
| return hb_face_reference (source); |
| } |
| |
| return new_source; |
| } |
| |
| /** |
| * hb_subset_input_old_to_new_glyph_mapping: |
| * @input: a #hb_subset_input_t object. |
| * |
| * Returns a map which can be used to provide an explicit mapping from old to new glyph |
| * id's in the produced subset. The caller should populate the map as desired. |
| * If this map is left empty then glyph ids will be automatically mapped to new |
| * values by the subsetter. If populated, the mapping must be unique. That |
| * is no two original glyph ids can be mapped to the same new id. |
| * Additionally, if a mapping is provided then the retain gids option cannot |
| * be enabled. |
| * |
| * Any glyphs that are retained in the subset which are not specified |
| * in this mapping will be assigned glyph ids after the highest glyph |
| * id in the mapping. |
| * |
| * Note: this will accept and apply non-monotonic mappings, however this |
| * may result in unsorted Coverage tables. Such fonts may not work for all |
| * use cases (for example ots will reject unsorted coverage tables). So it's |
| * recommended, if possible, to supply a monotonic mapping. |
| * |
| * Return value: (transfer none): pointer to the #hb_map_t of the custom glyphs ID map. |
| * |
| * Since: 7.3.0 |
| **/ |
| HB_EXTERN hb_map_t* |
| hb_subset_input_old_to_new_glyph_mapping (hb_subset_input_t *input) |
| { |
| return &input->glyph_map; |
| } |
| |
| #ifdef HB_EXPERIMENTAL_API |
| /** |
| * hb_subset_input_override_name_table: |
| * @input: a #hb_subset_input_t object. |
| * @name_id: name_id of a nameRecord |
| * @platform_id: platform ID of a nameRecord |
| * @encoding_id: encoding ID of a nameRecord |
| * @language_id: language ID of a nameRecord |
| * @name_str: pointer to name string new value or null to indicate should remove |
| * @str_len: the size of @name_str, or -1 if it is `NULL`-terminated |
| * |
| * Override the name string of the NameRecord identified by name_id, |
| * platform_id, encoding_id and language_id. If a record with that name_id |
| * doesn't exist, create it and insert to the name table. |
| * |
| * Note: for mac platform, we only support name_str with all ascii characters, |
| * name_str with non-ascii characters will be ignored. |
| * |
| * XSince: EXPERIMENTAL |
| **/ |
| HB_EXTERN hb_bool_t |
| hb_subset_input_override_name_table (hb_subset_input_t *input, |
| hb_ot_name_id_t name_id, |
| unsigned platform_id, |
| unsigned encoding_id, |
| unsigned language_id, |
| const char *name_str, |
| int str_len /* -1 means nul-terminated */) |
| { |
| if (!name_str) |
| { |
| str_len = 0; |
| } |
| else if (str_len == -1) |
| { |
| str_len = strlen (name_str); |
| } |
| |
| hb_bytes_t name_bytes (nullptr, 0); |
| if (str_len) |
| { |
| if (platform_id == 1) |
| { |
| const uint8_t *src = reinterpret_cast<const uint8_t*> (name_str); |
| const uint8_t *src_end = src + str_len; |
| |
| hb_codepoint_t unicode; |
| const hb_codepoint_t replacement = HB_BUFFER_REPLACEMENT_CODEPOINT_DEFAULT; |
| while (src < src_end) |
| { |
| src = hb_utf8_t::next (src, src_end, &unicode, replacement); |
| if (unicode >= 0x0080u) |
| { |
| printf ("Non-ascii character detected, ignored...This API supports ascii characters only for mac platform\n"); |
| return false; |
| } |
| } |
| } |
| char *override_name = (char *) hb_malloc (str_len); |
| if (unlikely (!override_name)) return false; |
| |
| hb_memcpy (override_name, name_str, str_len); |
| name_bytes = hb_bytes_t (override_name, str_len); |
| } |
| input->name_table_overrides.set (hb_ot_name_record_ids_t (platform_id, encoding_id, language_id, name_id), name_bytes); |
| return true; |
| } |
| #endif |