Coverage Report

Created: 2024-01-20 12:25

/src/harfbuzz/src/hb-ot-shape-normalize.cc
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * Copyright © 2011,2012  Google, Inc.
3
 *
4
 *  This is part of HarfBuzz, a text shaping library.
5
 *
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
 * Google Author(s): Behdad Esfahbod
25
 */
26
27
#include "hb.hh"
28
29
#ifndef HB_NO_OT_SHAPE
30
31
#include "hb-ot-shape-normalize.hh"
32
#include "hb-ot-shaper.hh"
33
#include "hb-ot-shape.hh"
34
35
36
/*
37
 * HIGHLEVEL DESIGN:
38
 *
39
 * This file exports one main function: _hb_ot_shape_normalize().
40
 *
41
 * This function closely reflects the Unicode Normalization Algorithm,
42
 * yet it's different.
43
 *
44
 * Each shaper specifies whether it prefers decomposed (NFD) or composed (NFC).
45
 * The logic however tries to use whatever the font can support.
46
 *
47
 * In general what happens is that: each grapheme is decomposed in a chain
48
 * of 1:2 decompositions, marks reordered, and then recomposed if desired,
49
 * so far it's like Unicode Normalization.  However, the decomposition and
50
 * recomposition only happens if the font supports the resulting characters.
51
 *
52
 * The goals are:
53
 *
54
 *   - Try to render all canonically equivalent strings similarly.  To really
55
 *     achieve this we have to always do the full decomposition and then
56
 *     selectively recompose from there.  It's kinda too expensive though, so
57
 *     we skip some cases.  For example, if composed is desired, we simply
58
 *     don't touch 1-character clusters that are supported by the font, even
59
 *     though their NFC may be different.
60
 *
61
 *   - When a font has a precomposed character for a sequence but the 'ccmp'
62
 *     feature in the font is not adequate, use the precomposed character
63
 *     which typically has better mark positioning.
64
 *
65
 *   - When a font does not support a combining mark, but supports it precomposed
66
 *     with previous base, use that.  This needs the itemizer to have this
67
 *     knowledge too.  We need to provide assistance to the itemizer.
68
 *
69
 *   - When a font does not support a character but supports its canonical
70
 *     decomposition, well, use the decomposition.
71
 *
72
 *   - The shapers can customize the compose and decompose functions to
73
 *     offload some of their requirements to the normalizer.  For example, the
74
 *     Indic shaper may want to disallow recomposing of two matras.
75
 */
76
77
static bool
78
decompose_unicode (const hb_ot_shape_normalize_context_t *c,
79
       hb_codepoint_t  ab,
80
       hb_codepoint_t *a,
81
       hb_codepoint_t *b)
82
25.1M
{
83
25.1M
  return (bool) c->unicode->decompose (ab, a, b);
84
25.1M
}
85
86
static bool
87
compose_unicode (const hb_ot_shape_normalize_context_t *c,
88
     hb_codepoint_t  a,
89
     hb_codepoint_t  b,
90
     hb_codepoint_t *ab)
