1*2d1272b8SAndroid Build Coastguard Worker /*
2*2d1272b8SAndroid Build Coastguard Worker * Copyright © 2018 Google, Inc.
3*2d1272b8SAndroid Build Coastguard Worker *
4*2d1272b8SAndroid Build Coastguard Worker * This is part of HarfBuzz, a text shaping library.
5*2d1272b8SAndroid Build Coastguard Worker *
6*2d1272b8SAndroid Build Coastguard Worker * Permission is hereby granted, without written agreement and without
7*2d1272b8SAndroid Build Coastguard Worker * license or royalty fees, to use, copy, modify, and distribute this
8*2d1272b8SAndroid Build Coastguard Worker * software and its documentation for any purpose, provided that the
9*2d1272b8SAndroid Build Coastguard Worker * above copyright notice and the following two paragraphs appear in
10*2d1272b8SAndroid Build Coastguard Worker * all copies of this software.
11*2d1272b8SAndroid Build Coastguard Worker *
12*2d1272b8SAndroid Build Coastguard Worker * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
13*2d1272b8SAndroid Build Coastguard Worker * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
14*2d1272b8SAndroid Build Coastguard Worker * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
15*2d1272b8SAndroid Build Coastguard Worker * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
16*2d1272b8SAndroid Build Coastguard Worker * DAMAGE.
17*2d1272b8SAndroid Build Coastguard Worker *
18*2d1272b8SAndroid Build Coastguard Worker * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
19*2d1272b8SAndroid Build Coastguard Worker * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
20*2d1272b8SAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS
21*2d1272b8SAndroid Build Coastguard Worker * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
22*2d1272b8SAndroid Build Coastguard Worker * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
23*2d1272b8SAndroid Build Coastguard Worker *
24*2d1272b8SAndroid Build Coastguard Worker * Google Author(s): Garret Rieger, Rod Sheeter, Behdad Esfahbod
25*2d1272b8SAndroid Build Coastguard Worker */
26*2d1272b8SAndroid Build Coastguard Worker
27*2d1272b8SAndroid Build Coastguard Worker #include "hb.hh"
28*2d1272b8SAndroid Build Coastguard Worker #include "hb-open-type.hh"
29*2d1272b8SAndroid Build Coastguard Worker
30*2d1272b8SAndroid Build Coastguard Worker #include "hb-subset.hh"
31*2d1272b8SAndroid Build Coastguard Worker
32*2d1272b8SAndroid Build Coastguard Worker #include "hb-open-file.hh"
33*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-cmap-table.hh"
34*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-glyf-table.hh"
35*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-hdmx-table.hh"
36*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-head-table.hh"
37*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-hhea-table.hh"
38*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-hmtx-table.hh"
39*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-maxp-table.hh"
40*2d1272b8SAndroid Build Coastguard Worker #include "OT/Color/CBDT/CBDT.hh"
41*2d1272b8SAndroid Build Coastguard Worker #include "OT/Color/COLR/COLR.hh"
42*2d1272b8SAndroid Build Coastguard Worker #include "OT/Color/CPAL/CPAL.hh"
43*2d1272b8SAndroid Build Coastguard Worker #include "OT/Color/sbix/sbix.hh"
44*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-os2-table.hh"
45*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-post-table.hh"
46*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-post-table-v2subset.hh"
47*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-cff1-table.hh"
48*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-cff2-table.hh"
49*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-vorg-table.hh"
50*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-name-table.hh"
51*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-layout-base-table.hh"
52*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-layout-gsub-table.hh"
53*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-layout-gpos-table.hh"
54*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-var-avar-table.hh"
55*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-var-cvar-table.hh"
56*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-var-fvar-table.hh"
57*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-var-gvar-table.hh"
58*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-var-hvar-table.hh"
59*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-var-mvar-table.hh"
60*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-math-table.hh"
61*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-stat-table.hh"
62*2d1272b8SAndroid Build Coastguard Worker #include "hb-repacker.hh"
63*2d1272b8SAndroid Build Coastguard Worker #include "hb-subset-accelerator.hh"
64*2d1272b8SAndroid Build Coastguard Worker
65*2d1272b8SAndroid Build Coastguard Worker using OT::Layout::GSUB;
66*2d1272b8SAndroid Build Coastguard Worker using OT::Layout::GPOS;
67*2d1272b8SAndroid Build Coastguard Worker
68*2d1272b8SAndroid Build Coastguard Worker
69*2d1272b8SAndroid Build Coastguard Worker #ifndef HB_NO_SUBSET_CFF
70*2d1272b8SAndroid Build Coastguard Worker template<>
71*2d1272b8SAndroid Build Coastguard Worker struct hb_subset_plan_t::source_table_loader<const OT::cff1>
72*2d1272b8SAndroid Build Coastguard Worker {
73*2d1272b8SAndroid Build Coastguard Worker auto operator () (hb_subset_plan_t *plan)
74*2d1272b8SAndroid Build Coastguard Worker HB_AUTO_RETURN (plan->accelerator ? plan->accelerator->cff1_accel :
75*2d1272b8SAndroid Build Coastguard Worker plan->inprogress_accelerator ? plan->inprogress_accelerator->cff1_accel :
76*2d1272b8SAndroid Build Coastguard Worker plan->cff1_accel)
77*2d1272b8SAndroid Build Coastguard Worker };
78*2d1272b8SAndroid Build Coastguard Worker template<>
79*2d1272b8SAndroid Build Coastguard Worker struct hb_subset_plan_t::source_table_loader<const OT::cff2>
80*2d1272b8SAndroid Build Coastguard Worker {
81*2d1272b8SAndroid Build Coastguard Worker auto operator () (hb_subset_plan_t *plan)
82*2d1272b8SAndroid Build Coastguard Worker HB_AUTO_RETURN (plan->accelerator ? plan->accelerator->cff2_accel :
83*2d1272b8SAndroid Build Coastguard Worker plan->inprogress_accelerator ? plan->inprogress_accelerator->cff2_accel :
84*2d1272b8SAndroid Build Coastguard Worker plan->cff2_accel)
85*2d1272b8SAndroid Build Coastguard Worker };
86*2d1272b8SAndroid Build Coastguard Worker #endif
87*2d1272b8SAndroid Build Coastguard Worker
88*2d1272b8SAndroid Build Coastguard Worker
89*2d1272b8SAndroid Build Coastguard Worker /**
90*2d1272b8SAndroid Build Coastguard Worker * SECTION:hb-subset
91*2d1272b8SAndroid Build Coastguard Worker * @title: hb-subset
92*2d1272b8SAndroid Build Coastguard Worker * @short_description: Subsets font files.
93*2d1272b8SAndroid Build Coastguard Worker * @include: hb-subset.h
94*2d1272b8SAndroid Build Coastguard Worker *
95*2d1272b8SAndroid Build Coastguard Worker * Subsetting reduces the codepoint coverage of font files and removes all data
96*2d1272b8SAndroid Build Coastguard Worker * that is no longer needed. A subset input describes the desired subset. The input is
97*2d1272b8SAndroid Build Coastguard Worker * provided along with a font to the subsetting operation. Output is a new font file
98*2d1272b8SAndroid Build Coastguard Worker * containing only the data specified in the input.
99*2d1272b8SAndroid Build Coastguard Worker *
100*2d1272b8SAndroid Build Coastguard Worker * Currently most outline and bitmap tables are supported: glyf, CFF, CFF2, sbix,
101*2d1272b8SAndroid Build Coastguard Worker * COLR, and CBDT/CBLC. This also includes fonts with variable outlines via OpenType
102*2d1272b8SAndroid Build Coastguard Worker * variations. Notably EBDT/EBLC and SVG are not supported. Layout subsetting is supported
103*2d1272b8SAndroid Build Coastguard Worker * only for OpenType Layout tables (GSUB, GPOS, GDEF). Notably subsetting of graphite or AAT tables
104*2d1272b8SAndroid Build Coastguard Worker * is not yet supported.
105*2d1272b8SAndroid Build Coastguard Worker *
106*2d1272b8SAndroid Build Coastguard Worker * Fonts with graphite or AAT tables may still be subsetted but will likely need to use the
107*2d1272b8SAndroid Build Coastguard Worker * retain glyph ids option and configure the subset to pass through the layout tables untouched.
108*2d1272b8SAndroid Build Coastguard Worker */
109*2d1272b8SAndroid Build Coastguard Worker
110*2d1272b8SAndroid Build Coastguard Worker
111*2d1272b8SAndroid Build Coastguard Worker hb_user_data_key_t _hb_subset_accelerator_user_data_key = {};
112*2d1272b8SAndroid Build Coastguard Worker
113*2d1272b8SAndroid Build Coastguard Worker
114*2d1272b8SAndroid Build Coastguard Worker /*
115*2d1272b8SAndroid Build Coastguard Worker * The list of tables in the open type spec. Used to check for tables that may need handling
116*2d1272b8SAndroid Build Coastguard Worker * if we are unable to list the tables in a face.
117*2d1272b8SAndroid Build Coastguard Worker */
118*2d1272b8SAndroid Build Coastguard Worker static hb_tag_t known_tables[] {
119*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('a', 'v', 'a', 'r'),
120*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_BASE,
121*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_CBDT,
122*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_CBLC,
123*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_CFF1,
124*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_CFF2,
125*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_cmap,
126*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_COLR,
127*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_CPAL,
128*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('c', 'v', 'a', 'r'),
129*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('c', 'v', 't', ' '),
130*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('D', 'S', 'I', 'G'),
131*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('E', 'B', 'D', 'T'),
132*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('E', 'B', 'L', 'C'),
133*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('E', 'B', 'S', 'C'),
134*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('f', 'p', 'g', 'm'),
135*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('f', 'v', 'a', 'r'),
136*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('g', 'a', 's', 'p'),
137*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_GDEF,
138*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_glyf,
139*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_GPOS,
140*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_GSUB,
141*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_gvar,
142*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_hdmx,
143*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_head,
144*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_hhea,
145*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_hmtx,
146*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_HVAR,
147*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_JSTF,
148*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('k', 'e', 'r', 'n'),
149*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_loca,
150*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('L', 'T', 'S', 'H'),
151*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_MATH,
152*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_maxp,
153*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('M', 'E', 'R', 'G'),
154*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('m', 'e', 't', 'a'),
155*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('M', 'V', 'A', 'R'),
156*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('P', 'C', 'L', 'T'),
157*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_post,
158*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('p', 'r', 'e', 'p'),
159*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_sbix,
160*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('S', 'T', 'A', 'T'),
161*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('S', 'V', 'G', ' '),
162*2d1272b8SAndroid Build Coastguard Worker HB_TAG ('V', 'D', 'M', 'X'),
163*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_vhea,
164*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_vmtx,
165*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_VORG,
166*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_VVAR,
167*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_name,
168*2d1272b8SAndroid Build Coastguard Worker HB_OT_TAG_OS2
169*2d1272b8SAndroid Build Coastguard Worker };
170*2d1272b8SAndroid Build Coastguard Worker
_table_is_empty(const hb_face_t * face,hb_tag_t tag)171*2d1272b8SAndroid Build Coastguard Worker static bool _table_is_empty (const hb_face_t *face, hb_tag_t tag)
172*2d1272b8SAndroid Build Coastguard Worker {
173*2d1272b8SAndroid Build Coastguard Worker hb_blob_t* blob = hb_face_reference_table (face, tag);
174*2d1272b8SAndroid Build Coastguard Worker bool result = (blob == hb_blob_get_empty ());
175*2d1272b8SAndroid Build Coastguard Worker hb_blob_destroy (blob);
176*2d1272b8SAndroid Build Coastguard Worker return result;
177*2d1272b8SAndroid Build Coastguard Worker }
178*2d1272b8SAndroid Build Coastguard Worker
179*2d1272b8SAndroid Build Coastguard Worker static unsigned int
_get_table_tags(const hb_subset_plan_t * plan,unsigned int start_offset,unsigned int * table_count,hb_tag_t * table_tags)180*2d1272b8SAndroid Build Coastguard Worker _get_table_tags (const hb_subset_plan_t* plan,
181*2d1272b8SAndroid Build Coastguard Worker unsigned int start_offset,
182*2d1272b8SAndroid Build Coastguard Worker unsigned int *table_count, /* IN/OUT */
183*2d1272b8SAndroid Build Coastguard Worker hb_tag_t *table_tags /* OUT */)
184*2d1272b8SAndroid Build Coastguard Worker {
185*2d1272b8SAndroid Build Coastguard Worker unsigned num_tables = hb_face_get_table_tags (plan->source, 0, nullptr, nullptr);
186*2d1272b8SAndroid Build Coastguard Worker if (num_tables)
187*2d1272b8SAndroid Build Coastguard Worker return hb_face_get_table_tags (plan->source, start_offset, table_count, table_tags);
188*2d1272b8SAndroid Build Coastguard Worker
189*2d1272b8SAndroid Build Coastguard Worker // If face has 0 tables associated with it, assume that it was built from
190*2d1272b8SAndroid Build Coastguard Worker // hb_face_create_tables and thus is unable to list its tables. Fallback to
191*2d1272b8SAndroid Build Coastguard Worker // checking each table type we can handle for existence instead.
192*2d1272b8SAndroid Build Coastguard Worker auto it =
193*2d1272b8SAndroid Build Coastguard Worker hb_concat (
194*2d1272b8SAndroid Build Coastguard Worker + hb_array (known_tables)
195*2d1272b8SAndroid Build Coastguard Worker | hb_filter ([&] (hb_tag_t tag) {
196*2d1272b8SAndroid Build Coastguard Worker return !_table_is_empty (plan->source, tag) && !plan->no_subset_tables.has (tag);
197*2d1272b8SAndroid Build Coastguard Worker })
198*2d1272b8SAndroid Build Coastguard Worker | hb_map ([] (hb_tag_t tag) -> hb_tag_t { return tag; }),
199*2d1272b8SAndroid Build Coastguard Worker
200*2d1272b8SAndroid Build Coastguard Worker plan->no_subset_tables.iter ()
201*2d1272b8SAndroid Build Coastguard Worker | hb_filter([&] (hb_tag_t tag) {
202*2d1272b8SAndroid Build Coastguard Worker return !_table_is_empty (plan->source, tag);
203*2d1272b8SAndroid Build Coastguard Worker }));
204*2d1272b8SAndroid Build Coastguard Worker
205*2d1272b8SAndroid Build Coastguard Worker it += start_offset;
206*2d1272b8SAndroid Build Coastguard Worker
207*2d1272b8SAndroid Build Coastguard Worker unsigned num_written = 0;
208*2d1272b8SAndroid Build Coastguard Worker while (bool (it) && num_written < *table_count)
209*2d1272b8SAndroid Build Coastguard Worker table_tags[num_written++] = *it++;
210*2d1272b8SAndroid Build Coastguard Worker
211*2d1272b8SAndroid Build Coastguard Worker *table_count = num_written;
212*2d1272b8SAndroid Build Coastguard Worker return num_written;
213*2d1272b8SAndroid Build Coastguard Worker }
214*2d1272b8SAndroid Build Coastguard Worker
215*2d1272b8SAndroid Build Coastguard Worker
216*2d1272b8SAndroid Build Coastguard Worker static unsigned
_plan_estimate_subset_table_size(hb_subset_plan_t * plan,unsigned table_len,hb_tag_t table_tag)217*2d1272b8SAndroid Build Coastguard Worker _plan_estimate_subset_table_size (hb_subset_plan_t *plan,
218*2d1272b8SAndroid Build Coastguard Worker unsigned table_len,
219*2d1272b8SAndroid Build Coastguard Worker hb_tag_t table_tag)
220*2d1272b8SAndroid Build Coastguard Worker {
221*2d1272b8SAndroid Build Coastguard Worker unsigned src_glyphs = plan->source->get_num_glyphs ();
222*2d1272b8SAndroid Build Coastguard Worker unsigned dst_glyphs = plan->glyphset ()->get_population ();
223*2d1272b8SAndroid Build Coastguard Worker
224*2d1272b8SAndroid Build Coastguard Worker unsigned bulk = 8192;
225*2d1272b8SAndroid Build Coastguard Worker /* Tables that we want to allocate same space as the source table. For GSUB/GPOS it's
226*2d1272b8SAndroid Build Coastguard Worker * because those are expensive to subset, so giving them more room is fine. */
227*2d1272b8SAndroid Build Coastguard Worker bool same_size = table_tag == HB_OT_TAG_GSUB ||
228*2d1272b8SAndroid Build Coastguard Worker table_tag == HB_OT_TAG_GPOS ||
229*2d1272b8SAndroid Build Coastguard Worker table_tag == HB_OT_TAG_name;
230*2d1272b8SAndroid Build Coastguard Worker
231*2d1272b8SAndroid Build Coastguard Worker if (plan->flags & HB_SUBSET_FLAGS_RETAIN_GIDS)
232*2d1272b8SAndroid Build Coastguard Worker {
233*2d1272b8SAndroid Build Coastguard Worker if (table_tag == HB_OT_TAG_CFF1)
234*2d1272b8SAndroid Build Coastguard Worker {
235*2d1272b8SAndroid Build Coastguard Worker /* Add some extra room for the CFF charset. */
236*2d1272b8SAndroid Build Coastguard Worker bulk += src_glyphs * 16;
237*2d1272b8SAndroid Build Coastguard Worker }
238*2d1272b8SAndroid Build Coastguard Worker else if (table_tag == HB_OT_TAG_CFF2)
239*2d1272b8SAndroid Build Coastguard Worker {
240*2d1272b8SAndroid Build Coastguard Worker /* Just extra CharString offsets. */
241*2d1272b8SAndroid Build Coastguard Worker bulk += src_glyphs * 4;
242*2d1272b8SAndroid Build Coastguard Worker }
243*2d1272b8SAndroid Build Coastguard Worker }
244*2d1272b8SAndroid Build Coastguard Worker
245*2d1272b8SAndroid Build Coastguard Worker if (unlikely (!src_glyphs) || same_size)
246*2d1272b8SAndroid Build Coastguard Worker return bulk + table_len;
247*2d1272b8SAndroid Build Coastguard Worker
248*2d1272b8SAndroid Build Coastguard Worker return bulk + (unsigned) (table_len * sqrt ((double) dst_glyphs / src_glyphs));
249*2d1272b8SAndroid Build Coastguard Worker }
250*2d1272b8SAndroid Build Coastguard Worker
251*2d1272b8SAndroid Build Coastguard Worker /*
252*2d1272b8SAndroid Build Coastguard Worker * Repack the serialization buffer if any offset overflows exist.
253*2d1272b8SAndroid Build Coastguard Worker */
254*2d1272b8SAndroid Build Coastguard Worker static hb_blob_t*
_repack(hb_tag_t tag,const hb_serialize_context_t & c)255*2d1272b8SAndroid Build Coastguard Worker _repack (hb_tag_t tag, const hb_serialize_context_t& c)
256*2d1272b8SAndroid Build Coastguard Worker {
257*2d1272b8SAndroid Build Coastguard Worker if (!c.offset_overflow ())
258*2d1272b8SAndroid Build Coastguard Worker return c.copy_blob ();
259*2d1272b8SAndroid Build Coastguard Worker
260*2d1272b8SAndroid Build Coastguard Worker hb_blob_t* result = hb_resolve_overflows (c.object_graph (), tag);
261*2d1272b8SAndroid Build Coastguard Worker
262*2d1272b8SAndroid Build Coastguard Worker if (unlikely (!result))
263*2d1272b8SAndroid Build Coastguard Worker {
264*2d1272b8SAndroid Build Coastguard Worker DEBUG_MSG (SUBSET, nullptr, "OT::%c%c%c%c offset overflow resolution failed.",
265*2d1272b8SAndroid Build Coastguard Worker HB_UNTAG (tag));
266*2d1272b8SAndroid Build Coastguard Worker return nullptr;
267*2d1272b8SAndroid Build Coastguard Worker }
268*2d1272b8SAndroid Build Coastguard Worker
269*2d1272b8SAndroid Build Coastguard Worker return result;
270*2d1272b8SAndroid Build Coastguard Worker }
271*2d1272b8SAndroid Build Coastguard Worker
272*2d1272b8SAndroid Build Coastguard Worker template<typename TableType>
273*2d1272b8SAndroid Build Coastguard Worker static
274*2d1272b8SAndroid Build Coastguard Worker bool
_try_subset(const TableType * table,hb_vector_t<char> * buf,hb_subset_context_t * c)275*2d1272b8SAndroid Build Coastguard Worker _try_subset (const TableType *table,
276*2d1272b8SAndroid Build Coastguard Worker hb_vector_t<char>* buf,
277*2d1272b8SAndroid Build Coastguard Worker hb_subset_context_t* c /* OUT */)
278*2d1272b8SAndroid Build Coastguard Worker {
279*2d1272b8SAndroid Build Coastguard Worker c->serializer->start_serialize ();
280*2d1272b8SAndroid Build Coastguard Worker if (c->serializer->in_error ()) return false;
281*2d1272b8SAndroid Build Coastguard Worker
282*2d1272b8SAndroid Build Coastguard Worker bool needed = table->subset (c);
283*2d1272b8SAndroid Build Coastguard Worker if (!c->serializer->ran_out_of_room ())
284*2d1272b8SAndroid Build Coastguard Worker {
285*2d1272b8SAndroid Build Coastguard Worker c->serializer->end_serialize ();
286*2d1272b8SAndroid Build Coastguard Worker return needed;
287*2d1272b8SAndroid Build Coastguard Worker }
288*2d1272b8SAndroid Build Coastguard Worker
289*2d1272b8SAndroid Build Coastguard Worker unsigned buf_size = buf->allocated;
290*2d1272b8SAndroid Build Coastguard Worker buf_size = buf_size * 2 + 16;
291*2d1272b8SAndroid Build Coastguard Worker
292*2d1272b8SAndroid Build Coastguard Worker
293*2d1272b8SAndroid Build Coastguard Worker
294*2d1272b8SAndroid Build Coastguard Worker
295*2d1272b8SAndroid Build Coastguard Worker DEBUG_MSG (SUBSET, nullptr, "OT::%c%c%c%c ran out of room; reallocating to %u bytes.",
296*2d1272b8SAndroid Build Coastguard Worker HB_UNTAG (c->table_tag), buf_size);
297*2d1272b8SAndroid Build Coastguard Worker
298*2d1272b8SAndroid Build Coastguard Worker if (unlikely (buf_size > c->source_blob->length * 16 ||
299*2d1272b8SAndroid Build Coastguard Worker !buf->alloc (buf_size, true)))
300*2d1272b8SAndroid Build Coastguard Worker {
301*2d1272b8SAndroid Build Coastguard Worker DEBUG_MSG (SUBSET, nullptr, "OT::%c%c%c%c failed to reallocate %u bytes.",
302*2d1272b8SAndroid Build Coastguard Worker HB_UNTAG (c->table_tag), buf_size);
303*2d1272b8SAndroid Build Coastguard Worker return needed;
304*2d1272b8SAndroid Build Coastguard Worker }
305*2d1272b8SAndroid Build Coastguard Worker
306*2d1272b8SAndroid Build Coastguard Worker c->serializer->reset (buf->arrayZ, buf->allocated);
307*2d1272b8SAndroid Build Coastguard Worker return _try_subset (table, buf, c);
308*2d1272b8SAndroid Build Coastguard Worker }
309*2d1272b8SAndroid Build Coastguard Worker
310*2d1272b8SAndroid Build Coastguard Worker template <typename T>
_do_destroy(T & t,hb_priority<1>)311*2d1272b8SAndroid Build Coastguard Worker static auto _do_destroy (T &t, hb_priority<1>) HB_RETURN (void, t.destroy ())
312*2d1272b8SAndroid Build Coastguard Worker
313*2d1272b8SAndroid Build Coastguard Worker template <typename T>
314*2d1272b8SAndroid Build Coastguard Worker static void _do_destroy (T &t, hb_priority<0>) {}
315*2d1272b8SAndroid Build Coastguard Worker
316*2d1272b8SAndroid Build Coastguard Worker template<typename TableType>
317*2d1272b8SAndroid Build Coastguard Worker static bool
_subset(hb_subset_plan_t * plan,hb_vector_t<char> & buf)318*2d1272b8SAndroid Build Coastguard Worker _subset (hb_subset_plan_t *plan, hb_vector_t<char> &buf)
319*2d1272b8SAndroid Build Coastguard Worker {
320*2d1272b8SAndroid Build Coastguard Worker auto &&source_blob = plan->source_table<TableType> ();
321*2d1272b8SAndroid Build Coastguard Worker auto *table = source_blob.get ();
322*2d1272b8SAndroid Build Coastguard Worker
323*2d1272b8SAndroid Build Coastguard Worker hb_tag_t tag = TableType::tableTag;
324*2d1272b8SAndroid Build Coastguard Worker hb_blob_t *blob = source_blob.get_blob();
325*2d1272b8SAndroid Build Coastguard Worker if (unlikely (!blob || !blob->data))
326*2d1272b8SAndroid Build Coastguard Worker {
327*2d1272b8SAndroid Build Coastguard Worker DEBUG_MSG (SUBSET, nullptr,
328*2d1272b8SAndroid Build Coastguard Worker "OT::%c%c%c%c::subset sanitize failed on source table.", HB_UNTAG (tag));
329*2d1272b8SAndroid Build Coastguard Worker _do_destroy (source_blob, hb_prioritize);
330*2d1272b8SAndroid Build Coastguard Worker return false;
331*2d1272b8SAndroid Build Coastguard Worker }
332*2d1272b8SAndroid Build Coastguard Worker
333*2d1272b8SAndroid Build Coastguard Worker unsigned buf_size = _plan_estimate_subset_table_size (plan, blob->length, TableType::tableTag);
334*2d1272b8SAndroid Build Coastguard Worker DEBUG_MSG (SUBSET, nullptr,
335*2d1272b8SAndroid Build Coastguard Worker "OT::%c%c%c%c initial estimated table size: %u bytes.", HB_UNTAG (tag), buf_size);
336*2d1272b8SAndroid Build Coastguard Worker if (unlikely (!buf.alloc (buf_size)))
337*2d1272b8SAndroid Build Coastguard Worker {
338*2d1272b8SAndroid Build Coastguard Worker DEBUG_MSG (SUBSET, nullptr, "OT::%c%c%c%c failed to allocate %u bytes.", HB_UNTAG (tag), buf_size);
339*2d1272b8SAndroid Build Coastguard Worker _do_destroy (source_blob, hb_prioritize);
340*2d1272b8SAndroid Build Coastguard Worker return false;
341*2d1272b8SAndroid Build Coastguard Worker }
342*2d1272b8SAndroid Build Coastguard Worker
343*2d1272b8SAndroid Build Coastguard Worker bool needed = false;
344*2d1272b8SAndroid Build Coastguard Worker hb_serialize_context_t serializer (buf.arrayZ, buf.allocated);
345*2d1272b8SAndroid Build Coastguard Worker {
346*2d1272b8SAndroid Build Coastguard Worker hb_subset_context_t c (blob, plan, &serializer, tag);
347*2d1272b8SAndroid Build Coastguard Worker needed = _try_subset (table, &buf, &c);
348*2d1272b8SAndroid Build Coastguard Worker }
349*2d1272b8SAndroid Build Coastguard Worker _do_destroy (source_blob, hb_prioritize);
350*2d1272b8SAndroid Build Coastguard Worker
351*2d1272b8SAndroid Build Coastguard Worker if (serializer.in_error () && !serializer.only_offset_overflow ())
352*2d1272b8SAndroid Build Coastguard Worker {
353*2d1272b8SAndroid Build Coastguard Worker DEBUG_MSG (SUBSET, nullptr, "OT::%c%c%c%c::subset FAILED!", HB_UNTAG (tag));
354*2d1272b8SAndroid Build Coastguard Worker return false;
355*2d1272b8SAndroid Build Coastguard Worker }
356*2d1272b8SAndroid Build Coastguard Worker
357*2d1272b8SAndroid Build Coastguard Worker if (!needed)
358*2d1272b8SAndroid Build Coastguard Worker {
359*2d1272b8SAndroid Build Coastguard Worker DEBUG_MSG (SUBSET, nullptr, "OT::%c%c%c%c::subset table subsetted to empty.", HB_UNTAG (tag));
360*2d1272b8SAndroid Build Coastguard Worker return true;
361*2d1272b8SAndroid Build Coastguard Worker }
362*2d1272b8SAndroid Build Coastguard Worker
363*2d1272b8SAndroid Build Coastguard Worker bool result = false;
364*2d1272b8SAndroid Build Coastguard Worker hb_blob_t *dest_blob = _repack (tag, serializer);
365*2d1272b8SAndroid Build Coastguard Worker if (dest_blob)
366*2d1272b8SAndroid Build Coastguard Worker {
367*2d1272b8SAndroid Build Coastguard Worker DEBUG_MSG (SUBSET, nullptr,
368*2d1272b8SAndroid Build Coastguard Worker "OT::%c%c%c%c final subset table size: %u bytes.",
369*2d1272b8SAndroid Build Coastguard Worker HB_UNTAG (tag), dest_blob->length);
370*2d1272b8SAndroid Build Coastguard Worker result = plan->add_table (tag, dest_blob);
371*2d1272b8SAndroid Build Coastguard Worker hb_blob_destroy (dest_blob);
372*2d1272b8SAndroid Build Coastguard Worker }
373*2d1272b8SAndroid Build Coastguard Worker
374*2d1272b8SAndroid Build Coastguard Worker DEBUG_MSG (SUBSET, nullptr, "OT::%c%c%c%c::subset %s",
375*2d1272b8SAndroid Build Coastguard Worker HB_UNTAG (tag), result ? "success" : "FAILED!");
376*2d1272b8SAndroid Build Coastguard Worker return result;
377*2d1272b8SAndroid Build Coastguard Worker }
378*2d1272b8SAndroid Build Coastguard Worker
379*2d1272b8SAndroid Build Coastguard Worker static bool
_is_table_present(hb_face_t * source,hb_tag_t tag)380*2d1272b8SAndroid Build Coastguard Worker _is_table_present (hb_face_t *source, hb_tag_t tag)
381*2d1272b8SAndroid Build Coastguard Worker {
382*2d1272b8SAndroid Build Coastguard Worker
383*2d1272b8SAndroid Build Coastguard Worker if (!hb_face_get_table_tags (source, 0, nullptr, nullptr)) {
384*2d1272b8SAndroid Build Coastguard Worker // If face has 0 tables associated with it, assume that it was built from
385*2d1272b8SAndroid Build Coastguard Worker // hb_face_create_tables and thus is unable to list its tables. Fallback to
386*2d1272b8SAndroid Build Coastguard Worker // checking if the blob associated with tag is empty.
387*2d1272b8SAndroid Build Coastguard Worker return !_table_is_empty (source, tag);
388*2d1272b8SAndroid Build Coastguard Worker }
389*2d1272b8SAndroid Build Coastguard Worker
390*2d1272b8SAndroid Build Coastguard Worker hb_tag_t table_tags[32];
391*2d1272b8SAndroid Build Coastguard Worker unsigned offset = 0, num_tables = ARRAY_LENGTH (table_tags);
392*2d1272b8SAndroid Build Coastguard Worker while (((void) hb_face_get_table_tags (source, offset, &num_tables, table_tags), num_tables))
393*2d1272b8SAndroid Build Coastguard Worker {
394*2d1272b8SAndroid Build Coastguard Worker for (unsigned i = 0; i < num_tables; ++i)
395*2d1272b8SAndroid Build Coastguard Worker if (table_tags[i] == tag)
396*2d1272b8SAndroid Build Coastguard Worker return true;
397*2d1272b8SAndroid Build Coastguard Worker offset += num_tables;
398*2d1272b8SAndroid Build Coastguard Worker }
399*2d1272b8SAndroid Build Coastguard Worker return false;
400*2d1272b8SAndroid Build Coastguard Worker }
401*2d1272b8SAndroid Build Coastguard Worker
402*2d1272b8SAndroid Build Coastguard Worker static bool
_should_drop_table(hb_subset_plan_t * plan,hb_tag_t tag)403*2d1272b8SAndroid Build Coastguard Worker _should_drop_table (hb_subset_plan_t *plan, hb_tag_t tag)
404*2d1272b8SAndroid Build Coastguard Worker {
405*2d1272b8SAndroid Build Coastguard Worker if (plan->drop_tables.has (tag))
406*2d1272b8SAndroid Build Coastguard Worker return true;
407*2d1272b8SAndroid Build Coastguard Worker
408*2d1272b8SAndroid Build Coastguard Worker switch (tag)
409*2d1272b8SAndroid Build Coastguard Worker {
410*2d1272b8SAndroid Build Coastguard Worker case HB_TAG ('c','v','a','r'): /* hint table, fallthrough */
411*2d1272b8SAndroid Build Coastguard Worker return plan->all_axes_pinned || (plan->flags & HB_SUBSET_FLAGS_NO_HINTING);
412*2d1272b8SAndroid Build Coastguard Worker
413*2d1272b8SAndroid Build Coastguard Worker case HB_TAG ('c','v','t',' '): /* hint table, fallthrough */
414*2d1272b8SAndroid Build Coastguard Worker case HB_TAG ('f','p','g','m'): /* hint table, fallthrough */
415*2d1272b8SAndroid Build Coastguard Worker case HB_TAG ('p','r','e','p'): /* hint table, fallthrough */
416*2d1272b8SAndroid Build Coastguard Worker case HB_TAG ('h','d','m','x'): /* hint table, fallthrough */
417*2d1272b8SAndroid Build Coastguard Worker case HB_TAG ('V','D','M','X'): /* hint table, fallthrough */
418*2d1272b8SAndroid Build Coastguard Worker return plan->flags & HB_SUBSET_FLAGS_NO_HINTING;
419*2d1272b8SAndroid Build Coastguard Worker
420*2d1272b8SAndroid Build Coastguard Worker #ifdef HB_NO_SUBSET_LAYOUT
421*2d1272b8SAndroid Build Coastguard Worker // Drop Layout Tables if requested.
422*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_GDEF:
423*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_GPOS:
424*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_GSUB:
425*2d1272b8SAndroid Build Coastguard Worker case HB_TAG ('m','o','r','x'):
426*2d1272b8SAndroid Build Coastguard Worker case HB_TAG ('m','o','r','t'):
427*2d1272b8SAndroid Build Coastguard Worker case HB_TAG ('k','e','r','x'):
428*2d1272b8SAndroid Build Coastguard Worker case HB_TAG ('k','e','r','n'):
429*2d1272b8SAndroid Build Coastguard Worker return true;
430*2d1272b8SAndroid Build Coastguard Worker #endif
431*2d1272b8SAndroid Build Coastguard Worker
432*2d1272b8SAndroid Build Coastguard Worker case HB_TAG ('a','v','a','r'):
433*2d1272b8SAndroid Build Coastguard Worker case HB_TAG ('f','v','a','r'):
434*2d1272b8SAndroid Build Coastguard Worker case HB_TAG ('g','v','a','r'):
435*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_HVAR:
436*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_VVAR:
437*2d1272b8SAndroid Build Coastguard Worker case HB_TAG ('M','V','A','R'):
438*2d1272b8SAndroid Build Coastguard Worker return plan->all_axes_pinned;
439*2d1272b8SAndroid Build Coastguard Worker
440*2d1272b8SAndroid Build Coastguard Worker default:
441*2d1272b8SAndroid Build Coastguard Worker return false;
442*2d1272b8SAndroid Build Coastguard Worker }
443*2d1272b8SAndroid Build Coastguard Worker }
444*2d1272b8SAndroid Build Coastguard Worker
445*2d1272b8SAndroid Build Coastguard Worker static bool
_passthrough(hb_subset_plan_t * plan,hb_tag_t tag)446*2d1272b8SAndroid Build Coastguard Worker _passthrough (hb_subset_plan_t *plan, hb_tag_t tag)
447*2d1272b8SAndroid Build Coastguard Worker {
448*2d1272b8SAndroid Build Coastguard Worker hb_blob_t *source_table = hb_face_reference_table (plan->source, tag);
449*2d1272b8SAndroid Build Coastguard Worker bool result = plan->add_table (tag, source_table);
450*2d1272b8SAndroid Build Coastguard Worker hb_blob_destroy (source_table);
451*2d1272b8SAndroid Build Coastguard Worker return result;
452*2d1272b8SAndroid Build Coastguard Worker }
453*2d1272b8SAndroid Build Coastguard Worker
454*2d1272b8SAndroid Build Coastguard Worker static bool
_dependencies_satisfied(hb_subset_plan_t * plan,hb_tag_t tag,const hb_set_t & subsetted_tags,const hb_set_t & pending_subset_tags)455*2d1272b8SAndroid Build Coastguard Worker _dependencies_satisfied (hb_subset_plan_t *plan, hb_tag_t tag,
456*2d1272b8SAndroid Build Coastguard Worker const hb_set_t &subsetted_tags,
457*2d1272b8SAndroid Build Coastguard Worker const hb_set_t &pending_subset_tags)
458*2d1272b8SAndroid Build Coastguard Worker {
459*2d1272b8SAndroid Build Coastguard Worker switch (tag)
460*2d1272b8SAndroid Build Coastguard Worker {
461*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_hmtx:
462*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_vmtx:
463*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_maxp:
464*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_OS2:
465*2d1272b8SAndroid Build Coastguard Worker return !plan->normalized_coords || !pending_subset_tags.has (HB_OT_TAG_glyf);
466*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_GPOS:
467*2d1272b8SAndroid Build Coastguard Worker return plan->all_axes_pinned || !pending_subset_tags.has (HB_OT_TAG_GDEF);
468*2d1272b8SAndroid Build Coastguard Worker default:
469*2d1272b8SAndroid Build Coastguard Worker return true;
470*2d1272b8SAndroid Build Coastguard Worker }
471*2d1272b8SAndroid Build Coastguard Worker }
472*2d1272b8SAndroid Build Coastguard Worker
473*2d1272b8SAndroid Build Coastguard Worker static bool
_subset_table(hb_subset_plan_t * plan,hb_vector_t<char> & buf,hb_tag_t tag)474*2d1272b8SAndroid Build Coastguard Worker _subset_table (hb_subset_plan_t *plan,
475*2d1272b8SAndroid Build Coastguard Worker hb_vector_t<char> &buf,
476*2d1272b8SAndroid Build Coastguard Worker hb_tag_t tag)
477*2d1272b8SAndroid Build Coastguard Worker {
478*2d1272b8SAndroid Build Coastguard Worker if (plan->no_subset_tables.has (tag)) {
479*2d1272b8SAndroid Build Coastguard Worker return _passthrough (plan, tag);
480*2d1272b8SAndroid Build Coastguard Worker }
481*2d1272b8SAndroid Build Coastguard Worker
482*2d1272b8SAndroid Build Coastguard Worker DEBUG_MSG (SUBSET, nullptr, "subset %c%c%c%c", HB_UNTAG (tag));
483*2d1272b8SAndroid Build Coastguard Worker switch (tag)
484*2d1272b8SAndroid Build Coastguard Worker {
485*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_glyf: return _subset<const OT::glyf> (plan, buf);
486*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_hdmx: return _subset<const OT::hdmx> (plan, buf);
487*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_name: return _subset<const OT::name> (plan, buf);
488*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_head:
489*2d1272b8SAndroid Build Coastguard Worker if (_is_table_present (plan->source, HB_OT_TAG_glyf) && !_should_drop_table (plan, HB_OT_TAG_glyf))
490*2d1272b8SAndroid Build Coastguard Worker return true; /* skip head, handled by glyf */
491*2d1272b8SAndroid Build Coastguard Worker return _subset<const OT::head> (plan, buf);
492*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_hhea: return true; /* skip hhea, handled by hmtx */
493*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_hmtx: return _subset<const OT::hmtx> (plan, buf);
494*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_vhea: return true; /* skip vhea, handled by vmtx */
495*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_vmtx: return _subset<const OT::vmtx> (plan, buf);
496*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_maxp: return _subset<const OT::maxp> (plan, buf);
497*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_sbix: return _subset<const OT::sbix> (plan, buf);
498*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_loca: return true; /* skip loca, handled by glyf */
499*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_cmap: return _subset<const OT::cmap> (plan, buf);
500*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_OS2 : return _subset<const OT::OS2 > (plan, buf);
501*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_post: return _subset<const OT::post> (plan, buf);
502*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_COLR: return _subset<const OT::COLR> (plan, buf);
503*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_CPAL: return _subset<const OT::CPAL> (plan, buf);
504*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_CBLC: return _subset<const OT::CBLC> (plan, buf);
505*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_CBDT: return true; /* skip CBDT, handled by CBLC */
506*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_MATH: return _subset<const OT::MATH> (plan, buf);
507*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_BASE: return _subset<const OT::BASE> (plan, buf);
508*2d1272b8SAndroid Build Coastguard Worker
509*2d1272b8SAndroid Build Coastguard Worker #ifndef HB_NO_SUBSET_CFF
510*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_CFF1: return _subset<const OT::cff1> (plan, buf);
511*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_CFF2: return _subset<const OT::cff2> (plan, buf);
512*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_VORG: return _subset<const OT::VORG> (plan, buf);
513*2d1272b8SAndroid Build Coastguard Worker #endif
514*2d1272b8SAndroid Build Coastguard Worker
515*2d1272b8SAndroid Build Coastguard Worker #ifndef HB_NO_SUBSET_LAYOUT
516*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_GDEF: return _subset<const OT::GDEF> (plan, buf);
517*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_GSUB: return _subset<const GSUB> (plan, buf);
518*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_GPOS: return _subset<const GPOS> (plan, buf);
519*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_gvar: return _subset<const OT::gvar> (plan, buf);
520*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_HVAR: return _subset<const OT::HVAR> (plan, buf);
521*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_VVAR: return _subset<const OT::VVAR> (plan, buf);
522*2d1272b8SAndroid Build Coastguard Worker #endif
523*2d1272b8SAndroid Build Coastguard Worker
524*2d1272b8SAndroid Build Coastguard Worker #ifndef HB_NO_VAR
525*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_fvar:
526*2d1272b8SAndroid Build Coastguard Worker if (plan->user_axes_location.is_empty ()) return _passthrough (plan, tag);
527*2d1272b8SAndroid Build Coastguard Worker return _subset<const OT::fvar> (plan, buf);
528*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_avar:
529*2d1272b8SAndroid Build Coastguard Worker if (plan->user_axes_location.is_empty ()) return _passthrough (plan, tag);
530*2d1272b8SAndroid Build Coastguard Worker return _subset<const OT::avar> (plan, buf);
531*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_cvar:
532*2d1272b8SAndroid Build Coastguard Worker if (plan->user_axes_location.is_empty ()) return _passthrough (plan, tag);
533*2d1272b8SAndroid Build Coastguard Worker return _subset<const OT::cvar> (plan, buf);
534*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_MVAR:
535*2d1272b8SAndroid Build Coastguard Worker if (plan->user_axes_location.is_empty ()) return _passthrough (plan, tag);
536*2d1272b8SAndroid Build Coastguard Worker return _subset<const OT::MVAR> (plan, buf);
537*2d1272b8SAndroid Build Coastguard Worker #endif
538*2d1272b8SAndroid Build Coastguard Worker
539*2d1272b8SAndroid Build Coastguard Worker case HB_OT_TAG_STAT:
540*2d1272b8SAndroid Build Coastguard Worker if (!plan->user_axes_location.is_empty ()) return _subset<const OT::STAT> (plan, buf);
541*2d1272b8SAndroid Build Coastguard Worker else return _passthrough (plan, tag);
542*2d1272b8SAndroid Build Coastguard Worker
543*2d1272b8SAndroid Build Coastguard Worker case HB_TAG ('c', 'v', 't', ' '):
544*2d1272b8SAndroid Build Coastguard Worker #ifndef HB_NO_VAR
545*2d1272b8SAndroid Build Coastguard Worker if (_is_table_present (plan->source, HB_OT_TAG_cvar) &&
546*2d1272b8SAndroid Build Coastguard Worker plan->normalized_coords && !plan->pinned_at_default)
547*2d1272b8SAndroid Build Coastguard Worker {
548*2d1272b8SAndroid Build Coastguard Worker auto &cvar = *plan->source->table.cvar;
549*2d1272b8SAndroid Build Coastguard Worker return OT::cvar::add_cvt_and_apply_deltas (plan, cvar.get_tuple_var_data (), &cvar);
550*2d1272b8SAndroid Build Coastguard Worker }
551*2d1272b8SAndroid Build Coastguard Worker #endif
552*2d1272b8SAndroid Build Coastguard Worker return _passthrough (plan, tag);
553*2d1272b8SAndroid Build Coastguard Worker
554*2d1272b8SAndroid Build Coastguard Worker default:
555*2d1272b8SAndroid Build Coastguard Worker if (plan->flags & HB_SUBSET_FLAGS_PASSTHROUGH_UNRECOGNIZED)
556*2d1272b8SAndroid Build Coastguard Worker return _passthrough (plan, tag);
557*2d1272b8SAndroid Build Coastguard Worker
558*2d1272b8SAndroid Build Coastguard Worker // Drop table
559*2d1272b8SAndroid Build Coastguard Worker return true;
560*2d1272b8SAndroid Build Coastguard Worker }
561*2d1272b8SAndroid Build Coastguard Worker }
562*2d1272b8SAndroid Build Coastguard Worker
_attach_accelerator_data(hb_subset_plan_t * plan,hb_face_t * face)563*2d1272b8SAndroid Build Coastguard Worker static void _attach_accelerator_data (hb_subset_plan_t* plan,
564*2d1272b8SAndroid Build Coastguard Worker hb_face_t* face /* IN/OUT */)
565*2d1272b8SAndroid Build Coastguard Worker {
566*2d1272b8SAndroid Build Coastguard Worker if (!plan->inprogress_accelerator) return;
567*2d1272b8SAndroid Build Coastguard Worker
568*2d1272b8SAndroid Build Coastguard Worker // Transfer the accelerator from the plan to us.
569*2d1272b8SAndroid Build Coastguard Worker hb_subset_accelerator_t* accel = plan->inprogress_accelerator;
570*2d1272b8SAndroid Build Coastguard Worker plan->inprogress_accelerator = nullptr;
571*2d1272b8SAndroid Build Coastguard Worker
572*2d1272b8SAndroid Build Coastguard Worker if (accel->in_error ())
573*2d1272b8SAndroid Build Coastguard Worker {
574*2d1272b8SAndroid Build Coastguard Worker hb_subset_accelerator_t::destroy (accel);
575*2d1272b8SAndroid Build Coastguard Worker return;
576*2d1272b8SAndroid Build Coastguard Worker }
577*2d1272b8SAndroid Build Coastguard Worker
578*2d1272b8SAndroid Build Coastguard Worker // Populate caches that need access to the final tables.
579*2d1272b8SAndroid Build Coastguard Worker hb_blob_ptr_t<OT::cmap> cmap_ptr (hb_sanitize_context_t ().reference_table<OT::cmap> (face));
580*2d1272b8SAndroid Build Coastguard Worker accel->cmap_cache = OT::cmap::create_filled_cache (cmap_ptr);
581*2d1272b8SAndroid Build Coastguard Worker accel->destroy_cmap_cache = OT::SubtableUnicodesCache::destroy;
582*2d1272b8SAndroid Build Coastguard Worker
583*2d1272b8SAndroid Build Coastguard Worker if (!hb_face_set_user_data(face,
584*2d1272b8SAndroid Build Coastguard Worker hb_subset_accelerator_t::user_data_key(),
585*2d1272b8SAndroid Build Coastguard Worker accel,
586*2d1272b8SAndroid Build Coastguard Worker hb_subset_accelerator_t::destroy,
587*2d1272b8SAndroid Build Coastguard Worker true))
588*2d1272b8SAndroid Build Coastguard Worker hb_subset_accelerator_t::destroy (accel);
589*2d1272b8SAndroid Build Coastguard Worker }
590*2d1272b8SAndroid Build Coastguard Worker
591*2d1272b8SAndroid Build Coastguard Worker /**
592*2d1272b8SAndroid Build Coastguard Worker * hb_subset_or_fail:
593*2d1272b8SAndroid Build Coastguard Worker * @source: font face data to be subset.
594*2d1272b8SAndroid Build Coastguard Worker * @input: input to use for the subsetting.
595*2d1272b8SAndroid Build Coastguard Worker *
596*2d1272b8SAndroid Build Coastguard Worker * Subsets a font according to provided input. Returns nullptr
597*2d1272b8SAndroid Build Coastguard Worker * if the subset operation fails or the face has no glyphs.
598*2d1272b8SAndroid Build Coastguard Worker *
599*2d1272b8SAndroid Build Coastguard Worker * Since: 2.9.0
600*2d1272b8SAndroid Build Coastguard Worker **/
601*2d1272b8SAndroid Build Coastguard Worker hb_face_t *
hb_subset_or_fail(hb_face_t * source,const hb_subset_input_t * input)602*2d1272b8SAndroid Build Coastguard Worker hb_subset_or_fail (hb_face_t *source, const hb_subset_input_t *input)
603*2d1272b8SAndroid Build Coastguard Worker {
604*2d1272b8SAndroid Build Coastguard Worker if (unlikely (!input || !source)) return nullptr;
605*2d1272b8SAndroid Build Coastguard Worker
606*2d1272b8SAndroid Build Coastguard Worker if (unlikely (!source->get_num_glyphs ()))
607*2d1272b8SAndroid Build Coastguard Worker {
608*2d1272b8SAndroid Build Coastguard Worker DEBUG_MSG (SUBSET, nullptr, "No glyphs in source font.");
609*2d1272b8SAndroid Build Coastguard Worker return nullptr;
610*2d1272b8SAndroid Build Coastguard Worker }
611*2d1272b8SAndroid Build Coastguard Worker
612*2d1272b8SAndroid Build Coastguard Worker hb_subset_plan_t *plan = hb_subset_plan_create_or_fail (source, input);
613*2d1272b8SAndroid Build Coastguard Worker if (unlikely (!plan)) {
614*2d1272b8SAndroid Build Coastguard Worker return nullptr;
615*2d1272b8SAndroid Build Coastguard Worker }
616*2d1272b8SAndroid Build Coastguard Worker
617*2d1272b8SAndroid Build Coastguard Worker hb_face_t * result = hb_subset_plan_execute_or_fail (plan);
618*2d1272b8SAndroid Build Coastguard Worker hb_subset_plan_destroy (plan);
619*2d1272b8SAndroid Build Coastguard Worker return result;
620*2d1272b8SAndroid Build Coastguard Worker }
621*2d1272b8SAndroid Build Coastguard Worker
622*2d1272b8SAndroid Build Coastguard Worker
623*2d1272b8SAndroid Build Coastguard Worker /**
624*2d1272b8SAndroid Build Coastguard Worker * hb_subset_plan_execute_or_fail:
625*2d1272b8SAndroid Build Coastguard Worker * @plan: a subsetting plan.
626*2d1272b8SAndroid Build Coastguard Worker *
627*2d1272b8SAndroid Build Coastguard Worker * Executes the provided subsetting @plan.
628*2d1272b8SAndroid Build Coastguard Worker *
629*2d1272b8SAndroid Build Coastguard Worker * Return value:
630*2d1272b8SAndroid Build Coastguard Worker * on success returns a reference to generated font subset. If the subsetting operation fails
631*2d1272b8SAndroid Build Coastguard Worker * returns nullptr.
632*2d1272b8SAndroid Build Coastguard Worker *
633*2d1272b8SAndroid Build Coastguard Worker * Since: 4.0.0
634*2d1272b8SAndroid Build Coastguard Worker **/
635*2d1272b8SAndroid Build Coastguard Worker hb_face_t *
hb_subset_plan_execute_or_fail(hb_subset_plan_t * plan)636*2d1272b8SAndroid Build Coastguard Worker hb_subset_plan_execute_or_fail (hb_subset_plan_t *plan)
637*2d1272b8SAndroid Build Coastguard Worker {
638*2d1272b8SAndroid Build Coastguard Worker if (unlikely (!plan || plan->in_error ())) {
639*2d1272b8SAndroid Build Coastguard Worker return nullptr;
640*2d1272b8SAndroid Build Coastguard Worker }
641*2d1272b8SAndroid Build Coastguard Worker
642*2d1272b8SAndroid Build Coastguard Worker hb_tag_t table_tags[32];
643*2d1272b8SAndroid Build Coastguard Worker unsigned offset = 0, num_tables = ARRAY_LENGTH (table_tags);
644*2d1272b8SAndroid Build Coastguard Worker
645*2d1272b8SAndroid Build Coastguard Worker hb_set_t subsetted_tags, pending_subset_tags;
646*2d1272b8SAndroid Build Coastguard Worker while (((void) _get_table_tags (plan, offset, &num_tables, table_tags), num_tables))
647*2d1272b8SAndroid Build Coastguard Worker {
648*2d1272b8SAndroid Build Coastguard Worker for (unsigned i = 0; i < num_tables; ++i)
649*2d1272b8SAndroid Build Coastguard Worker {
650*2d1272b8SAndroid Build Coastguard Worker hb_tag_t tag = table_tags[i];
651*2d1272b8SAndroid Build Coastguard Worker if (_should_drop_table (plan, tag)) continue;
652*2d1272b8SAndroid Build Coastguard Worker pending_subset_tags.add (tag);
653*2d1272b8SAndroid Build Coastguard Worker }
654*2d1272b8SAndroid Build Coastguard Worker
655*2d1272b8SAndroid Build Coastguard Worker offset += num_tables;
656*2d1272b8SAndroid Build Coastguard Worker }
657*2d1272b8SAndroid Build Coastguard Worker
658*2d1272b8SAndroid Build Coastguard Worker bool success = true;
659*2d1272b8SAndroid Build Coastguard Worker
660*2d1272b8SAndroid Build Coastguard Worker {
661*2d1272b8SAndroid Build Coastguard Worker // Grouping to deallocate buf before calling hb_face_reference (plan->dest).
662*2d1272b8SAndroid Build Coastguard Worker
663*2d1272b8SAndroid Build Coastguard Worker hb_vector_t<char> buf;
664*2d1272b8SAndroid Build Coastguard Worker buf.alloc (8192 - 16);
665*2d1272b8SAndroid Build Coastguard Worker
666*2d1272b8SAndroid Build Coastguard Worker while (!pending_subset_tags.is_empty ())
667*2d1272b8SAndroid Build Coastguard Worker {
668*2d1272b8SAndroid Build Coastguard Worker if (subsetted_tags.in_error ()
669*2d1272b8SAndroid Build Coastguard Worker || pending_subset_tags.in_error ()) {
670*2d1272b8SAndroid Build Coastguard Worker success = false;
671*2d1272b8SAndroid Build Coastguard Worker goto end;
672*2d1272b8SAndroid Build Coastguard Worker }
673*2d1272b8SAndroid Build Coastguard Worker
674*2d1272b8SAndroid Build Coastguard Worker bool made_changes = false;
675*2d1272b8SAndroid Build Coastguard Worker for (hb_tag_t tag : pending_subset_tags)
676*2d1272b8SAndroid Build Coastguard Worker {
677*2d1272b8SAndroid Build Coastguard Worker if (!_dependencies_satisfied (plan, tag,
678*2d1272b8SAndroid Build Coastguard Worker subsetted_tags,
679*2d1272b8SAndroid Build Coastguard Worker pending_subset_tags))
680*2d1272b8SAndroid Build Coastguard Worker {
681*2d1272b8SAndroid Build Coastguard Worker // delayed subsetting for some tables since they might have dependency on other tables
682*2d1272b8SAndroid Build Coastguard Worker // in some cases: e.g: during instantiating glyf tables, hmetrics/vmetrics are updated
683*2d1272b8SAndroid Build Coastguard Worker // and saved in subset plan, hmtx/vmtx subsetting need to use these updated metrics values
684*2d1272b8SAndroid Build Coastguard Worker continue;
685*2d1272b8SAndroid Build Coastguard Worker }
686*2d1272b8SAndroid Build Coastguard Worker
687*2d1272b8SAndroid Build Coastguard Worker pending_subset_tags.del (tag);
688*2d1272b8SAndroid Build Coastguard Worker subsetted_tags.add (tag);
689*2d1272b8SAndroid Build Coastguard Worker made_changes = true;
690*2d1272b8SAndroid Build Coastguard Worker
691*2d1272b8SAndroid Build Coastguard Worker success = _subset_table (plan, buf, tag);
692*2d1272b8SAndroid Build Coastguard Worker if (unlikely (!success)) goto end;
693*2d1272b8SAndroid Build Coastguard Worker }
694*2d1272b8SAndroid Build Coastguard Worker
695*2d1272b8SAndroid Build Coastguard Worker if (!made_changes)
696*2d1272b8SAndroid Build Coastguard Worker {
697*2d1272b8SAndroid Build Coastguard Worker DEBUG_MSG (SUBSET, nullptr, "Table dependencies unable to be satisfied. Subset failed.");
698*2d1272b8SAndroid Build Coastguard Worker success = false;
699*2d1272b8SAndroid Build Coastguard Worker goto end;
700*2d1272b8SAndroid Build Coastguard Worker }
701*2d1272b8SAndroid Build Coastguard Worker }
702*2d1272b8SAndroid Build Coastguard Worker }
703*2d1272b8SAndroid Build Coastguard Worker
704*2d1272b8SAndroid Build Coastguard Worker if (success && plan->attach_accelerator_data) {
705*2d1272b8SAndroid Build Coastguard Worker _attach_accelerator_data (plan, plan->dest);
706*2d1272b8SAndroid Build Coastguard Worker }
707*2d1272b8SAndroid Build Coastguard Worker
708*2d1272b8SAndroid Build Coastguard Worker end:
709*2d1272b8SAndroid Build Coastguard Worker return success ? hb_face_reference (plan->dest) : nullptr;
710*2d1272b8SAndroid Build Coastguard Worker }
711