[subset] ChainContext subsetting fix: add lookup index remapping
This commit is contained in:
parent
e3af529e51
commit
593e58c842
|
@ -2323,6 +2323,7 @@ struct ChainRule
|
||||||
}
|
}
|
||||||
|
|
||||||
ChainRule* copy (hb_serialize_context_t *c,
|
ChainRule* copy (hb_serialize_context_t *c,
|
||||||
|
const hb_map_t *lookup_map,
|
||||||
const hb_map_t *backtrack_map,
|
const hb_map_t *backtrack_map,
|
||||||
const hb_map_t *input_map = nullptr,
|
const hb_map_t *input_map = nullptr,
|
||||||
const hb_map_t *lookahead_map = nullptr) const
|
const hb_map_t *lookahead_map = nullptr) const
|
||||||
|
@ -2345,13 +2346,19 @@ struct ChainRule
|
||||||
serialize_array (c, lookahead.len, + lookahead.iter ()
|
serialize_array (c, lookahead.len, + lookahead.iter ()
|
||||||
| hb_map (mapping));
|
| hb_map (mapping));
|
||||||
|
|
||||||
const ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord>> (lookahead);
|
const ArrayOf<LookupRecord> &lookupRecord = StructAfter<ArrayOf<LookupRecord>> (lookahead);
|
||||||
c->copy (lookup);
|
HBUINT16 lookupCount;
|
||||||
|
lookupCount = lookupRecord.len;
|
||||||
|
if (!c->copy (lookupCount)) return_trace (nullptr);
|
||||||
|
|
||||||
|
for (unsigned i = 0; i < (unsigned) lookupCount; i++)
|
||||||
|
if (!c->copy (lookupRecord[i], lookup_map)) return_trace (nullptr);
|
||||||
|
|
||||||
return_trace (out);
|
return_trace (out);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool subset (hb_subset_context_t *c,
|
bool subset (hb_subset_context_t *c,
|
||||||
|
const hb_map_t *lookup_map,
|
||||||
const hb_map_t *backtrack_map = nullptr,
|
const hb_map_t *backtrack_map = nullptr,
|
||||||
const hb_map_t *input_map = nullptr,
|
const hb_map_t *input_map = nullptr,
|
||||||
const hb_map_t *lookahead_map = nullptr) const
|
const hb_map_t *lookahead_map = nullptr) const
|
||||||
|
@ -2369,7 +2376,7 @@ struct ChainRule
|
||||||
!hb_all (lookahead, glyphset))
|
!hb_all (lookahead, glyphset))
|
||||||
return_trace (false);
|
return_trace (false);
|
||||||
|
|
||||||
copy (c->serializer, c->plan->glyph_map);
|
copy (c->serializer, lookup_map, c->plan->glyph_map);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -2378,7 +2385,7 @@ struct ChainRule
|
||||||
!hb_all (lookahead, lookahead_map))
|
!hb_all (lookahead, lookahead_map))
|
||||||
return_trace (false);
|
return_trace (false);
|
||||||
|
|
||||||
copy (c->serializer, backtrack_map, input_map, lookahead_map);
|
copy (c->serializer, lookup_map, backtrack_map, input_map, lookahead_map);
|
||||||
}
|
}
|
||||||
|
|
||||||
return_trace (true);
|
return_trace (true);
|
||||||
|
@ -2479,6 +2486,7 @@ struct ChainRuleSet
|
||||||
}
|
}
|
||||||
|
|
||||||
bool subset (hb_subset_context_t *c,
|
bool subset (hb_subset_context_t *c,
|
||||||
|
const hb_map_t *lookup_map,
|
||||||
const hb_map_t *backtrack_klass_map = nullptr,
|
const hb_map_t *backtrack_klass_map = nullptr,
|
||||||
const hb_map_t *input_klass_map = nullptr,
|
const hb_map_t *input_klass_map = nullptr,
|
||||||
const hb_map_t *lookahead_klass_map = nullptr) const
|
const hb_map_t *lookahead_klass_map = nullptr) const
|
||||||
|
@ -2497,6 +2505,7 @@ struct ChainRuleSet
|
||||||
|
|
||||||
auto o_snap = c->serializer->snapshot ();
|
auto o_snap = c->serializer->snapshot ();
|
||||||
if (!o->serialize_subset (c, _, this,
|
if (!o->serialize_subset (c, _, this,
|
||||||
|
lookup_map,
|
||||||
backtrack_klass_map,
|
backtrack_klass_map,
|
||||||
input_klass_map,
|
input_klass_map,
|
||||||
lookahead_klass_map))
|
lookahead_klass_map))
|
||||||
|
@ -2621,10 +2630,11 @@ struct ChainContextFormat1
|
||||||
if (unlikely (!c->serializer->extend_min (out))) return_trace (false);
|
if (unlikely (!c->serializer->extend_min (out))) return_trace (false);
|
||||||
out->format = format;
|
out->format = format;
|
||||||
|
|
||||||
|
const hb_map_t *lookup_map = c->table_tag == HB_OT_TAG_GSUB ? c->plan->gsub_lookups : c->plan->gpos_lookups;
|
||||||
hb_sorted_vector_t<hb_codepoint_t> new_coverage;
|
hb_sorted_vector_t<hb_codepoint_t> new_coverage;
|
||||||
+ hb_zip (this+coverage, ruleSet)
|
+ hb_zip (this+coverage, ruleSet)
|
||||||
| hb_filter (glyphset, hb_first)
|
| hb_filter (glyphset, hb_first)
|
||||||
| hb_filter (subset_offset_array (c, out->ruleSet, this), hb_second)
|
| hb_filter (subset_offset_array (c, out->ruleSet, this, lookup_map), hb_second)
|
||||||
| hb_map (hb_first)
|
| hb_map (hb_first)
|
||||||
| hb_map (glyph_map)
|
| hb_map (glyph_map)
|
||||||
| hb_sink (new_coverage)
|
| hb_sink (new_coverage)
|
||||||
|
@ -2795,8 +2805,9 @@ struct ChainContextFormat2
|
||||||
hb_map_t lookahead_klass_map;
|
hb_map_t lookahead_klass_map;
|
||||||
out->lookaheadClassDef.serialize_subset (c, lookaheadClassDef, this, &lookahead_klass_map);
|
out->lookaheadClassDef.serialize_subset (c, lookaheadClassDef, this, &lookahead_klass_map);
|
||||||
|
|
||||||
hb_vector_t<unsigned> rulesets;
|
unsigned non_zero_index = 0, index = 0;
|
||||||
bool ret = true;
|
bool ret = true;
|
||||||
|
const hb_map_t *lookup_map = c->table_tag == HB_OT_TAG_GSUB ? c->plan->gsub_lookups : c->plan->gpos_lookups;
|
||||||
for (const OffsetTo<ChainRuleSet>& _ : + hb_enumerate (ruleSet)
|
for (const OffsetTo<ChainRuleSet>& _ : + hb_enumerate (ruleSet)
|
||||||
| hb_filter (input_klass_map, hb_first)
|
| hb_filter (input_klass_map, hb_first)
|
||||||
| hb_map (hb_second))
|
| hb_map (hb_second))
|
||||||
|
@ -2807,24 +2818,24 @@ struct ChainContextFormat2
|
||||||
ret = false;
|
ret = false;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (!o->serialize_subset (c, _, this,
|
if (o->serialize_subset (c, _, this,
|
||||||
&backtrack_klass_map,
|
lookup_map,
|
||||||
&input_klass_map,
|
&backtrack_klass_map,
|
||||||
&lookahead_klass_map))
|
&input_klass_map,
|
||||||
{
|
&lookahead_klass_map))
|
||||||
rulesets.push (0);
|
non_zero_index = index;
|
||||||
}
|
|
||||||
else rulesets.push (1);
|
index++;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!ret) return_trace (ret);
|
if (!ret) return_trace (ret);
|
||||||
|
|
||||||
//prune empty trailing ruleSets
|
//prune empty trailing ruleSets
|
||||||
unsigned count = rulesets.length;
|
--index;
|
||||||
while (count > 0 && rulesets[count-1] == 0)
|
while (index > non_zero_index)
|
||||||
{
|
{
|
||||||
out->ruleSet.pop ();
|
out->ruleSet.pop ();
|
||||||
count--;
|
index--;
|
||||||
}
|
}
|
||||||
|
|
||||||
return_trace (bool (out->ruleSet));
|
return_trace (bool (out->ruleSet));
|
||||||
|
@ -3014,8 +3025,16 @@ struct ChainContextFormat3
|
||||||
if (!serialize_coverage_offsets (c, lookahead.iter (), this))
|
if (!serialize_coverage_offsets (c, lookahead.iter (), this))
|
||||||
return_trace (false);
|
return_trace (false);
|
||||||
|
|
||||||
const ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord>> (lookahead);
|
const ArrayOf<LookupRecord> &lookupRecord = StructAfter<ArrayOf<LookupRecord>> (lookahead);
|
||||||
return_trace (c->serializer->copy (lookup));
|
HBUINT16 lookupCount;
|
||||||
|
lookupCount = lookupRecord.len;
|
||||||
|
if (!c->serializer->copy (lookupCount)) return_trace (false);
|
||||||
|
|
||||||
|
const hb_map_t *lookup_map = c->table_tag == HB_OT_TAG_GSUB ? c->plan->gsub_lookups : c->plan->gpos_lookups;
|
||||||
|
for (unsigned i = 0; i < (unsigned) lookupCount; i++)
|
||||||
|
if (!c->serializer->copy (lookupRecord[i], lookup_map)) return_trace (false);
|
||||||
|
|
||||||
|
return_trace (true);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool sanitize (hb_sanitize_context_t *c) const
|
bool sanitize (hb_sanitize_context_t *c) const
|
||||||
|
|
|
@ -18,6 +18,7 @@ EXTRA_DIST += \
|
||||||
expected/layout.gpos3 \
|
expected/layout.gpos3 \
|
||||||
expected/layout.gpos4 \
|
expected/layout.gpos4 \
|
||||||
expected/layout.gpos6 \
|
expected/layout.gpos6 \
|
||||||
|
expected/layout.gpos8 \
|
||||||
expected/layout.gsub3 \
|
expected/layout.gsub3 \
|
||||||
expected/layout.gsub6 \
|
expected/layout.gsub6 \
|
||||||
expected/layout.gdef \
|
expected/layout.gdef \
|
||||||
|
|
|
@ -18,6 +18,7 @@ DISABLED_TESTS = \
|
||||||
tests/layout.gpos3.tests \
|
tests/layout.gpos3.tests \
|
||||||
tests/layout.gpos4.tests \
|
tests/layout.gpos4.tests \
|
||||||
tests/layout.gpos6.tests \
|
tests/layout.gpos6.tests \
|
||||||
|
tests/layout.gpos8.tests \
|
||||||
tests/layout.gsub3.tests \
|
tests/layout.gsub3.tests \
|
||||||
tests/layout.gsub6.tests \
|
tests/layout.gsub6.tests \
|
||||||
tests/layout.gdef.tests \
|
tests/layout.gdef.tests \
|
||||||
|
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
|
@ -0,0 +1,13 @@
|
||||||
|
FONTS:
|
||||||
|
gpos_chaining1_multiple_subrules_f1.otf
|
||||||
|
gpos_chaining2_multiple_subrules_f1.otf
|
||||||
|
gpos_chaining3_simple_f1.otf
|
||||||
|
|
||||||
|
PROFILES:
|
||||||
|
keep-layout.txt
|
||||||
|
keep-layout-retain-gids.txt
|
||||||
|
|
||||||
|
SUBSETS:
|
||||||
|
0123
|
||||||
|
ABC
|
||||||
|
*
|
Loading…
Reference in New Issue