[subset] First pass at hmtx
This commit is contained in:
parent
e5edcc81bf
commit
fa87770372
|
@ -599,13 +599,13 @@ struct cmap
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline hb_blob_t * subset (hb_subset_plan_t *plan, hb_face_t *source) const
|
inline hb_bool_t subset (hb_subset_plan_t *plan) const
|
||||||
{
|
{
|
||||||
hb_auto_array_t<CmapSubtableLongGroup> groups;
|
hb_auto_array_t<CmapSubtableLongGroup> groups;
|
||||||
|
|
||||||
if (unlikely(!populate_groups(plan, &groups)))
|
if (unlikely(!populate_groups(plan, &groups)))
|
||||||
{
|
{
|
||||||
return nullptr;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// We now know how big our blob needs to be
|
// We now know how big our blob needs to be
|
||||||
|
@ -617,21 +617,22 @@ struct cmap
|
||||||
void *dest = calloc(dest_sz, 1);
|
void *dest = calloc(dest_sz, 1);
|
||||||
if (unlikely(!dest)) {
|
if (unlikely(!dest)) {
|
||||||
DEBUG_MSG(SUBSET, nullptr, "Unable to alloc %ld for cmap subset output", dest_sz);
|
DEBUG_MSG(SUBSET, nullptr, "Unable to alloc %ld for cmap subset output", dest_sz);
|
||||||
return nullptr;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (unlikely(!_subset(groups, dest_sz, dest)))
|
if (unlikely(!_subset(groups, dest_sz, dest)))
|
||||||
{
|
{
|
||||||
free(dest);
|
free(dest);
|
||||||
return nullptr;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// all done, write the blob into dest
|
// all done, write the blob into dest
|
||||||
return hb_blob_create ((const char *)dest,
|
hb_blob_t *cmap_prime = hb_blob_create ((const char *)dest,
|
||||||
dest_sz,
|
dest_sz,
|
||||||
HB_MEMORY_MODE_READONLY,
|
HB_MEMORY_MODE_READONLY,
|
||||||
/* userdata */ nullptr,
|
/* userdata */ nullptr,
|
||||||
free);
|
free);
|
||||||
|
return hb_subset_plan_add_table(plan, HB_OT_TAG_cmap, cmap_prime);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct accelerator_t
|
struct accelerator_t
|
||||||
|
|
|
@ -41,7 +41,7 @@ namespace OT {
|
||||||
#define HB_OT_TAG_hhea HB_TAG('h','h','e','a')
|
#define HB_OT_TAG_hhea HB_TAG('h','h','e','a')
|
||||||
#define HB_OT_TAG_vhea HB_TAG('v','h','e','a')
|
#define HB_OT_TAG_vhea HB_TAG('v','h','e','a')
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
struct _hea
|
struct _hea
|
||||||
{
|
{
|
||||||
inline bool sanitize (hb_sanitize_context_t *c) const
|
inline bool sanitize (hb_sanitize_context_t *c) const
|
||||||
|
@ -84,10 +84,10 @@ struct _hea
|
||||||
DEFINE_SIZE_STATIC (36);
|
DEFINE_SIZE_STATIC (36);
|
||||||
};
|
};
|
||||||
|
|
||||||
struct hhea : _hea {
|
struct hhea : _hea<hhea> {
|
||||||
static const hb_tag_t tableTag = HB_OT_TAG_hhea;
|
static const hb_tag_t tableTag = HB_OT_TAG_hhea;
|
||||||
};
|
};
|
||||||
struct vhea : _hea {
|
struct vhea : _hea<vhea> {
|
||||||
static const hb_tag_t tableTag = HB_OT_TAG_vhea;
|
static const hb_tag_t tableTag = HB_OT_TAG_vhea;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -53,7 +53,7 @@ struct LongMetric
|
||||||
DEFINE_SIZE_STATIC (4);
|
DEFINE_SIZE_STATIC (4);
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename T>
|
template <typename T, typename H>
|
||||||
struct hmtxvmtx
|
struct hmtxvmtx
|
||||||
{
|
{
|
||||||
inline bool sanitize (hb_sanitize_context_t *c) const
|
inline bool sanitize (hb_sanitize_context_t *c) const
|
||||||
|
@ -64,8 +64,129 @@ struct hmtxvmtx
|
||||||
return_trace (true);
|
return_trace (true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
inline bool subset_update_header(hb_subset_plan_t *plan,
|
||||||
|
unsigned int num_hmetrics) const
|
||||||
|
{
|
||||||
|
/* alloc the new table */
|
||||||
|
size_t dest_sz = sizeof(H);
|
||||||
|
void *dest = (void *) calloc(dest_sz, 1);
|
||||||
|
if (unlikely(!dest))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
hb_blob_t *src_blob = OT::Sanitizer<H>().sanitize (plan->source->reference_table (T::tableTag));
|
||||||
|
if (unlikely(!src_blob))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
unsigned int src_length;
|
||||||
|
const char *src_raw = hb_blob_get_data(src_blob, &src_length);
|
||||||
|
|
||||||
|
memcpy(dest, src_raw, src_length);
|
||||||
|
|
||||||
|
H *table = (H *) dest;
|
||||||
|
table->numberOfLongMetrics.set(num_hmetrics);
|
||||||
|
|
||||||
|
hb_blob_t *dest_blob = hb_blob_create ((const char *)dest,
|
||||||
|
dest_sz,
|
||||||
|
HB_MEMORY_MODE_READONLY,
|
||||||
|
/* userdata */ nullptr,
|
||||||
|
free);
|
||||||
|
return hb_subset_plan_add_table(plan, H::tableTag, dest_blob);
|
||||||
|
}
|
||||||
|
|
||||||
|
inline bool subset (hb_subset_plan_t *plan) const
|
||||||
|
{
|
||||||
|
typename T::accelerator_t _mtx;
|
||||||
|
_mtx.init(plan->source);
|
||||||
|
|
||||||
|
/* All the trailing glyphs with the same advance can use one LongMetric
|
||||||
|
* and just keep LSB */
|
||||||
|
hb_prealloced_array_t<hb_codepoint_t> &gids = plan->gids_to_retain_sorted;
|
||||||
|
unsigned int num_advances = gids.len;
|
||||||
|
unsigned int last_advance = _mtx.get_advance(gids[num_advances - 1]);
|
||||||
|
while (num_advances > 1
|
||||||
|
&& last_advance == _mtx.get_advance(gids[num_advances - 2]))
|
||||||
|
{
|
||||||
|
num_advances--;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* alloc the new table */
|
||||||
|
size_t dest_sz = num_advances * 4
|
||||||
|
+ (gids.len - num_advances) * 2;
|
||||||
|
void *dest = (void *) calloc(dest_sz, 1);
|
||||||
|
if (unlikely(!dest))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
DEBUG_MSG(SUBSET, nullptr, "%c%c%c%c in src has %d advances, %d lsbs", HB_UNTAG(T::tableTag), _mtx.num_advances, _mtx.num_metrics - _mtx.num_advances);
|
||||||
|
DEBUG_MSG(SUBSET, nullptr, "%c%c%c%c in dest has %d advances, %d lsbs, %d bytes", HB_UNTAG(T::tableTag), num_advances, gids.len - num_advances, dest_sz);
|
||||||
|
|
||||||
|
const char *source_table = hb_blob_get_data(_mtx.blob, nullptr);
|
||||||
|
// Copy everything over
|
||||||
|
LongMetric * old_metrics = (LongMetric *) source_table;
|
||||||
|
FWORD *lsbs = (FWORD *) (old_metrics + _mtx.num_advances);
|
||||||
|
char * dest_pos = (char *) dest;
|
||||||
|
for (unsigned int i = 0; i < gids.len; i++)
|
||||||
|
{
|
||||||
|
/* the last metric or the one for gids[i] */
|
||||||
|
LongMetric *src_metric = old_metrics + MIN(_mtx.num_advances - 1, gids[i]);
|
||||||
|
if (gids[i] < _mtx.num_advances)
|
||||||
|
{
|
||||||
|
/* src is a LongMetric */
|
||||||
|
if (i < num_advances)
|
||||||
|
{
|
||||||
|
/* dest is a LongMetric, copy it */
|
||||||
|
*((LongMetric *) dest_pos) = *src_metric;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* dest just lsb */
|
||||||
|
*((FWORD *) dest_pos) = src_metric->lsb;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
FWORD src_lsb = *(lsbs + gids[i] - _mtx.num_advances);
|
||||||
|
if (i < num_advances)
|
||||||
|
{
|
||||||
|
/* dest needs a full LongMetric */
|
||||||
|
LongMetric *metric = (LongMetric *)dest_pos;
|
||||||
|
metric->advance = src_metric->advance;
|
||||||
|
metric->lsb = src_lsb;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* dest just needs an lsb */
|
||||||
|
*((FWORD *) dest_pos) = src_lsb;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
dest_pos += (i < num_advances ? 4 : 2);
|
||||||
|
}
|
||||||
|
_mtx.fini();
|
||||||
|
|
||||||
|
// Amend header num hmetrics
|
||||||
|
if (unlikely(!subset_update_header(plan, num_advances)))
|
||||||
|
{
|
||||||
|
free(dest);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
hb_blob_t *result = hb_blob_create ((const char *)dest,
|
||||||
|
dest_sz,
|
||||||
|
HB_MEMORY_MODE_READONLY,
|
||||||
|
/* userdata */ nullptr,
|
||||||
|
free);
|
||||||
|
return hb_subset_plan_add_table(plan, T::tableTag, result);
|
||||||
|
}
|
||||||
|
|
||||||
struct accelerator_t
|
struct accelerator_t
|
||||||
{
|
{
|
||||||
|
friend struct hmtxvmtx;
|
||||||
|
|
||||||
inline void init (hb_face_t *face,
|
inline void init (hb_face_t *face,
|
||||||
unsigned int default_advance_ = 0)
|
unsigned int default_advance_ = 0)
|
||||||
{
|
{
|
||||||
|
@ -87,8 +208,8 @@ struct hmtxvmtx
|
||||||
hb_blob_destroy (os2_blob);
|
hb_blob_destroy (os2_blob);
|
||||||
}
|
}
|
||||||
|
|
||||||
hb_blob_t *_hea_blob = Sanitizer<_hea>().sanitize (face->reference_table (T::headerTag));
|
hb_blob_t *_hea_blob = Sanitizer<H>().sanitize (face->reference_table (H::tableTag));
|
||||||
const _hea *_hea_table = Sanitizer<_hea>::lock_instance (_hea_blob);
|
const H *_hea_table = Sanitizer<H>::lock_instance (_hea_blob);
|
||||||
num_advances = _hea_table->numberOfLongMetrics;
|
num_advances = _hea_table->numberOfLongMetrics;
|
||||||
if (!got_font_extents)
|
if (!got_font_extents)
|
||||||
{
|
{
|
||||||
|
@ -129,22 +250,27 @@ struct hmtxvmtx
|
||||||
hb_blob_destroy (var_blob);
|
hb_blob_destroy (var_blob);
|
||||||
}
|
}
|
||||||
|
|
||||||
inline unsigned int get_advance (hb_codepoint_t glyph,
|
inline unsigned int get_advance (hb_codepoint_t glyph) const
|
||||||
hb_font_t *font) const
|
|
||||||
{
|
{
|
||||||
if (unlikely (glyph >= num_metrics))
|
if (unlikely (glyph >= num_metrics))
|
||||||
{
|
{
|
||||||
/* If num_metrics is zero, it means we don't have the metrics table
|
/* If num_metrics is zero, it means we don't have the metrics table
|
||||||
* for this direction: return default advance. Otherwise, it means that the
|
* for this direction: return default advance. Otherwise, it means that the
|
||||||
* glyph index is out of bound: return zero. */
|
* glyph index is out of bound: return zero. */
|
||||||
if (num_metrics)
|
if (num_metrics)
|
||||||
return 0;
|
return 0;
|
||||||
else
|
else
|
||||||
return default_advance;
|
return default_advance;
|
||||||
}
|
}
|
||||||
|
|
||||||
return table->longMetric[MIN (glyph, (uint32_t) num_advances - 1)].advance
|
return table->longMetric[MIN (glyph, (uint32_t) num_advances - 1)].advance;
|
||||||
+ (font->num_coords ? var_table->get_advance_var (glyph, font->coords, font->num_coords) : 0); // TODO Optimize?!
|
}
|
||||||
|
|
||||||
|
inline unsigned int get_advance (hb_codepoint_t glyph,
|
||||||
|
hb_font_t *font) const
|
||||||
|
{
|
||||||
|
return get_advance(glyph)
|
||||||
|
+ (font->num_coords ? var_table->get_advance_var (glyph, font->coords, font->num_coords) : 0); // TODO Optimize?!
|
||||||
}
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
@ -153,11 +279,12 @@ struct hmtxvmtx
|
||||||
unsigned short descender;
|
unsigned short descender;
|
||||||
unsigned short line_gap;
|
unsigned short line_gap;
|
||||||
|
|
||||||
private:
|
protected:
|
||||||
unsigned int num_metrics;
|
unsigned int num_metrics;
|
||||||
unsigned int num_advances;
|
unsigned int num_advances;
|
||||||
unsigned int default_advance;
|
unsigned int default_advance;
|
||||||
|
|
||||||
|
private:
|
||||||
const hmtxvmtx *table;
|
const hmtxvmtx *table;
|
||||||
hb_blob_t *blob;
|
hb_blob_t *blob;
|
||||||
const HVARVVAR *var_table;
|
const HVARVVAR *var_table;
|
||||||
|
@ -190,15 +317,13 @@ struct hmtxvmtx
|
||||||
DEFINE_SIZE_ARRAY (0, longMetric);
|
DEFINE_SIZE_ARRAY (0, longMetric);
|
||||||
};
|
};
|
||||||
|
|
||||||
struct hmtx : hmtxvmtx<hmtx> {
|
struct hmtx : hmtxvmtx<hmtx, hhea> {
|
||||||
static const hb_tag_t tableTag = HB_OT_TAG_hmtx;
|
static const hb_tag_t tableTag = HB_OT_TAG_hmtx;
|
||||||
static const hb_tag_t headerTag = HB_OT_TAG_hhea;
|
|
||||||
static const hb_tag_t variationsTag = HB_OT_TAG_HVAR;
|
static const hb_tag_t variationsTag = HB_OT_TAG_HVAR;
|
||||||
static const hb_tag_t os2Tag = HB_OT_TAG_os2;
|
static const hb_tag_t os2Tag = HB_OT_TAG_os2;
|
||||||
};
|
};
|
||||||
struct vmtx : hmtxvmtx<vmtx> {
|
struct vmtx : hmtxvmtx<vmtx, vhea> {
|
||||||
static const hb_tag_t tableTag = HB_OT_TAG_vmtx;
|
static const hb_tag_t tableTag = HB_OT_TAG_vmtx;
|
||||||
static const hb_tag_t headerTag = HB_OT_TAG_vhea;
|
|
||||||
static const hb_tag_t variationsTag = HB_OT_TAG_VVAR;
|
static const hb_tag_t variationsTag = HB_OT_TAG_VVAR;
|
||||||
static const hb_tag_t os2Tag = HB_TAG_NONE;
|
static const hb_tag_t os2Tag = HB_TAG_NONE;
|
||||||
};
|
};
|
||||||
|
|
|
@ -61,21 +61,21 @@ struct maxp
|
||||||
(version.major == 0 && version.minor == 0x5000u)));
|
(version.major == 0 && version.minor == 0x5000u)));
|
||||||
}
|
}
|
||||||
|
|
||||||
inline hb_blob_t * subset (hb_subset_plan_t *plan, hb_face_t *source) const
|
inline hb_bool_t subset (hb_subset_plan_t *plan) const
|
||||||
{
|
{
|
||||||
hb_blob_t *maxp_blob = OT::Sanitizer<OT::maxp>().sanitize (hb_face_reference_table (source, HB_OT_TAG_maxp));
|
hb_blob_t *maxp_blob = OT::Sanitizer<OT::maxp>().sanitize (hb_face_reference_table (plan->source, HB_OT_TAG_maxp));
|
||||||
hb_blob_t *maxp_prime_blob = hb_blob_create_sub_blob (maxp_blob, 0, -1);
|
hb_blob_t *maxp_prime_blob = hb_blob_create_sub_blob (maxp_blob, 0, -1);
|
||||||
hb_blob_destroy (maxp_blob);
|
hb_blob_destroy (maxp_blob);
|
||||||
|
|
||||||
OT::maxp *maxp_prime = (OT::maxp *) hb_blob_get_data_writable (maxp_prime_blob, nullptr);
|
OT::maxp *maxp_prime = (OT::maxp *) hb_blob_get_data_writable (maxp_prime_blob, nullptr);
|
||||||
if (unlikely (!maxp_prime)) {
|
if (unlikely (!maxp_prime)) {
|
||||||
hb_blob_destroy (maxp_prime_blob);
|
hb_blob_destroy (maxp_prime_blob);
|
||||||
return nullptr;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
maxp_prime->set_num_glyphs (plan->gids_to_retain_sorted.len);
|
maxp_prime->set_num_glyphs (plan->gids_to_retain_sorted.len);
|
||||||
|
|
||||||
return maxp_prime_blob;
|
return hb_subset_plan_add_table(plan, HB_OT_TAG_maxp, maxp_prime_blob);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* We only implement version 0.5 as none of the extra fields in version 1.0 are useful. */
|
/* We only implement version 0.5 as none of the extra fields in version 1.0 are useful. */
|
||||||
|
|
|
@ -49,16 +49,16 @@ struct os2
|
||||||
return_trace (c->check_struct (this));
|
return_trace (c->check_struct (this));
|
||||||
}
|
}
|
||||||
|
|
||||||
inline hb_blob_t * subset (hb_subset_plan_t *plan, hb_face_t *source) const
|
inline hb_bool_t subset (hb_subset_plan_t *plan) const
|
||||||
{
|
{
|
||||||
hb_blob_t *os2_blob = OT::Sanitizer<OT::os2>().sanitize (hb_face_reference_table (source, HB_OT_TAG_os2));
|
hb_blob_t *os2_blob = OT::Sanitizer<OT::os2>().sanitize (hb_face_reference_table (plan->source, HB_OT_TAG_os2));
|
||||||
hb_blob_t *os2_prime_blob = hb_blob_create_sub_blob (os2_blob, 0, -1);
|
hb_blob_t *os2_prime_blob = hb_blob_create_sub_blob (os2_blob, 0, -1);
|
||||||
hb_blob_destroy (os2_blob);
|
hb_blob_destroy (os2_blob);
|
||||||
|
|
||||||
OT::os2 *os2_prime = (OT::os2 *) hb_blob_get_data_writable (os2_prime_blob, nullptr);
|
OT::os2 *os2_prime = (OT::os2 *) hb_blob_get_data_writable (os2_prime_blob, nullptr);
|
||||||
if (unlikely (!os2_prime)) {
|
if (unlikely (!os2_prime)) {
|
||||||
hb_blob_destroy (os2_prime_blob);
|
hb_blob_destroy (os2_prime_blob);
|
||||||
return nullptr;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint16_t min_cp, max_cp;
|
uint16_t min_cp, max_cp;
|
||||||
|
@ -66,7 +66,7 @@ struct os2
|
||||||
os2_prime->usFirstCharIndex.set (min_cp);
|
os2_prime->usFirstCharIndex.set (min_cp);
|
||||||
os2_prime->usLastCharIndex.set (max_cp);
|
os2_prime->usLastCharIndex.set (max_cp);
|
||||||
|
|
||||||
return os2_prime_blob;
|
return hb_subset_plan_add_table(plan, HB_OT_TAG_os2, os2_prime_blob);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void find_min_and_max_codepoint (hb_subset_plan_t *plan,
|
static inline void find_min_and_max_codepoint (hb_subset_plan_t *plan,
|
||||||
|
|
|
@ -157,16 +157,15 @@ _hb_subset_glyf_and_loca (const OT::glyf::accelerator_t &glyf,
|
||||||
**/
|
**/
|
||||||
bool
|
bool
|
||||||
hb_subset_glyf_and_loca (hb_subset_plan_t *plan,
|
hb_subset_glyf_and_loca (hb_subset_plan_t *plan,
|
||||||
hb_face_t *face,
|
|
||||||
bool *use_short_loca, /* OUT */
|
bool *use_short_loca, /* OUT */
|
||||||
hb_blob_t **glyf_prime, /* OUT */
|
hb_blob_t **glyf_prime, /* OUT */
|
||||||
hb_blob_t **loca_prime /* OUT */)
|
hb_blob_t **loca_prime /* OUT */)
|
||||||
{
|
{
|
||||||
hb_blob_t *glyf_blob = OT::Sanitizer<OT::glyf>().sanitize (face->reference_table (HB_OT_TAG_glyf));
|
hb_blob_t *glyf_blob = OT::Sanitizer<OT::glyf>().sanitize (plan->source->reference_table (HB_OT_TAG_glyf));
|
||||||
const char *glyf_data = hb_blob_get_data(glyf_blob, nullptr);
|
const char *glyf_data = hb_blob_get_data(glyf_blob, nullptr);
|
||||||
|
|
||||||
OT::glyf::accelerator_t glyf;
|
OT::glyf::accelerator_t glyf;
|
||||||
glyf.init(face);
|
glyf.init(plan->source);
|
||||||
bool result = _hb_subset_glyf_and_loca (glyf,
|
bool result = _hb_subset_glyf_and_loca (glyf,
|
||||||
glyf_data,
|
glyf_data,
|
||||||
plan->gids_to_retain_sorted,
|
plan->gids_to_retain_sorted,
|
||||||
|
|
|
@ -33,7 +33,6 @@
|
||||||
|
|
||||||
HB_INTERNAL bool
|
HB_INTERNAL bool
|
||||||
hb_subset_glyf_and_loca (hb_subset_plan_t *plan,
|
hb_subset_glyf_and_loca (hb_subset_plan_t *plan,
|
||||||
hb_face_t *face,
|
|
||||||
bool *use_short_loca, /* OUT */
|
bool *use_short_loca, /* OUT */
|
||||||
hb_blob_t **glyf_prime /* OUT */,
|
hb_blob_t **glyf_prime /* OUT */,
|
||||||
hb_blob_t **loca_prime /* OUT */);
|
hb_blob_t **loca_prime /* OUT */);
|
||||||
|
|
|
@ -72,6 +72,14 @@ hb_subset_plan_new_gid_for_old_id (hb_subset_plan_t *plan,
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
hb_bool_t
|
||||||
|
hb_subset_plan_add_table (hb_subset_plan_t *plan,
|
||||||
|
hb_tag_t tag,
|
||||||
|
hb_blob_t *contents)
|
||||||
|
{
|
||||||
|
return hb_subset_face_add_table(plan->dest, tag, contents);
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
_populate_codepoints (hb_set_t *input_codepoints,
|
_populate_codepoints (hb_set_t *input_codepoints,
|
||||||
hb_prealloced_array_t<hb_codepoint_t>& plan_codepoints)
|
hb_prealloced_array_t<hb_codepoint_t>& plan_codepoints)
|
||||||
|
@ -110,6 +118,7 @@ _populate_gids_to_retain (hb_face_t *face,
|
||||||
*(old_gids.push ()) = gid;
|
*(old_gids.push ()) = gid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Generally there shouldn't be any */
|
||||||
while (bad_indices.len > 0) {
|
while (bad_indices.len > 0) {
|
||||||
unsigned int i = bad_indices[bad_indices.len - 1];
|
unsigned int i = bad_indices[bad_indices.len - 1];
|
||||||
bad_indices.pop ();
|
bad_indices.pop ();
|
||||||
|
@ -154,12 +163,15 @@ hb_subset_plan_create (hb_face_t *face,
|
||||||
plan->codepoints.init();
|
plan->codepoints.init();
|
||||||
plan->gids_to_retain.init();
|
plan->gids_to_retain.init();
|
||||||
plan->gids_to_retain_sorted.init();
|
plan->gids_to_retain_sorted.init();
|
||||||
|
plan->source = face;
|
||||||
|
plan->dest = hb_subset_face_create ();
|
||||||
|
|
||||||
_populate_codepoints (input->unicodes, plan->codepoints);
|
_populate_codepoints (input->unicodes, plan->codepoints);
|
||||||
_populate_gids_to_retain (face,
|
_populate_gids_to_retain (face,
|
||||||
plan->codepoints,
|
plan->codepoints,
|
||||||
plan->gids_to_retain,
|
plan->gids_to_retain,
|
||||||
plan->gids_to_retain_sorted);
|
plan->gids_to_retain_sorted);
|
||||||
|
|
||||||
return plan;
|
return plan;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -38,9 +38,14 @@ struct hb_subset_plan_t {
|
||||||
// TODO(Q1) actual map, drop this crap
|
// TODO(Q1) actual map, drop this crap
|
||||||
// Look at me ma, I'm a poor mans map codepoint : new gid
|
// Look at me ma, I'm a poor mans map codepoint : new gid
|
||||||
// codepoints is sorted and aligned with gids_to_retain.
|
// codepoints is sorted and aligned with gids_to_retain.
|
||||||
|
// TODO Also you should init/fini those arrays
|
||||||
hb_prealloced_array_t<hb_codepoint_t> codepoints;
|
hb_prealloced_array_t<hb_codepoint_t> codepoints;
|
||||||
hb_prealloced_array_t<hb_codepoint_t> gids_to_retain;
|
hb_prealloced_array_t<hb_codepoint_t> gids_to_retain;
|
||||||
hb_prealloced_array_t<hb_codepoint_t> gids_to_retain_sorted;
|
hb_prealloced_array_t<hb_codepoint_t> gids_to_retain_sorted;
|
||||||
|
|
||||||
|
// Plan is only good for a specific source/dest so keep them with it
|
||||||
|
hb_face_t *source;
|
||||||
|
hb_face_t *dest;
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef struct hb_subset_plan_t hb_subset_plan_t;
|
typedef struct hb_subset_plan_t hb_subset_plan_t;
|
||||||
|
@ -60,6 +65,11 @@ hb_subset_plan_new_gid_for_codepoint(hb_subset_plan_t *plan,
|
||||||
hb_codepoint_t codepont,
|
hb_codepoint_t codepont,
|
||||||
hb_codepoint_t *new_gid /* OUT */);
|
hb_codepoint_t *new_gid /* OUT */);
|
||||||
|
|
||||||
|
HB_INTERNAL hb_bool_t
|
||||||
|
hb_subset_plan_add_table(hb_subset_plan_t *plan,
|
||||||
|
hb_tag_t tag,
|
||||||
|
hb_blob_t *contents);
|
||||||
|
|
||||||
HB_INTERNAL void
|
HB_INTERNAL void
|
||||||
hb_subset_plan_destroy (hb_subset_plan_t *plan);
|
hb_subset_plan_destroy (hb_subset_plan_t *plan);
|
||||||
|
|
||||||
|
|
|
@ -34,6 +34,8 @@
|
||||||
|
|
||||||
#include "hb-font-private.hh"
|
#include "hb-font-private.hh"
|
||||||
|
|
||||||
|
typedef struct hb_subset_face_data_t hb_subset_face_data_t;
|
||||||
|
|
||||||
struct hb_subset_input_t {
|
struct hb_subset_input_t {
|
||||||
hb_object_header_t header;
|
hb_object_header_t header;
|
||||||
ASSERT_POD ();
|
ASSERT_POD ();
|
||||||
|
@ -50,4 +52,13 @@ struct hb_subset_input_t {
|
||||||
*/
|
*/
|
||||||
};
|
};
|
||||||
|
|
||||||
|
HB_INTERNAL hb_face_t *
|
||||||
|
hb_subset_face_create (void);
|
||||||
|
|
||||||
|
HB_INTERNAL hb_bool_t
|
||||||
|
hb_subset_face_add_table (hb_face_t *face, hb_tag_t tag, hb_blob_t *blob);
|
||||||
|
|
||||||
|
HB_INTERNAL void
|
||||||
|
hb_subset_face_data_destroy (void *user_data);
|
||||||
|
|
||||||
#endif /* HB_SUBSET_PRIVATE_HH */
|
#endif /* HB_SUBSET_PRIVATE_HH */
|
||||||
|
|
|
@ -21,12 +21,14 @@
|
||||||
* ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
|
* ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
|
||||||
* PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
|
* PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
|
||||||
*
|
*
|
||||||
* Google Author(s): Garret Rieger, Rod Sheeter, Behdad Esfahbod
|
* Google Author(s): Garret Rieger, Rod Sheeter
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "hb-object-private.hh"
|
#include "hb-object-private.hh"
|
||||||
#include "hb-open-type-private.hh"
|
#include "hb-open-type-private.hh"
|
||||||
|
|
||||||
|
#include "hb-private.hh"
|
||||||
|
|
||||||
#include "hb-subset-glyf.hh"
|
#include "hb-subset-glyf.hh"
|
||||||
#include "hb-subset-private.hh"
|
#include "hb-subset-private.hh"
|
||||||
#include "hb-subset-plan.hh"
|
#include "hb-subset-plan.hh"
|
||||||
|
@ -35,6 +37,8 @@
|
||||||
#include "hb-ot-cmap-table.hh"
|
#include "hb-ot-cmap-table.hh"
|
||||||
#include "hb-ot-glyf-table.hh"
|
#include "hb-ot-glyf-table.hh"
|
||||||
#include "hb-ot-head-table.hh"
|
#include "hb-ot-head-table.hh"
|
||||||
|
#include "hb-ot-hhea-table.hh"
|
||||||
|
#include "hb-ot-hmtx-table.hh"
|
||||||
#include "hb-ot-maxp-table.hh"
|
#include "hb-ot-maxp-table.hh"
|
||||||
#include "hb-ot-os2-table.hh"
|
#include "hb-ot-os2-table.hh"
|
||||||
|
|
||||||
|
@ -76,13 +80,13 @@ hb_subset_profile_destroy (hb_subset_profile_t *profile)
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename TableType>
|
template<typename TableType>
|
||||||
static hb_blob_t *
|
static hb_bool_t
|
||||||
_subset (hb_subset_plan_t *plan, hb_face_t *source)
|
_subset (hb_subset_plan_t *plan)
|
||||||
{
|
{
|
||||||
OT::Sanitizer<TableType> sanitizer;
|
OT::Sanitizer<TableType> sanitizer;
|
||||||
hb_blob_t *source_blob = sanitizer.sanitize (source->reference_table (TableType::tableTag));
|
hb_blob_t *source_blob = sanitizer.sanitize (plan->source->reference_table (TableType::tableTag));
|
||||||
const TableType *table = OT::Sanitizer<TableType>::lock_instance (source_blob);
|
const TableType *table = OT::Sanitizer<TableType>::lock_instance (source_blob);
|
||||||
hb_blob_t *result = table->subset(plan, source);
|
hb_bool_t result = table->subset(plan);
|
||||||
|
|
||||||
hb_blob_destroy (source_blob);
|
hb_blob_destroy (source_blob);
|
||||||
|
|
||||||
|
@ -124,8 +128,8 @@ _hb_subset_face_data_create (void)
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
void
|
||||||
_hb_subset_face_data_destroy (void *user_data)
|
hb_subset_face_data_destroy (void *user_data)
|
||||||
{
|
{
|
||||||
hb_subset_face_data_t *data = (hb_subset_face_data_t *) user_data;
|
hb_subset_face_data_t *data = (hb_subset_face_data_t *) user_data;
|
||||||
|
|
||||||
|
@ -188,7 +192,7 @@ _hb_subset_face_reference_table (hb_face_t *face, hb_tag_t tag, void *user_data)
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
static hb_face_t *
|
hb_face_t *
|
||||||
hb_subset_face_create (void)
|
hb_subset_face_create (void)
|
||||||
{
|
{
|
||||||
hb_subset_face_data_t *data = _hb_subset_face_data_create ();
|
hb_subset_face_data_t *data = _hb_subset_face_data_create ();
|
||||||
|
@ -196,13 +200,13 @@ hb_subset_face_create (void)
|
||||||
|
|
||||||
return hb_face_create_for_tables (_hb_subset_face_reference_table,
|
return hb_face_create_for_tables (_hb_subset_face_reference_table,
|
||||||
data,
|
data,
|
||||||
_hb_subset_face_data_destroy);
|
hb_subset_face_data_destroy);
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool
|
hb_bool_t
|
||||||
hb_subset_face_add_table (hb_face_t *face, hb_tag_t tag, hb_blob_t *blob)
|
hb_subset_face_add_table (hb_face_t *face, hb_tag_t tag, hb_blob_t *blob)
|
||||||
{
|
{
|
||||||
if (unlikely (face->destroy != _hb_subset_face_data_destroy))
|
if (unlikely (face->destroy != hb_subset_face_data_destroy))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
hb_subset_face_data_t *data = (hb_subset_face_data_t *) face->user_data;
|
hb_subset_face_data_t *data = (hb_subset_face_data_t *) face->user_data;
|
||||||
|
@ -221,7 +225,7 @@ _add_head_and_set_loca_version (hb_face_t *source, bool use_short_loca, hb_face_
|
||||||
{
|
{
|
||||||
hb_blob_t *head_blob = OT::Sanitizer<OT::head>().sanitize (hb_face_reference_table (source, HB_OT_TAG_head));
|
hb_blob_t *head_blob = OT::Sanitizer<OT::head>().sanitize (hb_face_reference_table (source, HB_OT_TAG_head));
|
||||||
const OT::head *head = OT::Sanitizer<OT::head>::lock_instance (head_blob);
|
const OT::head *head = OT::Sanitizer<OT::head>::lock_instance (head_blob);
|
||||||
bool has_head = (head != nullptr);
|
hb_bool_t has_head = (head != nullptr);
|
||||||
|
|
||||||
if (has_head) {
|
if (has_head) {
|
||||||
OT::head *head_prime = (OT::head *) calloc (OT::head::static_size, 1);
|
OT::head *head_prime = (OT::head *) calloc (OT::head::static_size, 1);
|
||||||
|
@ -242,7 +246,7 @@ _add_head_and_set_loca_version (hb_face_t *source, bool use_short_loca, hb_face_
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool
|
static bool
|
||||||
_subset_glyf (hb_subset_plan_t *plan, hb_face_t *source, hb_face_t *dest)
|
_subset_glyf (hb_subset_plan_t *plan)
|
||||||
{
|
{
|
||||||
hb_blob_t *glyf_prime = nullptr;
|
hb_blob_t *glyf_prime = nullptr;
|
||||||
hb_blob_t *loca_prime = nullptr;
|
hb_blob_t *loca_prime = nullptr;
|
||||||
|
@ -250,10 +254,10 @@ _subset_glyf (hb_subset_plan_t *plan, hb_face_t *source, hb_face_t *dest)
|
||||||
bool success = true;
|
bool success = true;
|
||||||
bool use_short_loca = false;
|
bool use_short_loca = false;
|
||||||
// TODO(grieger): Migrate to subset function on the table like cmap.
|
// TODO(grieger): Migrate to subset function on the table like cmap.
|
||||||
if (hb_subset_glyf_and_loca (plan, source, &use_short_loca, &glyf_prime, &loca_prime)) {
|
if (hb_subset_glyf_and_loca (plan, &use_short_loca, &glyf_prime, &loca_prime)) {
|
||||||
success = success && hb_subset_face_add_table (dest, HB_OT_TAG_glyf, glyf_prime);
|
success = success && hb_subset_plan_add_table (plan, HB_OT_TAG_glyf, glyf_prime);
|
||||||
success = success && hb_subset_face_add_table (dest, HB_OT_TAG_loca, loca_prime);
|
success = success && hb_subset_plan_add_table (plan, HB_OT_TAG_loca, loca_prime);
|
||||||
success = success && _add_head_and_set_loca_version (source, use_short_loca, dest);
|
success = success && _add_head_and_set_loca_version (plan->source, use_short_loca, plan->dest);
|
||||||
} else {
|
} else {
|
||||||
success = false;
|
success = false;
|
||||||
}
|
}
|
||||||
|
@ -265,39 +269,50 @@ _subset_glyf (hb_subset_plan_t *plan, hb_face_t *source, hb_face_t *dest)
|
||||||
|
|
||||||
static bool
|
static bool
|
||||||
_subset_table (hb_subset_plan_t *plan,
|
_subset_table (hb_subset_plan_t *plan,
|
||||||
hb_face_t *source,
|
hb_tag_t tag)
|
||||||
hb_tag_t tag,
|
|
||||||
hb_blob_t *source_blob,
|
|
||||||
hb_face_t *dest)
|
|
||||||
{
|
{
|
||||||
// TODO (grieger): Handle updating the head table (loca format + num glyphs)
|
// TODO (grieger): Handle updating the head table (loca format + num glyphs)
|
||||||
DEBUG_MSG(SUBSET, nullptr, "begin subset %c%c%c%c", HB_UNTAG(tag));
|
DEBUG_MSG(SUBSET, nullptr, "begin subset %c%c%c%c", HB_UNTAG(tag));
|
||||||
hb_blob_t *dest_blob;
|
bool result = true;
|
||||||
switch (tag) {
|
switch (tag) {
|
||||||
case HB_OT_TAG_glyf:
|
case HB_OT_TAG_glyf:
|
||||||
return _subset_glyf (plan, source, dest);
|
result = _subset_glyf (plan);
|
||||||
|
break;
|
||||||
case HB_OT_TAG_head:
|
case HB_OT_TAG_head:
|
||||||
// SKIP head, it's handled by glyf
|
// SKIP head, it's handled by glyf
|
||||||
|
result = true;
|
||||||
|
break;
|
||||||
|
case HB_OT_TAG_hhea:
|
||||||
|
// SKIP hhea, it's handled by hmtx
|
||||||
return true;
|
return true;
|
||||||
|
case HB_OT_TAG_hmtx:
|
||||||
|
result = _subset<const OT::hmtx> (plan);
|
||||||
|
break;
|
||||||
case HB_OT_TAG_maxp:
|
case HB_OT_TAG_maxp:
|
||||||
dest_blob = _subset<const OT::maxp> (plan, source);
|
result = _subset<const OT::maxp> (plan);
|
||||||
break;
|
break;
|
||||||
case HB_OT_TAG_loca:
|
case HB_OT_TAG_loca:
|
||||||
// SKIP loca, it's handle by glyf
|
// SKIP loca, it's handle by glyf
|
||||||
return true;
|
return true;
|
||||||
case HB_OT_TAG_cmap:
|
case HB_OT_TAG_cmap:
|
||||||
dest_blob = _subset<const OT::cmap> (plan, source);
|
result = _subset<const OT::cmap> (plan);
|
||||||
break;
|
break;
|
||||||
case HB_OT_TAG_os2:
|
case HB_OT_TAG_os2:
|
||||||
dest_blob = _subset<const OT::os2> (plan, source);
|
result = _subset<const OT::os2> (plan);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
dest_blob = source_blob;
|
hb_blob_t *source_table = hb_face_reference_table(plan->source, tag);
|
||||||
|
if (likely(source_table))
|
||||||
|
{
|
||||||
|
result = hb_subset_plan_add_table(plan, tag, source_table);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
result = false;
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
DEBUG_MSG(SUBSET, nullptr, "subset %c%c%c%c %s", HB_UNTAG(tag), dest_blob ? "ok" : "FAILED");
|
DEBUG_MSG(SUBSET, nullptr, "subset %c%c%c%c %s", HB_UNTAG(tag), result ? "ok" : "FAILED");
|
||||||
if (unlikely(!dest_blob)) return false;
|
|
||||||
if (unlikely(!hb_subset_face_add_table (dest, tag, dest_blob))) return false;
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -325,14 +340,13 @@ _should_drop_table(hb_tag_t tag)
|
||||||
**/
|
**/
|
||||||
hb_face_t *
|
hb_face_t *
|
||||||
hb_subset (hb_face_t *source,
|
hb_subset (hb_face_t *source,
|
||||||
hb_subset_profile_t *profile,
|
hb_subset_profile_t *profile,
|
||||||
hb_subset_input_t *input)
|
hb_subset_input_t *input)
|
||||||
{
|
{
|
||||||
if (unlikely (!profile || !input || !source)) return hb_face_get_empty();
|
if (unlikely (!profile || !input || !source)) return hb_face_get_empty();
|
||||||
|
|
||||||
hb_subset_plan_t *plan = hb_subset_plan_create (source, profile, input);
|
hb_subset_plan_t *plan = hb_subset_plan_create (source, profile, input);
|
||||||
|
|
||||||
hb_face_t *dest = hb_subset_face_create ();
|
|
||||||
hb_tag_t table_tags[32];
|
hb_tag_t table_tags[32];
|
||||||
unsigned int offset = 0, count;
|
unsigned int offset = 0, count;
|
||||||
bool success = true;
|
bool success = true;
|
||||||
|
@ -347,12 +361,11 @@ hb_subset (hb_face_t *source,
|
||||||
DEBUG_MSG(SUBSET, nullptr, "drop %c%c%c%c", HB_UNTAG(tag));
|
DEBUG_MSG(SUBSET, nullptr, "drop %c%c%c%c", HB_UNTAG(tag));
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
hb_blob_t *blob = hb_face_reference_table (source, tag);
|
success = success && _subset_table (plan, tag);
|
||||||
success = success && _subset_table (plan, source, tag, blob, dest);
|
|
||||||
hb_blob_destroy (blob);
|
|
||||||
}
|
}
|
||||||
} while (count == ARRAY_LENGTH (table_tags));
|
} while (count == ARRAY_LENGTH (table_tags));
|
||||||
|
|
||||||
|
hb_face_t *result = success ? hb_face_reference(plan->dest) : hb_face_get_empty();
|
||||||
hb_subset_plan_destroy (plan);
|
hb_subset_plan_destroy (plan);
|
||||||
return success ? dest : hb_face_get_empty ();
|
return result;
|
||||||
}
|
}
|
||||||
|
|
|
@ -31,6 +31,7 @@ TEST_PROGS = \
|
||||||
test-shape \
|
test-shape \
|
||||||
test-subset-cmap \
|
test-subset-cmap \
|
||||||
test-subset-glyf \
|
test-subset-glyf \
|
||||||
|
test-subset-hmtx \
|
||||||
test-subset-os2 \
|
test-subset-os2 \
|
||||||
test-unicode \
|
test-unicode \
|
||||||
test-version \
|
test-version \
|
||||||
|
@ -39,6 +40,7 @@ TEST_PROGS = \
|
||||||
test_subset_LDADD = $(LDADD) $(top_builddir)/src/libharfbuzz-subset.la
|
test_subset_LDADD = $(LDADD) $(top_builddir)/src/libharfbuzz-subset.la
|
||||||
test_subset_cmap_LDADD = $(LDADD) $(top_builddir)/src/libharfbuzz-subset.la
|
test_subset_cmap_LDADD = $(LDADD) $(top_builddir)/src/libharfbuzz-subset.la
|
||||||
test_subset_glyf_LDADD = $(LDADD) $(top_builddir)/src/libharfbuzz-subset.la
|
test_subset_glyf_LDADD = $(LDADD) $(top_builddir)/src/libharfbuzz-subset.la
|
||||||
|
test_subset_hmtx_LDADD = $(LDADD) $(top_builddir)/src/libharfbuzz-subset.la
|
||||||
test_subset_os2_LDADD = $(LDADD) $(top_builddir)/src/libharfbuzz-subset.la
|
test_subset_os2_LDADD = $(LDADD) $(top_builddir)/src/libharfbuzz-subset.la
|
||||||
|
|
||||||
test_unicode_CPPFLAGS = \
|
test_unicode_CPPFLAGS = \
|
||||||
|
@ -65,6 +67,11 @@ TEST_PROGS += \
|
||||||
test_ot_math_LDADD = $(LDADD) $(FREETYPE_LIBS)
|
test_ot_math_LDADD = $(LDADD) $(FREETYPE_LIBS)
|
||||||
test_ot_math_CPPFLAGS = $(AM_CPPFLAGS) $(FREETYPE_CFLAGS)
|
test_ot_math_CPPFLAGS = $(AM_CPPFLAGS) $(FREETYPE_CFLAGS)
|
||||||
EXTRA_DIST += \
|
EXTRA_DIST += \
|
||||||
|
fonts/Inconsolata-Regular.ab.ttf \
|
||||||
|
fonts/Inconsolata-Regular.abc.ttf \
|
||||||
|
fonts/Inconsolata-Regular.abc.widerc.ttf \
|
||||||
|
fonts/Inconsolata-Regular.ac.ttf \
|
||||||
|
fonts/Inconsolata-Regular.ac.widerc.ttf \
|
||||||
fonts/Roboto-Regular.abc.ttf \
|
fonts/Roboto-Regular.abc.ttf \
|
||||||
fonts/Roboto-Regular.ac.ttf \
|
fonts/Roboto-Regular.ac.ttf \
|
||||||
fonts/MathTestFontEmpty.otf \
|
fonts/MathTestFontEmpty.otf \
|
||||||
|
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
|
@ -1 +1,3 @@
|
||||||
cmap-format12-only files created by ttx & remove all other cmap entries
|
cmap-format12-only files created by ttx & remove all other cmap entries
|
||||||
|
|
||||||
|
Inconsolata-Regular.abc.widerc.ttf has the hmtx width of "c" set to 600; everything else is 500. Subsetting out c should reduce numberOfHMetrics to 1.
|
||||||
|
|
|
@ -0,0 +1,164 @@
|
||||||
|
/*
|
||||||
|
* Copyright © 2018 Google, Inc.
|
||||||
|
*
|
||||||
|
* This is part of HarfBuzz, a text shaping library.
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, without written agreement and without
|
||||||
|
* license or royalty fees, to use, copy, modify, and distribute this
|
||||||
|
* software and its documentation for any purpose, provided that the
|
||||||
|
* above copyright notice and the following two paragraphs appear in
|
||||||
|
* all copies of this software.
|
||||||
|
*
|
||||||
|
* IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
|
||||||
|
* DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
|
||||||
|
* IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
|
||||||
|
* DAMAGE.
|
||||||
|
*
|
||||||
|
* THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
|
||||||
|
* BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS
|
||||||
|
* ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
|
||||||
|
* PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
|
||||||
|
*
|
||||||
|
* Google Author(s): Roderick Sheeter
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdbool.h>
|
||||||
|
|
||||||
|
#include "hb-test.h"
|
||||||
|
#include "hb-subset-test.h"
|
||||||
|
|
||||||
|
/* Unit tests for hmtx subsetting */
|
||||||
|
|
||||||
|
static void check_num_hmetrics(hb_face_t *face, uint16_t expected_num_hmetrics)
|
||||||
|
{
|
||||||
|
hb_blob_t *hhea_blob = hb_face_reference_table (face, HB_TAG ('h','h','e','a'));
|
||||||
|
hb_blob_t *hmtx_blob = hb_face_reference_table (face, HB_TAG ('h','m','t','x'));
|
||||||
|
|
||||||
|
// TODO I sure wish I could just use the hmtx table struct!
|
||||||
|
unsigned int hmtx_len = hb_blob_get_length(hmtx_blob);
|
||||||
|
unsigned int hhea_len;
|
||||||
|
uint8_t *raw_hhea = (uint8_t *) hb_blob_get_data(hhea_blob, &hhea_len);
|
||||||
|
uint16_t num_hmetrics = (raw_hhea[hhea_len - 2] << 8) + raw_hhea[hhea_len - 1];
|
||||||
|
g_assert_cmpuint(expected_num_hmetrics, ==, num_hmetrics);
|
||||||
|
|
||||||
|
hb_blob_destroy (hhea_blob);
|
||||||
|
hb_blob_destroy (hmtx_blob);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
test_subset_hmtx_simple_subset (void)
|
||||||
|
{
|
||||||
|
hb_face_t *face_abc = hb_subset_test_open_font ("fonts/Roboto-Regular.abc.ttf");
|
||||||
|
hb_face_t *face_ac = hb_subset_test_open_font ("fonts/Roboto-Regular.ac.ttf");
|
||||||
|
|
||||||
|
hb_set_t *codepoints = hb_set_create ();
|
||||||
|
hb_set_add (codepoints, 'a');
|
||||||
|
hb_set_add (codepoints, 'c');
|
||||||
|
hb_face_t *face_abc_subset = hb_subset_test_create_subset (face_abc, codepoints);
|
||||||
|
hb_set_destroy (codepoints);
|
||||||
|
|
||||||
|
check_num_hmetrics(face_abc_subset, 3); /* nothing has same width */
|
||||||
|
hb_subset_test_check (face_ac, face_abc_subset, HB_TAG ('h','m','t','x'));
|
||||||
|
|
||||||
|
hb_face_destroy (face_abc_subset);
|
||||||
|
hb_face_destroy (face_abc);
|
||||||
|
hb_face_destroy (face_ac);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void
|
||||||
|
test_subset_hmtx_monospace (void)
|
||||||
|
{
|
||||||
|
hb_face_t *face_abc = hb_subset_test_open_font ("fonts/Inconsolata-Regular.abc.ttf");
|
||||||
|
hb_face_t *face_ac = hb_subset_test_open_font ("fonts/Inconsolata-Regular.ac.ttf");
|
||||||
|
|
||||||
|
hb_set_t *codepoints = hb_set_create ();
|
||||||
|
hb_set_add (codepoints, 'a');
|
||||||
|
hb_set_add (codepoints, 'c');
|
||||||
|
hb_face_t *face_abc_subset = hb_subset_test_create_subset (face_abc, codepoints);
|
||||||
|
hb_set_destroy (codepoints);
|
||||||
|
|
||||||
|
check_num_hmetrics(face_abc_subset, 1); /* everything has same width */
|
||||||
|
hb_subset_test_check (face_ac, face_abc_subset, HB_TAG ('h','m','t','x'));
|
||||||
|
// TODO hhea
|
||||||
|
|
||||||
|
hb_face_destroy (face_abc_subset);
|
||||||
|
hb_face_destroy (face_abc);
|
||||||
|
hb_face_destroy (face_ac);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void
|
||||||
|
test_subset_hmtx_keep_num_metrics (void)
|
||||||
|
{
|
||||||
|
hb_face_t *face_abc = hb_subset_test_open_font ("fonts/Inconsolata-Regular.abc.widerc.ttf");
|
||||||
|
hb_face_t *face_ac = hb_subset_test_open_font ("fonts/Inconsolata-Regular.ac.widerc.ttf");
|
||||||
|
|
||||||
|
hb_set_t *codepoints = hb_set_create ();
|
||||||
|
hb_set_add (codepoints, 'a');
|
||||||
|
hb_set_add (codepoints, 'c');
|
||||||
|
hb_face_t *face_abc_subset = hb_subset_test_create_subset (face_abc, codepoints);
|
||||||
|
hb_set_destroy (codepoints);
|
||||||
|
|
||||||
|
check_num_hmetrics(face_abc_subset, 3); /* c is wider */
|
||||||
|
hb_subset_test_check (face_ac, face_abc_subset, HB_TAG ('h','m','t','x'));
|
||||||
|
|
||||||
|
hb_face_destroy (face_abc_subset);
|
||||||
|
hb_face_destroy (face_abc);
|
||||||
|
hb_face_destroy (face_ac);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
test_subset_hmtx_decrease_num_metrics (void)
|
||||||
|
{
|
||||||
|
hb_face_t *face_abc = hb_subset_test_open_font ("fonts/Inconsolata-Regular.abc.widerc.ttf");
|
||||||
|
hb_face_t *face_ab = hb_subset_test_open_font ("fonts/Inconsolata-Regular.ab.ttf");
|
||||||
|
|
||||||
|
hb_set_t *codepoints = hb_set_create ();
|
||||||
|
hb_set_add (codepoints, 'a');
|
||||||
|
hb_set_add (codepoints, 'b');
|
||||||
|
hb_face_t *face_abc_subset = hb_subset_test_create_subset (face_abc, codepoints);
|
||||||
|
hb_set_destroy (codepoints);
|
||||||
|
|
||||||
|
check_num_hmetrics(face_abc_subset, 1); /* everything left has same width */
|
||||||
|
hb_subset_test_check (face_ab, face_abc_subset, HB_TAG ('h','m','t','x'));
|
||||||
|
|
||||||
|
hb_face_destroy (face_abc_subset);
|
||||||
|
hb_face_destroy (face_abc);
|
||||||
|
hb_face_destroy (face_ab);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
test_subset_hmtx_noop (void)
|
||||||
|
{
|
||||||
|
hb_face_t *face_abc = hb_subset_test_open_font("fonts/Roboto-Regular.abc.ttf");
|
||||||
|
|
||||||
|
hb_set_t *codepoints = hb_set_create();
|
||||||
|
hb_set_add (codepoints, 'a');
|
||||||
|
hb_set_add (codepoints, 'b');
|
||||||
|
hb_set_add (codepoints, 'c');
|
||||||
|
hb_face_t *face_abc_subset = hb_subset_test_create_subset (face_abc, codepoints);
|
||||||
|
hb_set_destroy (codepoints);
|
||||||
|
|
||||||
|
check_num_hmetrics(face_abc_subset, 4); /* nothing has same width */
|
||||||
|
hb_subset_test_check (face_abc, face_abc_subset, HB_TAG ('h','m','t','x'));
|
||||||
|
|
||||||
|
hb_face_destroy (face_abc_subset);
|
||||||
|
hb_face_destroy (face_abc);
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
main (int argc, char **argv)
|
||||||
|
{
|
||||||
|
hb_test_init (&argc, &argv);
|
||||||
|
|
||||||
|
hb_test_add (test_subset_hmtx_simple_subset);
|
||||||
|
hb_test_add (test_subset_hmtx_monospace);
|
||||||
|
hb_test_add (test_subset_hmtx_keep_num_metrics);
|
||||||
|
hb_test_add (test_subset_hmtx_decrease_num_metrics);
|
||||||
|
hb_test_add (test_subset_hmtx_noop);
|
||||||
|
|
||||||
|
return hb_test_run();
|
||||||
|
}
|
Loading…
Reference in New Issue