nyx_space/dynamics/mod.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197
/*
Nyx, blazing fast astrodynamics
Copyright (C) 2018-onwards Christopher Rabotin <christopher.rabotin@gmail.com>
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
use crate::cosmic::{AstroError, Orbit};
use crate::linalg::allocator::Allocator;
use crate::linalg::{DefaultAllocator, DimName, Matrix3, Matrix4x3, OMatrix, OVector, Vector3};
use crate::State;
use anise::almanac::planetary::PlanetaryDataError;
use anise::almanac::Almanac;
use anise::errors::AlmanacError;
use hyperdual::Owned;
use snafu::Snafu;
use std::fmt;
use std::sync::Arc;
pub use crate::errors::NyxError;
/// The orbital module handles all Cartesian based orbital dynamics.
///
/// It is up to the engineer to ensure that the coordinate frames of the different dynamics borrowed
/// from this module match, or perform the appropriate coordinate transformations.
pub mod orbital;
use self::guidance::GuidanceError;
pub use self::orbital::*;
/// The gravity module handles spherical harmonics only. It _must_ be combined with a OrbitalDynamics dynamics
///
/// This module allows loading gravity models from [PDS](http://pds-geosciences.wustl.edu/), [EGM2008](http://earth-info.nga.mil/GandG/wgs84/gravitymod/egm2008/) and GMAT's own COF files.
// pub mod gravity;
/// The spacecraft module allows for simulation of spacecraft dynamics in general, including propulsion/maneuvers.
pub mod spacecraft;
pub use self::spacecraft::*;
/// Defines a few examples of guidance laws.
pub mod guidance;
/// Defines some velocity change controllers.
pub mod deltavctrl;
/// Defines solar radiation pressure models
pub mod solarpressure;
pub use self::solarpressure::*;
/// The drag module handles drag in a very basic fashion. Do not use for high fidelity dynamics.
pub mod drag;
pub use self::drag::*;
/// Define the spherical harmonic models.
pub mod sph_harmonics;
pub use self::sph_harmonics::*;
/// The `Dynamics` trait handles and stores any equation of motion *and* the state is integrated.
///
/// Its design is such that several of the provided dynamics can be combined fairly easily. However,
/// when combining the dynamics (e.g. integrating both the attitude of a spaceraft and its orbital
/// parameters), it is up to the implementor to handle time and state organization correctly.
/// For time management, I highly recommend using `hifitime` which is thoroughly validated.
#[allow(clippy::type_complexity)]
pub trait Dynamics: Clone + Sync + Send
where
DefaultAllocator: Allocator<<Self::StateType as State>::Size>
+ Allocator<<Self::StateType as State>::VecLength>
+ Allocator<<Self::StateType as State>::Size, <Self::StateType as State>::Size>,
{
/// The state of the associated hyperdual state, almost always StateType + U1
type HyperdualSize: DimName;
type StateType: State;
/// Defines the equations of motion for these dynamics, or a combination of provided dynamics.
/// The time delta_t is in **seconds** PAST the context epoch. The state vector is the state which
/// changes for every intermediate step of the integration. The state context is the state of
/// what is being propagated, it should allow rebuilding a new state context from the
/// provided state vector.
fn eom(
&self,
delta_t: f64,
state_vec: &OVector<f64, <Self::StateType as State>::VecLength>,
state_ctx: &Self::StateType,
almanac: Arc<Almanac>,
) -> Result<OVector<f64, <Self::StateType as State>::VecLength>, DynamicsError>
where
DefaultAllocator: Allocator<<Self::StateType as State>::VecLength>;
/// Defines the equations of motion for Dual numbers for these dynamics.
/// _All_ dynamics need to allow for automatic differentiation. However, if differentiation is not supported,
/// then the dynamics should prevent initialization with a context which has an STM defined.
fn dual_eom(
&self,
_delta_t: f64,
_osculating_state: &Self::StateType,
_almanac: Arc<Almanac>,
) -> Result<
(
OVector<f64, <Self::StateType as State>::Size>,
OMatrix<f64, <Self::StateType as State>::Size, <Self::StateType as State>::Size>,
),
DynamicsError,
>
where
DefaultAllocator: Allocator<Self::HyperdualSize>
+ Allocator<<Self::StateType as State>::Size>
+ Allocator<<Self::StateType as State>::Size, <Self::StateType as State>::Size>,
Owned<f64, Self::HyperdualSize>: Copy,
{
Err(DynamicsError::StateTransitionMatrixUnset)
}
/// Optionally performs some final changes after each successful integration of the equations of motion.
/// For example, this can be used to update the Guidance mode.
/// NOTE: This function is also called just prior to very first integration step in order to update the initial state if needed.
fn finally(
&self,
next_state: Self::StateType,
_almanac: Arc<Almanac>,
) -> Result<Self::StateType, DynamicsError> {
Ok(next_state)
}
}
/// The `ForceModel` trait handles immutable dynamics which return a force. Those will be divided by the mass of the spacecraft to compute the acceleration (F = ma).
///
/// Examples include Solar Radiation Pressure, drag, etc., i.e. forces which do not need to save the current state, only act on it.
pub trait ForceModel: Send + Sync + fmt::Display {
/// If a parameter of this force model is stored in the spacecraft state, then this function should return the index where this parameter is being affected
fn estimation_index(&self) -> Option<usize>;
/// Defines the equations of motion for this force model from the provided osculating state.
fn eom(&self, ctx: &Spacecraft, almanac: Arc<Almanac>) -> Result<Vector3<f64>, DynamicsError>;
/// Force models must implement their partials, although those will only be called if the propagation requires the
/// computation of the STM. The `osc_ctx` is the osculating context, i.e. it changes for each sub-step of the integrator.
/// The last row corresponds to the partials of the parameter of this force model wrt the position, i.e. this only applies to conservative forces.
fn dual_eom(
&self,
osc_ctx: &Spacecraft,
almanac: Arc<Almanac>,
) -> Result<(Vector3<f64>, Matrix4x3<f64>), DynamicsError>;
}
/// The `AccelModel` trait handles immutable dynamics which return an acceleration. Those can be added directly to Orbital Dynamics for example.
///
/// Examples include spherical harmonics, i.e. accelerations which do not need to save the current state, only act on it.
pub trait AccelModel: Send + Sync + fmt::Display {
/// Defines the equations of motion for this force model from the provided osculating state in the integration frame.
fn eom(&self, osc: &Orbit, almanac: Arc<Almanac>) -> Result<Vector3<f64>, DynamicsError>;
/// Acceleration models must implement their partials, although those will only be called if the propagation requires the
/// computation of the STM.
fn dual_eom(
&self,
osc_ctx: &Orbit,
almanac: Arc<Almanac>,
) -> Result<(Vector3<f64>, Matrix3<f64>), DynamicsError>;
}
/// Stores dynamical model errors
#[derive(Debug, PartialEq, Snafu)]
#[snafu(visibility(pub(crate)))]
pub enum DynamicsError {
/// Fuel exhausted at the provided spacecraft state
#[snafu(display("fuel exhausted at {sc}"))]
FuelExhausted { sc: Box<Spacecraft> },
#[snafu(display("expected STM to be set"))]
StateTransitionMatrixUnset,
#[snafu(display("dynamical model encountered an astro error: {source}"))]
DynamicsAstro { source: AstroError },
#[snafu(display("dynamical model encountered an issue with the guidance: {source}"))]
DynamicsGuidance { source: GuidanceError },
#[snafu(display("dynamical model issue due to Almanac: {action} {source}"))]
DynamicsAlmanacError {
action: &'static str,
#[snafu(source(from(AlmanacError, Box::new)))]
source: Box<AlmanacError>,
},
#[snafu(display("dynamical model issue due to planetary data: {action} {source}"))]
DynamicsPlanetaryError {
action: &'static str,
source: PlanetaryDataError,
},
}