#![deny(clippy::arithmetic_side_effects)]
use types::{FixedSize, Scalar, Tag};
use crate::font_data::FontData;
pub trait FontRead<'a>: Sized {
fn read(data: FontData<'a>) -> Result<Self, ReadError>;
}
pub trait ReadArgs {
type Args: Copy;
}
pub trait FontReadWithArgs<'a>: Sized + ReadArgs {
fn read_with_args(data: FontData<'a>, args: &Self::Args) -> Result<Self, ReadError>;
}
impl<'a, T: FontRead<'a>> ReadArgs for T {
type Args = ();
}
impl<'a, T: FontRead<'a>> FontReadWithArgs<'a> for T {
fn read_with_args(data: FontData<'a>, _: &Self::Args) -> Result<Self, ReadError> {
Self::read(data)
}
}
pub trait Format<T> {
const FORMAT: T;
}
pub trait ComputeSize: ReadArgs {
fn compute_size(args: &Self::Args) -> Result<usize, ReadError>;
}
pub trait VarSize {
type Size: Scalar + Into<u32>;
#[doc(hidden)]
fn read_len_at(data: FontData, pos: usize) -> Option<usize> {
let asu32 = data.read_at::<Self::Size>(pos).ok()?.into();
(asu32 as usize).checked_add(Self::Size::RAW_BYTE_LEN)
}
#[doc(hidden)]
fn total_len_for_count(data: FontData, count: usize) -> Result<usize, ReadError> {
let mut current_pos = 0;
for _ in 0..count {
let len = Self::read_len_at(data, current_pos).ok_or(ReadError::OutOfBounds)?;
if len == 0 {
return Ok(current_pos);
}
current_pos = current_pos.checked_add(len).ok_or(ReadError::OutOfBounds)?;
}
Ok(current_pos)
}
}
#[derive(Debug, Clone, PartialEq)]
pub enum ReadError {
OutOfBounds,
InvalidFormat(i64),
InvalidSfnt(u32),
InvalidTtc(Tag),
InvalidCollectionIndex(u32),
InvalidArrayLen,
ValidationError,
NullOffset,
TableIsMissing(Tag),
MetricIsMissing(Tag),
MalformedData(&'static str),
}
impl std::fmt::Display for ReadError {
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
match self {
ReadError::OutOfBounds => write!(f, "An offset was out of bounds"),
ReadError::InvalidFormat(x) => write!(f, "Invalid format '{x}'"),
ReadError::InvalidSfnt(ver) => write!(f, "Invalid sfnt version 0x{ver:08X}"),
ReadError::InvalidTtc(tag) => write!(f, "Invalid ttc tag {tag}"),
ReadError::InvalidCollectionIndex(ix) => {
write!(f, "Invalid index {ix} for font collection")
}
ReadError::InvalidArrayLen => {
write!(f, "Specified array length not a multiple of item size")
}
ReadError::ValidationError => write!(f, "A validation error occurred"),
ReadError::NullOffset => write!(f, "An offset was unexpectedly null"),
ReadError::TableIsMissing(tag) => write!(f, "the {tag} table is missing"),
ReadError::MetricIsMissing(tag) => write!(f, "the {tag} metric is missing"),
ReadError::MalformedData(msg) => write!(f, "Malformed data: '{msg}'"),
}
}
}
impl core::error::Error for ReadError {}
#[cfg(test)]
mod tests {
use font_test_data::bebuffer::BeBuffer;
use super::*;
struct DummyVarSize {}
impl VarSize for DummyVarSize {
type Size = u16;
fn read_len_at(data: FontData, pos: usize) -> Option<usize> {
data.read_at::<u16>(pos).map(|v| v as usize).ok()
}
}
#[test]
fn total_var_size_with_zero_length_element() {
const PAYLOAD_NOT_SIZE: u16 = 1;
let buf = BeBuffer::new().extend([2u16, 4u16, PAYLOAD_NOT_SIZE, 0u16, 20u16]);
let total_len =
DummyVarSize::total_len_for_count(FontData::new(buf.data()), usize::MAX).unwrap();
assert_eq!(total_len, 6);
}
}