blob: 26c47e373d9e8727bf287de1a5303f463d840224 [file]
#![allow(non_upper_case_globals)]
// C enum becomes i32 on some systems (eg. Windows).
#![allow(clippy::unnecessary_cast)]
use super::hb::*;
use std::ffi::c_void;
use std::mem::{align_of, offset_of, size_of, transmute};
use std::ptr::null_mut;
use harfrust::{
funcs::{AdvanceWidthBatch, BuiltinFontFuncs, FontFuncs},
FontRef, GlyphExtents, GlyphFlags as HRGlyphFlags, GlyphInfo as HRGlyphInfo,
GlyphPosition as HRGlyphPosition, NormalizedCoord, ShapeOptions, Shaper, ShaperData,
ShaperInstance, Tag,
};
use read_fonts::types::GlyphId;
use smallvec::SmallVec;
type HRFeatureVec = SmallVec<[harfrust::Feature; 4]>;
const _: () = {
assert!(size_of::<hb_glyph_info_t>() == size_of::<HRGlyphInfo>());
assert!(align_of::<hb_glyph_info_t>() == align_of::<HRGlyphInfo>());
assert!(offset_of!(hb_glyph_info_t, codepoint) == offset_of!(HRGlyphInfo, glyph_id));
assert!(offset_of!(hb_glyph_info_t, cluster) == offset_of!(HRGlyphInfo, cluster));
assert!(
hb_glyph_flags_t_HB_GLYPH_FLAG_UNSAFE_TO_BREAK as u32
== HRGlyphFlags::UNSAFE_TO_BREAK.to_bits()
);
assert!(
hb_glyph_flags_t_HB_GLYPH_FLAG_UNSAFE_TO_CONCAT as u32
== HRGlyphFlags::UNSAFE_TO_CONCAT.to_bits()
);
assert!(
hb_glyph_flags_t_HB_GLYPH_FLAG_SAFE_TO_INSERT_TATWEEL as u32
== HRGlyphFlags::SAFE_TO_INSERT_TATWEEL.to_bits()
);
assert!(
hb_glyph_flags_t_HB_GLYPH_FLAG_DEFINED as u32
== (HRGlyphFlags::UNSAFE_TO_BREAK.to_bits()
| HRGlyphFlags::UNSAFE_TO_CONCAT.to_bits()
| HRGlyphFlags::SAFE_TO_INSERT_TATWEEL.to_bits())
);
assert!(size_of::<hb_glyph_position_t>() == size_of::<HRGlyphPosition>());
assert!(align_of::<hb_glyph_position_t>() == align_of::<HRGlyphPosition>());
assert!(offset_of!(hb_glyph_position_t, x_advance) == offset_of!(HRGlyphPosition, x_advance));
assert!(offset_of!(hb_glyph_position_t, y_advance) == offset_of!(HRGlyphPosition, y_advance));
assert!(offset_of!(hb_glyph_position_t, x_offset) == offset_of!(HRGlyphPosition, x_offset));
assert!(offset_of!(hb_glyph_position_t, y_offset) == offset_of!(HRGlyphPosition, y_offset));
};
pub struct HBHarfRustFaceData<'a> {
face_blob: *mut hb_blob_t,
font_ref: FontRef<'a>,
shaper_data: ShaperData,
}
#[unsafe(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: *const u8 = hb_blob_get_data(face_blob, null_mut()).cast();
if blob_data.is_null() {
return null_mut();
}
let face_data = std::slice::from_raw_parts(blob_data, 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
}
#[unsafe(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 {
// Keep the instance alive because `shaper` borrows variation data from it.
_shaper_instance: Box<ShaperInstance>,
shaper: Shaper<'static>,
}
struct HBHarfBuzzFontFuncs {
font: *mut hb_font_t,
}
impl FontFuncs for HBHarfBuzzFontFuncs {
fn nominal_glyph(&mut self, _: &BuiltinFontFuncs, c: u32) -> Option<GlyphId> {
let mut glyph = 0;
if unsafe { hb_font_get_nominal_glyph(self.font, c, &mut glyph) } != 0 {
Some(GlyphId::new(glyph))
} else {
None
}
}
fn variant_glyph(&mut self, _: &BuiltinFontFuncs, c: u32, vs: u32) -> Option<GlyphId> {
let mut glyph = 0;
if unsafe { hb_font_get_variation_glyph(self.font, c, vs, &mut glyph) } != 0 {
Some(GlyphId::new(glyph))
} else {
None
}
}
fn advance_width(&mut self, _: &BuiltinFontFuncs, glyph: GlyphId) -> i32 {
unsafe { hb_font_get_glyph_h_advance(self.font, glyph.to_u32()) }
}
fn populate_advance_widths(&mut self, _: &BuiltinFontFuncs, batch: AdvanceWidthBatch<'_>) {
let raw = batch.into_raw();
unsafe {
hb_font_get_glyph_h_advances(
self.font,
raw.len as u32,
raw.gids,
raw.gid_stride as u32,
raw.advances,
raw.advance_stride as u32,
);
}
}
fn advance_height(&mut self, _: &BuiltinFontFuncs, glyph: GlyphId) -> i32 {
unsafe { hb_font_get_glyph_v_advance(self.font, glyph.to_u32()) }
}
fn vertical_origin(&mut self, _: &BuiltinFontFuncs, glyph: GlyphId) -> (i32, i32) {
let mut x = 0;
let mut y = 0;
unsafe {
hb_font_get_glyph_v_origin(self.font, glyph.to_u32(), &mut x, &mut y);
}
(x, y)
}
fn extents(&mut self, _: &BuiltinFontFuncs, glyph: GlyphId) -> Option<GlyphExtents> {
let mut extents = hb_glyph_extents_t {
x_bearing: 0,
y_bearing: 0,
width: 0,
height: 0,
};
if unsafe { hb_font_get_glyph_extents(self.font, glyph.to_u32(), &mut extents) } != 0 {
Some(GlyphExtents {
x_bearing: extents.x_bearing,
y_bearing: extents.y_bearing,
width: extents.width,
height: extents.height,
})
} else {
None
}
}
}
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)
}
#[unsafe(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_instance,
shaper: transmute::<harfrust::Shaper<'_>, harfrust::Shaper<'_>>(shaper),
});
let hr_font_data_ptr = Box::into_raw(hr_font_data);
hr_font_data_ptr as *mut c_void
}
#[unsafe(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<harfrust::Language> {
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) };
harfrust::Language::new(language_str.to_bytes())
}
#[unsafe(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
}
#[unsafe(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);
}
fn hb_feature_to_hr_feature(features: *const hb_feature_t, num_features: u32) -> HRFeatureVec {
if features.is_null() {
SmallVec::new()
} else {
let features = unsafe { 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()
}
}
#[unsafe(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,
features: *const hb_feature_t,
num_features: u32,
) -> *mut c_void {
let font_data = font_data as *const HBHarfRustFontData;
let script = harfrust::Script::from_iso15924_tag(Tag::from_u32(script as u32));
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 features = hb_feature_to_hr_feature(features, num_features);
let shaper = &(*font_data).shaper;
let hr_shape_plan =
harfrust::ShapePlan::new(shaper, direction, script, language.as_ref(), &features);
let hr_shape_plan = Box::new(hr_shape_plan);
Box::into_raw(hr_shape_plan) as *mut c_void
}
#[unsafe(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);
}
#[unsafe(no_mangle)]
pub unsafe extern "C" fn _hb_harfrust_shape_rs(
font_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 u32,
pre_context_length: u32,
post_context: *const u32,
post_context_length: u32,
features: *const hb_feature_t,
num_features: u32,
) -> hb_bool_t {
let font_data = font_data as *const HBHarfRustFontData;
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 as u32));
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 as u32))
.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());
let infos = std::slice::from_raw_parts(infos.cast(), count as usize);
if !hr_buffer.push_glyph_infos(infos) {
return false as hb_bool_t;
}
let pre_context = std::slice::from_raw_parts(pre_context, pre_context_length as usize);
hr_buffer.set_pre_context_codepoints(pre_context);
let post_context = std::slice::from_raw_parts(post_context, post_context_length as usize);
hr_buffer.set_post_context_codepoints(post_context);
let ptem = hb_font_get_ptem(font);
let ptem = if ptem > 0.0 { Some(ptem) } else { None };
let features = hb_feature_to_hr_feature(features, num_features);
let shape_plan = (shape_plan as *const harfrust::ShapePlan).as_ref();
let mut x_scale = 0;
let mut y_scale = 0;
hb_font_get_scale(font, &mut x_scale, &mut y_scale);
let mut font_funcs = HBHarfBuzzFontFuncs { font };
let glyphs = (*font_data).shaper.shape(
hr_buffer,
ShapeOptions::new()
.plan(shape_plan)
.scale_separate(Some((x_scale, y_scale)))
.point_size(ptem)
.features(&features)
.font_funcs(Some(&mut font_funcs)),
);
hb_buffer_set_content_type(
buffer,
hb_buffer_content_type_t_HB_BUFFER_CONTENT_TYPE_GLYPHS,
);
let count = glyphs.len();
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;
}
std::ptr::copy_nonoverlapping(glyphs.glyph_infos().as_ptr().cast(), infos, count);
std::ptr::copy_nonoverlapping(glyphs.glyph_positions().as_ptr().cast(), positions, count);
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
}