blob: 703239035f0056dd4a1e1a57f27d3f13297d05c2 [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 Esfahbod88474c62010-10-27 14:42:15 -040039ASSERT_STATIC (sizeof (hb_glyph_info_t) == 20);
Behdad Esfahbodf1322e52009-08-01 22:53:04 -040040ASSERT_STATIC (sizeof (hb_glyph_info_t) == sizeof (hb_glyph_position_t));
41
Behdad Esfahbod2f87ceb2012-07-27 04:02:38 -040042
43/*
Behdad Esfahbod2f87ceb2012-07-27 04:02:38 -040044 * hb_buffer_t
45 */
Behdad Esfahbodf1322e52009-08-01 22:53:04 -040046
Behdad Esfahbod1bc1cb32012-06-16 15:21:55 -040047struct hb_buffer_t {
Behdad Esfahbodfca368c2011-04-21 18:24:02 -040048 hb_object_header_t header;
Behdad Esfahbod6220e5f2012-06-06 03:30:09 -040049 ASSERT_POD ();
Behdad Esfahbodd0316a82010-05-12 23:34:52 -040050
51 /* Information about how the text in the buffer should be treated */
Behdad Esfahbod4e4ef242010-07-23 17:22:11 -040052
Behdad Esfahboda7c50462010-10-08 18:47:47 -040053 hb_unicode_funcs_t *unicode; /* Unicode functions */
54 hb_segment_properties_t props; /* Script, language, direction */
Behdad Esfahbod0c7df222012-11-13 14:42:35 -080055 hb_buffer_flags_t flags; /* BOT / EOT / etc. */
Behdad Esfahbodd0316a82010-05-12 23:34:52 -040056
57 /* Buffer contents */
58
Behdad Esfahbod96fdc042012-09-06 22:26:16 -040059 hb_buffer_content_type_t content_type;
60
Behdad Esfahbode0db4b82011-04-28 12:56:49 -040061 bool in_error; /* Allocation failed */
Behdad Esfahbodfca368c2011-04-21 18:24:02 -040062 bool have_output; /* Whether we have an output buffer going on */
63 bool have_positions; /* Whether we have positions */
Behdad Esfahbod910a33f2010-05-14 22:13:38 -040064
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -040065 unsigned int idx; /* Cursor into ->info and ->pos arrays */
Behdad Esfahbod910a33f2010-05-14 22:13:38 -040066 unsigned int len; /* Length of ->info and ->pos arrays */
Behdad Esfahbodcc1a8a92011-01-06 14:58:52 -050067 unsigned int out_len; /* Length of ->out array if have_output */
Behdad Esfahbodd0316a82010-05-12 23:34:52 -040068
Behdad Esfahbodc0af1932011-04-15 19:26:24 -040069 unsigned int allocated; /* Length of allocated arrays */
Behdad Esfahbod88474c62010-10-27 14:42:15 -040070 hb_glyph_info_t *info;
71 hb_glyph_info_t *out_info;
72 hb_glyph_position_t *pos;
Behdad Esfahbodd0316a82010-05-12 23:34:52 -040073
Behdad Esfahbod99c26952012-05-13 15:45:18 +020074 inline hb_glyph_info_t &cur (unsigned int i = 0) { return info[idx + i]; }
75 inline hb_glyph_info_t cur (unsigned int i = 0) const { return info[idx + i]; }
76
77 inline hb_glyph_position_t &cur_pos (unsigned int i = 0) { return pos[idx + i]; }
78 inline hb_glyph_position_t cur_pos (unsigned int i = 0) const { return pos[idx + i]; }
79
80 inline hb_glyph_info_t &prev (void) { return out_info[out_len - 1]; }
81 inline hb_glyph_info_t prev (void) const { return info[out_len - 1]; }
82
Behdad Esfahbod8b63efb2013-05-02 14:29:32 -040083 inline bool has_separate_output (void) const { return info != out_info; }
84
Behdad Esfahbodf4a579b2011-07-25 16:20:16 -040085 unsigned int serial;
Behdad Esfahbod05207a72012-09-25 17:44:53 -040086
87 /* These reflect current allocations of the bytes in glyph_info_t's var1 and var2. */
Behdad Esfahbodf4a579b2011-07-25 16:20:16 -040088 uint8_t allocated_var_bytes[8];
89 const char *allocated_var_owner[8];
90
Behdad Esfahbod05207a72012-09-25 17:44:53 -040091 /* Text before / after the main buffer contents.
92 * Always in Unicode, and ordered outward.
93 * Index 0 is for "pre-context", 1 for "post-context". */
94 static const unsigned int CONTEXT_LENGTH = 5;
95 hb_codepoint_t context[2][CONTEXT_LENGTH];
96 unsigned int context_len[2];
97
Behdad Esfahbodd0316a82010-05-12 23:34:52 -040098
99 /* Methods */
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400100
101 HB_INTERNAL void reset (void);
Behdad Esfahbod82ecaff2012-11-13 13:57:52 -0800102 HB_INTERNAL void clear (void);
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400103
Behdad Esfahbodcc1a8a92011-01-06 14:58:52 -0500104 inline unsigned int backtrack_len (void) const
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400105 { return have_output? out_len : idx; }
Behdad Esfahbod6b65a762013-10-14 18:51:39 +0200106 inline unsigned int lookahead_len (void) const
107 { return len - idx; }
Behdad Esfahboddbf56b12010-10-27 17:06:12 -0400108 inline unsigned int next_serial (void) { return serial++; }
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400109
Behdad Esfahbodf4a579b2011-07-25 16:20:16 -0400110 HB_INTERNAL void allocate_var (unsigned int byte_i, unsigned int count, const char *owner);
111 HB_INTERNAL void deallocate_var (unsigned int byte_i, unsigned int count, const char *owner);
Behdad Esfahbod965c2802012-08-29 13:59:16 -0400112 HB_INTERNAL void assert_var (unsigned int byte_i, unsigned int count, const char *owner);
Behdad Esfahboda9ad3d32011-07-28 15:42:18 -0400113 HB_INTERNAL void deallocate_var_all (void);
Behdad Esfahbodf4a579b2011-07-25 16:20:16 -0400114
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400115 HB_INTERNAL void add (hb_codepoint_t codepoint,
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400116 unsigned int cluster);
Behdad Esfahbod847794e2013-02-27 17:59:28 -0500117 HB_INTERNAL void add_info (const hb_glyph_info_t &glyph_info);
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400118
119 HB_INTERNAL void reverse_range (unsigned int start, unsigned int end);
120 HB_INTERNAL void reverse (void);
121 HB_INTERNAL void reverse_clusters (void);
Behdad Esfahbod3f82f8f2012-11-15 18:45:31 -0800122 HB_INTERNAL void guess_segment_properties (void);
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400123
124 HB_INTERNAL void swap_buffers (void);
Behdad Esfahbod0bc7a382012-10-29 22:02:45 -0700125 HB_INTERNAL void remove_output (void);
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400126 HB_INTERNAL void clear_output (void);
127 HB_INTERNAL void clear_positions (void);
Behdad Esfahbod69cc4922012-07-31 14:51:36 -0400128
Behdad Esfahbod9ebe8c02011-08-26 09:29:42 +0200129 HB_INTERNAL void replace_glyphs (unsigned int num_in,
130 unsigned int num_out,
Behdad Esfahbod8e3715f2012-04-23 22:18:54 -0400131 const hb_codepoint_t *glyph_data);
Behdad Esfahbod69cc4922012-07-31 14:51:36 -0400132
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400133 HB_INTERNAL void replace_glyph (hb_codepoint_t glyph_index);
134 /* Makes a copy of the glyph at idx to output and replace glyph_index */
135 HB_INTERNAL void output_glyph (hb_codepoint_t glyph_index);
Behdad Esfahbod847794e2013-02-27 17:59:28 -0500136 HB_INTERNAL void output_info (const hb_glyph_info_t &glyph_info);
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400137 /* Copies glyph at idx to output but doesn't advance idx */
138 HB_INTERNAL void copy_glyph (void);
Behdad Esfahbod6b65a762013-10-14 18:51:39 +0200139 HB_INTERNAL bool move_to (unsigned int i); /* i is output-buffer index. */
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400140 /* Copies glyph at idx to output and advance idx.
141 * If there's no output, just advance idx. */
Behdad Esfahbod1be368e2012-08-31 16:29:17 -0400142 inline void
143 next_glyph (void)
144 {
145 if (have_output)
146 {
147 if (unlikely (out_info != info || out_len != idx)) {
148 if (unlikely (!make_room_for (1, 1))) return;
149 out_info[out_len] = info[idx];
150 }
151 out_len++;
152 }
153
154 idx++;
155 }
156
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400157 /* Advance idx without copying to output. */
158 inline void skip_glyph (void) { idx++; }
Behdad Esfahbod1ce7b872010-05-21 17:31:45 +0100159
Behdad Esfahbodbd7378b2010-10-13 18:33:16 -0400160 inline void reset_masks (hb_mask_t mask)
161 {
Behdad Esfahbod31f18ab2011-06-15 09:49:58 -0400162 for (unsigned int j = 0; j < len; j++)
163 info[j].mask = mask;
Behdad Esfahbodbd7378b2010-10-13 18:33:16 -0400164 }
165 inline void add_masks (hb_mask_t mask)
166 {
Behdad Esfahbod31f18ab2011-06-15 09:49:58 -0400167 for (unsigned int j = 0; j < len; j++)
168 info[j].mask |= mask;
Behdad Esfahbodbd7378b2010-10-13 18:33:16 -0400169 }
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400170 HB_INTERNAL void set_masks (hb_mask_t value,
171 hb_mask_t mask,
172 unsigned int cluster_start,
173 unsigned int cluster_end);
174
Behdad Esfahbodd4cc4472012-04-07 21:52:28 -0400175 HB_INTERNAL void merge_clusters (unsigned int start,
176 unsigned int end);
177 HB_INTERNAL void merge_out_clusters (unsigned int start,
178 unsigned int end);
179
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400180 /* Internal methods */
181 HB_INTERNAL bool enlarge (unsigned int size);
182
183 inline bool ensure (unsigned int size)
Behdad Esfahbodd16ccc42012-07-18 15:43:55 -0400184 { return likely (size < allocated) ? true : enlarge (size); }
Behdad Esfahbod468e9cb2011-07-22 11:28:07 -0400185
186 HB_INTERNAL bool make_room_for (unsigned int num_in, unsigned int num_out);
Behdad Esfahbod6b65a762013-10-14 18:51:39 +0200187 HB_INTERNAL bool shift_forward (unsigned int count);
Behdad Esfahbode62df432011-08-03 17:38:54 -0400188
189 HB_INTERNAL void *get_scratch_buffer (unsigned int *size);
Behdad Esfahbod05207a72012-09-25 17:44:53 -0400190
191 inline void clear_context (unsigned int side) { context_len[side] = 0; }
Behdad Esfahbodd0316a82010-05-12 23:34:52 -0400192};
193
194
Behdad Esfahbodb65c0602011-07-28 16:48:43 -0400195#define HB_BUFFER_XALLOCATE_VAR(b, func, var, owner) \
196 b->func (offsetof (hb_glyph_info_t, var) - offsetof(hb_glyph_info_t, var1), \
197 sizeof (b->info[0].var), owner)
198#define HB_BUFFER_ALLOCATE_VAR(b, var) \
199 HB_BUFFER_XALLOCATE_VAR (b, allocate_var, var (), #var)
200#define HB_BUFFER_DEALLOCATE_VAR(b, var) \
201 HB_BUFFER_XALLOCATE_VAR (b, deallocate_var, var (), #var)
Behdad Esfahbod965c2802012-08-29 13:59:16 -0400202#define HB_BUFFER_ASSERT_VAR(b, var) \
203 HB_BUFFER_XALLOCATE_VAR (b, assert_var, var (), #var)
Behdad Esfahbodb65c0602011-07-28 16:48:43 -0400204
205
Behdad Esfahbodda8edbb2010-06-09 07:15:39 -0400206#endif /* HB_BUFFER_PRIVATE_HH */