use crate::int256;
macro_rules! make_lerp {
($i:ident, $u:ident, $ii:ident, $uu:ident, $uu1:expr) => {
pub const fn $i(r: $i, start: $i, end: $i, frac_bits: u32) -> ($i, bool) {
let (r_abs, r_neg) = if r >= 0 {
(r as $u, false)
} else {
(r.wrapping_neg() as $u, true)
};
let (range_abs, range_neg) = if end >= start {
(end.wrapping_sub(start) as $u, false)
} else {
(start.wrapping_sub(end) as $u, true)
};
let wide_abs = (r_abs as $uu) * (range_abs as $uu);
let wide_neg = r_neg != range_neg;
let wide_uns = if wide_neg {
let add = ($uu1 << frac_bits) - 1;
let shifted = (wide_abs + add) >> frac_bits;
shifted.wrapping_neg()
} else {
wide_abs >> frac_bits
};
let wide = wide_uns as $ii;
let (wide_ret, overflow1) = wide.overflowing_add(start as $ii);
let ret = wide_ret as $i;
let overflow2 = wide_ret != ret as $ii;
(ret, overflow1 | overflow2)
}
pub const fn $u(r: $u, start: $u, end: $u, frac_bits: u32) -> ($u, bool) {
let (range_abs, range_neg) = if end >= start {
(end - start, false)
} else {
(start - end, true)
};
let wide_abs = (r as $uu) * (range_abs as $uu);
let (wide_ret, overflow1) = if range_neg {
let add = ($uu1 << frac_bits) - 1;
let shifted = (wide_abs + add) >> frac_bits;
(start as $uu).overflowing_sub(shifted)
} else {
let shifted = wide_abs >> frac_bits;
(start as $uu).overflowing_add(shifted)
};
let ret = wide_ret as $u;
let overflow2 = wide_ret != ret as $uu;
(ret, overflow1 | overflow2)
}
};
}
make_lerp! { i8, u8, i16, u16, 1u16 }
make_lerp! { i16, u16, i32, u32, 1u32 }
make_lerp! { i32, u32, i64, u64, 1u64 }
make_lerp! { i64, u64, i128, u128, 1u128 }
pub const fn i128(r: i128, start: i128, end: i128, frac_bits: u32) -> (i128, bool) {
let (r_abs, r_neg) = if r >= 0 {
(r as u128, false)
} else {
(r.wrapping_neg() as u128, true)
};
let (range_abs, range_neg) = if end >= start {
(end.wrapping_sub(start) as u128, false)
} else {
(start.wrapping_sub(end) as u128, true)
};
let wide_abs = int256::wide_mul_u128(r_abs, range_abs);
let wide_neg = r_neg != range_neg;
let wide_uns = if wide_neg {
let add = if frac_bits == 128 {
u128::MAX
} else {
(1u128 << frac_bits) - 1
};
let sum = int256::wrapping_add_u256_u128(wide_abs, add);
let shifted = int256::shl_u256_max_128(sum, frac_bits);
int256::wrapping_neg_u256(shifted)
} else {
int256::shl_u256_max_128(wide_abs, frac_bits)
};
let wide = int256::u256_wrapping_as_i256(wide_uns);
let (wide_ret, overflow1) = int256::overflowing_add_i256_i128(wide, start);
let ret = wide_ret.lo as i128;
let overflow2 = if ret < 0 {
wide_ret.hi != -1
} else {
wide_ret.hi != 0
};
(ret, overflow1 | overflow2)
}
pub const fn u128(r: u128, start: u128, end: u128, frac_bits: u32) -> (u128, bool) {
let (range_abs, range_neg) = if end >= start {
(end - start, false)
} else {
(start - end, true)
};
let wide_abs = int256::wide_mul_u128(r, range_abs);
if range_neg {
let add = if frac_bits == 128 {
u128::MAX
} else {
(1u128 << frac_bits) - 1
};
let sum = int256::wrapping_add_u256_u128(wide_abs, add);
let shifted = int256::shl_u256_max_128(sum, frac_bits);
int256::overflowing_sub_u128_u256(start, shifted)
} else {
let shifted = int256::shl_u256_max_128(wide_abs, frac_bits);
int256::overflowing_add_u128_u256(start, shifted)
}
}
#[cfg(test)]
mod tests {
use crate::lerp;
#[test]
fn lerp_i8() {
assert_eq!(lerp::i8(0, -128, 127, 0), (-128, false));
assert_eq!(lerp::i8(0, 127, -128, 8), (127, false));
assert_eq!(lerp::i8(1, -128, 127, 1), (-1, false));
assert_eq!(lerp::i8(-64, -100, -110, 6), (-90, false));
assert_eq!(lerp::i8(2 << 2, 0, 50, 2), (100, false));
assert_eq!(lerp::i8(2 << 2, 0, 100, 2), (-56, true));
assert_eq!(lerp::i8((-1) << 2, 50, 0, 2), (100, false));
assert_eq!(lerp::i8((-1) << 2, 100, 0, 2), (-56, true));
}
#[test]
fn lerp_u8() {
assert_eq!(lerp::u8(0, 0, 255, 0), (0, false));
assert_eq!(lerp::u8(0, 255, 0, 8), (255, false));
assert_eq!(lerp::u8(1, 0, 255, 0), (255, false));
assert_eq!(lerp::u8(128, 255, 0, 7), (0, false));
assert_eq!(lerp::u8(2 << 2, 0, 100, 2), (200, false));
assert_eq!(lerp::u8(2 << 2, 0, 200, 2), (200 - 56, true));
}
#[test]
fn lerp_i128() {
assert_eq!(lerp::i128(0, -128, 127, 0), (-128, false));
assert_eq!(lerp::i128(0, 127, -128, 128), (127, false));
assert_eq!(lerp::i128(1, -128, 127, 1), (-1, false));
assert_eq!(lerp::i128(1 << 98, 127, -128, 100), (63, false));
assert_eq!(lerp::i128((-64) << 120, -100, -110, 126), (-90, false));
assert_eq!(
lerp::i128(2 << 2, 0, i128::MAX / 2, 2),
(i128::MAX / 2 * 2, false)
);
assert_eq!(
lerp::i128(2 << 2, 0, i128::MAX, 2),
(i128::MAX.wrapping_mul(2), true)
);
assert_eq!(
lerp::i128((-1) << 2, i128::MAX / 2, 0, 2),
(i128::MAX / 2 * 2, false)
);
assert_eq!(
lerp::i128((-1) << 2, i128::MAX, 0, 2),
(i128::MAX.wrapping_mul(2), true)
);
}
#[test]
fn lerp_u128() {
assert_eq!(lerp::u128(0, 0, 255, 0), (0, false));
assert_eq!(lerp::u128(0, 255, 0, 128), (255, false));
assert_eq!(lerp::u128(1, 0, 255, 0), (255, false));
assert_eq!(lerp::u128(128 << 120, 255, 0, 127), (0, false));
assert_eq!(
lerp::u128(2 << 2, 0, u128::MAX / 2, 2),
(u128::MAX / 2 * 2, false)
);
assert_eq!(
lerp::u128(2 << 2, 0, u128::MAX, 2),
(u128::MAX.wrapping_mul(2), true)
);
}
}