blob: 6cfb115c37dca142083449b4e6cb45eb20992fe6 [file] [log] [blame]
Behdad Esfahboda2a9a022008-01-15 22:46:32 +00001/*
Behdad Esfahbod2409d5f2011-04-21 17:14:28 -04002 * Copyright © 1998-2004 David Turner and Werner Lemberg
3 * Copyright © 2004,2007,2009,2010 Red Hat, Inc.
Behdad Esfahbod05207a72012-09-25 17:44:53 -04004 * Copyright © 2011,2012 Google, Inc.
Behdad Esfahbod6b347132007-10-11 08:30:50 +00005 *
Behdad Esfahbod8f0d7e02011-04-15 18:59:56 -04006 * This is part of HarfBuzz, a text shaping library.
Behdad Esfahbod6b347132007-10-11 08:30:50 +00007 *
Behdad Esfahboda2a9a022008-01-15 22:46:32 +00008 * Permission is hereby granted, without written agreement and without
9 * license or royalty fees, to use, copy, modify, and distribute this
10 * software and its documentation for any purpose, provided that the
11 * above copyright notice and the following two paragraphs appear in
12 * all copies of this software.
Behdad Esfahbod6b347132007-10-11 08:30:50 +000013 *
Behdad Esfahboda2a9a022008-01-15 22:46:32 +000014 * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
15 * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
16 * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
17 * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
18 * DAMAGE.
19 *
20 * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
21 * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
22 * FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS
23 * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
24 * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
25 *
26 * Red Hat Author(s): Owen Taylor, Behdad Esfahbod
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -040027 * Google Author(s): Behdad Esfahbod
Behdad Esfahbod6b347132007-10-11 08:30:50 +000028 */
Behdad Esfahboda2a9a022008-01-15 22:46:32 +000029
Behdad Esfahbodda8edbb2010-06-09 07:15:39 -040030#ifndef HB_BUFFER_PRIVATE_HH
31#define HB_BUFFER_PRIVATE_HH
Behdad Esfahbod6b347132007-10-11 08:30:50 +000032
Behdad Esfahbodc57d4542011-04-20 18:50:27 -040033#include "hb-private.hh"
Behdad Esfahbod5c0adce2009-05-20 05:42:12 -040034#include "hb-buffer.h"
Behdad Esfahbodfca368c2011-04-21 18:24:02 -040035#include "hb-object-private.hh"
Behdad Esfahbodfb194b82011-04-20 02:00:47 -040036#include "hb-unicode-private.hh"
Behdad Esfahbod6b347132007-10-11 08:30:50 +000037
Behdad Esfahbod6b347132007-10-11 08:30:50 +000038
Behdad Esfahbodacdba3f2010-07-23 15:11:18 -040039
Behdad Esfahbod88474c62010-10-27 14:42:15 -040040ASSERT_STATIC (sizeof (hb_glyph_info_t) == 20);
Behdad Esfahbodf1322e52009-08-01 22:53:04 -040041ASSERT_STATIC (sizeof (hb_glyph_info_t) == sizeof (hb_glyph_position_t));
42
Behdad Esfahbod2f87ceb2012-07-27 04:02:38 -040043
44/*
45 * hb_segment_properties_t
46 */
47
Behdad Esfahbod1bc1cb32012-06-16 15:21:55 -040048typedef struct hb_segment_properties_t {
Behdad Esfahboda7c50462010-10-08 18:47:47 -040049 hb_direction_t direction;
50 hb_script_t script;
51 hb_language_t language;
Behdad Esfahbod6220e5f2012-06-06 03:30:09 -040052 ASSERT_POD ();
Behdad Esfahboda7c50462010-10-08 18:47:47 -040053} hb_segment_properties_t;
Behdad Esfahbod2f87ceb2012-07-27 04:02:38 -040054
Behdad Esfahbodea278d32012-07-27 02:12:28 -040055#define _HB_BUFFER_PROPS_DEFAULT { HB_DIRECTION_INVALID, HB_SCRIPT_INVALID, HB_LANGUAGE_INVALID }
Behdad Esfahboda7c50462010-10-08 18:47:47 -040056
Behdad Esfahbod2f87ceb2012-07-27 04:02:38 -040057static inline hb_bool_t
58hb_segment_properties_equal (const hb_segment_properties_t *a,
59 const hb_segment_properties_t *b)
60{
61 return a->direction == b->direction &&
62 a->script == b->script &&
63 a->language == b->language;
64}
65
66
Behdad Esfahbod801298b2012-08-08 14:26:36 -040067#if 0
68static inline unsigned int
Behdad Esfahbod2f87ceb2012-07-27 04:02:38 -040069hb_segment_properties_hash (const hb_segment_properties_t *p)
70{
71 /* TODO improve */
Behdad Esfahbod801298b2012-08-08 14:26:36 -040072 return (unsigned int) p->direction +
73 (unsigned int) p->script +
74 (intptr_t) (p->language);
Behdad Esfahbod2f87ceb2012-07-27 04:02:38 -040075}
Behdad Esfahbod801298b2012-08-08 14:26:36 -040076#endif
Behdad Esfahbod2f87ceb2012-07-27 04:02:38 -040077
78
79
80/*
81 * hb_buffer_t
82 */
Behdad Esfahbodf1322e52009-08-01 22:53:04 -040083
Behdad Esfahbod1bc1cb32012-06-16 15:21:55 -040084struct hb_buffer_t {
Behdad Esfahbodfca368c2011-04-21 18:24:02 -040085 hb_object_header_t header;
Behdad Esfahbod6220e5f2012-06-06 03:30:09 -040086 ASSERT_POD ();
Behdad Esfahbodd0316a82010-05-12 23:34:52 -040087
88 /* Information about how the text in the buffer should be treated */
Behdad Esfahbod4e4ef242010-07-23 17:22:11 -040089
Behdad Esfahboda7c50462010-10-08 18:47:47 -040090 hb_unicode_funcs_t *unicode; /* Unicode functions */
91 hb_segment_properties_t props; /* Script, language, direction */
Behdad Esfahbod0c7df222012-11-13 14:42:35 -080092 hb_buffer_flags_t flags; /* BOT / EOT / etc. */
Behdad Esfahbodd0316a82010-05-12 23:34:52 -040093
94 /* Buffer contents */
95
Behdad Esfahbod96fdc042012-09-06 22:26:16 -040096 hb_buffer_content_type_t content_type;
97
Behdad Esfahbode0db4b82011-04-28 12:56:49 -040098 bool in_error; /* Allocation failed */
Behdad Esfahbodfca368c2011-04-21 18:24:02 -040099 bool have_output; /* Whether we have an output buffer going on */
100 bool have_positions; /* Whether we have positions */
Behdad Esfahbod910a33f2010-05-14 22:13:38 -0400101
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400102 unsigned int idx; /* Cursor into ->info and ->pos arrays */
Behdad Esfahbod910a33f2010-05-14 22:13:38 -0400103 unsigned int len; /* Length of ->info and ->pos arrays */
Behdad Esfahbodcc1a8a92011-01-06 14:58:52 -0500104 unsigned int out_len; /* Length of ->out array if have_output */
Behdad Esfahbodd0316a82010-05-12 23:34:52 -0400105
Behdad Esfahbodc0af1932011-04-15 19:26:24 -0400106 unsigned int allocated; /* Length of allocated arrays */
Behdad Esfahbod88474c62010-10-27 14:42:15 -0400107 hb_glyph_info_t *info;
108 hb_glyph_info_t *out_info;
109 hb_glyph_position_t *pos;
Behdad Esfahbodd0316a82010-05-12 23:34:52 -0400110
Behdad Esfahbod99c26952012-05-13 15:45:18 +0200111 inline hb_glyph_info_t &cur (unsigned int i = 0) { return info[idx + i]; }
112 inline hb_glyph_info_t cur (unsigned int i = 0) const { return info[idx + i]; }
113
114 inline hb_glyph_position_t &cur_pos (unsigned int i = 0) { return pos[idx + i]; }
115 inline hb_glyph_position_t cur_pos (unsigned int i = 0) const { return pos[idx + i]; }
116
117 inline hb_glyph_info_t &prev (void) { return out_info[out_len - 1]; }
118 inline hb_glyph_info_t prev (void) const { return info[out_len - 1]; }
119
Behdad Esfahbodf4a579b2011-07-25 16:20:16 -0400120 unsigned int serial;
Behdad Esfahbod05207a72012-09-25 17:44:53 -0400121
122 /* These reflect current allocations of the bytes in glyph_info_t's var1 and var2. */
Behdad Esfahbodf4a579b2011-07-25 16:20:16 -0400123 uint8_t allocated_var_bytes[8];
124 const char *allocated_var_owner[8];
125
Behdad Esfahbod05207a72012-09-25 17:44:53 -0400126 /* Text before / after the main buffer contents.
127 * Always in Unicode, and ordered outward.
128 * Index 0 is for "pre-context", 1 for "post-context". */
129 static const unsigned int CONTEXT_LENGTH = 5;
130 hb_codepoint_t context[2][CONTEXT_LENGTH];
131 unsigned int context_len[2];
132
Behdad Esfahbodd0316a82010-05-12 23:34:52 -0400133
134 /* Methods */
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400135
136 HB_INTERNAL void reset (void);
Behdad Esfahbod82ecaff2012-11-13 13:57:52 -0800137 HB_INTERNAL void clear (void);
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400138
Behdad Esfahbodcc1a8a92011-01-06 14:58:52 -0500139 inline unsigned int backtrack_len (void) const
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400140 { return have_output? out_len : idx; }
Behdad Esfahboddbf56b12010-10-27 17:06:12 -0400141 inline unsigned int next_serial (void) { return serial++; }
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400142
Behdad Esfahbodf4a579b2011-07-25 16:20:16 -0400143 HB_INTERNAL void allocate_var (unsigned int byte_i, unsigned int count, const char *owner);
144 HB_INTERNAL void deallocate_var (unsigned int byte_i, unsigned int count, const char *owner);
Behdad Esfahbod965c2802012-08-29 13:59:16 -0400145 HB_INTERNAL void assert_var (unsigned int byte_i, unsigned int count, const char *owner);
Behdad Esfahboda9ad3d32011-07-28 15:42:18 -0400146 HB_INTERNAL void deallocate_var_all (void);
Behdad Esfahbodf4a579b2011-07-25 16:20:16 -0400147
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400148 HB_INTERNAL void add (hb_codepoint_t codepoint,
149 hb_mask_t mask,
150 unsigned int cluster);
151
152 HB_INTERNAL void reverse_range (unsigned int start, unsigned int end);
153 HB_INTERNAL void reverse (void);
154 HB_INTERNAL void reverse_clusters (void);
Behdad Esfahbod02aeca92011-08-04 22:31:05 -0400155 HB_INTERNAL void guess_properties (void);
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400156
157 HB_INTERNAL void swap_buffers (void);
Behdad Esfahbod0bc7a382012-10-29 22:02:45 -0700158 HB_INTERNAL void remove_output (void);
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400159 HB_INTERNAL void clear_output (void);
160 HB_INTERNAL void clear_positions (void);
Behdad Esfahbod69cc4922012-07-31 14:51:36 -0400161
Behdad Esfahbod9ebe8c02011-08-26 09:29:42 +0200162 HB_INTERNAL void replace_glyphs (unsigned int num_in,
163 unsigned int num_out,
Behdad Esfahbod8e3715f2012-04-23 22:18:54 -0400164 const hb_codepoint_t *glyph_data);
Behdad Esfahbod69cc4922012-07-31 14:51:36 -0400165
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400166 HB_INTERNAL void replace_glyph (hb_codepoint_t glyph_index);
167 /* Makes a copy of the glyph at idx to output and replace glyph_index */
168 HB_INTERNAL void output_glyph (hb_codepoint_t glyph_index);
Behdad Esfahbodb85800f2012-08-31 18:12:01 -0400169 HB_INTERNAL void output_info (hb_glyph_info_t &glyph_info);
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400170 /* Copies glyph at idx to output but doesn't advance idx */
171 HB_INTERNAL void copy_glyph (void);
172 /* Copies glyph at idx to output and advance idx.
173 * If there's no output, just advance idx. */
Behdad Esfahbod1be368e2012-08-31 16:29:17 -0400174 inline void
175 next_glyph (void)
176 {
177 if (have_output)
178 {
179 if (unlikely (out_info != info || out_len != idx)) {
180 if (unlikely (!make_room_for (1, 1))) return;
181 out_info[out_len] = info[idx];
182 }
183 out_len++;
184 }
185
186 idx++;
187 }
188
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400189 /* Advance idx without copying to output. */
190 inline void skip_glyph (void) { idx++; }
Behdad Esfahbod1ce7b872010-05-21 17:31:45 +0100191
Behdad Esfahbodbd7378b2010-10-13 18:33:16 -0400192 inline void reset_masks (hb_mask_t mask)
193 {
Behdad Esfahbod31f18ab2011-06-15 09:49:58 -0400194 for (unsigned int j = 0; j < len; j++)
195 info[j].mask = mask;
Behdad Esfahbodbd7378b2010-10-13 18:33:16 -0400196 }
197 inline void add_masks (hb_mask_t mask)
198 {
Behdad Esfahbod31f18ab2011-06-15 09:49:58 -0400199 for (unsigned int j = 0; j < len; j++)
200 info[j].mask |= mask;
Behdad Esfahbodbd7378b2010-10-13 18:33:16 -0400201 }
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400202 HB_INTERNAL void set_masks (hb_mask_t value,
203 hb_mask_t mask,
204 unsigned int cluster_start,
205 unsigned int cluster_end);
206
Behdad Esfahbodd4cc4472012-04-07 21:52:28 -0400207 HB_INTERNAL void merge_clusters (unsigned int start,
208 unsigned int end);
209 HB_INTERNAL void merge_out_clusters (unsigned int start,
210 unsigned int end);
211
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400212 /* Internal methods */
213 HB_INTERNAL bool enlarge (unsigned int size);
214
215 inline bool ensure (unsigned int size)
Behdad Esfahbodd16ccc42012-07-18 15:43:55 -0400216 { return likely (size < allocated) ? true : enlarge (size); }
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400217
218 HB_INTERNAL bool make_room_for (unsigned int num_in, unsigned int num_out);
Behdad Esfahbode62df432011-08-03 17:38:54 -0400219
220 HB_INTERNAL void *get_scratch_buffer (unsigned int *size);
Behdad Esfahbod05207a72012-09-25 17:44:53 -0400221
222 inline void clear_context (unsigned int side) { context_len[side] = 0; }
Behdad Esfahbodd0316a82010-05-12 23:34:52 -0400223};
224
225
Behdad Esfahbodb65c0602011-07-28 16:48:43 -0400226#define HB_BUFFER_XALLOCATE_VAR(b, func, var, owner) \
227 b->func (offsetof (hb_glyph_info_t, var) - offsetof(hb_glyph_info_t, var1), \
228 sizeof (b->info[0].var), owner)
229#define HB_BUFFER_ALLOCATE_VAR(b, var) \
230 HB_BUFFER_XALLOCATE_VAR (b, allocate_var, var (), #var)
231#define HB_BUFFER_DEALLOCATE_VAR(b, var) \
232 HB_BUFFER_XALLOCATE_VAR (b, deallocate_var, var (), #var)
Behdad Esfahbod965c2802012-08-29 13:59:16 -0400233#define HB_BUFFER_ASSERT_VAR(b, var) \
234 HB_BUFFER_XALLOCATE_VAR (b, assert_var, var (), #var)
Behdad Esfahbodb65c0602011-07-28 16:48:43 -0400235
236
Behdad Esfahbodda8edbb2010-06-09 07:15:39 -0400237#endif /* HB_BUFFER_PRIVATE_HH */