[amalgam] Fix most duplicate-id instances in Indic-like shapers

Part of https://github.com/harfbuzz/harfbuzz/issues/1809
This commit is contained in:
Behdad Esfahbod 2019-07-02 14:42:45 -07:00
parent c073233f45
commit d115a9e022
14 changed files with 231 additions and 303 deletions

View File

@ -168,6 +168,17 @@ _hb_next_syllable (hb_buffer_t *buffer, unsigned int start)
return start; return start;
} }
static inline void
_hb_clear_syllables (const hb_ot_shape_plan_t *plan HB_UNUSED,
hb_font_t *font HB_UNUSED,
hb_buffer_t *buffer)
{
hb_glyph_info_t *info = buffer->info;
unsigned int count = buffer->len;
for (unsigned int i = 0; i < count; i++)
info[i].syllable() = 0;
}
/* unicode_props */ /* unicode_props */
@ -551,6 +562,17 @@ _hb_glyph_info_clear_substituted (hb_glyph_info_t *info)
info->glyph_props() &= ~(HB_OT_LAYOUT_GLYPH_PROPS_SUBSTITUTED); info->glyph_props() &= ~(HB_OT_LAYOUT_GLYPH_PROPS_SUBSTITUTED);
} }
static inline void
_hb_clear_substitution_flags (const hb_ot_shape_plan_t *plan HB_UNUSED,
hb_font_t *font HB_UNUSED,
hb_buffer_t *buffer)
{
hb_glyph_info_t *info = buffer->info;
unsigned int count = buffer->len;
for (unsigned int i = 0; i < count; i++)
_hb_glyph_info_clear_substituted (&info[i]);
}
/* Allocation / deallocation. */ /* Allocation / deallocation. */

View File

@ -395,13 +395,13 @@ static const int indic_syllable_machine_en_main = 39;
HB_STMT_START { \ HB_STMT_START { \
if (0) fprintf (stderr, "syllable %d..%d %s\n", ts, te, #syllable_type); \ if (0) fprintf (stderr, "syllable %d..%d %s\n", ts, te, #syllable_type); \
for (unsigned int i = ts; i < te; i++) \ for (unsigned int i = ts; i < te; i++) \
info[i].syllable() = (syllable_serial << 4) | syllable_type; \ info[i].syllable() = (syllable_serial << 4) | indic_##syllable_type; \
syllable_serial++; \ syllable_serial++; \
if (unlikely (syllable_serial == 16)) syllable_serial = 1; \ if (unlikely (syllable_serial == 16)) syllable_serial = 1; \
} HB_STMT_END } HB_STMT_END
static void static void
find_syllables (hb_buffer_t *buffer) find_syllables_indic (hb_buffer_t *buffer)
{ {
unsigned int p, pe, eof, ts, te, act; unsigned int p, pe, eof, ts, te, act;
int cs; int cs;
@ -569,4 +569,6 @@ _again:
} }
#undef found_syllable
#endif /* HB_OT_SHAPE_COMPLEX_INDIC_MACHINE_HH */ #endif /* HB_OT_SHAPE_COMPLEX_INDIC_MACHINE_HH */

View File

@ -96,13 +96,13 @@ main := |*
HB_STMT_START { \ HB_STMT_START { \
if (0) fprintf (stderr, "syllable %d..%d %s\n", ts, te, #syllable_type); \ if (0) fprintf (stderr, "syllable %d..%d %s\n", ts, te, #syllable_type); \
for (unsigned int i = ts; i < te; i++) \ for (unsigned int i = ts; i < te; i++) \
info[i].syllable() = (syllable_serial << 4) | syllable_type; \ info[i].syllable() = (syllable_serial << 4) | indic_##syllable_type; \
syllable_serial++; \ syllable_serial++; \
if (unlikely (syllable_serial == 16)) syllable_serial = 1; \ if (unlikely (syllable_serial == 16)) syllable_serial = 1; \
} HB_STMT_END } HB_STMT_END
static void static void
find_syllables (hb_buffer_t *buffer) find_syllables_indic (hb_buffer_t *buffer)
{ {
unsigned int p, pe, eof, ts, te, act; unsigned int p, pe, eof, ts, te, act;
int cs; int cs;
@ -121,4 +121,6 @@ find_syllables (hb_buffer_t *buffer)
}%% }%%
} }
#undef found_syllable
#endif /* HB_OT_SHAPE_COMPLEX_INDIC_MACHINE_HH */ #endif /* HB_OT_SHAPE_COMPLEX_INDIC_MACHINE_HH */

View File

