chromium/third_party/rust/chromium_crates_io/vendor/skrifa-0.20.0/src/outline/glyf/hint/math.rs

//! Fixed point math helpers that are specific to TrueType hinting.
//!
//! These are implemented in terms of font-types types when possible. It
//! likely makes sense to use more strongly typed fixed point values
//! in the future.

use read_fonts::types::{Fixed, Point};

pub fn floor(x: i32) -> i32 {
    x & !63
}

pub fn round(x: i32) -> i32 {
    floor(x + 32)
}

pub fn ceil(x: i32) -> i32 {
    floor(x + 63)
}

fn floor_pad(x: i32, n: i32) -> i32 {
    x & !(n - 1)
}

pub fn round_pad(x: i32, n: i32) -> i32 {
    floor_pad(x + n / 2, n)
}

#[inline(always)]
pub fn mul(a: i32, b: i32) -> i32 {
    (Fixed::from_bits(a) * Fixed::from_bits(b)).to_bits()
}

pub fn div(a: i32, b: i32) -> i32 {
    (Fixed::from_bits(a) / Fixed::from_bits(b)).to_bits()
}

/// Fixed point multiply and divide: a * b / c
pub fn mul_div(a: i32, b: i32, c: i32) -> i32 {
    Fixed::from_bits(a)
        .mul_div(Fixed::from_bits(b), Fixed::from_bits(c))
        .to_bits()
}

/// Fixed point multiply and divide without rounding: a * b / c
///
/// Based on <https://gitlab.freedesktop.org/freetype/freetype/-/blob/57617782464411201ce7bbc93b086c1b4d7d84a5/src/base/ftcalc.c#L200>
pub fn mul_div_no_round(mut a: i32, mut b: i32, mut c: i32) -> i32 {
    let mut s = 1;
    if a < 0 {
        a = -a;
        s = -1;
    }
    if b < 0 {
        b = -b;
        s = -s;
    }
    if c < 0 {
        c = -c;
        s = -s;
    }
    let d = if c > 0 {
        ((a as i64) * (b as i64)) / c as i64
    } else {
        0x7FFFFFFF
    };
    if s < 0 {
        -(d as i32)
    } else {
        d as i32
    }
}

/// Multiplication for 2.14 fixed point.
///
/// Based on <https://gitlab.freedesktop.org/freetype/freetype/-/blob/57617782464411201ce7bbc93b086c1b4d7d84a5/src/truetype/ttinterp.c#L1234>
pub fn mul14(a: i32, b: i32) -> i32 {
    let mut v = a as i64 * b as i64;
    v += 0x2000 + (v >> 63);
    (v >> 14) as i32
}

/// Normalize a vector in 2.14 fixed point.
///
/// Direct port of <https://gitlab.freedesktop.org/freetype/freetype/-/blob/57617782464411201ce7bbc93b086c1b4d7d84a5/src/base/ftcalc.c#L800>
pub fn normalize14(x: i32, y: i32) -> Point<i32> {
    use core::num::Wrapping;
    let (mut sx, mut sy) = (Wrapping(1i32), Wrapping(1i32));
    let mut ux = Wrapping(x as u32);
    let mut uy = Wrapping(y as u32);
    const ZERO: Wrapping<u32> = Wrapping(0);
    let mut result = Point::default();
    if x < 0 {
        ux = ZERO - ux;
        sx = -sx;
    }
    if y < 0 {
        uy = ZERO - uy;
        sy = -sy;
    }
    if ux == ZERO {
        result.x = x / 4;
        if uy.0 > 0 {
            result.y = (sy * Wrapping(0x10000) / Wrapping(4)).0;
        }
        return result;
    }
    if uy == ZERO {
        result.y = y / 4;
        if ux.0 > 0 {
            result.x = (sx * Wrapping(0x10000) / Wrapping(4)).0;
        }
        return result;
    }
    let mut len = if ux > uy {
        ux + (uy >> 1)
    } else {
        uy + (ux >> 1)
    };
    let mut shift = Wrapping(len.0.leading_zeros() as i32);
    shift -= Wrapping(15)
        + if len >= (Wrapping(0xAAAAAAAAu32) >> shift.0 as usize) {
            Wrapping(1)
        } else {
            Wrapping(0)
        };
    if shift.0 > 0 {
        let s = shift.0 as usize;
        ux <<= s;
        uy <<= s;
        len = if ux > uy {
            ux + (uy >> 1)
        } else {
            uy + (ux >> 1)
        };
    } else {
        let s = -shift.0 as usize;
        ux >>= s;
        uy >>= s;
        len >>= s;
    }
    let mut b = Wrapping(0x10000) - Wrapping(len.0 as i32);
    let x = Wrapping(ux.0 as i32);
    let y = Wrapping(uy.0 as i32);
    let mut z;
    let mut u;
    let mut v;
    loop {
        u = Wrapping((x + ((x * b) >> 16)).0 as u32);
        v = Wrapping((y + ((y * b) >> 16)).0 as u32);
        z = Wrapping(-((u * u + v * v).0 as i32)) / Wrapping(0x200);
        z = z * ((Wrapping(0x10000) + b) >> 8) / Wrapping(0x10000);
        b += z;
        if z <= Wrapping(0) {
            break;
        }
    }
    Point::new(
        (Wrapping(u.0 as i32) * sx / Wrapping(4)).0,
        (Wrapping(v.0 as i32) * sy / Wrapping(4)).0,
    )
}

