[subset] Fix cpal subsetting when there are partial palette overlaps.

The existing code doesn't correctly handle the case where palettes partially overlap in the color record array. This changes the subsetting to only share entries in the color record array when palettes have the same first color index. Partially overlapping palettes will be converted to disjoint segments in the color record array.

Updates one of the color tests to use multiple palettes.

Also fixes fuzzer: https://oss-fuzz.com/testcase-detail/5568200165687296.
This commit is contained in:
Garret Rieger 2022-05-05 23:27:34 +00:00 committed by Behdad Esfahbod
parent 2884eb97bf
commit b051f3fa83
7 changed files with 37 additions and 23 deletions

View File

@ -197,30 +197,38 @@ struct CPAL
public: public:
bool serialize (hb_serialize_context_t *c, bool serialize (hb_serialize_context_t *c,
const hb_array_t<const BGRAColor> &color_records,
const hb_array_t<const HBUINT16> &color_record_indices, const hb_array_t<const HBUINT16> &color_record_indices,
const hb_map_t &color_record_index_map, const hb_array_t<const BGRAColor> &color_records,
const hb_set_t &retained_color_record_indices) const const hb_vector_t<unsigned>& first_color_index_for_layer,
const hb_map_t& first_color_to_layer_index,
const hb_set_t &retained_color_indices) const
{ {
TRACE_SERIALIZE (this); TRACE_SERIALIZE (this);
// TODO(grieger): limit total final size.
for (const auto idx : color_record_indices) for (const auto idx : color_record_indices)
{ {
hb_codepoint_t layer_index = first_color_to_layer_index[idx];
HBUINT16 new_idx; HBUINT16 new_idx;
if (idx == 0) new_idx = 0; new_idx = layer_index * retained_color_indices.get_population ();
else new_idx = color_record_index_map.get (idx);
if (!c->copy<HBUINT16> (new_idx)) return_trace (false); if (!c->copy<HBUINT16> (new_idx)) return_trace (false);
} }
c->push (); c->push ();
for (const auto _ : retained_color_record_indices.iter ()) for (unsigned first_color_index : first_color_index_for_layer)
{ {
if (!c->copy<BGRAColor> (color_records[_])) for (hb_codepoint_t color_index : retained_color_indices)
{
if (!c->copy<BGRAColor> (color_records[first_color_index + color_index]))
{ {
c->pop_discard (); c->pop_discard ();
return_trace (false); return_trace (false);
} }
} }
}
c->add_link (colorRecordsZ, c->pop_pack ()); c->add_link (colorRecordsZ, c->pop_pack ());
return_trace (true); return_trace (true);
} }
@ -228,6 +236,8 @@ struct CPAL
bool subset (hb_subset_context_t *c) const bool subset (hb_subset_context_t *c) const
{ {
TRACE_SUBSET (this); TRACE_SUBSET (this);
if (!numPalettes) return_trace (false);
const hb_map_t *color_index_map = c->plan->colr_palettes; const hb_map_t *color_index_map = c->plan->colr_palettes;
if (color_index_map->is_empty ()) return_trace (false); if (color_index_map->is_empty ()) return_trace (false);
@ -242,30 +252,34 @@ struct CPAL
auto *out = c->serializer->start_embed (*this); auto *out = c->serializer->start_embed (*this);
if (unlikely (!c->serializer->extend_min (out))) return_trace (false); if (unlikely (!c->serializer->extend_min (out))) return_trace (false);
out->version = version; out->version = version;
out->numColors = retained_color_indices.get_population (); out->numColors = retained_color_indices.get_population ();
out->numPalettes = numPalettes; out->numPalettes = numPalettes;
const hb_array_t<const HBUINT16> colorRecordIndices = colorRecordIndicesZ.as_array (numPalettes); hb_vector_t<unsigned> first_color_index_for_layer;
hb_map_t color_record_index_map; hb_map_t first_color_to_layer_index;
hb_set_t retained_color_record_indices;
unsigned record_count = 0; const hb_array_t<const HBUINT16> colorRecordIndices = colorRecordIndicesZ.as_array (numPalettes);
for (const auto first_color_record_idx : colorRecordIndices) for (const auto first_color_record_idx : colorRecordIndices)
{ {
for (unsigned retained_color_idx : retained_color_indices.iter ()) if (first_color_to_layer_index.has (first_color_record_idx)) continue;
{
unsigned color_record_idx = first_color_record_idx + retained_color_idx; first_color_index_for_layer.push (first_color_record_idx);
if (color_record_index_map.has (color_record_idx)) continue; first_color_to_layer_index.set (first_color_record_idx,
color_record_index_map.set (color_record_idx, record_count); first_color_index_for_layer.length - 1);
retained_color_record_indices.add (color_record_idx);
record_count++;
}
} }
out->numColorRecords = record_count; out->numColorRecords = first_color_index_for_layer.length
* retained_color_indices.get_population ();
const hb_array_t<const BGRAColor> color_records = (this+colorRecordsZ).as_array (numColorRecords); const hb_array_t<const BGRAColor> color_records = (this+colorRecordsZ).as_array (numColorRecords);
if (!out->serialize (c->serializer, color_records, colorRecordIndices, color_record_index_map, retained_color_record_indices)) if (!out->serialize (c->serializer,
colorRecordIndices,
color_records,
first_color_index_for_layer,
first_color_to_layer_index,
retained_color_indices))
return_trace (false); return_trace (false);
if (version == 1) if (version == 1)