use super::super::FloatType;
use super::super::Roots;
pub fn find_roots_cubic_depressed<F: FloatType>(a1: F, a0: F) -> Roots<F> {
let _2 = F::from(2i16);
let _3 = F::from(3i16);
let _4 = F::from(4i16);
let _9 = F::from(9i16);
let _18 = F::from(18i16);
let _27 = F::from(27i16);
let _54 = F::from(54i16);
if a1 == F::zero() {
Roots::One([-a0.cbrt()])
} else if a0 == F::zero() {
super::quadratic::find_roots_quadratic(F::one(), F::zero(), a1).add_new_root(F::zero())
} else {
let d = a0 * a0 / _4 + a1 * a1 * a1 / _27;
if d < F::zero() {
let a = (-_4 * a1 / _3).sqrt();
let phi = (-_4 * a0 / (a * a * a)).acos() / _3;
Roots::One([a * phi.cos()])
.add_new_root(a * (phi + F::two_third_pi()).cos())
.add_new_root(a * (phi - F::two_third_pi()).cos())
} else {
let sqrt_d = d.sqrt();
let a0_div_2 = a0 / _2;
let x1 = (sqrt_d - a0_div_2).cbrt() - (sqrt_d + a0_div_2).cbrt();
if d == F::zero() {
Roots::One([x1]).add_new_root(a0_div_2)
} else {
Roots::One([x1])
}
}
}
}
#[cfg(test)]
mod test {
use super::super::super::*;
#[test]
fn test_find_roots_cubic_depressed() {
assert_eq!(find_roots_cubic_depressed(0f32, 0f32), Roots::One([0f32]));
assert_eq!(find_roots_cubic_depressed(-1f64, 0f64), Roots::Three([-1f64, 0f64, 1f64]));
match find_roots_cubic_depressed(-2f64, 2f64) {
Roots::One(x) => {
assert_float_array_eq!(1e-15, x, [-1.769292354238631415240409f64]);
}
_ => {
assert!(false);
}
}
match find_roots_cubic_depressed(-3f64, 2f64) {
Roots::Two(x) => {
assert_float_array_eq!(1e-15, x, [-2f64, 1f64]);
}
_ => {
assert!(false);
}
}
match find_roots_cubic_depressed(-2f64, 1f64) {
Roots::Three(x) => {
assert_float_array_eq!(1e-15, x, [(-1f64 - 5f64.sqrt()) / 2f64, (-1f64 + 5f64.sqrt()) / 2f64, 1f64]);
}
_ => {
assert!(false);
}
}
}
}