use crate::astro::math::mat3::{self, Mat3};
use crate::astro::math::vec3;
use crate::validate;
use nalgebra::SMatrix;
const ZERO_POSITION_EPS: f64 = 1.0e-30;
const PARALLEL_RV_EPS: f64 = 1.0e-30;
const PSD_DIAGONAL_EPS: f64 = 1.0e-15;
const PSD_MINOR_EPS: f64 = 1.0e-12;
const SYMMETRY_EPS: f64 = 1.0e-12;
const SYMMETRY_REL_EPS6: f64 = 1.0e-12;
const PSD6_EIGEN_REL_EPS: f64 = 1.0e-10;
pub type Mat6 = [[f64; 6]; 6];
#[derive(Debug, Clone, Copy, PartialEq)]
pub struct Covariance6 {
matrix: Mat6,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum Covariance6Error {
NonFinite,
Asymmetric,
NotPositiveSemidefinite,
}
impl Covariance6 {
pub fn try_from_matrix(matrix: Mat6) -> Result<Self, Covariance6Error> {
if !finite6(&matrix) {
return Err(Covariance6Error::NonFinite);
}
if !symmetric6(&matrix) {
return Err(Covariance6Error::Asymmetric);
}
if !positive_semidefinite6(&matrix) {
return Err(Covariance6Error::NotPositiveSemidefinite);
}
Ok(Self { matrix })
}
pub fn from_diagonal(diagonal: [f64; 6]) -> Result<Self, Covariance6Error> {
let mut matrix = [[0.0_f64; 6]; 6];
for (idx, value) in diagonal.into_iter().enumerate() {
matrix[idx][idx] = value;
}
Self::try_from_matrix(matrix)
}
pub const fn from_matrix_unchecked(matrix: Mat6) -> Self {
Self { matrix }
}
pub const fn as_matrix(&self) -> &Mat6 {
&self.matrix
}
pub const fn into_matrix(self) -> Mat6 {
self.matrix
}
pub fn position_covariance_km2(&self) -> Mat3 {
[
[self.matrix[0][0], self.matrix[0][1], self.matrix[0][2]],
[self.matrix[1][0], self.matrix[1][1], self.matrix[1][2]],
[self.matrix[2][0], self.matrix[2][1], self.matrix[2][2]],
]
}
pub fn is_symmetric(&self) -> bool {
symmetric6(&self.matrix)
}
pub fn is_positive_semidefinite(&self) -> bool {
positive_semidefinite6(&self.matrix)
}
#[allow(clippy::needless_range_loop)]
pub fn propagate_with_stm(&self, stm: &Mat6) -> Result<Self, Covariance6Error> {
if !finite6(stm) {
return Err(Covariance6Error::NonFinite);
}
let mut temp = [[0.0_f64; 6]; 6];
for i in 0..6 {
for j in 0..6 {
for k in 0..6 {
temp[i][j] += stm[i][k] * self.matrix[k][j];
}
}
}
let mut propagated = [[0.0_f64; 6]; 6];
for i in 0..6 {
for j in 0..6 {
for k in 0..6 {
propagated[i][j] += temp[i][k] * stm[j][k];
}
}
}
symmetrize6(&mut propagated);
Self::try_from_matrix(propagated)
}
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum RtnFrameError {
InvalidInput {
field: &'static str,
reason: &'static str,
},
ZeroPosition,
ParallelPositionVelocity,
}
impl RtnFrameError {
pub fn message(self) -> &'static str {
match self {
RtnFrameError::InvalidInput { .. } => "invalid input",
RtnFrameError::ZeroPosition => "zero position vector",
RtnFrameError::ParallelPositionVelocity => "position and velocity are parallel",
}
}
}
fn invalid_input(field: &'static str, reason: &'static str) -> RtnFrameError {
RtnFrameError::InvalidInput { field, reason }
}
fn validate_vec3(field: &'static str, values: [f64; 3]) -> Result<(), RtnFrameError> {
if values.iter().all(|value| value.is_finite()) {
Ok(())
} else {
Err(invalid_input(field, "components must be finite"))
}
}
fn validate_covariance(field: &'static str, values: &Mat3) -> Result<(), RtnFrameError> {
validate::validate_covariance_psd(values, field).map_err(|error| match error {
validate::FieldError::NonFinite { field } => {
invalid_input(field, "components must be finite")
}
validate::FieldError::NotPositive { field } => invalid_input(field, "not positive"),
validate::FieldError::Negative { field } => invalid_input(field, "negative"),
validate::FieldError::OutOfRange { field, .. } => invalid_input(field, "out of range"),
validate::FieldError::Missing { field }
| validate::FieldError::FloatParse { field, .. }
| validate::FieldError::IntParse { field, .. }
| validate::FieldError::InvalidCivilDate { field, .. }
| validate::FieldError::InvalidCivilTime { field, .. } => invalid_input(field, "invalid"),
})
}
fn validate_mat3_finite(field: &'static str, values: &Mat3) -> Result<(), RtnFrameError> {
for row in values {
validate_vec3(field, *row)?;
}
Ok(())
}
pub fn rtn_to_eci_rotation(r: [f64; 3], v: [f64; 3]) -> Result<Mat3, RtnFrameError> {
validate_vec3("position", r)?;
validate_vec3("velocity", v)?;
if vec3::norm3(r) < ZERO_POSITION_EPS {
return Err(RtnFrameError::ZeroPosition);
}
let r_hat = vec3::unit3_ref_unchecked(&r);
let h = vec3::cross3(r, v);
if vec3::norm3(h) < PARALLEL_RV_EPS {
return Err(RtnFrameError::ParallelPositionVelocity);
}
let n_hat = vec3::unit3_ref_unchecked(&h);
let t_hat = vec3::cross3(n_hat, r_hat);
Ok([
[r_hat[0], t_hat[0], n_hat[0]],
[r_hat[1], t_hat[1], n_hat[1]],
[r_hat[2], t_hat[2], n_hat[2]],
])
}
pub fn rtn_to_eci(cov_rtn: &Mat3, r: [f64; 3], v: [f64; 3]) -> Result<Mat3, RtnFrameError> {
validate_covariance("cov_rtn", cov_rtn)?;
let rot = rtn_to_eci_rotation(r, v)?;
let rot_t = mat3::inline_tr(&rot);
let cov_eci = mat3::inline_rxr(&mat3::inline_rxr(&rot, cov_rtn), &rot_t);
validate_mat3_finite("cov_eci", &cov_eci)?;
Ok(cov_eci)
}
pub fn symmetric(m: &Mat3) -> bool {
(m[0][1] - m[1][0]).abs() < SYMMETRY_EPS
&& (m[0][2] - m[2][0]).abs() < SYMMETRY_EPS
&& (m[1][2] - m[2][1]).abs() < SYMMETRY_EPS
}
fn det3x3(m: &Mat3) -> f64 {
let (a, b, c) = (m[0][0], m[0][1], m[0][2]);
let (d, e, f) = (m[1][0], m[1][1], m[1][2]);
let (g, h, i) = (m[2][0], m[2][1], m[2][2]);
a * (e * i - f * h) - b * (d * i - f * g) + c * (d * h - e * g)
}
pub fn positive_semidefinite(m: &Mat3) -> bool {
if !symmetric(m) {
return false;
}
let m11 = m[0][0];
let m22 = m[1][1];
let m33 = m[2][2];
let m12 = m[0][1];
let m13 = m[0][2];
let m23 = m[1][2];
let det12 = m11 * m22 - m12 * m12;
let det13 = m11 * m33 - m13 * m13;
let det23 = m22 * m33 - m23 * m23;
let det123 = det3x3(m);
m11 >= -PSD_DIAGONAL_EPS
&& m22 >= -PSD_DIAGONAL_EPS
&& m33 >= -PSD_DIAGONAL_EPS
&& det12 >= -PSD_MINOR_EPS
&& det13 >= -PSD_MINOR_EPS
&& det23 >= -PSD_MINOR_EPS
&& det123 >= -PSD_MINOR_EPS
}
fn finite6(m: &Mat6) -> bool {
m.iter().flatten().all(|value| value.is_finite())
}
fn covariance_scale6(m: &Mat6) -> f64 {
(0..6).fold(0.0_f64, |scale, idx| scale.max(m[idx][idx].abs()))
}
#[allow(clippy::needless_range_loop)]
fn symmetric6(m: &Mat6) -> bool {
let tolerance = SYMMETRY_REL_EPS6 * covariance_scale6(m);
for i in 0..6 {
for j in (i + 1)..6 {
if (m[i][j] - m[j][i]).abs() > tolerance {
return false;
}
}
}
true
}
fn positive_semidefinite6(m: &Mat6) -> bool {
if !finite6(m) || !symmetric6(m) {
return false;
}
let matrix = SMatrix::<f64, 6, 6>::from_fn(|i, j| m[i][j]);
let eigenvalues = matrix.symmetric_eigen().eigenvalues;
let scale = covariance_scale6(m);
let floor = -PSD6_EIGEN_REL_EPS * scale;
eigenvalues.iter().all(|&lambda| lambda >= floor)
}
#[allow(clippy::needless_range_loop)]
fn symmetrize6(m: &mut Mat6) {
for i in 0..6 {
for j in (i + 1)..6 {
let value = 0.5 * (m[i][j] + m[j][i]);
m[i][j] = value;
m[j][i] = value;
}
}
}
#[cfg(test)]
mod tests {
use super::*;
const RTN_TO_ECI_GOLDEN_BITS: [u64; 9] = [
0x4010077f74cce7ac,
0xbfd92b0043adb450,
0x3fe26dc422b0767a,
0xbfd92b0043adb44a,
0x402207fb1ad4c218,
0xbfb9ef5fd1874930,
0x3fe26dc422b0767a,
0xbfb9ef5fd1874930,
0x402ff4452ac4ca0f,
];
#[test]
fn rtn_to_eci_matches_frozen_elixir_bits() {
let r = [7000.123, 1234.5, -250.7];
let v = [1.2, 7.4, 0.3];
let cov_rtn = [[4.0, 0.5, 0.1], [0.5, 9.0, 0.2], [0.1, 0.2, 16.0]];
let eci = rtn_to_eci(&cov_rtn, r, v).expect("non-degenerate state");
let mut flat = [0u64; 9];
for (idx, slot) in flat.iter_mut().enumerate() {
*slot = eci[idx / 3][idx % 3].to_bits();
}
assert_eq!(flat, RTN_TO_ECI_GOLDEN_BITS);
}
#[test]
fn rtn_to_eci_aligned_state_is_exactly_the_rtn_diagonal() {
let r = [7000.0, 0.0, 0.0];
let v = [0.0, 7.5, 0.0];
let cov_rtn = [[1.0, 0.0, 0.0], [0.0, 2.0, 0.0], [0.0, 0.0, 3.0]];
let eci = rtn_to_eci(&cov_rtn, r, v).expect("non-degenerate state");
assert_eq!(eci[0][0].to_bits(), 1.0_f64.to_bits());
assert_eq!(eci[1][1].to_bits(), 2.0_f64.to_bits());
assert_eq!(eci[2][2].to_bits(), 3.0_f64.to_bits());
}
#[test]
fn rtn_to_eci_rejects_zero_position() {
let err = rtn_to_eci(&identity(), [0.0, 0.0, 0.0], [0.0, 7.5, 0.0]).unwrap_err();
assert_eq!(err, RtnFrameError::ZeroPosition);
assert_eq!(err.message(), "zero position vector");
}
#[test]
fn rtn_to_eci_rejects_parallel_position_velocity() {
let err = rtn_to_eci(&identity(), [7000.0, 0.0, 0.0], [1.0, 0.0, 0.0]).unwrap_err();
assert_eq!(err, RtnFrameError::ParallelPositionVelocity);
assert_eq!(err.message(), "position and velocity are parallel");
}
#[test]
fn rtn_to_eci_rejects_nonfinite_geometry_and_covariance() {
let err = rtn_to_eci(&identity(), [7000.0, f64::NAN, 0.0], [0.0, 7.5, 0.0]).unwrap_err();
assert_eq!(
err,
RtnFrameError::InvalidInput {
field: "position",
reason: "components must be finite",
}
);
let err =
rtn_to_eci(&identity(), [7000.0, 0.0, 0.0], [0.0, f64::INFINITY, 0.0]).unwrap_err();
assert_eq!(
err,
RtnFrameError::InvalidInput {
field: "velocity",
reason: "components must be finite",
}
);
let mut cov = identity();
cov[2][1] = f64::NEG_INFINITY;
let err = rtn_to_eci(&cov, [7000.0, 0.0, 0.0], [0.0, 7.5, 0.0]).unwrap_err();
assert_eq!(
err,
RtnFrameError::InvalidInput {
field: "cov_rtn",
reason: "components must be finite",
}
);
}
#[test]
fn rtn_to_eci_rejects_invalid_covariance_geometry() {
let r = [7000.0, 0.0, 0.0];
let v = [0.0, 7.5, 0.0];
let mut negative_variance = identity();
negative_variance[0][0] = -1.0;
let err = rtn_to_eci(&negative_variance, r, v).unwrap_err();
assert_eq!(
err,
RtnFrameError::InvalidInput {
field: "cov_rtn",
reason: "not positive",
}
);
let asymmetric = [[1.0, 0.5, 0.0], [0.4, 1.0, 0.0], [0.0, 0.0, 1.0]];
let err = rtn_to_eci(&asymmetric, r, v).unwrap_err();
assert_eq!(
err,
RtnFrameError::InvalidInput {
field: "cov_rtn",
reason: "not positive",
}
);
let indefinite = [[1.0, 2.0, 0.0], [2.0, 1.0, 0.0], [0.0, 0.0, 1.0]];
let err = rtn_to_eci(&indefinite, r, v).unwrap_err();
assert_eq!(
err,
RtnFrameError::InvalidInput {
field: "cov_rtn",
reason: "not positive",
}
);
}
#[test]
fn positive_semidefinite_accepts_identity_rejects_negative_and_asymmetric() {
assert!(positive_semidefinite(&identity()));
let negative_diag = [[-1.0, 0.0, 0.0], [0.0, 1.0, 0.0], [0.0, 0.0, 1.0]];
assert!(!positive_semidefinite(&negative_diag));
let asymmetric = [[1.0, 0.5, 0.0], [0.4, 1.0, 0.0], [0.0, 0.0, 1.0]];
assert!(!symmetric(&asymmetric));
assert!(!positive_semidefinite(&asymmetric));
}
#[test]
fn positive_semidefinite_rejects_symmetric_indefinite_matrix() {
let indefinite = [[1.0, 2.0, 0.0], [2.0, 1.0, 0.0], [0.0, 0.0, 1.0]];
assert!(symmetric(&indefinite));
assert!(!positive_semidefinite(&indefinite));
}
#[test]
fn covariance6_accepts_diagonal_and_rejects_bad_matrices() {
let covariance =
Covariance6::from_diagonal([1.0, 2.0, 3.0, 1.0e-6, 2.0e-6, 3.0e-6]).unwrap();
assert!(covariance.is_symmetric());
assert!(covariance.is_positive_semidefinite());
let mut asymmetric = *covariance.as_matrix();
asymmetric[0][1] = 1.0e-3;
assert_eq!(
Covariance6::try_from_matrix(asymmetric),
Err(Covariance6Error::Asymmetric)
);
let mut indefinite = *covariance.as_matrix();
indefinite[5][5] = -1.0;
assert_eq!(
Covariance6::try_from_matrix(indefinite),
Err(Covariance6Error::NotPositiveSemidefinite)
);
}
#[test]
fn covariance6_scales_psd_tolerance_to_covariance_magnitude() {
let mut large = [[0.0_f64; 6]; 6];
for (idx, row) in large.iter_mut().enumerate() {
row[idx] = 1.0e18;
}
large[0][1] = 2.5e17;
large[1][0] = 2.5e17 + 1.0e3;
let covariance = Covariance6::try_from_matrix(large).expect("large PSD covariance");
assert!(covariance.is_symmetric());
assert!(covariance.is_positive_semidefinite());
let mut indefinite = large;
indefinite[2][2] = -1.0e9;
assert_eq!(
Covariance6::try_from_matrix(indefinite),
Err(Covariance6Error::NotPositiveSemidefinite)
);
let small =
Covariance6::from_diagonal([1.0e-18, 2.0e-18, 3.0e-18, 4.0e-18, 5.0e-18, 6.0e-18])
.expect("small PSD covariance");
assert!(small.is_symmetric());
assert!(small.is_positive_semidefinite());
let mut small_indefinite = *small.as_matrix();
small_indefinite[0][0] = -1.0e-20;
assert_eq!(
Covariance6::try_from_matrix(small_indefinite),
Err(Covariance6Error::NotPositiveSemidefinite)
);
}
fn identity() -> Mat3 {
[[1.0, 0.0, 0.0], [0.0, 1.0, 0.0], [0.0, 0.0, 1.0]]
}
}