Skip to content

Commit

Permalink
[fontbe] Add a 'PendingLookup' type
Browse files Browse the repository at this point in the history
This will let us pre-compute and store the lookupflags and any mark
filtering set.
  • Loading branch information
cmyr committed Mar 15, 2024
1 parent 7de98d2 commit 2378b80
Show file tree
Hide file tree
Showing 4 changed files with 75 additions and 26 deletions.
10 changes: 9 additions & 1 deletion fontbe/src/features.rs
Original file line number Diff line number Diff line change
Expand Up @@ -39,11 +39,13 @@ use crate::{
orchestration::{AnyWorkId, BeWork, Context, FeaAst, FeaRsKerns, FeaRsMarks, WorkId},
};

mod common;
mod kern;
mod marks;
mod ot_tags;
mod properties;

pub(crate) use common::PendingLookup;
pub use kern::{create_gather_ir_kerning_work, create_kern_segment_work, create_kerns_work};
pub use marks::create_mark_work;

Expand Down Expand Up @@ -219,7 +221,13 @@ impl<'a> FeatureWriter<'a> {
.kerning
.lookups
.iter()
.map(|lookups| builder.add_lookup(LookupFlag::empty(), None, lookups.to_owned()))
.map(|lookup| {
builder.add_lookup(
lookup.flags,
lookup.mark_filter_set.clone(),
lookup.subtables.clone(),
)
})
.collect::<Vec<_>>();

for (feature, ids) in &self.kerning.features {
Expand Down
26 changes: 26 additions & 0 deletions fontbe/src/features/common.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
use fea_rs::GlyphSet;
use write_fonts::tables::layout::LookupFlag;

/// A lookup generated outside of user FEA
///
/// This will be merged into any user-provided features during compilation.
#[derive(Debug, Default, Clone, PartialEq, serde::Serialize, serde::Deserialize)]
pub struct PendingLookup<T> {
pub(crate) subtables: Vec<T>,
pub(crate) flags: LookupFlag,
pub(crate) mark_filter_set: Option<GlyphSet>,
}

impl<T> PendingLookup<T> {
pub(crate) fn new(
subtables: Vec<T>,
flags: LookupFlag,
mark_filter_set: Option<GlyphSet>,
) -> Self {
Self {
subtables,
flags,
mark_filter_set,
}
}
}
61 changes: 38 additions & 23 deletions fontbe/src/features/kern.rs
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ use log::debug;
use ordered_float::OrderedFloat;
use write_fonts::{
read::{tables::gsub::Gsub, FontRead, ReadError},
tables::gdef::GlyphClassDef,
tables::{gdef::GlyphClassDef, layout::LookupFlag},
types::{GlyphId, Tag},
};

Expand All @@ -43,7 +43,7 @@ use crate::{
},
};

use super::properties::CharMap;
use super::{properties::CharMap, PendingLookup};

/// On Linux it took ~0.01 ms per loop, try to get enough to make fan out worthwhile
/// based on empirical testing
Expand Down Expand Up @@ -463,11 +463,14 @@ impl KerningGatherWork {
/// <https://github.com/googlefonts/ufo2ft/blob/f6b4f42460b/Lib/ufo2ft/featureWriters/kernFeatureWriter.py#L772>
fn assign_lookups_to_scripts(
&self,
lookups: BTreeMap<BTreeSet<UnicodeShortName>, Vec<PairPosBuilder>>,
lookups: BTreeMap<BTreeSet<UnicodeShortName>, Vec<PendingLookup<PairPosBuilder>>>,
ast: &ParseTree,
// one of 'kern' or 'dist'
current_feature: Tag,
) -> (Vec<Vec<PairPosBuilder>>, BTreeMap<FeatureKey, Vec<usize>>) {
) -> (
Vec<PendingLookup<PairPosBuilder>>,
BTreeMap<FeatureKey, Vec<usize>>,
) {
let dflt_langs = vec![DFLT_LANG];

let is_kern_feature = current_feature == KERN;
Expand All @@ -482,13 +485,15 @@ impl KerningGatherWork {

// in python this part happens earlier, as part of splitKerning.
for (scripts, lookups) in lookups {
let idx = ordered_lookups.len();
ordered_lookups.push(lookups);
for script in scripts {
lookups_by_script
.entry(script)
.or_insert(Vec::new())
.push(idx);
for lookup in lookups {
let idx = ordered_lookups.len();
ordered_lookups.push(lookup);
for script in &scripts {
lookups_by_script
.entry(script.to_owned())
.or_insert(Vec::new())
.push(idx);
}
}
}

Expand Down Expand Up @@ -568,10 +573,10 @@ impl KerningGatherWork {

fn debug_ordered_lookups(
features: &BTreeMap<FeatureKey, Vec<usize>>,
lookups: &[Vec<PairPosBuilder>],
lookups: &[PendingLookup<PairPosBuilder>],
) {
for (i, subtables) in lookups.iter().enumerate() {
let total_rules = subtables.iter().map(|x| x.len()).sum::<usize>();
for (i, lookup) in lookups.iter().enumerate() {
let total_rules = lookup.subtables.iter().map(|x| x.len()).sum::<usize>();
log::trace!("lookup {i}, {total_rules} rules");
}

Expand Down Expand Up @@ -641,7 +646,7 @@ impl KernSplitContext {
fn make_lookups(
&self,
pairs: &[&PairPosEntry],
) -> BTreeMap<BTreeSet<UnicodeShortName>, Vec<PairPosBuilder>> {
) -> BTreeMap<BTreeSet<UnicodeShortName>, Vec<PendingLookup<PairPosBuilder>>> {
if !self.opts.ignore_marks {
let pairs = pairs.iter().map(|x| Cow::Borrowed(*x)).collect::<Vec<_>>();
return self.make_split_script_kern_lookups(&pairs, false);
Expand All @@ -665,21 +670,18 @@ impl KernSplitContext {
/// Returns a map of scripts: `[lookup]`, where 'scripts' is a set of scripts
/// referencing the lookups.
///
/// Although this method always/only returns a single lookup per unique set
/// of scripts, we wrap it in a vec because that is what we need at the
/// call site.
// <https://github.com/googlefonts/ufo2ft/blob/f6b4f42460b/Lib/ufo2ft/featureWriters/kernFeatureWriter.py#L694>
fn make_split_script_kern_lookups(
&self,
pairs: &[Cow<PairPosEntry>],
//TODO: handle marks
_are_marks: bool,
) -> BTreeMap<BTreeSet<UnicodeShortName>, Vec<PairPosBuilder>> {
) -> BTreeMap<BTreeSet<UnicodeShortName>, Vec<PendingLookup<PairPosBuilder>>> {
let mut lookups_by_script = BTreeMap::new();
let kerning_per_script = self.split_kerns(pairs);
let mut bidi_buf = HashSet::new(); // we can reuse this for each pair
for (scripts, pairs) in kerning_per_script {
let mut lookup = PairPosBuilder::default();
let mut builder = PairPosBuilder::default();
for mut pair in pairs {
bidi_buf.clear();
for (direction, glyphs) in &self.bidi_glyphs {
Expand All @@ -706,8 +708,9 @@ impl KernSplitContext {
if pair_is_rtl {
pair.make_rtl_compatible();
}
pair.add_to(&mut lookup);
pair.add_to(&mut builder);
}
let lookup = PendingLookup::new(vec![builder], LookupFlag::empty(), None);
lookups_by_script.insert(scripts, vec![lookup]);
}
lookups_by_script
Expand Down Expand Up @@ -1092,9 +1095,21 @@ mod tests {
.collect();

let cyr_rules = result.get(&cyrillic).unwrap();
assert_eq!(cyr_rules.iter().map(|x| x.len()).sum::<usize>(), 1);
assert_eq!(
cyr_rules
.iter()
.flat_map(|x| x.subtables.iter().map(|sub| sub.len()))
.sum::<usize>(),
1
);

let latn_rules = result.get(&latn).unwrap();
assert_eq!(latn_rules.iter().map(|x| x.len()).sum::<usize>(), 2);
assert_eq!(
latn_rules
.iter()
.flat_map(|x| x.subtables.iter().map(|sub| sub.len()))
.sum::<usize>(),
2
);
}
}
4 changes: 2 additions & 2 deletions fontbe/src/orchestration.rs
Original file line number Diff line number Diff line change
Expand Up @@ -64,7 +64,7 @@ use write_fonts::{
FontWrite,
};

use crate::{error::Error, paths::Paths};
use crate::{error::Error, features::PendingLookup, paths::Paths};

/// What exactly is being assembled from glyphs?
#[derive(Clone, Debug, PartialEq, Eq, Hash, PartialOrd, Ord)]
Expand Down Expand Up @@ -375,7 +375,7 @@ impl Persistable for FeaRsMarks {
#[derive(Debug, Default, Clone, Serialize, Deserialize, PartialEq)]
pub struct FeaRsKerns {
/// ordered!
pub lookups: Vec<Vec<PairPosBuilder>>,
pub lookups: Vec<PendingLookup<PairPosBuilder>>,
/// each value is a set of lookups, referenced by their order in array above
pub features: BTreeMap<FeatureKey, Vec<usize>>,
}
Expand Down

0 comments on commit 2378b80

Please sign in to comment.