@ -172,19 +172,15 @@ enum {
}; };
static void static void
setup_syllables (const hb_ot_shape_plan_t *plan, setup_syllables_indic (const hb_ot_shape_plan_t *plan,
hb_font_t *font, hb_font_t *font,
hb_buffer_t *buffer); hb_buffer_t *buffer);
static void static void
initial_reordering (const hb_ot_shape_plan_t *plan, initial_reordering_indic (const hb_ot_shape_plan_t *plan,
hb_font_t *font, hb_font_t *font,
hb_buffer_t *buffer); hb_buffer_t *buffer);
static void static void
final_reordering (const hb_ot_shape_plan_t *plan, final_reordering_indic (const hb_ot_shape_plan_t *plan,
hb_font_t *font,
hb_buffer_t *buffer);
static void
clear_syllables (const hb_ot_shape_plan_t *plan,
hb_font_t *font, hb_font_t *font,
hb_buffer_t *buffer); hb_buffer_t *buffer);
@ -194,7 +190,7 @@ collect_features_indic (hb_ot_shape_planner_t *plan)
hb_ot_map_builder_t *map = &plan->map; hb_ot_map_builder_t *map = &plan->map;
/* Do this before any lookups have been applied. */ /* Do this before any lookups have been applied. */
map->add_gsub_pause (setup_syllables); map->add_gsub_pause (setup_syllables_indic);
map->enable_feature (HB_TAG('l','o','c','l')); map->enable_feature (HB_TAG('l','o','c','l'));
/* The Indic specs do not require ccmp, but we apply it here since if /* The Indic specs do not require ccmp, but we apply it here since if
@ -203,14 +199,14 @@ collect_features_indic (hb_ot_shape_planner_t *plan)
unsigned int i = 0; unsigned int i = 0;
map->add_gsub_pause (initial_reordering); map->add_gsub_pause (initial_reordering_indic);
for (; i < INDIC_BASIC_FEATURES; i++) { for (; i < INDIC_BASIC_FEATURES; i++) {
map->add_feature (indic_features[i]); map->add_feature (indic_features[i]);
map->add_gsub_pause (nullptr); map->add_gsub_pause (nullptr);
} }
map->add_gsub_pause (final_reordering); map->add_gsub_pause (final_reordering_indic);
for (; i < INDIC_NUM_FEATURES; i++) for (; i < INDIC_NUM_FEATURES; i++)
map->add_feature (indic_features[i]); map->add_feature (indic_features[i]);
@ -218,7 +214,7 @@ collect_features_indic (hb_ot_shape_planner_t *plan)
map->enable_feature (HB_TAG('c','a','l','t')); map->enable_feature (HB_TAG('c','a','l','t'));
map->enable_feature (HB_TAG('c','l','i','g')); map->enable_feature (HB_TAG('c','l','i','g'));
map->add_gsub_pause (clear_syllables); map->add_gsub_pause (_hb_clear_syllables);
} }
static void static void
@ -228,32 +224,6 @@ override_features_indic (hb_ot_shape_planner_t *plan)
} }
struct would_substitute_feature_t
{
void init (const hb_ot_map_t *map, hb_tag_t feature_tag, bool zero_context_)
{
zero_context = zero_context_;
map->get_stage_lookups (0/*GSUB*/,
map->get_feature_stage (0/*GSUB*/, feature_tag),
&lookups, &count);
}
bool would_substitute (const hb_codepoint_t *glyphs,
unsigned int glyphs_count,
hb_face_t *face) const
{
for (unsigned int i = 0; i < count; i++)
if (hb_ot_layout_lookup_would_substitute (face, lookups[i].index, glyphs, glyphs_count, zero_context))
return true;
return false;
}
private:
const hb_ot_map_t::lookup_map_t *lookups;
unsigned int count;
bool zero_context;
};
struct indic_shape_plan_t struct indic_shape_plan_t
{ {
bool load_virama_glyph (hb_font_t *font, hb_codepoint_t *pglyph) const bool load_virama_glyph (hb_font_t *font, hb_codepoint_t *pglyph) const
@ -285,10 +255,10 @@ struct indic_shape_plan_t
#endif #endif
mutable hb_atomic_int_t virama_glyph; mutable hb_atomic_int_t virama_glyph;
would_substitute_feature_t rphf; hb_indic_would_substitute_feature_t rphf;
would_substitute_feature_t pref; hb_indic_would_substitute_feature_t pref;
would_substitute_feature_t blwf; hb_indic_would_substitute_feature_t blwf;
would_substitute_feature_t pstf; hb_indic_would_substitute_feature_t pstf;
hb_mask_t mask_array[INDIC_NUM_FEATURES]; hb_mask_t mask_array[INDIC_NUM_FEATURES];
}; };
@ -371,13 +341,13 @@ consonant_position_from_face (const indic_shape_plan_t *indic_plan,
} }
enum syllable_type_t { enum indic_syllable_type_t {
consonant_syllable, indic_consonant_syllable,
vowel_syllable, indic_vowel_syllable,
standalone_cluster, indic_standalone_cluster,
symbol_cluster, indic_symbol_cluster,
broken_cluster, indic_broken_cluster,
non_indic_cluster, indic_non_indic_cluster,
}; };
#include "hb-ot-shape-complex-indic-machine.hh" #include "hb-ot-shape-complex-indic-machine.hh"
@ -401,11 +371,11 @@ setup_masks_indic (const hb_ot_shape_plan_t *plan HB_UNUSED,
} }
static void static void
setup_syllables (const hb_ot_shape_plan_t *plan HB_UNUSED, setup_syllables_indic (const hb_ot_shape_plan_t *plan HB_UNUSED,
hb_font_t *font HB_UNUSED, hb_font_t *font HB_UNUSED,
hb_buffer_t *buffer) hb_buffer_t *buffer)
{ {
find_syllables (buffer); find_syllables_indic (buffer);
foreach_syllable (buffer, start, end) foreach_syllable (buffer, start, end)
buffer->unsafe_to_break (start, end); buffer->unsafe_to_break (start, end);
} }
@ -950,21 +920,21 @@ initial_reordering_syllable (const hb_ot_shape_plan_t *plan,
hb_buffer_t *buffer, hb_buffer_t *buffer,
unsigned int start, unsigned int end) unsigned int start, unsigned int end)
{ {
syllable_type_t syllable_type = (syllable_type_t) (buffer->info[start].syllable() & 0x0F); indic_syllable_type_t syllable_type = (indic_syllable_type_t) (buffer->info[start].syllable() & 0x0F);
switch (syllable_type) switch (syllable_type)
{ {
case vowel_syllable: /* We made the vowels look like consonants. So let's call the consonant logic! */ case indic_vowel_syllable: /* We made the vowels look like consonants. So let's call the consonant logic! */
case consonant_syllable: case indic_consonant_syllable:
initial_reordering_consonant_syllable (plan, face, buffer, start, end); initial_reordering_consonant_syllable (plan, face, buffer, start, end);
break; break;
case broken_cluster: /* We already inserted dotted-circles, so just call the standalone_cluster. */ case indic_broken_cluster: /* We already inserted dotted-circles, so just call the standalone_cluster. */
case standalone_cluster: case indic_standalone_cluster:
initial_reordering_standalone_cluster (plan, face, buffer, start, end); initial_reordering_standalone_cluster (plan, face, buffer, start, end);
break; break;
case symbol_cluster: case indic_symbol_cluster:
case non_indic_cluster: case indic_non_indic_cluster:
break; break;
} }
} }
@ -983,7 +953,7 @@ insert_dotted_circles (const hb_ot_shape_plan_t *plan HB_UNUSED,
unsigned int count = buffer->len; unsigned int count = buffer->len;
hb_glyph_info_t *info = buffer->info; hb_glyph_info_t *info = buffer->info;
for (unsigned int i = 0; i < count; i++) for (unsigned int i = 0; i < count; i++)
if ((info[i].syllable() & 0x0F) == broken_cluster) if ((info[i].syllable() & 0x0F) == indic_broken_cluster)
{ {
has_broken_syllables = true; has_broken_syllables = true;
break; break;
@ -1008,8 +978,8 @@ insert_dotted_circles (const hb_ot_shape_plan_t *plan HB_UNUSED,
while (buffer->idx < buffer->len && buffer->successful) while (buffer->idx < buffer->len && buffer->successful)
{ {
unsigned int syllable = buffer->cur().syllable(); unsigned int syllable = buffer->cur().syllable();
syllable_type_t syllable_type = (syllable_type_t) (syllable & 0x0F); indic_syllable_type_t syllable_type = (indic_syllable_type_t) (syllable & 0x0F);
if (unlikely (last_syllable != syllable && syllable_type == broken_cluster)) if (unlikely (last_syllable != syllable && syllable_type == indic_broken_cluster))
{ {
last_syllable = syllable; last_syllable = syllable;
@ -1033,7 +1003,7 @@ insert_dotted_circles (const hb_ot_shape_plan_t *plan HB_UNUSED,
} }
static void static void
initial_reordering (const hb_ot_shape_plan_t *plan, initial_reordering_indic (const hb_ot_shape_plan_t *plan,
hb_font_t *font, hb_font_t *font,
hb_buffer_t *buffer) hb_buffer_t *buffer)
{ {
@ -1508,7 +1478,7 @@ final_reordering_syllable (const hb_ot_shape_plan_t *plan,
static void static void
final_reordering (const hb_ot_shape_plan_t *plan, final_reordering_indic (const hb_ot_shape_plan_t *plan,
hb_font_t *font HB_UNUSED, hb_font_t *font HB_UNUSED,
hb_buffer_t *buffer) hb_buffer_t *buffer)
{ {
@ -1523,18 +1493,6 @@ final_reordering (const hb_ot_shape_plan_t *plan,
} }
static void
clear_syllables (const hb_ot_shape_plan_t *plan HB_UNUSED,
hb_font_t *font HB_UNUSED,
hb_buffer_t *buffer)
{
hb_glyph_info_t *info = buffer->info;
unsigned int count = buffer->len;
for (unsigned int i = 0; i < count; i++)
info[i].syllable() = 0;
}
static void static void
preprocess_text_indic (const hb_ot_shape_plan_t *plan, preprocess_text_indic (const hb_ot_shape_plan_t *plan,
hb_buffer_t *buffer, hb_buffer_t *buffer,

View File

@ -398,5 +398,31 @@ set_indic_properties (hb_glyph_info_t &info)
info.indic_position() = pos; info.indic_position() = pos;
} }
struct hb_indic_would_substitute_feature_t
{
void init (const hb_ot_map_t *map, hb_tag_t feature_tag, bool zero_context_)
{
zero_context = zero_context_;
map->get_stage_lookups (0/*GSUB*/,
map->get_feature_stage (0/*GSUB*/, feature_tag),
&lookups, &count);
}
bool would_substitute (const hb_codepoint_t *glyphs,
unsigned int glyphs_count,
hb_face_t *face) const
{
for (unsigned int i = 0; i < count; i++)
if (hb_ot_layout_lookup_would_substitute (face, lookups[i].index, glyphs, glyphs_count, zero_context))
return true;
return false;
}
private:
const hb_ot_map_t::lookup_map_t *lookups;
unsigned int count;
bool zero_context;
};
#endif /* HB_OT_SHAPE_COMPLEX_INDIC_HH */ #endif /* HB_OT_SHAPE_COMPLEX_INDIC_HH */

View File

@ -226,13 +226,13 @@ static const int khmer_syllable_machine_en_main = 20;
HB_STMT_START { \ HB_STMT_START { \
if (0) fprintf (stderr, "syllable %d..%d %s\n", ts, te, #syllable_type); \ if (0) fprintf (stderr, "syllable %d..%d %s\n", ts, te, #syllable_type); \
for (unsigned int i = ts; i < te; i++) \ for (unsigned int i = ts; i < te; i++) \
info[i].syllable() = (syllable_serial << 4) | syllable_type; \ info[i].syllable() = (syllable_serial << 4) | khmer_##syllable_type; \
syllable_serial++; \ syllable_serial++; \
if (unlikely (syllable_serial == 16)) syllable_serial = 1; \ if (unlikely (syllable_serial == 16)) syllable_serial = 1; \
} HB_STMT_END } HB_STMT_END
static void static void
find_syllables (hb_buffer_t *buffer) find_syllables_khmer (hb_buffer_t *buffer)
{ {
unsigned int p, pe, eof, ts, te, act HB_UNUSED; unsigned int p, pe, eof, ts, te, act HB_UNUSED;
int cs; int cs;
@ -367,4 +367,6 @@ _again:
} }
#undef found_syllable
#endif /* HB_OT_SHAPE_COMPLEX_KHMER_MACHINE_HH */ #endif /* HB_OT_SHAPE_COMPLEX_KHMER_MACHINE_HH */

View File

@ -83,13 +83,13 @@ main := |*
HB_STMT_START { \ HB_STMT_START { \
if (0) fprintf (stderr, "syllable %d..%d %s\n", ts, te, #syllable_type); \ if (0) fprintf (stderr, "syllable %d..%d %s\n", ts, te, #syllable_type); \
for (unsigned int i = ts; i < te; i++) \ for (unsigned int i = ts; i < te; i++) \
info[i].syllable() = (syllable_serial << 4) | syllable_type; \ info[i].syllable() = (syllable_serial << 4) | khmer_##syllable_type; \
syllable_serial++; \ syllable_serial++; \
if (unlikely (syllable_serial == 16)) syllable_serial = 1; \ if (unlikely (syllable_serial == 16)) syllable_serial = 1; \
} HB_STMT_END } HB_STMT_END
static void static void
find_syllables (hb_buffer_t *buffer) find_syllables_khmer (hb_buffer_t *buffer)
{ {
unsigned int p, pe, eof, ts, te, act HB_UNUSED; unsigned int p, pe, eof, ts, te, act HB_UNUSED;
int cs; int cs;
@ -108,4 +108,6 @@ find_syllables (hb_buffer_t *buffer)
}%% }%%
} }
#undef found_syllable
#endif /* HB_OT_SHAPE_COMPLEX_KHMER_MACHINE_HH */ #endif /* HB_OT_SHAPE_COMPLEX_KHMER_MACHINE_HH */

View File

@ -89,15 +89,11 @@ enum {
}; };
static void static void
setup_syllables (const hb_ot_shape_plan_t *plan, setup_syllables_khmer (const hb_ot_shape_plan_t *plan,
hb_font_t *font, hb_font_t *font,
hb_buffer_t *buffer); hb_buffer_t *buffer);
static void static void
reorder (const hb_ot_shape_plan_t *plan, reorder_khmer (const hb_ot_shape_plan_t *plan,
hb_font_t *font,
hb_buffer_t *buffer);
static void
clear_syllables (const hb_ot_shape_plan_t *plan,
hb_font_t *font, hb_font_t *font,
hb_buffer_t *buffer); hb_buffer_t *buffer);
@ -107,8 +103,8 @@ collect_features_khmer (hb_ot_shape_planner_t *plan)
hb_ot_map_builder_t *map = &plan->map; hb_ot_map_builder_t *map = &plan->map;
/* Do this before any lookups have been applied. */ /* Do this before any lookups have been applied. */
map->add_gsub_pause (setup_syllables); map->add_gsub_pause (setup_syllables_khmer);
map->add_gsub_pause (reorder); map->add_gsub_pause (reorder_khmer);
/* Testing suggests that Uniscribe does NOT pause between basic /* Testing suggests that Uniscribe does NOT pause between basic
* features. Test with KhmerUI.ttf and the following three * features. Test with KhmerUI.ttf and the following three
@ -127,7 +123,7 @@ collect_features_khmer (hb_ot_shape_planner_t *plan)
for (; i < KHMER_BASIC_FEATURES; i++) for (; i < KHMER_BASIC_FEATURES; i++)
map->add_feature (khmer_features[i]); map->add_feature (khmer_features[i]);
map->add_gsub_pause (clear_syllables); map->add_gsub_pause (_hb_clear_syllables);
for (; i < KHMER_NUM_FEATURES; i++) for (; i < KHMER_NUM_FEATURES; i++)
map->add_feature (khmer_features[i]); map->add_feature (khmer_features[i]);
@ -153,32 +149,6 @@ override_features_khmer (hb_ot_shape_planner_t *plan)
} }
struct would_substitute_feature_t
{
void init (const hb_ot_map_t *map, hb_tag_t feature_tag, bool zero_context_)
{
zero_context = zero_context_;
map->get_stage_lookups (0/*GSUB*/,
map->get_feature_stage (0/*GSUB*/, feature_tag),
&lookups, &count);
}
bool would_substitute (const hb_codepoint_t *glyphs,
unsigned int glyphs_count,
hb_face_t *face) const
{
for (unsigned int i = 0; i < count; i++)
if (hb_ot_layout_lookup_would_substitute (face, lookups[i].index, glyphs, glyphs_count, zero_context))
return true;
return false;
}
private:
const hb_ot_map_t::lookup_map_t *lookups;
unsigned int count;
bool zero_context;
};
struct khmer_shape_plan_t struct khmer_shape_plan_t
{ {
bool get_virama_glyph (hb_font_t *font, hb_codepoint_t *pglyph) const bool get_virama_glyph (hb_font_t *font, hb_codepoint_t *pglyph) const
@ -202,7 +172,7 @@ struct khmer_shape_plan_t
mutable hb_codepoint_t virama_glyph; mutable hb_codepoint_t virama_glyph;
would_substitute_feature_t pref; hb_indic_would_substitute_feature_t pref;
hb_mask_t mask_array[KHMER_NUM_FEATURES]; hb_mask_t mask_array[KHMER_NUM_FEATURES];
}; };
@ -232,10 +202,10 @@ data_destroy_khmer (void *data)
} }
enum syllable_type_t { enum khmer_syllable_type_t {
consonant_syllable, khmer_consonant_syllable,
broken_cluster, khmer_broken_cluster,
non_khmer_cluster, khmer_non_khmer_cluster,
}; };
#include "hb-ot-shape-complex-khmer-machine.hh" #include "hb-ot-shape-complex-khmer-machine.hh"
@ -257,11 +227,11 @@ setup_masks_khmer (const hb_ot_shape_plan_t *plan HB_UNUSED,
} }
static void static void
setup_syllables (const hb_ot_shape_plan_t *plan HB_UNUSED, setup_syllables_khmer (const hb_ot_shape_plan_t *plan HB_UNUSED,
hb_font_t *font HB_UNUSED, hb_font_t *font HB_UNUSED,
hb_buffer_t *buffer) hb_buffer_t *buffer)
{ {
find_syllables (buffer); find_syllables_khmer (buffer);
foreach_syllable (buffer, start, end) foreach_syllable (buffer, start, end)
buffer->unsafe_to_break (start, end); buffer->unsafe_to_break (start, end);
} }
@ -351,15 +321,15 @@ initial_reordering_syllable (const hb_ot_shape_plan_t *plan,
hb_buffer_t *buffer, hb_buffer_t *buffer,
unsigned int start, unsigned int end) unsigned int start, unsigned int end)
{ {
syllable_type_t syllable_type = (syllable_type_t) (buffer->info[start].syllable() & 0x0F); khmer_syllable_type_t syllable_type = (khmer_syllable_type_t) (buffer->info[start].syllable() & 0x0F);
switch (syllable_type) switch (syllable_type)
{ {
case broken_cluster: /* We already inserted dotted-circles, so just call the consonant_syllable. */ case khmer_broken_cluster: /* We already inserted dotted-circles, so just call the consonant_syllable. */
case consonant_syllable: case khmer_consonant_syllable:
reorder_consonant_syllable (plan, face, buffer, start, end); reorder_consonant_syllable (plan, face, buffer, start, end);
break; break;
case non_khmer_cluster: case khmer_non_khmer_cluster:
break; break;
} }
} }
@ -378,7 +348,7 @@ insert_dotted_circles (const hb_ot_shape_plan_t *plan HB_UNUSED,
unsigned int count = buffer->len; unsigned int count = buffer->len;
hb_glyph_info_t *info = buffer->info; hb_glyph_info_t *info = buffer->info;
for (unsigned int i = 0; i < count; i++) for (unsigned int i = 0; i < count; i++)
if ((info[i].syllable() & 0x0F) == broken_cluster) if ((info[i].syllable() & 0x0F) == khmer_broken_cluster)
{ {
has_broken_syllables = true; has_broken_syllables = true;
break; break;
@ -403,8 +373,8 @@ insert_dotted_circles (const hb_ot_shape_plan_t *plan HB_UNUSED,
while (buffer->idx < buffer->len && buffer->successful) while (buffer->idx < buffer->len && buffer->successful)
{ {
unsigned int syllable = buffer->cur().syllable(); unsigned int syllable = buffer->cur().syllable();
syllable_type_t syllable_type = (syllable_type_t) (syllable & 0x0F); khmer_syllable_type_t syllable_type = (khmer_syllable_type_t) (syllable & 0x0F);
if (unlikely (last_syllable != syllable && syllable_type == broken_cluster)) if (unlikely (last_syllable != syllable && syllable_type == khmer_broken_cluster))
{ {
last_syllable = syllable; last_syllable = syllable;
@ -428,7 +398,7 @@ insert_dotted_circles (const hb_ot_shape_plan_t *plan HB_UNUSED,
} }
static void static void
reorder (const hb_ot_shape_plan_t *plan, reorder_khmer (const hb_ot_shape_plan_t *plan,
hb_font_t *font, hb_font_t *font,
hb_buffer_t *buffer) hb_buffer_t *buffer)
{ {
@ -440,17 +410,6 @@ reorder (const hb_ot_shape_plan_t *plan,
HB_BUFFER_DEALLOCATE_VAR (buffer, khmer_category); HB_BUFFER_DEALLOCATE_VAR (buffer, khmer_category);
} }
static void
clear_syllables (const hb_ot_shape_plan_t *plan HB_UNUSED,
hb_font_t *font HB_UNUSED,
hb_buffer_t *buffer)
{
hb_glyph_info_t *info = buffer->info;
unsigned int count = buffer->len;
for (unsigned int i = 0; i < count; i++)
info[i].syllable() = 0;
}
static bool static bool
decompose_khmer (const hb_ot_shape_normalize_context_t *c, decompose_khmer (const hb_ot_shape_normalize_context_t *c,

View File

@ -304,13 +304,13 @@ static const int myanmar_syllable_machine_en_main = 0;
HB_STMT_START { \ HB_STMT_START { \
if (0) fprintf (stderr, "syllable %d..%d %s\n", ts, te, #syllable_type); \ if (0) fprintf (stderr, "syllable %d..%d %s\n", ts, te, #syllable_type); \
for (unsigned int i = ts; i < te; i++) \ for (unsigned int i = ts; i < te; i++) \
info[i].syllable() = (syllable_serial << 4) | syllable_type; \ info[i].syllable() = (syllable_serial << 4) | myanmar_##syllable_type; \
syllable_serial++; \ syllable_serial++; \
if (unlikely (syllable_serial == 16)) syllable_serial = 1; \ if (unlikely (syllable_serial == 16)) syllable_serial = 1; \
} HB_STMT_END } HB_STMT_END
static void static void
find_syllables (hb_buffer_t *buffer) find_syllables_myanmar (hb_buffer_t *buffer)
{ {
unsigned int p, pe, eof, ts, te, act HB_UNUSED; unsigned int p, pe, eof, ts, te, act HB_UNUSED;
int cs; int cs;

View File

@ -97,13 +97,13 @@ main := |*
HB_STMT_START { \ HB_STMT_START { \
if (0) fprintf (stderr, "syllable %d..%d %s\n", ts, te, #syllable_type); \ if (0) fprintf (stderr, "syllable %d..%d %s\n", ts, te, #syllable_type); \
for (unsigned int i = ts; i < te; i++) \ for (unsigned int i = ts; i < te; i++) \
info[i].syllable() = (syllable_serial << 4) | syllable_type; \ info[i].syllable() = (syllable_serial << 4) | myanmar_##syllable_type; \
syllable_serial++; \ syllable_serial++; \
if (unlikely (syllable_serial == 16)) syllable_serial = 1; \ if (unlikely (syllable_serial == 16)) syllable_serial = 1; \
} HB_STMT_END } HB_STMT_END
static void static void
find_syllables (hb_buffer_t *buffer) find_syllables_myanmar (hb_buffer_t *buffer)
{ {
unsigned int p, pe, eof, ts, te, act HB_UNUSED; unsigned int p, pe, eof, ts, te, act HB_UNUSED;
int cs; int cs;

View File

@ -80,15 +80,11 @@ positioning_features[] =
}; };
static void static void
setup_syllables (const hb_ot_shape_plan_t *plan, setup_syllables_myanmar (const hb_ot_shape_plan_t *plan,
hb_font_t *font, hb_font_t *font,
hb_buffer_t *buffer); hb_buffer_t *buffer);
static void static void
reorder (const hb_ot_shape_plan_t *plan, reorder_myanmar (const hb_ot_shape_plan_t *plan,
hb_font_t *font,
hb_buffer_t *buffer);
static void
clear_syllables (const hb_ot_shape_plan_t *plan,
hb_font_t *font, hb_font_t *font,
hb_buffer_t *buffer); hb_buffer_t *buffer);
@ -98,7 +94,7 @@ collect_features_myanmar (hb_ot_shape_planner_t *plan)
hb_ot_map_builder_t *map = &plan->map; hb_ot_map_builder_t *map = &plan->map;
/* Do this before any lookups have been applied. */ /* Do this before any lookups have been applied. */
map->add_gsub_pause (setup_syllables); map->add_gsub_pause (setup_syllables_myanmar);
map->enable_feature (HB_TAG('l','o','c','l')); map->enable_feature (HB_TAG('l','o','c','l'));
/* The Indic specs do not require ccmp, but we apply it here since if /* The Indic specs do not require ccmp, but we apply it here since if
@ -106,7 +102,7 @@ collect_features_myanmar (hb_ot_shape_planner_t *plan)
map->enable_feature (HB_TAG('c','c','m','p')); map->enable_feature (HB_TAG('c','c','m','p'));
map->add_gsub_pause (reorder); map->add_gsub_pause (reorder_myanmar);
for (unsigned int i = 0; i < ARRAY_LENGTH (basic_features); i++) for (unsigned int i = 0; i < ARRAY_LENGTH (basic_features); i++)
{ {
@ -114,7 +110,7 @@ collect_features_myanmar (hb_ot_shape_planner_t *plan)
map->add_gsub_pause (nullptr); map->add_gsub_pause (nullptr);
} }
map->add_gsub_pause (clear_syllables); map->add_gsub_pause (_hb_clear_syllables);
for (unsigned int i = 0; i < ARRAY_LENGTH (other_features); i++) for (unsigned int i = 0; i < ARRAY_LENGTH (other_features); i++)
map->enable_feature (other_features[i], F_MANUAL_ZWJ); map->enable_feature (other_features[i], F_MANUAL_ZWJ);
@ -130,11 +126,11 @@ override_features_myanmar (hb_ot_shape_planner_t *plan)
} }
enum syllable_type_t { enum myanmar_syllable_type_t {
consonant_syllable, myanmar_consonant_syllable,
punctuation_cluster, myanmar_punctuation_cluster,
broken_cluster, myanmar_broken_cluster,
non_myanmar_cluster, myanmar_non_myanmar_cluster,
}; };
#include "hb-ot-shape-complex-myanmar-machine.hh" #include "hb-ot-shape-complex-myanmar-machine.hh"
@ -158,11 +154,11 @@ setup_masks_myanmar (const hb_ot_shape_plan_t *plan HB_UNUSED,
} }
static void static void
setup_syllables (const hb_ot_shape_plan_t *plan HB_UNUSED, setup_syllables_myanmar (const hb_ot_shape_plan_t *plan HB_UNUSED,
hb_font_t *font HB_UNUSED, hb_font_t *font HB_UNUSED,
hb_buffer_t *buffer) hb_buffer_t *buffer)
{ {
find_syllables (buffer); find_syllables_myanmar (buffer);
foreach_syllable (buffer, start, end) foreach_syllable (buffer, start, end)
buffer->unsafe_to_break (start, end); buffer->unsafe_to_break (start, end);
} }
@ -283,16 +279,16 @@ initial_reordering_syllable (const hb_ot_shape_plan_t *plan HB_UNUSED,
hb_buffer_t *buffer, hb_buffer_t *buffer,
unsigned int start, unsigned int end) unsigned int start, unsigned int end)
{ {
syllable_type_t syllable_type = (syllable_type_t) (buffer->info[start].syllable() & 0x0F); myanmar_syllable_type_t syllable_type = (myanmar_syllable_type_t) (buffer->info[start].syllable() & 0x0F);
switch (syllable_type) { switch (syllable_type) {
case broken_cluster: /* We already inserted dotted-circles, so just call the consonant_syllable. */ case myanmar_broken_cluster: /* We already inserted dotted-circles, so just call the consonant_syllable. */
case consonant_syllable: case myanmar_consonant_syllable:
initial_reordering_consonant_syllable (buffer, start, end); initial_reordering_consonant_syllable (buffer, start, end);
break; break;
case punctuation_cluster: case myanmar_punctuation_cluster:
case non_myanmar_cluster: case myanmar_non_myanmar_cluster:
break; break;
} }
} }
@ -311,7 +307,7 @@ insert_dotted_circles (const hb_ot_shape_plan_t *plan HB_UNUSED,
unsigned int count = buffer->len; unsigned int count = buffer->len;
hb_glyph_info_t *info = buffer->info; hb_glyph_info_t *info = buffer->info;
for (unsigned int i = 0; i < count; i++) for (unsigned int i = 0; i < count; i++)
if ((info[i].syllable() & 0x0F) == broken_cluster) if ((info[i].syllable() & 0x0F) == myanmar_broken_cluster)
{ {
has_broken_syllables = true; has_broken_syllables = true;
break; break;
@ -336,8 +332,8 @@ insert_dotted_circles (const hb_ot_shape_plan_t *plan HB_UNUSED,
while (buffer->idx < buffer->len && buffer->successful) while (buffer->idx < buffer->len && buffer->successful)
{ {
unsigned int syllable = buffer->cur().syllable(); unsigned int syllable = buffer->cur().syllable();
syllable_type_t syllable_type = (syllable_type_t) (syllable & 0x0F); myanmar_syllable_type_t syllable_type = (myanmar_syllable_type_t) (syllable & 0x0F);
if (unlikely (last_syllable != syllable && syllable_type == broken_cluster)) if (unlikely (last_syllable != syllable && syllable_type == myanmar_broken_cluster))
{ {
last_syllable = syllable; last_syllable = syllable;
@ -355,7 +351,7 @@ insert_dotted_circles (const hb_ot_shape_plan_t *plan HB_UNUSED,
} }
static void static void
reorder (const hb_ot_shape_plan_t *plan, reorder_myanmar (const hb_ot_shape_plan_t *plan,
hb_font_t *font, hb_font_t *font,
hb_buffer_t *buffer) hb_buffer_t *buffer)
{ {
@ -368,17 +364,6 @@ reorder (const hb_ot_shape_plan_t *plan,
HB_BUFFER_DEALLOCATE_VAR (buffer, myanmar_position); HB_BUFFER_DEALLOCATE_VAR (buffer, myanmar_position);
} }
static void
clear_syllables (const hb_ot_shape_plan_t *plan HB_UNUSED,
hb_font_t *font HB_UNUSED,
hb_buffer_t *buffer)
{
hb_glyph_info_t *info = buffer->info;
unsigned int count = buffer->len;
for (unsigned int i = 0; i < count; i++)
info[i].syllable() = 0;
}
const hb_ot_complex_shaper_t _hb_ot_complex_shaper_myanmar = const hb_ot_complex_shaper_t _hb_ot_complex_shaper_myanmar =
{ {

View File

@ -380,13 +380,13 @@ static const int use_syllable_machine_en_main = 5;
HB_STMT_START { \ HB_STMT_START { \
if (0) fprintf (stderr, "syllable %d..%d %s\n", ts, te, #syllable_type); \ if (0) fprintf (stderr, "syllable %d..%d %s\n", ts, te, #syllable_type); \
for (unsigned int i = ts; i < te; i++) \ for (unsigned int i = ts; i < te; i++) \
info[i].syllable() = (syllable_serial << 4) | syllable_type; \ info[i].syllable() = (syllable_serial << 4) | use_##syllable_type; \
syllable_serial++; \ syllable_serial++; \
if (unlikely (syllable_serial == 16)) syllable_serial = 1; \ if (unlikely (syllable_serial == 16)) syllable_serial = 1; \
} HB_STMT_END } HB_STMT_END
static void static void
find_syllables (hb_buffer_t *buffer) find_syllables_use (hb_buffer_t *buffer)
{ {
unsigned int p, pe, eof, ts, te, act; unsigned int p, pe, eof, ts, te, act;
int cs; int cs;

View File

@ -165,13 +165,13 @@ main := |*
HB_STMT_START { \ HB_STMT_START { \
if (0) fprintf (stderr, "syllable %d..%d %s\n", ts, te, #syllable_type); \ if (0) fprintf (stderr, "syllable %d..%d %s\n", ts, te, #syllable_type); \
for (unsigned int i = ts; i < te; i++) \ for (unsigned int i = ts; i < te; i++) \
info[i].syllable() = (syllable_serial << 4) | syllable_type; \ info[i].syllable() = (syllable_serial << 4) | use_##syllable_type; \
syllable_serial++; \ syllable_serial++; \
if (unlikely (syllable_serial == 16)) syllable_serial = 1; \ if (unlikely (syllable_serial == 16)) syllable_serial = 1; \
} HB_STMT_END } HB_STMT_END
static void static void
find_syllables (hb_buffer_t *buffer) find_syllables_use (hb_buffer_t *buffer)
{ {
unsigned int p, pe, eof, ts, te, act; unsigned int p, pe, eof, ts, te, act;
int cs; int cs;

View File

@ -106,27 +106,19 @@ positioning_features[] =
}; };
static void static void
setup_syllables (const hb_ot_shape_plan_t *plan, setup_syllables_use (const hb_ot_shape_plan_t *plan,
hb_font_t *font, hb_font_t *font,
hb_buffer_t *buffer); hb_buffer_t *buffer);
static void static void
clear_substitution_flags (const hb_ot_shape_plan_t *plan, record_rphf_use (const hb_ot_shape_plan_t *plan,
hb_font_t *font, hb_font_t *font,
hb_buffer_t *buffer); hb_buffer_t *buffer);
static void static void
record_rphf (const hb_ot_shape_plan_t *plan, record_pref_use (const hb_ot_shape_plan_t *plan,
hb_font_t *font, hb_font_t *font,
hb_buffer_t *buffer); hb_buffer_t *buffer);
static void static void
record_pref (const hb_ot_shape_plan_t *plan, reorder_use (const hb_ot_shape_plan_t *plan,
hb_font_t *font,
hb_buffer_t *buffer);
static void
reorder (const hb_ot_shape_plan_t *plan,
hb_font_t *font,
hb_buffer_t *buffer);
static void
clear_syllables (const hb_ot_shape_plan_t *plan,
hb_font_t *font, hb_font_t *font,
hb_buffer_t *buffer); hb_buffer_t *buffer);
@ -136,7 +128,7 @@ collect_features_use (hb_ot_shape_planner_t *plan)
hb_ot_map_builder_t *map = &plan->map; hb_ot_map_builder_t *map = &plan->map;
/* Do this before any lookups have been applied. */ /* Do this before any lookups have been applied. */
map->add_gsub_pause (setup_syllables); map->add_gsub_pause (setup_syllables_use);
/* "Default glyph pre-processing group" */ /* "Default glyph pre-processing group" */
map->enable_feature (HB_TAG('l','o','c','l')); map->enable_feature (HB_TAG('l','o','c','l'));
@ -145,19 +137,19 @@ collect_features_use (hb_ot_shape_planner_t *plan)
map->enable_feature (HB_TAG('a','k','h','n'), F_MANUAL_ZWJ); map->enable_feature (HB_TAG('a','k','h','n'), F_MANUAL_ZWJ);
/* "Reordering group" */ /* "Reordering group" */
map->add_gsub_pause (clear_substitution_flags); map->add_gsub_pause (_hb_clear_substitution_flags);
map->add_feature (HB_TAG('r','p','h','f'), F_MANUAL_ZWJ); map->add_feature (HB_TAG('r','p','h','f'), F_MANUAL_ZWJ);
map->add_gsub_pause (record_rphf); map->add_gsub_pause (record_rphf_use);
map->add_gsub_pause (clear_substitution_flags); map->add_gsub_pause (_hb_clear_substitution_flags);
map->enable_feature (HB_TAG('p','r','e','f'), F_MANUAL_ZWJ); map->enable_feature (HB_TAG('p','r','e','f'), F_MANUAL_ZWJ);
map->add_gsub_pause (record_pref); map->add_gsub_pause (record_pref_use);
/* "Orthographic unit shaping group" */ /* "Orthographic unit shaping group" */
for (unsigned int i = 0; i < ARRAY_LENGTH (basic_features); i++) for (unsigned int i = 0; i < ARRAY_LENGTH (basic_features); i++)
map->enable_feature (basic_features[i], F_MANUAL_ZWJ); map->enable_feature (basic_features[i], F_MANUAL_ZWJ);
map->add_gsub_pause (reorder); map->add_gsub_pause (reorder_use);
map->add_gsub_pause (clear_syllables); map->add_gsub_pause (_hb_clear_syllables);
/* "Topographical features" */ /* "Topographical features" */
for (unsigned int i = 0; i < ARRAY_LENGTH (arabic_features); i++) for (unsigned int i = 0; i < ARRAY_LENGTH (arabic_features); i++)
@ -247,16 +239,16 @@ data_destroy_use (void *data)
free (data); free (data);
} }
enum syllable_type_t { enum use_syllable_type_t {
independent_cluster, use_independent_cluster,
virama_terminated_cluster, use_virama_terminated_cluster,
sakot_terminated_cluster, use_sakot_terminated_cluster,
standard_cluster, use_standard_cluster,
number_joiner_terminated_cluster, use_number_joiner_terminated_cluster,
numeral_cluster, use_numeral_cluster,
symbol_cluster, use_symbol_cluster,
broken_cluster, use_broken_cluster,
non_cluster, use_non_cluster,
}; };
#include "hb-ot-shape-complex-use-machine.hh" #include "hb-ot-shape-complex-use-machine.hh"
@ -331,22 +323,22 @@ setup_topographical_masks (const hb_ot_shape_plan_t *plan,
hb_glyph_info_t *info = buffer->info; hb_glyph_info_t *info = buffer->info;
foreach_syllable (buffer, start, end) foreach_syllable (buffer, start, end)
{ {
syllable_type_t syllable_type = (syllable_type_t) (info[start].syllable() & 0x0F); use_syllable_type_t syllable_type = (use_syllable_type_t) (info[start].syllable() & 0x0F);
switch (syllable_type) switch (syllable_type)
{ {
case independent_cluster: case use_independent_cluster:
case symbol_cluster: case use_symbol_cluster:
case non_cluster: case use_non_cluster:
/* These don't join. Nothing to do. */ /* These don't join. Nothing to do. */
last_form = _NONE; last_form = _NONE;
break; break;
case virama_terminated_cluster: case use_virama_terminated_cluster:
case sakot_terminated_cluster: case use_sakot_terminated_cluster:
case standard_cluster: case use_standard_cluster:
case number_joiner_terminated_cluster: case use_number_joiner_terminated_cluster:
case numeral_cluster: case use_numeral_cluster:
case broken_cluster: case use_broken_cluster:
bool join = last_form == FINA || last_form == ISOL; bool join = last_form == FINA || last_form == ISOL;
@ -371,11 +363,11 @@ setup_topographical_masks (const hb_ot_shape_plan_t *plan,
} }
static void static void
setup_syllables (const hb_ot_shape_plan_t *plan, setup_syllables_use (const hb_ot_shape_plan_t *plan,
hb_font_t *font HB_UNUSED, hb_font_t *font HB_UNUSED,
hb_buffer_t *buffer) hb_buffer_t *buffer)
{ {
find_syllables (buffer); find_syllables_use (buffer);
foreach_syllable (buffer, start, end) foreach_syllable (buffer, start, end)
buffer->unsafe_to_break (start, end); buffer->unsafe_to_break (start, end);
setup_rphf_mask (plan, buffer); setup_rphf_mask (plan, buffer);
@ -383,18 +375,7 @@ setup_syllables (const hb_ot_shape_plan_t *plan,
} }
static void static void
clear_substitution_flags (const hb_ot_shape_plan_t *plan HB_UNUSED, record_rphf_use (const hb_ot_shape_plan_t *plan,
hb_font_t *font HB_UNUSED,
hb_buffer_t *buffer)
{
hb_glyph_info_t *info = buffer->info;
unsigned int count = buffer->len;
for (unsigned int i = 0; i < count; i++)
_hb_glyph_info_clear_substituted (&info[i]);
}
static void
record_rphf (const hb_ot_shape_plan_t *plan,
hb_font_t *font HB_UNUSED, hb_font_t *font HB_UNUSED,
hb_buffer_t *buffer) hb_buffer_t *buffer)
{ {
@ -417,7 +398,7 @@ record_rphf (const hb_ot_shape_plan_t *plan,
} }
static void static void
record_pref (const hb_ot_shape_plan_t *plan HB_UNUSED, record_pref_use (const hb_ot_shape_plan_t *plan HB_UNUSED,
hb_font_t *font HB_UNUSED, hb_font_t *font HB_UNUSED,
hb_buffer_t *buffer) hb_buffer_t *buffer)
{ {
@ -443,15 +424,15 @@ is_halant (const hb_glyph_info_t &info)
} }
static void static void
reorder_syllable (hb_buffer_t *buffer, unsigned int start, unsigned int end) reorder_syllable_use (hb_buffer_t *buffer, unsigned int start, unsigned int end)
{ {
syllable_type_t syllable_type = (syllable_type_t) (buffer->info[start].syllable() & 0x0F); use_syllable_type_t syllable_type = (use_syllable_type_t) (buffer->info[start].syllable() & 0x0F);
/* Only a few syllable types need reordering. */ /* Only a few syllable types need reordering. */
if (unlikely (!(FLAG_UNSAFE (syllable_type) & if (unlikely (!(FLAG_UNSAFE (syllable_type) &
(FLAG (virama_terminated_cluster) | (FLAG (use_virama_terminated_cluster) |
FLAG (sakot_terminated_cluster) | FLAG (use_sakot_terminated_cluster) |
FLAG (standard_cluster) | FLAG (use_standard_cluster) |
FLAG (broken_cluster) | FLAG (use_broken_cluster) |
0)))) 0))))
return; return;
@ -539,7 +520,7 @@ insert_dotted_circles (const hb_ot_shape_plan_t *plan HB_UNUSED,
unsigned int count = buffer->len; unsigned int count = buffer->len;
hb_glyph_info_t *info = buffer->info; hb_glyph_info_t *info = buffer->info;
for (unsigned int i = 0; i < count; i++) for (unsigned int i = 0; i < count; i++)
if ((info[i].syllable() & 0x0F) == broken_cluster) if ((info[i].syllable() & 0x0F) == use_broken_cluster)
{ {
has_broken_syllables = true; has_broken_syllables = true;
break; break;
@ -559,8 +540,8 @@ insert_dotted_circles (const hb_ot_shape_plan_t *plan HB_UNUSED,
while (buffer->idx < buffer->len && buffer->successful) while (buffer->idx < buffer->len && buffer->successful)
{ {
unsigned int syllable = buffer->cur().syllable(); unsigned int syllable = buffer->cur().syllable();
syllable_type_t syllable_type = (syllable_type_t) (syllable & 0x0F); use_syllable_type_t syllable_type = (use_syllable_type_t) (syllable & 0x0F);
if (unlikely (last_syllable != syllable && syllable_type == broken_cluster)) if (unlikely (last_syllable != syllable && syllable_type == use_broken_cluster))
{ {
last_syllable = syllable; last_syllable = syllable;
@ -584,29 +565,18 @@ insert_dotted_circles (const hb_ot_shape_plan_t *plan HB_UNUSED,
} }
static void static void
reorder (const hb_ot_shape_plan_t *plan, reorder_use (const hb_ot_shape_plan_t *plan,
hb_font_t *font, hb_font_t *font,
hb_buffer_t *buffer) hb_buffer_t *buffer)
{ {
insert_dotted_circles (plan, font, buffer); insert_dotted_circles (plan, font, buffer);
foreach_syllable (buffer, start, end) foreach_syllable (buffer, start, end)
reorder_syllable (buffer, start, end); reorder_syllable_use (buffer, start, end);
HB_BUFFER_DEALLOCATE_VAR (buffer, use_category); HB_BUFFER_DEALLOCATE_VAR (buffer, use_category);
} }
static void
clear_syllables (const hb_ot_shape_plan_t *plan HB_UNUSED,
hb_font_t *font HB_UNUSED,
hb_buffer_t *buffer)
{
hb_glyph_info_t *info = buffer->info;
unsigned int count = buffer->len;
for (unsigned int i = 0; i < count; i++)
info[i].syllable() = 0;
}
static void static void
preprocess_text_use (const hb_ot_shape_plan_t *plan, preprocess_text_use (const hb_ot_shape_plan_t *plan,