Module: constraintDynamicEffector
Executive Summary
The constraint effector class is used to define a physical connection between two separate spacecraft. This class takes in the connection location on each spacecraft in their respective body frames as well as a relative vector between these two connection points. It also takes in two gain tuning parameters to tailor the stiffness and damping of the connection between the two spacecraft.
The constraint effector class is an instantiation of the dynamic effector abstract class. The included test is validating the interaction between two spacecraft rigid body hubs that are attached through a constraint effector. In this case, two identical spacecraft are connected by a 0.1 meter long arm which is enforced with high stiffness and damping to be virtually rigid.
Detailed Module Description
A constraint effector is a combination of two separate 3-degree-of-freedom holonomic constraints: a direction constraint which enforces the position of the connection point on each spacecraft relative to the other, and an attitude constraint which enforces the attitude of each spacecraft relative to the other. The constraint effector works by correcting violations of these constraints as well as their derivatives.
Mathematical Modeling
See the following conference paper for a detailed description of the mathematical model.
Note
J. Vaz Carneiro, A. Morell and H. Schaub, “Post-Docking Complex Spacecraft Dynamics Using Baumgarte Stabilization”, AAS/AIAA Astrodynamics Specialist Conference, Big Sky, Montana, Aug. 13-17, 2023
User Guide
This section outlines the steps needed to setup a Constraint Dynamic Effector in Python using Basilisk.
Import the constraintDynamicEffector class:
from Basilisk.simulation import constraintDynamicEffector
Create an instantiation of a holonomic constraint:
constraintEffector = constraintDynamicEffector.ConstraintDynamicEffector()
Define all physical parameters for the constraint:
constraintEffector.setR_P1B1_B1(r_P1B1_B1) constraintEffector.setR_P2B2_B2(r_P2B2_B2) constraintEffector.setR_P2P1_B1Init(r_P2P1_B1Init)
Define the stiffness and damping of the connection. See the recommended starting gains here:
constraintEffector.setAlpha(1E2) constraintEffector.setBeta(1e3)
(Optional) Define exact gains for the direction and attitude constraints separately. These are internally set based on alpha and beta during reset, but can be overridden in this way:
constraintEffector.setK_d(alpha**2) constraintEffector.setC_d(2*beta) constraintEffector.setK_a(alpha**2) constraintEffector.setC_a(2*beta)
Add the effector to both spacecraft:
scObject1.addDynamicEffector(constraintEffector) scObject2.addDynamicEffector(constraintEffector)
See Module: spacecraft documentation on how to set up a spacecraft object.
Add the module to the task list:
Sim.AddModelToTask(TaskName, constraintEffector)
Ensure that the dynamics integration is synced between the connected spacecraft. For best results use a variable timestep integrator:
integratorObject1 = svIntegrators.svIntegratorRKF45(scObject1) scObject1.setIntegrator(integratorObject1) scObject1.syncDynamicsIntegration(scObject2)
(Optional) Retrieve the constraint effector’s physical parameters, gain tuning parameters, or exact gains for the direction and attitude constraints:
constraintEffector.getR_P1B1_B1(r_P1B1_B1) constraintEffector.getR_P2B2_B2(r_P2B2_B2) constraintEffector.getR_P2P1_B1Init(r_P2P1_B1Init) constraintEffector.getAlpha(1E2) constraintEffector.getBeta(1e3) constraintEffector.getK_d(alpha**2) constraintEffector.getC_d(2*beta) constraintEffector.getK_a(alpha**2) constraintEffector.getC_a(2*beta)
(Optional) Define a input device status message.(1 means constraintEffector is connected.0 means constraintEffector is disconnected). If not set, it defaults to being connected:
effectorStatusMsgPayload = messaging.DeviceStatusMsgPayload() effectorStatusMsgPayload.deviceStatus = 1 effectorStatusMsg = messaging.DeviceStatusMsg().write(effectorStatusMsgPayload) constraintEffector.effectorStatusInMsg.subscribeTo(effectorStatusMsg)
(Optional) Setup Low Pass Filtering for the Constraint Forces and Torques acting on the two satellites. Define a positive cut-off frequency wc for the low-pass filter. If not set, defaults to 0:
constraintEffector.setFilterData(wc)
The constraintEffector output message records the raw and filtered constraint forces and torques acting on the two spacecraft using the variable
constraintElements
.
-
class ConstraintDynamicEffector : public SysModel, public DynamicEffector
- #include <constraintDynamicEffector.h>
constraint dynamic effector class
Public Functions
-
ConstraintDynamicEffector()
This is the constructor, nothing to report here
-
~ConstraintDynamicEffector()
This is the destructor, nothing to report here
-
void Reset(uint64_t CurrentSimNanos)
This method is used to reset the module.
-
void linkInStates(DynParamManager &states)
This method allows the constraint effector to have access to the parent states
- Parameters:
states – The states to link
-
void computeForceTorque(double integTime, double timeStep)
This method computes the forces on torques on each spacecraft body.
- Parameters:
integTime – Integration time
timeStep – Current integration time step used
-
void UpdateState(uint64_t CurrentSimNanos)
Update state method
- Parameters:
CurrentSimNanos – The current simulation time
-
void writeOutputStateMessage(uint64_t CurrentClock)
This method takes the computed constraint force and torque states and outputs them to the m messaging system.
- Parameters:
CurrentClock – The current simulation time (used for time stamping)
-
void computeFilteredForce(uint64_t CurrentClock)
Filtering method to calculate filtered Constraint Force
- Parameters:
CurrentClock – The current simulation time (used for time stamping)
-
void computeFilteredTorque(uint64_t CurrentClock)
Filtering method to calculate filtered Constraint Torque
- Parameters:
CurrentClock – The current simulation time (used for time stamping)
-
void readInputMessage()
This method allows the user to set the status of the constraint dynamic effector
-
void setR_P2P1_B1Init(Eigen::Vector3d r_P2P1_B1Init)
setter for
r_P2P1_B1Init
initial spacecraft separation
-
void setR_P1B1_B1(Eigen::Vector3d r_P1B1_B1)
setter for
r_P1B1_B1
connection point position on spacecraft 1
-
void setR_P2B2_B2(Eigen::Vector3d r_P2B2_B2)
setter for
r_P2B2_B2
connection point position on spacecraft 2
-
void setAlpha(double alpha)
setter for
alpha
gain tuning parameter
-
void setBeta(double beta)
setter for
beta
gain tuning parameter
-
void setK_d(double k_d)
setter for
k_d
gain
-
void setC_d(double c_d)
setter for
c_d
gain
-
void setK_a(double k_a)
setter for
k_a
gain
-
void setC_a(double c_a)
setter for
c_a
gain
-
void setFilter_Data(double wc, double h, double k)
setter for
a,b,s,c,d,e
coefficients of low pass filterThis method allows the user to set the cut-off frequency of the low pass filter which is then used to calculate the coefficients for numerical low pass filtering based on a second-order low pass filter design.
- Parameters:
wc – The cut-off frequency of the low pass filter.
h – The constant digital time step.
k – The damping coefficient
-
inline Eigen::Vector3d getR_P2P1_B1Init() const
getter for
r_P2P1_B1Init
initial spacecraft separation
-
inline Eigen::Vector3d getR_P1B1_B1() const
getter for
r_P1B1_B1
connection point position on spacecraft 1
-
inline Eigen::Vector3d getR_P2B2_B2() const
getter for
r_P2B2_B2
connection point position on spacecraft 2
-
inline double getAlpha() const
getter for
alpha
gain tuning parameter
-
inline double getBeta() const
getter for
beta
gain tuning parameter
-
inline double getK_d() const
getter for
k_d
gain
-
inline double getC_d() const
getter for
c_d
gain
-
inline double getK_a() const
getter for
k_a
gain
-
inline double getC_a() const
getter for
c_a
gain
Public Members
-
Message<ConstDynEffectorMsgPayload> constraintElements
output message with constraint force and torque on connected s/c
-
ReadFunctor<DeviceStatusMsgPayload> effectorStatusInMsg
input message to record device status
-
uint64_t effectorStatus = 1
internal variable to toggle effector status
Private Members
-
int scInitCounter = 0
counter to kill simulation if more than two spacecraft initialized
-
int scID = 1
0,1 alternating spacecraft tracker to output appropriate force/torque
-
Eigen::Vector3d r_P1B1_B1 = Eigen::Vector3d::Zero()
[m] position vector from spacecraft 1 hub to its connection point P1
-
Eigen::Vector3d r_P2B2_B2 = Eigen::Vector3d::Zero()
[m] position vector from spacecraft 2 hub to its connection point P2
-
Eigen::Vector3d r_P2P1_B1Init = Eigen::Vector3d::Zero()
[m] precribed position vector from spacecraft 1 connection point to spacecraft 2 connection point
-
double alpha = 0.0
Baumgarte stabilization gain tuning variable.
-
double beta = 0.0
Baumgarte stabilization gain tuning variable.
-
double k_d = 0.0
direction constraint proportional gain
-
double c_d = 0.0
direction constraint derivative gain
-
double k_a = 0.0
attitude constraint proportional gain
-
double c_a = 0.0
attitude constraint derivative gain
-
double a = 0.0
coefficient in numerical low pass filter
-
double b = 0.0
coefficient in numerical low pass filter
-
double c = 0.0
coefficient in numerical low pass filter
-
double d = 0.0
coefficient in numerical low pass filter
-
double e = 0.0
coefficient in numerical low pass filter
-
double F_mag_tminus2 = 0.0
Magnitude of unfiltered constraint force at t-2 time step.
-
double F_mag_tminus1 = 0.0
Magnitude of unfiltered constraint force at t-1 time step.
-
double F_mag_t = 0.0
Magnitude of unfiltered constraint force at t time step.
-
double F_filtered_mag_t = 0.0
Magnitude of filtered constraint force at t time step.
-
double F_filtered_mag_tminus1 = 0.0
Magnitude of filtered constraint force at t-1 time step.
-
double F_filtered_mag_tminus2 = 0.0
Magnitude of filtered constraint force at t-2 time step.
-
double T1_mag_tminus2 = 0.0
Magnitude of unfiltered constraint torque on s/c 1 at t-2 time step.
-
double T1_mag_tminus1 = 0.0
Magnitude of unfiltered constraint torque on s/c 1 at t-1 time step.
-
double T1_mag_t = 0.0
Magnitude of unfiltered constraint torque on s/c 1 at t time step.
-
double T1_filtered_mag_t = 0.0
Magnitude of filtered constraint torque on s/c 1 at t time step.
-
double T1_filtered_mag_tminus1 = 0.0
Magnitude of filtered constraint torque on s/c 1 at t-1 time step.
-
double T1_filtered_mag_tminus2 = 0.0
Magnitude of filtered constraint torque on s/c 1 at t-2 time step.
-
double T2_mag_tminus2 = 0.0
Magnitude of unfiltered constraint torque on s/c 2 at t-2 time step.
-
double T2_mag_tminus1 = 0.0
Magnitude of unfiltered constraint torque on s/c 2at t-1 time step.
-
double T2_mag_t = 0.0
Magnitude of unfiltered constraint torque on s/c 2 at t time step.
-
double T2_filtered_mag_t = 0.0
Magnitude of filtered constraint torque on s/c 2 at t time step.
-
double T2_filtered_mag_tminus1 = 0.0
Magnitude of filtered constraint torque on s/c 2 at t-1 time step.
-
double T2_filtered_mag_tminus2 = 0.0
Magnitude of filtered constraint torque on s/c 2 at t-2 time step.
-
Eigen::Vector3d psi_N = Eigen::Vector3d::Zero()
[m] direction constraint violation in inertial frame
-
Eigen::Vector3d psiPrime_N = Eigen::Vector3d::Zero()
[m/s] direction rate constraint violation in inertial frame
-
Eigen::MRPd sigma_B2B1 = Eigen::MRPd::Identity()
attitude constraint violation
-
Eigen::Vector3d omega_B2B1_B2 = Eigen::Vector3d::Zero()
[rad/s] angular velocity constraint violation in spacecraft 2 body frame
-
Eigen::Vector3d Fc_N = Eigen::Vector3d::Zero()
[N] force applied on each spacecraft COM in the inertial frame
-
Eigen::Vector3d T_B2 = Eigen::Vector3d::Zero()
[N-m] torque applied on spacecraft 2 in its body frame
-
Eigen::Vector3d T_B1 = Eigen::Vector3d::Zero()
[N-m] torque applied on spacecraft 1 in its body frame
-
ConstraintDynamicEffector()