use alloc::vec::Vec;
use core::ops::Range;
use crate::ttf_parser::opentype_layout::{FeatureIndex, LanguageIndex, LookupIndex, VariationIndex, ScriptIndex};
use crate::{tag, Face, Language, Mask, Tag, Script};
use crate::buffer::{glyph_flag, Buffer};
use crate::plan::ShapePlan;
use super::{LayoutTableExt, TableIndex};
pub struct Map {
found_script: [bool; 2],
chosen_script: [Option<Tag>; 2],
global_mask: Mask,
features: Vec<FeatureMap>,
lookups: [Vec<LookupMap>; 2],
stages: [Vec<StageMap>; 2],
}
#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord)]
pub struct FeatureMap {
tag: Tag,
index: [Option<FeatureIndex>; 2],
stage: [usize; 2],
shift: u32,
mask: Mask,
one_mask: Mask,
auto_zwnj: bool,
auto_zwj: bool,
random: bool,
}
#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord)]
pub struct LookupMap {
pub index: LookupIndex,
pub auto_zwnj: bool,
pub auto_zwj: bool,
pub random: bool,
pub mask: Mask,
}
#[derive(Clone, Copy)]
pub struct StageMap {
pub last_lookup: usize,
pub pause_func: Option<PauseFunc>,
}
pub type PauseFunc = fn(&ShapePlan, &Face, &mut Buffer);
impl Map {
pub const MAX_BITS: u32 = 8;
pub const MAX_VALUE: u32 = (1 << Self::MAX_BITS) - 1;
#[inline]
pub fn found_script(&self, table_index: TableIndex) -> bool {
self.found_script[table_index]
}
#[inline]
pub fn chosen_script(&self, table_index: TableIndex) -> Option<Tag> {
self.chosen_script[table_index]
}
#[inline]
pub fn global_mask(&self) -> Mask {
self.global_mask
}
#[inline]
pub fn mask(&self, feature_tag: Tag) -> (Mask, u32) {
self.features
.binary_search_by_key(&feature_tag, |f| f.tag)
.map_or((0, 0), |idx| (self.features[idx].mask, self.features[idx].shift))
}
#[inline]
pub fn one_mask(&self, feature_tag: Tag) -> Mask {
self.features
.binary_search_by_key(&feature_tag, |f| f.tag)
.map_or(0, |idx| self.features[idx].one_mask)
}
#[inline]
pub fn feature_index(&self, table_index: TableIndex, feature_tag: Tag) -> Option<FeatureIndex> {
self.features
.binary_search_by_key(&feature_tag, |f| f.tag)
.ok()
.and_then(|idx| self.features[idx].index[table_index])
}
#[inline]
pub fn feature_stage(&self, table_index: TableIndex, feature_tag: Tag) -> Option<usize> {
self.features
.binary_search_by_key(&feature_tag, |f| f.tag)
.map(|idx| self.features[idx].stage[table_index])
.ok()
}
#[inline]
pub fn stages(&self, table_index: TableIndex) -> &[StageMap] {
&self.stages[table_index]
}
#[inline]
pub fn lookup(&self, table_index: TableIndex, index: usize) -> &LookupMap {
&self.lookups[table_index][index]
}
#[inline]
pub fn stage_lookups(&self, table_index: TableIndex, stage: usize) -> &[LookupMap] {
&self.lookups[table_index][self.stage_lookup_range(table_index, stage)]
}
#[inline]
pub fn stage_lookup_range(&self, table_index: TableIndex, stage: usize) -> Range<usize> {
let stages = &self.stages[table_index];
let lookups = &self.lookups[table_index];
let start = stage.checked_sub(1).map_or(0, |prev| stages[prev].last_lookup);
let end = stages.get(stage).map_or(lookups.len(), |curr| curr.last_lookup);
start..end
}
}
bitflags::bitflags! {
#[derive(Default)]
pub struct FeatureFlags: u32 {
const GLOBAL = 0x01;
const HAS_FALLBACK = 0x02;
const MANUAL_ZWNJ = 0x04;
const MANUAL_ZWJ = 0x08;
const GLOBAL_SEARCH = 0x10;
const RANDOM = 0x20;
const MANUAL_JOINERS = Self::MANUAL_ZWNJ.bits | Self::MANUAL_ZWJ.bits;
const GLOBAL_MANUAL_JOINERS = Self::GLOBAL.bits | Self::MANUAL_JOINERS.bits;
const GLOBAL_HAS_FALLBACK = Self::GLOBAL.bits | Self::HAS_FALLBACK.bits;
}
}
pub struct MapBuilder<'a> {
face: &'a Face<'a>,
found_script: [bool; 2],
script_index: [Option<ScriptIndex>; 2],
chosen_script: [Option<Tag>; 2],
lang_index: [Option<LanguageIndex>; 2],
current_stage: [usize; 2],
feature_infos: Vec<FeatureInfo>,
stages: [Vec<StageInfo>; 2],
}
#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord)]
struct FeatureInfo {
tag: Tag,
seq: usize,
max_value: u32,
flags: FeatureFlags,
default_value: u32,
stage: [usize; 2],
}
#[derive(Clone, Copy)]
struct StageInfo {
index: usize,
pause_func: Option<PauseFunc>,
}
impl<'a> MapBuilder<'a> {
pub fn new(face: &'a Face<'a>, script: Option<Script>, language: Option<&Language>) -> Self {
let (script_tags, lang_tags) = tag::tags_from_script_and_language(script, language);
let mut found_script = [false; 2];
let mut script_index = [None; 2];
let mut chosen_script = [None; 2];
let mut lang_index = [None; 2];
for (table_index, table) in face.layout_tables() {
if let Some((found, idx, tag)) = table.select_script(&script_tags) {
chosen_script[table_index] = Some(tag);
found_script[table_index] = found;
script_index[table_index] = Some(idx);
if let Some(idx) = table.select_script_language(idx, &lang_tags) {
lang_index[table_index] = Some(idx);
}
}
}
Self {
face,
found_script,
script_index,
chosen_script,
lang_index,
current_stage: [0, 0],
feature_infos: Vec::new(),
stages: [Vec::new(), Vec::new()],
}
}
#[inline]
pub fn chosen_script(&self, table_index: TableIndex) -> Option<Tag> {
self.chosen_script[table_index]
}
#[inline]
pub fn add_feature(&mut self, tag: Tag, flags: FeatureFlags, value: u32) {
if !tag.is_null() {
let seq = self.feature_infos.len();
self.feature_infos.push(FeatureInfo {
tag,
seq,
max_value: value,
flags,
default_value: if flags.contains(FeatureFlags::GLOBAL) { value } else { 0 },
stage: self.current_stage,
});
}
}
#[inline]
pub fn enable_feature(&mut self, tag: Tag, flags: FeatureFlags, value: u32) {
self.add_feature(tag, flags | FeatureFlags::GLOBAL, value);
}
#[inline]
pub fn disable_feature(&mut self, tag: Tag) {
self.add_feature(tag, FeatureFlags::GLOBAL, 0);
}
#[inline]
pub fn add_gsub_pause(&mut self, pause: Option<PauseFunc>) {
self.add_pause(TableIndex::GSUB, pause);
}
#[inline]
pub fn add_gpos_pause(&mut self, pause: Option<PauseFunc>) {
self.add_pause(TableIndex::GPOS, pause);
}
fn add_pause(&mut self, table_index: TableIndex, pause: Option<PauseFunc>) {
self.stages[table_index].push(StageInfo {
index: self.current_stage[table_index],
pause_func: pause,
});
self.current_stage[table_index] += 1;
}
const GLOBAL_BIT_MASK: Mask = glyph_flag::DEFINED + 1;
const GLOBAL_BIT_SHIFT: u32 = glyph_flag::DEFINED.count_ones();
pub fn compile(&mut self) -> Map {
let mut required_index = [None; 2];
let mut required_tag = [None; 2];
for (table_index, table) in self.face.layout_tables() {
if let Some(script) = self.script_index[table_index] {
let lang = self.lang_index[table_index];
if let Some((idx, tag)) = table.get_required_language_feature(script, lang) {
required_index[table_index] = Some(idx);
required_tag[table_index] = Some(tag);
}
}
}
let (features, required_stage, global_mask) = self.collect_feature_maps(required_tag);
self.add_gsub_pause(None);
self.add_gpos_pause(None);
let (lookups, stages) = self.collect_lookup_stages(&features, required_index, required_stage);
Map {
found_script: self.found_script,
chosen_script: self.chosen_script,
global_mask,
features,
lookups,
stages,
}
}
fn collect_feature_maps(
&mut self,
required_tag: [Option<Tag>; 2],
) -> (Vec<FeatureMap>, [usize; 2], Mask) {
let mut map_features = Vec::new();
let mut required_stage = [0; 2];
let mut global_mask = Self::GLOBAL_BIT_MASK;
let mut next_bit = Self::GLOBAL_BIT_SHIFT + 1;
self.dedup_feature_infos();
for info in &self.feature_infos {
let bits_needed = if info.flags.contains(FeatureFlags::GLOBAL) && info.max_value == 1 {
0
} else {
let v = info.max_value;
let num_bits = 8 * core::mem::size_of_val(&v) as u32 - v.leading_zeros();
Map::MAX_BITS.min(num_bits)
};
let bits_available = 8 * core::mem::size_of::<Mask>() as u32;
if info.max_value == 0 || next_bit + bits_needed > bits_available {
continue;
}
let mut found = false;
let mut feature_index = [None; 2];
for (table_index, table) in self.face.layout_tables() {
if required_tag[table_index] == Some(info.tag) {
required_stage[table_index] = info.stage[table_index];
}
if let Some(script) = self.script_index[table_index] {
let lang = self.lang_index[table_index];
if let Some(idx) = table.find_language_feature(script, lang, info.tag) {
feature_index[table_index] = Some(idx);
found = true;
}
}
}
if !found && info.flags.contains(FeatureFlags::GLOBAL_SEARCH) {
for (table_index, table) in self.face.layout_tables() {
if let Some(idx) = table.features.index(info.tag) {
feature_index[table_index] = Some(idx);
found = true;
}
}
}
if !found && !info.flags.contains(FeatureFlags::HAS_FALLBACK) {
continue;
}
let (shift, mask) = if info.flags.contains(FeatureFlags::GLOBAL) && info.max_value == 1 {
(Self::GLOBAL_BIT_SHIFT, Self::GLOBAL_BIT_MASK)
} else {
let shift = next_bit;
let mask = (1 << (next_bit + bits_needed)) - (1 << next_bit);
next_bit += bits_needed;
global_mask |= (info.default_value << shift) & mask;
(shift, mask)
};
map_features.push(FeatureMap {
tag: info.tag,
index: feature_index,
stage: info.stage,
shift,
mask,
one_mask: (1 << shift) & mask,
auto_zwnj: !info.flags.contains(FeatureFlags::MANUAL_ZWNJ),
auto_zwj: !info.flags.contains(FeatureFlags::MANUAL_ZWJ),
random: info.flags.contains(FeatureFlags::RANDOM),
});
}
(map_features, required_stage, global_mask)
}
fn dedup_feature_infos(&mut self) {
let feature_infos = &mut self.feature_infos;
if feature_infos.is_empty() {
return;
}
feature_infos.sort();
let mut j = 0;
for i in 1..feature_infos.len() {
if feature_infos[i].tag != feature_infos[j].tag {
j += 1;
feature_infos[j] = feature_infos[i];
} else {
if feature_infos[i].flags.contains(FeatureFlags::GLOBAL) {
feature_infos[j].flags |= FeatureFlags::GLOBAL;
feature_infos[j].max_value = feature_infos[i].max_value;
feature_infos[j].default_value = feature_infos[i].default_value;
} else {
if feature_infos[j].flags.contains(FeatureFlags::GLOBAL) {
feature_infos[j].flags ^= FeatureFlags::GLOBAL;
}
feature_infos[j].max_value = feature_infos[j].max_value.max(feature_infos[i].max_value);
}
let flags = feature_infos[i].flags & FeatureFlags::HAS_FALLBACK;
feature_infos[j].flags |= flags;
feature_infos[j].stage[0] = feature_infos[j].stage[0].min(feature_infos[i].stage[0]);
feature_infos[j].stage[1] = feature_infos[j].stage[1].min(feature_infos[i].stage[1]);
}
}
feature_infos.truncate(j + 1);
}
fn collect_lookup_stages(
&self,
map_features: &[FeatureMap],
required_feature_index: [Option<FeatureIndex>; 2],
required_feature_stage: [usize; 2],
) -> ([Vec<LookupMap>; 2], [Vec<StageMap>; 2]) {
let mut map_lookups = [Vec::new(), Vec::new()];
let mut map_stages = [Vec::new(), Vec::new()];
for table_index in TableIndex::iter() {
let mut stage_index = 0;
let mut last_lookup = 0;
let coords = self.face.ttfp_face.variation_coordinates();
let variation_index = self.face
.layout_table(table_index)
.and_then(|t| t.variations?.find_index(coords));
for stage in 0..self.current_stage[table_index] {
if let Some(feature_index) = required_feature_index[table_index] {
if required_feature_stage[table_index] == stage {
self.add_lookups(
&mut map_lookups[table_index],
table_index,
feature_index,
variation_index,
Self::GLOBAL_BIT_MASK,
true,
true,
false,
);
}
}
for feature in map_features {
if let Some(feature_index) = feature.index[table_index] {
if feature.stage[table_index] == stage {
self.add_lookups(
&mut map_lookups[table_index],
table_index,
feature_index,
variation_index,
feature.mask,
feature.auto_zwnj,
feature.auto_zwj,
feature.random,
);
}
}
}
let lookups = &mut map_lookups[table_index];
let len = lookups.len();
if last_lookup < len {
lookups[last_lookup..].sort();
let mut j = last_lookup;
for i in j+1..len {
if lookups[i].index != lookups[j].index {
j += 1;
lookups[j] = lookups[i];
} else {
lookups[j].mask |= lookups[i].mask;
lookups[j].auto_zwnj &= lookups[i].auto_zwnj;
lookups[j].auto_zwj &= lookups[i].auto_zwj;
}
}
lookups.truncate(j + 1);
}
last_lookup = lookups.len();
if let Some(info) = self.stages[table_index].get(stage_index) {
if info.index == stage {
map_stages[table_index].push(StageMap {
last_lookup,
pause_func: info.pause_func,
});
stage_index += 1;
}
}
}
}
(map_lookups, map_stages)
}
fn add_lookups(
&self,
lookups: &mut Vec<LookupMap>,
table_index: TableIndex,
feature_index: FeatureIndex,
variation_index: Option<VariationIndex>,
mask: Mask,
auto_zwnj: bool,
auto_zwj: bool,
random: bool,
) -> Option<()> {
let table = self.face.layout_table(table_index)?;
let lookup_count = table.lookups.len();
let feature = match variation_index {
Some(idx) => {
table.variations
.and_then(|var| var.find_substitute(feature_index, idx))
.or_else(|| table.features.get(feature_index))?
}
None => table.features.get(feature_index)?,
};
for index in feature.lookup_indices {
if index < lookup_count {
lookups.push(LookupMap { mask, index, auto_zwnj, auto_zwj, random });
}
}
Some(())
}
}