blob: 36af3beae72301bcc03c05415752bf091dd28a1b [file] [log] [blame]
Garret Rieger6a45e5d2018-02-06 16:04:09 -08001/*
2 * Copyright © 2018 Google, Inc.
3 *
4 * This is part of HarfBuzz, a text shaping library.
5 *
6 * Permission is hereby granted, without written agreement and without
7 * license or royalty fees, to use, copy, modify, and distribute this
8 * software and its documentation for any purpose, provided that the
9 * above copyright notice and the following two paragraphs appear in
10 * all copies of this software.
11 *
12 * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
13 * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
14 * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
15 * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
16 * DAMAGE.
17 *
18 * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
19 * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
20 * FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS
21 * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
22 * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
23 *
Rod Sheeter4f074372018-02-21 22:23:05 -080024 * Google Author(s): Garret Rieger, Roderick Sheeter
Garret Rieger6a45e5d2018-02-06 16:04:09 -080025 */
26
Garret Rieger0a5d1442018-02-07 13:09:54 -080027#include "hb-open-type-private.hh"
Garret Rieger53aa0e92018-02-06 17:05:22 -080028#include "hb-ot-glyf-table.hh"
Garret Rieger0a5d1442018-02-07 13:09:54 -080029#include "hb-set.h"
Garret Rieger6a45e5d2018-02-06 16:04:09 -080030#include "hb-subset-glyf.hh"
Garret Rieger21303922018-02-16 17:01:00 -070031#include "hb-subset-plan.hh"
Garret Rieger6a45e5d2018-02-06 16:04:09 -080032
Ebrahim Byagowi203b6472018-02-11 01:00:03 +033033static bool
Garret Rieger8e9fd6f2018-02-07 19:01:21 -080034_calculate_glyf_and_loca_prime_size (const OT::glyf::accelerator_t &glyf,
Behdad Esfahbod5c3112a2018-05-01 19:07:04 -040035 hb_vector_t<hb_codepoint_t> &glyph_ids,
Rod Sheeter4f074372018-02-21 22:23:05 -080036 hb_bool_t drop_hints,
37 bool *use_short_loca /* OUT */,
38 unsigned int *glyf_size /* OUT */,
39 unsigned int *loca_size /* OUT */,
Behdad Esfahbod5c3112a2018-05-01 19:07:04 -040040 hb_vector_t<unsigned int> *instruction_ranges /* OUT */)
Garret Rieger0a5d1442018-02-07 13:09:54 -080041{
42 unsigned int total = 0;
Behdad Esfahbodfd0bde62018-02-13 16:35:30 -080043 for (unsigned int i = 0; i < glyph_ids.len; i++)
44 {
Rod Sheeter59c658c2018-02-08 19:22:47 -080045 hb_codepoint_t next_glyph = glyph_ids[i];
Garret Rieger13966142018-03-20 16:55:42 -070046 if (!instruction_ranges->resize (instruction_ranges->len + 2))
47 {
Garret Rieger45def992018-03-20 17:28:47 -070048 DEBUG_MSG(SUBSET, nullptr, "Failed to resize instruction_ranges.");
Garret Rieger13966142018-03-20 16:55:42 -070049 return false;
50 }
51 unsigned int *instruction_start = &(*instruction_ranges)[instruction_ranges->len - 2];
Rod Sheeter903771b2018-02-26 19:50:06 -080052 *instruction_start = 0;
Garret Rieger13966142018-03-20 16:55:42 -070053 unsigned int *instruction_end = &(*instruction_ranges)[instruction_ranges->len - 1];
Rod Sheeter903771b2018-02-26 19:50:06 -080054 *instruction_end = 0;
Rod Sheeter9bd6d252018-02-23 13:05:58 -080055
Garret Rieger0a5d1442018-02-07 13:09:54 -080056 unsigned int start_offset, end_offset;
Rod Sheeter1454d822018-02-23 17:49:23 -080057 if (unlikely (!(glyf.get_offsets(next_glyph, &start_offset, &end_offset)
Rod Sheeter903771b2018-02-26 19:50:06 -080058 && glyf.remove_padding(start_offset, &end_offset))))
Rod Sheeter4f074372018-02-21 22:23:05 -080059 {
60 DEBUG_MSG(SUBSET, nullptr, "Invalid gid %d", next_glyph);
61 continue;
62 }
Rod Sheeter9bd6d252018-02-23 13:05:58 -080063 if (end_offset - start_offset < OT::glyf::GlyphHeader::static_size)
64 continue; /* 0-length glyph */
65
Rod Sheeter4f074372018-02-21 22:23:05 -080066 if (drop_hints)
67 {
68 if (unlikely (!glyf.get_instruction_offsets(start_offset, end_offset,
Rod Sheeter903771b2018-02-26 19:50:06 -080069 instruction_start, instruction_end)))
Rod Sheeter4f074372018-02-21 22:23:05 -080070 {
71 DEBUG_MSG(SUBSET, nullptr, "Unable to get instruction offsets for %d", next_glyph);
72 return false;
73 }
74 }
Garret Rieger0a5d1442018-02-07 13:09:54 -080075
Rod Sheeter903771b2018-02-26 19:50:06 -080076 total += end_offset - start_offset - (*instruction_end - *instruction_start);
Rod Sheeter9bd6d252018-02-23 13:05:58 -080077 /* round2 so short loca will work */
78 total += total % 2;
Garret Rieger0a5d1442018-02-07 13:09:54 -080079 }
80
Garret Rieger8e9fd6f2018-02-07 19:01:21 -080081 *glyf_size = total;
Garret Riegerd18decd2018-02-09 18:41:21 -080082 *use_short_loca = (total <= 131070);
Rod Sheeter4f074372018-02-21 22:23:05 -080083 *loca_size = (glyph_ids.len + 1)
Garret Riegerd18decd2018-02-09 18:41:21 -080084 * (*use_short_loca ? sizeof(OT::HBUINT16) : sizeof(OT::HBUINT32));
85
86 DEBUG_MSG(SUBSET, nullptr, "preparing to subset glyf: final size %d, loca size %d, using %s loca",
87 total,
88 *loca_size,
89 *use_short_loca ? "short" : "long");
Garret Rieger0a5d1442018-02-07 13:09:54 -080090 return true;
91}
92
Garret Rieger0ab73e52018-02-20 15:33:03 -080093static bool
94_write_loca_entry (unsigned int id,
95 unsigned int offset,
96 bool is_short,
97 void *loca_prime,
98 unsigned int loca_size)
99{
100 unsigned int entry_size = is_short ? sizeof (OT::HBUINT16) : sizeof (OT::HBUINT32);
101 if ((id + 1) * entry_size <= loca_size)
102 {
103 if (is_short) {
104 ((OT::HBUINT16*) loca_prime) [id].set (offset / 2);
105 } else {
106 ((OT::HBUINT32*) loca_prime) [id].set (offset);
107 }
108 return true;
Garret Riegerd18decd2018-02-09 18:41:21 -0800109 }
Garret Rieger0ab73e52018-02-20 15:33:03 -0800110
111 // Offset was not written because the write is out of bounds.
112 DEBUG_MSG (SUBSET,
113 nullptr,
114 "WARNING: Attempted to write an out of bounds loca entry at index %d. Loca size is %d.",
115 id,
116 loca_size);
117 return false;
Garret Riegerd18decd2018-02-09 18:41:21 -0800118}
119
Garret Rieger21303922018-02-16 17:01:00 -0700120static void
121_update_components (hb_subset_plan_t * plan,
122 char * glyph_start,
123 unsigned int length)
Garret Rieger21303922018-02-16 17:01:00 -0700124{
125 OT::glyf::CompositeGlyphHeader::Iterator iterator;
126 if (OT::glyf::CompositeGlyphHeader::get_iterator (glyph_start,
127 length,
128 &iterator))
129 {
130 do
131 {
132 hb_codepoint_t new_gid;
Garret Rieger251cc972018-05-30 12:23:51 -0700133 if (!plan->new_gid_for_old_gid (iterator.current->glyphIndex,
134 &new_gid))
Garret Rieger21303922018-02-16 17:01:00 -0700135 continue;
136
137 ((OT::glyf::CompositeGlyphHeader *) iterator.current)->glyphIndex.set (new_gid);
138 } while (iterator.move_to_next());
139 }
140}
141
Rod Sheeter20d57392018-02-28 11:15:08 -0800142static bool _remove_composite_instruction_flag(char *glyf_prime, unsigned int length)
143{
144 /* remove WE_HAVE_INSTRUCTIONS from flags in dest */
145 OT::glyf::CompositeGlyphHeader::Iterator composite_it;
146 if (unlikely (!OT::glyf::CompositeGlyphHeader::get_iterator (glyf_prime, length, &composite_it))) return false;
147 const OT::glyf::CompositeGlyphHeader *glyph;
148 do {
149 glyph = composite_it.current;
150 OT::HBUINT16 *flags = const_cast<OT::HBUINT16 *> (&glyph->flags);
151 flags->set ( (uint16_t) *flags & ~OT::glyf::CompositeGlyphHeader::WE_HAVE_INSTRUCTIONS);
152 } while (composite_it.move_to_next());
153 return true;
154}
155
Ebrahim Byagowi203b6472018-02-11 01:00:03 +0330156static bool
Garret Rieger21303922018-02-16 17:01:00 -0700157_write_glyf_and_loca_prime (hb_subset_plan_t *plan,
158 const OT::glyf::accelerator_t &glyf,
Garret Rieger8e9fd6f2018-02-07 19:01:21 -0800159 const char *glyf_data,
Garret Riegerd18decd2018-02-09 18:41:21 -0800160 bool use_short_loca,
Behdad Esfahbod5c3112a2018-05-01 19:07:04 -0400161 hb_vector_t<unsigned int> &instruction_ranges,
Garret Rieger0ab73e52018-02-20 15:33:03 -0800162 unsigned int glyf_prime_size,
Garret Rieger8e9fd6f2018-02-07 19:01:21 -0800163 char *glyf_prime_data /* OUT */,
Garret Rieger0ab73e52018-02-20 15:33:03 -0800164 unsigned int loca_prime_size,
Garret Rieger8e9fd6f2018-02-07 19:01:21 -0800165 char *loca_prime_data /* OUT */)
Garret Rieger0a5d1442018-02-07 13:09:54 -0800166{
Garret Rieger251cc972018-05-30 12:23:51 -0700167 hb_vector_t<hb_codepoint_t> &glyph_ids = plan->glyphs;
Garret Rieger0a5d1442018-02-07 13:09:54 -0800168 char *glyf_prime_data_next = glyf_prime_data;
169
Garret Rieger0ab73e52018-02-20 15:33:03 -0800170 bool success = true;
Behdad Esfahbodfd0bde62018-02-13 16:35:30 -0800171 for (unsigned int i = 0; i < glyph_ids.len; i++)
172 {
173 unsigned int start_offset, end_offset;
Rod Sheeter1454d822018-02-23 17:49:23 -0800174 if (unlikely (!(glyf.get_offsets (glyph_ids[i], &start_offset, &end_offset)
Rod Sheeter903771b2018-02-26 19:50:06 -0800175 && glyf.remove_padding(start_offset, &end_offset))))
Behdad Esfahbodfd0bde62018-02-13 16:35:30 -0800176 end_offset = start_offset = 0;
Garret Rieger07851aa2018-03-26 20:56:56 -0600177
Rod Sheeter4f074372018-02-21 22:23:05 -0800178 unsigned int instruction_start = instruction_ranges[i * 2];
179 unsigned int instruction_end = instruction_ranges[i * 2 + 1];
Garret Rieger0a5d1442018-02-07 13:09:54 -0800180
Rod Sheeter4f074372018-02-21 22:23:05 -0800181 int length = end_offset - start_offset - (instruction_end - instruction_start);
Garret Riegera998eee2018-02-20 16:48:52 -0800182
183 if (glyf_prime_data_next + length > glyf_prime_data + glyf_prime_size)
184 {
185 DEBUG_MSG (SUBSET,
186 nullptr,
Rod Sheeter4f074372018-02-21 22:23:05 -0800187 "WARNING: Attempted to write an out of bounds glyph entry for gid %d (length %d)",
188 i, length);
Garret Riegera998eee2018-02-20 16:48:52 -0800189 return false;
190 }
Rod Sheeter4f074372018-02-21 22:23:05 -0800191
192 if (instruction_start == instruction_end)
Rod Sheeter4f074372018-02-21 22:23:05 -0800193 memcpy (glyf_prime_data_next, glyf_data + start_offset, length);
Rod Sheeter4f074372018-02-21 22:23:05 -0800194 else
195 {
Rod Sheeter4f074372018-02-21 22:23:05 -0800196 memcpy (glyf_prime_data_next, glyf_data + start_offset, instruction_start - start_offset);
197 memcpy (glyf_prime_data_next + instruction_start - start_offset, glyf_data + instruction_end, end_offset - instruction_end);
Rod Sheeterb3f1a042018-02-28 11:41:24 -0800198 /* if the instructions end at the end this was a composite glyph, else simple */
Rod Sheeter4f074372018-02-21 22:23:05 -0800199 if (instruction_end == end_offset)
Rod Sheeter6836a822018-02-27 20:51:12 -0800200 {
Rod Sheeter20d57392018-02-28 11:15:08 -0800201 if (unlikely (!_remove_composite_instruction_flag (glyf_prime_data_next, length))) return false;
Rod Sheeter6836a822018-02-27 20:51:12 -0800202 }
Rod Sheeter4f074372018-02-21 22:23:05 -0800203 else
Rod Sheeter9bd6d252018-02-23 13:05:58 -0800204 /* zero instruction length, which is just before instruction_start */
205 memset (glyf_prime_data_next + instruction_start - start_offset - 2, 0, 2);
Rod Sheeter4f074372018-02-21 22:23:05 -0800206 }
Garret Riegerd18decd2018-02-09 18:41:21 -0800207
Garret Rieger0ab73e52018-02-20 15:33:03 -0800208 success = success && _write_loca_entry (i,
209 glyf_prime_data_next - glyf_prime_data,
210 use_short_loca,
211 loca_prime_data,
212 loca_prime_size);
Rod Sheeter4f074372018-02-21 22:23:05 -0800213 _update_components (plan, glyf_prime_data_next, length);
Garret Rieger8e9fd6f2018-02-07 19:01:21 -0800214
Garret Rieger07851aa2018-03-26 20:56:56 -0600215 // TODO: don't align to two bytes if using long loca.
216 glyf_prime_data_next += length + (length % 2); // Align to 2 bytes for short loca.
Garret Rieger0a5d1442018-02-07 13:09:54 -0800217 }
218
Garret Rieger0ab73e52018-02-20 15:33:03 -0800219 success = success && _write_loca_entry (glyph_ids.len,
220 glyf_prime_data_next - glyf_prime_data,
221 use_short_loca,
222 loca_prime_data,
223 loca_prime_size);
224 return success;
Garret Rieger0a5d1442018-02-07 13:09:54 -0800225}
226
Ebrahim Byagowi203b6472018-02-11 01:00:03 +0330227static bool
Garret Rieger8e9fd6f2018-02-07 19:01:21 -0800228_hb_subset_glyf_and_loca (const OT::glyf::accelerator_t &glyf,
229 const char *glyf_data,
Rod Sheeter4f074372018-02-21 22:23:05 -0800230 hb_subset_plan_t *plan,
Garret Riegerd18decd2018-02-09 18:41:21 -0800231 bool *use_short_loca,
Garret Rieger8e9fd6f2018-02-07 19:01:21 -0800232 hb_blob_t **glyf_prime /* OUT */,
233 hb_blob_t **loca_prime /* OUT */)
Garret Rieger4e1abe22018-02-07 13:28:11 -0800234{
Garret Rieger4e1abe22018-02-07 13:28:11 -0800235 // TODO(grieger): Sanity check allocation size for the new table.
Garret Rieger251cc972018-05-30 12:23:51 -0700236 hb_vector_t<hb_codepoint_t> &glyphs_to_retain = plan->glyphs;
Garret Rieger4e1abe22018-02-07 13:28:11 -0800237
238 unsigned int glyf_prime_size;
Garret Rieger8e9fd6f2018-02-07 19:01:21 -0800239 unsigned int loca_prime_size;
Behdad Esfahbod5c3112a2018-05-01 19:07:04 -0400240 hb_vector_t<unsigned int> instruction_ranges;
Rod Sheeter4f074372018-02-21 22:23:05 -0800241 instruction_ranges.init();
Garret Riegerd18decd2018-02-09 18:41:21 -0800242
Garret Rieger8e9fd6f2018-02-07 19:01:21 -0800243 if (unlikely (!_calculate_glyf_and_loca_prime_size (glyf,
244 glyphs_to_retain,
Rod Sheeter4f074372018-02-21 22:23:05 -0800245 plan->drop_hints,
Garret Riegerd18decd2018-02-09 18:41:21 -0800246 use_short_loca,
Garret Rieger8e9fd6f2018-02-07 19:01:21 -0800247 &glyf_prime_size,
Rod Sheeter4f074372018-02-21 22:23:05 -0800248 &loca_prime_size,
249 &instruction_ranges))) {
Behdad Esfahboda60ba792018-05-01 19:01:25 -0400250 instruction_ranges.fini();
Garret Rieger4e1abe22018-02-07 13:28:11 -0800251 return false;
252 }
253
Rod Sheeter0ac8c0c2018-02-23 17:43:00 -0800254 char *glyf_prime_data = (char *) calloc (1, glyf_prime_size);
255 char *loca_prime_data = (char *) calloc (1, loca_prime_size);
Garret Rieger21303922018-02-16 17:01:00 -0700256 if (unlikely (!_write_glyf_and_loca_prime (plan, glyf, glyf_data,
Garret Riegerd18decd2018-02-09 18:41:21 -0800257 *use_short_loca,
Rod Sheeter4f074372018-02-21 22:23:05 -0800258 instruction_ranges,
Garret Rieger8e9fd6f2018-02-07 19:01:21 -0800259 glyf_prime_size, glyf_prime_data,
260 loca_prime_size, loca_prime_data))) {
Garret Rieger4e1abe22018-02-07 13:28:11 -0800261 free (glyf_prime_data);
Garret Rieger0ab73e52018-02-20 15:33:03 -0800262 free (loca_prime_data);
Behdad Esfahboda60ba792018-05-01 19:01:25 -0400263 instruction_ranges.fini();
Garret Rieger4e1abe22018-02-07 13:28:11 -0800264 return false;
265 }
Behdad Esfahboda60ba792018-05-01 19:01:25 -0400266 instruction_ranges.fini();
Garret Rieger4e1abe22018-02-07 13:28:11 -0800267
268 *glyf_prime = hb_blob_create (glyf_prime_data,
269 glyf_prime_size,
270 HB_MEMORY_MODE_READONLY,
271 glyf_prime_data,
272 free);
Garret Rieger8e9fd6f2018-02-07 19:01:21 -0800273 *loca_prime = hb_blob_create (loca_prime_data,
274 loca_prime_size,
275 HB_MEMORY_MODE_READONLY,
276 loca_prime_data,
277 free);
Garret Rieger4e1abe22018-02-07 13:28:11 -0800278 return true;
279}
280
Garret Rieger6a45e5d2018-02-06 16:04:09 -0800281/**
282 * hb_subset_glyf:
283 * Subsets the glyph table according to a provided plan.
284 *
285 * Return value: subsetted glyf table.
286 *
287 * Since: 1.7.5
288 **/
289bool
Garret Riegerf9c665f2018-02-07 16:53:18 -0800290hb_subset_glyf_and_loca (hb_subset_plan_t *plan,
Garret Rieger1582eab2018-02-09 12:52:08 -0800291 bool *use_short_loca, /* OUT */
292 hb_blob_t **glyf_prime, /* OUT */
Garret Riegerf9c665f2018-02-07 16:53:18 -0800293 hb_blob_t **loca_prime /* OUT */)
Garret Rieger6a45e5d2018-02-06 16:04:09 -0800294{
Behdad Esfahboded7b2e52018-08-01 23:59:09 -0700295 hb_blob_t *glyf_blob = hb_sanitize_context_t ().reference_table<OT::glyf> (plan->source);
Garret Rieger0a5d1442018-02-07 13:09:54 -0800296 const char *glyf_data = hb_blob_get_data(glyf_blob, nullptr);
Garret Rieger53aa0e92018-02-06 17:05:22 -0800297
Garret Rieger0a5d1442018-02-07 13:09:54 -0800298 OT::glyf::accelerator_t glyf;
Rod Sheeterfa877702018-02-14 14:16:25 -0800299 glyf.init(plan->source);
Garret Riegerd18decd2018-02-09 18:41:21 -0800300 bool result = _hb_subset_glyf_and_loca (glyf,
301 glyf_data,
Garret Rieger21303922018-02-16 17:01:00 -0700302 plan,
Garret Riegerd18decd2018-02-09 18:41:21 -0800303 use_short_loca,
304 glyf_prime,
305 loca_prime);
Garret Rieger99967e22018-02-23 15:45:45 -0800306
307 hb_blob_destroy (glyf_blob);
Garret Rieger0a5d1442018-02-07 13:09:54 -0800308 glyf.fini();
Garret Rieger4e1abe22018-02-07 13:28:11 -0800309
310 return result;
Garret Rieger6a45e5d2018-02-06 16:04:09 -0800311}