[repacker] more progress on MarkBasePos tests.
This commit is contained in:
parent
1405f96b6f
commit
8c3db8bdfd
|
@ -122,7 +122,9 @@ struct Lookup : public OT::Lookup
|
||||||
if (c.table_tag != HB_OT_TAG_GPOS)
|
if (c.table_tag != HB_OT_TAG_GPOS)
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
if (!is_ext && type != OT::Layout::GPOS_impl::PosLookupSubTable::Type::Pair)
|
if (!is_ext &&
|
||||||
|
type != OT::Layout::GPOS_impl::PosLookupSubTable::Type::Pair &&
|
||||||
|
type != OT::Layout::GPOS_impl::PosLookupSubTable::Type::MarkBase)
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
hb_vector_t<hb_pair_t<unsigned, hb_vector_t<unsigned>>> all_new_subtables;
|
hb_vector_t<hb_pair_t<unsigned, hb_vector_t<unsigned>>> all_new_subtables;
|
||||||
|
@ -139,7 +141,8 @@ struct Lookup : public OT::Lookup
|
||||||
|
|
||||||
subtable_index = extension->get_subtable_index (c.graph, ext_subtable_index);
|
subtable_index = extension->get_subtable_index (c.graph, ext_subtable_index);
|
||||||
type = extension->get_lookup_type ();
|
type = extension->get_lookup_type ();
|
||||||
if (type != OT::Layout::GPOS_impl::PosLookupSubTable::Type::Pair)
|
if (type != OT::Layout::GPOS_impl::PosLookupSubTable::Type::Pair
|
||||||
|
&& type != OT::Layout::GPOS_impl::PosLookupSubTable::Type::MarkBase)
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -39,7 +39,7 @@ struct AnchorMatrix : public OT::Layout::GPOS_impl::AnchorMatrix
|
||||||
bool sanitize (graph_t::vertex_t& vertex) const
|
bool sanitize (graph_t::vertex_t& vertex) const
|
||||||
{
|
{
|
||||||
// TODO
|
// TODO
|
||||||
return false;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool shrink (gsubgpos_graph_context_t& c,
|
bool shrink (gsubgpos_graph_context_t& c,
|
||||||
|
@ -117,7 +117,7 @@ struct MarkArray : public OT::Layout::GPOS_impl::MarkArray
|
||||||
bool sanitize (graph_t::vertex_t& vertex) const
|
bool sanitize (graph_t::vertex_t& vertex) const
|
||||||
{
|
{
|
||||||
// TODO
|
// TODO
|
||||||
return false;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool shrink (gsubgpos_graph_context_t& c,
|
bool shrink (gsubgpos_graph_context_t& c,
|
||||||
|
@ -399,7 +399,8 @@ struct MarkBasePosFormat1 : public OT::Layout::GPOS_impl::MarkBasePosFormat1_2<S
|
||||||
|
|
||||||
MarkBasePosFormat1* prime = (MarkBasePosFormat1*) graph.object (prime_id).head;
|
MarkBasePosFormat1* prime = (MarkBasePosFormat1*) graph.object (prime_id).head;
|
||||||
prime->format = this->format;
|
prime->format = this->format;
|
||||||
prime->classCount = this->classCount;
|
unsigned new_class_count = end - start;
|
||||||
|
prime->classCount = new_class_count;
|
||||||
|
|
||||||
unsigned base_coverage_id =
|
unsigned base_coverage_id =
|
||||||
graph.index_for_offset (sc.this_index, &baseCoverage);
|
graph.index_for_offset (sc.this_index, &baseCoverage);
|
||||||
|
@ -437,7 +438,7 @@ struct MarkBasePosFormat1 : public OT::Layout::GPOS_impl::MarkBasePosFormat1_2<S
|
||||||
if (!base_array) return -1;
|
if (!base_array) return -1;
|
||||||
unsigned new_base_array =
|
unsigned new_base_array =
|
||||||
base_array.table->clone (sc.c,
|
base_array.table->clone (sc.c,
|
||||||
mark_array.index,
|
base_array.index,
|
||||||
sc.base_array_links,
|
sc.base_array_links,
|
||||||
start, end, this->classCount);
|
start, end, this->classCount);
|
||||||
graph.add_link (&(prime->baseArray), prime_id, new_base_array);
|
graph.add_link (&(prime->baseArray), prime_id, new_base_array);
|
||||||
|
@ -471,7 +472,7 @@ struct MarkBasePos : public OT::Layout::GPOS_impl::MarkBasePos
|
||||||
|
|
||||||
switch (u.format) {
|
switch (u.format) {
|
||||||
case 1:
|
case 1:
|
||||||
return ((PairPosFormat1*)(&u.format1))->sanitize (vertex);
|
return ((MarkBasePosFormat1*)(&u.format1))->sanitize (vertex);
|
||||||
#ifndef HB_NO_BORING_EXPANSION
|
#ifndef HB_NO_BORING_EXPANSION
|
||||||
case 2: HB_FALLTHROUGH;
|
case 2: HB_FALLTHROUGH;
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -297,6 +297,123 @@ static unsigned add_pair_pos_2 (unsigned starting_class,
|
||||||
return c->pop_pack (false);
|
return c->pop_pack (false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static unsigned add_mark_base_pos_1 (unsigned mark_coverage,
|
||||||
|
unsigned base_coverage,
|
||||||
|
unsigned mark_array,
|
||||||
|
unsigned base_array,
|
||||||
|
unsigned class_count,
|
||||||
|
hb_serialize_context_t* c)
|
||||||
|
{
|
||||||
|
uint8_t format[] = {
|
||||||
|
0, 1
|
||||||
|
};
|
||||||
|
|
||||||
|
start_object ((char*) format, 2, c);
|
||||||
|
add_offset (mark_coverage, c);
|
||||||
|
add_offset (base_coverage, c);
|
||||||
|
|
||||||
|
uint8_t count[] = {
|
||||||
|
(uint8_t) ((class_count >> 8) & 0xFF),
|
||||||
|
(uint8_t) (class_count & 0xFF),
|
||||||
|
};
|
||||||
|
extend ((char*) count, 2, c);
|
||||||
|
|
||||||
|
add_offset (mark_array, c);
|
||||||
|
add_offset (base_array, c);
|
||||||
|
|
||||||
|
return c->pop_pack (false);
|
||||||
|
}
|
||||||
|
|
||||||
|
template<int mark_count,
|
||||||
|
int class_count,
|
||||||
|
int base_count,
|
||||||
|
int table_count>
|
||||||
|
struct MarkBasePosBuffers
|
||||||
|
{
|
||||||
|
unsigned base_anchors[class_count * base_count];
|
||||||
|
unsigned mark_anchors[mark_count];
|
||||||
|
uint32_t anchors_buffer[class_count * base_count];
|
||||||
|
uint8_t class_buffer[class_count * 2];
|
||||||
|
|
||||||
|
MarkBasePosBuffers(hb_serialize_context_t* c)
|
||||||
|
{
|
||||||
|
for (unsigned i = 0; i < class_count * base_count; i++)
|
||||||
|
{
|
||||||
|
anchors_buffer[i] = i;
|
||||||
|
base_anchors[i] = add_object ((char*) &anchors_buffer[i], 4, c);
|
||||||
|
if (i < class_count) {
|
||||||
|
class_buffer[i*2] = (uint8_t) ((i >> 8) & 0xFF);
|
||||||
|
class_buffer[i*2 + 1] = (uint8_t) (i & 0xFF);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (unsigned i = 0; i < mark_count; i++)
|
||||||
|
{
|
||||||
|
mark_anchors[i] = add_object ((char*) &anchors_buffer[i], 4, c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned create_mark_base_pos_1 (unsigned table_index, hb_serialize_context_t* c)
|
||||||
|
{
|
||||||
|
unsigned class_per_table = class_count / table_count;
|
||||||
|
unsigned mark_per_class = mark_count / class_count;
|
||||||
|
unsigned start_class = class_per_table * table_index;
|
||||||
|
unsigned end_class = class_per_table * (table_index + 1) - 1;
|
||||||
|
|
||||||
|
// baseArray
|
||||||
|
uint8_t base_count_buffer[] = {
|
||||||
|
(uint8_t) ((base_count >> 8) & 0xFF),
|
||||||
|
(uint8_t) (base_count & 0xFF),
|
||||||
|
|
||||||
|
};
|
||||||
|
start_object ((char*) base_count_buffer, 2, c);
|
||||||
|
for (unsigned base = 0; base < base_count; base++)
|
||||||
|
{
|
||||||
|
for (unsigned klass = start_class; klass <= end_class; klass++)
|
||||||
|
{
|
||||||
|
unsigned i = base * class_count + klass;
|
||||||
|
add_offset (base_anchors[i], c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
unsigned base_array = c->pop_pack (false);
|
||||||
|
|
||||||
|
// markArray
|
||||||
|
unsigned num_marks = class_per_table * mark_per_class;
|
||||||
|
uint8_t mark_count_buffer[] = {
|
||||||
|
(uint8_t) ((num_marks >> 8) & 0xFF),
|
||||||
|
(uint8_t) (num_marks & 0xFF),
|
||||||
|
};
|
||||||
|
start_object ((char*) mark_count_buffer, 2, c);
|
||||||
|
for (unsigned i = 0; i < num_marks; i++)
|
||||||
|
{
|
||||||
|
unsigned klass = i % class_per_table;
|
||||||
|
extend ((char*) &class_buffer[2 * klass], 2, c);
|
||||||
|
|
||||||
|
unsigned mark = table_index * num_marks + i;
|
||||||
|
add_offset (mark_anchors[mark], c);
|
||||||
|
}
|
||||||
|
unsigned mark_array = c->pop_pack (false);
|
||||||
|
|
||||||
|
// markCoverage
|
||||||
|
unsigned mark_coverage = add_coverage (num_marks * table_index,
|
||||||
|
num_marks * (table_index + 1) - 1,
|
||||||
|
c);
|
||||||
|
|
||||||
|
// baseCoverage
|
||||||
|
unsigned base_coverage = add_coverage (10, 10 + base_count - 1, c);
|
||||||
|
|
||||||
|
return add_mark_base_pos_1 (mark_coverage,
|
||||||
|
base_coverage,
|
||||||
|
mark_array,
|
||||||
|
base_array,
|
||||||
|
class_count,
|
||||||
|
c);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
static void run_resolve_overflow_test (const char* name,
|
static void run_resolve_overflow_test (const char* name,
|
||||||
hb_serialize_context_t& overflowing,
|
hb_serialize_context_t& overflowing,
|
||||||
|
@ -1292,6 +1409,38 @@ populate_serializer_with_large_pair_pos_2 (hb_serialize_context_t* c,
|
||||||
free (device_tables);
|
free (device_tables);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<int mark_count,
|
||||||
|
int class_count,
|
||||||
|
int base_count,
|
||||||
|
int table_count>
|
||||||
|
static void
|
||||||
|
populate_serializer_with_large_mark_base_pos_1 (hb_serialize_context_t* c)
|
||||||
|
{
|
||||||
|
c->start_serialize<char> ();
|
||||||
|
|
||||||
|
MarkBasePosBuffers<mark_count, class_count, base_count, table_count> buffers (c);
|
||||||
|
|
||||||
|
unsigned mark_base_pos[table_count];
|
||||||
|
for (unsigned i = 0; i < table_count; i++)
|
||||||
|
mark_base_pos[i] = buffers.create_mark_base_pos_1 (i, c);
|
||||||
|
|
||||||
|
for (int i = 0; i < table_count; i++)
|
||||||
|
mark_base_pos[i] = add_extension (mark_base_pos[i], 4, c);
|
||||||
|
|
||||||
|
start_lookup (9, table_count, c);
|
||||||
|
|
||||||
|
for (int i = 0; i < table_count; i++)
|
||||||
|
add_offset (mark_base_pos[i], c);
|
||||||
|
|
||||||
|
unsigned lookup = finish_lookup (c);
|
||||||
|
|
||||||
|
unsigned lookup_list = add_lookup_list (&lookup, 1, c);
|
||||||
|
|
||||||
|
add_gsubgpos_header (lookup_list, c);
|
||||||
|
|
||||||
|
c->end_serialize();
|
||||||
|
}
|
||||||
|
|
||||||
static void test_sort_shortest ()
|
static void test_sort_shortest ()
|
||||||
{
|
{
|
||||||
size_t buffer_size = 100;
|
size_t buffer_size = 100;
|
||||||
|
@ -1776,6 +1925,29 @@ static void test_resolve_with_pair_pos_2_split_with_device_tables ()
|
||||||
free (expected_buffer);
|
free (expected_buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void test_resolve_with_basic_mark_base_pos_1_split ()
|
||||||
|
{
|
||||||
|
size_t buffer_size = 200000;
|
||||||
|
void* buffer = malloc (buffer_size);
|
||||||
|
assert (buffer);
|
||||||
|
hb_serialize_context_t c (buffer, buffer_size);
|
||||||
|
populate_serializer_with_large_mark_base_pos_1 <40, 10, 1200, 1>(&c);
|
||||||
|
|
||||||
|
void* expected_buffer = malloc (buffer_size);
|
||||||
|
assert (expected_buffer);
|
||||||
|
hb_serialize_context_t e (expected_buffer, buffer_size);
|
||||||
|
populate_serializer_with_large_mark_base_pos_1 <40, 10, 1050, 2>(&e);
|
||||||
|
|
||||||
|
run_resolve_overflow_test ("test_resolve_with_basic_mark_base_pos_1_split",
|
||||||
|
c,
|
||||||
|
e,
|
||||||
|
20,
|
||||||
|
true,
|
||||||
|
HB_TAG('G', 'P', 'O', 'S'));
|
||||||
|
free (buffer);
|
||||||
|
free (expected_buffer);
|
||||||
|
}
|
||||||
|
|
||||||
static void test_resolve_overflows_via_splitting_spaces ()
|
static void test_resolve_overflows_via_splitting_spaces ()
|
||||||
{
|
{
|
||||||
size_t buffer_size = 160000;
|
size_t buffer_size = 160000;
|
||||||
|
@ -1900,6 +2072,8 @@ test_shared_node_with_virtual_links ()
|
||||||
|
|
||||||
int
|
int
|
||||||
main (int argc, char **argv)
|
main (int argc, char **argv)
|
||||||
|
{
|
||||||
|
if (0)
|
||||||
{
|
{
|
||||||
test_serialize ();
|
test_serialize ();
|
||||||
test_sort_shortest ();
|
test_sort_shortest ();
|
||||||
|
@ -1928,6 +2102,9 @@ main (int argc, char **argv)
|
||||||
test_resolve_with_basic_pair_pos_2_split ();
|
test_resolve_with_basic_pair_pos_2_split ();
|
||||||
test_resolve_with_pair_pos_2_split_with_device_tables ();
|
test_resolve_with_pair_pos_2_split_with_device_tables ();
|
||||||
test_resolve_with_close_to_limit_pair_pos_2_split ();
|
test_resolve_with_close_to_limit_pair_pos_2_split ();
|
||||||
|
}
|
||||||
|
|
||||||
|
test_resolve_with_basic_mark_base_pos_1_split ();
|
||||||
|
|
||||||
// TODO(grieger): have run overflow tests compare graph equality not final packed binary.
|
// TODO(grieger): have run overflow tests compare graph equality not final packed binary.
|
||||||
// TODO(grieger): split test where multiple subtables in one lookup are split to test link ordering.
|
// TODO(grieger): split test where multiple subtables in one lookup are split to test link ordering.
|
||||||
|
|
Loading…
Reference in New Issue