[subset] Fix wrong offset base for subsetting LigatureArray.

Offsets from LigatureArray must be relative to the beginning of the LigatureArray table. For the serialization mechanism to use the correct beginning point the LigatureArray must be created using the push()/pop() mechanism. So convert LigatureArray subsetting to use serialize_subset() instead of a manually called serialize and subset.
This commit is contained in:
Garret Rieger 2020-10-05 13:14:53 -07:00
parent 147e93b910
commit 093909b2ff
1 changed files with 36 additions and 42 deletions

View File

@ -548,6 +548,7 @@ struct AnchorMatrix
const hb_map_t *layout_variation_idx_map, const hb_map_t *layout_variation_idx_map,
Iterator index_iter) Iterator index_iter)
{ {
// TODO(grieger): can serialize with an iterator of anchor's?
TRACE_SERIALIZE (this); TRACE_SERIALIZE (this);
if (!index_iter) return_trace (false); if (!index_iter) return_trace (false);
if (unlikely (!c->extend_min ((*this)))) return_trace (false); if (unlikely (!c->extend_min ((*this)))) return_trace (false);
@ -573,6 +574,7 @@ struct AnchorMatrix
TRACE_SUBSET (this); TRACE_SUBSET (this);
auto *out = c->serializer->start_embed (*this); auto *out = c->serializer->start_embed (*this);
// TODO(grieger): shouldn't need to use an allocated vector to do this.
hb_vector_t<unsigned> indexes; hb_vector_t<unsigned> indexes;
for (unsigned row : + hb_range ((unsigned) rows)) for (unsigned row : + hb_range ((unsigned) rows))
{ {
@ -2056,25 +2058,29 @@ struct LigatureArray : OffsetListOf<LigatureAttach>
template <typename Iterator, template <typename Iterator,
hb_requires (hb_is_iterator (Iterator))> hb_requires (hb_is_iterator (Iterator))>
bool subset (hb_subset_context_t *c, bool subset (hb_subset_context_t *c,
unsigned cols, Iterator coverage,
const hb_map_t *klass_mapping, unsigned class_count,
const LigatureArray &src_lig_array, const hb_map_t *klass_mapping) const
Iterator ligature_iter)
{ {
TRACE_SERIALIZE (this); TRACE_SUBSET (this);
const hb_set_t &glyphset = *c->plan->glyphset_gsub ();
if (!ligature_iter.len ()) return_trace (false); auto *out = c->serializer->start_embed (this);
if (unlikely (!c->serializer->extend_min ((*this)))) return_trace (false); if (unlikely (!c->serializer->extend_min (out))) return_trace (false);
unsigned ligature_count = 0;
for (unsigned i : ligature_iter) for (hb_codepoint_t gid : coverage)
{ {
auto *out = serialize_append (c->serializer); ligature_count++;
if (unlikely (!out)) break; if (!glyphset.has (gid)) continue;
out->serialize_subset (c,
src_lig_array.arrayZ[i], auto *matrix = out->serialize_append (c->serializer);
&src_lig_array, if (unlikely (!matrix)) return_trace (false);
cols,
matrix->serialize_subset (c,
this->arrayZ[ligature_count - 1],
this,
class_count,
klass_mapping); klass_mapping);
} }
return_trace (this->len); return_trace (this->len);
@ -2181,7 +2187,7 @@ struct MarkLigPosFormat1
bool subset (hb_subset_context_t *c) const bool subset (hb_subset_context_t *c) const
{ {
TRACE_SUBSET (this); TRACE_SUBSET (this);
const hb_set_t &glyphset = *c->plan->glyphset (); const hb_set_t &glyphset = *c->plan->glyphset_gsub ();
const hb_map_t &glyph_map = *c->plan->glyph_map; const hb_map_t &glyph_map = *c->plan->glyph_map;
auto *out = c->serializer->start_embed (*this); auto *out = c->serializer->start_embed (*this);
@ -2199,15 +2205,14 @@ struct MarkLigPosFormat1
| hb_filter (glyphset, hb_first) | hb_filter (glyphset, hb_first)
; ;
hb_sorted_vector_t<hb_codepoint_t> new_coverage; auto new_mark_coverage =
+ mark_iter + mark_iter
| hb_map (hb_first) | hb_map_retains_sorting (hb_first)
| hb_map (glyph_map) | hb_map_retains_sorting (glyph_map)
| hb_sink (new_coverage)
; ;
if (!out->markCoverage.serialize (c->serializer, out) if (!out->markCoverage.serialize (c->serializer, out)
.serialize (c->serializer, new_coverage.iter ())) .serialize (c->serializer, new_mark_coverage))
return_trace (false); return_trace (false);
out->markArray.serialize (c->serializer, out) out->markArray.serialize (c->serializer, out)
@ -2218,30 +2223,18 @@ struct MarkLigPosFormat1
+ mark_iter + mark_iter
| hb_map (hb_second)); | hb_map (hb_second));
unsigned ligcount = (this+ligatureArray).len; auto new_ligature_coverage =
auto ligature_iter = + hb_iter (this + ligatureCoverage)
+ hb_zip (this+ligatureCoverage, hb_range (ligcount)) | hb_filter (glyphset)
| hb_filter (glyphset, hb_first) | hb_map_retains_sorting (glyph_map)
;
new_coverage.reset ();
+ ligature_iter
| hb_map (hb_first)
| hb_map (glyph_map)
| hb_sink (new_coverage)
; ;
if (!out->ligatureCoverage.serialize (c->serializer, out) if (!out->ligatureCoverage.serialize (c->serializer, out)
.serialize (c->serializer, new_coverage.iter ())) .serialize (c->serializer, new_ligature_coverage))
return_trace (false); return_trace (false);
out->ligatureArray.serialize (c->serializer, out) out->ligatureArray.serialize_subset (c, ligatureArray, this,
.subset (c, hb_iter (this+ligatureCoverage), classCount, &klass_mapping);
(unsigned) classCount,
&klass_mapping,
this+ligatureArray,
+ ligature_iter
| hb_map (hb_second));
return_trace (true); return_trace (true);
} }
@ -2276,6 +2269,7 @@ struct MarkLigPosFormat1
DEFINE_SIZE_STATIC (12); DEFINE_SIZE_STATIC (12);
}; };
struct MarkLigPos struct MarkLigPos
{ {
template <typename context_t, typename ...Ts> template <typename context_t, typename ...Ts>