#[cfg(test)]
mod tests {
    use raw::types::{F2Dot14, Fixed};

    /// Tolerance value for floating point sanity checks.
    /// Tests with large sets of values show this is the best we can
    /// expect from the fixed point implementations.
    const FLOAT_TOLERANCE: f32 = 1e-4;

    #[test]
    fn mul_div_no_round() {
        let cases = [
            // computed with FT_MulDiv_NoRound():
            // ((a, b, c), result) where result = a * b / c
            ((-326, -11474, 9942), 376),
            ((-6781, 13948, 11973), -7899),
            ((3517, 15622, 8075), 6804),
            ((-6127, 15026, 2276), -40450),
            ((11257, 14828, 2542), 65664),
            ((-12797, -16280, -9086), -22929),
            ((-7994, -3340, 9583), 2786),
            ((-16101, -13780, -1427), -155481),
            ((10304, -16331, 15480), -10870),
            ((-15879, 11912, -4650), 40677),
            ((-5015, 6382, -15977), 2003),
            ((2080, -11930, -15457), 1605),
            ((-11071, 13350, 16138), -9158),
            ((16084, -13564, -770), 283329),
            ((14304, -10377, -21), 7068219),
            ((-14056, -8853, -5488), -22674),
            ((-10319, 14797, 8554), -17850),
            ((-7820, 6826, 10555), -5057),
            ((7257, 15928, 8159), 14167),
            ((14929, 11579, -13204), -13091),
            ((2808, 12070, -14697), -2306),
            ((-13818, 8544, -1649), 71595),
            ((3265, 7325, -1373), -17418),
            ((14832, 10586, -6440), -24380),
            ((4123, 8274, -2022), -16871),
            ((4645, -4149, -7242), 2661),
            ((-3891, 8366, 5771), -5640),
            ((-15447, -3428, -9335), -5672),
            ((13670, -14311, -11122), 17589),
            ((12590, -6592, 13159), -6306),
            ((-8369, -10193, 5051), 16888),
            ((-9539, 5167, 2595), -18993),
        ];
        for ((a, b, c), expected_result) in cases {
            let result = super::mul_div_no_round(a, b, c);
            assert_eq!(result, expected_result);
            let fa = Fixed::from_bits(a as _).to_f32();
            let fb = Fixed::from_bits(b as _).to_f32();
            let fc = Fixed::from_bits(c as _).to_f32();
            let fresult = fa * fb / fc;
            let fexpected_result = Fixed::from_bits(expected_result as _).to_f32();
            assert!((fresult - fexpected_result).abs() < FLOAT_TOLERANCE);
        }
    }

