blob: e542481df82cf2e6a2504669d0664fba836ff20e [file] [log] [blame]
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -04001/*
2 * Copyright (C) 2007,2008,2009 Red Hat, Inc.
3 *
Behdad Esfahbodc755cb32010-04-22 00:11:43 -04004 * This is part of HarfBuzz, a text shaping library.
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -04005 *
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 * Red Hat Author(s): Behdad Esfahbod
25 */
26
Behdad Esfahbod5f5b24f2009-08-02 20:03:12 -040027#ifndef HB_OT_LAYOUT_GSUBGPOS_PRIVATE_HH
28#define HB_OT_LAYOUT_GSUBGPOS_PRIVATE_HH
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040029
Behdad Esfahbod88a5f5a2009-05-25 03:39:11 -040030#include "hb-buffer-private.h"
Behdad Esfahbod5f5b24f2009-08-02 20:03:12 -040031#include "hb-ot-layout-gdef-private.hh"
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040032
Behdad Esfahbod6f20f722009-05-17 20:28:01 -040033
Behdad Esfahbod0535b502009-08-28 17:14:33 -040034#ifndef HB_DEBUG_APPLY
Behdad Esfahbod74e313c2010-04-28 15:15:09 -040035#define HB_DEBUG_APPLY HB_DEBUG+0
Behdad Esfahbod0535b502009-08-28 17:14:33 -040036#endif
37
Behdad Esfahbodbc200452010-04-29 01:40:26 -040038#define TRACE_APPLY() \
39 HB_STMT_START { \
40 if (HB_DEBUG_APPLY) \
Behdad Esfahbod7d3a1262010-04-29 13:54:01 -040041 _hb_trace ("APPLY", HB_FUNC, this, apply_depth, HB_DEBUG_APPLY); \
Behdad Esfahbodbc200452010-04-29 01:40:26 -040042 } HB_STMT_END
Behdad Esfahbod74e313c2010-04-28 15:15:09 -040043
Behdad Esfahbod0535b502009-08-28 17:14:33 -040044
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -040045#define APPLY_ARG_DEF \
Behdad Esfahbod6617ead2010-04-29 02:25:30 -040046 hb_apply_context_t *context, \
Behdad Esfahbodb4c7fab2010-04-29 02:24:25 -040047 hb_ot_layout_context_t *layout_context, \
Behdad Esfahbod1376fb72010-04-29 02:19:21 -040048 hb_buffer_t *buffer, \
Behdad Esfahbod33d13fd2010-04-29 13:56:44 -040049 unsigned int context_length HB_UNUSED, \
50 unsigned int apply_depth HB_UNUSED
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -040051#define APPLY_ARG \
Behdad Esfahbod6617ead2010-04-29 02:25:30 -040052 context, \
Behdad Esfahbodb4c7fab2010-04-29 02:24:25 -040053 layout_context, \
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040054 buffer, \
55 context_length, \
Behdad Esfahbodbc200452010-04-29 01:40:26 -040056 (HB_DEBUG_APPLY ? apply_depth + 1 : 0)
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040057
Behdad Esfahbod1376fb72010-04-29 02:19:21 -040058struct hb_apply_context_t
59{
60 unsigned int nesting_level_left;
61 unsigned int lookup_flag;
62 unsigned int property; /* propety of first glyph (TODO remove) */
63};
64
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040065
Behdad Esfahbod1922ffe2010-04-21 04:19:51 -040066typedef bool (*match_func_t) (hb_codepoint_t glyph_id, const USHORT &value, const char *data);
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -040067typedef bool (*apply_lookup_func_t) (APPLY_ARG_DEF, unsigned int lookup_index);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040068
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -040069struct ContextFuncs
70{
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -040071 match_func_t match;
72 apply_lookup_func_t apply;
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040073};
74
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -040075
Behdad Esfahbod33d13fd2010-04-29 13:56:44 -040076static inline bool match_glyph (hb_codepoint_t glyph_id, const USHORT &value, const char *data HB_UNUSED)
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -040077{
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -040078 return glyph_id == value;
79}
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040080
Behdad Esfahbod1922ffe2010-04-21 04:19:51 -040081static inline bool match_class (hb_codepoint_t glyph_id, const USHORT &value, const char *data)
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -040082{
Behdad Esfahbod2b5a59c2009-08-04 11:38:50 -040083 const ClassDef &class_def = *reinterpret_cast<const ClassDef *>(data);
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -040084 return class_def.get_class (glyph_id) == value;
85}
86
Behdad Esfahbod1922ffe2010-04-21 04:19:51 -040087static inline bool match_coverage (hb_codepoint_t glyph_id, const USHORT &value, const char *data)
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -040088{
Behdad Esfahbod6b54c5d2009-05-18 18:30:25 -040089 const OffsetTo<Coverage> &coverage = (const OffsetTo<Coverage>&)value;
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -040090 return (data+coverage) (glyph_id) != NOT_COVERED;
91}
92
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040093
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -040094static inline bool match_input (APPLY_ARG_DEF,
Behdad Esfahbode072c242009-05-18 03:47:31 -040095 unsigned int count, /* Including the first glyph (not matched) */
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -040096 const USHORT input[], /* Array of input values--start with second glyph */
97 match_func_t match_func,
Behdad Esfahbod1922ffe2010-04-21 04:19:51 -040098 const char *match_data,
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -040099 unsigned int *context_length_out)
100{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400101 unsigned int i, j;
Behdad Esfahbod7cff75b2009-05-18 04:09:05 -0400102 unsigned int end = MIN (buffer->in_length, buffer->in_pos + context_length);
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400103 if (unlikely (buffer->in_pos + count > end))
Behdad Esfahbod7cff75b2009-05-18 04:09:05 -0400104 return false;
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400105
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400106 for (i = 1, j = buffer->in_pos + 1; i < count; i++, j++)
107 {
Behdad Esfahbod6617ead2010-04-29 02:25:30 -0400108 while (_hb_ot_layout_skip_mark (layout_context->face, IN_INFO (j), context->lookup_flag, NULL))
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400109 {
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400110 if (unlikely (j + count - i == end))
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400111 return false;
112 j++;
113 }
114
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400115 if (likely (!match_func (IN_GLYPH (j), input[i - 1], match_data)))
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400116 return false;
117 }
118
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400119 *context_length_out = j - buffer->in_pos;
120
121 return true;
122}
123
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400124static inline bool match_backtrack (APPLY_ARG_DEF,
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400125 unsigned int count,
126 const USHORT backtrack[],
127 match_func_t match_func,
Behdad Esfahbod1922ffe2010-04-21 04:19:51 -0400128 const char *match_data)
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400129{
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400130 if (unlikely (buffer->out_pos < count))
Behdad Esfahbod7cff75b2009-05-18 04:09:05 -0400131 return false;
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400132
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400133 for (unsigned int i = 0, j = buffer->out_pos - 1; i < count; i++, j--)
134 {
Behdad Esfahbod6617ead2010-04-29 02:25:30 -0400135 while (_hb_ot_layout_skip_mark (layout_context->face, OUT_INFO (j), context->lookup_flag, NULL))
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400136 {
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400137 if (unlikely (j + 1 == count - i))
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400138 return false;
139 j--;
140 }
141
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400142 if (likely (!match_func (OUT_GLYPH (j), backtrack[i], match_data)))
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400143 return false;
144 }
145
146 return true;
147}
148
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400149static inline bool match_lookahead (APPLY_ARG_DEF,
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400150 unsigned int count,
151 const USHORT lookahead[],
152 match_func_t match_func,
Behdad Esfahbod1922ffe2010-04-21 04:19:51 -0400153 const char *match_data,
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400154 unsigned int offset)
155{
156 unsigned int i, j;
Behdad Esfahbod7cff75b2009-05-18 04:09:05 -0400157 unsigned int end = MIN (buffer->in_length, buffer->in_pos + context_length);
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400158 if (unlikely (buffer->in_pos + offset + count > end))
Behdad Esfahbod7cff75b2009-05-18 04:09:05 -0400159 return false;
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400160
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400161 for (i = 0, j = buffer->in_pos + offset; i < count; i++, j++)
162 {
Behdad Esfahbod6617ead2010-04-29 02:25:30 -0400163 while (_hb_ot_layout_skip_mark (layout_context->face, OUT_INFO (j), context->lookup_flag, NULL))
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400164 {
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400165 if (unlikely (j + count - i == end))
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400166 return false;
167 j++;
168 }
169
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400170 if (likely (!match_func (IN_GLYPH (j), lookahead[i], match_data)))
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400171 return false;
172 }
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400173
174 return true;
175}
176
177
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400178struct LookupRecord
179{
Behdad Esfahbod4b8487d2010-03-16 03:46:17 -0400180 static inline unsigned int get_size () { return sizeof (LookupRecord); }
181
Behdad Esfahbod39840472010-05-05 00:23:19 -0400182 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400183 TRACE_SANITIZE ();
Behdad Esfahbodcd3827e2009-08-04 02:09:34 -0400184 return SANITIZE_SELF ();
185 }
186
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400187 USHORT sequenceIndex; /* Index into current glyph
188 * sequence--first glyph = 0 */
189 USHORT lookupListIndex; /* Lookup to apply to that
190 * position--zero--based */
191};
192ASSERT_SIZE (LookupRecord, 4);
193
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400194static inline bool apply_lookup (APPLY_ARG_DEF,
Behdad Esfahbode072c242009-05-18 03:47:31 -0400195 unsigned int count, /* Including the first glyph */
196 unsigned int lookupCount,
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400197 const LookupRecord lookupRecord[], /* Array of LookupRecords--in design order */
198 apply_lookup_func_t apply_func)
199{
Behdad Esfahbode73a0c22009-05-18 04:15:25 -0400200 unsigned int end = MIN (buffer->in_length, buffer->in_pos + context_length);
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400201 if (unlikely (buffer->in_pos + count > end))
Behdad Esfahbode73a0c22009-05-18 04:15:25 -0400202 return false;
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400203
Behdad Esfahbod7cff75b2009-05-18 04:09:05 -0400204 /* TODO We don't support lookupRecord arrays that are not increasing:
205 * Should be easy for in_place ones at least. */
Behdad Esfahbod52e9a712009-09-21 13:58:56 -0400206
207 /* Note: If sublookup is reverse, i will underflow after the first loop
208 * and we jump out of it. Not entirely disastrous. So we don't check
209 * for reverse lookup here.
210 */
Behdad Esfahbodf9c0a2d2009-09-21 13:43:54 -0400211 for (unsigned int i = 0; i < count; /* NOP */)
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400212 {
Behdad Esfahbod6617ead2010-04-29 02:25:30 -0400213 while (_hb_ot_layout_skip_mark (layout_context->face, IN_CURINFO (), context->lookup_flag, NULL))
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400214 {
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400215 if (unlikely (buffer->in_pos == end))
Behdad Esfahbode73a0c22009-05-18 04:15:25 -0400216 return true;
217 /* No lookup applied for this index */
218 _hb_buffer_next_glyph (buffer);
219 }
220
Behdad Esfahbod47958de2009-05-18 04:17:47 -0400221 if (lookupCount && i == lookupRecord->sequenceIndex)
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400222 {
223 unsigned int old_pos = buffer->in_pos;
224
225 /* Apply a lookup */
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400226 bool done = apply_func (APPLY_ARG, lookupRecord->lookupListIndex);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400227
Behdad Esfahbod47958de2009-05-18 04:17:47 -0400228 lookupRecord++;
229 lookupCount--;
Behdad Esfahbodf9c0a2d2009-09-21 13:43:54 -0400230 /* Err, this is wrong if the lookup jumped over some glyphs */
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400231 i += buffer->in_pos - old_pos;
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400232 if (unlikely (buffer->in_pos == end))
Behdad Esfahbod2e8fb6c2009-05-18 04:37:37 -0400233 return true;
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400234
235 if (!done)
236 goto not_applied;
237 }
238 else
239 {
240 not_applied:
241 /* No lookup applied for this index */
242 _hb_buffer_next_glyph (buffer);
243 i++;
244 }
245 }
246
247 return true;
248}
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400249
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400250
251/* Contextual lookups */
252
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400253struct ContextLookupContext
254{
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400255 ContextFuncs funcs;
Behdad Esfahbod1922ffe2010-04-21 04:19:51 -0400256 const char *match_data;
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400257};
258
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400259static inline bool context_lookup (APPLY_ARG_DEF,
Behdad Esfahbode072c242009-05-18 03:47:31 -0400260 unsigned int inputCount, /* Including the first glyph (not matched) */
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400261 const USHORT input[], /* Array of input values--start with second glyph */
Behdad Esfahbode072c242009-05-18 03:47:31 -0400262 unsigned int lookupCount,
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400263 const LookupRecord lookupRecord[],
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400264 ContextLookupContext &lookup_context)
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400265{
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400266 return match_input (APPLY_ARG,
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400267 inputCount, input,
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400268 lookup_context.funcs.match, lookup_context.match_data,
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400269 &context_length) &&
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400270 apply_lookup (APPLY_ARG,
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400271 inputCount,
272 lookupCount, lookupRecord,
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400273 lookup_context.funcs.apply);
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400274}
275
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400276struct Rule
277{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400278 friend struct RuleSet;
279
280 private:
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400281 inline bool apply (APPLY_ARG_DEF, ContextLookupContext &lookup_context) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400282 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400283 TRACE_APPLY ();
Behdad Esfahbodfff9aa22010-05-05 00:32:21 -0400284 const LookupRecord *lookupRecord = &StructAtOffset<LookupRecord> (input, input[0].get_size () * (inputCount ? inputCount - 1 : 0));
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400285 return context_lookup (APPLY_ARG,
Behdad Esfahbod969afd72009-05-18 05:47:47 -0400286 inputCount, input,
Behdad Esfahbodfff9aa22010-05-05 00:32:21 -0400287 lookupCount, lookupRecord,
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400288 lookup_context);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400289 }
290
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400291 public:
Behdad Esfahbod39840472010-05-05 00:23:19 -0400292 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400293 TRACE_SANITIZE ();
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500294 if (!(SANITIZE (inputCount) && SANITIZE (lookupCount))) return false;
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400295 return SANITIZE_MEM (input,
Behdad Esfahbod4b8487d2010-03-16 03:46:17 -0400296 input[0].get_size () * inputCount +
297 lookupRecordX[0].get_size () * lookupCount);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400298 }
299
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400300 private:
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400301 USHORT inputCount; /* Total number of glyphs in input
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400302 * glyph sequence--includes the first
303 * glyph */
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400304 USHORT lookupCount; /* Number of LookupRecords */
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500305 USHORT input[VAR]; /* Array of match inputs--start with
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400306 * second glyph */
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500307 LookupRecord lookupRecordX[VAR]; /* Array of LookupRecords--in
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400308 * design order */
309};
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500310ASSERT_SIZE_VAR2 (Rule, 4, USHORT, LookupRecord);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400311
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400312struct RuleSet
313{
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400314 inline bool apply (APPLY_ARG_DEF, ContextLookupContext &lookup_context) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400315 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400316 TRACE_APPLY ();
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400317 unsigned int num_rules = rule.len;
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400318 for (unsigned int i = 0; i < num_rules; i++)
319 {
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400320 if ((this+rule[i]).apply (APPLY_ARG, lookup_context))
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400321 return true;
322 }
323
324 return false;
325 }
326
Behdad Esfahbod39840472010-05-05 00:23:19 -0400327 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400328 TRACE_SANITIZE ();
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400329 return SANITIZE_WITH_BASE (this, rule);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400330 }
331
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400332 private:
333 OffsetArrayOf<Rule>
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400334 rule; /* Array of Rule tables
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400335 * ordered by preference */
336};
337
338
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400339struct ContextFormat1
340{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400341 friend struct Context;
342
343 private:
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400344 inline bool apply (APPLY_ARG_DEF, apply_lookup_func_t apply_func) const
345 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400346 TRACE_APPLY ();
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400347 unsigned int index = (this+coverage) (IN_CURGLYPH ());
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400348 if (likely (index == NOT_COVERED))
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400349 return false;
350
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400351 const RuleSet &rule_set = this+ruleSet[index];
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400352 struct ContextLookupContext lookup_context = {
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400353 {match_glyph, apply_func},
354 NULL
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400355 };
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400356 return rule_set.apply (APPLY_ARG, lookup_context);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400357 }
358
Behdad Esfahbod39840472010-05-05 00:23:19 -0400359 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400360 TRACE_SANITIZE ();
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400361 return SANITIZE_WITH_BASE (this, coverage)
362 && SANITIZE_WITH_BASE (this, ruleSet);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400363 }
364
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400365 private:
366 USHORT format; /* Format identifier--format = 1 */
367 OffsetTo<Coverage>
368 coverage; /* Offset to Coverage table--from
369 * beginning of table */
370 OffsetArrayOf<RuleSet>
371 ruleSet; /* Array of RuleSet tables
372 * ordered by Coverage Index */
373};
374ASSERT_SIZE (ContextFormat1, 6);
375
376
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400377struct ContextFormat2
378{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400379 friend struct Context;
380
381 private:
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400382 inline bool apply (APPLY_ARG_DEF, apply_lookup_func_t apply_func) const
383 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400384 TRACE_APPLY ();
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400385 unsigned int index = (this+coverage) (IN_CURGLYPH ());
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400386 if (likely (index == NOT_COVERED))
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400387 return false;
388
389 const ClassDef &class_def = this+classDef;
390 index = class_def (IN_CURGLYPH ());
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400391 const RuleSet &rule_set = this+ruleSet[index];
392 /* LONGTERMTODO: Old code fetches glyph classes at most once and caches
393 * them across subrule lookups. Not sure it's worth it.
394 */
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400395 struct ContextLookupContext lookup_context = {
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400396 {match_class, apply_func},
Behdad Esfahbod0dfcc132010-04-21 23:41:26 -0400397 CharP(&class_def)
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400398 };
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400399 return rule_set.apply (APPLY_ARG, lookup_context);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400400 }
401
Behdad Esfahbod39840472010-05-05 00:23:19 -0400402 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400403 TRACE_SANITIZE ();
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400404 return SANITIZE_WITH_BASE (this, coverage)
405 && SANITIZE_WITH_BASE (this, classDef)
406 && SANITIZE_WITH_BASE (this, ruleSet);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400407 }
408
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400409 private:
410 USHORT format; /* Format identifier--format = 2 */
411 OffsetTo<Coverage>
412 coverage; /* Offset to Coverage table--from
413 * beginning of table */
414 OffsetTo<ClassDef>
415 classDef; /* Offset to glyph ClassDef table--from
416 * beginning of table */
417 OffsetArrayOf<RuleSet>
418 ruleSet; /* Array of RuleSet tables
419 * ordered by class */
420};
421ASSERT_SIZE (ContextFormat2, 8);
422
423
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400424struct ContextFormat3
425{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400426 friend struct Context;
427
428 private:
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400429 inline bool apply (APPLY_ARG_DEF, apply_lookup_func_t apply_func) const
430 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400431 TRACE_APPLY ();
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400432 unsigned int index = (this+coverage[0]) (IN_CURGLYPH ());
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400433 if (likely (index == NOT_COVERED))
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400434 return false;
435
Behdad Esfahbodfff9aa22010-05-05 00:32:21 -0400436 const LookupRecord *lookupRecord = &StructAtOffset<LookupRecord> (coverage, coverage[0].get_size () * glyphCount);
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400437 struct ContextLookupContext lookup_context = {
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400438 {match_coverage, apply_func},
Behdad Esfahbod0dfcc132010-04-21 23:41:26 -0400439 CharP(this)
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400440 };
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400441 return context_lookup (APPLY_ARG,
Behdad Esfahbod969afd72009-05-18 05:47:47 -0400442 glyphCount, (const USHORT *) (coverage + 1),
Behdad Esfahbodfff9aa22010-05-05 00:32:21 -0400443 lookupCount, lookupRecord,
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400444 lookup_context);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400445 }
446
Behdad Esfahbod39840472010-05-05 00:23:19 -0400447 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400448 TRACE_SANITIZE ();
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400449 if (!SANITIZE_SELF ()) return false;
450 unsigned int count = glyphCount;
Behdad Esfahbod9fc86842010-04-22 13:37:58 -0400451 if (!SANITIZE_ARRAY (coverage, OffsetTo<Coverage>::get_size (), glyphCount)) return false;
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400452 for (unsigned int i = 0; i < count; i++)
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400453 if (!SANITIZE_WITH_BASE (this, coverage[i])) return false;
Behdad Esfahbodfff9aa22010-05-05 00:32:21 -0400454 LookupRecord *lookupRecord = &StructAtOffset<LookupRecord> (coverage, OffsetTo<Coverage>::get_size () * glyphCount);
455 return SANITIZE_ARRAY (lookupRecord, LookupRecord::get_size (), lookupCount);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400456 }
457
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400458 private:
459 USHORT format; /* Format identifier--format = 3 */
460 USHORT glyphCount; /* Number of glyphs in the input glyph
461 * sequence */
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400462 USHORT lookupCount; /* Number of LookupRecords */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400463 OffsetTo<Coverage>
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500464 coverage[VAR]; /* Array of offsets to Coverage
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400465 * table in glyph sequence order */
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500466 LookupRecord lookupRecordX[VAR]; /* Array of LookupRecords--in
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400467 * design order */
468};
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500469ASSERT_SIZE_VAR2 (ContextFormat3, 6, OffsetTo<Coverage>, LookupRecord);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400470
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400471struct Context
472{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400473 protected:
Behdad Esfahbod20b035d2009-08-10 19:00:36 -0400474 inline bool apply (APPLY_ARG_DEF, apply_lookup_func_t apply_func) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400475 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400476 TRACE_APPLY ();
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400477 switch (u.format) {
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400478 case 1: return u.format1->apply (APPLY_ARG, apply_func);
479 case 2: return u.format2->apply (APPLY_ARG, apply_func);
480 case 3: return u.format3->apply (APPLY_ARG, apply_func);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400481 default:return false;
482 }
483 }
484
Behdad Esfahbod39840472010-05-05 00:23:19 -0400485 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400486 TRACE_SANITIZE ();
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400487 if (!SANITIZE (u.format)) return false;
488 switch (u.format) {
Behdad Esfahbod39840472010-05-05 00:23:19 -0400489 case 1: return u.format1->sanitize (context);
490 case 2: return u.format2->sanitize (context);
491 case 3: return u.format3->sanitize (context);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400492 default:return true;
493 }
494 }
495
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400496 private:
497 union {
Behdad Esfahbodf8dc67b2009-05-17 19:47:54 -0400498 USHORT format; /* Format identifier */
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500499 ContextFormat1 format1[VAR];
500 ContextFormat2 format2[VAR];
501 ContextFormat3 format3[VAR];
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400502 } u;
503};
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400504
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400505
506/* Chaining Contextual lookups */
507
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400508struct ChainContextLookupContext
509{
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400510 ContextFuncs funcs;
Behdad Esfahbod1922ffe2010-04-21 04:19:51 -0400511 const char *match_data[3];
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400512};
513
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400514static inline bool chain_context_lookup (APPLY_ARG_DEF,
Behdad Esfahbode072c242009-05-18 03:47:31 -0400515 unsigned int backtrackCount,
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400516 const USHORT backtrack[],
Behdad Esfahbode072c242009-05-18 03:47:31 -0400517 unsigned int inputCount, /* Including the first glyph (not matched) */
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400518 const USHORT input[], /* Array of input values--start with second glyph */
Behdad Esfahbode072c242009-05-18 03:47:31 -0400519 unsigned int lookaheadCount,
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400520 const USHORT lookahead[],
Behdad Esfahbode072c242009-05-18 03:47:31 -0400521 unsigned int lookupCount,
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400522 const LookupRecord lookupRecord[],
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400523 ChainContextLookupContext &lookup_context)
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400524{
Behdad Esfahbode072c242009-05-18 03:47:31 -0400525 /* First guess */
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400526 if (unlikely (buffer->out_pos < backtrackCount ||
Behdad Esfahbod122f21f2009-05-18 04:21:53 -0400527 buffer->in_pos + inputCount + lookaheadCount > buffer->in_length ||
528 inputCount + lookaheadCount > context_length))
Behdad Esfahbode072c242009-05-18 03:47:31 -0400529 return false;
530
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400531 unsigned int offset;
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400532 return match_backtrack (APPLY_ARG,
Behdad Esfahbode072c242009-05-18 03:47:31 -0400533 backtrackCount, backtrack,
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400534 lookup_context.funcs.match, lookup_context.match_data[0]) &&
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400535 match_input (APPLY_ARG,
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400536 inputCount, input,
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400537 lookup_context.funcs.match, lookup_context.match_data[1],
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400538 &offset) &&
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400539 match_lookahead (APPLY_ARG,
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400540 lookaheadCount, lookahead,
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400541 lookup_context.funcs.match, lookup_context.match_data[2],
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400542 offset) &&
543 (context_length = offset, true) &&
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400544 apply_lookup (APPLY_ARG,
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400545 inputCount,
546 lookupCount, lookupRecord,
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400547 lookup_context.funcs.apply);
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400548}
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400549
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400550struct ChainRule
551{
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400552 friend struct ChainRuleSet;
553
554 private:
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400555 inline bool apply (APPLY_ARG_DEF, ChainContextLookupContext &lookup_context) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400556 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400557 TRACE_APPLY ();
Behdad Esfahbode961c862010-04-21 15:56:11 -0400558 const HeadlessArrayOf<USHORT> &input = StructAfter<HeadlessArrayOf<USHORT> > (backtrack);
559 const ArrayOf<USHORT> &lookahead = StructAfter<ArrayOf<USHORT> > (input);
560 const ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord> > (lookahead);
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400561 return chain_context_lookup (APPLY_ARG,
Behdad Esfahbod2cb08452010-04-21 22:37:31 -0400562 backtrack.len, backtrack.array(),
563 input.len, input.array(),
564 lookahead.len, lookahead.array(),
565 lookup.len, lookup.array(),
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400566 lookup_context);
Behdad Esfahbod13ed4402009-05-18 02:14:37 -0400567 return false;
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400568 }
569
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400570 public:
Behdad Esfahbod39840472010-05-05 00:23:19 -0400571 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400572 TRACE_SANITIZE ();
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400573 if (!SANITIZE (backtrack)) return false;
Behdad Esfahbode961c862010-04-21 15:56:11 -0400574 HeadlessArrayOf<USHORT> &input = StructAfter<HeadlessArrayOf<USHORT> > (backtrack);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400575 if (!SANITIZE (input)) return false;
Behdad Esfahbode961c862010-04-21 15:56:11 -0400576 ArrayOf<USHORT> &lookahead = StructAfter<ArrayOf<USHORT> > (input);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400577 if (!SANITIZE (lookahead)) return false;
Behdad Esfahbode961c862010-04-21 15:56:11 -0400578 ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord> > (lookahead);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400579 return SANITIZE (lookup);
580 }
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400581
582 private:
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400583 ArrayOf<USHORT>
584 backtrack; /* Array of backtracking values
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400585 * (to be matched before the input
586 * sequence) */
Behdad Esfahbode8cbaaf2009-05-18 02:03:58 -0400587 HeadlessArrayOf<USHORT>
588 inputX; /* Array of input values (start with
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400589 * second glyph) */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400590 ArrayOf<USHORT>
591 lookaheadX; /* Array of lookahead values's (to be
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400592 * matched after the input sequence) */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400593 ArrayOf<LookupRecord>
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400594 lookupX; /* Array of LookupRecords--in
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400595 * design order) */
596};
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400597ASSERT_SIZE (ChainRule, 8);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400598
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400599struct ChainRuleSet
600{
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400601 inline bool apply (APPLY_ARG_DEF, ChainContextLookupContext &lookup_context) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400602 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400603 TRACE_APPLY ();
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400604 unsigned int num_rules = rule.len;
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400605 for (unsigned int i = 0; i < num_rules; i++)
606 {
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400607 if ((this+rule[i]).apply (APPLY_ARG, lookup_context))
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400608 return true;
609 }
610
611 return false;
612 }
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400613
Behdad Esfahbod39840472010-05-05 00:23:19 -0400614 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400615 TRACE_SANITIZE ();
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400616 return SANITIZE_WITH_BASE (this, rule);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400617 }
618
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400619 private:
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400620 OffsetArrayOf<ChainRule>
621 rule; /* Array of ChainRule tables
622 * ordered by preference */
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400623};
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400624ASSERT_SIZE (ChainRuleSet, 2);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400625
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400626struct ChainContextFormat1
627{
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400628 friend struct ChainContext;
629
630 private:
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400631 inline bool apply (APPLY_ARG_DEF, apply_lookup_func_t apply_func) const
632 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400633 TRACE_APPLY ();
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400634 unsigned int index = (this+coverage) (IN_CURGLYPH ());
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400635 if (likely (index == NOT_COVERED))
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400636 return false;
637
638 const ChainRuleSet &rule_set = this+ruleSet[index];
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400639 struct ChainContextLookupContext lookup_context = {
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400640 {match_glyph, apply_func},
641 {NULL, NULL, NULL}
642 };
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400643 return rule_set.apply (APPLY_ARG, lookup_context);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400644 }
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400645
Behdad Esfahbod39840472010-05-05 00:23:19 -0400646 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400647 TRACE_SANITIZE ();
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400648 return SANITIZE_WITH_BASE (this, coverage)
649 && SANITIZE_WITH_BASE (this, ruleSet);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400650 }
651
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400652 private:
653 USHORT format; /* Format identifier--format = 1 */
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400654 OffsetTo<Coverage>
655 coverage; /* Offset to Coverage table--from
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400656 * beginning of table */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400657 OffsetArrayOf<ChainRuleSet>
658 ruleSet; /* Array of ChainRuleSet tables
659 * ordered by Coverage Index */
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400660};
661ASSERT_SIZE (ChainContextFormat1, 6);
662
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400663struct ChainContextFormat2
664{
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400665 friend struct ChainContext;
666
667 private:
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400668 inline bool apply (APPLY_ARG_DEF, apply_lookup_func_t apply_func) const
669 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400670 TRACE_APPLY ();
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400671 unsigned int index = (this+coverage) (IN_CURGLYPH ());
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400672 if (likely (index == NOT_COVERED))
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400673 return false;
674
675 const ClassDef &backtrack_class_def = this+backtrackClassDef;
676 const ClassDef &input_class_def = this+inputClassDef;
677 const ClassDef &lookahead_class_def = this+lookaheadClassDef;
678
679 index = input_class_def (IN_CURGLYPH ());
680 const ChainRuleSet &rule_set = this+ruleSet[index];
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400681 /* LONGTERMTODO: Old code fetches glyph classes at most once and caches
682 * them across subrule lookups. Not sure it's worth it.
683 */
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400684 struct ChainContextLookupContext lookup_context = {
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400685 {match_class, apply_func},
Behdad Esfahbod0dfcc132010-04-21 23:41:26 -0400686 {CharP(&backtrack_class_def),
687 CharP(&input_class_def),
688 CharP(&lookahead_class_def)}
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400689 };
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400690 return rule_set.apply (APPLY_ARG, lookup_context);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400691 }
692
Behdad Esfahbod39840472010-05-05 00:23:19 -0400693 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400694 TRACE_SANITIZE ();
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400695 return SANITIZE_WITH_BASE (this, coverage)
696 && SANITIZE_WITH_BASE (this, backtrackClassDef)
697 && SANITIZE_WITH_BASE (this, inputClassDef)
698 && SANITIZE_WITH_BASE (this, lookaheadClassDef)
699 && SANITIZE_WITH_BASE (this, ruleSet);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400700 }
701
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400702 private:
703 USHORT format; /* Format identifier--format = 2 */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400704 OffsetTo<Coverage>
705 coverage; /* Offset to Coverage table--from
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400706 * beginning of table */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400707 OffsetTo<ClassDef>
708 backtrackClassDef; /* Offset to glyph ClassDef table
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400709 * containing backtrack sequence
710 * data--from beginning of table */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400711 OffsetTo<ClassDef>
712 inputClassDef; /* Offset to glyph ClassDef
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400713 * table containing input sequence
714 * data--from beginning of table */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400715 OffsetTo<ClassDef>
716 lookaheadClassDef; /* Offset to glyph ClassDef table
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400717 * containing lookahead sequence
718 * data--from beginning of table */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400719 OffsetArrayOf<ChainRuleSet>
720 ruleSet; /* Array of ChainRuleSet tables
721 * ordered by class */
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400722};
723ASSERT_SIZE (ChainContextFormat2, 12);
724
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400725struct ChainContextFormat3
726{
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400727 friend struct ChainContext;
728
729 private:
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400730
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400731 inline bool apply (APPLY_ARG_DEF, apply_lookup_func_t apply_func) const
732 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400733 TRACE_APPLY ();
Behdad Esfahbode961c862010-04-21 15:56:11 -0400734 const OffsetArrayOf<Coverage> &input = StructAfter<OffsetArrayOf<Coverage> > (backtrack);
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400735
736 unsigned int index = (this+input[0]) (IN_CURGLYPH ());
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400737 if (likely (index == NOT_COVERED))
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400738 return false;
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400739
Behdad Esfahbode961c862010-04-21 15:56:11 -0400740 const OffsetArrayOf<Coverage> &lookahead = StructAfter<OffsetArrayOf<Coverage> > (input);
741 const ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord> > (lookahead);
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400742 struct ChainContextLookupContext lookup_context = {
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400743 {match_coverage, apply_func},
Behdad Esfahbod0dfcc132010-04-21 23:41:26 -0400744 {CharP(this), CharP(this), CharP(this)}
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400745 };
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400746 return chain_context_lookup (APPLY_ARG,
Behdad Esfahbod2cb08452010-04-21 22:37:31 -0400747 backtrack.len, (const USHORT *) backtrack.array(),
748 input.len, (const USHORT *) input.array() + 1,
749 lookahead.len, (const USHORT *) lookahead.array(),
750 lookup.len, lookup.array(),
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400751 lookup_context);
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400752 return false;
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400753 }
754
Behdad Esfahbod39840472010-05-05 00:23:19 -0400755 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400756 TRACE_SANITIZE ();
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400757 if (!SANITIZE_WITH_BASE (this, backtrack)) return false;
Behdad Esfahbode961c862010-04-21 15:56:11 -0400758 OffsetArrayOf<Coverage> &input = StructAfter<OffsetArrayOf<Coverage> > (backtrack);
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400759 if (!SANITIZE_WITH_BASE (this, input)) return false;
Behdad Esfahbode961c862010-04-21 15:56:11 -0400760 OffsetArrayOf<Coverage> &lookahead = StructAfter<OffsetArrayOf<Coverage> > (input);
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400761 if (!SANITIZE_WITH_BASE (this, lookahead)) return false;
Behdad Esfahbode961c862010-04-21 15:56:11 -0400762 ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord> > (lookahead);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400763 return SANITIZE (lookup);
764 }
765
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400766 private:
767 USHORT format; /* Format identifier--format = 3 */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400768 OffsetArrayOf<Coverage>
Behdad Esfahbod13ed4402009-05-18 02:14:37 -0400769 backtrack; /* Array of coverage tables
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400770 * in backtracking sequence, in glyph
771 * sequence order */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400772 OffsetArrayOf<Coverage>
Behdad Esfahbod13ed4402009-05-18 02:14:37 -0400773 inputX ; /* Array of coverage
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400774 * tables in input sequence, in glyph
775 * sequence order */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400776 OffsetArrayOf<Coverage>
Behdad Esfahbod13ed4402009-05-18 02:14:37 -0400777 lookaheadX; /* Array of coverage tables
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400778 * in lookahead sequence, in glyph
779 * sequence order */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400780 ArrayOf<LookupRecord>
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400781 lookupX; /* Array of LookupRecords--in
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400782 * design order) */
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400783};
784ASSERT_SIZE (ChainContextFormat3, 10);
785
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400786struct ChainContext
787{
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400788 protected:
Behdad Esfahbod20b035d2009-08-10 19:00:36 -0400789 inline bool apply (APPLY_ARG_DEF, apply_lookup_func_t apply_func) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400790 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400791 TRACE_APPLY ();
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400792 switch (u.format) {
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400793 case 1: return u.format1->apply (APPLY_ARG, apply_func);
794 case 2: return u.format2->apply (APPLY_ARG, apply_func);
795 case 3: return u.format3->apply (APPLY_ARG, apply_func);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400796 default:return false;
797 }
798 }
799
Behdad Esfahbod39840472010-05-05 00:23:19 -0400800 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400801 TRACE_SANITIZE ();
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400802 if (!SANITIZE (u.format)) return false;
803 switch (u.format) {
Behdad Esfahbod39840472010-05-05 00:23:19 -0400804 case 1: return u.format1->sanitize (context);
805 case 2: return u.format2->sanitize (context);
806 case 3: return u.format3->sanitize (context);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400807 default:return true;
808 }
809 }
810
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400811 private:
812 union {
813 USHORT format; /* Format identifier */
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500814 ChainContextFormat1 format1[VAR];
815 ChainContextFormat2 format2[VAR];
816 ChainContextFormat3 format3[VAR];
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400817 } u;
818};
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400819
820
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400821struct ExtensionFormat1
822{
823 friend struct Extension;
824
Behdad Esfahbod18939482009-08-04 14:27:56 -0400825 protected:
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400826 inline unsigned int get_type (void) const { return extensionLookupType; }
Behdad Esfahbod3b2c2df2010-04-22 16:51:42 -0400827 inline unsigned int get_offset (void) const { return extensionOffset; }
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400828
Behdad Esfahbod39840472010-05-05 00:23:19 -0400829 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400830 TRACE_SANITIZE ();
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400831 return SANITIZE_SELF ();
832 }
833
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400834 private:
835 USHORT format; /* Format identifier. Set to 1. */
836 USHORT extensionLookupType; /* Lookup type of subtable referenced
837 * by ExtensionOffset (i.e. the
838 * extension subtable). */
Behdad Esfahbod81f2af42010-04-22 00:58:49 -0400839 ULONG extensionOffset; /* Offset to the extension subtable,
840 * of lookup type subtable. */
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400841};
842ASSERT_SIZE (ExtensionFormat1, 8);
843
844struct Extension
845{
846 inline unsigned int get_type (void) const
847 {
848 switch (u.format) {
849 case 1: return u.format1->get_type ();
850 default:return 0;
851 }
852 }
Behdad Esfahbod3b2c2df2010-04-22 16:51:42 -0400853 inline unsigned int get_offset (void) const
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400854 {
855 switch (u.format) {
Behdad Esfahbod3b2c2df2010-04-22 16:51:42 -0400856 case 1: return u.format1->get_offset ();
857 default:return 0;
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400858 }
859 }
860
Behdad Esfahbod39840472010-05-05 00:23:19 -0400861 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400862 TRACE_SANITIZE ();
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400863 if (!SANITIZE (u.format)) return false;
864 switch (u.format) {
Behdad Esfahbod39840472010-05-05 00:23:19 -0400865 case 1: return u.format1->sanitize (context);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400866 default:return true;
867 }
868 }
869
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400870 private:
871 union {
872 USHORT format; /* Format identifier */
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500873 ExtensionFormat1 format1[VAR];
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400874 } u;
875};
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400876
877
Behdad Esfahbodf45107f2009-05-17 20:13:02 -0400878/*
879 * GSUB/GPOS Common
880 */
881
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400882struct GSUBGPOS
883{
Behdad Esfahboda328d662009-08-04 20:27:05 -0400884 static const hb_tag_t GSUBTag = HB_OT_TAG_GSUB;
885 static const hb_tag_t GPOSTag = HB_OT_TAG_GPOS;
Behdad Esfahbodf45107f2009-05-17 20:13:02 -0400886
Behdad Esfahbodbff3c0f2009-08-07 19:46:30 -0400887 inline unsigned int get_script_count (void) const
888 { return (this+scriptList).len; }
889 inline const Tag& get_script_tag (unsigned int i) const
890 { return (this+scriptList).get_tag (i); }
Behdad Esfahbode21899b2009-11-04 16:36:14 -0500891 inline unsigned int get_script_tags (unsigned int start_offset,
892 unsigned int *script_count /* IN/OUT */,
893 hb_tag_t *script_tags /* OUT */) const
894 { return (this+scriptList).get_tags (start_offset, script_count, script_tags); }
Behdad Esfahbodbff3c0f2009-08-07 19:46:30 -0400895 inline const Script& get_script (unsigned int i) const
896 { return (this+scriptList)[i]; }
897 inline bool find_script_index (hb_tag_t tag, unsigned int *index) const
898 { return (this+scriptList).find_index (tag, index); }
Behdad Esfahbodf45107f2009-05-17 20:13:02 -0400899
Behdad Esfahbodbff3c0f2009-08-07 19:46:30 -0400900 inline unsigned int get_feature_count (void) const
901 { return (this+featureList).len; }
902 inline const Tag& get_feature_tag (unsigned int i) const
903 { return (this+featureList).get_tag (i); }
Behdad Esfahbode21899b2009-11-04 16:36:14 -0500904 inline unsigned int get_feature_tags (unsigned int start_offset,
905 unsigned int *feature_count /* IN/OUT */,
906 hb_tag_t *feature_tags /* OUT */) const
907 { return (this+featureList).get_tags (start_offset, feature_count, feature_tags); }
Behdad Esfahbodbff3c0f2009-08-07 19:46:30 -0400908 inline const Feature& get_feature (unsigned int i) const
909 { return (this+featureList)[i]; }
910 inline bool find_feature_index (hb_tag_t tag, unsigned int *index) const
911 { return (this+featureList).find_index (tag, index); }
912
913 inline unsigned int get_lookup_count (void) const
914 { return (this+lookupList).len; }
915 inline const Lookup& get_lookup (unsigned int i) const
916 { return (this+lookupList)[i]; }
Behdad Esfahbodf45107f2009-05-17 20:13:02 -0400917
Behdad Esfahbod39840472010-05-05 00:23:19 -0400918 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400919 TRACE_SANITIZE ();
Behdad Esfahbod26bfcb62010-05-04 14:49:45 -0400920 return SANITIZE (version) && likely (version.major == 1)
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400921 && SANITIZE_WITH_BASE (this, scriptList)
922 && SANITIZE_WITH_BASE (this, featureList)
923 && SANITIZE_WITH_BASE (this, lookupList);
Behdad Esfahbodcd3827e2009-08-04 02:09:34 -0400924 }
925
Behdad Esfahbod212aba62009-05-24 00:50:27 -0400926 protected:
Behdad Esfahbod87fcdcb2009-05-24 01:03:24 -0400927 FixedVersion version; /* Version of the GSUB/GPOS table--initially set
Behdad Esfahbodf45107f2009-05-17 20:13:02 -0400928 * to 0x00010000 */
929 OffsetTo<ScriptList>
930 scriptList; /* ScriptList table */
931 OffsetTo<FeatureList>
932 featureList; /* FeatureList table */
933 OffsetTo<LookupList>
934 lookupList; /* LookupList table */
935};
936ASSERT_SIZE (GSUBGPOS, 10);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400937
Behdad Esfahbod6f20f722009-05-17 20:28:01 -0400938
Behdad Esfahbod5f5b24f2009-08-02 20:03:12 -0400939#endif /* HB_OT_LAYOUT_GSUBGPOS_PRIVATE_HH */