blob: e91bae5c5e13bb0c239a1e03d446daaa68f4573a [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 Esfahbod1376fb72010-04-29 02:19:21 -040047 hb_buffer_t *buffer, \
Behdad Esfahbod33d13fd2010-04-29 13:56:44 -040048 unsigned int context_length HB_UNUSED, \
49 unsigned int apply_depth HB_UNUSED
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -040050#define APPLY_ARG \
Behdad Esfahbod6617ead2010-04-29 02:25:30 -040051 context, \
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040052 buffer, \
53 context_length, \
Behdad Esfahbodbc200452010-04-29 01:40:26 -040054 (HB_DEBUG_APPLY ? apply_depth + 1 : 0)
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040055
Behdad Esfahbod1376fb72010-04-29 02:19:21 -040056struct hb_apply_context_t
57{
Behdad Esfahbod63493f92010-05-05 01:01:05 -040058 hb_ot_layout_context_t *layout;
Behdad Esfahbod1376fb72010-04-29 02:19:21 -040059 unsigned int nesting_level_left;
60 unsigned int lookup_flag;
61 unsigned int property; /* propety of first glyph (TODO remove) */
62};
63
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040064
Behdad Esfahbod1922ffe2010-04-21 04:19:51 -040065typedef bool (*match_func_t) (hb_codepoint_t glyph_id, const USHORT &value, const char *data);
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -040066typedef bool (*apply_lookup_func_t) (APPLY_ARG_DEF, unsigned int lookup_index);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040067
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -040068struct ContextFuncs
69{
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -040070 match_func_t match;
71 apply_lookup_func_t apply;
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040072};
73
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -040074
Behdad Esfahbod33d13fd2010-04-29 13:56:44 -040075static inline bool match_glyph (hb_codepoint_t glyph_id, const USHORT &value, const char *data HB_UNUSED)
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -040076{
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -040077 return glyph_id == value;
78}
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040079
Behdad Esfahbod1922ffe2010-04-21 04:19:51 -040080static inline bool match_class (hb_codepoint_t glyph_id, const USHORT &value, const char *data)
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -040081{
Behdad Esfahbod2b5a59c2009-08-04 11:38:50 -040082 const ClassDef &class_def = *reinterpret_cast<const ClassDef *>(data);
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -040083 return class_def.get_class (glyph_id) == value;
84}
85
Behdad Esfahbod1922ffe2010-04-21 04:19:51 -040086static inline bool match_coverage (hb_codepoint_t glyph_id, const USHORT &value, const char *data)
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -040087{
Behdad Esfahbod6b54c5d2009-05-18 18:30:25 -040088 const OffsetTo<Coverage> &coverage = (const OffsetTo<Coverage>&)value;
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -040089 return (data+coverage) (glyph_id) != NOT_COVERED;
90}
91
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -040092
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -040093static inline bool match_input (APPLY_ARG_DEF,
Behdad Esfahbode072c242009-05-18 03:47:31 -040094 unsigned int count, /* Including the first glyph (not matched) */
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -040095 const USHORT input[], /* Array of input values--start with second glyph */
96 match_func_t match_func,
Behdad Esfahbod1922ffe2010-04-21 04:19:51 -040097 const char *match_data,
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -040098 unsigned int *context_length_out)
99{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400100 unsigned int i, j;
Behdad Esfahbod7cff75b2009-05-18 04:09:05 -0400101 unsigned int end = MIN (buffer->in_length, buffer->in_pos + context_length);
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400102 if (unlikely (buffer->in_pos + count > end))
Behdad Esfahbod7cff75b2009-05-18 04:09:05 -0400103 return false;
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400104
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400105 for (i = 1, j = buffer->in_pos + 1; i < count; i++, j++)
106 {
Behdad Esfahbod63493f92010-05-05 01:01:05 -0400107 while (_hb_ot_layout_skip_mark (context->layout->face, IN_INFO (j), context->lookup_flag, NULL))
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400108 {
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400109 if (unlikely (j + count - i == end))
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400110 return false;
111 j++;
112 }
113
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400114 if (likely (!match_func (IN_GLYPH (j), input[i - 1], match_data)))
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400115 return false;
116 }
117
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400118 *context_length_out = j - buffer->in_pos;
119
120 return true;
121}
122
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400123static inline bool match_backtrack (APPLY_ARG_DEF,
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400124 unsigned int count,
125 const USHORT backtrack[],
126 match_func_t match_func,
Behdad Esfahbod1922ffe2010-04-21 04:19:51 -0400127 const char *match_data)
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400128{
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400129 if (unlikely (buffer->out_pos < count))
Behdad Esfahbod7cff75b2009-05-18 04:09:05 -0400130 return false;
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400131
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400132 for (unsigned int i = 0, j = buffer->out_pos - 1; i < count; i++, j--)
133 {
Behdad Esfahbod63493f92010-05-05 01:01:05 -0400134 while (_hb_ot_layout_skip_mark (context->layout->face, OUT_INFO (j), context->lookup_flag, NULL))
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400135 {
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400136 if (unlikely (j + 1 == count - i))
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400137 return false;
138 j--;
139 }
140
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400141 if (likely (!match_func (OUT_GLYPH (j), backtrack[i], match_data)))
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400142 return false;
143 }
144
145 return true;
146}
147
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400148static inline bool match_lookahead (APPLY_ARG_DEF,
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400149 unsigned int count,
150 const USHORT lookahead[],
151 match_func_t match_func,
Behdad Esfahbod1922ffe2010-04-21 04:19:51 -0400152 const char *match_data,
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400153 unsigned int offset)
154{
155 unsigned int i, j;
Behdad Esfahbod7cff75b2009-05-18 04:09:05 -0400156 unsigned int end = MIN (buffer->in_length, buffer->in_pos + context_length);
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400157 if (unlikely (buffer->in_pos + offset + count > end))
Behdad Esfahbod7cff75b2009-05-18 04:09:05 -0400158 return false;
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400159
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400160 for (i = 0, j = buffer->in_pos + offset; i < count; i++, j++)
161 {
Behdad Esfahbod63493f92010-05-05 01:01:05 -0400162 while (_hb_ot_layout_skip_mark (context->layout->face, OUT_INFO (j), context->lookup_flag, NULL))
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400163 {
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400164 if (unlikely (j + count - i == end))
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400165 return false;
166 j++;
167 }
168
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400169 if (likely (!match_func (IN_GLYPH (j), lookahead[i], match_data)))
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400170 return false;
171 }
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400172
173 return true;
174}
175
176
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400177struct LookupRecord
178{
Behdad Esfahbod4b8487d2010-03-16 03:46:17 -0400179 static inline unsigned int get_size () { return sizeof (LookupRecord); }
180
Behdad Esfahbod39840472010-05-05 00:23:19 -0400181 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400182 TRACE_SANITIZE ();
Behdad Esfahbodcd3827e2009-08-04 02:09:34 -0400183 return SANITIZE_SELF ();
184 }
185
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400186 USHORT sequenceIndex; /* Index into current glyph
187 * sequence--first glyph = 0 */
188 USHORT lookupListIndex; /* Lookup to apply to that
189 * position--zero--based */
190};
191ASSERT_SIZE (LookupRecord, 4);
192
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400193static inline bool apply_lookup (APPLY_ARG_DEF,
Behdad Esfahbode072c242009-05-18 03:47:31 -0400194 unsigned int count, /* Including the first glyph */
195 unsigned int lookupCount,
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400196 const LookupRecord lookupRecord[], /* Array of LookupRecords--in design order */
197 apply_lookup_func_t apply_func)
198{
Behdad Esfahbode73a0c22009-05-18 04:15:25 -0400199 unsigned int end = MIN (buffer->in_length, buffer->in_pos + context_length);
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400200 if (unlikely (buffer->in_pos + count > end))
Behdad Esfahbode73a0c22009-05-18 04:15:25 -0400201 return false;
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400202
Behdad Esfahbod7cff75b2009-05-18 04:09:05 -0400203 /* TODO We don't support lookupRecord arrays that are not increasing:
204 * Should be easy for in_place ones at least. */
Behdad Esfahbod52e9a712009-09-21 13:58:56 -0400205
206 /* Note: If sublookup is reverse, i will underflow after the first loop
207 * and we jump out of it. Not entirely disastrous. So we don't check
208 * for reverse lookup here.
209 */
Behdad Esfahbodf9c0a2d2009-09-21 13:43:54 -0400210 for (unsigned int i = 0; i < count; /* NOP */)
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400211 {
Behdad Esfahbod63493f92010-05-05 01:01:05 -0400212 while (_hb_ot_layout_skip_mark (context->layout->face, IN_CURINFO (), context->lookup_flag, NULL))
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400213 {
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400214 if (unlikely (buffer->in_pos == end))
Behdad Esfahbode73a0c22009-05-18 04:15:25 -0400215 return true;
216 /* No lookup applied for this index */
217 _hb_buffer_next_glyph (buffer);
218 }
219
Behdad Esfahbod47958de2009-05-18 04:17:47 -0400220 if (lookupCount && i == lookupRecord->sequenceIndex)
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400221 {
222 unsigned int old_pos = buffer->in_pos;
223
224 /* Apply a lookup */
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400225 bool done = apply_func (APPLY_ARG, lookupRecord->lookupListIndex);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400226
Behdad Esfahbod47958de2009-05-18 04:17:47 -0400227 lookupRecord++;
228 lookupCount--;
Behdad Esfahbodf9c0a2d2009-09-21 13:43:54 -0400229 /* Err, this is wrong if the lookup jumped over some glyphs */
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400230 i += buffer->in_pos - old_pos;
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400231 if (unlikely (buffer->in_pos == end))
Behdad Esfahbod2e8fb6c2009-05-18 04:37:37 -0400232 return true;
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400233
234 if (!done)
235 goto not_applied;
236 }
237 else
238 {
239 not_applied:
240 /* No lookup applied for this index */
241 _hb_buffer_next_glyph (buffer);
242 i++;
243 }
244 }
245
246 return true;
247}
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400248
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400249
250/* Contextual lookups */
251
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400252struct ContextLookupContext
253{
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400254 ContextFuncs funcs;
Behdad Esfahbod1922ffe2010-04-21 04:19:51 -0400255 const char *match_data;
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400256};
257
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400258static inline bool context_lookup (APPLY_ARG_DEF,
Behdad Esfahbode072c242009-05-18 03:47:31 -0400259 unsigned int inputCount, /* Including the first glyph (not matched) */
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400260 const USHORT input[], /* Array of input values--start with second glyph */
Behdad Esfahbode072c242009-05-18 03:47:31 -0400261 unsigned int lookupCount,
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400262 const LookupRecord lookupRecord[],
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400263 ContextLookupContext &lookup_context)
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400264{
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400265 return match_input (APPLY_ARG,
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400266 inputCount, input,
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400267 lookup_context.funcs.match, lookup_context.match_data,
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400268 &context_length) &&
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400269 apply_lookup (APPLY_ARG,
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400270 inputCount,
271 lookupCount, lookupRecord,
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400272 lookup_context.funcs.apply);
Behdad Esfahbodf14c2b72009-05-18 02:36:18 -0400273}
274
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400275struct Rule
276{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400277 friend struct RuleSet;
278
279 private:
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400280 inline bool apply (APPLY_ARG_DEF, ContextLookupContext &lookup_context) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400281 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400282 TRACE_APPLY ();
Behdad Esfahbodfff9aa22010-05-05 00:32:21 -0400283 const LookupRecord *lookupRecord = &StructAtOffset<LookupRecord> (input, input[0].get_size () * (inputCount ? inputCount - 1 : 0));
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400284 return context_lookup (APPLY_ARG,
Behdad Esfahbod969afd72009-05-18 05:47:47 -0400285 inputCount, input,
Behdad Esfahbodfff9aa22010-05-05 00:32:21 -0400286 lookupCount, lookupRecord,
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400287 lookup_context);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400288 }
289
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400290 public:
Behdad Esfahbod39840472010-05-05 00:23:19 -0400291 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400292 TRACE_SANITIZE ();
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500293 if (!(SANITIZE (inputCount) && SANITIZE (lookupCount))) return false;
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400294 return SANITIZE_MEM (input,
Behdad Esfahbod4b8487d2010-03-16 03:46:17 -0400295 input[0].get_size () * inputCount +
296 lookupRecordX[0].get_size () * lookupCount);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400297 }
298
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400299 private:
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400300 USHORT inputCount; /* Total number of glyphs in input
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400301 * glyph sequence--includes the first
302 * glyph */
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400303 USHORT lookupCount; /* Number of LookupRecords */
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500304 USHORT input[VAR]; /* Array of match inputs--start with
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400305 * second glyph */
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500306 LookupRecord lookupRecordX[VAR]; /* Array of LookupRecords--in
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400307 * design order */
308};
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500309ASSERT_SIZE_VAR2 (Rule, 4, USHORT, LookupRecord);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400310
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400311struct RuleSet
312{
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400313 inline bool apply (APPLY_ARG_DEF, ContextLookupContext &lookup_context) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400314 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400315 TRACE_APPLY ();
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400316 unsigned int num_rules = rule.len;
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400317 for (unsigned int i = 0; i < num_rules; i++)
318 {
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400319 if ((this+rule[i]).apply (APPLY_ARG, lookup_context))
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400320 return true;
321 }
322
323 return false;
324 }
325
Behdad Esfahbod39840472010-05-05 00:23:19 -0400326 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400327 TRACE_SANITIZE ();
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400328 return SANITIZE_WITH_BASE (this, rule);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400329 }
330
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400331 private:
332 OffsetArrayOf<Rule>
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400333 rule; /* Array of Rule tables
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400334 * ordered by preference */
335};
336
337
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400338struct ContextFormat1
339{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400340 friend struct Context;
341
342 private:
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400343 inline bool apply (APPLY_ARG_DEF, apply_lookup_func_t apply_func) const
344 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400345 TRACE_APPLY ();
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400346 unsigned int index = (this+coverage) (IN_CURGLYPH ());
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400347 if (likely (index == NOT_COVERED))
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400348 return false;
349
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400350 const RuleSet &rule_set = this+ruleSet[index];
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400351 struct ContextLookupContext lookup_context = {
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400352 {match_glyph, apply_func},
353 NULL
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400354 };
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400355 return rule_set.apply (APPLY_ARG, lookup_context);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400356 }
357
Behdad Esfahbod39840472010-05-05 00:23:19 -0400358 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400359 TRACE_SANITIZE ();
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400360 return SANITIZE_WITH_BASE (this, coverage)
361 && SANITIZE_WITH_BASE (this, ruleSet);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400362 }
363
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400364 private:
365 USHORT format; /* Format identifier--format = 1 */
366 OffsetTo<Coverage>
367 coverage; /* Offset to Coverage table--from
368 * beginning of table */
369 OffsetArrayOf<RuleSet>
370 ruleSet; /* Array of RuleSet tables
371 * ordered by Coverage Index */
372};
373ASSERT_SIZE (ContextFormat1, 6);
374
375
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400376struct ContextFormat2
377{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400378 friend struct Context;
379
380 private:
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400381 inline bool apply (APPLY_ARG_DEF, apply_lookup_func_t apply_func) const
382 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400383 TRACE_APPLY ();
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400384 unsigned int index = (this+coverage) (IN_CURGLYPH ());
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400385 if (likely (index == NOT_COVERED))
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400386 return false;
387
388 const ClassDef &class_def = this+classDef;
389 index = class_def (IN_CURGLYPH ());
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400390 const RuleSet &rule_set = this+ruleSet[index];
391 /* LONGTERMTODO: Old code fetches glyph classes at most once and caches
392 * them across subrule lookups. Not sure it's worth it.
393 */
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400394 struct ContextLookupContext lookup_context = {
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400395 {match_class, apply_func},
Behdad Esfahbod0dfcc132010-04-21 23:41:26 -0400396 CharP(&class_def)
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400397 };
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400398 return rule_set.apply (APPLY_ARG, lookup_context);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400399 }
400
Behdad Esfahbod39840472010-05-05 00:23:19 -0400401 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400402 TRACE_SANITIZE ();
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400403 return SANITIZE_WITH_BASE (this, coverage)
404 && SANITIZE_WITH_BASE (this, classDef)
405 && SANITIZE_WITH_BASE (this, ruleSet);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400406 }
407
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400408 private:
409 USHORT format; /* Format identifier--format = 2 */
410 OffsetTo<Coverage>
411 coverage; /* Offset to Coverage table--from
412 * beginning of table */
413 OffsetTo<ClassDef>
414 classDef; /* Offset to glyph ClassDef table--from
415 * beginning of table */
416 OffsetArrayOf<RuleSet>
417 ruleSet; /* Array of RuleSet tables
418 * ordered by class */
419};
420ASSERT_SIZE (ContextFormat2, 8);
421
422
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400423struct ContextFormat3
424{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400425 friend struct Context;
426
427 private:
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400428 inline bool apply (APPLY_ARG_DEF, apply_lookup_func_t apply_func) const
429 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400430 TRACE_APPLY ();
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400431 unsigned int index = (this+coverage[0]) (IN_CURGLYPH ());
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400432 if (likely (index == NOT_COVERED))
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400433 return false;
434
Behdad Esfahbodfff9aa22010-05-05 00:32:21 -0400435 const LookupRecord *lookupRecord = &StructAtOffset<LookupRecord> (coverage, coverage[0].get_size () * glyphCount);
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400436 struct ContextLookupContext lookup_context = {
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400437 {match_coverage, apply_func},
Behdad Esfahbod0dfcc132010-04-21 23:41:26 -0400438 CharP(this)
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400439 };
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400440 return context_lookup (APPLY_ARG,
Behdad Esfahbod969afd72009-05-18 05:47:47 -0400441 glyphCount, (const USHORT *) (coverage + 1),
Behdad Esfahbodfff9aa22010-05-05 00:32:21 -0400442 lookupCount, lookupRecord,
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400443 lookup_context);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400444 }
445
Behdad Esfahbod39840472010-05-05 00:23:19 -0400446 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400447 TRACE_SANITIZE ();
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400448 if (!SANITIZE_SELF ()) return false;
449 unsigned int count = glyphCount;
Behdad Esfahbod9fc86842010-04-22 13:37:58 -0400450 if (!SANITIZE_ARRAY (coverage, OffsetTo<Coverage>::get_size (), glyphCount)) return false;
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400451 for (unsigned int i = 0; i < count; i++)
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400452 if (!SANITIZE_WITH_BASE (this, coverage[i])) return false;
Behdad Esfahbodfff9aa22010-05-05 00:32:21 -0400453 LookupRecord *lookupRecord = &StructAtOffset<LookupRecord> (coverage, OffsetTo<Coverage>::get_size () * glyphCount);
454 return SANITIZE_ARRAY (lookupRecord, LookupRecord::get_size (), lookupCount);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400455 }
456
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400457 private:
458 USHORT format; /* Format identifier--format = 3 */
459 USHORT glyphCount; /* Number of glyphs in the input glyph
460 * sequence */
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400461 USHORT lookupCount; /* Number of LookupRecords */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400462 OffsetTo<Coverage>
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500463 coverage[VAR]; /* Array of offsets to Coverage
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400464 * table in glyph sequence order */
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500465 LookupRecord lookupRecordX[VAR]; /* Array of LookupRecords--in
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400466 * design order */
467};
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500468ASSERT_SIZE_VAR2 (ContextFormat3, 6, OffsetTo<Coverage>, LookupRecord);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400469
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400470struct Context
471{
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400472 protected:
Behdad Esfahbod20b035d2009-08-10 19:00:36 -0400473 inline bool apply (APPLY_ARG_DEF, apply_lookup_func_t apply_func) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400474 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400475 TRACE_APPLY ();
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400476 switch (u.format) {
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400477 case 1: return u.format1->apply (APPLY_ARG, apply_func);
478 case 2: return u.format2->apply (APPLY_ARG, apply_func);
479 case 3: return u.format3->apply (APPLY_ARG, apply_func);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400480 default:return false;
481 }
482 }
483
Behdad Esfahbod39840472010-05-05 00:23:19 -0400484 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400485 TRACE_SANITIZE ();
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400486 if (!SANITIZE (u.format)) return false;
487 switch (u.format) {
Behdad Esfahbod39840472010-05-05 00:23:19 -0400488 case 1: return u.format1->sanitize (context);
489 case 2: return u.format2->sanitize (context);
490 case 3: return u.format3->sanitize (context);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400491 default:return true;
492 }
493 }
494
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400495 private:
496 union {
Behdad Esfahbodf8dc67b2009-05-17 19:47:54 -0400497 USHORT format; /* Format identifier */
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500498 ContextFormat1 format1[VAR];
499 ContextFormat2 format2[VAR];
500 ContextFormat3 format3[VAR];
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400501 } u;
502};
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400503
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400504
505/* Chaining Contextual lookups */
506
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400507struct ChainContextLookupContext
508{
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400509 ContextFuncs funcs;
Behdad Esfahbod1922ffe2010-04-21 04:19:51 -0400510 const char *match_data[3];
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400511};
512
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400513static inline bool chain_context_lookup (APPLY_ARG_DEF,
Behdad Esfahbode072c242009-05-18 03:47:31 -0400514 unsigned int backtrackCount,
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400515 const USHORT backtrack[],
Behdad Esfahbode072c242009-05-18 03:47:31 -0400516 unsigned int inputCount, /* Including the first glyph (not matched) */
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400517 const USHORT input[], /* Array of input values--start with second glyph */
Behdad Esfahbode072c242009-05-18 03:47:31 -0400518 unsigned int lookaheadCount,
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400519 const USHORT lookahead[],
Behdad Esfahbode072c242009-05-18 03:47:31 -0400520 unsigned int lookupCount,
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400521 const LookupRecord lookupRecord[],
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400522 ChainContextLookupContext &lookup_context)
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400523{
Behdad Esfahbode072c242009-05-18 03:47:31 -0400524 /* First guess */
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400525 if (unlikely (buffer->out_pos < backtrackCount ||
Behdad Esfahbod122f21f2009-05-18 04:21:53 -0400526 buffer->in_pos + inputCount + lookaheadCount > buffer->in_length ||
527 inputCount + lookaheadCount > context_length))
Behdad Esfahbode072c242009-05-18 03:47:31 -0400528 return false;
529
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400530 unsigned int offset;
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400531 return match_backtrack (APPLY_ARG,
Behdad Esfahbode072c242009-05-18 03:47:31 -0400532 backtrackCount, backtrack,
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400533 lookup_context.funcs.match, lookup_context.match_data[0]) &&
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400534 match_input (APPLY_ARG,
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400535 inputCount, input,
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400536 lookup_context.funcs.match, lookup_context.match_data[1],
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400537 &offset) &&
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400538 match_lookahead (APPLY_ARG,
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400539 lookaheadCount, lookahead,
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400540 lookup_context.funcs.match, lookup_context.match_data[2],
Behdad Esfahbodd0ba0552009-05-18 03:56:39 -0400541 offset) &&
542 (context_length = offset, true) &&
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400543 apply_lookup (APPLY_ARG,
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400544 inputCount,
545 lookupCount, lookupRecord,
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400546 lookup_context.funcs.apply);
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400547}
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400548
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400549struct ChainRule
550{
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400551 friend struct ChainRuleSet;
552
553 private:
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400554 inline bool apply (APPLY_ARG_DEF, ChainContextLookupContext &lookup_context) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400555 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400556 TRACE_APPLY ();
Behdad Esfahbode961c862010-04-21 15:56:11 -0400557 const HeadlessArrayOf<USHORT> &input = StructAfter<HeadlessArrayOf<USHORT> > (backtrack);
558 const ArrayOf<USHORT> &lookahead = StructAfter<ArrayOf<USHORT> > (input);
559 const ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord> > (lookahead);
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400560 return chain_context_lookup (APPLY_ARG,
Behdad Esfahbod2cb08452010-04-21 22:37:31 -0400561 backtrack.len, backtrack.array(),
562 input.len, input.array(),
563 lookahead.len, lookahead.array(),
564 lookup.len, lookup.array(),
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400565 lookup_context);
Behdad Esfahbod13ed4402009-05-18 02:14:37 -0400566 return false;
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400567 }
568
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400569 public:
Behdad Esfahbod39840472010-05-05 00:23:19 -0400570 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400571 TRACE_SANITIZE ();
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400572 if (!SANITIZE (backtrack)) return false;
Behdad Esfahbode961c862010-04-21 15:56:11 -0400573 HeadlessArrayOf<USHORT> &input = StructAfter<HeadlessArrayOf<USHORT> > (backtrack);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400574 if (!SANITIZE (input)) return false;
Behdad Esfahbode961c862010-04-21 15:56:11 -0400575 ArrayOf<USHORT> &lookahead = StructAfter<ArrayOf<USHORT> > (input);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400576 if (!SANITIZE (lookahead)) return false;
Behdad Esfahbode961c862010-04-21 15:56:11 -0400577 ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord> > (lookahead);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400578 return SANITIZE (lookup);
579 }
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400580
581 private:
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400582 ArrayOf<USHORT>
583 backtrack; /* Array of backtracking values
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400584 * (to be matched before the input
585 * sequence) */
Behdad Esfahbode8cbaaf2009-05-18 02:03:58 -0400586 HeadlessArrayOf<USHORT>
587 inputX; /* Array of input values (start with
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400588 * second glyph) */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400589 ArrayOf<USHORT>
590 lookaheadX; /* Array of lookahead values's (to be
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400591 * matched after the input sequence) */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400592 ArrayOf<LookupRecord>
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400593 lookupX; /* Array of LookupRecords--in
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400594 * design order) */
595};
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400596ASSERT_SIZE (ChainRule, 8);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400597
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400598struct ChainRuleSet
599{
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400600 inline bool apply (APPLY_ARG_DEF, ChainContextLookupContext &lookup_context) const
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400601 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400602 TRACE_APPLY ();
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400603 unsigned int num_rules = rule.len;
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400604 for (unsigned int i = 0; i < num_rules; i++)
605 {
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400606 if ((this+rule[i]).apply (APPLY_ARG, lookup_context))
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400607 return true;
608 }
609
610 return false;
611 }
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400612
Behdad Esfahbod39840472010-05-05 00:23:19 -0400613 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400614 TRACE_SANITIZE ();
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400615 return SANITIZE_WITH_BASE (this, rule);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400616 }
617
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400618 private:
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400619 OffsetArrayOf<ChainRule>
620 rule; /* Array of ChainRule tables
621 * ordered by preference */
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400622};
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400623ASSERT_SIZE (ChainRuleSet, 2);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400624
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400625struct ChainContextFormat1
626{
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400627 friend struct ChainContext;
628
629 private:
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400630 inline bool apply (APPLY_ARG_DEF, apply_lookup_func_t apply_func) const
631 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400632 TRACE_APPLY ();
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400633 unsigned int index = (this+coverage) (IN_CURGLYPH ());
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400634 if (likely (index == NOT_COVERED))
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400635 return false;
636
637 const ChainRuleSet &rule_set = this+ruleSet[index];
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400638 struct ChainContextLookupContext lookup_context = {
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400639 {match_glyph, apply_func},
640 {NULL, NULL, NULL}
641 };
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400642 return rule_set.apply (APPLY_ARG, lookup_context);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400643 }
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400644
Behdad Esfahbod39840472010-05-05 00:23:19 -0400645 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400646 TRACE_SANITIZE ();
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400647 return SANITIZE_WITH_BASE (this, coverage)
648 && SANITIZE_WITH_BASE (this, ruleSet);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400649 }
650
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400651 private:
652 USHORT format; /* Format identifier--format = 1 */
Behdad Esfahbod48f16ed2009-05-17 22:11:30 -0400653 OffsetTo<Coverage>
654 coverage; /* Offset to Coverage table--from
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400655 * beginning of table */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400656 OffsetArrayOf<ChainRuleSet>
657 ruleSet; /* Array of ChainRuleSet tables
658 * ordered by Coverage Index */
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400659};
660ASSERT_SIZE (ChainContextFormat1, 6);
661
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400662struct ChainContextFormat2
663{
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400664 friend struct ChainContext;
665
666 private:
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400667 inline bool apply (APPLY_ARG_DEF, apply_lookup_func_t apply_func) const
668 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400669 TRACE_APPLY ();
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400670 unsigned int index = (this+coverage) (IN_CURGLYPH ());
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400671 if (likely (index == NOT_COVERED))
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400672 return false;
673
674 const ClassDef &backtrack_class_def = this+backtrackClassDef;
675 const ClassDef &input_class_def = this+inputClassDef;
676 const ClassDef &lookahead_class_def = this+lookaheadClassDef;
677
678 index = input_class_def (IN_CURGLYPH ());
679 const ChainRuleSet &rule_set = this+ruleSet[index];
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400680 /* LONGTERMTODO: Old code fetches glyph classes at most once and caches
681 * them across subrule lookups. Not sure it's worth it.
682 */
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400683 struct ChainContextLookupContext lookup_context = {
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400684 {match_class, apply_func},
Behdad Esfahbod0dfcc132010-04-21 23:41:26 -0400685 {CharP(&backtrack_class_def),
686 CharP(&input_class_def),
687 CharP(&lookahead_class_def)}
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400688 };
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400689 return rule_set.apply (APPLY_ARG, lookup_context);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400690 }
691
Behdad Esfahbod39840472010-05-05 00:23:19 -0400692 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400693 TRACE_SANITIZE ();
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400694 return SANITIZE_WITH_BASE (this, coverage)
695 && SANITIZE_WITH_BASE (this, backtrackClassDef)
696 && SANITIZE_WITH_BASE (this, inputClassDef)
697 && SANITIZE_WITH_BASE (this, lookaheadClassDef)
698 && SANITIZE_WITH_BASE (this, ruleSet);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400699 }
700
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400701 private:
702 USHORT format; /* Format identifier--format = 2 */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400703 OffsetTo<Coverage>
704 coverage; /* Offset to Coverage table--from
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400705 * beginning of table */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400706 OffsetTo<ClassDef>
707 backtrackClassDef; /* Offset to glyph ClassDef table
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400708 * containing backtrack sequence
709 * data--from beginning of table */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400710 OffsetTo<ClassDef>
711 inputClassDef; /* Offset to glyph ClassDef
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400712 * table containing input sequence
713 * data--from beginning of table */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400714 OffsetTo<ClassDef>
715 lookaheadClassDef; /* Offset to glyph ClassDef table
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400716 * containing lookahead sequence
717 * data--from beginning of table */
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400718 OffsetArrayOf<ChainRuleSet>
719 ruleSet; /* Array of ChainRuleSet tables
720 * ordered by class */
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400721};
722ASSERT_SIZE (ChainContextFormat2, 12);
723
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400724struct ChainContextFormat3
725{
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400726 friend struct ChainContext;
727
728 private:
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400729
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400730 inline bool apply (APPLY_ARG_DEF, apply_lookup_func_t apply_func) const
731 {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400732 TRACE_APPLY ();
Behdad Esfahbode961c862010-04-21 15:56:11 -0400733 const OffsetArrayOf<Coverage> &input = StructAfter<OffsetArrayOf<Coverage> > (backtrack);
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400734
735 unsigned int index = (this+input[0]) (IN_CURGLYPH ());
Behdad Esfahbod64d3fc82010-05-03 22:51:19 -0400736 if (likely (index == NOT_COVERED))
Behdad Esfahbodaa3d7ad2009-05-17 23:17:56 -0400737 return false;
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400738
Behdad Esfahbode961c862010-04-21 15:56:11 -0400739 const OffsetArrayOf<Coverage> &lookahead = StructAfter<OffsetArrayOf<Coverage> > (input);
740 const ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord> > (lookahead);
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400741 struct ChainContextLookupContext lookup_context = {
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400742 {match_coverage, apply_func},
Behdad Esfahbod0dfcc132010-04-21 23:41:26 -0400743 {CharP(this), CharP(this), CharP(this)}
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400744 };
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400745 return chain_context_lookup (APPLY_ARG,
Behdad Esfahbod2cb08452010-04-21 22:37:31 -0400746 backtrack.len, (const USHORT *) backtrack.array(),
747 input.len, (const USHORT *) input.array() + 1,
748 lookahead.len, (const USHORT *) lookahead.array(),
749 lookup.len, lookup.array(),
Behdad Esfahbod7cda6fa2009-07-29 18:37:57 -0400750 lookup_context);
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400751 return false;
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400752 }
753
Behdad Esfahbod39840472010-05-05 00:23:19 -0400754 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400755 TRACE_SANITIZE ();
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400756 if (!SANITIZE_WITH_BASE (this, backtrack)) return false;
Behdad Esfahbode961c862010-04-21 15:56:11 -0400757 OffsetArrayOf<Coverage> &input = StructAfter<OffsetArrayOf<Coverage> > (backtrack);
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400758 if (!SANITIZE_WITH_BASE (this, input)) return false;
Behdad Esfahbode961c862010-04-21 15:56:11 -0400759 OffsetArrayOf<Coverage> &lookahead = StructAfter<OffsetArrayOf<Coverage> > (input);
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400760 if (!SANITIZE_WITH_BASE (this, lookahead)) return false;
Behdad Esfahbode961c862010-04-21 15:56:11 -0400761 ArrayOf<LookupRecord> &lookup = StructAfter<ArrayOf<LookupRecord> > (lookahead);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400762 return SANITIZE (lookup);
763 }
764
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400765 private:
766 USHORT format; /* Format identifier--format = 3 */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400767 OffsetArrayOf<Coverage>
Behdad Esfahbod13ed4402009-05-18 02:14:37 -0400768 backtrack; /* Array of coverage tables
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400769 * in backtracking sequence, in glyph
770 * sequence order */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400771 OffsetArrayOf<Coverage>
Behdad Esfahbod13ed4402009-05-18 02:14:37 -0400772 inputX ; /* Array of coverage
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400773 * tables in input sequence, in glyph
774 * sequence order */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400775 OffsetArrayOf<Coverage>
Behdad Esfahbod13ed4402009-05-18 02:14:37 -0400776 lookaheadX; /* Array of coverage tables
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400777 * in lookahead sequence, in glyph
778 * sequence order */
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400779 ArrayOf<LookupRecord>
Behdad Esfahbod02e1e5c2009-05-18 02:47:57 -0400780 lookupX; /* Array of LookupRecords--in
Behdad Esfahboddcb6b602009-05-18 01:49:57 -0400781 * design order) */
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400782};
783ASSERT_SIZE (ChainContextFormat3, 10);
784
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400785struct ChainContext
786{
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400787 protected:
Behdad Esfahbod20b035d2009-08-10 19:00:36 -0400788 inline bool apply (APPLY_ARG_DEF, 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 Esfahbodca5290f2009-05-17 20:48:27 -0400791 switch (u.format) {
Behdad Esfahbodeb0dfc82009-05-18 18:22:44 -0400792 case 1: return u.format1->apply (APPLY_ARG, apply_func);
793 case 2: return u.format2->apply (APPLY_ARG, apply_func);
794 case 3: return u.format3->apply (APPLY_ARG, apply_func);
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400795 default:return false;
796 }
797 }
798
Behdad Esfahbod39840472010-05-05 00:23:19 -0400799 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400800 TRACE_SANITIZE ();
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400801 if (!SANITIZE (u.format)) return false;
802 switch (u.format) {
Behdad Esfahbod39840472010-05-05 00:23:19 -0400803 case 1: return u.format1->sanitize (context);
804 case 2: return u.format2->sanitize (context);
805 case 3: return u.format3->sanitize (context);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400806 default:return true;
807 }
808 }
809
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400810 private:
811 union {
812 USHORT format; /* Format identifier */
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500813 ChainContextFormat1 format1[VAR];
814 ChainContextFormat2 format2[VAR];
815 ChainContextFormat3 format3[VAR];
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400816 } u;
817};
Behdad Esfahbodca5290f2009-05-17 20:48:27 -0400818
819
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400820struct ExtensionFormat1
821{
822 friend struct Extension;
823
Behdad Esfahbod18939482009-08-04 14:27:56 -0400824 protected:
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400825 inline unsigned int get_type (void) const { return extensionLookupType; }
Behdad Esfahbod3b2c2df2010-04-22 16:51:42 -0400826 inline unsigned int get_offset (void) const { return extensionOffset; }
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400827
Behdad Esfahbod39840472010-05-05 00:23:19 -0400828 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400829 TRACE_SANITIZE ();
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400830 return SANITIZE_SELF ();
831 }
832
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400833 private:
834 USHORT format; /* Format identifier. Set to 1. */
835 USHORT extensionLookupType; /* Lookup type of subtable referenced
836 * by ExtensionOffset (i.e. the
837 * extension subtable). */
Behdad Esfahbod81f2af42010-04-22 00:58:49 -0400838 ULONG extensionOffset; /* Offset to the extension subtable,
839 * of lookup type subtable. */
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400840};
841ASSERT_SIZE (ExtensionFormat1, 8);
842
843struct Extension
844{
845 inline unsigned int get_type (void) const
846 {
847 switch (u.format) {
848 case 1: return u.format1->get_type ();
849 default:return 0;
850 }
851 }
Behdad Esfahbod3b2c2df2010-04-22 16:51:42 -0400852 inline unsigned int get_offset (void) const
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400853 {
854 switch (u.format) {
Behdad Esfahbod3b2c2df2010-04-22 16:51:42 -0400855 case 1: return u.format1->get_offset ();
856 default:return 0;
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400857 }
858 }
859
Behdad Esfahbod39840472010-05-05 00:23:19 -0400860 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400861 TRACE_SANITIZE ();
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400862 if (!SANITIZE (u.format)) return false;
863 switch (u.format) {
Behdad Esfahbod39840472010-05-05 00:23:19 -0400864 case 1: return u.format1->sanitize (context);
Behdad Esfahbod70de50c2009-08-04 00:58:28 -0400865 default:return true;
866 }
867 }
868
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400869 private:
870 union {
871 USHORT format; /* Format identifier */
Behdad Esfahbodd3480ba2009-11-03 10:47:29 -0500872 ExtensionFormat1 format1[VAR];
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400873 } u;
874};
Behdad Esfahbodd468f9a2009-05-21 22:31:33 -0400875
876
Behdad Esfahbodf45107f2009-05-17 20:13:02 -0400877/*
878 * GSUB/GPOS Common
879 */
880
Behdad Esfahbod60d77cf2009-05-19 23:58:54 -0400881struct GSUBGPOS
882{
Behdad Esfahboda328d662009-08-04 20:27:05 -0400883 static const hb_tag_t GSUBTag = HB_OT_TAG_GSUB;
884 static const hb_tag_t GPOSTag = HB_OT_TAG_GPOS;
Behdad Esfahbodf45107f2009-05-17 20:13:02 -0400885
Behdad Esfahbodbff3c0f2009-08-07 19:46:30 -0400886 inline unsigned int get_script_count (void) const
887 { return (this+scriptList).len; }
888 inline const Tag& get_script_tag (unsigned int i) const
889 { return (this+scriptList).get_tag (i); }
Behdad Esfahbode21899b2009-11-04 16:36:14 -0500890 inline unsigned int get_script_tags (unsigned int start_offset,
891 unsigned int *script_count /* IN/OUT */,
892 hb_tag_t *script_tags /* OUT */) const
893 { return (this+scriptList).get_tags (start_offset, script_count, script_tags); }
Behdad Esfahbodbff3c0f2009-08-07 19:46:30 -0400894 inline const Script& get_script (unsigned int i) const
895 { return (this+scriptList)[i]; }
896 inline bool find_script_index (hb_tag_t tag, unsigned int *index) const
897 { return (this+scriptList).find_index (tag, index); }
Behdad Esfahbodf45107f2009-05-17 20:13:02 -0400898
Behdad Esfahbodbff3c0f2009-08-07 19:46:30 -0400899 inline unsigned int get_feature_count (void) const
900 { return (this+featureList).len; }
901 inline const Tag& get_feature_tag (unsigned int i) const
902 { return (this+featureList).get_tag (i); }
Behdad Esfahbode21899b2009-11-04 16:36:14 -0500903 inline unsigned int get_feature_tags (unsigned int start_offset,
904 unsigned int *feature_count /* IN/OUT */,
905 hb_tag_t *feature_tags /* OUT */) const
906 { return (this+featureList).get_tags (start_offset, feature_count, feature_tags); }
Behdad Esfahbodbff3c0f2009-08-07 19:46:30 -0400907 inline const Feature& get_feature (unsigned int i) const
908 { return (this+featureList)[i]; }
909 inline bool find_feature_index (hb_tag_t tag, unsigned int *index) const
910 { return (this+featureList).find_index (tag, index); }
911
912 inline unsigned int get_lookup_count (void) const
913 { return (this+lookupList).len; }
914 inline const Lookup& get_lookup (unsigned int i) const
915 { return (this+lookupList)[i]; }
Behdad Esfahbodf45107f2009-05-17 20:13:02 -0400916
Behdad Esfahbod39840472010-05-05 00:23:19 -0400917 inline bool sanitize (hb_sanitize_context_t *context) {
Behdad Esfahbod3e2401f2009-08-28 17:17:11 -0400918 TRACE_SANITIZE ();
Behdad Esfahbod26bfcb62010-05-04 14:49:45 -0400919 return SANITIZE (version) && likely (version.major == 1)
Behdad Esfahbodbb029af2010-05-04 15:28:52 -0400920 && SANITIZE_WITH_BASE (this, scriptList)
921 && SANITIZE_WITH_BASE (this, featureList)
922 && SANITIZE_WITH_BASE (this, lookupList);
Behdad Esfahbodcd3827e2009-08-04 02:09:34 -0400923 }
924
Behdad Esfahbod212aba62009-05-24 00:50:27 -0400925 protected:
Behdad Esfahbod87fcdcb2009-05-24 01:03:24 -0400926 FixedVersion version; /* Version of the GSUB/GPOS table--initially set
Behdad Esfahbodf45107f2009-05-17 20:13:02 -0400927 * to 0x00010000 */
928 OffsetTo<ScriptList>
929 scriptList; /* ScriptList table */
930 OffsetTo<FeatureList>
931 featureList; /* FeatureList table */
932 OffsetTo<LookupList>
933 lookupList; /* LookupList table */
934};
935ASSERT_SIZE (GSUBGPOS, 10);
Behdad Esfahbod66bf7ce2009-05-17 08:28:42 -0400936
Behdad Esfahbod6f20f722009-05-17 20:28:01 -0400937
Behdad Esfahbod5f5b24f2009-08-02 20:03:12 -0400938#endif /* HB_OT_LAYOUT_GSUBGPOS_PRIVATE_HH */