    #[test]
    fn mul14() {
        let cases = [
            // computed with TT_MulFix14():
            // ((a, b), result) where result = a * b
            ((6236, -10078), -3836),
            ((-6803, -5405), 2244),
            ((-10006, -12852), 7849),
            ((-15434, -4102), 3864),
            ((-8681, 9269), -4911),
            ((9449, -9130), -5265),
            ((12643, 2161), 1668),
            ((-6115, 9284), -3465),
            ((316, 3390), 65),
            ((15077, -12901), -11872),
            ((-12182, 11613), -8635),
            ((-7213, 8246), -3630),
            ((13482, 8096), 6662),
            ((5690, 15016), 5215),
            ((-5991, 12613), -4612),
            ((13112, -8404), -6726),
            ((13524, 6786), 5601),
            ((7156, 3291), 1437),
            ((-2978, 353), -64),
            ((-1755, 14626), -1567),
            ((14402, 7886), 6932),
            ((7124, 15730), 6840),
            ((-12679, 14830), -11476),
            ((-9374, -12999), 7437),
            ((12301, -4685), -3517),
            ((5324, 2066), 671),
            ((6783, -4946), -2048),
            ((12078, -968), -714),
            ((-10137, 14116), -8734),
            ((-13946, 11585), -9861),
            ((-678, -2205), 91),
            ((-2629, -3319), 533),
        ];
        for ((a, b), expected_result) in cases {
            let result = super::mul14(a, b);
            assert_eq!(result, expected_result);
            let fa = F2Dot14::from_bits(a as _).to_f32();
            let fb = F2Dot14::from_bits(b as _).to_f32();
            let fresult = fa * fb;
            let fexpected_result = F2Dot14::from_bits(expected_result as _).to_f32();
            assert!((fresult - fexpected_result).abs() < FLOAT_TOLERANCE);
        }
    }

    #[test]
    fn normalize14() {
        let cases = [
            // computed with FT_Vector_NormLen():
            // (input vector, expected normalized vector)
            ((-13660, 11807), (-12395, 10713)),
            ((-10763, 9293), (-12401, 10707)),
            ((-3673, 673), (-16115, 2952)),
            ((15886, -2964), (16106, -3005)),
            ((15442, -2871), (16108, -2994)),
            ((-6308, 5744), (-12114, 11031)),
            ((9410, -10415), (10983, -12156)),
            ((-10620, -14856), (-9528, -13328)),
            ((-9372, 12029), (-10069, 12924)),
            ((-1272, -1261), (-11635, -11534)),
            ((-7076, -5517), (-12920, -10074)),
            ((-10297, 179), (-16381, 284)),
            ((9256, -13235), (9389, -13426)),
            ((5315, -12449), (6433, -15068)),
            ((8064, 15213), (7673, 14476)),
            ((-8665, 41), (-16383, 77)),
            ((-3455, -4720), (-9677, -13220)),
            ((13449, -5152), (15299, -5861)),
            ((-15605, 8230), (-14492, 7643)),
            ((4716, -13690), (5336, -15490)),
            ((12904, -11422), (12268, -10859)),
            ((2825, -6396), (6619, -14987)),
            ((4654, 15245), (4783, 15670)),
            ((-14769, 15133), (-11443, 11725)),
            ((-8090, -9057), (-10914, -12219)),
            ((-472, 1953), (-3848, 15925)),
            ((-12563, 1040), (-16328, 1351)),
            ((-7938, 15587), (-7435, 14599)),
            ((-9701, 5356), (-14343, 7919)),
            ((-642, -14484), (-725, -16367)),
            ((12963, -9690), (13123, -9809)),
            ((7067, 5361), (13053, 9902)),
            ((0x4000, 0), (0x4000, 0)),
            ((0, 0x4000), (0, 0x4000)),
            ((-0x4000, 0), (-0x4000, 0)),
            ((0, -0x4000), (0, -0x4000)),
        ];
        for ((x, y), expected) in cases {
            let n = super::normalize14(x, y);
            assert_eq!((n.x, n.y), expected);
            // Ensure the length of the vector is nearly 1.0
            let fx = F2Dot14::from_bits(n.x as _).to_f32();
            let fy = F2Dot14::from_bits(n.y as _).to_f32();
            let flen = (fx * fx + fy * fy).sqrt();
            assert!((flen - 1.0).abs() <= FLOAT_TOLERANCE);
        }
    }
}