91
121k
{
92
121k
  return (bool) c->unicode->compose (a, b, ab);
93
121k
}
94
95
static inline void
96
set_glyph (hb_glyph_info_t &info, hb_font_t *font)
97
15.8k
{
98
15.8k
  (void) font->get_nominal_glyph (info.codepoint, &info.glyph_index());
99
15.8k
}
100
101
static inline void
102
output_char (hb_buffer_t *buffer, hb_codepoint_t unichar, hb_codepoint_t glyph)
103
25.9k
{
104
  /* This is very confusing indeed. */
105
25.9k
  buffer->cur().glyph_index() = glyph;
106
25.9k
  (void) buffer->output_glyph (unichar);
107
25.9k
  _hb_glyph_info_set_unicode_props (&buffer->prev(), buffer);
108
25.9k
}
109
110
static inline void
111
next_char (hb_buffer_t *buffer, hb_codepoint_t glyph)
112
25.6M
{
113
25.6M
  buffer->cur().glyph_index() = glyph;
114
25.6M
  (void) buffer->next_glyph ();
115
25.6M
}
116
117
static inline void
118
skip_char (hb_buffer_t *buffer)
119
11.1k
{
120
11.1k
  buffer->skip_glyph ();
121
11.1k
}
122
123
/* Returns 0 if didn't decompose, number of resulting characters otherwise. */
124
static inline unsigned int
125
decompose (const hb_ot_shape_normalize_context_t *c, bool shortest, hb_codepoint_t ab)
126
25.5M
{
127
25.5M
  hb_codepoint_t a = 0, b = 0, a_glyph = 0, b_glyph = 0;
128
25.5M
  hb_buffer_t * const buffer = c->buffer;
129
25.5M
  hb_font_t * const font = c->font;
130
131
25.5M
  if (!c->decompose (c, ab, &a, &b) ||
132
25.5M
      (b && !font->get_nominal_glyph (b, &b_glyph)))
133
25.5M
    return 0;
134
135
27.7k
  bool has_a = (bool) font->get_nominal_glyph (a, &a_glyph);
136
27.7k
  if (shortest && has_a) {
137
    /* Output a and b */
138
2.67k
    output_char (buffer, a, a_glyph);
139
2.67k
    if (likely (b)) {
140
2.50k
      output_char (buffer, b, b_glyph);
141
2.50k
      return 2;
142
2.50k
    }
143
174
    return 1;
144
2.67k
  }
145
146
25.1k
  if (unsigned ret = decompose (c, shortest, a)) {
147
4.09k
    if (b) {
148
4.05k
      output_char (buffer, b, b_glyph);
149
4.05k
      return ret + 1;
150
4.05k
    }
151
49
    return ret;
152
4.09k
  }
153
154
21.0k
  if (has_a) {
155
8.52k
    output_char (buffer, a, a_glyph);
156
8.52k
    if (likely (b)) {
157
8.18k
      output_char (buffer, b, b_glyph);
158
8.18k
      return 2;
159
8.18k
    }
160
334
    return 1;
161
8.52k
  }
162
163
12.4k
  return 0;
164
21.0k
}
165
166
static inline void
167
decompose_current_character (const hb_ot_shape_normalize_context_t *c, bool shortest)
168
25.6M
{
169
25.6M
  hb_buffer_t * const buffer = c->buffer;
170
25.6M
  hb_codepoint_t u = buffer->cur().codepoint;
171
25.6M
  hb_codepoint_t glyph = 0;
172
173
25.6M
  if (shortest && c->font->get_nominal_glyph (u, &glyph, c->not_found))
174
108k
  {
175
108k
    next_char (buffer, glyph);
176
108k
    return;
177
108k
  }
178
179
25.5M
  if (decompose (c, shortest, u))
180
11.1k
  {
181
11.1k
    skip_char (buffer);
182
11.1k
    return;
183
11.1k
  }
184
185
25.5M
  if (!shortest && c->font->get_nominal_glyph (u, &glyph, c->not_found))
186
30.6k
  {
187
30.6k
    next_char (buffer, glyph);
188
30.6k
    return;
189
30.6k
  }
190
191
25.4M
  if (_hb_glyph_info_is_unicode_space (&buffer->cur()))
192
21.4k
  {
193
21.4k
    hb_codepoint_t space_glyph;
194
21.4k
    hb_unicode_funcs_t::space_t space_type = buffer->unicode->space_fallback_type (u);
195
21.4k
    if (space_type != hb_unicode_funcs_t::NOT_SPACE &&
196
21.4k
  (c->font->get_nominal_glyph (0x0020, &space_glyph) || (space_glyph = buffer->invisible)))
197
4.96k
    {
198
4.96k
      _hb_glyph_info_set_unicode_space_fallback_type (&buffer->cur(), space_type);
199
4.96k
      next_char (buffer, space_glyph);
200
4.96k
      buffer->scratch_flags |= HB_BUFFER_SCRATCH_FLAG_HAS_SPACE_FALLBACK;
201
4.96k
      return;
202
4.96k
    }
203
21.4k
  }
204
205
25.4M
  if (u == 0x2011u)
206
2.21k
  {
207
    /* U+2011 is the only sensible character that is a no-break version of another character
208
     * and not a space.  The space ones are handled already.  Handle this lone one. */
209
2.21k
    hb_codepoint_t other_glyph;
210
2.21k
    if (c->font->get_nominal_glyph (0x2010u, &other_glyph))
211
36
    {
212
36
      next_char (buffer, other_glyph);
213
36
      return;
214
36
    }
215
2.21k
  }
216
217
25.4M
  next_char (buffer, glyph); /* glyph is initialized in earlier branches. */
218
25.4M
}
219
220
static inline void
221
handle_variation_selector_cluster (const hb_ot_shape_normalize_context_t *c,
222
           unsigned int end,
223
           bool short_circuit HB_UNUSED)
