xref: /aosp_15_r20/external/harfbuzz_ng/src/hb-kern.hh (revision 2d1272b857b1f7575e6e246373e1cb218663db8a)
1*2d1272b8SAndroid Build Coastguard Worker /*
2*2d1272b8SAndroid Build Coastguard Worker  * Copyright © 2017  Google, Inc.
3*2d1272b8SAndroid Build Coastguard Worker  *
4*2d1272b8SAndroid Build Coastguard Worker  *  This is part of HarfBuzz, a text shaping library.
5*2d1272b8SAndroid Build Coastguard Worker  *
6*2d1272b8SAndroid Build Coastguard Worker  * Permission is hereby granted, without written agreement and without
7*2d1272b8SAndroid Build Coastguard Worker  * license or royalty fees, to use, copy, modify, and distribute this
8*2d1272b8SAndroid Build Coastguard Worker  * software and its documentation for any purpose, provided that the
9*2d1272b8SAndroid Build Coastguard Worker  * above copyright notice and the following two paragraphs appear in
10*2d1272b8SAndroid Build Coastguard Worker  * all copies of this software.
11*2d1272b8SAndroid Build Coastguard Worker  *
12*2d1272b8SAndroid Build Coastguard Worker  * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
13*2d1272b8SAndroid Build Coastguard Worker  * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
14*2d1272b8SAndroid Build Coastguard Worker  * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
15*2d1272b8SAndroid Build Coastguard Worker  * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
16*2d1272b8SAndroid Build Coastguard Worker  * DAMAGE.
17*2d1272b8SAndroid Build Coastguard Worker  *
18*2d1272b8SAndroid Build Coastguard Worker  * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
19*2d1272b8SAndroid Build Coastguard Worker  * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
20*2d1272b8SAndroid Build Coastguard Worker  * FITNESS FOR A PARTICULAR PURPOSE.  THE SOFTWARE PROVIDED HEREUNDER IS
21*2d1272b8SAndroid Build Coastguard Worker  * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
22*2d1272b8SAndroid Build Coastguard Worker  * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
23*2d1272b8SAndroid Build Coastguard Worker  *
24*2d1272b8SAndroid Build Coastguard Worker  * Google Author(s): Behdad Esfahbod
25*2d1272b8SAndroid Build Coastguard Worker  */
26*2d1272b8SAndroid Build Coastguard Worker 
27*2d1272b8SAndroid Build Coastguard Worker #ifndef HB_KERN_HH
28*2d1272b8SAndroid Build Coastguard Worker #define HB_KERN_HH
29*2d1272b8SAndroid Build Coastguard Worker 
30*2d1272b8SAndroid Build Coastguard Worker #include "hb-open-type.hh"
31*2d1272b8SAndroid Build Coastguard Worker #include "hb-aat-layout-common.hh"
32*2d1272b8SAndroid Build Coastguard Worker #include "hb-ot-layout-gpos-table.hh"
33*2d1272b8SAndroid Build Coastguard Worker 
34*2d1272b8SAndroid Build Coastguard Worker 
35*2d1272b8SAndroid Build Coastguard Worker namespace OT {
36*2d1272b8SAndroid Build Coastguard Worker 
37*2d1272b8SAndroid Build Coastguard Worker 
38*2d1272b8SAndroid Build Coastguard Worker template <typename Driver>
39*2d1272b8SAndroid Build Coastguard Worker struct hb_kern_machine_t
40*2d1272b8SAndroid Build Coastguard Worker {
hb_kern_machine_tOT::hb_kern_machine_t41*2d1272b8SAndroid Build Coastguard Worker   hb_kern_machine_t (const Driver &driver_,
42*2d1272b8SAndroid Build Coastguard Worker 		     bool crossStream_ = false) :
43*2d1272b8SAndroid Build Coastguard Worker 		       driver (driver_),
44*2d1272b8SAndroid Build Coastguard Worker 		       crossStream (crossStream_) {}
45*2d1272b8SAndroid Build Coastguard Worker 
46*2d1272b8SAndroid Build Coastguard Worker   HB_NO_SANITIZE_SIGNED_INTEGER_OVERFLOW
kernOT::hb_kern_machine_t47*2d1272b8SAndroid Build Coastguard Worker   void kern (hb_font_t   *font,
48*2d1272b8SAndroid Build Coastguard Worker 	     hb_buffer_t *buffer,
49*2d1272b8SAndroid Build Coastguard Worker 	     hb_mask_t    kern_mask,
50*2d1272b8SAndroid Build Coastguard Worker 	     bool         scale = true) const
51*2d1272b8SAndroid Build Coastguard Worker   {
52*2d1272b8SAndroid Build Coastguard Worker     if (!buffer->message (font, "start kern"))
53*2d1272b8SAndroid Build Coastguard Worker       return;
54*2d1272b8SAndroid Build Coastguard Worker 
55*2d1272b8SAndroid Build Coastguard Worker     buffer->unsafe_to_concat ();
56*2d1272b8SAndroid Build Coastguard Worker     OT::hb_ot_apply_context_t c (1, font, buffer, hb_blob_get_empty ());
57*2d1272b8SAndroid Build Coastguard Worker     c.set_lookup_mask (kern_mask);
58*2d1272b8SAndroid Build Coastguard Worker     c.set_lookup_props (OT::LookupFlag::IgnoreMarks);
59*2d1272b8SAndroid Build Coastguard Worker     auto &skippy_iter = c.iter_input;
60*2d1272b8SAndroid Build Coastguard Worker 
61*2d1272b8SAndroid Build Coastguard Worker     bool horizontal = HB_DIRECTION_IS_HORIZONTAL (buffer->props.direction);
62*2d1272b8SAndroid Build Coastguard Worker     unsigned int count = buffer->len;
63*2d1272b8SAndroid Build Coastguard Worker     hb_glyph_info_t *info = buffer->info;
64*2d1272b8SAndroid Build Coastguard Worker     hb_glyph_position_t *pos = buffer->pos;
65*2d1272b8SAndroid Build Coastguard Worker     for (unsigned int idx = 0; idx < count;)
66*2d1272b8SAndroid Build Coastguard Worker     {
67*2d1272b8SAndroid Build Coastguard Worker       if (!(info[idx].mask & kern_mask))
68*2d1272b8SAndroid Build Coastguard Worker       {
69*2d1272b8SAndroid Build Coastguard Worker 	idx++;
70*2d1272b8SAndroid Build Coastguard Worker 	continue;
71*2d1272b8SAndroid Build Coastguard Worker       }
72*2d1272b8SAndroid Build Coastguard Worker 
73*2d1272b8SAndroid Build Coastguard Worker       skippy_iter.reset (idx);
74*2d1272b8SAndroid Build Coastguard Worker       unsigned unsafe_to;
75*2d1272b8SAndroid Build Coastguard Worker       if (!skippy_iter.next (&unsafe_to))
76*2d1272b8SAndroid Build Coastguard Worker       {
77*2d1272b8SAndroid Build Coastguard Worker 	idx++;
78*2d1272b8SAndroid Build Coastguard Worker 	continue;
79*2d1272b8SAndroid Build Coastguard Worker       }
80*2d1272b8SAndroid Build Coastguard Worker 
81*2d1272b8SAndroid Build Coastguard Worker       unsigned int i = idx;
82*2d1272b8SAndroid Build Coastguard Worker       unsigned int j = skippy_iter.idx;
83*2d1272b8SAndroid Build Coastguard Worker 
84*2d1272b8SAndroid Build Coastguard Worker       hb_position_t kern = driver.get_kerning (info[i].codepoint,
85*2d1272b8SAndroid Build Coastguard Worker 					       info[j].codepoint);
86*2d1272b8SAndroid Build Coastguard Worker 
87*2d1272b8SAndroid Build Coastguard Worker 
88*2d1272b8SAndroid Build Coastguard Worker       if (likely (!kern))
89*2d1272b8SAndroid Build Coastguard Worker 	goto skip;
90*2d1272b8SAndroid Build Coastguard Worker 
91*2d1272b8SAndroid Build Coastguard Worker       if (horizontal)
92*2d1272b8SAndroid Build Coastguard Worker       {
93*2d1272b8SAndroid Build Coastguard Worker 	if (scale)
94*2d1272b8SAndroid Build Coastguard Worker 	  kern = font->em_scale_x (kern);
95*2d1272b8SAndroid Build Coastguard Worker 	if (crossStream)
96*2d1272b8SAndroid Build Coastguard Worker 	{
97*2d1272b8SAndroid Build Coastguard Worker 	  pos[j].y_offset = kern;
98*2d1272b8SAndroid Build Coastguard Worker 	  buffer->scratch_flags |= HB_BUFFER_SCRATCH_FLAG_HAS_GPOS_ATTACHMENT;
99*2d1272b8SAndroid Build Coastguard Worker 	}
100*2d1272b8SAndroid Build Coastguard Worker 	else
101*2d1272b8SAndroid Build Coastguard Worker 	{
102*2d1272b8SAndroid Build Coastguard Worker 	  hb_position_t kern1 = kern >> 1;
103*2d1272b8SAndroid Build Coastguard Worker 	  hb_position_t kern2 = kern - kern1;
104*2d1272b8SAndroid Build Coastguard Worker 	  pos[i].x_advance += kern1;
105*2d1272b8SAndroid Build Coastguard Worker 	  pos[j].x_advance += kern2;
106*2d1272b8SAndroid Build Coastguard Worker 	  pos[j].x_offset += kern2;
107*2d1272b8SAndroid Build Coastguard Worker 	}
108*2d1272b8SAndroid Build Coastguard Worker       }
109*2d1272b8SAndroid Build Coastguard Worker       else
110*2d1272b8SAndroid Build Coastguard Worker       {
111*2d1272b8SAndroid Build Coastguard Worker 	if (scale)
112*2d1272b8SAndroid Build Coastguard Worker 	  kern = font->em_scale_y (kern);
113*2d1272b8SAndroid Build Coastguard Worker 	if (crossStream)
114*2d1272b8SAndroid Build Coastguard Worker 	{
115*2d1272b8SAndroid Build Coastguard Worker 	  pos[j].x_offset = kern;
116*2d1272b8SAndroid Build Coastguard Worker 	  buffer->scratch_flags |= HB_BUFFER_SCRATCH_FLAG_HAS_GPOS_ATTACHMENT;
117*2d1272b8SAndroid Build Coastguard Worker 	}
118*2d1272b8SAndroid Build Coastguard Worker 	else
119*2d1272b8SAndroid Build Coastguard Worker 	{
120*2d1272b8SAndroid Build Coastguard Worker 	  hb_position_t kern1 = kern >> 1;
121*2d1272b8SAndroid Build Coastguard Worker 	  hb_position_t kern2 = kern - kern1;
122*2d1272b8SAndroid Build Coastguard Worker 	  pos[i].y_advance += kern1;
123*2d1272b8SAndroid Build Coastguard Worker 	  pos[j].y_advance += kern2;
124*2d1272b8SAndroid Build Coastguard Worker 	  pos[j].y_offset += kern2;
125*2d1272b8SAndroid Build Coastguard Worker 	}
126*2d1272b8SAndroid Build Coastguard Worker       }
127*2d1272b8SAndroid Build Coastguard Worker 
128*2d1272b8SAndroid Build Coastguard Worker       buffer->unsafe_to_break (i, j + 1);
129*2d1272b8SAndroid Build Coastguard Worker 
130*2d1272b8SAndroid Build Coastguard Worker     skip:
131*2d1272b8SAndroid Build Coastguard Worker       idx = skippy_iter.idx;
132*2d1272b8SAndroid Build Coastguard Worker     }
133*2d1272b8SAndroid Build Coastguard Worker 
134*2d1272b8SAndroid Build Coastguard Worker     (void) buffer->message (font, "end kern");
135*2d1272b8SAndroid Build Coastguard Worker   }
136*2d1272b8SAndroid Build Coastguard Worker 
137*2d1272b8SAndroid Build Coastguard Worker   const Driver &driver;
138*2d1272b8SAndroid Build Coastguard Worker   bool crossStream;
139*2d1272b8SAndroid Build Coastguard Worker };
140*2d1272b8SAndroid Build Coastguard Worker 
141*2d1272b8SAndroid Build Coastguard Worker 
142*2d1272b8SAndroid Build Coastguard Worker } /* namespace OT */
143*2d1272b8SAndroid Build Coastguard Worker 
144*2d1272b8SAndroid Build Coastguard Worker 
145*2d1272b8SAndroid Build Coastguard Worker #endif /* HB_KERN_HH */
146