blob: 2554f42e128d78390697bbcd25ab2a4e5bbff597 [file] [log] [blame]
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -04001/*
Behdad Esfahbod2409d5f2011-04-21 17:14:28 -04002 * Copyright © 2007,2008,2009,2010 Red Hat, Inc.
Behdad Esfahbod5b93e8d2012-04-23 22:26:13 -04003 * Copyright © 2010,2012 Google, Inc.
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -04004 *
Behdad Esfahbodc755cb32010-04-22 00:11:43 -04005 * This is part of HarfBuzz, a text shaping library.
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -04006 *
7 * Permission is hereby granted, without written agreement and without
8 * license or royalty fees, to use, copy, modify, and distribute this
9 * software and its documentation for any purpose, provided that the
10 * above copyright notice and the following two paragraphs appear in
11 * all copies of this software.
12 *
13 * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
14 * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
15 * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
16 * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
17 * DAMAGE.
18 *
19 * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
20 * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
21 * FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS
22 * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
23 * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
24 *
25 * Red Hat Author(s): Behdad Esfahbod
Behdad Esfahbod98370e82010-10-27 17:39:01 -040026 * Google Author(s): Behdad Esfahbod
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040027 */
28
Behdad Esfahbod5f5b24f2009-08-02 20:03:12 -040029#ifndef HB_OT_LAYOUT_GSUBGPOS_PRIVATE_HH
30#define HB_OT_LAYOUT_GSUBGPOS_PRIVATE_HH
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040031
Behdad Esfahbod22da7fd2010-05-12 18:23:21 -040032#include "hb-buffer-private.hh"
Behdad Esfahbod7a750ac2011-08-17 14:19:59 +020033#include "hb-ot-layout-gdef-table.hh"
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -040034
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040035
Behdad Esfahbodacdba3f2010-07-23 15:11:18 -040036
Behdad Esfahboda9844d42012-05-09 17:53:13 +020037/* unique ligature id */
38/* component number in the ligature (0 = base) */
39static inline void
40set_lig_props (hb_glyph_info_t &info, unsigned int lig_id, unsigned int lig_comp)
41{
42 info.lig_props() = (lig_id << 4) | (lig_comp & 0x0F);
43}
44static inline unsigned int
45get_lig_id (hb_glyph_info_t &info)
46{
47 return info.lig_props() >> 4;
48}
49static inline unsigned int
50get_lig_comp (hb_glyph_info_t &info)
51{
52 return info.lig_props() & 0x0F;
53}
Behdad Esfahbodc2709112010-10-27 23:18:51 -040054
Behdad Esfahbod18c42852011-07-25 00:36:58 -040055static inline uint8_t allocate_lig_id (hb_buffer_t *buffer) {
Behdad Esfahboda9844d42012-05-09 17:53:13 +020056 uint8_t lig_id = buffer->next_serial () & 0x0F;
57 if (unlikely (!lig_id))
58 lig_id = allocate_lig_id (buffer); /* in case of overflow */
Behdad Esfahbodd8787492011-07-25 00:36:01 -040059 return lig_id;
60}
61
62
Behdad Esfahbodc2709112010-10-27 23:18:51 -040063
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -040064#ifndef HB_DEBUG_CLOSURE
65#define HB_DEBUG_CLOSURE (HB_DEBUG+0)
66#endif
67
68#define TRACE_CLOSURE() \
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +020069 hb_auto_trace_t<HB_DEBUG_CLOSURE> trace (&c->debug_depth, "CLOSURE", this, HB_FUNC, "");
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -040070
71
Behdad Esfahbod98619ce2012-05-11 02:34:06 +020072/* TODO Add TRACE_RETURN annotation for would_apply */
73
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -040074
75struct hb_closure_context_t
76{
77 hb_face_t *face;
Behdad Esfahbod6a9be5b2012-04-23 22:23:17 -040078 hb_set_t *glyphs;
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -040079 unsigned int nesting_level_left;
80 unsigned int debug_depth;
81
82
83 hb_closure_context_t (hb_face_t *face_,
Behdad Esfahbod6a9be5b2012-04-23 22:23:17 -040084 hb_set_t *glyphs_,
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -040085 unsigned int nesting_level_left_ = MAX_NESTING_LEVEL) :
86 face (face_), glyphs (glyphs_),
87 nesting_level_left (nesting_level_left_),
88 debug_depth (0) {}
89};
90
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -040091
92
Behdad Esfahbod0535b502009-08-28 17:14:33 -040093#ifndef HB_DEBUG_APPLY
Behdad Esfahbod11e3ec42010-11-03 15:11:04 -040094#define HB_DEBUG_APPLY (HB_DEBUG+0)
Behdad Esfahbod0535b502009-08-28 17:14:33 -040095#endif
96
Behdad Esfahbodbc200452010-04-29 01:40:26 -040097#define TRACE_APPLY() \
Behdad Esfahbod68b76122012-06-08 10:43:49 -040098 hb_auto_trace_t<HB_DEBUG_APPLY> trace (&c->debug_depth, "APPLY", this, HB_FUNC, "idx %d codepoint %u", c->buffer->idx, c->buffer->cur().codepoint);
Behdad Esfahbod74e313c2010-04-28 15:15:09 -040099
Behdad Esfahbod0535b502009-08-28 17:14:33 -0400100
Behdad Esfahbodacdba3f2010-07-23 15:11:18 -0400101
Behdad Esfahbod1376fb72010-04-29 02:19:21 -0400102struct hb_apply_context_t
103{
Behdad Esfahbodabcfe9b2011-05-11 00:02:02 -0400104 hb_font_t *font;
105 hb_face_t *face;
Behdad Esfahbod94a23aa2010-05-05 01:13:09 -0400106 hb_buffer_t *buffer;
Behdad Esfahbod744970a2011-05-16 18:15:37 -0400107 hb_direction_t direction;
Behdad Esfahbodf7acd8d2010-05-20 17:26:35 +0100108 hb_mask_t lookup_mask;
Behdad Esfahbod1376fb72010-04-29 02:19:21 -0400109 unsigned int nesting_level_left;
Behdad Esfahbod8c69e652010-10-27 22:07:49 -0400110 unsigned int lookup_props;
Behdad Esfahbod98370e82010-10-27 17:39:01 -0400111 unsigned int property; /* propety of first glyph */
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -0400112 unsigned int debug_depth;
Behdad Esfahbod98370e82010-10-27 17:39:01 -0400113
Behdad Esfahbod41ae6742012-04-11 17:11:05 -0400114
115 hb_apply_context_t (hb_font_t *font_,
116 hb_face_t *face_,
117 hb_buffer_t *buffer_,
Behdad Esfahbod6736f3c2012-05-13 15:21:06 +0200118 hb_mask_t lookup_mask_) :
Behdad Esfahbod41ae6742012-04-11 17:11:05 -0400119 font (font_), face (face_), buffer (buffer_),
120 direction (buffer_->props.direction),
121 lookup_mask (lookup_mask_),
Behdad Esfahbod6736f3c2012-05-13 15:21:06 +0200122 nesting_level_left (MAX_NESTING_LEVEL),
Behdad Esfahbod650ac002012-04-23 13:17:09 -0400123 lookup_props (0), property (0), debug_depth (0) {}
Behdad Esfahbod41ae6742012-04-11 17:11:05 -0400124
Behdad Esfahbod650ac002012-04-23 13:17:09 -0400125 void set_lookup (const Lookup &l) {
Behdad Esfahbod41ae6742012-04-11 17:11:05 -0400126 lookup_props = l.get_props ();
127 }
128
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500129 struct mark_skipping_forward_iterator_t
130 {
131 inline mark_skipping_forward_iterator_t (hb_apply_context_t *c_,
132 unsigned int start_index_,
Behdad Esfahbod203d7102012-05-11 16:01:44 +0200133 unsigned int num_items_,
Behdad Esfahbod28b9d502012-05-13 15:04:00 +0200134 bool context_match = false)
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500135 {
136 c = c_;
137 idx = start_index_;
138 num_items = num_items_;
Behdad Esfahbod28b9d502012-05-13 15:04:00 +0200139 mask = context_match ? -1 : c->lookup_mask;
Behdad Esfahbod99c26952012-05-13 15:45:18 +0200140 syllable = context_match ? 0 : c->buffer->cur().syllable ();
Behdad Esfahbod5df809b2012-05-13 15:17:51 +0200141 end = c->buffer->len;
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500142 }
143 inline bool has_no_chance (void) const
144 {
145 return unlikely (num_items && idx + num_items >= end);
146 }
147 inline bool next (unsigned int *property_out,
148 unsigned int lookup_props)
149 {
Behdad Esfahbod506ffeb2012-01-18 16:07:53 -0500150 assert (num_items > 0);
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500151 do
152 {
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500153 if (has_no_chance ())
154 return false;
Behdad Esfahboda4a48fe2012-01-17 18:08:41 -0500155 idx++;
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500156 } while (_hb_ot_layout_skip_mark (c->face, &c->buffer->info[idx], lookup_props, property_out));
157 num_items--;
Behdad Esfahbod5f131d32012-05-11 17:29:40 +0200158 return (c->buffer->info[idx].mask & mask) && (!syllable || syllable == c->buffer->info[idx].syllable ());
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500159 }
160 inline bool next (unsigned int *property_out = NULL)
161 {
162 return next (property_out, c->lookup_props);
163 }
164
165 unsigned int idx;
166 private:
167 hb_apply_context_t *c;
168 unsigned int num_items;
Behdad Esfahbod203d7102012-05-11 16:01:44 +0200169 hb_mask_t mask;
Behdad Esfahbod5f131d32012-05-11 17:29:40 +0200170 uint8_t syllable;
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500171 unsigned int end;
172 };
173
174 struct mark_skipping_backward_iterator_t
175 {
176 inline mark_skipping_backward_iterator_t (hb_apply_context_t *c_,
177 unsigned int start_index_,
Behdad Esfahbod203d7102012-05-11 16:01:44 +0200178 unsigned int num_items_,
Behdad Esfahbod5f131d32012-05-11 17:29:40 +0200179 hb_mask_t mask_ = 0,
Behdad Esfahbod6736f3c2012-05-13 15:21:06 +0200180 bool match_syllable_ = true)
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500181 {
182 c = c_;
183 idx = start_index_;
184 num_items = num_items_;
Behdad Esfahbod203d7102012-05-11 16:01:44 +0200185 mask = mask_ ? mask_ : c->lookup_mask;
Behdad Esfahbod99c26952012-05-13 15:45:18 +0200186 syllable = match_syllable_ ? c->buffer->cur().syllable () : 0;
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500187 }
188 inline bool has_no_chance (void) const
189 {
Behdad Esfahbod506ffeb2012-01-18 16:07:53 -0500190 return unlikely (idx < num_items);
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500191 }
192 inline bool prev (unsigned int *property_out,
193 unsigned int lookup_props)
194 {
Behdad Esfahbod506ffeb2012-01-18 16:07:53 -0500195 assert (num_items > 0);
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500196 do
197 {
198 if (has_no_chance ())
199 return false;
200 idx--;
201 } while (_hb_ot_layout_skip_mark (c->face, &c->buffer->out_info[idx], lookup_props, property_out));
202 num_items--;
Behdad Esfahbod5f131d32012-05-11 17:29:40 +0200203 return (c->buffer->out_info[idx].mask & mask) && (!syllable || syllable == c->buffer->out_info[idx].syllable ());
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500204 }
205 inline bool prev (unsigned int *property_out = NULL)
206 {
207 return prev (property_out, c->lookup_props);
208 }
209
210 unsigned int idx;
211 private:
212 hb_apply_context_t *c;
213 unsigned int num_items;
Behdad Esfahbod203d7102012-05-11 16:01:44 +0200214 hb_mask_t mask;
Behdad Esfahbod5f131d32012-05-11 17:29:40 +0200215 uint8_t syllable;
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500216 };
217
Behdad Esfahbod7d479902012-01-18 21:19:32 -0500218 inline bool should_mark_skip_current_glyph (void) const
219 {
Behdad Esfahbod99c26952012-05-13 15:45:18 +0200220 return _hb_ot_layout_skip_mark (face, &buffer->cur(), lookup_props, NULL);
Behdad Esfahbod7d479902012-01-18 21:19:32 -0500221 }
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500222
223
224
Behdad Esfahboded2f1362012-05-22 22:12:22 -0400225 inline void replace_glyph (hb_codepoint_t glyph_index,
226 unsigned int klass = 0) const
Behdad Esfahbod98370e82010-10-27 17:39:01 -0400227 {
Behdad Esfahboded2f1362012-05-22 22:12:22 -0400228 buffer->cur().props_cache() = klass; /*XXX if has gdef? */
Behdad Esfahbod98370e82010-10-27 17:39:01 -0400229 buffer->replace_glyph (glyph_index);
230 }
231 inline void replace_glyphs_be16 (unsigned int num_in,
232 unsigned int num_out,
Behdad Esfahbod2a3d9112012-06-07 17:31:46 -0400233 const char *glyph_data_be,
Behdad Esfahboded2f1362012-05-22 22:12:22 -0400234 unsigned int klass = 0) const
Behdad Esfahbod98370e82010-10-27 17:39:01 -0400235 {
Behdad Esfahboded2f1362012-05-22 22:12:22 -0400236 buffer->cur().props_cache() = klass; /* XXX if has gdef? */
Behdad Esfahbod98370e82010-10-27 17:39:01 -0400237 buffer->replace_glyphs_be16 (num_in, num_out, glyph_data_be);
238 }
Behdad Esfahbod1376fb72010-04-29 02:19:21 -0400239};
240
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400241
Behdad Esfahbod94a23aa2010-05-05 01:13:09 -0400242
Behdad Esfahbod6a9be5b2012-04-23 22:23:17 -0400243typedef bool (*intersects_func_t) (hb_set_t *glyphs, const USHORT &value, const void *data);
Behdad Esfahbod40cbefe2010-05-10 17:47:22 -0400244typedef bool (*match_func_t) (hb_codepoint_t glyph_id, const USHORT &value, const void *data);
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400245typedef void (*closure_lookup_func_t) (hb_closure_context_t *c, unsigned int lookup_index);
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400246typedef bool (*apply_lookup_func_t) (hb_apply_context_t *c, unsigned int lookup_index);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400247
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400248struct ContextClosureFuncs
249{
250 intersects_func_t intersects;
251 closure_lookup_func_t closure;
252};
253struct ContextApplyFuncs
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400254{
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400255 match_func_t match;
256 apply_lookup_func_t apply;
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400257};
258
Behdad Esfahbod6a9be5b2012-04-23 22:23:17 -0400259static inline bool intersects_glyph (hb_set_t *glyphs, const USHORT &value, const void *data HB_UNUSED)
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400260{
261 return glyphs->has (value);
262}
Behdad Esfahbod6a9be5b2012-04-23 22:23:17 -0400263static inline bool intersects_class (hb_set_t *glyphs, const USHORT &value, const void *data)
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400264{
265 const ClassDef &class_def = *reinterpret_cast<const ClassDef *>(data);
266 return class_def.intersects_class (glyphs, value);
267}
Behdad Esfahbod6a9be5b2012-04-23 22:23:17 -0400268static inline bool intersects_coverage (hb_set_t *glyphs, const USHORT &value, const void *data)
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400269{
270 const OffsetTo<Coverage> &coverage = (const OffsetTo<Coverage>&)value;
271 return (data+coverage).intersects (glyphs);
272}
273
274static inline bool intersects_array (hb_closure_context_t *c,
275 unsigned int count,
276 const USHORT values[],
277 intersects_func_t intersects_func,
278 const void *intersects_data)
279{
280 for (unsigned int i = 0; i < count; i++)
281 if (likely (!intersects_func (c->glyphs, values[i], intersects_data)))
282 return false;
283 return true;
284}
285
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400286
Behdad Esfahbod40cbefe2010-05-10 17:47:22 -0400287static inline bool match_glyph (hb_codepoint_t glyph_id, const USHORT &value, const void *data HB_UNUSED)
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400288{
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400289 return glyph_id == value;
290}
Behdad Esfahbod40cbefe2010-05-10 17:47:22 -0400291static inline bool match_class (hb_codepoint_t glyph_id, const USHORT &value, const void *data)
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400292{
Behdad Esfahbod2b5a59c2009-08-04 11:38:50 -0400293 const ClassDef &class_def = *reinterpret_cast<const ClassDef *>(data);
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400294 return class_def.get_class (glyph_id) == value;
295}
Behdad Esfahbod40cbefe2010-05-10 17:47:22 -0400296static inline bool match_coverage (hb_codepoint_t glyph_id, const USHORT &value, const void *data)
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400297{
Behdad Esfahbod6b54c5d2009-05-18 18:30:25 -0400298 const OffsetTo<Coverage> &coverage = (const OffsetTo<Coverage>&)value;
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400299 return (data+coverage).get_coverage (glyph_id) != NOT_COVERED;
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400300}
301
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400302
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400303static inline bool match_input (hb_apply_context_t *c,
Behdad Esfahbode072c242009-05-18 03:47:31 -0400304 unsigned int count, /* Including the first glyph (not matched) */
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400305 const USHORT input[], /* Array of input values--start with second glyph */
306 match_func_t match_func,
Behdad Esfahbod40cbefe2010-05-10 17:47:22 -0400307 const void *match_data,
Behdad Esfahbod5df809b2012-05-13 15:17:51 +0200308 unsigned int *end_offset = NULL)
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400309{
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500310 hb_apply_context_t::mark_skipping_forward_iterator_t skippy_iter (c, c->buffer->idx, count - 1);
311 if (skippy_iter.has_no_chance ())
Behdad Esfahbod7cff75b2009-05-18 04:09:05 -0400312 return false;
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400313
Behdad Esfahbod370f03e2012-01-16 17:03:55 -0500314 for (unsigned int i = 1; i < count; i++)
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400315 {
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500316 if (!skippy_iter.next ())
317 return false;
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400318
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500319 if (likely (!match_func (c->buffer->info[skippy_iter.idx].codepoint, input[i - 1], match_data)))
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400320 return false;
321 }
322
Behdad Esfahbod5df809b2012-05-13 15:17:51 +0200323 if (end_offset)
324 *end_offset = skippy_iter.idx - c->buffer->idx + 1;
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400325
326 return true;
327}
328
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400329static inline bool match_backtrack (hb_apply_context_t *c,
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400330 unsigned int count,
331 const USHORT backtrack[],
332 match_func_t match_func,
Behdad Esfahbod40cbefe2010-05-10 17:47:22 -0400333 const void *match_data)
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400334{
Behdad Esfahbod28b9d502012-05-13 15:04:00 +0200335 hb_apply_context_t::mark_skipping_backward_iterator_t skippy_iter (c, c->buffer->backtrack_len (), count, true);
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500336 if (skippy_iter.has_no_chance ())
337 return false;
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400338
Behdad Esfahbod4d3aeb82012-01-16 16:43:26 -0500339 for (unsigned int i = 0; i < count; i++)
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400340 {
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500341 if (!skippy_iter.prev ())
342 return false;
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400343
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500344 if (likely (!match_func (c->buffer->out_info[skippy_iter.idx].codepoint, backtrack[i], match_data)))
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400345 return false;
346 }
347
348 return true;
349}
350
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400351static inline bool match_lookahead (hb_apply_context_t *c,
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400352 unsigned int count,
353 const USHORT lookahead[],
354 match_func_t match_func,
Behdad Esfahbod40cbefe2010-05-10 17:47:22 -0400355 const void *match_data,
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400356 unsigned int offset)
357{
Behdad Esfahbod28b9d502012-05-13 15:04:00 +0200358 hb_apply_context_t::mark_skipping_forward_iterator_t skippy_iter (c, c->buffer->idx + offset - 1, count, true);
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500359 if (skippy_iter.has_no_chance ())
360 return false;
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400361
Behdad Esfahbod370f03e2012-01-16 17:03:55 -0500362 for (unsigned int i = 0; i < count; i++)
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400363 {
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500364 if (!skippy_iter.next ())
365 return false;
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400366
Behdad Esfahbod4ab97312012-01-16 22:05:08 -0500367 if (likely (!match_func (c->buffer->info[skippy_iter.idx].codepoint, lookahead[i], match_data)))
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400368 return false;
369 }
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400370
371 return true;
372}
373
Behdad Esfahbodacdba3f2010-07-23 15:11:18 -0400374
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400375
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400376struct LookupRecord
377{
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400378 inline bool sanitize (hb_sanitize_context_t *c) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400379 TRACE_SANITIZE ();
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +0200380 return TRACE_RETURN (c->check_struct (this));
Behdad Esfahbodcd3827e2009-08-04 02:09:34 -0400381 }
382
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400383 USHORT sequenceIndex; /* Index into current glyph
384 * sequence--first glyph = 0 */
385 USHORT lookupListIndex; /* Lookup to apply to that
386 * position--zero--based */
Behdad Esfahbod569da922010-05-10 16:38:32 -0400387 public:
388 DEFINE_SIZE_STATIC (4);
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400389};
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400390
Behdad Esfahbodacdba3f2010-07-23 15:11:18 -0400391
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400392static inline void closure_lookup (hb_closure_context_t *c,
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400393 unsigned int lookupCount,
394 const LookupRecord lookupRecord[], /* Array of LookupRecords--in design order */
395 closure_lookup_func_t closure_func)
396{
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400397 for (unsigned int i = 0; i < lookupCount; i++)
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400398 closure_func (c, lookupRecord->lookupListIndex);
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400399}
Behdad Esfahbodacdba3f2010-07-23 15:11:18 -0400400
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400401static inline bool apply_lookup (hb_apply_context_t *c,
Behdad Esfahbode072c242009-05-18 03:47:31 -0400402 unsigned int count, /* Including the first glyph */
403 unsigned int lookupCount,
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400404 const LookupRecord lookupRecord[], /* Array of LookupRecords--in design order */
405 apply_lookup_func_t apply_func)
406{
Behdad Esfahbod5df809b2012-05-13 15:17:51 +0200407 unsigned int end = c->buffer->len;
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400408 if (unlikely (count == 0 || c->buffer->idx + count > end))
Behdad Esfahbode73a0c22009-05-18 04:15:25 -0400409 return false;
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400410
Behdad Esfahbod7cff75b2009-05-18 04:09:05 -0400411 /* TODO We don't support lookupRecord arrays that are not increasing:
412 * Should be easy for in_place ones at least. */
Behdad Esfahbod52e9a712009-09-21 13:58:56 -0400413
Behdad Esfahbod7eb87512010-12-13 14:13:35 -0500414 /* Note: If sublookup is reverse, it will underflow after the first loop
Behdad Esfahbod52e9a712009-09-21 13:58:56 -0400415 * and we jump out of it. Not entirely disastrous. So we don't check
416 * for reverse lookup here.
417 */
Behdad Esfahbodf9c0a2d2009-09-21 13:43:54 -0400418 for (unsigned int i = 0; i < count; /* NOP */)
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400419 {
Behdad Esfahbod03408ce2012-01-18 21:28:34 -0500420 if (unlikely (c->buffer->idx == end))
421 return true;
Behdad Esfahbod7d479902012-01-18 21:19:32 -0500422 while (c->should_mark_skip_current_glyph ())
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400423 {
Behdad Esfahbode73a0c22009-05-18 04:15:25 -0400424 /* No lookup applied for this index */
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400425 c->buffer->next_glyph ();
Behdad Esfahbod03408ce2012-01-18 21:28:34 -0500426 if (unlikely (c->buffer->idx == end))
427 return true;
Behdad Esfahbode73a0c22009-05-18 04:15:25 -0400428 }
429
Behdad Esfahbod47958de2009-05-18 04:17:47 -0400430 if (lookupCount && i == lookupRecord->sequenceIndex)
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400431 {
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400432 unsigned int old_pos = c->buffer->idx;
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400433
434 /* Apply a lookup */
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400435 bool done = apply_func (c, lookupRecord->lookupListIndex);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400436
Behdad Esfahbod47958de2009-05-18 04:17:47 -0400437 lookupRecord++;
438 lookupCount--;
Behdad Esfahbodf9c0a2d2009-09-21 13:43:54 -0400439 /* Err, this is wrong if the lookup jumped over some glyphs */
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400440 i += c->buffer->idx - old_pos;
441 if (unlikely (c->buffer->idx == end))
Behdad Esfahbod2e8fb6c2009-05-18 04:37:37 -0400442 return true;
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400443
444 if (!done)
445 goto not_applied;
446 }
447 else
448 {
449 not_applied:
450 /* No lookup applied for this index */
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400451 c->buffer->next_glyph ();
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400452 i++;
453 }
454 }
455
456 return true;
457}
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400458
Behdad Esfahbodacdba3f2010-07-23 15:11:18 -0400459
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400460
461/* Contextual lookups */
462
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400463struct ContextClosureLookupContext
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400464{
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400465 ContextClosureFuncs funcs;
466 const void *intersects_data;
467};
468
469struct ContextApplyLookupContext
470{
471 ContextApplyFuncs funcs;
Behdad Esfahbod40cbefe2010-05-10 17:47:22 -0400472 const void *match_data;
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400473};
474
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400475static inline void context_closure_lookup (hb_closure_context_t *c,
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400476 unsigned int inputCount, /* Including the first glyph (not matched) */
477 const USHORT input[], /* Array of input values--start with second glyph */
478 unsigned int lookupCount,
479 const LookupRecord lookupRecord[],
480 ContextClosureLookupContext &lookup_context)
481{
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400482 if (intersects_array (c,
483 inputCount ? inputCount - 1 : 0, input,
484 lookup_context.funcs.intersects, lookup_context.intersects_data))
485 closure_lookup (c,
486 lookupCount, lookupRecord,
487 lookup_context.funcs.closure);
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400488}
489
490
491static inline bool context_apply_lookup (hb_apply_context_t *c,
492 unsigned int inputCount, /* Including the first glyph (not matched) */
493 const USHORT input[], /* Array of input values--start with second glyph */
494 unsigned int lookupCount,
495 const LookupRecord lookupRecord[],
496 ContextApplyLookupContext &lookup_context)
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400497{
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400498 return match_input (c,
Behdad Esfahbod41697102010-05-05 01:37:58 -0400499 inputCount, input,
Behdad Esfahbod5df809b2012-05-13 15:17:51 +0200500 lookup_context.funcs.match, lookup_context.match_data)
501 && apply_lookup (c,
Behdad Esfahbod41697102010-05-05 01:37:58 -0400502 inputCount,
503 lookupCount, lookupRecord,
504 lookup_context.funcs.apply);
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400505}
506
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400507struct Rule
508{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400509 friend struct RuleSet;
510
511 private:
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400512
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400513 inline void closure (hb_closure_context_t *c, ContextClosureLookupContext &lookup_context) const
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400514 {
515 TRACE_CLOSURE ();
516 const LookupRecord *lookupRecord = &StructAtOffset<LookupRecord> (input, input[0].static_size * (inputCount ? inputCount - 1 : 0));
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400517 context_closure_lookup (c,
518 inputCount, input,
519 lookupCount, lookupRecord,
520 lookup_context);
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400521 }
522
523 inline bool apply (hb_apply_context_t *c, ContextApplyLookupContext &lookup_context) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400524 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400525 TRACE_APPLY ();
Behdad Esfahbode45d3f82010-05-06 19:33:31 -0400526 const LookupRecord *lookupRecord = &StructAtOffset<LookupRecord> (input, input[0].static_size * (inputCount ? inputCount - 1 : 0));
Behdad Esfahbodacea1832012-05-11 02:33:11 +0200527 return TRACE_RETURN (context_apply_lookup (c, inputCount, input, lookupCount, lookupRecord, lookup_context));
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400528 }
529
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400530 public:
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400531 inline bool sanitize (hb_sanitize_context_t *c) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400532 TRACE_SANITIZE ();
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400533 return inputCount.sanitize (c)
534 && lookupCount.sanitize (c)
535 && c->check_range (input,
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -0400536 input[0].static_size * inputCount
537 + lookupRecordX[0].static_size * lookupCount);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400538 }
539
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400540 private:
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400541 USHORT inputCount; /* Total number of glyphs in input
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400542 * glyph sequence--includes the first
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400543 * glyph */
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400544 USHORT lookupCount; /* Number of LookupRecords */
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500545 USHORT input[VAR]; /* Array of match inputs--start with
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400546 * second glyph */
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500547 LookupRecord lookupRecordX[VAR]; /* Array of LookupRecords--in
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400548 * design order */
Behdad Esfahbod569da922010-05-10 16:38:32 -0400549 public:
Behdad Esfahbod0eb9fc62010-05-10 19:01:17 -0400550 DEFINE_SIZE_ARRAY2 (4, input, lookupRecordX);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400551};
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400552
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400553struct RuleSet
554{
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400555 inline void closure (hb_closure_context_t *c, ContextClosureLookupContext &lookup_context) const
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400556 {
557 TRACE_CLOSURE ();
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400558 unsigned int num_rules = rule.len;
559 for (unsigned int i = 0; i < num_rules; i++)
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400560 (this+rule[i]).closure (c, lookup_context);
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400561 }
562
563 inline bool apply (hb_apply_context_t *c, ContextApplyLookupContext &lookup_context) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400564 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400565 TRACE_APPLY ();
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400566 unsigned int num_rules = rule.len;
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400567 for (unsigned int i = 0; i < num_rules; i++)
568 {
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400569 if ((this+rule[i]).apply (c, lookup_context))
Behdad Esfahbodacea1832012-05-11 02:33:11 +0200570 return TRACE_RETURN (true);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400571 }
Behdad Esfahbodacea1832012-05-11 02:33:11 +0200572 return TRACE_RETURN (false);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400573 }
574
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400575 inline bool sanitize (hb_sanitize_context_t *c) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400576 TRACE_SANITIZE ();
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +0200577 return TRACE_RETURN (rule.sanitize (c, this));
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400578 }
579
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400580 private:
581 OffsetArrayOf<Rule>
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400582 rule; /* Array of Rule tables
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400583 * ordered by preference */
Behdad Esfahboded074222010-05-10 18:08:46 -0400584 public:
Behdad Esfahbod0eb9fc62010-05-10 19:01:17 -0400585 DEFINE_SIZE_ARRAY (2, rule);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400586};
587
588
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400589struct ContextFormat1
590{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400591 friend struct Context;
592
593 private:
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -0400594
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400595 inline void closure (hb_closure_context_t *c, closure_lookup_func_t closure_func) const
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -0400596 {
597 TRACE_CLOSURE ();
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400598
599 const Coverage &cov = (this+coverage);
600
601 struct ContextClosureLookupContext lookup_context = {
602 {intersects_glyph, closure_func},
603 NULL
604 };
605
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400606 unsigned int count = ruleSet.len;
607 for (unsigned int i = 0; i < count; i++)
608 if (cov.intersects_coverage (c->glyphs, i)) {
609 const RuleSet &rule_set = this+ruleSet[i];
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400610 rule_set.closure (c, lookup_context);
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400611 }
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -0400612 }
613
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400614 inline bool apply (hb_apply_context_t *c, apply_lookup_func_t apply_func) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400615 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400616 TRACE_APPLY ();
Behdad Esfahbod99c26952012-05-13 15:45:18 +0200617 unsigned int index = (this+coverage) (c->buffer->cur().codepoint);
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400618 if (likely (index == NOT_COVERED))
Behdad Esfahbodacea1832012-05-11 02:33:11 +0200619 return TRACE_RETURN (false);
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400620
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400621 const RuleSet &rule_set = this+ruleSet[index];
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400622 struct ContextApplyLookupContext lookup_context = {
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400623 {match_glyph, apply_func},
624 NULL
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400625 };
Behdad Esfahbodacea1832012-05-11 02:33:11 +0200626 return TRACE_RETURN (rule_set.apply (c, lookup_context));
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400627 }
628
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400629 inline bool sanitize (hb_sanitize_context_t *c) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400630 TRACE_SANITIZE ();
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +0200631 return TRACE_RETURN (coverage.sanitize (c, this) && ruleSet.sanitize (c, this));
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400632 }
633
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400634 private:
635 USHORT format; /* Format identifier--format = 1 */
636 OffsetTo<Coverage>
637 coverage; /* Offset to Coverage table--from
638 * beginning of table */
639 OffsetArrayOf<RuleSet>
640 ruleSet; /* Array of RuleSet tables
641 * ordered by Coverage Index */
Behdad Esfahbodb3651232010-05-10 16:57:29 -0400642 public:
Behdad Esfahbod0eb9fc62010-05-10 19:01:17 -0400643 DEFINE_SIZE_ARRAY (6, ruleSet);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400644};
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400645
646
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400647struct ContextFormat2
648{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400649 friend struct Context;
650
651 private:
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -0400652
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400653 inline void closure (hb_closure_context_t *c, closure_lookup_func_t closure_func) const
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -0400654 {
655 TRACE_CLOSURE ();
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400656 if (!(this+coverage).intersects (c->glyphs))
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400657 return;
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400658
659 const ClassDef &class_def = this+classDef;
660
661 struct ContextClosureLookupContext lookup_context = {
662 {intersects_class, closure_func},
663 NULL
664 };
665
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400666 unsigned int count = ruleSet.len;
667 for (unsigned int i = 0; i < count; i++)
668 if (class_def.intersects_class (c->glyphs, i)) {
669 const RuleSet &rule_set = this+ruleSet[i];
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400670 rule_set.closure (c, lookup_context);
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400671 }
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -0400672 }
673
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400674 inline bool apply (hb_apply_context_t *c, apply_lookup_func_t apply_func) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400675 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400676 TRACE_APPLY ();
Behdad Esfahbod99c26952012-05-13 15:45:18 +0200677 unsigned int index = (this+coverage) (c->buffer->cur().codepoint);
Behdad Esfahbodacea1832012-05-11 02:33:11 +0200678 if (likely (index == NOT_COVERED)) return TRACE_RETURN (false);
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400679
680 const ClassDef &class_def = this+classDef;
Behdad Esfahbod99c26952012-05-13 15:45:18 +0200681 index = class_def (c->buffer->cur().codepoint);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400682 const RuleSet &rule_set = this+ruleSet[index];
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400683 struct ContextApplyLookupContext lookup_context = {
Behdad Esfahbod40cbefe2010-05-10 17:47:22 -0400684 {match_class, apply_func},
685 &class_def
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400686 };
Behdad Esfahbodacea1832012-05-11 02:33:11 +0200687 return TRACE_RETURN (rule_set.apply (c, lookup_context));
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400688 }
689
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400690 inline bool sanitize (hb_sanitize_context_t *c) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400691 TRACE_SANITIZE ();
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +0200692 return TRACE_RETURN (coverage.sanitize (c, this) && classDef.sanitize (c, this) && ruleSet.sanitize (c, this));
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400693 }
694
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400695 private:
696 USHORT format; /* Format identifier--format = 2 */
697 OffsetTo<Coverage>
698 coverage; /* Offset to Coverage table--from
699 * beginning of table */
700 OffsetTo<ClassDef>
701 classDef; /* Offset to glyph ClassDef table--from
702 * beginning of table */
703 OffsetArrayOf<RuleSet>
704 ruleSet; /* Array of RuleSet tables
705 * ordered by class */
Behdad Esfahbodb3651232010-05-10 16:57:29 -0400706 public:
Behdad Esfahbod0eb9fc62010-05-10 19:01:17 -0400707 DEFINE_SIZE_ARRAY (8, ruleSet);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400708};
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400709
710
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400711struct ContextFormat3
712{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400713 friend struct Context;
714
715 private:
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -0400716
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400717 inline void closure (hb_closure_context_t *c, closure_lookup_func_t closure_func) const
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -0400718 {
719 TRACE_CLOSURE ();
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400720 if (!(this+coverage[0]).intersects (c->glyphs))
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400721 return;
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400722
723 const LookupRecord *lookupRecord = &StructAtOffset<LookupRecord> (coverage, coverage[0].static_size * glyphCount);
724 struct ContextClosureLookupContext lookup_context = {
725 {intersects_coverage, closure_func},
726 this
727 };
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400728 context_closure_lookup (c,
729 glyphCount, (const USHORT *) (coverage + 1),
730 lookupCount, lookupRecord,
731 lookup_context);
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -0400732 }
733
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400734 inline bool apply (hb_apply_context_t *c, apply_lookup_func_t apply_func) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400735 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400736 TRACE_APPLY ();
Behdad Esfahbod99c26952012-05-13 15:45:18 +0200737 unsigned int index = (this+coverage[0]) (c->buffer->cur().codepoint);
Behdad Esfahbodacea1832012-05-11 02:33:11 +0200738 if (likely (index == NOT_COVERED)) return TRACE_RETURN (false);
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400739
Behdad Esfahbode45d3f82010-05-06 19:33:31 -0400740 const LookupRecord *lookupRecord = &StructAtOffset<LookupRecord> (coverage, coverage[0].static_size * glyphCount);
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400741 struct ContextApplyLookupContext lookup_context = {
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400742 {match_coverage, apply_func},
Behdad Esfahbod40cbefe2010-05-10 17:47:22 -0400743 this
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400744 };
Behdad Esfahbodacea1832012-05-11 02:33:11 +0200745 return TRACE_RETURN (context_apply_lookup (c, glyphCount, (const USHORT *) (coverage + 1), lookupCount, lookupRecord, lookup_context));
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400746 }
747
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400748 inline bool sanitize (hb_sanitize_context_t *c) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400749 TRACE_SANITIZE ();
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +0200750 if (!c->check_struct (this)) return TRACE_RETURN (false);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400751 unsigned int count = glyphCount;
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +0200752 if (!c->check_array (coverage, coverage[0].static_size, count)) return TRACE_RETURN (false);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400753 for (unsigned int i = 0; i < count; i++)
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +0200754 if (!coverage[i].sanitize (c, this)) return TRACE_RETURN (false);
Behdad Esfahbode45d3f82010-05-06 19:33:31 -0400755 LookupRecord *lookupRecord = &StructAtOffset<LookupRecord> (coverage, coverage[0].static_size * count);
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +0200756 return TRACE_RETURN (c->check_array (lookupRecord, lookupRecord[0].static_size, lookupCount));
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400757 }
758
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400759 private:
760 USHORT format; /* Format identifier--format = 3 */
761 USHORT glyphCount; /* Number of glyphs in the input glyph
762 * sequence */
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400763 USHORT lookupCount; /* Number of LookupRecords */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400764 OffsetTo<Coverage>
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500765 coverage[VAR]; /* Array of offsets to Coverage
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400766 * table in glyph sequence order */
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500767 LookupRecord lookupRecordX[VAR]; /* Array of LookupRecords--in
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400768 * design order */
Behdad Esfahbod569da922010-05-10 16:38:32 -0400769 public:
Behdad Esfahbod0eb9fc62010-05-10 19:01:17 -0400770 DEFINE_SIZE_ARRAY2 (6, coverage, lookupRecordX);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400771};
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400772
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400773struct Context
774{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400775 protected:
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -0400776
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400777 inline void closure (hb_closure_context_t *c, closure_lookup_func_t closure_func) const
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -0400778 {
779 TRACE_CLOSURE ();
780 switch (u.format) {
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400781 case 1: u.format1.closure (c, closure_func); break;
782 case 2: u.format2.closure (c, closure_func); break;
783 case 3: u.format3.closure (c, closure_func); break;
784 default: break;
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -0400785 }
786 }
787
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400788 inline bool apply (hb_apply_context_t *c, apply_lookup_func_t apply_func) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400789 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400790 TRACE_APPLY ();
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400791 switch (u.format) {
Behdad Esfahbodacea1832012-05-11 02:33:11 +0200792 case 1: return TRACE_RETURN (u.format1.apply (c, apply_func));
793 case 2: return TRACE_RETURN (u.format2.apply (c, apply_func));
794 case 3: return TRACE_RETURN (u.format3.apply (c, apply_func));
795 default:return TRACE_RETURN (false);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400796 }
797 }
798
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400799 inline bool sanitize (hb_sanitize_context_t *c) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400800 TRACE_SANITIZE ();
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +0200801 if (!u.format.sanitize (c)) return TRACE_RETURN (false);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400802 switch (u.format) {
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +0200803 case 1: return TRACE_RETURN (u.format1.sanitize (c));
804 case 2: return TRACE_RETURN (u.format2.sanitize (c));
805 case 3: return TRACE_RETURN (u.format3.sanitize (c));
806 default:return TRACE_RETURN (true);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400807 }
808 }
809
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400810 private:
811 union {
Behdad Esfahbodf8dc67b2009-05-17 19:47:54 -0400812 USHORT format; /* Format identifier */
Behdad Esfahboddacebca2010-05-10 19:45:41 -0400813 ContextFormat1 format1;
814 ContextFormat2 format2;
815 ContextFormat3 format3;
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400816 } u;
817};
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400818
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400819
820/* Chaining Contextual lookups */
821
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400822struct ChainContextClosureLookupContext
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400823{
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400824 ContextClosureFuncs funcs;
825 const void *intersects_data[3];
826};
827
828struct ChainContextApplyLookupContext
829{
830 ContextApplyFuncs funcs;
Behdad Esfahbod40cbefe2010-05-10 17:47:22 -0400831 const void *match_data[3];
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400832};
833
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400834static inline void chain_context_closure_lookup (hb_closure_context_t *c,
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400835 unsigned int backtrackCount,
836 const USHORT backtrack[],
837 unsigned int inputCount, /* Including the first glyph (not matched) */
838 const USHORT input[], /* Array of input values--start with second glyph */
839 unsigned int lookaheadCount,
840 const USHORT lookahead[],
841 unsigned int lookupCount,
842 const LookupRecord lookupRecord[],
843 ChainContextClosureLookupContext &lookup_context)
844{
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400845 if (intersects_array (c,
846 backtrackCount, backtrack,
847 lookup_context.funcs.intersects, lookup_context.intersects_data[0])
848 && intersects_array (c,
849 inputCount ? inputCount - 1 : 0, input,
850 lookup_context.funcs.intersects, lookup_context.intersects_data[1])
851 && intersects_array (c,
852 lookaheadCount, lookahead,
853 lookup_context.funcs.intersects, lookup_context.intersects_data[2]))
854 closure_lookup (c,
855 lookupCount, lookupRecord,
856 lookup_context.funcs.closure);
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400857}
858
859static inline bool chain_context_apply_lookup (hb_apply_context_t *c,
860 unsigned int backtrackCount,
861 const USHORT backtrack[],
862 unsigned int inputCount, /* Including the first glyph (not matched) */
863 const USHORT input[], /* Array of input values--start with second glyph */
864 unsigned int lookaheadCount,
865 const USHORT lookahead[],
866 unsigned int lookupCount,
867 const LookupRecord lookupRecord[],
868 ChainContextApplyLookupContext &lookup_context)
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400869{
Behdad Esfahbod5df809b2012-05-13 15:17:51 +0200870 unsigned int lookahead_offset;
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400871 return match_backtrack (c,
Behdad Esfahbod41697102010-05-05 01:37:58 -0400872 backtrackCount, backtrack,
873 lookup_context.funcs.match, lookup_context.match_data[0])
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400874 && match_input (c,
Behdad Esfahbod41697102010-05-05 01:37:58 -0400875 inputCount, input,
876 lookup_context.funcs.match, lookup_context.match_data[1],
Behdad Esfahbod5df809b2012-05-13 15:17:51 +0200877 &lookahead_offset)
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400878 && match_lookahead (c,
Behdad Esfahbod41697102010-05-05 01:37:58 -0400879 lookaheadCount, lookahead,
880 lookup_context.funcs.match, lookup_context.match_data[2],
Behdad Esfahbod5df809b2012-05-13 15:17:51 +0200881 lookahead_offset)
882 && apply_lookup (c,
Behdad Esfahbod41697102010-05-05 01:37:58 -0400883 inputCount,
884 lookupCount, lookupRecord,
885 lookup_context.funcs.apply);
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400886}
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400887
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400888struct ChainRule
889{
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400890 friend struct ChainRuleSet;
891
892 private:
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400893
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400894 inline void closure (hb_closure_context_t *c, ChainContextClosureLookupContext &lookup_context) const
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400895 {
896 TRACE_CLOSURE ();
897 const HeadlessArrayOf<USHORT> &input = StructAfter<HeadlessArrayOf<USHORT> > (backtrack);
898 const ArrayOf<USHORT> &lookahead = StructAfter<ArrayOf<USHORT> > (input);
899 const ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord> > (lookahead);
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400900 chain_context_closure_lookup (c,
901 backtrack.len, backtrack.array,
902 input.len, input.array,
903 lookahead.len, lookahead.array,
904 lookup.len, lookup.array,
905 lookup_context);
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400906 }
907
908 inline bool apply (hb_apply_context_t *c, ChainContextApplyLookupContext &lookup_context) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400909 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400910 TRACE_APPLY ();
Behdad Esfahbode961c862010-04-21 15:56:11 -0400911 const HeadlessArrayOf<USHORT> &input = StructAfter<HeadlessArrayOf<USHORT> > (backtrack);
912 const ArrayOf<USHORT> &lookahead = StructAfter<ArrayOf<USHORT> > (input);
913 const ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord> > (lookahead);
Behdad Esfahbodacea1832012-05-11 02:33:11 +0200914 return TRACE_RETURN (chain_context_apply_lookup (c,
915 backtrack.len, backtrack.array,
916 input.len, input.array,
917 lookahead.len, lookahead.array, lookup.len,
918 lookup.array, lookup_context));
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400919 }
920
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400921 public:
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400922 inline bool sanitize (hb_sanitize_context_t *c) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400923 TRACE_SANITIZE ();
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +0200924 if (!backtrack.sanitize (c)) return TRACE_RETURN (false);
Behdad Esfahbode961c862010-04-21 15:56:11 -0400925 HeadlessArrayOf<USHORT> &input = StructAfter<HeadlessArrayOf<USHORT> > (backtrack);
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +0200926 if (!input.sanitize (c)) return TRACE_RETURN (false);
Behdad Esfahbode961c862010-04-21 15:56:11 -0400927 ArrayOf<USHORT> &lookahead = StructAfter<ArrayOf<USHORT> > (input);
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +0200928 if (!lookahead.sanitize (c)) return TRACE_RETURN (false);
Behdad Esfahbode961c862010-04-21 15:56:11 -0400929 ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord> > (lookahead);
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +0200930 return TRACE_RETURN (lookup.sanitize (c));
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400931 }
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400932
933 private:
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400934 ArrayOf<USHORT>
935 backtrack; /* Array of backtracking values
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400936 * (to be matched before the input
937 * sequence) */
Behdad Esfahbode8cbaaf2009-05-18 02:03:58 -0400938 HeadlessArrayOf<USHORT>
939 inputX; /* Array of input values (start with
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400940 * second glyph) */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400941 ArrayOf<USHORT>
942 lookaheadX; /* Array of lookahead values's (to be
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400943 * matched after the input sequence) */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400944 ArrayOf<LookupRecord>
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400945 lookupX; /* Array of LookupRecords--in
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400946 * design order) */
Behdad Esfahbodb3651232010-05-10 16:57:29 -0400947 public:
Behdad Esfahbodbea34c72010-05-10 17:28:16 -0400948 DEFINE_SIZE_MIN (8);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400949};
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400950
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400951struct ChainRuleSet
952{
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400953 inline void closure (hb_closure_context_t *c, ChainContextClosureLookupContext &lookup_context) const
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400954 {
955 TRACE_CLOSURE ();
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400956 unsigned int num_rules = rule.len;
957 for (unsigned int i = 0; i < num_rules; i++)
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400958 (this+rule[i]).closure (c, lookup_context);
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400959 }
960
961 inline bool apply (hb_apply_context_t *c, ChainContextApplyLookupContext &lookup_context) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400962 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400963 TRACE_APPLY ();
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400964 unsigned int num_rules = rule.len;
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400965 for (unsigned int i = 0; i < num_rules; i++)
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400966 if ((this+rule[i]).apply (c, lookup_context))
Behdad Esfahbodacea1832012-05-11 02:33:11 +0200967 return TRACE_RETURN (true);
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400968
Behdad Esfahbodacea1832012-05-11 02:33:11 +0200969 return TRACE_RETURN (false);
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400970 }
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400971
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -0400972 inline bool sanitize (hb_sanitize_context_t *c) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400973 TRACE_SANITIZE ();
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +0200974 return TRACE_RETURN (rule.sanitize (c, this));
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400975 }
976
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400977 private:
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400978 OffsetArrayOf<ChainRule>
979 rule; /* Array of ChainRule tables
980 * ordered by preference */
Behdad Esfahbodb3651232010-05-10 16:57:29 -0400981 public:
Behdad Esfahbod0eb9fc62010-05-10 19:01:17 -0400982 DEFINE_SIZE_ARRAY (2, rule);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400983};
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400984
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400985struct ChainContextFormat1
986{
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400987 friend struct ChainContext;
988
989 private:
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -0400990
Behdad Esfahbod5caece62012-04-23 23:03:12 -0400991 inline void closure (hb_closure_context_t *c, closure_lookup_func_t closure_func) const
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -0400992 {
993 TRACE_CLOSURE ();
Behdad Esfahbod31081f72012-04-23 16:54:58 -0400994 const Coverage &cov = (this+coverage);
995
996 struct ChainContextClosureLookupContext lookup_context = {
997 {intersects_glyph, closure_func},
998 {NULL, NULL, NULL}
999 };
1000
Behdad Esfahbod31081f72012-04-23 16:54:58 -04001001 unsigned int count = ruleSet.len;
1002 for (unsigned int i = 0; i < count; i++)
1003 if (cov.intersects_coverage (c->glyphs, i)) {
1004 const ChainRuleSet &rule_set = this+ruleSet[i];
Behdad Esfahbod5caece62012-04-23 23:03:12 -04001005 rule_set.closure (c, lookup_context);
Behdad Esfahbod31081f72012-04-23 16:54:58 -04001006 }
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -04001007 }
1008
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -04001009 inline bool apply (hb_apply_context_t *c, apply_lookup_func_t apply_func) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -04001010 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -04001011 TRACE_APPLY ();
Behdad Esfahbod99c26952012-05-13 15:45:18 +02001012 unsigned int index = (this+coverage) (c->buffer->cur().codepoint);
Behdad Esfahbodacea1832012-05-11 02:33:11 +02001013 if (likely (index == NOT_COVERED)) return TRACE_RETURN (false);
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -04001014
1015 const ChainRuleSet &rule_set = this+ruleSet[index];
Behdad Esfahbod31081f72012-04-23 16:54:58 -04001016 struct ChainContextApplyLookupContext lookup_context = {
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -04001017 {match_glyph, apply_func},
1018 {NULL, NULL, NULL}
1019 };
Behdad Esfahbodacea1832012-05-11 02:33:11 +02001020 return TRACE_RETURN (rule_set.apply (c, lookup_context));
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001021 }
Behdad Esfahbod70de50c2009-08-04 00:58:28 -04001022
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -04001023 inline bool sanitize (hb_sanitize_context_t *c) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -04001024 TRACE_SANITIZE ();
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +02001025 return TRACE_RETURN (coverage.sanitize (c, this) && ruleSet.sanitize (c, this));
Behdad Esfahbod70de50c2009-08-04 00:58:28 -04001026 }
1027
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001028 private:
1029 USHORT format; /* Format identifier--format = 1 */
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -04001030 OffsetTo<Coverage>
1031 coverage; /* Offset to Coverage table--from
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001032 * beginning of table */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -04001033 OffsetArrayOf<ChainRuleSet>
1034 ruleSet; /* Array of ChainRuleSet tables
1035 * ordered by Coverage Index */
Behdad Esfahbodb3651232010-05-10 16:57:29 -04001036 public:
Behdad Esfahbod0eb9fc62010-05-10 19:01:17 -04001037 DEFINE_SIZE_ARRAY (6, ruleSet);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001038};
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001039
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -04001040struct ChainContextFormat2
1041{
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001042 friend struct ChainContext;
1043
1044 private:
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -04001045
Behdad Esfahbod5caece62012-04-23 23:03:12 -04001046 inline void closure (hb_closure_context_t *c, closure_lookup_func_t closure_func) const
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -04001047 {
1048 TRACE_CLOSURE ();
Behdad Esfahbod31081f72012-04-23 16:54:58 -04001049 if (!(this+coverage).intersects (c->glyphs))
Behdad Esfahbod5caece62012-04-23 23:03:12 -04001050 return;
Behdad Esfahbod31081f72012-04-23 16:54:58 -04001051
1052 const ClassDef &backtrack_class_def = this+backtrackClassDef;
1053 const ClassDef &input_class_def = this+inputClassDef;
1054 const ClassDef &lookahead_class_def = this+lookaheadClassDef;
1055
1056 struct ChainContextClosureLookupContext lookup_context = {
1057 {intersects_class, closure_func},
1058 {&backtrack_class_def,
1059 &input_class_def,
1060 &lookahead_class_def}
1061 };
1062
Behdad Esfahbod31081f72012-04-23 16:54:58 -04001063 unsigned int count = ruleSet.len;
1064 for (unsigned int i = 0; i < count; i++)
1065 if (input_class_def.intersects_class (c->glyphs, i)) {
1066 const ChainRuleSet &rule_set = this+ruleSet[i];
Behdad Esfahbod5caece62012-04-23 23:03:12 -04001067 rule_set.closure (c, lookup_context);
Behdad Esfahbod31081f72012-04-23 16:54:58 -04001068 }
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -04001069 }
1070
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -04001071 inline bool apply (hb_apply_context_t *c, apply_lookup_func_t apply_func) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -04001072 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -04001073 TRACE_APPLY ();
Behdad Esfahbod99c26952012-05-13 15:45:18 +02001074 unsigned int index = (this+coverage) (c->buffer->cur().codepoint);
Behdad Esfahbodacea1832012-05-11 02:33:11 +02001075 if (likely (index == NOT_COVERED)) return TRACE_RETURN (false);
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -04001076
1077 const ClassDef &backtrack_class_def = this+backtrackClassDef;
1078 const ClassDef &input_class_def = this+inputClassDef;
1079 const ClassDef &lookahead_class_def = this+lookaheadClassDef;
1080
Behdad Esfahbod99c26952012-05-13 15:45:18 +02001081 index = input_class_def (c->buffer->cur().codepoint);
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -04001082 const ChainRuleSet &rule_set = this+ruleSet[index];
Behdad Esfahbod31081f72012-04-23 16:54:58 -04001083 struct ChainContextApplyLookupContext lookup_context = {
Behdad Esfahbod40cbefe2010-05-10 17:47:22 -04001084 {match_class, apply_func},
1085 {&backtrack_class_def,
1086 &input_class_def,
1087 &lookahead_class_def}
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -04001088 };
Behdad Esfahbodacea1832012-05-11 02:33:11 +02001089 return TRACE_RETURN (rule_set.apply (c, lookup_context));
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001090 }
1091
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -04001092 inline bool sanitize (hb_sanitize_context_t *c) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -04001093 TRACE_SANITIZE ();
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +02001094 return TRACE_RETURN (coverage.sanitize (c, this) && backtrackClassDef.sanitize (c, this) &&
1095 inputClassDef.sanitize (c, this) && lookaheadClassDef.sanitize (c, this) &&
1096 ruleSet.sanitize (c, this));
Behdad Esfahbod70de50c2009-08-04 00:58:28 -04001097 }
1098
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001099 private:
1100 USHORT format; /* Format identifier--format = 2 */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -04001101 OffsetTo<Coverage>
1102 coverage; /* Offset to Coverage table--from
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001103 * beginning of table */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -04001104 OffsetTo<ClassDef>
1105 backtrackClassDef; /* Offset to glyph ClassDef table
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001106 * containing backtrack sequence
1107 * data--from beginning of table */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -04001108 OffsetTo<ClassDef>
1109 inputClassDef; /* Offset to glyph ClassDef
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001110 * table containing input sequence
1111 * data--from beginning of table */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -04001112 OffsetTo<ClassDef>
1113 lookaheadClassDef; /* Offset to glyph ClassDef table
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001114 * containing lookahead sequence
1115 * data--from beginning of table */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -04001116 OffsetArrayOf<ChainRuleSet>
1117 ruleSet; /* Array of ChainRuleSet tables
1118 * ordered by class */
Behdad Esfahbodb3651232010-05-10 16:57:29 -04001119 public:
Behdad Esfahbod0eb9fc62010-05-10 19:01:17 -04001120 DEFINE_SIZE_ARRAY (12, ruleSet);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001121};
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001122
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -04001123struct ChainContextFormat3
1124{
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001125 friend struct ChainContext;
1126
1127 private:
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -04001128
Behdad Esfahbod5caece62012-04-23 23:03:12 -04001129 inline void closure (hb_closure_context_t *c, closure_lookup_func_t closure_func) const
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -04001130 {
1131 TRACE_CLOSURE ();
Behdad Esfahbod5caece62012-04-23 23:03:12 -04001132 const OffsetArrayOf<Coverage> &input = StructAfter<OffsetArrayOf<Coverage> > (backtrack);
1133
1134 if (!(this+input[0]).intersects (c->glyphs))
1135 return;
1136
1137 const OffsetArrayOf<Coverage> &lookahead = StructAfter<OffsetArrayOf<Coverage> > (input);
1138 const ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord> > (lookahead);
1139 struct ChainContextClosureLookupContext lookup_context = {
1140 {intersects_coverage, closure_func},
1141 {this, this, this}
1142 };
1143 chain_context_closure_lookup (c,
1144 backtrack.len, (const USHORT *) backtrack.array,
1145 input.len, (const USHORT *) input.array + 1,
1146 lookahead.len, (const USHORT *) lookahead.array,
1147 lookup.len, lookup.array,
1148 lookup_context);
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -04001149 }
1150
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -04001151 inline bool apply (hb_apply_context_t *c, apply_lookup_func_t apply_func) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -04001152 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -04001153 TRACE_APPLY ();
Behdad Esfahbode961c862010-04-21 15:56:11 -04001154 const OffsetArrayOf<Coverage> &input = StructAfter<OffsetArrayOf<Coverage> > (backtrack);
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -04001155
Behdad Esfahbod99c26952012-05-13 15:45:18 +02001156 unsigned int index = (this+input[0]) (c->buffer->cur().codepoint);
Behdad Esfahbodacea1832012-05-11 02:33:11 +02001157 if (likely (index == NOT_COVERED)) return TRACE_RETURN (false);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001158
Behdad Esfahbode961c862010-04-21 15:56:11 -04001159 const OffsetArrayOf<Coverage> &lookahead = StructAfter<OffsetArrayOf<Coverage> > (input);
1160 const ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord> > (lookahead);
Behdad Esfahbod31081f72012-04-23 16:54:58 -04001161 struct ChainContextApplyLookupContext lookup_context = {
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -04001162 {match_coverage, apply_func},
Behdad Esfahbod40cbefe2010-05-10 17:47:22 -04001163 {this, this, this}
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -04001164 };
Behdad Esfahbodacea1832012-05-11 02:33:11 +02001165 return TRACE_RETURN (chain_context_apply_lookup (c,
1166 backtrack.len, (const USHORT *) backtrack.array,
1167 input.len, (const USHORT *) input.array + 1,
1168 lookahead.len, (const USHORT *) lookahead.array,
1169 lookup.len, lookup.array, lookup_context));
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001170 }
1171
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -04001172 inline bool sanitize (hb_sanitize_context_t *c) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -04001173 TRACE_SANITIZE ();
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +02001174 if (!backtrack.sanitize (c, this)) return TRACE_RETURN (false);
Behdad Esfahbode961c862010-04-21 15:56:11 -04001175 OffsetArrayOf<Coverage> &input = StructAfter<OffsetArrayOf<Coverage> > (backtrack);
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +02001176 if (!input.sanitize (c, this)) return TRACE_RETURN (false);
Behdad Esfahbode961c862010-04-21 15:56:11 -04001177 OffsetArrayOf<Coverage> &lookahead = StructAfter<OffsetArrayOf<Coverage> > (input);
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +02001178 if (!lookahead.sanitize (c, this)) return TRACE_RETURN (false);
Behdad Esfahbode961c862010-04-21 15:56:11 -04001179 ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord> > (lookahead);
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +02001180 return TRACE_RETURN (lookup.sanitize (c));
Behdad Esfahbod70de50c2009-08-04 00:58:28 -04001181 }
1182
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001183 private:
1184 USHORT format; /* Format identifier--format = 3 */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -04001185 OffsetArrayOf<Coverage>
Behdad Esfahbod13ed4402009-05-18 02:14:37 -04001186 backtrack; /* Array of coverage tables
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001187 * in backtracking sequence, in glyph
1188 * sequence order */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -04001189 OffsetArrayOf<Coverage>
Behdad Esfahbod13ed4402009-05-18 02:14:37 -04001190 inputX ; /* Array of coverage
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001191 * tables in input sequence, in glyph
1192 * sequence order */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -04001193 OffsetArrayOf<Coverage>
Behdad Esfahbod13ed4402009-05-18 02:14:37 -04001194 lookaheadX; /* Array of coverage tables
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001195 * in lookahead sequence, in glyph
1196 * sequence order */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -04001197 ArrayOf<LookupRecord>
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -04001198 lookupX; /* Array of LookupRecords--in
Behdad Esfahboddcb6b602009-05-18 01:49:57 -04001199 * design order) */
Behdad Esfahbodb3651232010-05-10 16:57:29 -04001200 public:
Behdad Esfahbodbea34c72010-05-10 17:28:16 -04001201 DEFINE_SIZE_MIN (10);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001202};
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001203
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -04001204struct ChainContext
1205{
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001206 protected:
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -04001207
Behdad Esfahbod5caece62012-04-23 23:03:12 -04001208 inline void closure (hb_closure_context_t *c, closure_lookup_func_t closure_func) const
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -04001209 {
1210 TRACE_CLOSURE ();
1211 switch (u.format) {
Behdad Esfahbod5caece62012-04-23 23:03:12 -04001212 case 1: u.format1.closure (c, closure_func); break;
1213 case 2: u.format2.closure (c, closure_func); break;
1214 case 3: u.format3.closure (c, closure_func); break;
1215 default: break;
Behdad Esfahbodf94b0aa2012-04-23 13:04:38 -04001216 }
1217 }
1218
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -04001219 inline bool apply (hb_apply_context_t *c, apply_lookup_func_t apply_func) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -04001220 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -04001221 TRACE_APPLY ();
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001222 switch (u.format) {
Behdad Esfahbodacea1832012-05-11 02:33:11 +02001223 case 1: return TRACE_RETURN (u.format1.apply (c, apply_func));
1224 case 2: return TRACE_RETURN (u.format2.apply (c, apply_func));
1225 case 3: return TRACE_RETURN (u.format3.apply (c, apply_func));
1226 default:return TRACE_RETURN (false);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001227 }
1228 }
1229
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -04001230 inline bool sanitize (hb_sanitize_context_t *c) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -04001231 TRACE_SANITIZE ();
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +02001232 if (!u.format.sanitize (c)) return TRACE_RETURN (false);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -04001233 switch (u.format) {
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +02001234 case 1: return TRACE_RETURN (u.format1.sanitize (c));
1235 case 2: return TRACE_RETURN (u.format2.sanitize (c));
1236 case 3: return TRACE_RETURN (u.format3.sanitize (c));
1237 default:return TRACE_RETURN (true);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -04001238 }
1239 }
1240
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001241 private:
1242 union {
1243 USHORT format; /* Format identifier */
Behdad Esfahboddacebca2010-05-10 19:45:41 -04001244 ChainContextFormat1 format1;
1245 ChainContextFormat2 format2;
1246 ChainContextFormat3 format3;
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001247 } u;
1248};
Behdad Esfahbodca5290f2009-05-17 20:48:27 -04001249
1250
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -04001251struct ExtensionFormat1
1252{
1253 friend struct Extension;
1254
Behdad Esfahbod18939482009-08-04 14:27:56 -04001255 protected:
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -04001256 inline unsigned int get_type (void) const { return extensionLookupType; }
Behdad Esfahbod3b2c2df2010-04-22 16:51:42 -04001257 inline unsigned int get_offset (void) const { return extensionOffset; }
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -04001258
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -04001259 inline bool sanitize (hb_sanitize_context_t *c) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -04001260 TRACE_SANITIZE ();
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +02001261 return TRACE_RETURN (c->check_struct (this));
Behdad Esfahbod70de50c2009-08-04 00:58:28 -04001262 }
1263
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -04001264 private:
1265 USHORT format; /* Format identifier. Set to 1. */
1266 USHORT extensionLookupType; /* Lookup type of subtable referenced
1267 * by ExtensionOffset (i.e. the
1268 * extension subtable). */
Behdad Esfahbod81f2af42010-04-22 00:58:49 -04001269 ULONG extensionOffset; /* Offset to the extension subtable,
1270 * of lookup type subtable. */
Behdad Esfahbodb3651232010-05-10 16:57:29 -04001271 public:
1272 DEFINE_SIZE_STATIC (8);
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -04001273};
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -04001274
1275struct Extension
1276{
1277 inline unsigned int get_type (void) const
1278 {
1279 switch (u.format) {
Behdad Esfahboddacebca2010-05-10 19:45:41 -04001280 case 1: return u.format1.get_type ();
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -04001281 default:return 0;
1282 }
1283 }
Behdad Esfahbod3b2c2df2010-04-22 16:51:42 -04001284 inline unsigned int get_offset (void) const
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -04001285 {
1286 switch (u.format) {
Behdad Esfahboddacebca2010-05-10 19:45:41 -04001287 case 1: return u.format1.get_offset ();
Behdad Esfahbod3b2c2df2010-04-22 16:51:42 -04001288 default:return 0;
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -04001289 }
1290 }
1291
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -04001292 inline bool sanitize (hb_sanitize_context_t *c) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -04001293 TRACE_SANITIZE ();
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +02001294 if (!u.format.sanitize (c)) return TRACE_RETURN (false);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -04001295 switch (u.format) {
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +02001296 case 1: return TRACE_RETURN (u.format1.sanitize (c));
1297 default:return TRACE_RETURN (true);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -04001298 }
1299 }
1300
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -04001301 private:
1302 union {
1303 USHORT format; /* Format identifier */
Behdad Esfahboddacebca2010-05-10 19:45:41 -04001304 ExtensionFormat1 format1;
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -04001305 } u;
1306};
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -04001307
1308
Behdad Esfahbodf45107f2009-05-17 20:13:02 -04001309/*
1310 * GSUB/GPOS Common
1311 */
1312
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -04001313struct GSUBGPOS
1314{
Behdad Esfahboda328d662009-08-04 20:27:05 -04001315 static const hb_tag_t GSUBTag = HB_OT_TAG_GSUB;
1316 static const hb_tag_t GPOSTag = HB_OT_TAG_GPOS;
Behdad Esfahbodf45107f2009-05-17 20:13:02 -04001317
Behdad Esfahbodbff3c0f2009-08-07 19:46:30 -04001318 inline unsigned int get_script_count (void) const
1319 { return (this+scriptList).len; }
1320 inline const Tag& get_script_tag (unsigned int i) const
1321 { return (this+scriptList).get_tag (i); }
Behdad Esfahbode21899b2009-11-04 16:36:14 -05001322 inline unsigned int get_script_tags (unsigned int start_offset,
1323 unsigned int *script_count /* IN/OUT */,
1324 hb_tag_t *script_tags /* OUT */) const
1325 { return (this+scriptList).get_tags (start_offset, script_count, script_tags); }
Behdad Esfahbodbff3c0f2009-08-07 19:46:30 -04001326 inline const Script& get_script (unsigned int i) const
1327 { return (this+scriptList)[i]; }
1328 inline bool find_script_index (hb_tag_t tag, unsigned int *index) const
1329 { return (this+scriptList).find_index (tag, index); }
Behdad Esfahbodf45107f2009-05-17 20:13:02 -04001330
Behdad Esfahbodbff3c0f2009-08-07 19:46:30 -04001331 inline unsigned int get_feature_count (void) const
1332 { return (this+featureList).len; }
1333 inline const Tag& get_feature_tag (unsigned int i) const
1334 { return (this+featureList).get_tag (i); }
Behdad Esfahbode21899b2009-11-04 16:36:14 -05001335 inline unsigned int get_feature_tags (unsigned int start_offset,
1336 unsigned int *feature_count /* IN/OUT */,
1337 hb_tag_t *feature_tags /* OUT */) const
1338 { return (this+featureList).get_tags (start_offset, feature_count, feature_tags); }
Behdad Esfahbodbff3c0f2009-08-07 19:46:30 -04001339 inline const Feature& get_feature (unsigned int i) const
1340 { return (this+featureList)[i]; }
1341 inline bool find_feature_index (hb_tag_t tag, unsigned int *index) const
1342 { return (this+featureList).find_index (tag, index); }
1343
1344 inline unsigned int get_lookup_count (void) const
1345 { return (this+lookupList).len; }
1346 inline const Lookup& get_lookup (unsigned int i) const
1347 { return (this+lookupList)[i]; }
Behdad Esfahbodf45107f2009-05-17 20:13:02 -04001348
Behdad Esfahbodd7cfb3b2010-05-13 14:18:49 -04001349 inline bool sanitize (hb_sanitize_context_t *c) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -04001350 TRACE_SANITIZE ();
Behdad Esfahbod0ab8c862012-05-11 01:25:34 +02001351 return TRACE_RETURN (version.sanitize (c) && likely (version.major == 1) &&
1352 scriptList.sanitize (c, this) &&
1353 featureList.sanitize (c, this) &&
1354 lookupList.sanitize (c, this));
Behdad Esfahbodcd3827e2009-08-04 02:09:34 -04001355 }
1356
Behdad Esfahbod212aba62009-05-24 00:50:27 -04001357 protected:
Behdad Esfahbod87fcdcb2009-05-24 01:03:24 -04001358 FixedVersion version; /* Version of the GSUB/GPOS table--initially set
Behdad Esfahbodf45107f2009-05-17 20:13:02 -04001359 * to 0x00010000 */
1360 OffsetTo<ScriptList>
1361 scriptList; /* ScriptList table */
1362 OffsetTo<FeatureList>
1363 featureList; /* FeatureList table */
1364 OffsetTo<LookupList>
1365 lookupList; /* LookupList table */
Behdad Esfahbodb3651232010-05-10 16:57:29 -04001366 public:
1367 DEFINE_SIZE_STATIC (10);
Behdad Esfahbodf45107f2009-05-17 20:13:02 -04001368};
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -04001369
Behdad Esfahbod6f20f722009-05-17 20:28:01 -04001370
Behdad Esfahbodacdba3f2010-07-23 15:11:18 -04001371
Behdad Esfahbod5f5b24f2009-08-02 20:03:12 -04001372#endif /* HB_OT_LAYOUT_GSUBGPOS_PRIVATE_HH */