224
6.11k
{
225
  /* Currently if there's a variation-selector we give-up on normalization, it's just too hard. */
226
6.11k
  hb_buffer_t * const buffer = c->buffer;
227
6.11k
  hb_font_t * const font = c->font;
228
13.4k
  for (; buffer->idx < end - 1 && buffer->successful;) {
229
7.32k
    if (unlikely (buffer->unicode->is_variation_selector (buffer->cur(+1).codepoint))) {
230
6.17k
      if (font->get_variation_glyph (buffer->cur().codepoint, buffer->cur(+1).codepoint, &buffer->cur().glyph_index()))
231
200
      {
232
200
  hb_codepoint_t unicode = buffer->cur().codepoint;
233
200
  (void) buffer->replace_glyphs (2, 1, &unicode);
234
200
      }
235
5.97k
      else
236
5.97k
      {
237
  /* Just pass on the two characters separately, let GSUB do its magic. */
238
5.97k
  set_glyph (buffer->cur(), font);
239
5.97k
  (void) buffer->next_glyph ();
240
5.97k
  set_glyph (buffer->cur(), font);
241
5.97k
  (void) buffer->next_glyph ();
242
5.97k
      }
243
      /* Skip any further variation selectors. */
244
8.26k
      while (buffer->idx < end &&
245
8.26k
       buffer->successful &&
246
8.26k
       unlikely (buffer->unicode->is_variation_selector (buffer->cur().codepoint)))
247
2.09k
      {
248
2.09k
  set_glyph (buffer->cur(), font);
249
2.09k
  (void) buffer->next_glyph ();
250
2.09k
      }
251
6.17k
    }
252
1.15k
    else
253
1.15k
    {
254
1.15k
      set_glyph (buffer->cur(), font);
255
1.15k
      (void) buffer->next_glyph ();
256
1.15k
    }
257
7.32k
  }
258
6.11k
  if (likely (buffer->idx < end))
259
673
  {
260
673
    set_glyph (buffer->cur(), font);
261
673
    (void) buffer->next_glyph ();
262
673
  }
263
6.11k
}
264
265
static inline void
266
decompose_multi_char_cluster (const hb_ot_shape_normalize_context_t *c, unsigned int end, bool short_circuit)
267
187k
{
268
187k
  hb_buffer_t * const buffer = c->buffer;
269
640k
  for (unsigned int i = buffer->idx; i < end && buffer->successful; i++)
270
458k
    if (unlikely (buffer->unicode->is_variation_selector (buffer->info[i].codepoint))) {
271
6.11k
      handle_variation_selector_cluster (c, end, short_circuit);
272
6.11k
      return;
273
6.11k
    }
274
275
627k
  while (buffer->idx < end && buffer->successful)
276
446k
    decompose_current_character (c, short_circuit);
277
181k
}
278
279
280
static int
281
compare_combining_class (const hb_glyph_info_t *pa, const hb_glyph_info_t *pb)
282
40.0k
{
283
40.0k
  unsigned int a = _hb_glyph_info_get_modified_combining_class (pa);
284
40.0k
  unsigned int b = _hb_glyph_info_get_modified_combining_class (pb);
285
286
40.0k
  return a < b ? -1 : a == b ? 0 : +1;
287
40.0k
}
288
289
290
void
291
_hb_ot_shape_normalize (const hb_ot_shape_plan_t *plan,
292
      hb_buffer_t *buffer,
293
      hb_font_t *font)
