#![allow(non_upper_case_globals)] use super::hb::*; use std::ffi::c_void; use std::mem::transmute; use std::ptr::null_mut; use std::str::FromStr; use harfrust::{FontRef, NormalizedCoord, Shaper, ShaperData, ShaperInstance, Tag}; pub struct HBHarfRustFaceData<'a> { face_blob: *mut hb_blob_t, font_ref: FontRef<'a>, shaper_data: ShaperData, } #[no_mangle] pub unsafe extern "C" fn _hb_harfrust_shaper_face_data_create_rs( face: *mut hb_face_t, ) -> *mut c_void { let face_index = hb_face_get_index(face); let face_blob = hb_face_reference_blob(face); let blob_length = hb_blob_get_length(face_blob); let blob_data = hb_blob_get_data(face_blob, null_mut()); if blob_data.is_null() { return null_mut(); } let face_data = std::slice::from_raw_parts(blob_data as *const u8, blob_length as usize); let font_ref = match FontRef::from_index(face_data, face_index) { Ok(f) => f, Err(_) => return null_mut(), }; let shaper_data = ShaperData::new(&font_ref); let hr_face_data = Box::new(HBHarfRustFaceData { face_blob, font_ref, shaper_data, }); Box::into_raw(hr_face_data) as *mut c_void } #[no_mangle] pub unsafe extern "C" fn _hb_harfrust_shaper_face_data_destroy_rs(data: *mut c_void) { let data = data as *mut HBHarfRustFaceData; let hr_face_data = Box::from_raw(data); let blob = hr_face_data.face_blob; hb_blob_destroy(blob); } pub struct HBHarfRustFontData { shaper_instance: Box, shaper: Shaper<'static>, } fn font_to_shaper_instance(font: *mut hb_font_t, font_ref: &FontRef<'_>) -> ShaperInstance { let mut num_coords: u32 = 0; let coords = unsafe { hb_font_get_var_coords_normalized(font, &mut num_coords) }; let coords = if coords.is_null() { &[] } else { unsafe { std::slice::from_raw_parts(coords, num_coords as usize) } }; let coords = coords.iter().map(|&v| NormalizedCoord::from_bits(v as i16)); ShaperInstance::from_coords(font_ref, coords) } #[no_mangle] pub unsafe extern "C" fn _hb_harfrust_shaper_font_data_create_rs( font: *mut hb_font_t, face_data: *const c_void, ) -> *mut c_void { let face_data = face_data as *const HBHarfRustFaceData; let font_ref = &(*face_data).font_ref; let shaper_instance = Box::new(font_to_shaper_instance(font, font_ref)); let shaper_instance_ref = &*(&*shaper_instance as *const _); let shaper = (*face_data) .shaper_data .shaper(font_ref) .instance(Some(shaper_instance_ref)) .build(); let hr_font_data = Box::new(HBHarfRustFontData { shaper_instance, shaper: transmute::, harfrust::Shaper<'_>>(shaper), }); let hr_font_data_ptr = Box::into_raw(hr_font_data); hr_font_data_ptr as *mut c_void } #[no_mangle] pub unsafe extern "C" fn _hb_harfrust_shaper_font_data_destroy_rs(data: *mut c_void) { let data = data as *mut HBHarfRustFontData; let _hr_font_data = Box::from_raw(data); } fn hb_language_to_hr_language(language: hb_language_t) -> Option { let language_str = unsafe { hb_language_to_string(language) }; if language_str.is_null() { return None; } let language_str = unsafe { std::ffi::CStr::from_ptr(language_str) }; let language_str = language_str.to_str().unwrap_or_default(); Some(harfrust::Language::from_str(language_str).unwrap()) } #[no_mangle] pub unsafe extern "C" fn _hb_harfrust_buffer_create_rs() -> *mut c_void { let hr_buffer = Box::new(harfrust::UnicodeBuffer::new()); Box::into_raw(hr_buffer) as *mut c_void } #[no_mangle] pub unsafe extern "C" fn _hb_harfrust_buffer_destroy_rs(data: *mut c_void) { let data = data as *mut harfrust::UnicodeBuffer; let _hr_buffer = Box::from_raw(data); } #[no_mangle] pub unsafe extern "C" fn _hb_harfrust_shape_plan_create_rs( font_data: *const c_void, script: hb_script_t, language: hb_language_t, direction: hb_direction_t, ) -> *mut c_void { let font_data = font_data as *const HBHarfRustFontData; let script = harfrust::Script::from_iso15924_tag(Tag::from_u32(script)); let language = hb_language_to_hr_language(language); let direction = match direction { hb_direction_t_HB_DIRECTION_LTR => harfrust::Direction::LeftToRight, hb_direction_t_HB_DIRECTION_RTL => harfrust::Direction::RightToLeft, hb_direction_t_HB_DIRECTION_TTB => harfrust::Direction::TopToBottom, hb_direction_t_HB_DIRECTION_BTT => harfrust::Direction::BottomToTop, _ => harfrust::Direction::Invalid, }; let shaper = &(*font_data).shaper; let hr_shape_plan = harfrust::ShapePlan::new(shaper, direction, script, language.as_ref(), &[]); let hr_shape_plan = Box::new(hr_shape_plan); Box::into_raw(hr_shape_plan) as *mut c_void } #[no_mangle] pub unsafe extern "C" fn _hb_harfrust_shape_plan_destroy_rs(data: *mut c_void) { let data = data as *mut harfrust::ShapePlan; let _hr_shape_plan = Box::from_raw(data); } #[no_mangle] pub unsafe extern "C" fn _hb_harfrust_shape_rs( font_data: *const c_void, face_data: *const c_void, shape_plan: *const c_void, hr_buffer_box: *const c_void, font: *mut hb_font_t, buffer: *mut hb_buffer_t, pre_context: *const u8, pre_context_length: u32, post_context: *const u8, post_context_length: u32, features: *const hb_feature_t, num_features: u32, ) -> hb_bool_t { let font_data = font_data as *const HBHarfRustFontData; let face_data = face_data as *const HBHarfRustFaceData; let font_ref = &(*face_data).font_ref; let hr_buffer_box = hr_buffer_box as *mut harfrust::UnicodeBuffer; let mut hr_buffer_box = Box::from_raw(hr_buffer_box); let mut hr_buffer = *hr_buffer_box; // Set buffer properties let cluster_level = hb_buffer_get_cluster_level(buffer); let cluster_level = match cluster_level { hb_buffer_cluster_level_t_HB_BUFFER_CLUSTER_LEVEL_MONOTONE_GRAPHEMES => { harfrust::BufferClusterLevel::MonotoneGraphemes } hb_buffer_cluster_level_t_HB_BUFFER_CLUSTER_LEVEL_MONOTONE_CHARACTERS => { harfrust::BufferClusterLevel::MonotoneCharacters } hb_buffer_cluster_level_t_HB_BUFFER_CLUSTER_LEVEL_CHARACTERS => { harfrust::BufferClusterLevel::Characters } hb_buffer_cluster_level_t_HB_BUFFER_CLUSTER_LEVEL_GRAPHEMES => { harfrust::BufferClusterLevel::Graphemes } _ => harfrust::BufferClusterLevel::default(), }; hr_buffer.set_cluster_level(cluster_level); let flags = hb_buffer_get_flags(buffer); hr_buffer.set_flags(harfrust::BufferFlags::from_bits_truncate(flags)); let not_found_variation_selector_glyph = hb_buffer_get_not_found_variation_selector_glyph(buffer); if not_found_variation_selector_glyph != u32::MAX { hr_buffer.set_not_found_variation_selector_glyph(not_found_variation_selector_glyph); } // Segment properties: let script = hb_buffer_get_script(buffer); let language = hb_buffer_get_language(buffer); let direction = hb_buffer_get_direction(buffer); // Convert to HarfRust types let script = harfrust::Script::from_iso15924_tag(Tag::from_u32(script)) .unwrap_or(harfrust::script::UNKNOWN); let language = hb_language_to_hr_language(language); let direction = match direction { hb_direction_t_HB_DIRECTION_LTR => harfrust::Direction::LeftToRight, hb_direction_t_HB_DIRECTION_RTL => harfrust::Direction::RightToLeft, hb_direction_t_HB_DIRECTION_TTB => harfrust::Direction::TopToBottom, hb_direction_t_HB_DIRECTION_BTT => harfrust::Direction::BottomToTop, _ => harfrust::Direction::Invalid, }; // Set properties on the buffer hr_buffer.set_script(script); if let Some(lang) = language { hr_buffer.set_language(lang); } hr_buffer.set_direction(direction); // Populate buffer let count = hb_buffer_get_length(buffer); let infos = hb_buffer_get_glyph_infos(buffer, null_mut()); hr_buffer.reserve(count as usize); for i in 0..count { let info = &*infos.add(i as usize); let unicode = info.codepoint; let cluster = info.cluster; hr_buffer.add(char::from_u32_unchecked(unicode), cluster); } let pre_context = std::slice::from_raw_parts(pre_context, pre_context_length as usize); hr_buffer.set_pre_context(str::from_utf8(pre_context).unwrap()); let post_context = std::slice::from_raw_parts(post_context, post_context_length as usize); hr_buffer.set_post_context(str::from_utf8(post_context).unwrap()); let ptem = hb_font_get_ptem(font); let ptem = if ptem > 0.0 { Some(ptem) } else { None }; let shaper = if ptem.is_some() { (*face_data) .shaper_data .shaper(font_ref) .instance(Some(&(*font_data).shaper_instance)) .point_size(ptem) .build() } else { (*font_data).shaper.clone() }; let features = if features.is_null() { Vec::new() } else { let features = std::slice::from_raw_parts(features, num_features as usize); features .iter() .map(|f| { let tag = f.tag; let value = f.value; let start = f.start; let end = f.end; harfrust::Feature { tag: Tag::from_u32(tag), value, start, end, } }) .collect::>() }; let glyphs = if shape_plan.is_null() { shaper.shape(hr_buffer, &features) } else { let shape_plan = shape_plan as *const harfrust::ShapePlan; shaper.shape_with_plan(shape_plan.as_ref().unwrap(), hr_buffer, &features) }; let count = glyphs.len(); hb_buffer_set_length(buffer, 0u32); hb_buffer_set_content_type( buffer, hb_buffer_content_type_t_HB_BUFFER_CONTENT_TYPE_GLYPHS, ); hb_buffer_set_length(buffer, count as u32); let mut count_out: u32 = 0; let infos = hb_buffer_get_glyph_infos(buffer, &mut count_out); let positions = hb_buffer_get_glyph_positions(buffer, null_mut()); if count != count_out as usize { return false as hb_bool_t; } let mut x_scale: i32 = 0; let mut y_scale: i32 = 0; hb_font_get_scale(font, &mut x_scale, &mut y_scale); let upem = shaper.units_per_em(); let upem = if upem > 0 { upem } else { 1000 }; let x_mult = if x_scale < 0 { -((-x_scale as i64) << 16) } else { (x_scale as i64) << 16 } / upem as i64; let y_mult = if y_scale < 0 { -((-y_scale as i64) << 16) } else { (y_scale as i64) << 16 } / upem as i64; let em_mult = |v: i32, mult: i64| -> hb_position_t { ((v as i64 * mult + 32768) >> 16) as hb_position_t }; for (i, (hr_info, hr_pos)) in glyphs .glyph_infos() .iter() .zip(glyphs.glyph_positions()) .enumerate() { let info = &mut *infos.add(i); let pos = &mut *positions.add(i); info.codepoint = hr_info.glyph_id; info.cluster = hr_info.cluster; info.mask = 0; if hr_info.unsafe_to_break() { info.mask |= hb_glyph_flags_t_HB_GLYPH_FLAG_UNSAFE_TO_BREAK; } if hr_info.unsafe_to_concat() { info.mask |= hb_glyph_flags_t_HB_GLYPH_FLAG_UNSAFE_TO_CONCAT; } if hr_info.safe_to_insert_tatweel() { info.mask |= hb_glyph_flags_t_HB_GLYPH_FLAG_SAFE_TO_INSERT_TATWEEL; } pos.x_advance = em_mult(hr_pos.x_advance, x_mult); pos.y_advance = em_mult(hr_pos.y_advance, y_mult); pos.x_offset = em_mult(hr_pos.x_offset, x_mult); pos.y_offset = em_mult(hr_pos.y_offset, y_mult); } let hr_buffer = glyphs.clear(); *hr_buffer_box = hr_buffer; // Move the buffer back into the box let _ = Box::into_raw(hr_buffer_box); // Prevent double free true as hb_bool_t }