1#![expect(unsafe_code)]
6
7use std::os::raw::{c_char, c_int, c_uint, c_void};
8use std::sync::LazyLock;
9use std::{char, ptr};
10
11use app_units::Au;
12use euclid::default::Point2D;
13use harfbuzz_sys::{
16 HB_BUFFER_CLUSTER_LEVEL_MONOTONE_CHARACTERS, HB_DIRECTION_LTR, HB_DIRECTION_RTL,
17 HB_MEMORY_MODE_READONLY, HB_OT_LAYOUT_BASELINE_TAG_HANGING,
18 HB_OT_LAYOUT_BASELINE_TAG_IDEO_EMBOX_BOTTOM_OR_LEFT, HB_OT_LAYOUT_BASELINE_TAG_ROMAN,
19 hb_blob_create, hb_blob_t, hb_bool_t, hb_buffer_add_utf8, hb_buffer_create, hb_buffer_destroy,
20 hb_buffer_get_glyph_infos, hb_buffer_get_glyph_positions, hb_buffer_get_length,
21 hb_buffer_set_cluster_level, hb_buffer_set_direction, hb_buffer_set_language,
22 hb_buffer_set_script, hb_buffer_t, hb_codepoint_t, hb_face_create_for_tables, hb_face_destroy,
23 hb_face_t, hb_feature_t, hb_font_create, hb_font_destroy, hb_font_funcs_create,
24 hb_font_funcs_set_glyph_h_advance_func, hb_font_funcs_set_nominal_glyph_func, hb_font_funcs_t,
25 hb_font_set_funcs, hb_font_set_ppem, hb_font_set_scale, hb_font_set_variations, hb_font_t,
26 hb_glyph_info_t, hb_glyph_position_t, hb_language_from_string, hb_ot_layout_get_baseline,
27 hb_position_t, hb_script_from_iso15924_tag, hb_shape, hb_tag_t, hb_variation_t,
28};
29use num_traits::Zero;
30use read_fonts::types::Tag;
31
32use super::{GlyphShapingResult, ShapedGlyph, unicode_script_to_iso15924_tag};
33use crate::platform::font::FontTable;
34use crate::{
35 BASE, Font, FontBaseline, FontTableMethods, GlyphId, ShapedText, ShapingFlags, ShapingOptions,
36 fixed_to_float, float_to_fixed,
37};
38
39const HB_OT_TAG_DEFAULT_SCRIPT: hb_tag_t = u32::from_be_bytes(Tag::new(b"DFLT").to_be_bytes());
40const HB_OT_TAG_DEFAULT_LANGUAGE: hb_tag_t = u32::from_be_bytes(Tag::new(b"dflt").to_be_bytes());
41
42pub(crate) struct HarfbuzzGlyphShapingResult {
43 count: usize,
44 buffer: *mut hb_buffer_t,
45 glyph_infos: *mut hb_glyph_info_t,
46 pos_infos: *mut hb_glyph_position_t,
47}
48
49impl HarfbuzzGlyphShapingResult {
50 unsafe fn new(buffer: *mut hb_buffer_t) -> HarfbuzzGlyphShapingResult {
58 let mut glyph_count = 0;
59 let glyph_infos = unsafe { hb_buffer_get_glyph_infos(buffer, &mut glyph_count) };
60 assert!(!glyph_infos.is_null());
61 let mut pos_count = 0;
62 let pos_infos = unsafe { hb_buffer_get_glyph_positions(buffer, &mut pos_count) };
63 assert!(!pos_infos.is_null());
64 assert_eq!(glyph_count, pos_count);
65
66 HarfbuzzGlyphShapingResult {
67 count: glyph_count as usize,
68 buffer,
69 glyph_infos,
70 pos_infos,
71 }
72 }
73}
74
75impl Drop for HarfbuzzGlyphShapingResult {
76 fn drop(&mut self) {
77 unsafe { hb_buffer_destroy(self.buffer) }
78 }
79}
80
81struct ShapedGlyphIterator<'a> {
82 shaped_glyph_data: &'a HarfbuzzGlyphShapingResult,
83 current_glyph_offset: usize,
84 y_position: Au,
85}
86
87impl<'a> Iterator for ShapedGlyphIterator<'a> {
88 type Item = ShapedGlyph;
89
90 fn next(&mut self) -> Option<Self::Item> {
91 if self.current_glyph_offset >= self.shaped_glyph_data.count {
92 return None;
93 }
94
95 unsafe {
96 let glyph_info_i = self
97 .shaped_glyph_data
98 .glyph_infos
99 .add(self.current_glyph_offset);
100 let pos_info_i = self
101 .shaped_glyph_data
102 .pos_infos
103 .add(self.current_glyph_offset);
104 let x_offset = Shaper::fixed_to_float((*pos_info_i).x_offset);
105 let y_offset = Shaper::fixed_to_float((*pos_info_i).y_offset);
106 let x_advance = Shaper::fixed_to_float((*pos_info_i).x_advance);
107 let y_advance = Shaper::fixed_to_float((*pos_info_i).y_advance);
108
109 let x_offset = Au::from_f64_px(x_offset);
110 let y_offset = Au::from_f64_px(y_offset);
111 let x_advance = Au::from_f64_px(x_advance);
112 let y_advance = Au::from_f64_px(y_advance);
113
114 let offset = if x_offset.is_zero() && y_offset.is_zero() && y_advance.is_zero() {
115 None
116 } else {
117 if y_advance > Au::zero() {
119 self.y_position -= y_advance;
120 }
121
122 Some(Point2D::new(x_offset, self.y_position - y_offset))
123 };
124
125 self.current_glyph_offset += 1;
126 Some(ShapedGlyph {
127 glyph_id: (*glyph_info_i).codepoint as GlyphId,
128 string_byte_offset: (*glyph_info_i).cluster as usize,
129 advance: x_advance,
130 offset,
131 })
132 }
133 }
134}
135
136impl GlyphShapingResult for HarfbuzzGlyphShapingResult {
137 #[inline]
138 fn len(&self) -> usize {
139 self.count
140 }
141
142 fn iter(&self) -> impl Iterator<Item = ShapedGlyph> {
143 ShapedGlyphIterator {
144 shaped_glyph_data: self,
145 current_glyph_offset: 0,
146 y_position: Au::zero(),
147 }
148 }
149
150 fn is_rtl(&self) -> bool {
151 if self.count == 0 {
152 return false;
153 }
154 unsafe {
158 let first_glyph_info = self.glyph_infos.add(0);
159 let last_glyph_info = self.glyph_infos.add(self.count - 1);
160 (*last_glyph_info).cluster < (*first_glyph_info).cluster
161 }
162 }
163}
164
165#[derive(Debug)]
166pub(crate) struct Shaper {
167 hb_face: *mut hb_face_t,
168 hb_font: *mut hb_font_t,
169 font: *const Font,
170}
171
172unsafe impl Sync for Shaper {}
176unsafe impl Send for Shaper {}
177
178impl Drop for Shaper {
179 fn drop(&mut self) {
180 unsafe {
181 assert!(!self.hb_face.is_null());
182 hb_face_destroy(self.hb_face);
183
184 assert!(!self.hb_font.is_null());
185 hb_font_destroy(self.hb_font);
186 }
187 }
188}
189
190impl Shaper {
191 pub(crate) fn new(font: &Font) -> Shaper {
192 unsafe {
193 let hb_face: *mut hb_face_t = hb_face_create_for_tables(
194 Some(font_table_func),
195 font as *const Font as *mut c_void,
196 None,
197 );
198 let hb_font: *mut hb_font_t = hb_font_create(hb_face);
199
200 let pt_size = font.descriptor.pt_size.to_f64_px();
202 hb_font_set_ppem(hb_font, pt_size as c_uint, pt_size as c_uint);
203
204 hb_font_set_scale(
206 hb_font,
207 Shaper::float_to_fixed(pt_size) as c_int,
208 Shaper::float_to_fixed(pt_size) as c_int,
209 );
210
211 hb_font_set_funcs(
213 hb_font,
214 HB_FONT_FUNCS.0,
215 font as *const Font as *mut c_void,
216 None,
217 );
218
219 if servo_config::pref!(layout_variable_fonts_enabled) {
220 let variations = &font.variations();
221 if !variations.is_empty() {
222 let variations: Vec<_> = variations
223 .iter()
224 .map(|variation| hb_variation_t {
225 tag: variation.tag,
226
227 value: variation.value,
228 })
229 .collect();
230
231 hb_font_set_variations(hb_font, variations.as_ptr(), variations.len() as u32);
232 }
233 }
234
235 Shaper {
236 hb_face,
237 hb_font,
238 font,
239 }
240 }
241 }
242
243 fn shaped_glyph_data(
245 &self,
246 text: &str,
247 options: &ShapingOptions,
248 font_features: &[(Tag, u32)],
249 ) -> HarfbuzzGlyphShapingResult {
250 unsafe {
251 let hb_buffer: *mut hb_buffer_t = hb_buffer_create();
252 hb_buffer_set_cluster_level(hb_buffer, HB_BUFFER_CLUSTER_LEVEL_MONOTONE_CHARACTERS);
253 hb_buffer_set_direction(
254 hb_buffer,
255 if options.flags.contains(ShapingFlags::RTL_FLAG) {
256 HB_DIRECTION_RTL
257 } else {
258 HB_DIRECTION_LTR
259 },
260 );
261
262 let script =
263 hb_script_from_iso15924_tag(unicode_script_to_iso15924_tag(options.script));
264 hb_buffer_set_script(hb_buffer, script);
265
266 hb_buffer_add_utf8(
267 hb_buffer,
268 text.as_ptr() as *const c_char,
269 text.len() as c_int,
270 0,
271 text.len() as c_int,
272 );
273
274 let language = options.language;
275 let hb_language = hb_language_from_string(
276 language.as_str().as_ptr() as *const c_char,
277 language.as_str().len() as c_int,
278 );
279 hb_buffer_set_language(hb_buffer, hb_language);
280
281 let mut features: Vec<_> = font_features
282 .iter()
283 .map(|(tag, value)| hb_feature_t {
284 tag: u32::from_be_bytes(tag.to_be_bytes()),
285 value: *value,
286 start: 0,
287 end: hb_buffer_get_length(hb_buffer),
288 })
289 .collect();
290 hb_shape(
291 self.hb_font,
292 hb_buffer,
293 features.as_mut_ptr(),
294 features.len() as u32,
295 );
296
297 HarfbuzzGlyphShapingResult::new(hb_buffer)
298 }
299 }
300
301 pub(crate) fn shape_text(
302 &self,
303 text: &str,
304 options: &ShapingOptions,
305 font_features: &[(Tag, u32)],
306 ) -> ShapedText {
307 ShapedText::with_shaped_glyph_data(
308 text,
309 options,
310 &self.shaped_glyph_data(text, options, font_features),
311 )
312 }
313
314 pub(crate) fn baseline(&self) -> Option<FontBaseline> {
315 unsafe { (*self.font).table_for_tag(BASE)? };
316
317 let mut hanging_baseline = 0;
318 let mut alphabetic_baseline = 0;
319 let mut ideographic_baseline = 0;
320
321 unsafe {
322 hb_ot_layout_get_baseline(
323 self.hb_font,
324 HB_OT_LAYOUT_BASELINE_TAG_ROMAN,
325 HB_DIRECTION_LTR,
326 HB_OT_TAG_DEFAULT_SCRIPT,
327 HB_OT_TAG_DEFAULT_LANGUAGE,
328 &mut alphabetic_baseline as *mut _,
329 );
330
331 hb_ot_layout_get_baseline(
332 self.hb_font,
333 HB_OT_LAYOUT_BASELINE_TAG_HANGING,
334 HB_DIRECTION_LTR,
335 HB_OT_TAG_DEFAULT_SCRIPT,
336 HB_OT_TAG_DEFAULT_LANGUAGE,
337 &mut hanging_baseline as *mut _,
338 );
339
340 hb_ot_layout_get_baseline(
341 self.hb_font,
342 HB_OT_LAYOUT_BASELINE_TAG_IDEO_EMBOX_BOTTOM_OR_LEFT,
343 HB_DIRECTION_LTR,
344 HB_OT_TAG_DEFAULT_SCRIPT,
345 HB_OT_TAG_DEFAULT_LANGUAGE,
346 &mut ideographic_baseline as *mut _,
347 );
348 }
349
350 Some(FontBaseline {
351 ideographic_baseline: Shaper::fixed_to_float(ideographic_baseline) as f32,
352 alphabetic_baseline: Shaper::fixed_to_float(alphabetic_baseline) as f32,
353 hanging_baseline: Shaper::fixed_to_float(hanging_baseline) as f32,
354 })
355 }
356
357 fn float_to_fixed(f: f64) -> i32 {
358 float_to_fixed(16, f)
359 }
360
361 fn fixed_to_float(i: hb_position_t) -> f64 {
362 fixed_to_float(16, i)
363 }
364}
365
366struct FontFuncs(*mut hb_font_funcs_t);
368
369unsafe impl Sync for FontFuncs {}
370unsafe impl Send for FontFuncs {}
371
372static HB_FONT_FUNCS: LazyLock<FontFuncs> = LazyLock::new(|| unsafe {
373 let hb_funcs = hb_font_funcs_create();
374 hb_font_funcs_set_nominal_glyph_func(hb_funcs, Some(glyph_func), ptr::null_mut(), None);
375 hb_font_funcs_set_glyph_h_advance_func(
376 hb_funcs,
377 Some(glyph_h_advance_func),
378 ptr::null_mut(),
379 None,
380 );
381
382 FontFuncs(hb_funcs)
383});
384
385extern "C" fn glyph_func(
386 _: *mut hb_font_t,
387 font_data: *mut c_void,
388 unicode: hb_codepoint_t,
389 glyph: *mut hb_codepoint_t,
390 _: *mut c_void,
391) -> hb_bool_t {
392 let font: *const Font = font_data as *const Font;
393 assert!(!font.is_null());
394
395 match unsafe { (*font).glyph_index(char::from_u32(unicode).unwrap()) } {
396 Some(g) => {
397 unsafe { *glyph = g as hb_codepoint_t };
398 true as hb_bool_t
399 },
400 None => false as hb_bool_t,
401 }
402}
403
404extern "C" fn glyph_h_advance_func(
405 _: *mut hb_font_t,
406 font_data: *mut c_void,
407 glyph: hb_codepoint_t,
408 _: *mut c_void,
409) -> hb_position_t {
410 let font: *mut Font = font_data as *mut Font;
411 assert!(!font.is_null());
412
413 let advance = unsafe { (*font).glyph_h_advance(glyph as GlyphId) };
414 Shaper::float_to_fixed(advance)
415}
416
417extern "C" fn font_table_func(
419 _: *mut hb_face_t,
420 tag: hb_tag_t,
421 user_data: *mut c_void,
422) -> *mut hb_blob_t {
423 let font = user_data as *const Font;
425 assert!(!font.is_null());
426
427 let Some(font_table) = (unsafe { (*font).table_for_tag(Tag::from_u32(tag)) }) else {
429 return ptr::null_mut();
430 };
431
432 let font_table_ptr = Box::into_raw(Box::new(font_table));
436
437 let buf = unsafe { (*font_table_ptr).buffer() };
438 let blob = unsafe {
440 hb_blob_create(
441 buf.as_ptr() as *const c_char,
442 buf.len() as c_uint,
443 HB_MEMORY_MODE_READONLY,
444 font_table_ptr as *mut c_void,
445 Some(destroy_blob_func),
446 )
447 };
448
449 assert!(!blob.is_null());
450 blob
451}
452
453extern "C" fn destroy_blob_func(font_table_ptr: *mut c_void) {
454 unsafe {
455 drop(Box::from_raw(font_table_ptr as *mut FontTable));
456 }
457}