Skip to main content

nyx_space/dynamics/guidance/
mod.rs

1/*
2    Nyx, blazing fast astrodynamics
3    Copyright (C) 2018-onwards Christopher Rabotin <christopher.rabotin@gmail.com>
4
5    This program is free software: you can redistribute it and/or modify
6    it under the terms of the GNU Affero General Public License as published
7    by the Free Software Foundation, either version 3 of the License, or
8    (at your option) any later version.
9
10    This program is distributed in the hope that it will be useful,
11    but WITHOUT ANY WARRANTY; without even the implied warranty of
12    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13    GNU Affero General Public License for more details.
14
15    You should have received a copy of the GNU Affero General Public License
16    along with this program.  If not, see <https://www.gnu.org/licenses/>.
17*/
18
19use crate::cosmic::{GuidanceMode, Orbit, Spacecraft, STD_GRAVITY};
20use crate::errors::{NyxError, StateError};
21use crate::linalg::Vector3;
22use anise::astro::PhysicsResult;
23use anise::errors::PhysicsError;
24use anise::math::rotation::DCM;
25use anise::prelude::Almanac;
26use der::{Decode, Encode, Reader};
27use serde::{Deserialize, Serialize};
28use serde_dhall::StaticType;
29
30pub mod mnvr;
31pub use mnvr::{Maneuver, MnvrRepr};
32
33mod replay;
34pub use replay::ThrustDirectionReplay;
35
36mod ruggiero;
37pub use ruggiero::{Objective, Ruggiero, StateParameter};
38
39mod kluever;
40pub use kluever::Kluever;
41use snafu::Snafu;
42
43use std::fmt;
44use std::sync::Arc;
45
46#[cfg(feature = "python")]
47use pyo3::prelude::*;
48
49/// Defines a thruster with a maximum isp and a maximum thrust.
50#[allow(non_snake_case)]
51#[cfg_attr(feature = "python", pyclass(get_all, set_all))]
52#[derive(Copy, Clone, Debug, PartialEq, Serialize, Deserialize, StaticType)]
53pub struct Thruster {
54    /// The thrust is to be provided in Newtons
55    pub thrust_N: f64,
56    /// The Isp is to be provided in seconds
57    pub isp_s: f64,
58}
59
60#[cfg_attr(feature = "python", pymethods)]
61impl Thruster {
62    /// Returns the exhaust velocity v_e in meters per second
63    pub fn exhaust_velocity_m_s(&self) -> f64 {
64        self.isp_s * STD_GRAVITY
65    }
66}
67
68#[cfg(feature = "python")]
69#[cfg_attr(feature = "python", pymethods)]
70impl Thruster {
71    #[allow(non_snake_case)]
72    #[new]
73    fn py_new(thrust_N: f64, isp_s: f64) -> Self {
74        Self { thrust_N, isp_s }
75    }
76}
77
78impl Encode for Thruster {
79    fn encoded_len(&self) -> der::Result<der::Length> {
80        self.thrust_N.encoded_len()? + self.isp_s.encoded_len()?
81    }
82
83    fn encode(&self, encoder: &mut impl der::Writer) -> der::Result<()> {
84        self.thrust_N.encode(encoder)?;
85        self.isp_s.encode(encoder)
86    }
87}
88
89impl<'a> Decode<'a> for Thruster {
90    fn decode<R: Reader<'a>>(decoder: &mut R) -> der::Result<Self> {
91        Ok(Self {
92            thrust_N: decoder.decode()?,
93            isp_s: decoder.decode()?,
94        })
95    }
96}
97
98#[derive(Clone, Debug, Serialize, Deserialize, StaticType)]
99pub struct ObjectiveEfficiency {
100    pub objective: Objective,
101    pub efficiency: f64,
102}
103
104#[derive(Clone, Debug, Serialize, Deserialize, StaticType)]
105pub struct ObjectiveWeight {
106    pub objective: Objective,
107    pub weight: f64,
108}
109
110/// The `GuidanceLaw` trait handles guidance laws, optimizations, and other such methods for
111/// controlling the overall thrust direction when tied to a `Spacecraft`. For delta V control,
112/// tie the DeltaVctrl to a MissionArc.
113pub trait GuidanceLaw: fmt::Display + Send + Sync {
114    /// Returns a unit vector corresponding to the thrust direction in the inertial frame.
115    fn direction(&self, osc_state: &Spacecraft) -> Result<Vector3<f64>, GuidanceError>;
116
117    /// Returns a number between [0;1] corresponding to the engine throttle level.
118    /// For example, 0 means coasting, i.e. no thrusting, and 1 means maximum thrusting.
119    fn throttle(&self, osc_state: &Spacecraft) -> Result<f64, GuidanceError>;
120
121    /// Updates the state of the BaseSpacecraft for the next maneuver, e.g. prepares the controller for the next maneuver
122    fn next(&self, next_state: &mut Spacecraft, almanac: Arc<Almanac>);
123
124    /// Returns whether this thrust control has been achieved, if it has an objective
125    fn achieved(&self, _osc_state: &Spacecraft) -> Result<bool, GuidanceError> {
126        Err(GuidanceError::NoGuidanceObjectiveDefined)
127    }
128}
129
130/// Converts the alpha (in-plane) and beta (out-of-plane) angles in the RCN frame to the unit vector in the RCN frame
131fn unit_vector_from_plane_angles(alpha: f64, beta: f64) -> Vector3<f64> {
132    Vector3::new(
133        alpha.sin() * beta.cos(),
134        alpha.cos() * beta.cos(),
135        beta.sin(),
136    )
137}
138
139/// Converts the provided unit vector into in-plane and out-of-plane angles, returned in radians
140pub fn plane_angles_from_unit_vector(vhat: Vector3<f64>) -> (f64, f64) {
141    (vhat[1].atan2(vhat[0]), vhat[2].asin())
142}
143
144/// Converts the alpha (in-plane) and beta (out-of-plane) angles in the RCN frame to the unit vector
145pub(crate) fn unit_vector_from_ra_dec(alpha: f64, delta: f64) -> Vector3<f64> {
146    Vector3::new(
147        delta.cos() * alpha.cos(),
148        delta.cos() * alpha.sin(),
149        delta.sin(),
150    )
151}
152
153/// Converts the provided unit vector into in-plane and out-of-plane angles, returned in radians
154pub(crate) fn ra_dec_from_unit_vector(vhat: Vector3<f64>) -> (f64, f64) {
155    let alpha = vhat[1].atan2(vhat[0]);
156    let delta = vhat[2].asin();
157    (alpha, delta)
158}
159
160#[derive(Debug, PartialEq, Snafu)]
161pub enum GuidanceError {
162    #[snafu(display("No thruster attached to spacecraft"))]
163    NoThrustersDefined,
164    #[snafu(display("Throttle is not between 0.0 and 1.0: {ratio}"))]
165    ThrottleRatio { ratio: f64 },
166    #[snafu(display("Invalid finite burn control direction u = [{x}, {y}, {z}] => i-plane = {in_plane_deg} deg, Delta = {out_of_plane_deg} deg",))]
167    InvalidDirection {
168        x: f64,
169        y: f64,
170        z: f64,
171        in_plane_deg: f64,
172        out_of_plane_deg: f64,
173    },
174    #[snafu(display("Invalid finite burn control rate u = [{x}, {y}, {z}] => in-plane = {in_plane_deg_s} deg/s, out of plane = {out_of_plane_deg_s} deg/s",))]
175    InvalidRate {
176        x: f64,
177        y: f64,
178        z: f64,
179        in_plane_deg_s: f64,
180        out_of_plane_deg_s: f64,
181    },
182    #[snafu(display("Invalid finite burn control acceleration u = [{x}, {y}, {z}] => in-plane = {in_plane_deg_s2} deg/s^2, out of plane = {out_of_plane_deg_s2} deg/s^2",))]
183    InvalidAcceleration {
184        x: f64,
185        y: f64,
186        z: f64,
187        in_plane_deg_s2: f64,
188        out_of_plane_deg_s2: f64,
189    },
190    #[snafu(display("when {action} encountered {source}"))]
191    GuidancePhysicsError {
192        action: &'static str,
193        source: PhysicsError,
194    },
195    #[snafu(display(
196        "An objective based analysis or control was attempted, but no objective was defined"
197    ))]
198    NoGuidanceObjectiveDefined,
199    #[snafu(display("{param} is not a control variable in this guidance law"))]
200    InvalidControl { param: StateParameter },
201    #[snafu(display("guidance encountered {source}"))]
202    GuidState { source: StateError },
203}
204
205/// Local frame options, used notably for guidance laws.
206/// TODO: Replace with ANISE enum, which is identical
207#[derive(Copy, Clone, Debug, PartialEq, Eq, Serialize, Deserialize, StaticType)]
208pub enum LocalFrame {
209    Inertial,
210    RIC,
211    VNC,
212    RCN,
213}
214
215impl LocalFrame {
216    pub fn dcm_to_inertial(&self, state: Orbit) -> PhysicsResult<DCM> {
217        match self {
218            LocalFrame::Inertial => Ok(DCM::identity(
219                state.frame.orientation_id,
220                state.frame.orientation_id,
221            )),
222            LocalFrame::RIC => state.dcm_from_ric_to_inertial(),
223            LocalFrame::VNC => state.dcm_from_vnc_to_inertial(),
224            LocalFrame::RCN => state.dcm_from_rcn_to_inertial(),
225        }
226    }
227}
228
229#[test]
230fn ra_dec_from_vec() {
231    use std::f64::consts::{FRAC_PI_2, PI, TAU};
232    let mut delta = -FRAC_PI_2;
233    let mut alpha = 0.0;
234    loop {
235        loop {
236            let unit_v = unit_vector_from_ra_dec(alpha, delta);
237            let (alpha2, delta2) = ra_dec_from_unit_vector(unit_v);
238            assert!((alpha - alpha2).abs() < f64::EPSILON);
239            assert!((delta - delta2).abs() < f64::EPSILON);
240            alpha += TAU * 0.1; // Increment right ascension by one tenth of a circle
241            if alpha > PI {
242                alpha = 0.0;
243                break;
244            }
245        }
246        delta += TAU * 0.1; // Increment declination by one tenth of a circle
247        if delta > FRAC_PI_2 {
248            break;
249        }
250    }
251}