use core::cmp;
use core::convert::TryFrom;
use core::num::NonZeroU16;
use crate::{loca, GlyphId, OutlineBuilder, Rect, BBox, NormalizedCoord};
use crate::parser::{Stream, Offset, Offset16, Offset32, LazyArray16, F2DOT14};
use crate::glyf::{self, Transform};
const PHANTOM_POINTS_LEN: usize = 4;
#[derive(Clone, Copy)]
enum GlyphVariationDataOffsets<'a> {
Short(LazyArray16<'a, Offset16>),
Long(LazyArray16<'a, Offset32>),
}
#[derive(Clone, Copy)]
pub struct Table<'a> {
axis_count: NonZeroU16,
shared_tuple_records: LazyArray16<'a, F2DOT14>,
offsets: GlyphVariationDataOffsets<'a>,
glyphs_variation_data: &'a [u8],
}
impl<'a> Table<'a> {
pub fn parse(data: &'a [u8]) -> Option<Self> {
let mut s = Stream::new(data);
let version: u32 = s.read()?;
if version != 0x00010000 {
return None;
}
let axis_count: u16 = s.read()?;
let shared_tuple_count: u16 = s.read()?;
let shared_tuples_offset: Offset32 = s.read()?;
let glyph_count: u16 = s.read()?;
let flags: u16 = s.read()?;
let glyph_variation_data_array_offset: Offset32 = s.read()?;
let axis_count = NonZeroU16::new(axis_count)?;
let shared_tuple_records = {
let mut sub_s = Stream::new_at(data, shared_tuples_offset.to_usize())?;
sub_s.read_array16(shared_tuple_count.checked_mul(axis_count.get())?)?
};
let glyphs_variation_data = data.get(glyph_variation_data_array_offset.to_usize()..)?;
let offsets = {
let offsets_count = glyph_count.checked_add(1)?;
let is_long_format = flags & 1 == 1;
if is_long_format {
GlyphVariationDataOffsets::Long(s.read_array16(offsets_count)?)
} else {
GlyphVariationDataOffsets::Short(s.read_array16(offsets_count)?)
}
};
Some(Table {
axis_count,
shared_tuple_records,
offsets,
glyphs_variation_data,
})
}
#[inline]
fn parse_variation_data(
&self,
glyph_id: GlyphId,
coordinates: &[NormalizedCoord],
points_len: u16,
tuples: &mut VariationTuples<'a>,
) -> Option<()> {
tuples.len = 0;
if coordinates.len() != usize::from(self.axis_count.get()) {
return None;
}
let next_glyph_id = glyph_id.0.checked_add(1)?;
let (start, end) = match self.offsets {
GlyphVariationDataOffsets::Short(ref array) => {
(array.get(glyph_id.0)?.to_usize() * 2, array.get(next_glyph_id)?.to_usize() * 2)
}
GlyphVariationDataOffsets::Long(ref array) => {
(array.get(glyph_id.0)?.to_usize(), array.get(next_glyph_id)?.to_usize())
}
};
if start == end {
return Some(());
}
let data = self.glyphs_variation_data.get(start..end)?;
parse_variation_data(coordinates, &self.shared_tuple_records, points_len, data, tuples)
}
}
pub(crate) fn outline(
loca_table: loca::Table,
glyf_table: &[u8],
gvar_table: &Table,
coordinates: &[NormalizedCoord],
glyph_id: GlyphId,
builder: &mut dyn OutlineBuilder,
) -> Option<Rect> {
let mut b = glyf::Builder::new(Transform::default(), Some(BBox::new()), builder);
let range = loca_table.glyph_range(glyph_id)?;
let glyph_data = glyf_table.get(range)?;
outline_var_impl(loca_table, glyf_table, gvar_table,
glyph_id, glyph_data, coordinates, 0, &mut b);
b.bbox.and_then(|bbox| bbox.to_rect())
}
fn outline_var_impl<'a>(
loca_table: loca::Table,
glyf_table: &[u8],
gvar_table: &Table,
glyph_id: GlyphId,
data: &[u8],
coordinates: &[NormalizedCoord],
depth: u8,
builder: &mut glyf::Builder,
) -> Option<()> {
if depth >= glyf::MAX_COMPONENTS {
return None;
}
let mut s = Stream::new(data);
let number_of_contours: i16 = s.read()?;
s.advance(8);
let mut tuples = VariationTuples {
headers: [VariationTuple::default(); MAX_TUPLES_LEN as usize],
len: 0,
};
if number_of_contours > 0 {
let number_of_contours = NonZeroU16::new(number_of_contours as u16)?;
let mut glyph_points = glyf::parse_simple_outline(s.tail()?, number_of_contours)?;
let all_glyph_points = glyph_points.clone();
let points_len = glyph_points.points_left;
gvar_table.parse_variation_data(glyph_id, coordinates, points_len, &mut tuples)?;
while let Some(point) = glyph_points.next() {
let (x, y) = tuples.apply(all_glyph_points.clone(), glyph_points.clone(), point)?;
builder.push_point(x, y, point.on_curve_point, point.last_point);
}
Some(())
} else if number_of_contours < 0 {
let mut components = glyf::CompositeGlyphIter::new(s.tail()?);
let components_count = components.clone().count() as u16;
gvar_table.parse_variation_data(glyph_id, coordinates, components_count, &mut tuples)?;
while let Some(component) = components.next() {
let (tx, ty) = tuples.apply_null()?;
let mut transform = builder.transform;
if component.flags.args_are_xy_values() {
transform = Transform::combine(transform, Transform::new_translate(tx, ty));
}
transform = Transform::combine(transform, component.transform);
let mut b = glyf::Builder::new(transform, builder.bbox, builder.builder);
let range = loca_table.glyph_range(component.glyph_id)?;
let glyph_data = glyf_table.get(range)?;
outline_var_impl(
loca_table, glyf_table, gvar_table, component.glyph_id,
glyph_data, coordinates, depth + 1, &mut b,
)?;
builder.bbox = b.bbox;
}
Some(())
} else {
None
}
}
fn parse_variation_data<'a>(
coordinates: &[NormalizedCoord],
shared_tuple_records: &LazyArray16<F2DOT14>,
points_len: u16,
data: &'a [u8],
tuples: &mut VariationTuples<'a>,
) -> Option<()> {
const SHARED_POINT_NUMBERS_FLAG: u16 = 0x8000;
const COUNT_MASK: u16 = 0x0FFF;
let mut main_stream = Stream::new(data);
let tuple_variation_count: u16 = main_stream.read()?;
let data_offset: Offset16 = main_stream.read()?;
let has_shared_point_numbers = tuple_variation_count & SHARED_POINT_NUMBERS_FLAG != 0;
let tuple_variation_count = tuple_variation_count & COUNT_MASK;
if tuple_variation_count == 0 {
return None;
}
if tuple_variation_count >= MAX_TUPLES_LEN {
return None;
}
let mut serialized_stream = Stream::new_at(data, data_offset.to_usize())?;
let mut shared_point_numbers = None;
if has_shared_point_numbers {
shared_point_numbers = PackedPointsIter::new(&mut serialized_stream)?;
}
parse_variation_tuples(
tuple_variation_count,
coordinates,
shared_tuple_records,
shared_point_numbers,
points_len.checked_add(PHANTOM_POINTS_LEN as u16)?,
main_stream,
serialized_stream,
tuples,
)
}
#[derive(Clone, Copy, Default, Debug)]
struct PointAndDelta {
x: i16,
y: i16,
x_delta: f32,
y_delta: f32,
}
#[derive(Clone, Copy, Default)]
struct VariationTuple<'a> {
set_points: Option<SetPointsIter<'a>>,
deltas: PackedDeltasIter<'a>,
prev_point: Option<PointAndDelta>,
}
const MAX_TUPLES_LEN: u16 = 16;
struct VariationTuples<'a> {
headers: [VariationTuple<'a>; MAX_TUPLES_LEN as usize],
len: u16,
}
impl<'a> VariationTuples<'a> {
#[inline]
fn as_mut_slice(&mut self) -> &mut [VariationTuple<'a>] {
&mut self.headers[0..usize::from(self.len)]
}
fn apply(
&mut self,
all_points: glyf::GlyphPointsIter,
points: glyf::GlyphPointsIter,
point: glyf::GlyphPoint,
) -> Option<(f32, f32)> {
let mut x = f32::from(point.x);
let mut y = f32::from(point.y);
for tuple in self.as_mut_slice() {
if let Some(ref mut set_points) = tuple.set_points {
if set_points.next()? {
if let Some((x_delta, y_delta)) = tuple.deltas.next() {
tuple.prev_point = Some(PointAndDelta {
x: point.x, y: point.y, x_delta, y_delta
});
x += x_delta;
y += y_delta;
} else {
let set_points = set_points.clone();
let (x_delta, y_delta) = infer_deltas(
tuple, set_points, points.clone(), all_points.clone(), point
);
x += x_delta;
y += y_delta;
}
} else {
let set_points = set_points.clone();
let (x_delta, y_delta) = infer_deltas(
tuple, set_points, points.clone(), all_points.clone(), point
);
x += x_delta;
y += y_delta;
}
if point.last_point {
tuple.prev_point = None;
}
} else {
if let Some((x_delta, y_delta)) = tuple.deltas.next() {
x += x_delta;
y += y_delta;
}
}
}
Some((x, y))
}
fn apply_null(&mut self) -> Option<(f32, f32)> {
let mut x = 0.0;
let mut y = 0.0;
for tuple in self.as_mut_slice() {
if let Some(ref mut set_points) = tuple.set_points {
if set_points.next()? {
if let Some((x_delta, y_delta)) = tuple.deltas.next() {
x += x_delta;
y += y_delta;
}
}
} else {
if let Some((x_delta, y_delta)) = tuple.deltas.next() {
x += x_delta;
y += y_delta;
}
}
}
Some((x, y))
}
}
#[derive(Clone, Copy, Default, Debug)]
struct TupleVariationHeaderData {
scalar: f32,
has_private_point_numbers: bool,
serialized_data_len: u16,
}
fn parse_variation_tuples<'a>(
count: u16,
coordinates: &[NormalizedCoord],
shared_tuple_records: &LazyArray16<F2DOT14>,
shared_point_numbers: Option<PackedPointsIter<'a>>,
points_len: u16,
mut main_s: Stream<'a>,
mut serialized_s: Stream<'a>,
tuples: &mut VariationTuples<'a>,
) -> Option<()> {
debug_assert!(core::mem::size_of::<VariationTuple>() <= 80);
for _ in 0..count {
let header = parse_tuple_variation_header(coordinates, shared_tuple_records, &mut main_s)?;
if !(header.scalar > 0.0) {
serialized_s.advance(usize::from(header.serialized_data_len));
continue;
}
let serialized_data_start = serialized_s.offset();
let point_numbers = if header.has_private_point_numbers {
PackedPointsIter::new(&mut serialized_s)?
} else {
shared_point_numbers.clone()
};
let deltas_count = if let Some(point_numbers) = point_numbers.clone() {
u16::try_from(point_numbers.clone().count()).ok()?
} else {
points_len
};
let deltas = {
let left = usize::from(header.serialized_data_len)
.checked_sub(serialized_s.offset() - serialized_data_start)?;
let deltas_data = serialized_s.read_bytes(left)?;
PackedDeltasIter::new(header.scalar, deltas_count, deltas_data)
};
let tuple = VariationTuple {
set_points: point_numbers.map(SetPointsIter::new),
deltas,
prev_point: None,
};
tuples.headers[usize::from(tuples.len)] = tuple;
tuples.len += 1;
}
Some(())
}
fn parse_tuple_variation_header(
coordinates: &[NormalizedCoord],
shared_tuple_records: &LazyArray16<F2DOT14>,
s: &mut Stream,
) -> Option<TupleVariationHeaderData> {
const EMBEDDED_PEAK_TUPLE_FLAG: u16 = 0x8000;
const INTERMEDIATE_REGION_FLAG: u16 = 0x4000;
const PRIVATE_POINT_NUMBERS_FLAG: u16 = 0x2000;
const TUPLE_INDEX_MASK: u16 = 0x0FFF;
let serialized_data_size: u16 = s.read()?;
let tuple_index: u16 = s.read()?;
let has_embedded_peak_tuple = tuple_index & EMBEDDED_PEAK_TUPLE_FLAG != 0;
let has_intermediate_region = tuple_index & INTERMEDIATE_REGION_FLAG != 0;
let has_private_point_numbers = tuple_index & PRIVATE_POINT_NUMBERS_FLAG != 0;
let tuple_index = tuple_index & TUPLE_INDEX_MASK;
let axis_count = coordinates.len() as u16;
let peak_tuple = if has_embedded_peak_tuple {
s.read_array16(axis_count)?
} else {
let start = tuple_index.checked_mul(axis_count)?;
let end = start.checked_add(axis_count)?;
shared_tuple_records.slice(start..end)?
};
let (start_tuple, end_tuple) = if has_intermediate_region {
(s.read_array16(axis_count)?, s.read_array16(axis_count)?)
} else {
(LazyArray16::<F2DOT14>::default(), LazyArray16::<F2DOT14>::default())
};
let mut header = TupleVariationHeaderData {
scalar: 0.0,
has_private_point_numbers,
serialized_data_len: serialized_data_size,
};
let mut scalar = 1.0;
for i in 0..axis_count {
let v = coordinates[usize::from(i)].get();
let peak = peak_tuple.get(i)?.0;
if peak == 0 || v == peak {
continue;
}
if has_intermediate_region {
let start = start_tuple.get(i)?.0;
let end = end_tuple.get(i)?.0;
if start > peak || peak > end || (start < 0 && end > 0 && peak != 0) {
continue;
}
if v < start || v > end {
return Some(header);
}
if v < peak {
if peak != start {
scalar *= f32::from(v - start) / f32::from(peak - start);
}
} else {
if peak != end {
scalar *= f32::from(end - v) / f32::from(end - peak);
}
}
} else if v == 0 || v < cmp::min(0, peak) || v > cmp::max(0, peak) {
return Some(header);
} else {
scalar *= f32::from(v) / f32::from(peak);
}
}
header.scalar = scalar;
Some(header)
}
mod packed_points {
use crate::parser::{Stream, FromData};
struct Control(u8);
impl Control {
const POINTS_ARE_WORDS_FLAG: u8 = 0x80;
const POINT_RUN_COUNT_MASK: u8 = 0x7F;
#[inline]
fn is_points_are_words(&self) -> bool { self.0 & Self::POINTS_ARE_WORDS_FLAG != 0 }
#[inline]
fn run_count(&self) -> u8 { (self.0 & Self::POINT_RUN_COUNT_MASK) + 1 }
}
impl FromData for Control {
const SIZE: usize = 1;
#[inline]
fn parse(data: &[u8]) -> Option<Self> { data.get(0).copied().map(Control) }
}
#[derive(Clone, Copy, PartialEq)]
enum State {
Control,
ShortPoint,
LongPoint,
}
#[derive(Clone, Copy)]
pub struct PackedPointsIter<'a> {
data: &'a [u8],
offset: u16,
state: State,
points_left: u8,
}
impl<'a> PackedPointsIter<'a> {
pub fn new<'b>(s: &'b mut Stream<'a>) -> Option<Option<Self>> {
let b1: u8 = s.read()?;
let mut count = u16::from(b1);
if b1 & Control::POINTS_ARE_WORDS_FLAG != 0 {
let b2: u8 = s.read()?;
count = (u16::from(b1 & Control::POINT_RUN_COUNT_MASK) << 8) | u16::from(b2);
}
if count == 0 {
return Some(None);
}
let start = s.offset();
let tail = s.tail()?;
let mut i = 0;
while i < count {
let control: Control = s.read()?;
let run_count = u16::from(control.run_count());
let is_points_are_words = control.is_points_are_words();
s.advance_checked(if is_points_are_words { 2 } else { 1 } * usize::from(run_count))?;
i += run_count;
}
if i == 0 {
return Some(None);
}
if i > count {
return None;
}
let data_len = s.offset() - start;
if data_len > usize::from(core::u16::MAX) {
return None;
}
Some(Some(PackedPointsIter {
data: &tail[0..data_len],
offset: 0,
state: State::Control,
points_left: 0,
}))
}
}
impl<'a> Iterator for PackedPointsIter<'a> {
type Item = u16;
fn next(&mut self) -> Option<Self::Item> {
if usize::from(self.offset) >= self.data.len() {
return None;
}
if self.state == State::Control {
let control = Control(self.data[usize::from(self.offset)]);
self.offset += 1;
self.points_left = control.run_count();
self.state = if control.is_points_are_words() {
State::LongPoint
} else {
State::ShortPoint
};
self.next()
} else {
let mut s = Stream::new_at(self.data, usize::from(self.offset))?;
let point = if self.state == State::LongPoint {
self.offset += 2;
s.read::<u16>()?
} else {
self.offset += 1;
u16::from(s.read::<u8>()?)
};
self.points_left -= 1;
if self.points_left == 0 {
self.state = State::Control;
}
Some(point)
}
}
}
#[derive(Clone, Copy)]
pub struct SetPointsIter<'a> {
iter: PackedPointsIter<'a>,
unref_count: u16,
}
impl<'a> SetPointsIter<'a> {
#[inline]
pub fn new(mut iter: PackedPointsIter<'a>) -> Self {
let unref_count = iter.next().unwrap_or(0);
SetPointsIter { iter, unref_count }
}
#[inline]
pub fn restart(self) -> Self {
let mut iter = self.iter.clone();
iter.offset = 0;
iter.state = State::Control;
iter.points_left = 0;
let unref_count = iter.next().unwrap_or(0);
SetPointsIter { iter, unref_count }
}
}
impl<'a> Iterator for SetPointsIter<'a> {
type Item = bool;
#[inline]
fn next(&mut self) -> Option<Self::Item> {
if self.unref_count != 0 {
self.unref_count -= 1;
return Some(false);
}
if let Some(unref_count) = self.iter.next() {
self.unref_count = unref_count;
if self.unref_count != 0 {
self.unref_count -= 1;
}
}
Some(true)
}
}
#[cfg(test)]
mod tests {
use super::*;
struct NewControl {
deltas_are_words: bool,
run_count: u8,
}
fn gen_control(control: NewControl) -> u8 {
assert!(control.run_count > 0, "run count cannot be zero");
let mut n = 0;
if control.deltas_are_words { n |= 0x80; }
n |= (control.run_count - 1) & 0x7F;
n
}
#[test]
fn empty() {
let mut s = Stream::new(&[]);
assert!(PackedPointsIter::new(&mut s).is_none());
}
#[test]
fn single_zero_control() {
let mut s = Stream::new(&[0]);
assert!(PackedPointsIter::new(&mut s).unwrap().is_none());
}
#[test]
fn single_point() {
let data = vec![
1,
gen_control(NewControl { deltas_are_words: false, run_count: 1 }),
1
];
let points_iter = PackedPointsIter::new(&mut Stream::new(&data)).unwrap().unwrap();
let mut iter = SetPointsIter::new(points_iter);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), true);
}
#[test]
fn set_0_and_2() {
let data = vec![
2,
gen_control(NewControl { deltas_are_words: false, run_count: 2 }),
0, 2
];
let points_iter = PackedPointsIter::new(&mut Stream::new(&data)).unwrap().unwrap();
let mut iter = SetPointsIter::new(points_iter);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), true);
}
#[test]
fn set_1_and_2() {
let data = vec![
2,
gen_control(NewControl { deltas_are_words: false, run_count: 2 }),
1, 1
];
let points_iter = PackedPointsIter::new(&mut Stream::new(&data)).unwrap().unwrap();
let mut iter = SetPointsIter::new(points_iter);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), true);
}
#[test]
fn set_1_and_3() {
let data = vec![
2,
gen_control(NewControl { deltas_are_words: false, run_count: 2 }),
1, 2
];
let points_iter = PackedPointsIter::new(&mut Stream::new(&data)).unwrap().unwrap();
let mut iter = SetPointsIter::new(points_iter);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), true);
}
#[test]
fn set_2_5_7() {
let data = vec![
3,
gen_control(NewControl { deltas_are_words: false, run_count: 3 }),
2, 3, 2
];
let points_iter = PackedPointsIter::new(&mut Stream::new(&data)).unwrap().unwrap();
let mut iter = SetPointsIter::new(points_iter);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), true);
}
#[test]
fn more_than_127_points() {
let mut data = vec![];
data.push(Control::POINTS_ARE_WORDS_FLAG);
data.push(150);
data.push(gen_control(NewControl { deltas_are_words: false, run_count: 100 }));
for _ in 0..100 {
data.push(2);
}
data.push(gen_control(NewControl { deltas_are_words: false, run_count: 50 }));
for _ in 0..50 {
data.push(2);
}
let points_iter = PackedPointsIter::new(&mut Stream::new(&data)).unwrap().unwrap();
let mut iter = SetPointsIter::new(points_iter);
assert_eq!(iter.next().unwrap(), false);
for _ in 0..150 {
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), true);
}
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), true);
}
#[test]
fn long_points() {
let data = vec![
2,
gen_control(NewControl { deltas_are_words: true, run_count: 2 }),
0, 2, 0, 3
];
let points_iter = PackedPointsIter::new(&mut Stream::new(&data)).unwrap().unwrap();
let mut iter = SetPointsIter::new(points_iter);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), true);
}
#[test]
fn multiple_runs() {
let data = vec![
5,
gen_control(NewControl { deltas_are_words: true, run_count: 2 }),
0, 2, 0, 3,
gen_control(NewControl { deltas_are_words: false, run_count: 3 }),
2, 3, 2
];
let points_iter = PackedPointsIter::new(&mut Stream::new(&data)).unwrap().unwrap();
let mut iter = SetPointsIter::new(points_iter);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), false);
assert_eq!(iter.next().unwrap(), true);
assert_eq!(iter.next().unwrap(), true);
}
#[test]
fn runs_overflow() {
let data = vec![0xFF; 0xFFFF * 2];
assert!(PackedPointsIter::new(&mut Stream::new(&data)).is_none());
}
}
}
use packed_points::*;
mod packed_deltas {
use crate::parser::Stream;
struct Control(u8);
impl Control {
const DELTAS_ARE_ZERO_FLAG: u8 = 0x80;
const DELTAS_ARE_WORDS_FLAG: u8 = 0x40;
const DELTA_RUN_COUNT_MASK: u8 = 0x3F;
#[inline]
fn is_deltas_are_zero(&self) -> bool { self.0 & Self::DELTAS_ARE_ZERO_FLAG != 0 }
#[inline]
fn is_deltas_are_words(&self) -> bool { self.0 & Self::DELTAS_ARE_WORDS_FLAG != 0 }
#[inline]
fn run_count(&self) -> u8 { (self.0 & Self::DELTA_RUN_COUNT_MASK) + 1 }
}
#[derive(Clone, Copy, PartialEq, Debug)]
enum State {
Control,
ZeroDelta,
ShortDelta,
LongDelta,
}
impl Default for State {
#[inline]
fn default() -> Self {
State::Control
}
}
#[derive(Clone, Copy, Default)]
struct RunState {
data_offset: u16,
state: State,
run_deltas_left: u8,
}
impl RunState {
fn next(&mut self, data: &[u8], scalar: f32) -> Option<f32> {
if self.state == State::Control {
if usize::from(self.data_offset) == data.len() {
return None;
}
let control = Control(Stream::read_at(data, usize::from(self.data_offset))?);
self.data_offset += 1;
self.run_deltas_left = control.run_count();
self.state = if control.is_deltas_are_zero() {
State::ZeroDelta
} else if control.is_deltas_are_words() {
State::LongDelta
} else {
State::ShortDelta
};
self.next(data, scalar)
} else {
let mut s = Stream::new_at(data, usize::from(self.data_offset))?;
let delta = if self.state == State::LongDelta {
self.data_offset += 2;
f32::from(s.read::<i16>()?) * scalar
} else if self.state == State::ZeroDelta {
0.0
} else {
self.data_offset += 1;
f32::from(s.read::<i8>()?) * scalar
};
self.run_deltas_left -= 1;
if self.run_deltas_left == 0 {
self.state = State::Control;
}
Some(delta)
}
}
}
#[derive(Clone, Copy, Default)]
pub struct PackedDeltasIter<'a> {
data: &'a [u8],
x_run: RunState,
y_run: RunState,
total_count: u16,
scalar: f32,
}
impl<'a> PackedDeltasIter<'a> {
pub fn new(scalar: f32, count: u16, data: &'a [u8]) -> Self {
debug_assert!(core::mem::size_of::<PackedDeltasIter>() <= 32);
let mut iter = PackedDeltasIter {
data,
total_count: count,
scalar,
..PackedDeltasIter::default()
};
for _ in 0..count {
iter.y_run.next(data, scalar);
}
iter
}
#[inline]
pub fn restart(self) -> Self {
PackedDeltasIter::new(self.scalar, self.total_count, self.data)
}
#[inline]
pub fn next(&mut self) -> Option<(f32, f32)> {
let x = self.x_run.next(self.data, self.scalar)?;
let y = self.y_run.next(self.data, self.scalar)?;
Some((x, y))
}
}
#[cfg(test)]
mod tests {
use super::*;
struct NewControl {
deltas_are_zero: bool,
deltas_are_words: bool,
run_count: u8,
}
fn gen_control(control: NewControl) -> u8 {
assert!(control.run_count > 0, "run count cannot be zero");
let mut n = 0;
if control.deltas_are_zero { n |= 0x80; }
if control.deltas_are_words { n |= 0x40; }
n |= (control.run_count - 1) & 0x3F;
n
}
#[test]
fn empty() {
let mut iter = PackedDeltasIter::new(1.0, 1, &[]);
assert!(iter.next().is_none());
}
#[test]
fn single_delta() {
let data = vec![
gen_control(NewControl { deltas_are_zero: false, deltas_are_words: false, run_count: 2 }),
2, 3
];
let mut iter = PackedDeltasIter::new(1.0, 1, &data);
assert_eq!(iter.next().unwrap(), (2.0, 3.0));
assert!(iter.next().is_none());
}
#[test]
fn two_deltas() {
let data = vec![
gen_control(NewControl { deltas_are_zero: false, deltas_are_words: false, run_count: 4 }),
2, 3, 4, 5,
];
let mut iter = PackedDeltasIter::new(1.0, 2, &data);
assert_eq!(iter.next().unwrap(), (2.0, 4.0));
assert_eq!(iter.next().unwrap(), (3.0, 5.0));
assert!(iter.next().is_none());
}
#[test]
fn single_long_delta() {
let data = vec![
gen_control(NewControl { deltas_are_zero: false, deltas_are_words: true, run_count: 2 }),
0, 2, 0, 3
];
let mut iter = PackedDeltasIter::new(1.0, 1, &data);
assert_eq!(iter.next().unwrap(), (2.0, 3.0));
assert!(iter.next().is_none());
}
#[test]
fn zeros() {
let data = vec![
gen_control(NewControl { deltas_are_zero: true, deltas_are_words: false, run_count: 4 }),
];
let mut iter = PackedDeltasIter::new(1.0, 2, &data);
assert_eq!(iter.next().unwrap(), (0.0, 0.0));
assert_eq!(iter.next().unwrap(), (0.0, 0.0));
assert!(iter.next().is_none());
}
#[test]
fn zero_words() {
let data = vec![
gen_control(NewControl { deltas_are_zero: true, deltas_are_words: true, run_count: 4 }),
];
let mut iter = PackedDeltasIter::new(1.0, 2, &data);
assert_eq!(iter.next().unwrap(), (0.0, 0.0));
assert_eq!(iter.next().unwrap(), (0.0, 0.0));
assert!(iter.next().is_none());
}
#[test]
fn zero_runs() {
let data = vec![
gen_control(NewControl { deltas_are_zero: true, deltas_are_words: false, run_count: 2 }),
gen_control(NewControl { deltas_are_zero: true, deltas_are_words: false, run_count: 4 }),
gen_control(NewControl { deltas_are_zero: true, deltas_are_words: false, run_count: 6 }),
];
let mut iter = PackedDeltasIter::new(1.0, 6, &data);
assert_eq!(iter.next().unwrap(), (0.0, 0.0));
assert_eq!(iter.next().unwrap(), (0.0, 0.0));
assert_eq!(iter.next().unwrap(), (0.0, 0.0));
assert_eq!(iter.next().unwrap(), (0.0, 0.0));
assert_eq!(iter.next().unwrap(), (0.0, 0.0));
assert_eq!(iter.next().unwrap(), (0.0, 0.0));
assert!(iter.next().is_none());
}
#[test]
fn delta_after_zeros() {
let data = vec![
gen_control(NewControl { deltas_are_zero: true, deltas_are_words: false, run_count: 2 }),
gen_control(NewControl { deltas_are_zero: false, deltas_are_words: false, run_count: 2 }),
2, 3
];
let mut iter = PackedDeltasIter::new(1.0, 2, &data);
assert_eq!(iter.next().unwrap(), (0.0, 2.0));
assert_eq!(iter.next().unwrap(), (0.0, 3.0));
assert!(iter.next().is_none());
}
#[test]
fn unexpected_end_of_data_1() {
let data = vec![
gen_control(NewControl { deltas_are_zero: false, deltas_are_words: false, run_count: 2 }),
];
let mut iter = PackedDeltasIter::new(1.0, 1, &data);
assert!(iter.next().is_none());
}
#[test]
fn unexpected_end_of_data_2() {
let data = vec![
gen_control(NewControl { deltas_are_zero: false, deltas_are_words: false, run_count: 2 }),
1
];
let mut iter = PackedDeltasIter::new(1.0, 1, &data);
assert!(iter.next().is_none());
}
#[test]
fn unexpected_end_of_data_3() {
let data = vec![
gen_control(NewControl { deltas_are_zero: false, deltas_are_words: true, run_count: 2 }),
];
let mut iter = PackedDeltasIter::new(1.0, 1, &data);
assert!(iter.next().is_none());
}
#[test]
fn unexpected_end_of_data_4() {
let data = vec![
gen_control(NewControl { deltas_are_zero: false, deltas_are_words: true, run_count: 2 }),
1
];
let mut iter = PackedDeltasIter::new(1.0, 1, &data);
assert!(iter.next().is_none());
}
#[test]
fn unexpected_end_of_data_6() {
let data = vec![
gen_control(NewControl { deltas_are_zero: false, deltas_are_words: true, run_count: 2 }),
0, 1
];
let mut iter = PackedDeltasIter::new(1.0, 1, &data);
assert!(iter.next().is_none());
}
#[test]
fn unexpected_end_of_data_7() {
let data = vec![
gen_control(NewControl { deltas_are_zero: false, deltas_are_words: true, run_count: 2 }),
0, 1, 0
];
let mut iter = PackedDeltasIter::new(1.0, 1, &data);
assert!(iter.next().is_none());
}
#[test]
fn single_run() {
let data = vec![
gen_control(NewControl { deltas_are_zero: false, deltas_are_words: false, run_count: 1 }),
2, 3
];
let mut iter = PackedDeltasIter::new(1.0, 1, &data);
assert!(iter.next().is_none());
}
#[test]
fn too_many_pairs() {
let data = vec![
gen_control(NewControl { deltas_are_zero: false, deltas_are_words: false, run_count: 2 }),
2, 3
];
let mut iter = PackedDeltasIter::new(1.0, 10, &data);
assert!(iter.next().is_none());
}
#[test]
fn invalid_number_of_pairs() {
let data = vec![
gen_control(NewControl { deltas_are_zero: false, deltas_are_words: false, run_count: 2 }),
2, 3, 4, 5, 6, 7,
];
let mut iter = PackedDeltasIter::new(1.0, 4, &data);
assert_eq!(iter.next().unwrap(), (2.0, 7.0));
assert!(iter.next().is_none());
}
#[test]
fn mixed_runs() {
let data = vec![
gen_control(NewControl { deltas_are_zero: false, deltas_are_words: false, run_count: 3 }),
2, 3, 4,
gen_control(NewControl { deltas_are_zero: false, deltas_are_words: true, run_count: 2 }),
0, 5, 0, 6,
gen_control(NewControl { deltas_are_zero: true, deltas_are_words: false, run_count: 1 }),
];
let mut iter = PackedDeltasIter::new(1.0, 3, &data);
assert_eq!(iter.next().unwrap(), (2.0, 5.0));
assert_eq!(iter.next().unwrap(), (3.0, 6.0));
assert_eq!(iter.next().unwrap(), (4.0, 0.0));
assert!(iter.next().is_none());
}
#[test]
fn non_default_scalar() {
let data = vec![
gen_control(NewControl { deltas_are_zero: false, deltas_are_words: false, run_count: 2 }),
2, 3
];
let mut iter = PackedDeltasIter::new(0.5, 1, &data);
assert_eq!(iter.next().unwrap(), (1.0, 1.5));
assert!(iter.next().is_none());
}
#[test]
fn runs_overflow() {
let data = vec![0xFF; 0xFFFF];
let mut iter = PackedDeltasIter::new(1.0, 0xFFFF, &data);
assert_eq!(iter.next().unwrap(), (0.0, 0.0));
}
}
}
use packed_deltas::PackedDeltasIter;
fn infer_deltas(
tuple: &VariationTuple,
points_set: SetPointsIter,
points: glyf::GlyphPointsIter,
all_points: glyf::GlyphPointsIter,
curr_point: glyf::GlyphPoint,
) -> (f32, f32) {
let mut current_contour = points.current_contour();
if curr_point.last_point && current_contour != 0 {
current_contour -= 1;
}
let prev_point = if let Some(prev_point) = tuple.prev_point {
prev_point
} else {
let mut last_point = None;
let mut deltas = tuple.deltas.clone();
for (point, is_set) in points.clone().zip(points_set.clone()) {
if is_set {
if let Some((x_delta, y_delta)) = deltas.next() {
last_point = Some(PointAndDelta {
x: point.x,
y: point.y,
x_delta,
y_delta,
});
}
}
if point.last_point {
break;
}
}
match last_point {
Some(p) => p,
None => return (0.0, 0.0),
}
};
let mut next_point = None;
if !curr_point.last_point {
let mut deltas = tuple.deltas.clone();
for (point, is_set) in points.clone().zip(points_set.clone()) {
if is_set {
if let Some((x_delta, y_delta)) = deltas.next() {
next_point = Some(PointAndDelta {
x: point.x,
y: point.y,
x_delta,
y_delta,
});
}
break;
}
if point.last_point {
break;
}
}
}
if next_point.is_none() {
let mut all_points = all_points.clone();
let mut deltas = tuple.deltas.clone().restart();
let mut points_set = points_set.clone().restart();
let mut contour = 0;
while let (Some(point), Some(is_set)) = (all_points.next(), points_set.next()) {
if contour != current_contour {
if is_set {
let _ = deltas.next();
}
contour = all_points.current_contour();
continue;
}
if is_set {
let (x_delta, y_delta) = deltas.next().unwrap_or((0.0, 0.0));
next_point = Some(PointAndDelta {
x: point.x,
y: point.y,
x_delta,
y_delta,
});
break;
}
if point.last_point {
break;
}
}
}
let next_point = match next_point {
Some(p) => p,
None => return (0.0, 0.0),
};
let dx = infer_delta(prev_point.x, curr_point.x, next_point.x,
prev_point.x_delta, next_point.x_delta);
let dy = infer_delta(prev_point.y, curr_point.y, next_point.y,
prev_point.y_delta, next_point.y_delta);
(dx, dy)
}
fn infer_delta(
prev_point: i16,
target_point: i16,
next_point: i16,
prev_delta: f32,
next_delta: f32,
) -> f32 {
if prev_point == next_point {
if prev_delta == next_delta { prev_delta } else { 0.0 }
} else if target_point <= prev_point.min(next_point) {
if prev_point < next_point { prev_delta } else { next_delta }
} else if target_point >= prev_point.max(next_point) {
if prev_point > next_point { prev_delta } else { next_delta }
} else {
let d = f32::from(try_opt_or!(target_point.checked_sub(prev_point), 0.0))
/ f32::from(try_opt_or!(next_point.checked_sub(prev_point), 0.0));
(1.0 - d) * prev_delta + d * next_delta
}
}