[subset] GSUB Lookup Type 6: ChainContextSubst
This commit is contained in:
parent
eff91bbb4b
commit
b66094ada0
|
@ -66,6 +66,23 @@ namespace OT {
|
||||||
|
|
||||||
#define NOT_COVERED ((unsigned int) -1)
|
#define NOT_COVERED ((unsigned int) -1)
|
||||||
|
|
||||||
|
|
||||||
|
template<typename Iterator>
|
||||||
|
static inline void Coverage_serialize (hb_serialize_context_t *c,
|
||||||
|
Iterator it);
|
||||||
|
|
||||||
|
template<typename Iterator>
|
||||||
|
static inline void ClassDef_serialize (hb_serialize_context_t *c,
|
||||||
|
Iterator it);
|
||||||
|
|
||||||
|
static void ClassDef_remap_and_serialize (hb_serialize_context_t *c,
|
||||||
|
const hb_set_t &glyphset,
|
||||||
|
const hb_map_t &gid_klass_map,
|
||||||
|
hb_sorted_vector_t<HBGlyphID> glyphs,
|
||||||
|
hb_sorted_vector_t<unsigned> klasses,
|
||||||
|
hb_map_t *klass_map /*INOUT*/);
|
||||||
|
|
||||||
|
|
||||||
template<typename OutputArray>
|
template<typename OutputArray>
|
||||||
struct subset_offset_array_t
|
struct subset_offset_array_t
|
||||||
{
|
{
|
||||||
|
@ -120,7 +137,6 @@ struct
|
||||||
}
|
}
|
||||||
HB_FUNCOBJ (subset_offset_array);
|
HB_FUNCOBJ (subset_offset_array);
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
*
|
*
|
||||||
* OpenType Layout Common Table Formats
|
* OpenType Layout Common Table Formats
|
||||||
|
@ -1179,6 +1195,23 @@ struct Coverage
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool subset (hb_subset_context_t *c) const
|
||||||
|
{
|
||||||
|
TRACE_SUBSET (this);
|
||||||
|
const hb_set_t &glyphset = *c->plan->glyphset ();
|
||||||
|
const hb_map_t &glyph_map = *c->plan->glyph_map;
|
||||||
|
|
||||||
|
auto it =
|
||||||
|
+ iter ()
|
||||||
|
| hb_filter (glyphset)
|
||||||
|
| hb_map_retains_sorting (glyph_map)
|
||||||
|
;
|
||||||
|
|
||||||
|
bool ret = bool (it);
|
||||||
|
Coverage_serialize (c->serializer, it);
|
||||||
|
return_trace (ret);
|
||||||
|
}
|
||||||
|
|
||||||
bool sanitize (hb_sanitize_context_t *c) const
|
bool sanitize (hb_sanitize_context_t *c) const
|
||||||
{
|
{
|
||||||
TRACE_SANITIZE (this);
|
TRACE_SANITIZE (this);
|
||||||
|
@ -1297,15 +1330,51 @@ struct Coverage
|
||||||
DEFINE_SIZE_UNION (2, format);
|
DEFINE_SIZE_UNION (2, format);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
template<typename Iterator>
|
||||||
|
static inline void
|
||||||
|
Coverage_serialize (hb_serialize_context_t *c,
|
||||||
|
Iterator it)
|
||||||
|
{ c->start_embed<Coverage> ()->serialize (c, it); }
|
||||||
|
|
||||||
|
static void ClassDef_remap_and_serialize (hb_serialize_context_t *c,
|
||||||
|
const hb_set_t &glyphset,
|
||||||
|
const hb_map_t &gid_klass_map,
|
||||||
|
hb_sorted_vector_t<HBGlyphID> glyphs,
|
||||||
|
hb_sorted_vector_t<unsigned> klasses,
|
||||||
|
hb_map_t *klass_map /*INOUT*/)
|
||||||
|
{
|
||||||
|
bool has_no_match = glyphset.get_population () > gid_klass_map.get_population ();
|
||||||
|
|
||||||
|
hb_map_t m;
|
||||||
|
if (!klass_map) klass_map = &m;
|
||||||
|
|
||||||
|
if (has_no_match) klass_map->set (0, 0);
|
||||||
|
unsigned idx = klass_map->has (0) ? 1 : 0;
|
||||||
|
for (const unsigned k: klasses.iter ())
|
||||||
|
{
|
||||||
|
if (klass_map->has (k)) continue;
|
||||||
|
klass_map->set (k, idx);
|
||||||
|
idx++;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto it =
|
||||||
|
+ glyphs.iter ()
|
||||||
|
| hb_map_retains_sorting ([&] (const HBGlyphID& gid) -> hb_pair_t<hb_codepoint_t, HBUINT16>
|
||||||
|
{
|
||||||
|
HBUINT16 new_klass;
|
||||||
|
new_klass = klass_map->get (gid_klass_map[gid]);
|
||||||
|
return hb_pair ((hb_codepoint_t)gid, new_klass);
|
||||||
|
})
|
||||||
|
;
|
||||||
|
|
||||||
|
c->propagate_error (glyphs, klasses);
|
||||||
|
ClassDef_serialize (c, it);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Class Definition Table
|
* Class Definition Table
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static inline void ClassDef_serialize (hb_serialize_context_t *c,
|
|
||||||
hb_array_t<const HBGlyphID> glyphs,
|
|
||||||
hb_array_t<const HBUINT16> klasses);
|
|
||||||
|
|
||||||
struct ClassDefFormat1
|
struct ClassDefFormat1
|
||||||
{
|
{
|
||||||
friend struct ClassDef;
|
friend struct ClassDef;
|
||||||
|
@ -1316,53 +1385,53 @@ struct ClassDefFormat1
|
||||||
return classValue[(unsigned int) (glyph_id - startGlyph)];
|
return classValue[(unsigned int) (glyph_id - startGlyph)];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<typename Iterator,
|
||||||
|
hb_requires (hb_is_iterator (Iterator))>
|
||||||
bool serialize (hb_serialize_context_t *c,
|
bool serialize (hb_serialize_context_t *c,
|
||||||
hb_array_t<const HBGlyphID> glyphs,
|
Iterator it)
|
||||||
hb_array_t<const HBUINT16> klasses)
|
|
||||||
{
|
{
|
||||||
TRACE_SERIALIZE (this);
|
TRACE_SERIALIZE (this);
|
||||||
if (unlikely (!c->extend_min (*this))) return_trace (false);
|
if (unlikely (!c->extend_min (*this))) return_trace (false);
|
||||||
|
|
||||||
if (unlikely (!glyphs))
|
if (unlikely (!it))
|
||||||
{
|
{
|
||||||
startGlyph = 0;
|
startGlyph = 0;
|
||||||
classValue.len = 0;
|
classValue.len = 0;
|
||||||
return_trace (true);
|
return_trace (true);
|
||||||
}
|
}
|
||||||
|
|
||||||
hb_codepoint_t glyph_min = +glyphs | hb_reduce (hb_min, 0xFFFFu);
|
startGlyph = (*it).first;
|
||||||
hb_codepoint_t glyph_max = +glyphs | hb_reduce (hb_max, 0u);
|
classValue.serialize (c, + it
|
||||||
|
| hb_map (hb_second));
|
||||||
startGlyph = glyph_min;
|
|
||||||
c->check_assign (classValue.len, glyph_max - glyph_min + 1);
|
|
||||||
if (unlikely (!c->extend (classValue))) return_trace (false);
|
|
||||||
|
|
||||||
for (unsigned int i = 0; i < glyphs.length; i++)
|
|
||||||
classValue[glyphs[i] - glyph_min] = klasses[i];
|
|
||||||
|
|
||||||
return_trace (true);
|
return_trace (true);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool subset (hb_subset_context_t *c) const
|
bool subset (hb_subset_context_t *c,
|
||||||
|
hb_map_t *klass_map = nullptr /*OUT*/) const
|
||||||
{
|
{
|
||||||
TRACE_SUBSET (this);
|
TRACE_SUBSET (this);
|
||||||
const hb_set_t &glyphset = *c->plan->glyphset ();
|
const hb_set_t &glyphset = *c->plan->glyphset ();
|
||||||
const hb_map_t &glyph_map = *c->plan->glyph_map;
|
const hb_map_t &glyph_map = *c->plan->glyph_map;
|
||||||
|
|
||||||
hb_sorted_vector_t<HBGlyphID> glyphs;
|
hb_sorted_vector_t<HBGlyphID> glyphs;
|
||||||
hb_vector_t<HBUINT16> klasses;
|
hb_sorted_vector_t<unsigned> orig_klasses;
|
||||||
|
hb_map_t gid_org_klass_map;
|
||||||
|
|
||||||
hb_codepoint_t start = startGlyph;
|
hb_codepoint_t start = startGlyph;
|
||||||
hb_codepoint_t end = start + classValue.len;
|
hb_codepoint_t end = start + classValue.len;
|
||||||
for (hb_codepoint_t g = start; g < end; g++)
|
for (const hb_codepoint_t gid : + hb_range (start, end)
|
||||||
|
| hb_filter (glyphset))
|
||||||
{
|
{
|
||||||
if (!glyphset.has (g)) continue;
|
unsigned klass = classValue[gid - start];
|
||||||
unsigned int value = classValue[g - start];
|
if (!klass) continue;
|
||||||
if (!value) continue;
|
|
||||||
glyphs.push(glyph_map[g]);
|
glyphs.push (glyph_map[gid]);
|
||||||
klasses.push(value);
|
gid_org_klass_map.set (glyph_map[gid], klass);
|
||||||
|
orig_klasses.push (klass);
|
||||||
}
|
}
|
||||||
c->serializer->propagate_error (glyphs, klasses);
|
|
||||||
ClassDef_serialize (c->serializer, glyphs, klasses);
|
ClassDef_remap_and_serialize (c->serializer, glyphset, gid_org_klass_map,
|
||||||
|
glyphs, orig_klasses, klass_map);
|
||||||
return_trace ((bool) glyphs);
|
return_trace ((bool) glyphs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1452,70 +1521,87 @@ struct ClassDefFormat2
|
||||||
return rangeRecord.bsearch (glyph_id).value;
|
return rangeRecord.bsearch (glyph_id).value;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<typename Iterator,
|
||||||
|
hb_requires (hb_is_iterator (Iterator))>
|
||||||
bool serialize (hb_serialize_context_t *c,
|
bool serialize (hb_serialize_context_t *c,
|
||||||
hb_array_t<const HBGlyphID> glyphs,
|
Iterator it)
|
||||||
hb_array_t<const HBUINT16> klasses)
|
|
||||||
{
|
{
|
||||||
TRACE_SERIALIZE (this);
|
TRACE_SERIALIZE (this);
|
||||||
if (unlikely (!c->extend_min (*this))) return_trace (false);
|
if (unlikely (!c->extend_min (*this))) return_trace (false);
|
||||||
|
|
||||||
if (unlikely (!glyphs))
|
if (unlikely (!it))
|
||||||
{
|
{
|
||||||
rangeRecord.len = 0;
|
rangeRecord.len = 0;
|
||||||
return_trace (true);
|
return_trace (true);
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int count = glyphs.len ();
|
unsigned num_ranges = 1;
|
||||||
unsigned int num_ranges = 1;
|
hb_codepoint_t prev_gid = (*it).first;
|
||||||
for (unsigned int i = 1; i < count; i++)
|
unsigned prev_klass = (*it).second;
|
||||||
if (glyphs[i - 1] + 1 != glyphs[i] ||
|
|
||||||
klasses[i - 1] != klasses[i])
|
|
||||||
num_ranges++;
|
|
||||||
rangeRecord.len = num_ranges;
|
|
||||||
if (unlikely (!c->extend (rangeRecord))) return_trace (false);
|
|
||||||
|
|
||||||
unsigned int range = 0;
|
RangeRecord range_rec;
|
||||||
rangeRecord[range].start = glyphs[0];
|
range_rec.start = prev_gid;
|
||||||
rangeRecord[range].value = klasses[0];
|
range_rec.end = prev_gid;
|
||||||
for (unsigned int i = 1; i < count; i++)
|
range_rec.value = prev_klass;
|
||||||
|
|
||||||
|
RangeRecord *record = c->copy (range_rec);
|
||||||
|
|
||||||
|
for (const auto gid_klass_pair : + (++it))
|
||||||
{
|
{
|
||||||
if (glyphs[i - 1] + 1 != glyphs[i] ||
|
hb_codepoint_t cur_gid = gid_klass_pair.first;
|
||||||
klasses[i - 1] != klasses[i])
|
unsigned cur_klass = gid_klass_pair.second;
|
||||||
|
|
||||||
|
if (cur_gid != prev_gid + 1 ||
|
||||||
|
cur_klass != prev_klass)
|
||||||
{
|
{
|
||||||
rangeRecord[range].end = glyphs[i - 1];
|
record->end = prev_gid;
|
||||||
range++;
|
num_ranges++;
|
||||||
rangeRecord[range].start = glyphs[i];
|
|
||||||
rangeRecord[range].value = klasses[i];
|
range_rec.start = cur_gid;
|
||||||
|
range_rec.end = cur_gid;
|
||||||
|
range_rec.value = cur_klass;
|
||||||
|
|
||||||
|
record = c->copy (range_rec);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
prev_klass = cur_klass;
|
||||||
|
prev_gid = cur_gid;
|
||||||
}
|
}
|
||||||
rangeRecord[range].end = glyphs[count - 1];
|
|
||||||
|
record->end = prev_gid;
|
||||||
|
rangeRecord.len = num_ranges;
|
||||||
return_trace (true);
|
return_trace (true);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool subset (hb_subset_context_t *c) const
|
bool subset (hb_subset_context_t *c,
|
||||||
|
hb_map_t *klass_map = nullptr /*OUT*/) const
|
||||||
{
|
{
|
||||||
TRACE_SUBSET (this);
|
TRACE_SUBSET (this);
|
||||||
const hb_set_t &glyphset = *c->plan->glyphset ();
|
const hb_set_t &glyphset = *c->plan->glyphset ();
|
||||||
const hb_map_t &glyph_map = *c->plan->glyph_map;
|
const hb_map_t &glyph_map = *c->plan->glyph_map;
|
||||||
hb_vector_t<HBGlyphID> glyphs;
|
|
||||||
hb_vector_t<HBUINT16> klasses;
|
|
||||||
|
|
||||||
unsigned int count = rangeRecord.len;
|
hb_sorted_vector_t<HBGlyphID> glyphs;
|
||||||
for (unsigned int i = 0; i < count; i++)
|
hb_sorted_vector_t<unsigned> orig_klasses;
|
||||||
|
hb_map_t gid_org_klass_map;
|
||||||
|
|
||||||
|
unsigned count = rangeRecord.len;
|
||||||
|
for (unsigned i = 0; i < count; i++)
|
||||||
{
|
{
|
||||||
unsigned int value = rangeRecord[i].value;
|
unsigned klass = rangeRecord[i].value;
|
||||||
if (!value) continue;
|
if (!klass) continue;
|
||||||
hb_codepoint_t start = rangeRecord[i].start;
|
hb_codepoint_t start = rangeRecord[i].start;
|
||||||
hb_codepoint_t end = rangeRecord[i].end + 1;
|
hb_codepoint_t end = rangeRecord[i].end + 1;
|
||||||
for (hb_codepoint_t g = start; g < end; g++)
|
for (hb_codepoint_t g = start; g < end; g++)
|
||||||
{
|
{
|
||||||
if (!glyphset.has (g)) continue;
|
if (!glyphset.has (g)) continue;
|
||||||
glyphs.push (glyph_map[g]);
|
glyphs.push (glyph_map[g]);
|
||||||
klasses.push (value);
|
gid_org_klass_map.set (glyph_map[g], klass);
|
||||||
|
orig_klasses.push (klass);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
c->serializer->propagate_error (glyphs, klasses);
|
|
||||||
ClassDef_serialize (c->serializer, glyphs, klasses);
|
ClassDef_remap_and_serialize (c->serializer, glyphset, gid_org_klass_map,
|
||||||
|
glyphs, orig_klasses, klass_map);
|
||||||
return_trace ((bool) glyphs);
|
return_trace ((bool) glyphs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1612,25 +1698,36 @@ struct ClassDef
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool serialize (hb_serialize_context_t *c,
|
template<typename Iterator,
|
||||||
hb_array_t<const HBGlyphID> glyphs,
|
hb_requires (hb_is_iterator (Iterator))>
|
||||||
hb_array_t<const HBUINT16> klasses)
|
bool serialize (hb_serialize_context_t *c, Iterator it)
|
||||||
{
|
{
|
||||||
TRACE_SERIALIZE (this);
|
TRACE_SERIALIZE (this);
|
||||||
if (unlikely (!c->extend_min (*this))) return_trace (false);
|
if (unlikely (!c->extend_min (*this))) return_trace (false);
|
||||||
|
|
||||||
unsigned int format = 2;
|
unsigned format = 2;
|
||||||
if (likely (glyphs))
|
if (likely (it))
|
||||||
{
|
{
|
||||||
hb_codepoint_t glyph_min = +glyphs | hb_reduce (hb_min, 0xFFFFu);
|
hb_codepoint_t glyph_min = (*it).first;
|
||||||
hb_codepoint_t glyph_max = +glyphs | hb_reduce (hb_max, 0u);
|
hb_codepoint_t glyph_max = + it
|
||||||
|
| hb_map (hb_first)
|
||||||
|
| hb_reduce (hb_max, 0u);
|
||||||
|
|
||||||
unsigned int count = glyphs.len ();
|
unsigned num_ranges = 1;
|
||||||
unsigned int num_ranges = 1;
|
hb_codepoint_t prev_gid = glyph_min;
|
||||||
for (unsigned int i = 1; i < count; i++)
|
unsigned prev_klass = (*it).second;
|
||||||
if (glyphs[i - 1] + 1 != glyphs[i] ||
|
|
||||||
klasses[i - 1] != klasses[i])
|
for (const auto gid_klass_pair : it)
|
||||||
num_ranges++;
|
{
|
||||||
|
hb_codepoint_t cur_gid = gid_klass_pair.first;
|
||||||
|
unsigned cur_klass = gid_klass_pair.second;
|
||||||
|
if (cur_gid != prev_gid + 1 ||
|
||||||
|
cur_klass != prev_klass)
|
||||||
|
num_ranges++;
|
||||||
|
|
||||||
|
prev_gid = cur_gid;
|
||||||
|
prev_klass = cur_klass;
|
||||||
|
}
|
||||||
|
|
||||||
if (1 + (glyph_max - glyph_min + 1) < num_ranges * 3)
|
if (1 + (glyph_max - glyph_min + 1) < num_ranges * 3)
|
||||||
format = 1;
|
format = 1;
|
||||||
|
@ -1639,18 +1736,19 @@ struct ClassDef
|
||||||
|
|
||||||
switch (u.format)
|
switch (u.format)
|
||||||
{
|
{
|
||||||
case 1: return_trace (u.format1.serialize (c, glyphs, klasses));
|
case 1: return_trace (u.format1.serialize (c, it));
|
||||||
case 2: return_trace (u.format2.serialize (c, glyphs, klasses));
|
case 2: return_trace (u.format2.serialize (c, it));
|
||||||
default:return_trace (false);
|
default:return_trace (false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool subset (hb_subset_context_t *c) const
|
bool subset (hb_subset_context_t *c,
|
||||||
|
hb_map_t *klass_map = nullptr /*OUT*/) const
|
||||||
{
|
{
|
||||||
TRACE_SUBSET (this);
|
TRACE_SUBSET (this);
|
||||||
switch (u.format) {
|
switch (u.format) {
|
||||||
case 1: return_trace (u.format1.subset (c));
|
case 1: return_trace (u.format1.subset (c, klass_map));
|
||||||
case 2: return_trace (u.format2.subset (c));
|
case 2: return_trace (u.format2.subset (c, klass_map));
|
||||||
default:return_trace (false);
|
default:return_trace (false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1717,10 +1815,10 @@ struct ClassDef
|
||||||
DEFINE_SIZE_UNION (2, format);
|
DEFINE_SIZE_UNION (2, format);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
template<typename Iterator>
|
||||||
static inline void ClassDef_serialize (hb_serialize_context_t *c,
|
static inline void ClassDef_serialize (hb_serialize_context_t *c,
|
||||||
hb_array_t<const HBGlyphID> glyphs,
|
Iterator it)
|
||||||
hb_array_t<const HBUINT16> klasses)
|
{ c->start_embed<ClassDef> ()->serialize (c, it); }
|
||||||
{ c->start_embed<ClassDef> ()->serialize (c, glyphs, klasses); }
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -2002,6 +2002,80 @@ struct ChainRule
|
||||||
lookup.arrayZ, lookup_context));
|
lookup.arrayZ, lookup_context));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<typename Iterator,
|
||||||
|
hb_requires (hb_is_iterator (Iterator))>
|
||||||
|
void serialize_array (hb_serialize_context_t *c,
|
||||||
|
HBUINT16 len,
|
||||||
|
Iterator it) const
|
||||||
|
{
|
||||||
|
c->copy (len);
|
||||||
|
for (const auto g : it)
|
||||||
|
{
|
||||||
|
HBUINT16 gid;
|
||||||
|
gid = g;
|
||||||
|
c->copy (gid);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ChainRule* copy (hb_serialize_context_t *c,
|
||||||
|
const hb_map_t *backtrack_map,
|
||||||
|
const hb_map_t *input_map = nullptr,
|
||||||
|
const hb_map_t *lookahead_map = nullptr) const
|
||||||
|
{
|
||||||
|
TRACE_SERIALIZE (this);
|
||||||
|
auto *out = c->start_embed (this);
|
||||||
|
if (unlikely (!out)) return_trace (nullptr);
|
||||||
|
|
||||||
|
const hb_map_t *mapping = backtrack_map;
|
||||||
|
serialize_array (c, backtrack.len, + backtrack.iter ()
|
||||||
|
| hb_map (mapping));
|
||||||
|
|
||||||
|
const HeadlessArrayOf<HBUINT16> &input = StructAfter<HeadlessArrayOf<HBUINT16>> (backtrack);
|
||||||
|
if (input_map) mapping = input_map;
|
||||||
|
serialize_array (c, input.lenP1, + input.iter ()
|
||||||
|
| hb_map (mapping));
|
||||||
|
|
||||||
|
const ArrayOf<HBUINT16> &lookahead = StructAfter<ArrayOf<HBUINT16>> (input);
|
||||||
|
if (lookahead_map) mapping = lookahead_map;
|
||||||
|
serialize_array (c, lookahead.len, + lookahead.iter ()
|
||||||
|
| hb_map (mapping));
|
||||||
|
|
||||||
|
const ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord>> (lookahead);
|
||||||
|
c->copy (lookup);
|
||||||
|
|
||||||
|
return_trace (out);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool subset (hb_subset_context_t *c,
|
||||||
|
const hb_map_t *backtrack_map = nullptr,
|
||||||
|
const hb_map_t *input_map = nullptr,
|
||||||
|
const hb_map_t *lookahead_map = nullptr) const
|
||||||
|
{
|
||||||
|
TRACE_SUBSET (this);
|
||||||
|
|
||||||
|
if (!backtrack_map)
|
||||||
|
{
|
||||||
|
const hb_set_t &glyphset = *c->plan->glyphset ();
|
||||||
|
if (!hb_all (backtrack, glyphset) ||
|
||||||
|
!hb_all (inputX, glyphset) ||
|
||||||
|
!hb_all (lookaheadX, glyphset))
|
||||||
|
return_trace (false);
|
||||||
|
|
||||||
|
copy (c->serializer, c->plan->glyph_map);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (!hb_all (backtrack, backtrack_map) ||
|
||||||
|
!hb_all (inputX, input_map) ||
|
||||||
|
!hb_all (lookaheadX, lookahead_map))
|
||||||
|
return_trace (false);
|
||||||
|
|
||||||
|
copy (c->serializer, backtrack_map, input_map, lookahead_map);
|
||||||
|
}
|
||||||
|
|
||||||
|
return_trace (true);
|
||||||
|
}
|
||||||
|
|
||||||
bool sanitize (hb_sanitize_context_t *c) const
|
bool sanitize (hb_sanitize_context_t *c) const
|
||||||
{
|
{
|
||||||
TRACE_SANITIZE (this);
|
TRACE_SANITIZE (this);
|
||||||
|
@ -2083,6 +2157,40 @@ struct ChainRuleSet
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool subset (hb_subset_context_t *c,
|
||||||
|
const hb_map_t *backtrack_klass_map = nullptr,
|
||||||
|
const hb_map_t *input_klass_map = nullptr,
|
||||||
|
const hb_map_t *lookahead_klass_map = nullptr) const
|
||||||
|
{
|
||||||
|
TRACE_SUBSET (this);
|
||||||
|
|
||||||
|
auto snap = c->serializer->snapshot ();
|
||||||
|
auto *out = c->serializer->start_embed (*this);
|
||||||
|
if (unlikely (!c->serializer->extend_min (out))) return_trace (false);
|
||||||
|
|
||||||
|
for (const OffsetTo<ChainRule>& _ : rule)
|
||||||
|
{
|
||||||
|
if (!_) continue;
|
||||||
|
auto *o = out->rule.serialize_append (c->serializer);
|
||||||
|
if (unlikely (!o)) continue;
|
||||||
|
|
||||||
|
auto o_snap = c->serializer->snapshot ();
|
||||||
|
if (!o->serialize_subset (c, _, this, out,
|
||||||
|
backtrack_klass_map,
|
||||||
|
input_klass_map,
|
||||||
|
lookahead_klass_map))
|
||||||
|
{
|
||||||
|
out->rule.pop ();
|
||||||
|
c->serializer->revert (o_snap);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ret = bool (out->rule);
|
||||||
|
if (!ret) c->serializer->revert (snap);
|
||||||
|
|
||||||
|
return_trace (ret);
|
||||||
|
}
|
||||||
|
|
||||||
bool sanitize (hb_sanitize_context_t *c) const
|
bool sanitize (hb_sanitize_context_t *c) const
|
||||||
{
|
{
|
||||||
TRACE_SANITIZE (this);
|
TRACE_SANITIZE (this);
|
||||||
|
@ -2175,8 +2283,25 @@ struct ChainContextFormat1
|
||||||
bool subset (hb_subset_context_t *c) const
|
bool subset (hb_subset_context_t *c) const
|
||||||
{
|
{
|
||||||
TRACE_SUBSET (this);
|
TRACE_SUBSET (this);
|
||||||
// TODO(subset)
|
const hb_set_t &glyphset = *c->plan->glyphset ();
|
||||||
return_trace (false);
|
const hb_map_t &glyph_map = *c->plan->glyph_map;
|
||||||
|
|
||||||
|
auto *out = c->serializer->start_embed (*this);
|
||||||
|
if (unlikely (!c->serializer->extend_min (out))) return_trace (false);
|
||||||
|
out->format = format;
|
||||||
|
|
||||||
|
hb_sorted_vector_t<hb_codepoint_t> new_coverage;
|
||||||
|
+ hb_zip (this+coverage, ruleSet)
|
||||||
|
| hb_filter (glyphset, hb_first)
|
||||||
|
| hb_filter (subset_offset_array (c, out->ruleSet, this, out), hb_second)
|
||||||
|
| hb_map (hb_first)
|
||||||
|
| hb_map (glyph_map)
|
||||||
|
| hb_sink (new_coverage)
|
||||||
|
;
|
||||||
|
|
||||||
|
out->coverage.serialize (c->serializer, out)
|
||||||
|
.serialize (c->serializer, new_coverage.iter ());
|
||||||
|
return_trace (bool (new_coverage));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool sanitize (hb_sanitize_context_t *c) const
|
bool sanitize (hb_sanitize_context_t *c) const
|
||||||
|
@ -2314,8 +2439,54 @@ struct ChainContextFormat2
|
||||||
bool subset (hb_subset_context_t *c) const
|
bool subset (hb_subset_context_t *c) const
|
||||||
{
|
{
|
||||||
TRACE_SUBSET (this);
|
TRACE_SUBSET (this);
|
||||||
// TODO(subset)
|
auto *out = c->serializer->start_embed (*this);
|
||||||
return_trace (false);
|
if (unlikely (!c->serializer->extend_min (out))) return_trace (false);
|
||||||
|
out->format = format;
|
||||||
|
out->coverage.serialize_subset (c, coverage, this, out);
|
||||||
|
|
||||||
|
hb_map_t backtrack_klass_map;
|
||||||
|
out->backtrackClassDef.serialize_subset (c, backtrackClassDef, this, out, &backtrack_klass_map);
|
||||||
|
|
||||||
|
// subset inputClassDef based on glyphs survived in Coverage subsetting
|
||||||
|
hb_map_t input_klass_map;
|
||||||
|
out->inputClassDef.serialize_subset (c, inputClassDef, this, out, &input_klass_map);
|
||||||
|
|
||||||
|
hb_map_t lookahead_klass_map;
|
||||||
|
out->lookaheadClassDef.serialize_subset (c, lookaheadClassDef, this, out, &lookahead_klass_map);
|
||||||
|
|
||||||
|
hb_vector_t<unsigned> rulesets;
|
||||||
|
bool ret = true;
|
||||||
|
for (const OffsetTo<ChainRuleSet>& _ : + hb_enumerate (ruleSet)
|
||||||
|
| hb_filter (input_klass_map, hb_first)
|
||||||
|
| hb_map (hb_second))
|
||||||
|
{
|
||||||
|
auto *o = out->ruleSet.serialize_append (c->serializer);
|
||||||
|
if (unlikely (!o))
|
||||||
|
{
|
||||||
|
ret = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (!o->serialize_subset (c, _, this, out,
|
||||||
|
&backtrack_klass_map,
|
||||||
|
&input_klass_map,
|
||||||
|
&lookahead_klass_map))
|
||||||
|
{
|
||||||
|
rulesets.push (0);
|
||||||
|
}
|
||||||
|
else rulesets.push (1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!ret) return_trace (ret);
|
||||||
|
|
||||||
|
//prune empty trailing ruleSets
|
||||||
|
unsigned count = rulesets.length;
|
||||||
|
while (count > 0 && rulesets[count-1] == 0)
|
||||||
|
{
|
||||||
|
out->ruleSet.pop ();
|
||||||
|
count--;
|
||||||
|
}
|
||||||
|
|
||||||
|
return_trace (bool (out->ruleSet));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool sanitize (hb_sanitize_context_t *c) const
|
bool sanitize (hb_sanitize_context_t *c) const
|
||||||
|
@ -2457,11 +2628,46 @@ struct ChainContextFormat3
|
||||||
lookup.len, lookup.arrayZ, lookup_context));
|
lookup.len, lookup.arrayZ, lookup_context));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<typename Iterator,
|
||||||
|
hb_requires (hb_is_iterator (Iterator))>
|
||||||
|
bool serialize_coverage_offsets (hb_subset_context_t *c,
|
||||||
|
Iterator it,
|
||||||
|
const void* src_base,
|
||||||
|
const void* dst_base) const
|
||||||
|
{
|
||||||
|
TRACE_SERIALIZE (this);
|
||||||
|
auto *out = c->serializer->start_embed<OffsetArrayOf<Coverage>> ();
|
||||||
|
|
||||||
|
if (unlikely (!c->serializer->allocate_size<HBUINT16> (HBUINT16::static_size))) return_trace (false);
|
||||||
|
|
||||||
|
+ it
|
||||||
|
| hb_apply (subset_offset_array (c, *out, src_base, dst_base))
|
||||||
|
;
|
||||||
|
|
||||||
|
return_trace (out->len);
|
||||||
|
}
|
||||||
|
|
||||||
bool subset (hb_subset_context_t *c) const
|
bool subset (hb_subset_context_t *c) const
|
||||||
{
|
{
|
||||||
TRACE_SUBSET (this);
|
TRACE_SUBSET (this);
|
||||||
// TODO(subset)
|
|
||||||
return_trace (false);
|
auto *out = c->serializer->start_embed (this);
|
||||||
|
if (unlikely (!out)) return_trace (false);
|
||||||
|
if (unlikely (!c->serializer->embed (this->format))) return_trace (false);
|
||||||
|
|
||||||
|
if (!serialize_coverage_offsets (c, backtrack.iter (), this, out))
|
||||||
|
return_trace (false);
|
||||||
|
|
||||||
|
const OffsetArrayOf<Coverage> &input = StructAfter<OffsetArrayOf<Coverage>> (backtrack);
|
||||||
|
if (!serialize_coverage_offsets (c, input.iter (), this, out))
|
||||||
|
return_trace (false);
|
||||||
|
|
||||||
|
const OffsetArrayOf<Coverage> &lookahead = StructAfter<OffsetArrayOf<Coverage>> (input);
|
||||||
|
if (!serialize_coverage_offsets (c, lookahead.iter (), this, out))
|
||||||
|
return_trace (false);
|
||||||
|
|
||||||
|
const ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord>> (lookahead);
|
||||||
|
return_trace (c->serializer->copy (lookup));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool sanitize (hb_sanitize_context_t *c) const
|
bool sanitize (hb_sanitize_context_t *c) const
|
||||||
|
|
|
@ -15,6 +15,7 @@ EXTRA_DIST += \
|
||||||
expected/layout \
|
expected/layout \
|
||||||
expected/layout.gpos \
|
expected/layout.gpos \
|
||||||
expected/layout.gpos3 \
|
expected/layout.gpos3 \
|
||||||
|
expected/layout.gsub6 \
|
||||||
expected/cmap14 \
|
expected/cmap14 \
|
||||||
fonts \
|
fonts \
|
||||||
profiles \
|
profiles \
|
||||||
|
|
|
@ -7,6 +7,7 @@ TESTS = \
|
||||||
tests/layout.tests \
|
tests/layout.tests \
|
||||||
tests/layout.gpos.tests \
|
tests/layout.gpos.tests \
|
||||||
tests/layout.gpos3.tests \
|
tests/layout.gpos3.tests \
|
||||||
|
tests/layout.gsub6.tests \
|
||||||
tests/cmap14.tests \
|
tests/cmap14.tests \
|
||||||
$(NULL)
|
$(NULL)
|
||||||
|
|
||||||
|
|
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,12 @@
|
||||||
|
FONTS:
|
||||||
|
gsub_chaining1_multiple_subrules_f1.otf
|
||||||
|
gsub_chaining2_multiple_subrules_f1.otf
|
||||||
|
gsub_chaining3_simple_f2.otf
|
||||||
|
|
||||||
|
PROFILES:
|
||||||
|
keep-layout.txt
|
||||||
|
keep-layout-retain-gids.txt
|
||||||
|
|
||||||
|
SUBSETS:
|
||||||
|
0123
|
||||||
|
*
|
Loading…
Reference in New Issue