294
9.96M
{
295
9.96M
  if (unlikely (!buffer->len)) return;
296
297
9.96M
  _hb_buffer_assert_unicode_vars (buffer);
298
299
9.96M
  hb_ot_shape_normalization_mode_t mode = plan->shaper->normalization_preference;
300
9.96M
  if (mode == HB_OT_SHAPE_NORMALIZATION_MODE_AUTO)
301
9.91M
  {
302
9.91M
    if (plan->has_gpos_mark)
303
      // https://github.com/harfbuzz/harfbuzz/issues/653#issuecomment-423905920
304
      //mode = HB_OT_SHAPE_NORMALIZATION_MODE_DECOMPOSED;
305
138k
      mode = HB_OT_SHAPE_NORMALIZATION_MODE_COMPOSED_DIACRITICS;
306
9.78M
    else
307
9.78M
      mode = HB_OT_SHAPE_NORMALIZATION_MODE_COMPOSED_DIACRITICS;
308
9.91M
  }
309
310
9.96M
  const hb_ot_shape_normalize_context_t c = {
311
9.96M
    plan,
312
9.96M
    buffer,
313
9.96M
    font,
314
9.96M
    buffer->unicode,
315
9.96M
    buffer->not_found,
316
9.96M
    plan->shaper->decompose ? plan->shaper->decompose : decompose_unicode,
317
9.96M
    plan->shaper->compose   ? plan->shaper->compose   : compose_unicode
318
9.96M
  };
319
320
9.96M
  bool always_short_circuit = mode == HB_OT_SHAPE_NORMALIZATION_MODE_NONE;
321
9.96M
  bool might_short_circuit = always_short_circuit ||
322
9.96M
           (mode != HB_OT_SHAPE_NORMALIZATION_MODE_DECOMPOSED &&
323
9.96M
            mode != HB_OT_SHAPE_NORMALIZATION_MODE_COMPOSED_DIACRITICS_NO_SHORT_CIRCUIT);
324
9.96M
  unsigned int count;
325
326
  /* We do a fairly straightforward yet custom normalization process in three
327
   * separate rounds: decompose, reorder, recompose (if desired).  Currently
328
   * this makes two buffer swaps.  We can make it faster by moving the last
329
   * two rounds into the inner loop for the first round, but it's more readable
330
   * this way. */
331
332
333
  /* First round, decompose */
334
335
9.96M
  bool all_simple = true;
336
9.96M
  {
337
9.96M
    buffer->clear_output ();
338
9.96M
    count = buffer->len;
339
9.96M
    buffer->idx = 0;
340
9.96M
    do
341
10.1M
    {
342
10.1M
      unsigned int end;
343
26.2M
      for (end = buffer->idx + 1; end < count; end++)
344
16.2M
  if (_hb_glyph_info_is_unicode_mark (&buffer->info[end]))
345
187k
    break;
346
347
10.1M
      if (end < count)
348
187k
  end--; /* Leave one base for the marks to cluster with. */
349
350
      /* From idx to end are simple clusters. */
351
10.1M
      if (might_short_circuit)
352
10.0M
      {
353
10.0M
  unsigned int done = font->get_nominal_glyphs (end - buffer->idx,
354
10.0M
                  &buffer->cur().codepoint,
355
10.0M
                  sizeof (buffer->info[0]),
356
10.0M
                  &buffer->cur().glyph_index(),
357
10.0M
                  sizeof (buffer->info[0]));
358
10.0M
  if (unlikely (!buffer->next_glyphs (done))) break;
359
10.0M
      }
360
35.3M
      while (buffer->idx < end && buffer->successful)
361
25.1M
  decompose_current_character (&c, might_short_circuit);
362
363
10.1M
      if (buffer->idx == count || !buffer->successful)
364
9.95M
  break;
365
366
187k
      all_simple = false;
367
368
      /* Find all the marks now. */
369
462k
      for (end = buffer->idx + 1; end < count; end++)
370
447k
  if (!_hb_glyph_info_is_unicode_mark(&buffer->info[end]))
371
172k
    break;
372
373
      /* idx to end is one non-simple cluster. */
374
187k
      decompose_multi_char_cluster (&c, end, always_short_circuit);
375
187k
    }
376
9.96M
    while (buffer->idx < count && buffer->successful);
377
0
    buffer->sync ();
378
9.96M
  }
379
380
381
  /* Second round, reorder (inplace) */
382
383
9.96M
  if (!all_simple && buffer->message(font, "start reorder"))
384
95.1k
  {
385
95.1k
    count = buffer->len;
386
1.50M
    for (unsigned int i = 0; i < count; i++)
387
1.40M
    {
388
1.40M
      if (_hb_glyph_info_get_modified_combining_class (&buffer->info[i]) == 0)
389
1.29M
  continue;
390
391
111k
      unsigned int end;
392
143k
      for (end = i + 1; end < count; end++)
393
136k
  if (_hb_glyph_info_get_modified_combining_class (&buffer->info[end]) == 0)
394
104k
    break;
395
396
      /* We are going to do a O(n^2).  Only do this if the sequence is short. */
397
111k
      if (end - i > HB_OT_SHAPE_MAX_COMBINING_MARKS) {
398
0
  i = end;
399
0
  continue;
400
0
      }
401
402
111k
      buffer->sort (i, end, compare_combining_class);
403
404
111k
      if (plan->shaper->reorder_marks)
405
25.3k
  plan->shaper->reorder_marks (plan, buffer, i, end);
406
407
111k
      i = end;
408
111k
    }
409
95.1k
    (void) buffer->message(font, "end reorder");
410
95.1k
  }
411
9.96M
  if (buffer->scratch_flags & HB_BUFFER_SCRATCH_FLAG_HAS_CGJ)
412
463
  {
413
    /* For all CGJ, check if it prevented any reordering at all.
414
     * If it did NOT, then make it skippable.
415
     * https://github.com/harfbuzz/harfbuzz/issues/554
416
     */
417
7.78k
    for (unsigned int i = 1; i + 1 < buffer->len; i++)
418
7.31k
      if (buffer->info[i].codepoint == 0x034Fu/*CGJ*/ &&
419
7.31k
    (info_cc(buffer->info[i+1]) == 0 || info_cc(buffer->info[i-1]) <= info_cc(buffer->info[i+1])))
420
1.20k
      {
421
1.20k
  _hb_glyph_info_unhide (&buffer->info[i]);
422
1.20k
      }
423
463
  }
424
425
426
  /* Third round, recompose */
427
428
9.96M
  if (!all_simple &&
429
9.96M
      buffer->successful &&
430
9.96M
      (mode == HB_OT_SHAPE_NORMALIZATION_MODE_COMPOSED_DIACRITICS ||
431
95.0k
       mode == HB_OT_SHAPE_NORMALIZATION_MODE_COMPOSED_DIACRITICS_NO_SHORT_CIRCUIT))
432
94.2k
  {
433
    /* As noted in the comment earlier, we don't try to combine
434
     * ccc=0 chars with their previous Starter. */
435
436
94.2k
    buffer->clear_output ();
437
94.2k
    count = buffer->len;
438
94.2k
    unsigned int starter = 0;
439
94.2k
    (void) buffer->next_glyph ();
440
1.52M
    while (buffer->idx < count /* No need for: && buffer->successful */)
441
1.43M
    {
442
1.43M
      hb_codepoint_t composed, glyph;
443
1.43M
      if (/* We don't try to compose a non-mark character with it's preceding starter.
444
     * This is both an optimization to avoid trying to compose every two neighboring
445
     * glyphs in most scripts AND a desired feature for Hangul.  Apparently Hangul
446
     * fonts are not designed to mix-and-match pre-composed syllables and Jamo. */
447
1.43M
    _hb_glyph_info_is_unicode_mark(&buffer->cur()))
448
279k
      {
449
279k
  if (/* If there's anything between the starter and this char, they should have CCC
450
       * smaller than this character's. */
451
279k
      (starter == buffer->out_len - 1 ||
452
279k
       info_cc (buffer->prev()) < info_cc (buffer->cur())) &&
453
      /* And compose. */
454
279k
      c.compose (&c,
455
255k
           buffer->out_info[starter].codepoint,
456
255k
           buffer->cur().codepoint,
457
255k
           &composed) &&
458
      /* And the font has glyph for the composite. */
459
279k
      font->get_nominal_glyph (composed, &glyph))
460
3.77k
  {
461
    /* Composes. */
462
3.77k
    if (unlikely (!buffer->next_glyph ())) break; /* Copy to out-buffer. */
463
3.77k
    buffer->merge_out_clusters (starter, buffer->out_len);
464
3.77k
    buffer->out_len--; /* Remove the second composable. */
465
    /* Modify starter and carry on. */
466
3.77k
    buffer->out_info[starter].codepoint = composed;
467
3.77k
    buffer->out_info[starter].glyph_index() = glyph;
468
3.77k
    _hb_glyph_info_set_unicode_props (&buffer->out_info[starter], buffer);
469
470
3.77k
    continue;
471
3.77k
  }
472
279k
      }
473
474
      /* Blocked, or doesn't compose. */
475
1.43M
      if (unlikely (!buffer->next_glyph ())) break;
476
477
1.43M
      if (info_cc (buffer->prev()) == 0)
478
1.29M
  starter = buffer->out_len - 1;
479
1.43M
    }
480
94.2k
    buffer->sync ();
481
94.2k
  }
482
9.96M
}
483
484
485
#endif