blob: 27ade28fe193e7ad57f54df4447ca06d13f05df8 [file] [log] [blame]
Behdad Esfahboda0175e72017-08-17 16:55:54 -07001/*
2 * Copyright © 2017 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 *
24 * Google Author(s): Behdad Esfahbod
25 */
26
Behdad Esfahbodc77ae402018-08-25 22:36:36 -070027#ifndef HB_AAT_LAYOUT_COMMON_HH
28#define HB_AAT_LAYOUT_COMMON_HH
Behdad Esfahboda0175e72017-08-17 16:55:54 -070029
Behdad Esfahbodc77ae402018-08-25 22:36:36 -070030#include "hb-aat-layout.hh"
Ebrahim Byagowi47cf9a92018-12-08 10:20:25 +033031#include "hb-open-type.hh"
Behdad Esfahboda0175e72017-08-17 16:55:54 -070032
33
34namespace AAT {
35
36using namespace OT;
37
38
39/*
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +010040 * Lookup Table
41 */
42
43template <typename T> struct Lookup;
44
45template <typename T>
46struct LookupFormat0
47{
48 friend struct Lookup<T>;
49
50 private:
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +033051 const T* get_value (hb_codepoint_t glyph_id, unsigned int num_glyphs) const
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +010052 {
Behdad Esfahbod748b9892018-01-09 17:55:17 +010053 if (unlikely (glyph_id >= num_glyphs)) return nullptr;
54 return &arrayZ[glyph_id];
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +010055 }
56
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +033057 bool sanitize (hb_sanitize_context_t *c) const
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +010058 {
59 TRACE_SANITIZE (this);
Behdad Esfahbodbe707382018-07-17 18:45:25 +020060 return_trace (arrayZ.sanitize (c, c->get_num_glyphs ()));
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +010061 }
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +033062 bool sanitize (hb_sanitize_context_t *c, const void *base) const
Ebrahim Byagowib8b00fb2018-11-08 18:53:14 +033063 {
64 TRACE_SANITIZE (this);
65 return_trace (arrayZ.sanitize (c, c->get_num_glyphs (), base));
66 }
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +010067
68 protected:
Behdad Esfahbod6b191782018-01-10 03:07:30 +010069 HBUINT16 format; /* Format identifier--format = 0 */
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +010070 UnsizedArrayOf<T>
71 arrayZ; /* Array of lookup values, indexed by glyph index. */
72 public:
Behdad Esfahbod3b9fd172018-11-22 01:18:55 -050073 DEFINE_SIZE_UNBOUNDED (2);
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +010074};
75
76
77template <typename T>
78struct LookupSegmentSingle
79{
Behdad Esfahbod5d4b0372019-01-22 12:11:24 +010080 static constexpr unsigned TerminationWordCount = 2u;
Behdad Esfahbod3d309722018-11-24 23:12:28 -050081
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +033082 int cmp (hb_codepoint_t g) const
83 { return g < first ? -1 : g <= last ? 0 : +1 ; }
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +010084
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +033085 bool sanitize (hb_sanitize_context_t *c) const
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +010086 {
87 TRACE_SANITIZE (this);
88 return_trace (c->check_struct (this) && value.sanitize (c));
89 }
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +033090 bool sanitize (hb_sanitize_context_t *c, const void *base) const
Ebrahim Byagowib8b00fb2018-11-08 18:53:14 +033091 {
92 TRACE_SANITIZE (this);
93 return_trace (c->check_struct (this) && value.sanitize (c, base));
94 }
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +010095
96 GlyphID last; /* Last GlyphID in this segment */
97 GlyphID first; /* First GlyphID in this segment */
98 T value; /* The lookup value (only one) */
99 public:
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100100 DEFINE_SIZE_STATIC (4 + T::static_size);
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100101};
102
103template <typename T>
104struct LookupFormat2
105{
106 friend struct Lookup<T>;
107
108 private:
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330109 const T* get_value (hb_codepoint_t glyph_id) const
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100110 {
111 const LookupSegmentSingle<T> *v = segments.bsearch (glyph_id);
Behdad Esfahbod748b9892018-01-09 17:55:17 +0100112 return v ? &v->value : nullptr;
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100113 }
114
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330115 bool sanitize (hb_sanitize_context_t *c) const
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100116 {
117 TRACE_SANITIZE (this);
118 return_trace (segments.sanitize (c));
119 }
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330120 bool sanitize (hb_sanitize_context_t *c, const void *base) const
Ebrahim Byagowib8b00fb2018-11-08 18:53:14 +0330121 {
122 TRACE_SANITIZE (this);
123 return_trace (segments.sanitize (c, base));
124 }
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100125
126 protected:
Behdad Esfahbod6b191782018-01-10 03:07:30 +0100127 HBUINT16 format; /* Format identifier--format = 2 */
Behdad Esfahbod3515c8b2018-10-07 22:27:00 -0400128 VarSizedBinSearchArrayOf<LookupSegmentSingle<T> >
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100129 segments; /* The actual segments. These must already be sorted,
130 * according to the first word in each one (the last
131 * glyph in each segment). */
132 public:
133 DEFINE_SIZE_ARRAY (8, segments);
134};
135
136template <typename T>
137struct LookupSegmentArray
138{
Behdad Esfahbod5d4b0372019-01-22 12:11:24 +0100139 static constexpr unsigned TerminationWordCount = 2u;
Behdad Esfahbod3d309722018-11-24 23:12:28 -0500140
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330141 const T* get_value (hb_codepoint_t glyph_id, const void *base) const
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100142 {
Behdad Esfahbod748b9892018-01-09 17:55:17 +0100143 return first <= glyph_id && glyph_id <= last ? &(base+valuesZ)[glyph_id - first] : nullptr;
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100144 }
145
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330146 int cmp (hb_codepoint_t g) const
147 { return g < first ? -1 : g <= last ? 0 : +1; }
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100148
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330149 bool sanitize (hb_sanitize_context_t *c, const void *base) const
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100150 {
151 TRACE_SANITIZE (this);
152 return_trace (c->check_struct (this) &&
153 first <= last &&
154 valuesZ.sanitize (c, base, last - first + 1));
155 }
Ebrahim Byagowib8b00fb2018-11-08 18:53:14 +0330156 template <typename T2>
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330157 bool sanitize (hb_sanitize_context_t *c, const void *base, T2 user_data) const
Ebrahim Byagowib8b00fb2018-11-08 18:53:14 +0330158 {
159 TRACE_SANITIZE (this);
160 return_trace (c->check_struct (this) &&
161 first <= last &&
162 valuesZ.sanitize (c, base, last - first + 1, user_data));
163 }
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100164
165 GlyphID last; /* Last GlyphID in this segment */
166 GlyphID first; /* First GlyphID in this segment */
Behdad Esfahbod7e6bd512019-01-17 18:24:18 -0500167 NNOffsetTo<UnsizedArrayOf<T> >
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100168 valuesZ; /* A 16-bit offset from the start of
169 * the table to the data. */
170 public:
171 DEFINE_SIZE_STATIC (6);
172};
173
174template <typename T>
175struct LookupFormat4
176{
177 friend struct Lookup<T>;
178
179 private:
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330180 const T* get_value (hb_codepoint_t glyph_id) const
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100181 {
182 const LookupSegmentArray<T> *v = segments.bsearch (glyph_id);
Behdad Esfahbod748b9892018-01-09 17:55:17 +0100183 return v ? v->get_value (glyph_id, this) : nullptr;
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100184 }
185
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330186 bool sanitize (hb_sanitize_context_t *c) const
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100187 {
188 TRACE_SANITIZE (this);
189 return_trace (segments.sanitize (c, this));
190 }
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330191 bool sanitize (hb_sanitize_context_t *c, const void *base) const
Ebrahim Byagowib8b00fb2018-11-08 18:53:14 +0330192 {
193 TRACE_SANITIZE (this);
194 return_trace (segments.sanitize (c, this, base));
195 }
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100196
197 protected:
Behdad Esfahbod2c824d32018-10-11 16:41:01 -0400198 HBUINT16 format; /* Format identifier--format = 4 */
Behdad Esfahbod3515c8b2018-10-07 22:27:00 -0400199 VarSizedBinSearchArrayOf<LookupSegmentArray<T> >
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100200 segments; /* The actual segments. These must already be sorted,
201 * according to the first word in each one (the last
202 * glyph in each segment). */
203 public:
204 DEFINE_SIZE_ARRAY (8, segments);
205};
206
207template <typename T>
208struct LookupSingle
209{
Behdad Esfahbod5d4b0372019-01-22 12:11:24 +0100210 static constexpr unsigned TerminationWordCount = 1u;
Behdad Esfahbod3d309722018-11-24 23:12:28 -0500211
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330212 int cmp (hb_codepoint_t g) const { return glyph.cmp (g); }
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100213
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330214 bool sanitize (hb_sanitize_context_t *c) const
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100215 {
216 TRACE_SANITIZE (this);
217 return_trace (c->check_struct (this) && value.sanitize (c));
218 }
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330219 bool sanitize (hb_sanitize_context_t *c, const void *base) const
Ebrahim Byagowib8b00fb2018-11-08 18:53:14 +0330220 {
221 TRACE_SANITIZE (this);
222 return_trace (c->check_struct (this) && value.sanitize (c, base));
223 }
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100224
225 GlyphID glyph; /* Last GlyphID */
226 T value; /* The lookup value (only one) */
227 public:
Behdad Esfahbod2c824d32018-10-11 16:41:01 -0400228 DEFINE_SIZE_STATIC (2 + T::static_size);
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100229};
230
231template <typename T>
232struct LookupFormat6
233{
234 friend struct Lookup<T>;
235
236 private:
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330237 const T* get_value (hb_codepoint_t glyph_id) const
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100238 {
239 const LookupSingle<T> *v = entries.bsearch (glyph_id);
Behdad Esfahbod748b9892018-01-09 17:55:17 +0100240 return v ? &v->value : nullptr;
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100241 }
242
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330243 bool sanitize (hb_sanitize_context_t *c) const
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100244 {
245 TRACE_SANITIZE (this);
246 return_trace (entries.sanitize (c));
247 }
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330248 bool sanitize (hb_sanitize_context_t *c, const void *base) const
Ebrahim Byagowib8b00fb2018-11-08 18:53:14 +0330249 {
250 TRACE_SANITIZE (this);
251 return_trace (entries.sanitize (c, base));
252 }
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100253
254 protected:
Behdad Esfahbod6b191782018-01-10 03:07:30 +0100255 HBUINT16 format; /* Format identifier--format = 6 */
Behdad Esfahbod3515c8b2018-10-07 22:27:00 -0400256 VarSizedBinSearchArrayOf<LookupSingle<T> >
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100257 entries; /* The actual entries, sorted by glyph index. */
258 public:
259 DEFINE_SIZE_ARRAY (8, entries);
260};
261
262template <typename T>
263struct LookupFormat8
264{
265 friend struct Lookup<T>;
266
267 private:
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330268 const T* get_value (hb_codepoint_t glyph_id) const
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100269 {
Behdad Esfahbod44af1f92018-10-14 14:52:17 -0700270 return firstGlyph <= glyph_id && glyph_id - firstGlyph < glyphCount ?
271 &valueArrayZ[glyph_id - firstGlyph] : nullptr;
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100272 }
273
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330274 bool sanitize (hb_sanitize_context_t *c) const
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100275 {
276 TRACE_SANITIZE (this);
277 return_trace (c->check_struct (this) && valueArrayZ.sanitize (c, glyphCount));
278 }
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330279 bool sanitize (hb_sanitize_context_t *c, const void *base) const
Ebrahim Byagowib8b00fb2018-11-08 18:53:14 +0330280 {
281 TRACE_SANITIZE (this);
282 return_trace (c->check_struct (this) && valueArrayZ.sanitize (c, glyphCount, base));
283 }
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100284
285 protected:
Behdad Esfahbod44af1f92018-10-14 14:52:17 -0700286 HBUINT16 format; /* Format identifier--format = 8 */
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100287 GlyphID firstGlyph; /* First glyph index included in the trimmed array. */
Behdad Esfahbod6b191782018-01-10 03:07:30 +0100288 HBUINT16 glyphCount; /* Total number of glyphs (equivalent to the last
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100289 * glyph minus the value of firstGlyph plus 1). */
290 UnsizedArrayOf<T>
291 valueArrayZ; /* The lookup values (indexed by the glyph index
292 * minus the value of firstGlyph). */
293 public:
294 DEFINE_SIZE_ARRAY (6, valueArrayZ);
295};
296
297template <typename T>
Behdad Esfahbodf7c0b432018-10-19 15:23:49 -0700298struct LookupFormat10
299{
300 friend struct Lookup<T>;
301
302 private:
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330303 const typename T::type get_value_or_null (hb_codepoint_t glyph_id) const
Behdad Esfahbodf7c0b432018-10-19 15:23:49 -0700304 {
305 if (!(firstGlyph <= glyph_id && glyph_id - firstGlyph < glyphCount))
306 return Null(T);
307
308 const HBUINT8 *p = &valueArrayZ[(glyph_id - firstGlyph) * valueSize];
309
310 unsigned int v = 0;
311 unsigned int count = valueSize;
312 for (unsigned int i = 0; i < count; i++)
313 v = (v << 8) | *p++;
314
315 return v;
316 }
317
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330318 bool sanitize (hb_sanitize_context_t *c) const
Behdad Esfahbodf7c0b432018-10-19 15:23:49 -0700319 {
320 TRACE_SANITIZE (this);
321 return_trace (c->check_struct (this) &&
322 valueSize <= 4 &&
323 valueArrayZ.sanitize (c, glyphCount * valueSize));
324 }
325
326 protected:
327 HBUINT16 format; /* Format identifier--format = 8 */
328 HBUINT16 valueSize; /* Byte size of each value. */
329 GlyphID firstGlyph; /* First glyph index included in the trimmed array. */
330 HBUINT16 glyphCount; /* Total number of glyphs (equivalent to the last
331 * glyph minus the value of firstGlyph plus 1). */
332 UnsizedArrayOf<HBUINT8>
333 valueArrayZ; /* The lookup values (indexed by the glyph index
334 * minus the value of firstGlyph). */
335 public:
Behdad Esfahbod00fdbca2018-10-20 12:04:51 -0700336 DEFINE_SIZE_ARRAY (8, valueArrayZ);
Behdad Esfahbodf7c0b432018-10-19 15:23:49 -0700337};
338
339template <typename T>
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100340struct Lookup
341{
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330342 const T* get_value (hb_codepoint_t glyph_id, unsigned int num_glyphs) const
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100343 {
344 switch (u.format) {
345 case 0: return u.format0.get_value (glyph_id, num_glyphs);
346 case 2: return u.format2.get_value (glyph_id);
347 case 4: return u.format4.get_value (glyph_id);
348 case 6: return u.format6.get_value (glyph_id);
349 case 8: return u.format8.get_value (glyph_id);
Behdad Esfahbod748b9892018-01-09 17:55:17 +0100350 default:return nullptr;
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100351 }
352 }
353
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330354 const typename T::type get_value_or_null (hb_codepoint_t glyph_id, unsigned int num_glyphs) const
Behdad Esfahbod7727e732018-10-10 13:24:51 -0400355 {
Behdad Esfahbodf7c0b432018-10-19 15:23:49 -0700356 switch (u.format) {
357 /* Format 10 cannot return a pointer. */
358 case 10: return u.format10.get_value_or_null (glyph_id);
359 default:
360 const T *v = get_value (glyph_id, num_glyphs);
361 return v ? *v : Null(T);
362 }
Behdad Esfahbod7727e732018-10-10 13:24:51 -0400363 }
364
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330365 typename T::type get_class (hb_codepoint_t glyph_id,
366 unsigned int num_glyphs,
367 unsigned int outOfRange) const
Behdad Esfahbod5b4a7892018-11-28 14:46:26 -0500368 {
369 const T *v = get_value (glyph_id, num_glyphs);
370 return v ? *v : outOfRange;
371 }
372
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330373 bool sanitize (hb_sanitize_context_t *c) const
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100374 {
375 TRACE_SANITIZE (this);
376 if (!u.format.sanitize (c)) return_trace (false);
377 switch (u.format) {
378 case 0: return_trace (u.format0.sanitize (c));
379 case 2: return_trace (u.format2.sanitize (c));
380 case 4: return_trace (u.format4.sanitize (c));
381 case 6: return_trace (u.format6.sanitize (c));
382 case 8: return_trace (u.format8.sanitize (c));
Behdad Esfahbodf7c0b432018-10-19 15:23:49 -0700383 case 10: return_trace (u.format10.sanitize (c));
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100384 default:return_trace (true);
385 }
386 }
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330387 bool sanitize (hb_sanitize_context_t *c, const void *base) const
Ebrahim Byagowib8b00fb2018-11-08 18:53:14 +0330388 {
389 TRACE_SANITIZE (this);
390 if (!u.format.sanitize (c)) return_trace (false);
391 switch (u.format) {
392 case 0: return_trace (u.format0.sanitize (c, base));
393 case 2: return_trace (u.format2.sanitize (c, base));
394 case 4: return_trace (u.format4.sanitize (c, base));
395 case 6: return_trace (u.format6.sanitize (c, base));
396 case 8: return_trace (u.format8.sanitize (c, base));
Behdad Esfahbodc99d13d2019-01-17 17:56:27 -0500397 case 10: return_trace (false); /* We don't support format10 here currently. */
Ebrahim Byagowib8b00fb2018-11-08 18:53:14 +0330398 default:return_trace (true);
399 }
400 }
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100401
402 protected:
403 union {
Behdad Esfahbod6b191782018-01-10 03:07:30 +0100404 HBUINT16 format; /* Format identifier */
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100405 LookupFormat0<T> format0;
406 LookupFormat2<T> format2;
407 LookupFormat4<T> format4;
408 LookupFormat6<T> format6;
409 LookupFormat8<T> format8;
Behdad Esfahbodf7c0b432018-10-19 15:23:49 -0700410 LookupFormat10<T> format10;
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100411 } u;
412 public:
Behdad Esfahbod9e8a9b82018-10-17 21:41:25 -0700413 DEFINE_SIZE_UNION (2, format);
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100414};
Behdad Esfahbod9e8a9b82018-10-17 21:41:25 -0700415/* Lookup 0 has unbounded size (dependant on num_glyphs). So we need to defined
416 * special NULL objects for Lookup<> objects, but since it's template our macros
417 * don't work. So we have to hand-code them here. UGLY. */
418} /* Close namespace. */
419/* Ugly hand-coded null objects for template Lookup<> :(. */
420extern HB_INTERNAL const unsigned char _hb_Null_AAT_Lookup[2];
421template <>
Ebrahim Byagowie4120082018-12-17 21:31:01 +0330422/*static*/ inline const AAT::Lookup<OT::HBUINT16>& Null<AAT::Lookup<OT::HBUINT16> > ()
423{ return *reinterpret_cast<const AAT::Lookup<OT::HBUINT16> *> (_hb_Null_AAT_Lookup); }
Behdad Esfahbod9e8a9b82018-10-17 21:41:25 -0700424template <>
Ebrahim Byagowie4120082018-12-17 21:31:01 +0330425/*static*/ inline const AAT::Lookup<OT::HBUINT32>& Null<AAT::Lookup<OT::HBUINT32> > ()
426{ return *reinterpret_cast<const AAT::Lookup<OT::HBUINT32> *> (_hb_Null_AAT_Lookup); }
Behdad Esfahbod9e8a9b82018-10-17 21:41:25 -0700427template <>
Ebrahim Byagowie4120082018-12-17 21:31:01 +0330428/*static*/ inline const AAT::Lookup<OT::Offset<OT::HBUINT16, false> >& Null<AAT::Lookup<OT::Offset<OT::HBUINT16, false> > > ()
429{ return *reinterpret_cast<const AAT::Lookup<OT::Offset<OT::HBUINT16, false> > *> (_hb_Null_AAT_Lookup); }
Behdad Esfahbod9e8a9b82018-10-17 21:41:25 -0700430namespace AAT {
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100431
Behdad Esfahbod102af612018-10-25 17:29:32 -0700432enum { DELETED_GLYPH = 0xFFFF };
Behdad Esfahbod470fe5b2018-01-09 15:48:51 +0100433
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100434/*
Behdad Esfahbodc0383c62018-11-06 15:07:19 -0500435 * (Extended) State Table
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100436 */
437
438template <typename T>
439struct Entry
440{
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330441 bool sanitize (hb_sanitize_context_t *c, unsigned int count) const
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100442 {
443 TRACE_SANITIZE (this);
Behdad Esfahbod17f01af2018-01-11 18:54:49 +0100444 /* Note, we don't recurse-sanitize data because we don't access it.
445 * That said, in our DEFINE_SIZE_STATIC we access T::static_size,
446 * which ensures that data has a simple sanitize(). To be determined
Behdad Esfahbod39bd07a2018-10-26 21:01:11 -0700447 * if I need to remove that as well.
448 *
Behdad Esfahbodba383782018-11-24 00:27:57 -0500449 * HOWEVER! Because we are a template, our DEFINE_SIZE_STATIC
450 * assertion wouldn't be checked, hence the line below. */
451 static_assert (T::static_size, "");
452
Behdad Esfahbod17f01af2018-01-11 18:54:49 +0100453 return_trace (c->check_struct (this));
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100454 }
455
456 public:
Behdad Esfahbod17f01af2018-01-11 18:54:49 +0100457 HBUINT16 newState; /* Byte offset from beginning of state table
458 * to the new state. Really?!?! Or just state
459 * number? The latter in morx for sure. */
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100460 HBUINT16 flags; /* Table specific. */
461 T data; /* Optional offsets to per-glyph tables. */
462 public:
463 DEFINE_SIZE_STATIC (4 + T::static_size);
464};
465
466template <>
467struct Entry<void>
468{
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330469 bool sanitize (hb_sanitize_context_t *c, unsigned int count /*XXX Unused?*/) const
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100470 {
471 TRACE_SANITIZE (this);
472 return_trace (c->check_struct (this));
473 }
474
475 public:
476 HBUINT16 newState; /* Byte offset from beginning of state table to the new state. */
477 HBUINT16 flags; /* Table specific. */
478 public:
479 DEFINE_SIZE_STATIC (4);
480};
481
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330482template <typename Types, typename Extra>
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100483struct StateTable
484{
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330485 typedef typename Types::HBUINT HBUINT;
486 typedef typename Types::HBUSHORT HBUSHORT;
Behdad Esfahbodf9a9c0f2018-11-28 14:51:56 -0500487 typedef typename Types::ClassTypeNarrow ClassType;
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330488
Behdad Esfahbod4653e6c2018-09-14 11:31:33 +0200489 enum State
490 {
491 STATE_START_OF_TEXT = 0,
492 STATE_START_OF_LINE = 1,
493 };
494 enum Class
495 {
496 CLASS_END_OF_TEXT = 0,
497 CLASS_OUT_OF_BOUNDS = 1,
498 CLASS_DELETED_GLYPH = 2,
499 CLASS_END_OF_LINE = 3,
500 };
501
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330502 int new_state (unsigned int newState) const
Behdad Esfahbod84efe042018-12-02 12:38:53 -0500503 { return Types::extended ? newState : ((int) newState - (int) stateArrayTable) / (int) nClasses; }
Behdad Esfahbod0cf282a2018-10-30 20:51:44 -0700504
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330505 unsigned int get_class (hb_codepoint_t glyph_id, unsigned int num_glyphs) const
Behdad Esfahbod1f1c85a2018-01-11 22:43:57 +0100506 {
Behdad Esfahbod102af612018-10-25 17:29:32 -0700507 if (unlikely (glyph_id == DELETED_GLYPH)) return CLASS_DELETED_GLYPH;
Behdad Esfahbodd5c0ca22018-11-07 12:08:44 -0500508 return (this+classTable).get_class (glyph_id, num_glyphs, 1);
Behdad Esfahbod1f1c85a2018-01-11 22:43:57 +0100509 }
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100510
Ebrahim Byagowie4120082018-12-17 21:31:01 +0330511 const Entry<Extra> *get_entries () const
Ebrahim Byagowi7ee5c522018-12-12 15:14:37 +0330512 { return (this+entryTable).arrayZ; }
Behdad Esfahbod17f01af2018-01-11 18:54:49 +0100513
Behdad Esfahbod299eca02019-01-24 17:17:00 +0100514 const Entry<Extra> &get_entry (int state, unsigned int klass) const
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100515 {
Behdad Esfahbod299eca02019-01-24 17:17:00 +0100516 if (unlikely (klass >= nClasses))
517 klass = StateTable<Types, Entry<Extra> >::CLASS_OUT_OF_BOUNDS;
Behdad Esfahbod680cbc22018-01-11 18:15:53 +0100518
Behdad Esfahbodc2527a12018-10-30 19:26:16 -0700519 const HBUSHORT *states = (this+stateArrayTable).arrayZ;
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100520 const Entry<Extra> *entries = (this+entryTable).arrayZ;
521
Behdad Esfahbod680cbc22018-01-11 18:15:53 +0100522 unsigned int entry = states[state * nClasses + klass];
Behdad Esfahbod006386b2018-11-07 18:04:53 -0500523 DEBUG_MSG (APPLY, nullptr, "e%u", entry);
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100524
Behdad Esfahbod299eca02019-01-24 17:17:00 +0100525 return entries[entry];
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100526 }
527
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330528 bool sanitize (hb_sanitize_context_t *c,
529 unsigned int *num_entries_out = nullptr) const
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100530 {
531 TRACE_SANITIZE (this);
Behdad Esfahbod12fffce2018-01-15 15:41:51 -0500532 if (unlikely (!(c->check_struct (this) &&
Behdad Esfahbodc4e36f92019-01-24 17:06:16 +0100533 nClasses >= 4 /* Ensure pre-defined classes fit. */ &&
Behdad Esfahbod12fffce2018-01-15 15:41:51 -0500534 classTable.sanitize (c, this)))) return_trace (false);
Behdad Esfahbod680cbc22018-01-11 18:15:53 +0100535
Behdad Esfahbodc2527a12018-10-30 19:26:16 -0700536 const HBUSHORT *states = (this+stateArrayTable).arrayZ;
Behdad Esfahbod680cbc22018-01-11 18:15:53 +0100537 const Entry<Extra> *entries = (this+entryTable).arrayZ;
538
Behdad Esfahbode9405302018-10-11 15:56:17 -0400539 unsigned int num_classes = nClasses;
Behdad Esfahbod006386b2018-11-07 18:04:53 -0500540 if (unlikely (hb_unsigned_mul_overflows (num_classes, states[0].static_size)))
541 return_trace (false);
542 unsigned int row_stride = num_classes * states[0].static_size;
Behdad Esfahbode9405302018-10-11 15:56:17 -0400543
Behdad Esfahbod006386b2018-11-07 18:04:53 -0500544 /* Apple 'kern' table has this peculiarity:
545 *
546 * "Because the stateTableOffset in the state table header is (strictly
547 * speaking) redundant, some 'kern' tables use it to record an initial
548 * state where that should not be StartOfText. To determine if this is
549 * done, calculate what the stateTableOffset should be. If it's different
550 * from the actual stateTableOffset, use it as the initial state."
551 *
552 * We implement this by calling the initial state zero, but allow *negative*
553 * states if the start state indeed was not the first state. Since the code
554 * is shared, this will also apply to 'mort' table. The 'kerx' / 'morx'
555 * tables are not affected since those address states by index, not offset.
556 */
557
558 int min_state = 0;
559 int max_state = 0;
Behdad Esfahbod680cbc22018-01-11 18:15:53 +0100560 unsigned int num_entries = 0;
561
Behdad Esfahbod006386b2018-11-07 18:04:53 -0500562 int state_pos = 0;
563 int state_neg = 0;
Behdad Esfahbod680cbc22018-01-11 18:15:53 +0100564 unsigned int entry = 0;
Behdad Esfahbod006386b2018-11-07 18:04:53 -0500565 while (min_state < state_neg || state_pos <= max_state)
Behdad Esfahbod680cbc22018-01-11 18:15:53 +0100566 {
Behdad Esfahbod006386b2018-11-07 18:04:53 -0500567 if (min_state < state_neg)
568 {
569 /* Negative states. */
570 if (unlikely (hb_unsigned_mul_overflows (min_state, num_classes)))
571 return_trace (false);
Behdad Esfahbode0144052018-11-12 14:23:31 -0500572 if (unlikely (!c->check_range (&states[min_state * num_classes],
573 -min_state,
574 row_stride)))
Behdad Esfahbod006386b2018-11-07 18:04:53 -0500575 return_trace (false);
Behdad Esfahbodc4623db2019-01-24 17:10:12 +0100576 if ((c->max_ops -= state_neg - min_state) <= 0)
Behdad Esfahbod006386b2018-11-07 18:04:53 -0500577 return_trace (false);
578 { /* Sweep new states. */
579 const HBUSHORT *stop = &states[min_state * num_classes];
580 if (unlikely (stop > states))
581 return_trace (false);
582 for (const HBUSHORT *p = states; stop < p; p--)
583 num_entries = MAX<unsigned int> (num_entries, *(p - 1) + 1);
584 state_neg = min_state;
585 }
586 }
Behdad Esfahbode9405302018-10-11 15:56:17 -0400587
Behdad Esfahbod006386b2018-11-07 18:04:53 -0500588 if (state_pos <= max_state)
589 {
590 /* Positive states. */
Behdad Esfahbode0144052018-11-12 14:23:31 -0500591 if (unlikely (!c->check_range (states,
592 max_state + 1,
593 row_stride)))
Behdad Esfahbod006386b2018-11-07 18:04:53 -0500594 return_trace (false);
Behdad Esfahbodc4623db2019-01-24 17:10:12 +0100595 if ((c->max_ops -= max_state - state_pos + 1) <= 0)
Behdad Esfahbod006386b2018-11-07 18:04:53 -0500596 return_trace (false);
597 { /* Sweep new states. */
598 if (unlikely (hb_unsigned_mul_overflows ((max_state + 1), num_classes)))
599 return_trace (false);
600 const HBUSHORT *stop = &states[(max_state + 1) * num_classes];
601 if (unlikely (stop < states))
602 return_trace (false);
603 for (const HBUSHORT *p = &states[state_pos * num_classes]; p < stop; p++)
604 num_entries = MAX<unsigned int> (num_entries, *p + 1);
605 state_pos = max_state + 1;
606 }
Behdad Esfahbod680cbc22018-01-11 18:15:53 +0100607 }
608
Behdad Esfahbod9507b052018-09-10 23:18:07 +0200609 if (unlikely (!c->check_array (entries, num_entries)))
Behdad Esfahbod680cbc22018-01-11 18:15:53 +0100610 return_trace (false);
Behdad Esfahbodc4623db2019-01-24 17:10:12 +0100611 if ((c->max_ops -= num_entries - entry) <= 0)
Behdad Esfahbod83780302018-10-17 22:34:16 -0700612 return_trace (false);
Behdad Esfahbod680cbc22018-01-11 18:15:53 +0100613 { /* Sweep new entries. */
614 const Entry<Extra> *stop = &entries[num_entries];
615 for (const Entry<Extra> *p = &entries[entry]; p < stop; p++)
Behdad Esfahbod0cf282a2018-10-30 20:51:44 -0700616 {
Behdad Esfahbod006386b2018-11-07 18:04:53 -0500617 int newState = new_state (p->newState);
618 min_state = MIN (min_state, newState);
619 max_state = MAX (max_state, newState);
Behdad Esfahbod0cf282a2018-10-30 20:51:44 -0700620 }
Behdad Esfahbod680cbc22018-01-11 18:15:53 +0100621 entry = num_entries;
622 }
623 }
624
Behdad Esfahbod17f01af2018-01-11 18:54:49 +0100625 if (num_entries_out)
626 *num_entries_out = num_entries;
627
Behdad Esfahbod680cbc22018-01-11 18:15:53 +0100628 return_trace (true);
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100629 }
630
631 protected:
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330632 HBUINT nClasses; /* Number of classes, which is the number of indices
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100633 * in a single line in the state array. */
Behdad Esfahbod7e6bd512019-01-17 18:24:18 -0500634 NNOffsetTo<ClassType, HBUINT>
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100635 classTable; /* Offset to the class table. */
Behdad Esfahbod7e6bd512019-01-17 18:24:18 -0500636 NNOffsetTo<UnsizedArrayOf<HBUSHORT>, HBUINT>
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100637 stateArrayTable;/* Offset to the state array. */
Behdad Esfahbod7e6bd512019-01-17 18:24:18 -0500638 NNOffsetTo<UnsizedArrayOf<Entry<Extra> >, HBUINT>
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100639 entryTable; /* Offset to the entry array. */
640
641 public:
Behdad Esfahbod11dbf0f2018-10-30 21:49:59 -0700642 DEFINE_SIZE_STATIC (4 * sizeof (HBUINT));
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100643};
644
Behdad Esfahbod5b4a7892018-11-28 14:46:26 -0500645template <typename HBUCHAR>
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330646struct ClassTable
647{
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330648 unsigned int get_class (hb_codepoint_t glyph_id, unsigned int outOfRange) const
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330649 {
Behdad Esfahbodb11830c2018-11-06 15:23:18 -0500650 unsigned int i = glyph_id - firstGlyph;
651 return i >= classArray.len ? outOfRange : classArray.arrayZ[i];
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330652 }
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330653 unsigned int get_class (hb_codepoint_t glyph_id,
654 unsigned int num_glyphs HB_UNUSED,
655 unsigned int outOfRange) const
Behdad Esfahbod5b4a7892018-11-28 14:46:26 -0500656 {
657 return get_class (glyph_id, outOfRange);
658 }
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330659 bool sanitize (hb_sanitize_context_t *c) const
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330660 {
661 TRACE_SANITIZE (this);
Behdad Esfahbodb11830c2018-11-06 15:23:18 -0500662 return_trace (c->check_struct (this) && classArray.sanitize (c));
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330663 }
664 protected:
Behdad Esfahbodb11830c2018-11-06 15:23:18 -0500665 GlyphID firstGlyph; /* First glyph index included in the trimmed array. */
Behdad Esfahbod5b4a7892018-11-28 14:46:26 -0500666 ArrayOf<HBUCHAR> classArray; /* The class codes (indexed by glyph index minus
Behdad Esfahbodb11830c2018-11-06 15:23:18 -0500667 * firstGlyph). */
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330668 public:
Behdad Esfahbodb11830c2018-11-06 15:23:18 -0500669 DEFINE_SIZE_ARRAY (4, classArray);
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330670};
671
Behdad Esfahbod241ba7d2018-11-07 11:51:40 -0500672struct ObsoleteTypes
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330673{
Behdad Esfahbod39e1b6d2019-01-22 12:07:43 +0100674 static constexpr bool extended = false;
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330675 typedef HBUINT16 HBUINT;
676 typedef HBUINT8 HBUSHORT;
Behdad Esfahbodf9a9c0f2018-11-28 14:51:56 -0500677 typedef ClassTable<HBUINT8> ClassTypeNarrow;
678 typedef ClassTable<HBUINT16> ClassTypeWide;
Behdad Esfahbod5b4a7892018-11-28 14:46:26 -0500679
Behdad Esfahbod36e90ef2018-10-31 15:09:09 -0700680 template <typename T>
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330681 static unsigned int offsetToIndex (unsigned int offset,
682 const void *base,
683 const T *array)
Behdad Esfahbod36e90ef2018-10-31 15:09:09 -0700684 {
685 return (offset - ((const char *) array - (const char *) base)) / sizeof (T);
686 }
687 template <typename T>
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330688 static unsigned int byteOffsetToIndex (unsigned int offset,
689 const void *base,
690 const T *array)
Behdad Esfahbod42a2b492018-11-28 15:24:30 -0500691 {
692 return offsetToIndex (offset, base, array);
693 }
694 template <typename T>
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330695 static unsigned int wordOffsetToIndex (unsigned int offset,
696 const void *base,
697 const T *array)
Behdad Esfahbod36e90ef2018-10-31 15:09:09 -0700698 {
699 return offsetToIndex (2 * offset, base, array);
700 }
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330701};
Behdad Esfahbod241ba7d2018-11-07 11:51:40 -0500702struct ExtendedTypes
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330703{
Behdad Esfahbod39e1b6d2019-01-22 12:07:43 +0100704 static constexpr bool extended = true;
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330705 typedef HBUINT32 HBUINT;
706 typedef HBUINT16 HBUSHORT;
Behdad Esfahbodf9a9c0f2018-11-28 14:51:56 -0500707 typedef Lookup<HBUINT16> ClassTypeNarrow;
708 typedef Lookup<HBUINT16> ClassTypeWide;
Behdad Esfahbod5b4a7892018-11-28 14:46:26 -0500709
Behdad Esfahbod36e90ef2018-10-31 15:09:09 -0700710 template <typename T>
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330711 static unsigned int offsetToIndex (unsigned int offset,
Behdad Esfahbodd25a2f12018-12-23 20:19:52 -0500712 const void *base HB_UNUSED,
713 const T *array HB_UNUSED)
Behdad Esfahbod36e90ef2018-10-31 15:09:09 -0700714 {
715 return offset;
716 }
717 template <typename T>
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330718 static unsigned int byteOffsetToIndex (unsigned int offset,
Behdad Esfahbodd25a2f12018-12-23 20:19:52 -0500719 const void *base HB_UNUSED,
720 const T *array HB_UNUSED)
Behdad Esfahbod42a2b492018-11-28 15:24:30 -0500721 {
722 return offset / 2;
723 }
724 template <typename T>
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330725 static unsigned int wordOffsetToIndex (unsigned int offset,
Behdad Esfahbodd25a2f12018-12-23 20:19:52 -0500726 const void *base HB_UNUSED,
727 const T *array HB_UNUSED)
Behdad Esfahbod36e90ef2018-10-31 15:09:09 -0700728 {
729 return offset;
730 }
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330731};
732
733template <typename Types, typename EntryData>
Behdad Esfahbod117cfe72018-01-12 00:01:36 +0100734struct StateTableDriver
735{
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330736 StateTableDriver (const StateTable<Types, EntryData> &machine_,
737 hb_buffer_t *buffer_,
738 hb_face_t *face_) :
Behdad Esfahbod117cfe72018-01-12 00:01:36 +0100739 machine (machine_),
740 buffer (buffer_),
Behdad Esfahbod54e6efa2018-02-04 14:58:02 -0500741 num_glyphs (face_->get_num_glyphs ()) {}
Behdad Esfahbod117cfe72018-01-12 00:01:36 +0100742
743 template <typename context_t>
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330744 void drive (context_t *c)
Behdad Esfahbod117cfe72018-01-12 00:01:36 +0100745 {
Behdad Esfahbode6f283e2018-01-19 18:08:56 -0800746 if (!c->in_place)
747 buffer->clear_output ();
748
Behdad Esfahbod006386b2018-11-07 18:04:53 -0500749 int state = StateTable<Types, EntryData>::STATE_START_OF_TEXT;
Behdad Esfahbod1bb8ed82018-10-25 17:33:48 -0700750 for (buffer->idx = 0; buffer->successful;)
Behdad Esfahbod117cfe72018-01-12 00:01:36 +0100751 {
Behdad Esfahbode6f283e2018-01-19 18:08:56 -0800752 unsigned int klass = buffer->idx < buffer->len ?
Behdad Esfahbod4831e612018-10-05 18:14:13 +0200753 machine.get_class (buffer->info[buffer->idx].codepoint, num_glyphs) :
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330754 (unsigned) StateTable<Types, EntryData>::CLASS_END_OF_TEXT;
Behdad Esfahbod006386b2018-11-07 18:04:53 -0500755 DEBUG_MSG (APPLY, nullptr, "c%u at %u", klass, buffer->idx);
Behdad Esfahbod1ec90512019-01-24 17:21:41 +0100756 const Entry<EntryData> &entry = machine.get_entry (state, klass);
Behdad Esfahbod117cfe72018-01-12 00:01:36 +0100757
Behdad Esfahbod54e6efa2018-02-04 14:58:02 -0500758 /* Unsafe-to-break before this if not in state 0, as things might
Behdad Esfahbod7ee50af2018-10-06 21:31:44 +0200759 * go differently if we start from state 0 here.
760 *
761 * Ugh. The indexing here is ugly... */
762 if (state && buffer->backtrack_len () && buffer->idx < buffer->len)
Behdad Esfahbod54e6efa2018-02-04 14:58:02 -0500763 {
Behdad Esfahbod2971e9d2018-02-06 11:48:04 -0500764 /* If there's no action and we're just epsilon-transitioning to state 0,
765 * safe to break. */
766 if (c->is_actionable (this, entry) ||
Behdad Esfahbod1ec90512019-01-24 17:21:41 +0100767 !(entry.newState == StateTable<Types, EntryData>::STATE_START_OF_TEXT &&
768 entry.flags == context_t::DontAdvance))
Behdad Esfahbod7ee50af2018-10-06 21:31:44 +0200769 buffer->unsafe_to_break_from_outbuffer (buffer->backtrack_len () - 1, buffer->idx + 1);
Behdad Esfahbod54e6efa2018-02-04 14:58:02 -0500770 }
771
772 /* Unsafe-to-break if end-of-text would kick in here. */
773 if (buffer->idx + 2 <= buffer->len)
774 {
Behdad Esfahbod1ec90512019-01-24 17:21:41 +0100775 const Entry<EntryData> &end_entry = machine.get_entry (state, StateTable<Types, EntryData>::CLASS_END_OF_TEXT);
Behdad Esfahbod54e6efa2018-02-04 14:58:02 -0500776 if (c->is_actionable (this, end_entry))
777 buffer->unsafe_to_break (buffer->idx, buffer->idx + 2);
778 }
779
Behdad Esfahbodb9769402019-01-24 18:01:07 +0100780 c->transition (this, entry);
Behdad Esfahbodf7600222018-01-12 11:09:21 +0100781
Behdad Esfahbod1ec90512019-01-24 17:21:41 +0100782 state = machine.new_state (entry.newState);
Behdad Esfahbod19090722018-11-07 16:42:16 -0500783 DEBUG_MSG (APPLY, nullptr, "s%d", state);
Behdad Esfahbode6f283e2018-01-19 18:08:56 -0800784
785 if (buffer->idx == buffer->len)
Behdad Esfahbod0cf282a2018-10-30 20:51:44 -0700786 break;
Behdad Esfahbode6f283e2018-01-19 18:08:56 -0800787
Behdad Esfahbod1ec90512019-01-24 17:21:41 +0100788 if (!(entry.flags & context_t::DontAdvance) || buffer->max_ops-- <= 0)
Behdad Esfahbod0cf282a2018-10-30 20:51:44 -0700789 buffer->next_glyph ();
Behdad Esfahbod117cfe72018-01-12 00:01:36 +0100790 }
791
Behdad Esfahbod57051b42018-01-12 11:42:25 +0100792 if (!c->in_place)
793 {
Behdad Esfahbod7efa3822018-10-14 19:30:44 -0700794 for (; buffer->successful && buffer->idx < buffer->len;)
795 buffer->next_glyph ();
Behdad Esfahbod17335a82018-11-04 02:25:07 -0500796 buffer->swap_buffers ();
Behdad Esfahbod57051b42018-01-12 11:42:25 +0100797 }
Behdad Esfahbod117cfe72018-01-12 00:01:36 +0100798 }
799
800 public:
Ebrahim Byagowib053cab2018-10-30 18:41:34 +0330801 const StateTable<Types, EntryData> &machine;
Behdad Esfahbod117cfe72018-01-12 00:01:36 +0100802 hb_buffer_t *buffer;
Behdad Esfahbod117cfe72018-01-12 00:01:36 +0100803 unsigned int num_glyphs;
Behdad Esfahbod117cfe72018-01-12 00:01:36 +0100804};
805
Behdad Esfahbodca42d962018-01-11 09:15:34 +0100806
Behdad Esfahbod7bb4da72018-10-11 00:52:07 -0400807struct ankr;
Behdad Esfahbod046690a2018-01-17 16:59:55 -0800808
809struct hb_aat_apply_context_t :
810 hb_dispatch_context_t<hb_aat_apply_context_t, bool, HB_DEBUG_APPLY>
811{
Ebrahim Byagowie4120082018-12-17 21:31:01 +0330812 const char *get_name () { return "APPLY"; }
Behdad Esfahbod046690a2018-01-17 16:59:55 -0800813 template <typename T>
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330814 return_t dispatch (const T &obj) { return obj.apply (this); }
Ebrahim Byagowie4120082018-12-17 21:31:01 +0330815 static return_t default_return_value () { return false; }
Behdad Esfahbod046690a2018-01-17 16:59:55 -0800816 bool stop_sublookup_iteration (return_t r) const { return r; }
817
Behdad Esfahbodc221dc02018-11-14 14:49:34 -0500818 const hb_ot_shape_plan_t *plan;
Behdad Esfahbod046690a2018-01-17 16:59:55 -0800819 hb_font_t *font;
820 hb_face_t *face;
821 hb_buffer_t *buffer;
Behdad Esfahbodf07ce662018-01-19 16:52:01 -0800822 hb_sanitize_context_t sanitizer;
Behdad Esfahbodb605db22018-11-04 12:58:02 -0500823 const ankr *ankr_table;
Behdad Esfahbod046690a2018-01-17 16:59:55 -0800824
Behdad Esfahbod126ffdb2018-02-07 12:26:41 -0500825 /* Unused. For debug tracing only. */
826 unsigned int lookup_index;
827 unsigned int debug_depth;
828
Behdad Esfahbodc221dc02018-11-14 14:49:34 -0500829 HB_INTERNAL hb_aat_apply_context_t (const hb_ot_shape_plan_t *plan_,
Behdad Esfahbodb605db22018-11-04 12:58:02 -0500830 hb_font_t *font_,
831 hb_buffer_t *buffer_,
832 hb_blob_t *blob = const_cast<hb_blob_t *> (&Null(hb_blob_t)));
833
Ebrahim Byagowie4120082018-12-17 21:31:01 +0330834 HB_INTERNAL ~hb_aat_apply_context_t ();
Behdad Esfahbodb605db22018-11-04 12:58:02 -0500835
Behdad Esfahbod347ad452019-01-17 17:47:29 -0500836 HB_INTERNAL void set_ankr_table (const AAT::ankr *ankr_table_);
Behdad Esfahbodf07ce662018-01-19 16:52:01 -0800837
Ebrahim Byagowib2ebaa92018-12-16 22:38:10 +0330838 void set_lookup_index (unsigned int i) { lookup_index = i; }
Behdad Esfahbod046690a2018-01-17 16:59:55 -0800839};
840
841
Behdad Esfahboda0175e72017-08-17 16:55:54 -0700842} /* namespace AAT */
843
844
Behdad Esfahbodc77ae402018-08-25 22:36:36 -0700845#endif /* HB_AAT_LAYOUT_COMMON_HH */