Mercurial > hgrepos > Python2 > PyMuPDF
comparison mupdf-source/thirdparty/harfbuzz/src/hb-ot-map.cc @ 2:b50eed0cc0ef upstream
ADD: MuPDF v1.26.7: the MuPDF source as downloaded by a default build of PyMuPDF 1.26.4.
The directory name has changed: no version number in the expanded directory now.
| author | Franz Glasner <fzglas.hg@dom66.de> |
|---|---|
| date | Mon, 15 Sep 2025 11:43:07 +0200 |
| parents | |
| children |
comparison
equal
deleted
inserted
replaced
| 1:1d09e1dec1d9 | 2:b50eed0cc0ef |
|---|---|
| 1 /* | |
| 2 * Copyright © 2009,2010 Red Hat, Inc. | |
| 3 * Copyright © 2010,2011,2013 Google, Inc. | |
| 4 * | |
| 5 * This is part of HarfBuzz, a text shaping library. | |
| 6 * | |
| 7 * Permission is hereby granted, without written agreement and without | |
| 8 * license or royalty fees, to use, copy, modify, and distribute this | |
| 9 * software and its documentation for any purpose, provided that the | |
| 10 * above copyright notice and the following two paragraphs appear in | |
| 11 * all copies of this software. | |
| 12 * | |
| 13 * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR | |
| 14 * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES | |
| 15 * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN | |
| 16 * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH | |
| 17 * DAMAGE. | |
| 18 * | |
| 19 * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING, | |
| 20 * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND | |
| 21 * FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS | |
| 22 * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO | |
| 23 * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS. | |
| 24 * | |
| 25 * Red Hat Author(s): Behdad Esfahbod | |
| 26 * Google Author(s): Behdad Esfahbod | |
| 27 */ | |
| 28 | |
| 29 #include "hb.hh" | |
| 30 | |
| 31 #ifndef HB_NO_OT_SHAPE | |
| 32 | |
| 33 #include "hb-ot-map.hh" | |
| 34 #include "hb-ot-shape.hh" | |
| 35 #include "hb-ot-layout.hh" | |
| 36 | |
| 37 | |
| 38 void hb_ot_map_t::collect_lookups (unsigned int table_index, hb_set_t *lookups_out) const | |
| 39 { | |
| 40 for (unsigned int i = 0; i < lookups[table_index].length; i++) | |
| 41 lookups_out->add (lookups[table_index][i].index); | |
| 42 } | |
| 43 | |
| 44 | |
| 45 hb_ot_map_builder_t::hb_ot_map_builder_t (hb_face_t *face_, | |
| 46 const hb_segment_properties_t &props_) | |
| 47 { | |
| 48 hb_memset (this, 0, sizeof (*this)); | |
| 49 | |
| 50 feature_infos.init (); | |
| 51 for (unsigned int table_index = 0; table_index < 2; table_index++) | |
| 52 stages[table_index].init (); | |
| 53 | |
| 54 face = face_; | |
| 55 props = props_; | |
| 56 | |
| 57 /* Fetch script/language indices for GSUB/GPOS. We need these later to skip | |
| 58 * features not available in either table and not waste precious bits for them. */ | |
| 59 | |
| 60 unsigned int script_count = HB_OT_MAX_TAGS_PER_SCRIPT; | |
| 61 unsigned int language_count = HB_OT_MAX_TAGS_PER_LANGUAGE; | |
| 62 hb_tag_t script_tags[HB_OT_MAX_TAGS_PER_SCRIPT]; | |
| 63 hb_tag_t language_tags[HB_OT_MAX_TAGS_PER_LANGUAGE]; | |
| 64 | |
| 65 hb_ot_tags_from_script_and_language (props.script, | |
| 66 props.language, | |
| 67 &script_count, | |
| 68 script_tags, | |
| 69 &language_count, | |
| 70 language_tags); | |
| 71 | |
| 72 for (unsigned int table_index = 0; table_index < 2; table_index++) | |
| 73 { | |
| 74 hb_tag_t table_tag = table_tags[table_index]; | |
| 75 found_script[table_index] = (bool) hb_ot_layout_table_select_script (face, | |
| 76 table_tag, | |
| 77 script_count, | |
| 78 script_tags, | |
| 79 &script_index[table_index], | |
| 80 &chosen_script[table_index]); | |
| 81 hb_ot_layout_script_select_language (face, | |
| 82 table_tag, | |
| 83 script_index[table_index], | |
| 84 language_count, | |
| 85 language_tags, | |
| 86 &language_index[table_index]); | |
| 87 } | |
| 88 } | |
| 89 | |
| 90 hb_ot_map_builder_t::~hb_ot_map_builder_t () | |
| 91 { | |
| 92 feature_infos.fini (); | |
| 93 for (unsigned int table_index = 0; table_index < 2; table_index++) | |
| 94 stages[table_index].fini (); | |
| 95 } | |
| 96 | |
| 97 void hb_ot_map_builder_t::add_feature (hb_tag_t tag, | |
| 98 hb_ot_map_feature_flags_t flags, | |
| 99 unsigned int value) | |
| 100 { | |
| 101 if (unlikely (!tag)) return; | |
| 102 feature_info_t *info = feature_infos.push(); | |
| 103 info->tag = tag; | |
| 104 info->seq = feature_infos.length; | |
| 105 info->max_value = value; | |
| 106 info->flags = flags; | |
| 107 info->default_value = (flags & F_GLOBAL) ? value : 0; | |
| 108 info->stage[0] = current_stage[0]; | |
| 109 info->stage[1] = current_stage[1]; | |
| 110 } | |
| 111 | |
| 112 bool hb_ot_map_builder_t::has_feature (hb_tag_t tag) | |
| 113 { | |
| 114 for (unsigned int table_index = 0; table_index < 2; table_index++) | |
| 115 { | |
| 116 if (hb_ot_layout_language_find_feature (face, | |
| 117 table_tags[table_index], | |
| 118 script_index[table_index], | |
| 119 language_index[table_index], | |
| 120 tag, | |
| 121 nullptr)) | |
| 122 return true; | |
| 123 } | |
| 124 return false; | |
| 125 } | |
| 126 | |
| 127 void | |
| 128 hb_ot_map_builder_t::add_lookups (hb_ot_map_t &m, | |
| 129 unsigned int table_index, | |
| 130 unsigned int feature_index, | |
| 131 unsigned int variations_index, | |
| 132 hb_mask_t mask, | |
| 133 bool auto_zwnj, | |
| 134 bool auto_zwj, | |
| 135 bool random, | |
| 136 bool per_syllable, | |
| 137 hb_tag_t feature_tag) | |
| 138 { | |
| 139 unsigned int lookup_indices[32]; | |
| 140 unsigned int offset, len; | |
| 141 unsigned int table_lookup_count; | |
| 142 | |
| 143 table_lookup_count = hb_ot_layout_table_get_lookup_count (face, table_tags[table_index]); | |
| 144 | |
| 145 offset = 0; | |
| 146 do { | |
| 147 len = ARRAY_LENGTH (lookup_indices); | |
| 148 hb_ot_layout_feature_with_variations_get_lookups (face, | |
| 149 table_tags[table_index], | |
| 150 feature_index, | |
| 151 variations_index, | |
| 152 offset, &len, | |
| 153 lookup_indices); | |
| 154 | |
| 155 for (unsigned int i = 0; i < len; i++) | |
| 156 { | |
| 157 if (lookup_indices[i] >= table_lookup_count) | |
| 158 continue; | |
| 159 hb_ot_map_t::lookup_map_t *lookup = m.lookups[table_index].push (); | |
| 160 lookup->mask = mask; | |
| 161 lookup->index = lookup_indices[i]; | |
| 162 lookup->auto_zwnj = auto_zwnj; | |
| 163 lookup->auto_zwj = auto_zwj; | |
| 164 lookup->random = random; | |
| 165 lookup->per_syllable = per_syllable; | |
| 166 lookup->feature_tag = feature_tag; | |
| 167 } | |
| 168 | |
| 169 offset += len; | |
| 170 } while (len == ARRAY_LENGTH (lookup_indices)); | |
| 171 } | |
| 172 | |
| 173 | |
| 174 void hb_ot_map_builder_t::add_pause (unsigned int table_index, hb_ot_map_t::pause_func_t pause_func) | |
| 175 { | |
| 176 stage_info_t *s = stages[table_index].push (); | |
| 177 s->index = current_stage[table_index]; | |
| 178 s->pause_func = pause_func; | |
| 179 | |
| 180 current_stage[table_index]++; | |
| 181 } | |
| 182 | |
| 183 void | |
| 184 hb_ot_map_builder_t::compile (hb_ot_map_t &m, | |
| 185 const hb_ot_shape_plan_key_t &key) | |
| 186 { | |
| 187 unsigned int global_bit_shift = 8 * sizeof (hb_mask_t) - 1; | |
| 188 unsigned int global_bit_mask = 1u << global_bit_shift; | |
| 189 | |
| 190 m.global_mask = global_bit_mask; | |
| 191 | |
| 192 unsigned int required_feature_index[2]; | |
| 193 hb_tag_t required_feature_tag[2]; | |
| 194 /* We default to applying required feature in stage 0. If the required | |
| 195 * feature has a tag that is known to the shaper, we apply required feature | |
| 196 * in the stage for that tag. | |
| 197 */ | |
| 198 unsigned int required_feature_stage[2] = {0, 0}; | |
| 199 | |
| 200 for (unsigned int table_index = 0; table_index < 2; table_index++) | |
| 201 { | |
| 202 m.chosen_script[table_index] = chosen_script[table_index]; | |
| 203 m.found_script[table_index] = found_script[table_index]; | |
| 204 | |
| 205 hb_ot_layout_language_get_required_feature (face, | |
| 206 table_tags[table_index], | |
| 207 script_index[table_index], | |
| 208 language_index[table_index], | |
| 209 &required_feature_index[table_index], | |
| 210 &required_feature_tag[table_index]); | |
| 211 } | |
| 212 | |
| 213 /* Sort features and merge duplicates */ | |
| 214 if (feature_infos.length) | |
| 215 { | |
| 216 feature_infos.qsort (); | |
| 217 auto *f = feature_infos.arrayZ; | |
| 218 unsigned int j = 0; | |
| 219 unsigned count = feature_infos.length; | |
| 220 for (unsigned int i = 1; i < count; i++) | |
| 221 if (f[i].tag != f[j].tag) | |
| 222 f[++j] = f[i]; | |
| 223 else { | |
| 224 if (f[i].flags & F_GLOBAL) { | |
| 225 f[j].flags |= F_GLOBAL; | |
| 226 f[j].max_value = f[i].max_value; | |
| 227 f[j].default_value = f[i].default_value; | |
| 228 } else { | |
| 229 if (f[j].flags & F_GLOBAL) | |
| 230 f[j].flags ^= F_GLOBAL; | |
| 231 f[j].max_value = hb_max (f[j].max_value, f[i].max_value); | |
| 232 /* Inherit default_value from j */ | |
| 233 } | |
| 234 f[j].flags |= (f[i].flags & F_HAS_FALLBACK); | |
| 235 f[j].stage[0] = hb_min (f[j].stage[0], f[i].stage[0]); | |
| 236 f[j].stage[1] = hb_min (f[j].stage[1], f[i].stage[1]); | |
| 237 } | |
| 238 feature_infos.shrink (j + 1); | |
| 239 } | |
| 240 | |
| 241 | |
| 242 /* Allocate bits now */ | |
| 243 static_assert ((!(HB_GLYPH_FLAG_DEFINED & (HB_GLYPH_FLAG_DEFINED + 1))), ""); | |
| 244 unsigned int next_bit = hb_popcount (HB_GLYPH_FLAG_DEFINED) + 1; | |
| 245 | |
| 246 unsigned count = feature_infos.length; | |
| 247 for (unsigned int i = 0; i < count; i++) | |
| 248 { | |
| 249 const feature_info_t *info = &feature_infos[i]; | |
| 250 | |
| 251 unsigned int bits_needed; | |
| 252 | |
| 253 if ((info->flags & F_GLOBAL) && info->max_value == 1) | |
| 254 /* Uses the global bit */ | |
| 255 bits_needed = 0; | |
| 256 else | |
| 257 /* Limit bits per feature. */ | |
| 258 bits_needed = hb_min (HB_OT_MAP_MAX_BITS, hb_bit_storage (info->max_value)); | |
| 259 | |
| 260 if (!info->max_value || next_bit + bits_needed >= global_bit_shift) | |
| 261 continue; /* Feature disabled, or not enough bits. */ | |
| 262 | |
| 263 | |
| 264 bool found = false; | |
| 265 unsigned int feature_index[2]; | |
| 266 for (unsigned int table_index = 0; table_index < 2; table_index++) | |
| 267 { | |
| 268 if (required_feature_tag[table_index] == info->tag) | |
| 269 required_feature_stage[table_index] = info->stage[table_index]; | |
| 270 | |
| 271 found |= (bool) hb_ot_layout_language_find_feature (face, | |
| 272 table_tags[table_index], | |
| 273 script_index[table_index], | |
| 274 language_index[table_index], | |
| 275 info->tag, | |
| 276 &feature_index[table_index]); | |
| 277 } | |
| 278 if (!found && (info->flags & F_GLOBAL_SEARCH)) | |
| 279 { | |
| 280 for (unsigned int table_index = 0; table_index < 2; table_index++) | |
| 281 { | |
| 282 found |= (bool) hb_ot_layout_table_find_feature (face, | |
| 283 table_tags[table_index], | |
| 284 info->tag, | |
| 285 &feature_index[table_index]); | |
| 286 } | |
| 287 } | |
| 288 if (!found && !(info->flags & F_HAS_FALLBACK)) | |
| 289 continue; | |
| 290 | |
| 291 | |
| 292 hb_ot_map_t::feature_map_t *map = m.features.push (); | |
| 293 | |
| 294 map->tag = info->tag; | |
| 295 map->index[0] = feature_index[0]; | |
| 296 map->index[1] = feature_index[1]; | |
| 297 map->stage[0] = info->stage[0]; | |
| 298 map->stage[1] = info->stage[1]; | |
| 299 map->auto_zwnj = !(info->flags & F_MANUAL_ZWNJ); | |
| 300 map->auto_zwj = !(info->flags & F_MANUAL_ZWJ); | |
| 301 map->random = !!(info->flags & F_RANDOM); | |
| 302 map->per_syllable = !!(info->flags & F_PER_SYLLABLE); | |
| 303 if ((info->flags & F_GLOBAL) && info->max_value == 1) { | |
| 304 /* Uses the global bit */ | |
| 305 map->shift = global_bit_shift; | |
| 306 map->mask = global_bit_mask; | |
| 307 } else { | |
| 308 map->shift = next_bit; | |
| 309 map->mask = (1u << (next_bit + bits_needed)) - (1u << next_bit); | |
| 310 next_bit += bits_needed; | |
| 311 m.global_mask |= (info->default_value << map->shift) & map->mask; | |
| 312 } | |
| 313 map->_1_mask = (1u << map->shift) & map->mask; | |
| 314 map->needs_fallback = !found; | |
| 315 } | |
| 316 //feature_infos.shrink (0); /* Done with these */ | |
| 317 | |
| 318 | |
| 319 add_gsub_pause (nullptr); | |
| 320 add_gpos_pause (nullptr); | |
| 321 | |
| 322 for (unsigned int table_index = 0; table_index < 2; table_index++) | |
| 323 { | |
| 324 /* Collect lookup indices for features */ | |
| 325 auto &lookups = m.lookups[table_index]; | |
| 326 | |
| 327 unsigned int stage_index = 0; | |
| 328 unsigned int last_num_lookups = 0; | |
| 329 for (unsigned stage = 0; stage < current_stage[table_index]; stage++) | |
| 330 { | |
| 331 if (required_feature_index[table_index] != HB_OT_LAYOUT_NO_FEATURE_INDEX && | |
| 332 required_feature_stage[table_index] == stage) | |
| 333 add_lookups (m, table_index, | |
| 334 required_feature_index[table_index], | |
| 335 key.variations_index[table_index], | |
| 336 global_bit_mask); | |
| 337 | |
| 338 for (auto &feature : m.features) | |
| 339 { | |
| 340 if (feature.stage[table_index] == stage) | |
| 341 add_lookups (m, table_index, | |
| 342 feature.index[table_index], | |
| 343 key.variations_index[table_index], | |
| 344 feature.mask, | |
| 345 feature.auto_zwnj, | |
| 346 feature.auto_zwj, | |
| 347 feature.random, | |
| 348 feature.per_syllable, | |
| 349 feature.tag); | |
| 350 } | |
| 351 | |
| 352 /* Sort lookups and merge duplicates */ | |
| 353 if (last_num_lookups < lookups.length) | |
| 354 { | |
| 355 lookups.as_array ().sub_array (last_num_lookups, lookups.length - last_num_lookups).qsort (); | |
| 356 | |
| 357 unsigned int j = last_num_lookups; | |
| 358 for (unsigned int i = j + 1; i < lookups.length; i++) | |
| 359 if (lookups.arrayZ[i].index != lookups.arrayZ[j].index) | |
| 360 lookups.arrayZ[++j] = lookups.arrayZ[i]; | |
| 361 else | |
| 362 { | |
| 363 lookups.arrayZ[j].mask |= lookups.arrayZ[i].mask; | |
| 364 lookups.arrayZ[j].auto_zwnj &= lookups.arrayZ[i].auto_zwnj; | |
| 365 lookups.arrayZ[j].auto_zwj &= lookups.arrayZ[i].auto_zwj; | |
| 366 } | |
| 367 lookups.shrink (j + 1); | |
| 368 } | |
| 369 | |
| 370 last_num_lookups = lookups.length; | |
| 371 | |
| 372 if (stage_index < stages[table_index].length && stages[table_index][stage_index].index == stage) { | |
| 373 hb_ot_map_t::stage_map_t *stage_map = m.stages[table_index].push (); | |
| 374 stage_map->last_lookup = last_num_lookups; | |
| 375 stage_map->pause_func = stages[table_index][stage_index].pause_func; | |
| 376 | |
| 377 stage_index++; | |
| 378 } | |
| 379 } | |
| 380 } | |
| 381 } | |
| 382 | |
| 383 | |
| 384 #endif |
