added CFF1 desubr api test cases & bug fixes

This commit is contained in:
Michiharu Ariza 2018-11-02 15:28:01 -07:00
parent 0b2870085d
commit 1bc710a8c9
15 changed files with 232 additions and 106 deletions

View File

@ -111,16 +111,9 @@ struct CFF1CSOpSet : CSOpSet<Number, OPSET, CFF1CSInterpEnv, PARAM, PATH>
break; break;
} }
env.set_width (has_width); env.set_width (has_width);
if (has_width)
{
OPSET::process_width (env, param);
}
} }
} }
static inline void process_width (CFF1CSInterpEnv &env, PARAM& param)
{}
static inline void flush_args (CFF1CSInterpEnv &env, PARAM& param) static inline void flush_args (CFF1CSInterpEnv &env, PARAM& param)
{ {
SUPER::flush_args (env, param); SUPER::flush_args (env, param);

View File

@ -93,7 +93,12 @@ struct CFFIndex
{ return calculate_offset_array_size (offSize, count); } { return calculate_offset_array_size (offSize, count); }
inline static unsigned int calculate_serialized_size (unsigned int offSize, unsigned int count, unsigned int dataSize) inline static unsigned int calculate_serialized_size (unsigned int offSize, unsigned int count, unsigned int dataSize)
{ return min_size + calculate_offset_array_size (offSize, count) + dataSize; } {
if (count == 0)
return COUNT::static_size;
else
return min_size + calculate_offset_array_size (offSize, count) + dataSize;
}
inline bool serialize (hb_serialize_context_t *c, const CFFIndex &src) inline bool serialize (hb_serialize_context_t *c, const CFFIndex &src)
{ {
@ -110,30 +115,39 @@ struct CFFIndex
const ByteStrArray &byteArray) const ByteStrArray &byteArray)
{ {
TRACE_SERIALIZE (this); TRACE_SERIALIZE (this);
/* serialize CFFIndex header */ if (byteArray.len == 0)
if (unlikely (!c->extend_min (*this))) return_trace (false);
this->count.set (byteArray.len);
this->offSize.set (offSize_);
if (!unlikely (c->allocate_size<HBUINT8> (offSize_ * (byteArray.len + 1))))
return_trace (false);
/* serialize indices */
unsigned int offset = 1;
unsigned int i = 0;
for (; i < byteArray.len; i++)
{ {
set_offset_at (i, offset); COUNT *dest = c->allocate_min<COUNT> ();
offset += byteArray[i].get_size (); if (unlikely (dest == nullptr)) return_trace (false);
dest->set (0);
} }
set_offset_at (i, offset); else
/* serialize data */
for (unsigned int i = 0; i < byteArray.len; i++)
{ {
ByteStr *dest = c->start_embed<ByteStr> (); /* serialize CFFIndex header */
if (unlikely (dest == nullptr || if (unlikely (!c->extend_min (*this))) return_trace (false);
!dest->serialize (c, byteArray[i]))) this->count.set (byteArray.len);
this->offSize.set (offSize_);
if (!unlikely (c->allocate_size<HBUINT8> (offSize_ * (byteArray.len + 1))))
return_trace (false); return_trace (false);
/* serialize indices */
unsigned int offset = 1;
unsigned int i = 0;
for (; i < byteArray.len; i++)
{
set_offset_at (i, offset);
offset += byteArray[i].get_size ();
}
set_offset_at (i, offset);
/* serialize data */
for (unsigned int i = 0; i < byteArray.len; i++)
{
ByteStr *dest = c->start_embed<ByteStr> ();
if (unlikely (dest == nullptr ||
!dest->serialize (c, byteArray[i])))
return_trace (false);
}
} }
return_trace (true); return_trace (true);
} }

View File

@ -378,8 +378,10 @@ struct ParsedCSOp : OpStr
inline void init (unsigned int subr_num_ = 0) inline void init (unsigned int subr_num_ = 0)
{ {
OpStr::init (); OpStr::init ();
flags = kDropFlag_None;
subr_num = subr_num_; subr_num = subr_num_;
drop_flag = false;
keep_flag = false;
skip_flag = false;
} }
inline void fini (void) inline void fini (void)
@ -387,20 +389,19 @@ struct ParsedCSOp : OpStr
OpStr::fini (); OpStr::fini ();
} }
inline bool for_keep (void) const { return (flags & kDropFlag_Keep) != 0; } inline bool for_drop (void) const { return drop_flag; }
inline bool for_drop (void) const { return (flags & kDropFlag_Drop) != 0; } inline void set_drop (void) { if (!for_keep ()) drop_flag = true; }
inline void set_drop (void) { if (!for_keep ()) flags |= kDropFlag_Drop; } inline bool for_keep (void) const { return keep_flag; }
inline void set_keep (void) { flags |= kDropFlag_Keep; } inline void set_keep (void) { keep_flag = true; }
inline bool for_skip (void) const { return skip_flag; }
inline void set_skip (void) { skip_flag = true; }
enum DropFlag
{
kDropFlag_None = 0,
kDropFlag_Drop = 1,
kDropFlag_Keep = 2
};
unsigned int flags;
unsigned int subr_num; unsigned int subr_num;
protected:
bool drop_flag : 1;
bool keep_flag : 1;
bool skip_flag : 1;
}; };
struct ParsedCStr : ParsedValues<ParsedCSOp> struct ParsedCStr : ParsedValues<ParsedCSOp>
@ -425,7 +426,7 @@ struct ParsedCStr : ParsedValues<ParsedCSOp>
{ {
unsigned int parsed_len = get_count (); unsigned int parsed_len = get_count ();
if (likely (parsed_len > 0)) if (likely (parsed_len > 0))
values[parsed_len-1].set_drop (); values[parsed_len-1].set_skip ();
ParsedCSOp val; ParsedCSOp val;
val.init (subr_num); val.init (subr_num);
@ -629,8 +630,8 @@ struct SubrSubsetter
* 4. re-encode all charstrings and subroutines with new subroutine numbers * 4. re-encode all charstrings and subroutines with new subroutine numbers
* *
* Phases #1 and #2 are done at the same time in collect_subrs (). * Phases #1 and #2 are done at the same time in collect_subrs ().
* Phase #3 requires walking charstrings/subroutines forward then backward (hence parsing), because * Phase #3 walks charstrings/subroutines forward then backward (hence parsing required),
* we can't tell if a number belongs to a hint op until we see the first moveto. * because we can't tell if a number belongs to a hint op until we see the first moveto.
* *
* Assumption: a callsubr/callgsubr operator must immediately follow a (biased) subroutine number * Assumption: a callsubr/callgsubr operator must immediately follow a (biased) subroutine number
* within the same charstring/subroutine, e.g., not split across a charstring and a subroutine. * within the same charstring/subroutine, e.g., not split across a charstring and a subroutine.
@ -686,7 +687,8 @@ struct SubrSubsetter
drop_hints); drop_hints);
bool seen_moveto = false; bool seen_moveto = false;
if (drop_hints_in_str (parsed_charstrings[i], param, seen_moveto)) bool ends_in_hint = false;
if (drop_hints_in_str (parsed_charstrings[i], param, seen_moveto, ends_in_hint))
parsed_charstrings[i].set_hint_removed (); parsed_charstrings[i].set_hint_removed ();
} }
@ -755,39 +757,39 @@ struct SubrSubsetter
ParsedCStrs &subrs, unsigned int subr_num, ParsedCStrs &subrs, unsigned int subr_num,
const SubrSubsetParam &param, bool &seen_moveto) const SubrSubsetParam &param, bool &seen_moveto)
{ {
if (drop_hints_in_str (subrs[subr_num], param, seen_moveto)) bool ends_in_hint = false;
{ bool has_hint = drop_hints_in_str (subrs[subr_num], param, seen_moveto, ends_in_hint);
/* if the first op in the subr is a hint op, then all args/ops (especially including other subr calls)
* preceding this subr no and call op are hints */ /* if this subr ends with a stem hint (i.e., not a number a potential argument for moveto),
/* TODO CFF2 vsindex */ * then this entire subroutine must be a hint. drop its call. */
for (unsigned int i = 0; i + 1 < pos; i++) if (ends_in_hint)
str.values[i].set_drop (); str.values[pos].set_drop ();
return true;
} return has_hint;
else
return false;
} }
/* returns true if it sees a hint op before moveto */ /* returns true if it sees a hint op before the first moveto */
inline bool drop_hints_in_str (ParsedCStr &str, const SubrSubsetParam &param, bool &seen_moveto) inline bool drop_hints_in_str (ParsedCStr &str, const SubrSubsetParam &param,
bool &seen_moveto, bool &ends_in_hint)
{ {
bool seen_hint = false; bool seen_hint = false;
unsigned int next_check_pos = 0;
for (unsigned int pos = 0; pos < str.values.len; pos++) for (unsigned int pos = 0; pos < str.values.len; pos++)
{ {
bool has_hint = false;
switch (str.values[pos].op) switch (str.values[pos].op)
{ {
case OpCode_callsubr: case OpCode_callsubr:
seen_hint |= drop_hints_in_subr (str, pos, has_hint = drop_hints_in_subr (str, pos,
*param.parsed_local_subrs, str.values[pos].subr_num, *param.parsed_local_subrs, str.values[pos].subr_num,
param, seen_moveto); param, seen_moveto);
break; break;
case OpCode_callgsubr: case OpCode_callgsubr:
seen_hint |= drop_hints_in_subr (str, pos, has_hint = drop_hints_in_subr (str, pos,
*param.parsed_global_subrs, str.values[pos].subr_num, *param.parsed_global_subrs, str.values[pos].subr_num,
param, seen_moveto); param, seen_moveto);
break; break;
case OpCode_rmoveto: case OpCode_rmoveto:
@ -809,19 +811,27 @@ struct SubrSubsetter
case OpCode_vstemhm: case OpCode_vstemhm:
case OpCode_hstem: case OpCode_hstem:
case OpCode_vstem: case OpCode_vstem:
seen_hint = true; has_hint = true;
for (unsigned int i = next_check_pos; i <= pos; i++) str.values[pos].set_drop ();
{ if ((pos + 1 >= str.values.len) /* CFF2 */
/* TODO: CFF2 vsindex */ || (str.values[pos + 1].op == OpCode_return))
str.values[i].set_drop (); ends_in_hint = true;
}
next_check_pos = pos + 1;
break; break;
default: default:
/* NONE */ /* NONE */
break; break;
} }
if (has_hint)
{
for (int i = pos - 1; i >= 0; i--)
{
if (str.values[i].for_drop ())
break;
str.values[i].set_drop ();
}
seen_hint |= has_hint;
}
} }
return seen_hint; return seen_hint;
@ -878,7 +888,7 @@ struct SubrSubsetter
for (unsigned int i = 0; i < str.get_count(); i++) for (unsigned int i = 0; i < str.get_count(); i++)
{ {
const ParsedCSOp &opstr = str.values[i]; const ParsedCSOp &opstr = str.values[i];
if (!opstr.for_drop ()) if (!opstr.for_drop () && !opstr.for_skip ())
{ {
switch (opstr.op) switch (opstr.op)
{ {

View File

@ -380,11 +380,6 @@ struct CFF1CSOpSet_SubrSubset : CFF1CSOpSet<CFF1CSOpSet_SubrSubset, SubrSubsetPa
} }
} }
static inline void process_width (CFF1CSInterpEnv &env, SubrSubsetParam& param)
{
}
protected: protected:
static inline void process_call_subr (OpCode op, CSType type, static inline void process_call_subr (OpCode op, CSType type,
CFF1CSInterpEnv &env, SubrSubsetParam& param, CFF1CSInterpEnv &env, SubrSubsetParam& param,
@ -405,7 +400,7 @@ struct CFF1SubrSubsetter : SubrSubsetter<CFF1SubrSubsetter, CFF1Subrs, const OT:
{ {
static inline void set_parsed_prefix (const CFF1CSInterpEnv &env, ParsedCStr &charstring) static inline void set_parsed_prefix (const CFF1CSInterpEnv &env, ParsedCStr &charstring)
{ {
if (env.processed_width) if (env.has_width)
charstring.set_prefix (env.width); charstring.set_prefix (env.width);
} }
}; };
@ -676,7 +671,7 @@ struct cff_subset_plan {
return false; return false;
/* no global/local subroutines */ /* no global/local subroutines */
offsets.globalSubrsInfo.size = HBUINT16::static_size; /* count 0 only */ offsets.globalSubrsInfo.size = CFF1Subrs::calculate_serialized_size (1, 0, 0);
} }
else else
{ {
@ -704,14 +699,19 @@ struct cff_subset_plan {
for (unsigned int fd = 0; fd < orig_fdcount; fd++) for (unsigned int fd = 0; fd < orig_fdcount; fd++)
{ {
subset_localsubrs[fd].init (); subset_localsubrs[fd].init ();
offsets.localSubrsInfos[fd].init ();
if (fdmap.includes (fd)) if (fdmap.includes (fd))
{ {
if (!subr_subsetter.encode_localsubrs (fd, subset_localsubrs[fd])) if (!subr_subsetter.encode_localsubrs (fd, subset_localsubrs[fd]))
return false; return false;
unsigned int dataSize = subset_localsubrs[fd].total_size (); unsigned int dataSize = subset_localsubrs[fd].total_size ();
offsets.localSubrsInfos[fd].offSize = calcOffSize (dataSize); if (dataSize > 0)
offsets.localSubrsInfos[fd].size = CFF1Subrs::calculate_serialized_size (offsets.localSubrsInfos[fd].offSize, subset_localsubrs[fd].len, dataSize); {
offsets.localSubrsInfos[fd].offset = final_size;
offsets.localSubrsInfos[fd].offSize = calcOffSize (dataSize);
offsets.localSubrsInfos[fd].size = CFF1Subrs::calculate_serialized_size (offsets.localSubrsInfos[fd].offSize, subset_localsubrs[fd].len, dataSize);
}
} }
} }
} }
@ -779,8 +779,11 @@ struct cff_subset_plan {
fontdicts_mod.push (fontdict_mod); fontdicts_mod.push (fontdict_mod);
final_size += privInfo.size; final_size += privInfo.size;
if (!plan->desubroutinize) if (!plan->desubroutinize && (offsets.localSubrsInfos[i].size > 0))
{
offsets.localSubrsInfos[i].offset = final_size;
final_size += offsets.localSubrsInfos[i].size; final_size += offsets.localSubrsInfos[i].size;
}
} }
} }
@ -900,21 +903,12 @@ static inline bool _write_cff1 (const cff_subset_plan &plan,
assert (plan.offsets.globalSubrsInfo.offset != 0); assert (plan.offsets.globalSubrsInfo.offset != 0);
assert (plan.offsets.globalSubrsInfo.offset == c.head - c.start); assert (plan.offsets.globalSubrsInfo.offset == c.head - c.start);
if (plan.desubroutinize) CFF1Subrs *dest = c.start_embed <CFF1Subrs> ();
if (unlikely (dest == nullptr)) return false;
if (unlikely (!dest->serialize (&c, plan.offsets.globalSubrsInfo.offSize, plan.subset_globalsubrs)))
{ {
CFF1Subrs *dest = c.allocate_size <CFF1Subrs> (HBUINT16::static_size); DEBUG_MSG (SUBSET, nullptr, "failed to serialize global subroutines");
if (unlikely (dest == nullptr)) return false; return false;
dest->count.set (0);
}
else
{
CFF1Subrs *dest = c.start_embed <CFF1Subrs> ();
if (unlikely (dest == nullptr)) return false;
if (unlikely (!dest->serialize (&c, plan.offsets.globalSubrsInfo.offSize, plan.subset_globalsubrs)))
{
DEBUG_MSG (SUBSET, nullptr, "failed to serialize global subroutines");
return false;
}
} }
} }
@ -1010,7 +1004,7 @@ static inline bool _write_cff1 (const cff_subset_plan &plan,
assert (plan.offsets.privateDictInfo.offset == c.head - c.start); assert (plan.offsets.privateDictInfo.offset == c.head - c.start);
for (unsigned int i = 0; i < acc.privateDicts.len; i++) for (unsigned int i = 0; i < acc.privateDicts.len; i++)
{ {
if (!plan.fdmap.excludes (i)) if (plan.fdmap.includes (i))
{ {
PrivateDict *pd = c.start_embed<PrivateDict> (); PrivateDict *pd = c.start_embed<PrivateDict> ();
if (unlikely (pd == nullptr)) return false; if (unlikely (pd == nullptr)) return false;
@ -1024,14 +1018,7 @@ static inline bool _write_cff1 (const cff_subset_plan &plan,
DEBUG_MSG (SUBSET, nullptr, "failed to serialize CFF Private Dict[%d]", i); DEBUG_MSG (SUBSET, nullptr, "failed to serialize CFF Private Dict[%d]", i);
return false; return false;
} }
} if (plan.offsets.localSubrsInfos[i].size > 0)
}
if (!plan.desubroutinize)
{
for (unsigned int i = 0; i < acc.privateDicts.len; i++)
{
if (!plan.fdmap.excludes (i))
{ {
CFF1Subrs *dest = c.start_embed <CFF1Subrs> (); CFF1Subrs *dest = c.start_embed <CFF1Subrs> ();
if (unlikely (dest == nullptr)) return false; if (unlikely (dest == nullptr)) return false;

Binary file not shown.

View File

@ -91,6 +91,53 @@ test_subset_cff1_strip_hints (void)
hb_face_destroy (face_ac); hb_face_destroy (face_ac);
} }
static void
test_subset_cff1_desubr (void)
{
hb_face_t *face_abc = hb_test_open_font_file ("fonts/SourceSansPro-Regular.abc.otf");
hb_face_t *face_ac = hb_test_open_font_file ("fonts/SourceSansPro-Regular.ac.nosubrs.otf");
hb_set_t *codepoints = hb_set_create ();
hb_subset_input_t *input;
hb_face_t *face_abc_subset;
hb_set_add (codepoints, 'a');
hb_set_add (codepoints, 'c');
input = hb_subset_test_create_input (codepoints);
hb_subset_input_set_desubroutinize (input, true);
face_abc_subset = hb_subset_test_create_subset (face_abc, input);
hb_set_destroy (codepoints);
hb_subset_test_check (face_ac, face_abc_subset, HB_TAG ('C','F','F',' '));
hb_face_destroy (face_abc_subset);
hb_face_destroy (face_abc);
hb_face_destroy (face_ac);
}
static void
test_subset_cff1_desubr_strip_hints (void)
{
hb_face_t *face_abc = hb_test_open_font_file ("fonts/SourceSansPro-Regular.abc.otf");
hb_face_t *face_ac = hb_test_open_font_file ("fonts/SourceSansPro-Regular.ac.nosubrs.nohints.otf");
hb_set_t *codepoints = hb_set_create ();
hb_subset_input_t *input;
hb_face_t *face_abc_subset;
hb_set_add (codepoints, 'a');
hb_set_add (codepoints, 'c');
input = hb_subset_test_create_input (codepoints);
hb_subset_input_set_drop_hints (input, true);
hb_subset_input_set_desubroutinize (input, true);
face_abc_subset = hb_subset_test_create_subset (face_abc, input);
hb_set_destroy (codepoints);
hb_subset_test_check (face_ac, face_abc_subset, HB_TAG ('C', 'F', 'F', ' '));
hb_face_destroy (face_abc_subset);
hb_face_destroy (face_abc);
hb_face_destroy (face_ac);
}
static void static void
test_subset_cff1_j (void) test_subset_cff1_j (void)
{ {
@ -111,6 +158,76 @@ test_subset_cff1_j (void)
hb_face_destroy (face_41_4c2e); hb_face_destroy (face_41_4c2e);
} }
static void
test_subset_cff1_j_strip_hints (void)
{
hb_face_t *face_41_3041_4c2e = hb_test_open_font_file ("fonts/SourceHanSans-Regular.41,3041,4C2E.otf");
hb_face_t *face_41_4c2e = hb_test_open_font_file ("fonts/SourceHanSans-Regular.41,4C2E.nohints.otf");
hb_set_t *codepoints = hb_set_create ();
hb_face_t *face_41_3041_4c2e_subset;
hb_subset_input_t *input;
hb_set_add (codepoints, 0x41);
hb_set_add (codepoints, 0x4C2E);
input = hb_subset_test_create_input (codepoints);
hb_subset_input_set_drop_hints (input, true);
face_41_3041_4c2e_subset = hb_subset_test_create_subset (face_41_3041_4c2e, input);
hb_set_destroy (codepoints);
hb_subset_test_check (face_41_4c2e, face_41_3041_4c2e_subset, HB_TAG ('C','F','F',' '));
hb_face_destroy (face_41_3041_4c2e_subset);
hb_face_destroy (face_41_3041_4c2e);
hb_face_destroy (face_41_4c2e);
}
static void
test_subset_cff1_j_desubr (void)
{
hb_face_t *face_41_3041_4c2e = hb_test_open_font_file ("fonts/SourceHanSans-Regular.41,3041,4C2E.otf");
hb_face_t *face_41_4c2e = hb_test_open_font_file ("fonts/SourceHanSans-Regular.41,4C2E.nosubrs.otf");
hb_set_t *codepoints = hb_set_create ();
hb_face_t *face_41_3041_4c2e_subset;
hb_subset_input_t *input;
hb_set_add (codepoints, 0x41);
hb_set_add (codepoints, 0x4C2E);
input = hb_subset_test_create_input (codepoints);
hb_subset_input_set_desubroutinize (input, true);
face_41_3041_4c2e_subset = hb_subset_test_create_subset (face_41_3041_4c2e, input);
hb_set_destroy (codepoints);
hb_subset_test_check (face_41_4c2e, face_41_3041_4c2e_subset, HB_TAG ('C','F','F',' '));
hb_face_destroy (face_41_3041_4c2e_subset);
hb_face_destroy (face_41_3041_4c2e);
hb_face_destroy (face_41_4c2e);
}
static void
test_subset_cff1_j_desubr_strip_hints (void)
{
hb_face_t *face_41_3041_4c2e = hb_test_open_font_file ("fonts/SourceHanSans-Regular.41,3041,4C2E.otf");
hb_face_t *face_41_4c2e = hb_test_open_font_file ("fonts/SourceHanSans-Regular.41,4C2E.nosubrs.nohints.otf");
hb_set_t *codepoints = hb_set_create ();
hb_face_t *face_41_3041_4c2e_subset;
hb_subset_input_t *input;
hb_set_add (codepoints, 0x41);
hb_set_add (codepoints, 0x4C2E);
input = hb_subset_test_create_input (codepoints);
hb_subset_input_set_drop_hints (input, true);
hb_subset_input_set_desubroutinize (input, true);
face_41_3041_4c2e_subset = hb_subset_test_create_subset (face_41_3041_4c2e, input);
hb_set_destroy (codepoints);
hb_subset_test_check (face_41_4c2e, face_41_3041_4c2e_subset, HB_TAG ('C','F','F',' '));
hb_face_destroy (face_41_3041_4c2e_subset);
hb_face_destroy (face_41_3041_4c2e);
hb_face_destroy (face_41_4c2e);
}
static void static void
test_subset_cff1_expert (void) test_subset_cff1_expert (void)
{ {
@ -139,7 +256,12 @@ main (int argc, char **argv)
hb_test_add (test_subset_cff1_noop); hb_test_add (test_subset_cff1_noop);
hb_test_add (test_subset_cff1); hb_test_add (test_subset_cff1);
hb_test_add (test_subset_cff1_strip_hints); hb_test_add (test_subset_cff1_strip_hints);
hb_test_add (test_subset_cff1_desubr);
hb_test_add (test_subset_cff1_desubr_strip_hints);
hb_test_add (test_subset_cff1_j); hb_test_add (test_subset_cff1_j);
hb_test_add (test_subset_cff1_j_strip_hints);
hb_test_add (test_subset_cff1_j_desubr);
hb_test_add (test_subset_cff1_j_desubr_strip_hints);
hb_test_add (test_subset_cff1_expert); hb_test_add (test_subset_cff1_expert);
return hb_test_run (); return hb_test_run ();