/* * Copyright © 2023 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 */ #ifndef HELPER_SUBSET_HH #define HELPER_SUBSET_HH #include "glib.h" #include #include #include #include "hb-subset.h" #ifndef HB_NO_VAR // Parses an axis position string and sets min, default, and max to // the requested values. If a value should be set to it's default value // then it will be set to NaN. static gboolean parse_axis_position(const char* s, float* min, float* def, float* max, gboolean* drop, GError **error) { const char* part = strpbrk(s, ":"); *drop = false; if (!part) { // Single value. if (strcmp (s, "drop") == 0) { *min = NAN; *def = NAN; *max = NAN; *drop = true; return true; } errno = 0; char *p; float axis_value = strtof (s, &p); if (errno || s == p) { g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, "Failed parsing axis value at: '%s'", s); return false; } *min = axis_value; *def = axis_value; *max = axis_value; return true; } float values[3]; int count = 0; for (int i = 0; i < 3; i++) { errno = 0; count++; if (!*s || part == s) { values[i] = NAN; if (part == NULL) break; s = part + 1; part = strpbrk(s, ":"); continue; } char *pend; values[i] = strtof (s, &pend); if (errno || s == pend || (part && pend != part)) { g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, "Failed parsing axis value at: '%s'", s); return false; } if (part == NULL) break; s = pend + 1; part = strpbrk(s, ":"); } if (count == 2) { *min = values[0]; *def = NAN; *max = values[1]; return true; } else if (count == 3) { *min = values[0]; *def = values[1]; *max = values[2]; return true; } g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, "Failed parsing axis value at: '%s'", s); return false; } static gboolean parse_instancing_spec (const char *arg, hb_face_t* face, hb_subset_input_t* input, GError **error) { char* s; while ((s = strtok((char *) arg, "="))) { arg = NULL; unsigned len = strlen (s); if (len > 4) //Axis tags are 4 bytes. { g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, "Failed parsing axis tag at: '%s'", s); return false; } /* support *=drop */ if (0 == strcmp (s, "*")) { s = strtok(NULL, ", "); if (0 != strcmp (s, "drop")) { g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, "Failed parsing axis position at: '%s'", s); return false; } if (!hb_subset_input_pin_all_axes_to_default (input, face)) { g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, "Failed pinning all axes to default."); return false; } continue; } hb_tag_t axis_tag = hb_tag_from_string (s, len); s = strtok(NULL, ", "); if (!s) { g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, "Value not specified for axis: %c%c%c%c", HB_UNTAG (axis_tag)); return false; } gboolean drop; float min, def, max; if (!parse_axis_position(s, &min, &def, &max, &drop, error)) return false; if (drop) { if (!hb_subset_input_pin_axis_to_default (input, face, axis_tag)) { g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, "Cannot pin axis: '%c%c%c%c', not present in fvar", HB_UNTAG (axis_tag)); return false; } continue; } if (min == def && def == max) { if (!hb_subset_input_pin_axis_location (input, face, axis_tag, def)) { g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, "Cannot pin axis: '%c%c%c%c', not present in fvar", HB_UNTAG (axis_tag)); return false; } continue; } if (!hb_subset_input_set_axis_range (input, face, axis_tag, min, max, def)) { g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, "Cannot pin axis: '%c%c%c%c', not present in fvar", HB_UNTAG (axis_tag)); return false; } continue; g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, "Partial instancing is not supported."); return false; } return true; } #endif #endif