1 /*
2 * Copyright © 1998-2004 David Turner and Werner Lemberg
3 * Copyright © 2004,2007,2009,2010 Red Hat, Inc.
4 * Copyright © 2011,2012 Google, Inc.
5 *
6 * This is part of HarfBuzz, a text shaping library.
7 *
8 * 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.
13 *
14 * 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
27 * Google Author(s): Behdad Esfahbod
28 */
29
30 #include "hb-buffer-private.hh"
31 #include "hb-utf-private.hh"
32
33
34 #ifndef HB_DEBUG_BUFFER
35 #define HB_DEBUG_BUFFER (HB_DEBUG+0)
36 #endif
37
38
39 hb_bool_t
hb_segment_properties_equal(const hb_segment_properties_t * a,const hb_segment_properties_t * b)40 hb_segment_properties_equal (const hb_segment_properties_t *a,
41 const hb_segment_properties_t *b)
42 {
43 return a->direction == b->direction &&
44 a->script == b->script &&
45 a->language == b->language &&
46 a->reserved1 == b->reserved1 &&
47 a->reserved2 == b->reserved2;
48
49 }
50
51 unsigned int
hb_segment_properties_hash(const hb_segment_properties_t * p)52 hb_segment_properties_hash (const hb_segment_properties_t *p)
53 {
54 return (unsigned int) p->direction ^
55 (unsigned int) p->script ^
56 (intptr_t) (p->language);
57 }
58
59
60
61 /* Here is how the buffer works internally:
62 *
63 * There are two info pointers: info and out_info. They always have
64 * the same allocated size, but different lengths.
65 *
66 * As an optimization, both info and out_info may point to the
67 * same piece of memory, which is owned by info. This remains the
68 * case as long as out_len doesn't exceed i at any time.
69 * In that case, swap_buffers() is no-op and the glyph operations operate
70 * mostly in-place.
71 *
72 * As soon as out_info gets longer than info, out_info is moved over
73 * to an alternate buffer (which we reuse the pos buffer for!), and its
74 * current contents (out_len entries) are copied to the new place.
75 * This should all remain transparent to the user. swap_buffers() then
76 * switches info and out_info.
77 */
78
79
80
81 /* Internal API */
82
83 bool
enlarge(unsigned int size)84 hb_buffer_t::enlarge (unsigned int size)
85 {
86 if (unlikely (in_error))
87 return false;
88
89 unsigned int new_allocated = allocated;
90 hb_glyph_position_t *new_pos = NULL;
91 hb_glyph_info_t *new_info = NULL;
92 bool separate_out = out_info != info;
93
94 if (unlikely (_hb_unsigned_int_mul_overflows (size, sizeof (info[0]))))
95 goto done;
96
97 while (size >= new_allocated)
98 new_allocated += (new_allocated >> 1) + 32;
99
100 ASSERT_STATIC (sizeof (info[0]) == sizeof (pos[0]));
101 if (unlikely (_hb_unsigned_int_mul_overflows (new_allocated, sizeof (info[0]))))
102 goto done;
103
104 new_pos = (hb_glyph_position_t *) realloc (pos, new_allocated * sizeof (pos[0]));
105 new_info = (hb_glyph_info_t *) realloc (info, new_allocated * sizeof (info[0]));
106
107 done:
108 if (unlikely (!new_pos || !new_info))
109 in_error = true;
110
111 if (likely (new_pos))
112 pos = new_pos;
113
114 if (likely (new_info))
115 info = new_info;
116
117 out_info = separate_out ? (hb_glyph_info_t *) pos : info;
118 if (likely (!in_error))
119 allocated = new_allocated;
120
121 return likely (!in_error);
122 }
123
124 bool
make_room_for(unsigned int num_in,unsigned int num_out)125 hb_buffer_t::make_room_for (unsigned int num_in,
126 unsigned int num_out)
127 {
128 if (unlikely (!ensure (out_len + num_out))) return false;
129
130 if (out_info == info &&
131 out_len + num_out > idx + num_in)
132 {
133 assert (have_output);
134
135 out_info = (hb_glyph_info_t *) pos;
136 memcpy (out_info, info, out_len * sizeof (out_info[0]));
137 }
138
139 return true;
140 }
141
142 bool
shift_forward(unsigned int count)143 hb_buffer_t::shift_forward (unsigned int count)
144 {
145 assert (have_output);
146 if (unlikely (!ensure (len + count))) return false;
147
148 memmove (info + idx + count, info + idx, (len - idx) * sizeof (info[0]));
149 len += count;
150 idx += count;
151
152 return true;
153 }
154
155 hb_buffer_t::scratch_buffer_t *
get_scratch_buffer(unsigned int * size)156 hb_buffer_t::get_scratch_buffer (unsigned int *size)
157 {
158 have_output = false;
159 have_positions = false;
160
161 out_len = 0;
162 out_info = info;
163
164 assert ((uintptr_t) pos % sizeof (scratch_buffer_t) == 0);
165 *size = allocated * sizeof (pos[0]) / sizeof (scratch_buffer_t);
166 return (scratch_buffer_t *) (void *) pos;
167 }
168
169
170
171 /* HarfBuzz-Internal API */
172
173 void
reset(void)174 hb_buffer_t::reset (void)
175 {
176 if (unlikely (hb_object_is_inert (this)))
177 return;
178
179 hb_unicode_funcs_destroy (unicode);
180 unicode = hb_unicode_funcs_get_default ();
181
182 clear ();
183 }
184
185 void
clear(void)186 hb_buffer_t::clear (void)
187 {
188 if (unlikely (hb_object_is_inert (this)))
189 return;
190
191 hb_segment_properties_t default_props = HB_SEGMENT_PROPERTIES_DEFAULT;
192 props = default_props;
193 flags = HB_BUFFER_FLAG_DEFAULT;
194
195 content_type = HB_BUFFER_CONTENT_TYPE_INVALID;
196 in_error = false;
197 have_output = false;
198 have_positions = false;
199
200 idx = 0;
201 len = 0;
202 out_len = 0;
203 out_info = info;
204
205 serial = 0;
206 memset (allocated_var_bytes, 0, sizeof allocated_var_bytes);
207 memset (allocated_var_owner, 0, sizeof allocated_var_owner);
208
209 memset (context, 0, sizeof context);
210 memset (context_len, 0, sizeof context_len);
211 }
212
213 void
add(hb_codepoint_t codepoint,unsigned int cluster)214 hb_buffer_t::add (hb_codepoint_t codepoint,
215 unsigned int cluster)
216 {
217 hb_glyph_info_t *glyph;
218
219 if (unlikely (!ensure (len + 1))) return;
220
221 glyph = &info[len];
222
223 memset (glyph, 0, sizeof (*glyph));
224 glyph->codepoint = codepoint;
225 glyph->mask = 1;
226 glyph->cluster = cluster;
227
228 len++;
229 }
230
231 void
add_info(const hb_glyph_info_t & glyph_info)232 hb_buffer_t::add_info (const hb_glyph_info_t &glyph_info)
233 {
234 if (unlikely (!ensure (len + 1))) return;
235
236 info[len] = glyph_info;
237
238 len++;
239 }
240
241
242 void
remove_output(void)243 hb_buffer_t::remove_output (void)
244 {
245 if (unlikely (hb_object_is_inert (this)))
246 return;
247
248 have_output = false;
249 have_positions = false;
250
251 out_len = 0;
252 out_info = info;
253 }
254
255 void
clear_output(void)256 hb_buffer_t::clear_output (void)
257 {
258 if (unlikely (hb_object_is_inert (this)))
259 return;
260
261 have_output = true;
262 have_positions = false;
263
264 out_len = 0;
265 out_info = info;
266 }
267
268 void
clear_positions(void)269 hb_buffer_t::clear_positions (void)
270 {
271 if (unlikely (hb_object_is_inert (this)))
272 return;
273
274 have_output = false;
275 have_positions = true;
276
277 out_len = 0;
278 out_info = info;
279
280 memset (pos, 0, sizeof (pos[0]) * len);
281 }
282
283 void
swap_buffers(void)284 hb_buffer_t::swap_buffers (void)
285 {
286 if (unlikely (in_error)) return;
287
288 assert (have_output);
289 have_output = false;
290
291 if (out_info != info)
292 {
293 hb_glyph_info_t *tmp_string;
294 tmp_string = info;
295 info = out_info;
296 out_info = tmp_string;
297 pos = (hb_glyph_position_t *) out_info;
298 }
299
300 unsigned int tmp;
301 tmp = len;
302 len = out_len;
303 out_len = tmp;
304
305 idx = 0;
306 }
307
308
309 void
replace_glyphs(unsigned int num_in,unsigned int num_out,const uint32_t * glyph_data)310 hb_buffer_t::replace_glyphs (unsigned int num_in,
311 unsigned int num_out,
312 const uint32_t *glyph_data)
313 {
314 if (unlikely (!make_room_for (num_in, num_out))) return;
315
316 merge_clusters (idx, idx + num_in);
317
318 hb_glyph_info_t orig_info = info[idx];
319 hb_glyph_info_t *pinfo = &out_info[out_len];
320 for (unsigned int i = 0; i < num_out; i++)
321 {
322 *pinfo = orig_info;
323 pinfo->codepoint = glyph_data[i];
324 pinfo++;
325 }
326
327 idx += num_in;
328 out_len += num_out;
329 }
330
331 void
output_glyph(hb_codepoint_t glyph_index)332 hb_buffer_t::output_glyph (hb_codepoint_t glyph_index)
333 {
334 if (unlikely (!make_room_for (0, 1))) return;
335
336 out_info[out_len] = info[idx];
337 out_info[out_len].codepoint = glyph_index;
338
339 out_len++;
340 }
341
342 void
output_info(const hb_glyph_info_t & glyph_info)343 hb_buffer_t::output_info (const hb_glyph_info_t &glyph_info)
344 {
345 if (unlikely (!make_room_for (0, 1))) return;
346
347 out_info[out_len] = glyph_info;
348
349 out_len++;
350 }
351
352 void
copy_glyph(void)353 hb_buffer_t::copy_glyph (void)
354 {
355 if (unlikely (!make_room_for (0, 1))) return;
356
357 out_info[out_len] = info[idx];
358
359 out_len++;
360 }
361
362 bool
move_to(unsigned int i)363 hb_buffer_t::move_to (unsigned int i)
364 {
365 if (!have_output)
366 {
367 assert (i <= len);
368 idx = i;
369 return true;
370 }
371
372 assert (i <= out_len + (len - idx));
373
374 if (out_len < i)
375 {
376 unsigned int count = i - out_len;
377 if (unlikely (!make_room_for (count, count))) return false;
378
379 memmove (out_info + out_len, info + idx, count * sizeof (out_info[0]));
380 idx += count;
381 out_len += count;
382 }
383 else if (out_len > i)
384 {
385 /* Tricky part: rewinding... */
386 unsigned int count = out_len - i;
387
388 if (unlikely (idx < count && !shift_forward (count + 32))) return false;
389
390 assert (idx >= count);
391
392 idx -= count;
393 out_len -= count;
394 memmove (info + idx, out_info + out_len, count * sizeof (out_info[0]));
395 }
396
397 return true;
398 }
399
400 void
replace_glyph(hb_codepoint_t glyph_index)401 hb_buffer_t::replace_glyph (hb_codepoint_t glyph_index)
402 {
403 if (unlikely (out_info != info || out_len != idx)) {
404 if (unlikely (!make_room_for (1, 1))) return;
405 out_info[out_len] = info[idx];
406 }
407 out_info[out_len].codepoint = glyph_index;
408
409 idx++;
410 out_len++;
411 }
412
413
414 void
set_masks(hb_mask_t value,hb_mask_t mask,unsigned int cluster_start,unsigned int cluster_end)415 hb_buffer_t::set_masks (hb_mask_t value,
416 hb_mask_t mask,
417 unsigned int cluster_start,
418 unsigned int cluster_end)
419 {
420 hb_mask_t not_mask = ~mask;
421 value &= mask;
422
423 if (!mask)
424 return;
425
426 if (cluster_start == 0 && cluster_end == (unsigned int)-1) {
427 unsigned int count = len;
428 for (unsigned int i = 0; i < count; i++)
429 info[i].mask = (info[i].mask & not_mask) | value;
430 return;
431 }
432
433 unsigned int count = len;
434 for (unsigned int i = 0; i < count; i++)
435 if (cluster_start <= info[i].cluster && info[i].cluster < cluster_end)
436 info[i].mask = (info[i].mask & not_mask) | value;
437 }
438
439 void
reverse_range(unsigned int start,unsigned int end)440 hb_buffer_t::reverse_range (unsigned int start,
441 unsigned int end)
442 {
443 unsigned int i, j;
444
445 if (start == end - 1)
446 return;
447
448 for (i = start, j = end - 1; i < j; i++, j--) {
449 hb_glyph_info_t t;
450
451 t = info[i];
452 info[i] = info[j];
453 info[j] = t;
454 }
455
456 if (pos) {
457 for (i = start, j = end - 1; i < j; i++, j--) {
458 hb_glyph_position_t t;
459
460 t = pos[i];
461 pos[i] = pos[j];
462 pos[j] = t;
463 }
464 }
465 }
466
467 void
reverse(void)468 hb_buffer_t::reverse (void)
469 {
470 if (unlikely (!len))
471 return;
472
473 reverse_range (0, len);
474 }
475
476 void
reverse_clusters(void)477 hb_buffer_t::reverse_clusters (void)
478 {
479 unsigned int i, start, count, last_cluster;
480
481 if (unlikely (!len))
482 return;
483
484 reverse ();
485
486 count = len;
487 start = 0;
488 last_cluster = info[0].cluster;
489 for (i = 1; i < count; i++) {
490 if (last_cluster != info[i].cluster) {
491 reverse_range (start, i);
492 start = i;
493 last_cluster = info[i].cluster;
494 }
495 }
496 reverse_range (start, i);
497 }
498
499 void
merge_clusters(unsigned int start,unsigned int end)500 hb_buffer_t::merge_clusters (unsigned int start,
501 unsigned int end)
502 {
503 if (unlikely (end - start < 2))
504 return;
505
506 unsigned int cluster = info[start].cluster;
507
508 for (unsigned int i = start + 1; i < end; i++)
509 cluster = MIN (cluster, info[i].cluster);
510
511 /* Extend end */
512 while (end < len && info[end - 1].cluster == info[end].cluster)
513 end++;
514
515 /* Extend start */
516 while (idx < start && info[start - 1].cluster == info[start].cluster)
517 start--;
518
519 /* If we hit the start of buffer, continue in out-buffer. */
520 if (idx == start)
521 for (unsigned i = out_len; i && out_info[i - 1].cluster == info[start].cluster; i--)
522 out_info[i - 1].cluster = cluster;
523
524 for (unsigned int i = start; i < end; i++)
525 info[i].cluster = cluster;
526 }
527 void
merge_out_clusters(unsigned int start,unsigned int end)528 hb_buffer_t::merge_out_clusters (unsigned int start,
529 unsigned int end)
530 {
531 if (unlikely (end - start < 2))
532 return;
533
534 unsigned int cluster = out_info[start].cluster;
535
536 for (unsigned int i = start + 1; i < end; i++)
537 cluster = MIN (cluster, out_info[i].cluster);
538
539 /* Extend start */
540 while (start && out_info[start - 1].cluster == out_info[start].cluster)
541 start--;
542
543 /* Extend end */
544 while (end < out_len && out_info[end - 1].cluster == out_info[end].cluster)
545 end++;
546
547 /* If we hit the end of out-buffer, continue in buffer. */
548 if (end == out_len)
549 for (unsigned i = idx; i < len && info[i].cluster == out_info[end - 1].cluster; i++)
550 info[i].cluster = cluster;
551
552 for (unsigned int i = start; i < end; i++)
553 out_info[i].cluster = cluster;
554 }
555
556 void
guess_segment_properties(void)557 hb_buffer_t::guess_segment_properties (void)
558 {
559 assert (content_type == HB_BUFFER_CONTENT_TYPE_UNICODE ||
560 (!len && content_type == HB_BUFFER_CONTENT_TYPE_INVALID));
561
562 /* If script is set to INVALID, guess from buffer contents */
563 if (props.script == HB_SCRIPT_INVALID) {
564 for (unsigned int i = 0; i < len; i++) {
565 hb_script_t script = unicode->script (info[i].codepoint);
566 if (likely (script != HB_SCRIPT_COMMON &&
567 script != HB_SCRIPT_INHERITED &&
568 script != HB_SCRIPT_UNKNOWN)) {
569 props.script = script;
570 break;
571 }
572 }
573 }
574
575 /* If direction is set to INVALID, guess from script */
576 if (props.direction == HB_DIRECTION_INVALID) {
577 props.direction = hb_script_get_horizontal_direction (props.script);
578 }
579
580 /* If language is not set, use default language from locale */
581 if (props.language == HB_LANGUAGE_INVALID) {
582 /* TODO get_default_for_script? using $LANGUAGE */
583 props.language = hb_language_get_default ();
584 }
585 }
586
587
588 static inline void
dump_var_allocation(const hb_buffer_t * buffer)589 dump_var_allocation (const hb_buffer_t *buffer)
590 {
591 char buf[80];
592 for (unsigned int i = 0; i < 8; i++)
593 buf[i] = '0' + buffer->allocated_var_bytes[7 - i];
594 buf[8] = '\0';
595 DEBUG_MSG (BUFFER, buffer,
596 "Current var allocation: %s",
597 buf);
598 }
599
allocate_var(unsigned int byte_i,unsigned int count,const char * owner)600 void hb_buffer_t::allocate_var (unsigned int byte_i, unsigned int count, const char *owner)
601 {
602 assert (byte_i < 8 && byte_i + count <= 8);
603
604 if (DEBUG_ENABLED (BUFFER))
605 dump_var_allocation (this);
606 DEBUG_MSG (BUFFER, this,
607 "Allocating var bytes %d..%d for %s",
608 byte_i, byte_i + count - 1, owner);
609
610 for (unsigned int i = byte_i; i < byte_i + count; i++) {
611 assert (!allocated_var_bytes[i]);
612 allocated_var_bytes[i]++;
613 allocated_var_owner[i] = owner;
614 }
615 }
616
deallocate_var(unsigned int byte_i,unsigned int count,const char * owner)617 void hb_buffer_t::deallocate_var (unsigned int byte_i, unsigned int count, const char *owner)
618 {
619 if (DEBUG_ENABLED (BUFFER))
620 dump_var_allocation (this);
621
622 DEBUG_MSG (BUFFER, this,
623 "Deallocating var bytes %d..%d for %s",
624 byte_i, byte_i + count - 1, owner);
625
626 assert (byte_i < 8 && byte_i + count <= 8);
627 for (unsigned int i = byte_i; i < byte_i + count; i++) {
628 assert (allocated_var_bytes[i]);
629 assert (0 == strcmp (allocated_var_owner[i], owner));
630 allocated_var_bytes[i]--;
631 }
632 }
633
assert_var(unsigned int byte_i,unsigned int count,const char * owner)634 void hb_buffer_t::assert_var (unsigned int byte_i, unsigned int count, const char *owner)
635 {
636 if (DEBUG_ENABLED (BUFFER))
637 dump_var_allocation (this);
638
639 DEBUG_MSG (BUFFER, this,
640 "Asserting var bytes %d..%d for %s",
641 byte_i, byte_i + count - 1, owner);
642
643 assert (byte_i < 8 && byte_i + count <= 8);
644 for (unsigned int i = byte_i; i < byte_i + count; i++) {
645 assert (allocated_var_bytes[i]);
646 assert (0 == strcmp (allocated_var_owner[i], owner));
647 }
648 }
649
deallocate_var_all(void)650 void hb_buffer_t::deallocate_var_all (void)
651 {
652 memset (allocated_var_bytes, 0, sizeof (allocated_var_bytes));
653 memset (allocated_var_owner, 0, sizeof (allocated_var_owner));
654 }
655
656 /* Public API */
657
658 /**
659 * hb_buffer_create: (Xconstructor)
660 *
661 *
662 *
663 * Return value: (transfer full)
664 *
665 * Since: 1.0
666 **/
667 hb_buffer_t *
hb_buffer_create(void)668 hb_buffer_create (void)
669 {
670 hb_buffer_t *buffer;
671
672 if (!(buffer = hb_object_create<hb_buffer_t> ()))
673 return hb_buffer_get_empty ();
674
675 buffer->reset ();
676
677 return buffer;
678 }
679
680 /**
681 * hb_buffer_get_empty:
682 *
683 *
684 *
685 * Return value: (transfer full):
686 *
687 * Since: 1.0
688 **/
689 hb_buffer_t *
hb_buffer_get_empty(void)690 hb_buffer_get_empty (void)
691 {
692 static const hb_buffer_t _hb_buffer_nil = {
693 HB_OBJECT_HEADER_STATIC,
694
695 const_cast<hb_unicode_funcs_t *> (&_hb_unicode_funcs_nil),
696 HB_SEGMENT_PROPERTIES_DEFAULT,
697 HB_BUFFER_FLAG_DEFAULT,
698
699 HB_BUFFER_CONTENT_TYPE_INVALID,
700 true, /* in_error */
701 true, /* have_output */
702 true /* have_positions */
703
704 /* Zero is good enough for everything else. */
705 };
706
707 return const_cast<hb_buffer_t *> (&_hb_buffer_nil);
708 }
709
710 /**
711 * hb_buffer_reference: (skip)
712 * @buffer: a buffer.
713 *
714 *
715 *
716 * Return value: (transfer full):
717 *
718 * Since: 1.0
719 **/
720 hb_buffer_t *
hb_buffer_reference(hb_buffer_t * buffer)721 hb_buffer_reference (hb_buffer_t *buffer)
722 {
723 return hb_object_reference (buffer);
724 }
725
726 /**
727 * hb_buffer_destroy: (skip)
728 * @buffer: a buffer.
729 *
730 *
731 *
732 * Since: 1.0
733 **/
734 void
hb_buffer_destroy(hb_buffer_t * buffer)735 hb_buffer_destroy (hb_buffer_t *buffer)
736 {
737 if (!hb_object_destroy (buffer)) return;
738
739 hb_unicode_funcs_destroy (buffer->unicode);
740
741 free (buffer->info);
742 free (buffer->pos);
743
744 free (buffer);
745 }
746
747 /**
748 * hb_buffer_set_user_data: (skip)
749 * @buffer: a buffer.
750 * @key:
751 * @data:
752 * @destroy:
753 * @replace:
754 *
755 *
756 *
757 * Return value:
758 *
759 * Since: 1.0
760 **/
761 hb_bool_t
hb_buffer_set_user_data(hb_buffer_t * buffer,hb_user_data_key_t * key,void * data,hb_destroy_func_t destroy,hb_bool_t replace)762 hb_buffer_set_user_data (hb_buffer_t *buffer,
763 hb_user_data_key_t *key,
764 void * data,
765 hb_destroy_func_t destroy,
766 hb_bool_t replace)
767 {
768 return hb_object_set_user_data (buffer, key, data, destroy, replace);
769 }
770
771 /**
772 * hb_buffer_get_user_data: (skip)
773 * @buffer: a buffer.
774 * @key:
775 *
776 *
777 *
778 * Return value:
779 *
780 * Since: 1.0
781 **/
782 void *
hb_buffer_get_user_data(hb_buffer_t * buffer,hb_user_data_key_t * key)783 hb_buffer_get_user_data (hb_buffer_t *buffer,
784 hb_user_data_key_t *key)
785 {
786 return hb_object_get_user_data (buffer, key);
787 }
788
789
790 /**
791 * hb_buffer_set_content_type:
792 * @buffer: a buffer.
793 * @content_type:
794 *
795 *
796 *
797 * Since: 1.0
798 **/
799 void
hb_buffer_set_content_type(hb_buffer_t * buffer,hb_buffer_content_type_t content_type)800 hb_buffer_set_content_type (hb_buffer_t *buffer,
801 hb_buffer_content_type_t content_type)
802 {
803 buffer->content_type = content_type;
804 }
805
806 /**
807 * hb_buffer_get_content_type:
808 * @buffer: a buffer.
809 *
810 *
811 *
812 * Return value:
813 *
814 * Since: 1.0
815 **/
816 hb_buffer_content_type_t
hb_buffer_get_content_type(hb_buffer_t * buffer)817 hb_buffer_get_content_type (hb_buffer_t *buffer)
818 {
819 return buffer->content_type;
820 }
821
822
823 /**
824 * hb_buffer_set_unicode_funcs:
825 * @buffer: a buffer.
826 * @unicode_funcs:
827 *
828 *
829 *
830 * Since: 1.0
831 **/
832 void
hb_buffer_set_unicode_funcs(hb_buffer_t * buffer,hb_unicode_funcs_t * unicode_funcs)833 hb_buffer_set_unicode_funcs (hb_buffer_t *buffer,
834 hb_unicode_funcs_t *unicode_funcs)
835 {
836 if (unlikely (hb_object_is_inert (buffer)))
837 return;
838
839 if (!unicode_funcs)
840 unicode_funcs = hb_unicode_funcs_get_default ();
841
842
843 hb_unicode_funcs_reference (unicode_funcs);
844 hb_unicode_funcs_destroy (buffer->unicode);
845 buffer->unicode = unicode_funcs;
846 }
847
848 /**
849 * hb_buffer_get_unicode_funcs:
850 * @buffer: a buffer.
851 *
852 *
853 *
854 * Return value:
855 *
856 * Since: 1.0
857 **/
858 hb_unicode_funcs_t *
hb_buffer_get_unicode_funcs(hb_buffer_t * buffer)859 hb_buffer_get_unicode_funcs (hb_buffer_t *buffer)
860 {
861 return buffer->unicode;
862 }
863
864 /**
865 * hb_buffer_set_direction:
866 * @buffer: a buffer.
867 * @direction:
868 *
869 *
870 *
871 * Since: 1.0
872 **/
873 void
hb_buffer_set_direction(hb_buffer_t * buffer,hb_direction_t direction)874 hb_buffer_set_direction (hb_buffer_t *buffer,
875 hb_direction_t direction)
876
877 {
878 if (unlikely (hb_object_is_inert (buffer)))
879 return;
880
881 buffer->props.direction = direction;
882 }
883
884 /**
885 * hb_buffer_get_direction:
886 * @buffer: a buffer.
887 *
888 *
889 *
890 * Return value:
891 *
892 * Since: 1.0
893 **/
894 hb_direction_t
hb_buffer_get_direction(hb_buffer_t * buffer)895 hb_buffer_get_direction (hb_buffer_t *buffer)
896 {
897 return buffer->props.direction;
898 }
899
900 /**
901 * hb_buffer_set_script:
902 * @buffer: a buffer.
903 * @script:
904 *
905 *
906 *
907 * Since: 1.0
908 **/
909 void
hb_buffer_set_script(hb_buffer_t * buffer,hb_script_t script)910 hb_buffer_set_script (hb_buffer_t *buffer,
911 hb_script_t script)
912 {
913 if (unlikely (hb_object_is_inert (buffer)))
914 return;
915
916 buffer->props.script = script;
917 }
918
919 /**
920 * hb_buffer_get_script:
921 * @buffer: a buffer.
922 *
923 *
924 *
925 * Return value:
926 *
927 * Since: 1.0
928 **/
929 hb_script_t
hb_buffer_get_script(hb_buffer_t * buffer)930 hb_buffer_get_script (hb_buffer_t *buffer)
931 {
932 return buffer->props.script;
933 }
934
935 /**
936 * hb_buffer_set_language:
937 * @buffer: a buffer.
938 * @language:
939 *
940 *
941 *
942 * Since: 1.0
943 **/
944 void
hb_buffer_set_language(hb_buffer_t * buffer,hb_language_t language)945 hb_buffer_set_language (hb_buffer_t *buffer,
946 hb_language_t language)
947 {
948 if (unlikely (hb_object_is_inert (buffer)))
949 return;
950
951 buffer->props.language = language;
952 }
953
954 /**
955 * hb_buffer_get_language:
956 * @buffer: a buffer.
957 *
958 *
959 *
960 * Return value:
961 *
962 * Since: 1.0
963 **/
964 hb_language_t
hb_buffer_get_language(hb_buffer_t * buffer)965 hb_buffer_get_language (hb_buffer_t *buffer)
966 {
967 return buffer->props.language;
968 }
969
970 /**
971 * hb_buffer_set_segment_properties:
972 * @buffer: a buffer.
973 * @props:
974 *
975 *
976 *
977 * Since: 1.0
978 **/
979 void
hb_buffer_set_segment_properties(hb_buffer_t * buffer,const hb_segment_properties_t * props)980 hb_buffer_set_segment_properties (hb_buffer_t *buffer,
981 const hb_segment_properties_t *props)
982 {
983 if (unlikely (hb_object_is_inert (buffer)))
984 return;
985
986 buffer->props = *props;
987 }
988
989 /**
990 * hb_buffer_get_segment_properties:
991 * @buffer: a buffer.
992 * @props:
993 *
994 *
995 *
996 * Since: 1.0
997 **/
998 void
hb_buffer_get_segment_properties(hb_buffer_t * buffer,hb_segment_properties_t * props)999 hb_buffer_get_segment_properties (hb_buffer_t *buffer,
1000 hb_segment_properties_t *props)
1001 {
1002 *props = buffer->props;
1003 }
1004
1005
1006 /**
1007 * hb_buffer_set_flags:
1008 * @buffer: a buffer.
1009 * @flags:
1010 *
1011 *
1012 *
1013 * Since: 1.0
1014 **/
1015 void
hb_buffer_set_flags(hb_buffer_t * buffer,hb_buffer_flags_t flags)1016 hb_buffer_set_flags (hb_buffer_t *buffer,
1017 hb_buffer_flags_t flags)
1018 {
1019 if (unlikely (hb_object_is_inert (buffer)))
1020 return;
1021
1022 buffer->flags = flags;
1023 }
1024
1025 /**
1026 * hb_buffer_get_flags:
1027 * @buffer: a buffer.
1028 *
1029 *
1030 *
1031 * Return value:
1032 *
1033 * Since: 1.0
1034 **/
1035 hb_buffer_flags_t
hb_buffer_get_flags(hb_buffer_t * buffer)1036 hb_buffer_get_flags (hb_buffer_t *buffer)
1037 {
1038 return buffer->flags;
1039 }
1040
1041
1042 /**
1043 * hb_buffer_reset:
1044 * @buffer: a buffer.
1045 *
1046 *
1047 *
1048 * Since: 1.0
1049 **/
1050 void
hb_buffer_reset(hb_buffer_t * buffer)1051 hb_buffer_reset (hb_buffer_t *buffer)
1052 {
1053 buffer->reset ();
1054 }
1055
1056 /**
1057 * hb_buffer_clear_contents:
1058 * @buffer: a buffer.
1059 *
1060 *
1061 *
1062 * Since: 1.0
1063 **/
1064 void
hb_buffer_clear_contents(hb_buffer_t * buffer)1065 hb_buffer_clear_contents (hb_buffer_t *buffer)
1066 {
1067 buffer->clear ();
1068 }
1069
1070 /**
1071 * hb_buffer_pre_allocate:
1072 * @buffer: a buffer.
1073 * @size:
1074 *
1075 *
1076 *
1077 * Return value:
1078 *
1079 * Since: 1.0
1080 **/
1081 hb_bool_t
hb_buffer_pre_allocate(hb_buffer_t * buffer,unsigned int size)1082 hb_buffer_pre_allocate (hb_buffer_t *buffer, unsigned int size)
1083 {
1084 return buffer->ensure (size);
1085 }
1086
1087 /**
1088 * hb_buffer_allocation_successful:
1089 * @buffer: a buffer.
1090 *
1091 *
1092 *
1093 * Return value:
1094 *
1095 * Since: 1.0
1096 **/
1097 hb_bool_t
hb_buffer_allocation_successful(hb_buffer_t * buffer)1098 hb_buffer_allocation_successful (hb_buffer_t *buffer)
1099 {
1100 return !buffer->in_error;
1101 }
1102
1103 /**
1104 * hb_buffer_add:
1105 * @buffer: a buffer.
1106 * @codepoint:
1107 * @cluster:
1108 *
1109 *
1110 *
1111 * Since: 1.0
1112 **/
1113 void
hb_buffer_add(hb_buffer_t * buffer,hb_codepoint_t codepoint,unsigned int cluster)1114 hb_buffer_add (hb_buffer_t *buffer,
1115 hb_codepoint_t codepoint,
1116 unsigned int cluster)
1117 {
1118 buffer->add (codepoint, cluster);
1119 buffer->clear_context (1);
1120 }
1121
1122 /**
1123 * hb_buffer_set_length:
1124 * @buffer: a buffer.
1125 * @length:
1126 *
1127 *
1128 *
1129 * Return value:
1130 *
1131 * Since: 1.0
1132 **/
1133 hb_bool_t
hb_buffer_set_length(hb_buffer_t * buffer,unsigned int length)1134 hb_buffer_set_length (hb_buffer_t *buffer,
1135 unsigned int length)
1136 {
1137 if (unlikely (hb_object_is_inert (buffer)))
1138 return length == 0;
1139
1140 if (!buffer->ensure (length))
1141 return false;
1142
1143 /* Wipe the new space */
1144 if (length > buffer->len) {
1145 memset (buffer->info + buffer->len, 0, sizeof (buffer->info[0]) * (length - buffer->len));
1146 if (buffer->have_positions)
1147 memset (buffer->pos + buffer->len, 0, sizeof (buffer->pos[0]) * (length - buffer->len));
1148 }
1149
1150 buffer->len = length;
1151
1152 if (!length)
1153 buffer->clear_context (0);
1154 buffer->clear_context (1);
1155
1156 return true;
1157 }
1158
1159 /**
1160 * hb_buffer_get_length:
1161 * @buffer: a buffer.
1162 *
1163 * Returns the number of items in the buffer.
1164 *
1165 * Return value: buffer length.
1166 *
1167 * Since: 1.0
1168 **/
1169 unsigned int
hb_buffer_get_length(hb_buffer_t * buffer)1170 hb_buffer_get_length (hb_buffer_t *buffer)
1171 {
1172 return buffer->len;
1173 }
1174
1175 /**
1176 * hb_buffer_get_glyph_infos:
1177 * @buffer: a buffer.
1178 * @length: (out): output array length.
1179 *
1180 * Returns buffer glyph information array. Returned pointer
1181 * is valid as long as buffer contents are not modified.
1182 *
1183 * Return value: (transfer none) (array length=length): buffer glyph information array.
1184 *
1185 * Since: 1.0
1186 **/
1187 hb_glyph_info_t *
hb_buffer_get_glyph_infos(hb_buffer_t * buffer,unsigned int * length)1188 hb_buffer_get_glyph_infos (hb_buffer_t *buffer,
1189 unsigned int *length)
1190 {
1191 if (length)
1192 *length = buffer->len;
1193
1194 return (hb_glyph_info_t *) buffer->info;
1195 }
1196
1197 /**
1198 * hb_buffer_get_glyph_positions:
1199 * @buffer: a buffer.
1200 * @length: (out): output length.
1201 *
1202 * Returns buffer glyph position array. Returned pointer
1203 * is valid as long as buffer contents are not modified.
1204 *
1205 * Return value: (transfer none) (array length=length): buffer glyph position array.
1206 *
1207 * Since: 1.0
1208 **/
1209 hb_glyph_position_t *
hb_buffer_get_glyph_positions(hb_buffer_t * buffer,unsigned int * length)1210 hb_buffer_get_glyph_positions (hb_buffer_t *buffer,
1211 unsigned int *length)
1212 {
1213 if (!buffer->have_positions)
1214 buffer->clear_positions ();
1215
1216 if (length)
1217 *length = buffer->len;
1218
1219 return (hb_glyph_position_t *) buffer->pos;
1220 }
1221
1222 /**
1223 * hb_buffer_reverse:
1224 * @buffer: a buffer.
1225 *
1226 * Reverses buffer contents.
1227 *
1228 * Since: 1.0
1229 **/
1230 void
hb_buffer_reverse(hb_buffer_t * buffer)1231 hb_buffer_reverse (hb_buffer_t *buffer)
1232 {
1233 buffer->reverse ();
1234 }
1235
1236 /**
1237 * hb_buffer_reverse_clusters:
1238 * @buffer: a buffer.
1239 *
1240 * Reverses buffer clusters. That is, the buffer contents are
1241 * reversed, then each cluster (consecutive items having the
1242 * same cluster number) are reversed again.
1243 *
1244 * Since: 1.0
1245 **/
1246 void
hb_buffer_reverse_clusters(hb_buffer_t * buffer)1247 hb_buffer_reverse_clusters (hb_buffer_t *buffer)
1248 {
1249 buffer->reverse_clusters ();
1250 }
1251
1252 /**
1253 * hb_buffer_guess_segment_properties:
1254 * @buffer: a buffer.
1255 *
1256 * Sets unset buffer segment properties based on buffer Unicode
1257 * contents. If buffer is not empty, it must have content type
1258 * %HB_BUFFER_CONTENT_TYPE_UNICODE.
1259 *
1260 * If buffer script is not set (ie. is %HB_SCRIPT_INVALID), it
1261 * will be set to the Unicode script of the first character in
1262 * the buffer that has a script other than %HB_SCRIPT_COMMON,
1263 * %HB_SCRIPT_INHERITED, and %HB_SCRIPT_UNKNOWN.
1264 *
1265 * Next, if buffer direction is not set (ie. is %HB_DIRECTION_INVALID),
1266 * it will be set to the natural horizontal direction of the
1267 * buffer script as returned by hb_script_get_horizontal_direction().
1268 *
1269 * Finally, if buffer language is not set (ie. is %HB_LANGUAGE_INVALID),
1270 * it will be set to the process's default language as returned by
1271 * hb_language_get_default(). This may change in the future by
1272 * taking buffer script into consideration when choosing a language.
1273 *
1274 * Since: 1.0
1275 **/
1276 void
hb_buffer_guess_segment_properties(hb_buffer_t * buffer)1277 hb_buffer_guess_segment_properties (hb_buffer_t *buffer)
1278 {
1279 buffer->guess_segment_properties ();
1280 }
1281
1282 template <typename T>
1283 static inline void
hb_buffer_add_utf(hb_buffer_t * buffer,const T * text,int text_length,unsigned int item_offset,int item_length)1284 hb_buffer_add_utf (hb_buffer_t *buffer,
1285 const T *text,
1286 int text_length,
1287 unsigned int item_offset,
1288 int item_length)
1289 {
1290 assert (buffer->content_type == HB_BUFFER_CONTENT_TYPE_UNICODE ||
1291 (!buffer->len && buffer->content_type == HB_BUFFER_CONTENT_TYPE_INVALID));
1292
1293 if (unlikely (hb_object_is_inert (buffer)))
1294 return;
1295
1296 if (text_length == -1)
1297 text_length = hb_utf_strlen (text);
1298
1299 if (item_length == -1)
1300 item_length = text_length - item_offset;
1301
1302 buffer->ensure (buffer->len + item_length * sizeof (T) / 4);
1303
1304 /* If buffer is empty and pre-context provided, install it.
1305 * This check is written this way, to make sure people can
1306 * provide pre-context in one add_utf() call, then provide
1307 * text in a follow-up call. See:
1308 *
1309 * https://bugzilla.mozilla.org/show_bug.cgi?id=801410#c13
1310 */
1311 if (!buffer->len && item_offset > 0)
1312 {
1313 /* Add pre-context */
1314 buffer->clear_context (0);
1315 const T *prev = text + item_offset;
1316 const T *start = text;
1317 while (start < prev && buffer->context_len[0] < buffer->CONTEXT_LENGTH)
1318 {
1319 hb_codepoint_t u;
1320 prev = hb_utf_prev (prev, start, &u);
1321 buffer->context[0][buffer->context_len[0]++] = u;
1322 }
1323 }
1324
1325 const T *next = text + item_offset;
1326 const T *end = next + item_length;
1327 while (next < end)
1328 {
1329 hb_codepoint_t u;
1330 const T *old_next = next;
1331 next = hb_utf_next (next, end, &u);
1332 buffer->add (u, old_next - (const T *) text);
1333 }
1334
1335 /* Add post-context */
1336 buffer->clear_context (1);
1337 end = text + text_length;
1338 while (next < end && buffer->context_len[1] < buffer->CONTEXT_LENGTH)
1339 {
1340 hb_codepoint_t u;
1341 next = hb_utf_next (next, end, &u);
1342 buffer->context[1][buffer->context_len[1]++] = u;
1343 }
1344
1345 buffer->content_type = HB_BUFFER_CONTENT_TYPE_UNICODE;
1346 }
1347
1348 /**
1349 * hb_buffer_add_utf8:
1350 * @buffer: a buffer.
1351 * @text: (array length=text_length):
1352 * @text_length:
1353 * @item_offset:
1354 * @item_length:
1355 *
1356 *
1357 *
1358 * Since: 1.0
1359 **/
1360 void
hb_buffer_add_utf8(hb_buffer_t * buffer,const char * text,int text_length,unsigned int item_offset,int item_length)1361 hb_buffer_add_utf8 (hb_buffer_t *buffer,
1362 const char *text,
1363 int text_length,
1364 unsigned int item_offset,
1365 int item_length)
1366 {
1367 hb_buffer_add_utf (buffer, (const uint8_t *) text, text_length, item_offset, item_length);
1368 }
1369
1370 /**
1371 * hb_buffer_add_utf16:
1372 * @buffer: a buffer.
1373 * @text: (array length=text_length):
1374 * @text_length:
1375 * @item_offset:
1376 * @item_length:
1377 *
1378 *
1379 *
1380 * Since: 1.0
1381 **/
1382 void
hb_buffer_add_utf16(hb_buffer_t * buffer,const uint16_t * text,int text_length,unsigned int item_offset,int item_length)1383 hb_buffer_add_utf16 (hb_buffer_t *buffer,
1384 const uint16_t *text,
1385 int text_length,
1386 unsigned int item_offset,
1387 int item_length)
1388 {
1389 hb_buffer_add_utf (buffer, text, text_length, item_offset, item_length);
1390 }
1391
1392 /**
1393 * hb_buffer_add_utf32:
1394 * @buffer: a buffer.
1395 * @text: (array length=text_length):
1396 * @text_length:
1397 * @item_offset:
1398 * @item_length:
1399 *
1400 *
1401 *
1402 * Since: 1.0
1403 **/
1404 void
hb_buffer_add_utf32(hb_buffer_t * buffer,const uint32_t * text,int text_length,unsigned int item_offset,int item_length)1405 hb_buffer_add_utf32 (hb_buffer_t *buffer,
1406 const uint32_t *text,
1407 int text_length,
1408 unsigned int item_offset,
1409 int item_length)
1410 {
1411 hb_buffer_add_utf (buffer, text, text_length, item_offset, item_length);
1412 }
1413
1414
1415 static int
compare_info_codepoint(const hb_glyph_info_t * pa,const hb_glyph_info_t * pb)1416 compare_info_codepoint (const hb_glyph_info_t *pa,
1417 const hb_glyph_info_t *pb)
1418 {
1419 return (int) pb->codepoint - (int) pa->codepoint;
1420 }
1421
1422 static inline void
normalize_glyphs_cluster(hb_buffer_t * buffer,unsigned int start,unsigned int end,bool backward)1423 normalize_glyphs_cluster (hb_buffer_t *buffer,
1424 unsigned int start,
1425 unsigned int end,
1426 bool backward)
1427 {
1428 hb_glyph_position_t *pos = buffer->pos;
1429
1430 /* Total cluster advance */
1431 hb_position_t total_x_advance = 0, total_y_advance = 0;
1432 for (unsigned int i = start; i < end; i++)
1433 {
1434 total_x_advance += pos[i].x_advance;
1435 total_y_advance += pos[i].y_advance;
1436 }
1437
1438 hb_position_t x_advance = 0, y_advance = 0;
1439 for (unsigned int i = start; i < end; i++)
1440 {
1441 pos[i].x_offset += x_advance;
1442 pos[i].y_offset += y_advance;
1443
1444 x_advance += pos[i].x_advance;
1445 y_advance += pos[i].y_advance;
1446
1447 pos[i].x_advance = 0;
1448 pos[i].y_advance = 0;
1449 }
1450
1451 if (backward)
1452 {
1453 /* Transfer all cluster advance to the last glyph. */
1454 pos[end - 1].x_advance = total_x_advance;
1455 pos[end - 1].y_advance = total_y_advance;
1456
1457 hb_bubble_sort (buffer->info + start, end - start - 1, compare_info_codepoint, buffer->pos + start);
1458 } else {
1459 /* Transfer all cluster advance to the first glyph. */
1460 pos[start].x_advance += total_x_advance;
1461 pos[start].y_advance += total_y_advance;
1462 for (unsigned int i = start + 1; i < end; i++) {
1463 pos[i].x_offset -= total_x_advance;
1464 pos[i].y_offset -= total_y_advance;
1465 }
1466 hb_bubble_sort (buffer->info + start + 1, end - start - 1, compare_info_codepoint, buffer->pos + start + 1);
1467 }
1468 }
1469
1470 /**
1471 * hb_buffer_normalize_glyphs:
1472 * @buffer: a buffer.
1473 *
1474 *
1475 *
1476 * Since: 1.0
1477 **/
1478 void
hb_buffer_normalize_glyphs(hb_buffer_t * buffer)1479 hb_buffer_normalize_glyphs (hb_buffer_t *buffer)
1480 {
1481 assert (buffer->have_positions);
1482 assert (buffer->content_type == HB_BUFFER_CONTENT_TYPE_GLYPHS);
1483
1484 bool backward = HB_DIRECTION_IS_BACKWARD (buffer->props.direction);
1485
1486 unsigned int count = buffer->len;
1487 if (unlikely (!count)) return;
1488 hb_glyph_info_t *info = buffer->info;
1489
1490 unsigned int start = 0;
1491 unsigned int end;
1492 for (end = start + 1; end < count; end++)
1493 if (info[start].cluster != info[end].cluster) {
1494 normalize_glyphs_cluster (buffer, start, end, backward);
1495 start = end;
1496 }
1497 normalize_glyphs_cluster (buffer, start, end, backward);
1498 }
1499