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): Roderick Sheeter
25*2d1272b8SAndroid Build Coastguard Worker */
26*2d1272b8SAndroid Build Coastguard Worker
27*2d1272b8SAndroid Build Coastguard Worker #include "hb-test.h"
28*2d1272b8SAndroid Build Coastguard Worker #include "hb-subset-test.h"
29*2d1272b8SAndroid Build Coastguard Worker
30*2d1272b8SAndroid Build Coastguard Worker /* Unit tests for hmtx subsetting */
31*2d1272b8SAndroid Build Coastguard Worker
check_num_hmetrics(hb_face_t * face,uint16_t expected_num_hmetrics)32*2d1272b8SAndroid Build Coastguard Worker static void check_num_hmetrics(hb_face_t *face, uint16_t expected_num_hmetrics)
33*2d1272b8SAndroid Build Coastguard Worker {
34*2d1272b8SAndroid Build Coastguard Worker hb_blob_t *hhea_blob = hb_face_reference_table (face, HB_TAG ('h','h','e','a'));
35*2d1272b8SAndroid Build Coastguard Worker hb_blob_t *hmtx_blob = hb_face_reference_table (face, HB_TAG ('h','m','t','x'));
36*2d1272b8SAndroid Build Coastguard Worker
37*2d1272b8SAndroid Build Coastguard Worker // TODO I sure wish I could just use the hmtx table struct!
38*2d1272b8SAndroid Build Coastguard Worker unsigned int hhea_len;
39*2d1272b8SAndroid Build Coastguard Worker uint8_t *raw_hhea = (uint8_t *) hb_blob_get_data(hhea_blob, &hhea_len);
40*2d1272b8SAndroid Build Coastguard Worker uint16_t num_hmetrics = (raw_hhea[hhea_len - 2] << 8) + raw_hhea[hhea_len - 1];
41*2d1272b8SAndroid Build Coastguard Worker g_assert_cmpuint(expected_num_hmetrics, ==, num_hmetrics);
42*2d1272b8SAndroid Build Coastguard Worker
43*2d1272b8SAndroid Build Coastguard Worker hb_blob_destroy (hhea_blob);
44*2d1272b8SAndroid Build Coastguard Worker hb_blob_destroy (hmtx_blob);
45*2d1272b8SAndroid Build Coastguard Worker }
46*2d1272b8SAndroid Build Coastguard Worker
47*2d1272b8SAndroid Build Coastguard Worker static void
test_subset_hmtx_simple_subset(void)48*2d1272b8SAndroid Build Coastguard Worker test_subset_hmtx_simple_subset (void)
49*2d1272b8SAndroid Build Coastguard Worker {
50*2d1272b8SAndroid Build Coastguard Worker hb_face_t *face_abc = hb_test_open_font_file ("fonts/Roboto-Regular.abc.ttf");
51*2d1272b8SAndroid Build Coastguard Worker hb_face_t *face_ac = hb_test_open_font_file ("fonts/Roboto-Regular.ac.ttf");
52*2d1272b8SAndroid Build Coastguard Worker
53*2d1272b8SAndroid Build Coastguard Worker hb_set_t *codepoints = hb_set_create ();
54*2d1272b8SAndroid Build Coastguard Worker hb_face_t *face_abc_subset;
55*2d1272b8SAndroid Build Coastguard Worker hb_set_add (codepoints, 'a');
56*2d1272b8SAndroid Build Coastguard Worker hb_set_add (codepoints, 'c');
57*2d1272b8SAndroid Build Coastguard Worker face_abc_subset = hb_subset_test_create_subset (face_abc, hb_subset_test_create_input (codepoints));
58*2d1272b8SAndroid Build Coastguard Worker hb_set_destroy (codepoints);
59*2d1272b8SAndroid Build Coastguard Worker
60*2d1272b8SAndroid Build Coastguard Worker check_num_hmetrics(face_abc_subset, 3); /* nothing has same width */
61*2d1272b8SAndroid Build Coastguard Worker hb_subset_test_check (face_ac, face_abc_subset, HB_TAG ('h','m','t','x'));
62*2d1272b8SAndroid Build Coastguard Worker
63*2d1272b8SAndroid Build Coastguard Worker hb_face_destroy (face_abc_subset);
64*2d1272b8SAndroid Build Coastguard Worker hb_face_destroy (face_abc);
65*2d1272b8SAndroid Build Coastguard Worker hb_face_destroy (face_ac);
66*2d1272b8SAndroid Build Coastguard Worker }
67*2d1272b8SAndroid Build Coastguard Worker
68*2d1272b8SAndroid Build Coastguard Worker
69*2d1272b8SAndroid Build Coastguard Worker static void
test_subset_hmtx_monospace(void)70*2d1272b8SAndroid Build Coastguard Worker test_subset_hmtx_monospace (void)
71*2d1272b8SAndroid Build Coastguard Worker {
72*2d1272b8SAndroid Build Coastguard Worker hb_face_t *face_abc = hb_test_open_font_file ("fonts/Inconsolata-Regular.abc.ttf");
73*2d1272b8SAndroid Build Coastguard Worker hb_face_t *face_ac = hb_test_open_font_file ("fonts/Inconsolata-Regular.ac.ttf");
74*2d1272b8SAndroid Build Coastguard Worker
75*2d1272b8SAndroid Build Coastguard Worker hb_set_t *codepoints = hb_set_create ();
76*2d1272b8SAndroid Build Coastguard Worker hb_face_t *face_abc_subset;
77*2d1272b8SAndroid Build Coastguard Worker hb_set_add (codepoints, 'a');
78*2d1272b8SAndroid Build Coastguard Worker hb_set_add (codepoints, 'c');
79*2d1272b8SAndroid Build Coastguard Worker face_abc_subset = hb_subset_test_create_subset (face_abc, hb_subset_test_create_input (codepoints));
80*2d1272b8SAndroid Build Coastguard Worker hb_set_destroy (codepoints);
81*2d1272b8SAndroid Build Coastguard Worker
82*2d1272b8SAndroid Build Coastguard Worker check_num_hmetrics(face_abc_subset, 1); /* everything has same width */
83*2d1272b8SAndroid Build Coastguard Worker hb_subset_test_check (face_ac, face_abc_subset, HB_TAG ('h','m','t','x'));
84*2d1272b8SAndroid Build Coastguard Worker
85*2d1272b8SAndroid Build Coastguard Worker hb_face_destroy (face_abc_subset);
86*2d1272b8SAndroid Build Coastguard Worker hb_face_destroy (face_abc);
87*2d1272b8SAndroid Build Coastguard Worker hb_face_destroy (face_ac);
88*2d1272b8SAndroid Build Coastguard Worker }
89*2d1272b8SAndroid Build Coastguard Worker
90*2d1272b8SAndroid Build Coastguard Worker
91*2d1272b8SAndroid Build Coastguard Worker static void
test_subset_hmtx_keep_num_metrics(void)92*2d1272b8SAndroid Build Coastguard Worker test_subset_hmtx_keep_num_metrics (void)
93*2d1272b8SAndroid Build Coastguard Worker {
94*2d1272b8SAndroid Build Coastguard Worker hb_face_t *face_abc = hb_test_open_font_file ("fonts/Inconsolata-Regular.abc.widerc.ttf");
95*2d1272b8SAndroid Build Coastguard Worker hb_face_t *face_ac = hb_test_open_font_file ("fonts/Inconsolata-Regular.ac.widerc.ttf");
96*2d1272b8SAndroid Build Coastguard Worker
97*2d1272b8SAndroid Build Coastguard Worker hb_set_t *codepoints = hb_set_create ();
98*2d1272b8SAndroid Build Coastguard Worker hb_face_t *face_abc_subset;
99*2d1272b8SAndroid Build Coastguard Worker hb_set_add (codepoints, 'a');
100*2d1272b8SAndroid Build Coastguard Worker hb_set_add (codepoints, 'c');
101*2d1272b8SAndroid Build Coastguard Worker face_abc_subset = hb_subset_test_create_subset (face_abc, hb_subset_test_create_input (codepoints));
102*2d1272b8SAndroid Build Coastguard Worker hb_set_destroy (codepoints);
103*2d1272b8SAndroid Build Coastguard Worker
104*2d1272b8SAndroid Build Coastguard Worker check_num_hmetrics(face_abc_subset, 3); /* c is wider */
105*2d1272b8SAndroid Build Coastguard Worker hb_subset_test_check (face_ac, face_abc_subset, HB_TAG ('h','m','t','x'));
106*2d1272b8SAndroid Build Coastguard Worker
107*2d1272b8SAndroid Build Coastguard Worker hb_face_destroy (face_abc_subset);
108*2d1272b8SAndroid Build Coastguard Worker hb_face_destroy (face_abc);
109*2d1272b8SAndroid Build Coastguard Worker hb_face_destroy (face_ac);
110*2d1272b8SAndroid Build Coastguard Worker }
111*2d1272b8SAndroid Build Coastguard Worker
112*2d1272b8SAndroid Build Coastguard Worker static void
test_subset_hmtx_decrease_num_metrics(void)113*2d1272b8SAndroid Build Coastguard Worker test_subset_hmtx_decrease_num_metrics (void)
114*2d1272b8SAndroid Build Coastguard Worker {
115*2d1272b8SAndroid Build Coastguard Worker hb_face_t *face_abc = hb_test_open_font_file ("fonts/Inconsolata-Regular.abc.widerc.ttf");
116*2d1272b8SAndroid Build Coastguard Worker hb_face_t *face_ab = hb_test_open_font_file ("fonts/Inconsolata-Regular.ab.ttf");
117*2d1272b8SAndroid Build Coastguard Worker
118*2d1272b8SAndroid Build Coastguard Worker hb_set_t *codepoints = hb_set_create ();
119*2d1272b8SAndroid Build Coastguard Worker hb_face_t *face_abc_subset;
120*2d1272b8SAndroid Build Coastguard Worker hb_set_add (codepoints, 'a');
121*2d1272b8SAndroid Build Coastguard Worker hb_set_add (codepoints, 'b');
122*2d1272b8SAndroid Build Coastguard Worker face_abc_subset = hb_subset_test_create_subset (face_abc, hb_subset_test_create_input (codepoints));
123*2d1272b8SAndroid Build Coastguard Worker hb_set_destroy (codepoints);
124*2d1272b8SAndroid Build Coastguard Worker
125*2d1272b8SAndroid Build Coastguard Worker check_num_hmetrics(face_abc_subset, 1); /* everything left has same width */
126*2d1272b8SAndroid Build Coastguard Worker hb_subset_test_check (face_ab, face_abc_subset, HB_TAG ('h','m','t','x'));
127*2d1272b8SAndroid Build Coastguard Worker
128*2d1272b8SAndroid Build Coastguard Worker hb_face_destroy (face_abc_subset);
129*2d1272b8SAndroid Build Coastguard Worker hb_face_destroy (face_abc);
130*2d1272b8SAndroid Build Coastguard Worker hb_face_destroy (face_ab);
131*2d1272b8SAndroid Build Coastguard Worker }
132*2d1272b8SAndroid Build Coastguard Worker
133*2d1272b8SAndroid Build Coastguard Worker static void
test_subset_hmtx_noop(void)134*2d1272b8SAndroid Build Coastguard Worker test_subset_hmtx_noop (void)
135*2d1272b8SAndroid Build Coastguard Worker {
136*2d1272b8SAndroid Build Coastguard Worker hb_face_t *face_abc = hb_test_open_font_file ("fonts/Roboto-Regular.abc.ttf");
137*2d1272b8SAndroid Build Coastguard Worker
138*2d1272b8SAndroid Build Coastguard Worker hb_set_t *codepoints = hb_set_create();
139*2d1272b8SAndroid Build Coastguard Worker hb_face_t *face_abc_subset;
140*2d1272b8SAndroid Build Coastguard Worker hb_set_add (codepoints, 'a');
141*2d1272b8SAndroid Build Coastguard Worker hb_set_add (codepoints, 'b');
142*2d1272b8SAndroid Build Coastguard Worker hb_set_add (codepoints, 'c');
143*2d1272b8SAndroid Build Coastguard Worker face_abc_subset = hb_subset_test_create_subset (face_abc, hb_subset_test_create_input (codepoints));
144*2d1272b8SAndroid Build Coastguard Worker hb_set_destroy (codepoints);
145*2d1272b8SAndroid Build Coastguard Worker
146*2d1272b8SAndroid Build Coastguard Worker check_num_hmetrics(face_abc_subset, 4); /* nothing has same width */
147*2d1272b8SAndroid Build Coastguard Worker hb_subset_test_check (face_abc, face_abc_subset, HB_TAG ('h','m','t','x'));
148*2d1272b8SAndroid Build Coastguard Worker
149*2d1272b8SAndroid Build Coastguard Worker hb_face_destroy (face_abc_subset);
150*2d1272b8SAndroid Build Coastguard Worker hb_face_destroy (face_abc);
151*2d1272b8SAndroid Build Coastguard Worker }
152*2d1272b8SAndroid Build Coastguard Worker
153*2d1272b8SAndroid Build Coastguard Worker static void
test_subset_invalid_hmtx(void)154*2d1272b8SAndroid Build Coastguard Worker test_subset_invalid_hmtx (void)
155*2d1272b8SAndroid Build Coastguard Worker {
156*2d1272b8SAndroid Build Coastguard Worker hb_face_t *face = hb_test_open_font_file ("../fuzzing/fonts/crash-e4e0bb1458a91b692eba492c907ae1f94e635480");
157*2d1272b8SAndroid Build Coastguard Worker hb_face_t *subset;
158*2d1272b8SAndroid Build Coastguard Worker
159*2d1272b8SAndroid Build Coastguard Worker hb_subset_input_t *input = hb_subset_input_create_or_fail ();
160*2d1272b8SAndroid Build Coastguard Worker hb_set_t *codepoints = hb_subset_input_unicode_set (input);
161*2d1272b8SAndroid Build Coastguard Worker hb_set_add (codepoints, 'a');
162*2d1272b8SAndroid Build Coastguard Worker hb_set_add (codepoints, 'b');
163*2d1272b8SAndroid Build Coastguard Worker hb_set_add (codepoints, 'c');
164*2d1272b8SAndroid Build Coastguard Worker
165*2d1272b8SAndroid Build Coastguard Worker subset = hb_subset_or_fail (face, input);
166*2d1272b8SAndroid Build Coastguard Worker g_assert (!subset);
167*2d1272b8SAndroid Build Coastguard Worker
168*2d1272b8SAndroid Build Coastguard Worker hb_subset_input_destroy (input);
169*2d1272b8SAndroid Build Coastguard Worker hb_face_destroy (subset);
170*2d1272b8SAndroid Build Coastguard Worker hb_face_destroy (face);
171*2d1272b8SAndroid Build Coastguard Worker }
172*2d1272b8SAndroid Build Coastguard Worker
173*2d1272b8SAndroid Build Coastguard Worker int
main(int argc,char ** argv)174*2d1272b8SAndroid Build Coastguard Worker main (int argc, char **argv)
175*2d1272b8SAndroid Build Coastguard Worker {
176*2d1272b8SAndroid Build Coastguard Worker hb_test_init (&argc, &argv);
177*2d1272b8SAndroid Build Coastguard Worker
178*2d1272b8SAndroid Build Coastguard Worker hb_test_add (test_subset_hmtx_simple_subset);
179*2d1272b8SAndroid Build Coastguard Worker hb_test_add (test_subset_hmtx_monospace);
180*2d1272b8SAndroid Build Coastguard Worker hb_test_add (test_subset_hmtx_keep_num_metrics);
181*2d1272b8SAndroid Build Coastguard Worker hb_test_add (test_subset_hmtx_decrease_num_metrics);
182*2d1272b8SAndroid Build Coastguard Worker hb_test_add (test_subset_hmtx_noop);
183*2d1272b8SAndroid Build Coastguard Worker hb_test_add (test_subset_invalid_hmtx);
184*2d1272b8SAndroid Build Coastguard Worker
185*2d1272b8SAndroid Build Coastguard Worker return hb_test_run();
186*2d1272b8SAndroid Build Coastguard Worker }
187